/* * Unix implementation of the OS query functions that detect Arm * architecture extensions. */ #include "putty.h" #include "ssh.h" #include "utils/arm_arch_queries.h" #if defined __arm__ || defined __aarch64__ bool platform_aes_neon_available(void) { #if defined HWCAP_AES return getauxval(AT_HWCAP) & HWCAP_AES; #elif defined HWCAP2_AES return getauxval(AT_HWCAP2) & HWCAP2_AES; #elif defined __APPLE__ SysctlResult res = test_sysctl_flag("hw.optional.arm.FEAT_AES"); /* Older M1 macOS didn't provide this flag, but as far as I know * implemented the crypto extension anyway, so treat 'feature * missing' as 'implemented' */ return res != SYSCTL_OFF; #else return false; #endif } bool platform_pmull_neon_available(void) { #if defined HWCAP_PMULL return getauxval(AT_HWCAP) & HWCAP_PMULL; #elif defined HWCAP2_PMULL return getauxval(AT_HWCAP2) & HWCAP2_PMULL; #elif defined __APPLE__ SysctlResult res = test_sysctl_flag("hw.optional.arm.FEAT_PMULL"); /* As above, treat 'missing' as enabled */ return res != SYSCTL_OFF; #else return false; #endif } bool platform_sha256_neon_available(void) { #if defined HWCAP_SHA2 return getauxval(AT_HWCAP) & HWCAP_SHA2; #elif defined HWCAP2_SHA2 return getauxval(AT_HWCAP2) & HWCAP2_SHA2; #elif defined __APPLE__ SysctlResult res = test_sysctl_flag("hw.optional.arm.FEAT_SHA256"); /* As above, treat 'missing' as enabled */ return res != SYSCTL_OFF; #else return false; #endif } bool platform_sha1_neon_available(void) { #if defined HWCAP_SHA1 return getauxval(AT_HWCAP) & HWCAP_SHA1; #elif defined HWCAP2_SHA1 return getauxval(AT_HWCAP2) & HWCAP2_SHA1; #elif defined __APPLE__ SysctlResult res = test_sysctl_flag("hw.optional.arm.FEAT_SHA1"); /* As above, treat 'missing' as enabled */ return res != SYSCTL_OFF; #else return false; #endif } bool platform_sha512_neon_available(void) { #if defined HWCAP_SHA512 return getauxval(AT_HWCAP) & HWCAP_SHA512; #elif defined HWCAP2_SHA512 return getauxval(AT_HWCAP2) & HWCAP2_SHA512; #elif defined __APPLE__ /* There are two sysctl flags for this, apparently invented at * different times. Try both, falling back to the older one. */ SysctlResult res = test_sysctl_flag("hw.optional.arm.FEAT_SHA512"); if (res != SYSCTL_MISSING) return res == SYSCTL_ON; res = test_sysctl_flag("hw.optional.armv8_2_sha512"); return res == SYSCTL_ON; #else return false; #endif } #else /* defined __arm__ || defined __aarch64__ */ /* * Include _something_ in this file to prevent an annoying compiler * warning, and to avoid having to condition out this file in * CMakeLists. It's in a library, so this variable shouldn't end up in * any actual program, because nothing will refer to it. */ const int arm_arch_queries_dummy_variable = 0; #endif /* defined __arm__ || defined __aarch64__ */