|  | // Copyright 2020 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | //go:build arm64 && linux | 
|  | // +build arm64,linux | 
|  |  | 
|  | package cpu | 
|  |  | 
|  | // HWCap may be initialized by archauxv and | 
|  | // should not be changed after it was initialized. | 
|  | var HWCap uint | 
|  |  | 
|  | // HWCAP bits. These are exposed by Linux. | 
|  | const ( | 
|  | hwcap_AES     = 1 << 3 | 
|  | hwcap_PMULL   = 1 << 4 | 
|  | hwcap_SHA1    = 1 << 5 | 
|  | hwcap_SHA2    = 1 << 6 | 
|  | hwcap_CRC32   = 1 << 7 | 
|  | hwcap_ATOMICS = 1 << 8 | 
|  | hwcap_CPUID   = 1 << 11 | 
|  | ) | 
|  |  | 
|  | func hwcapInit(os string) { | 
|  | // HWCap was populated by the runtime from the auxiliary vector. | 
|  | // Use HWCap information since reading aarch64 system registers | 
|  | // is not supported in user space on older linux kernels. | 
|  | ARM64.HasAES = isSet(HWCap, hwcap_AES) | 
|  | ARM64.HasPMULL = isSet(HWCap, hwcap_PMULL) | 
|  | ARM64.HasSHA1 = isSet(HWCap, hwcap_SHA1) | 
|  | ARM64.HasSHA2 = isSet(HWCap, hwcap_SHA2) | 
|  | ARM64.HasCRC32 = isSet(HWCap, hwcap_CRC32) | 
|  | ARM64.HasCPUID = isSet(HWCap, hwcap_CPUID) | 
|  |  | 
|  | // The Samsung S9+ kernel reports support for atomics, but not all cores | 
|  | // actually support them, resulting in SIGILL. See issue #28431. | 
|  | // TODO(elias.naur): Only disable the optimization on bad chipsets on android. | 
|  | ARM64.HasATOMICS = isSet(HWCap, hwcap_ATOMICS) && os != "android" | 
|  |  | 
|  | // Check to see if executing on a NeoverseN1 and in order to do that, | 
|  | // check the AUXV for the CPUID bit. The getMIDR function executes an | 
|  | // instruction which would normally be an illegal instruction, but it's | 
|  | // trapped by the kernel, the value sanitized and then returned. Without | 
|  | // the CPUID bit the kernel will not trap the instruction and the process | 
|  | // will be terminated with SIGILL. | 
|  | if ARM64.HasCPUID { | 
|  | midr := getMIDR() | 
|  | part_num := uint16((midr >> 4) & 0xfff) | 
|  | implementor := byte((midr >> 24) & 0xff) | 
|  |  | 
|  | if implementor == 'A' && part_num == 0xd0c { | 
|  | ARM64.IsNeoverseN1 = true | 
|  | } | 
|  | if implementor == 'A' && part_num == 0xd40 { | 
|  | ARM64.IsZeus = true | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func isSet(hwc uint, value uint) bool { | 
|  | return hwc&value != 0 | 
|  | } |