| //! Run-time feature detection for ARM on Linux. |
| |
| use super::auxvec; |
| use crate::detect::{bit, cache, Feature}; |
| |
| /// Try to read the features from the auxiliary vector, and if that fails, try |
| /// to read them from /proc/cpuinfo. |
| pub(crate) fn detect_features() -> cache::Initializer { |
| let mut value = cache::Initializer::default(); |
| let enable_feature = |value: &mut cache::Initializer, f, enable| { |
| if enable { |
| value.set(f as u32); |
| } |
| }; |
| |
| // The values are part of the platform-specific [asm/hwcap.h][hwcap] |
| // |
| // [hwcap]: https://github.com/torvalds/linux/blob/master/arch/arm/include/uapi/asm/hwcap.h |
| if let Ok(auxv) = auxvec::auxv() { |
| enable_feature(&mut value, Feature::neon, bit::test(auxv.hwcap, 12)); |
| enable_feature(&mut value, Feature::pmull, bit::test(auxv.hwcap2, 1)); |
| enable_feature(&mut value, Feature::crc, bit::test(auxv.hwcap2, 4)); |
| enable_feature( |
| &mut value, |
| Feature::crypto, |
| bit::test(auxv.hwcap2, 0) |
| && bit::test(auxv.hwcap2, 1) |
| && bit::test(auxv.hwcap2, 2) |
| && bit::test(auxv.hwcap2, 3), |
| ); |
| enable_feature(&mut value, Feature::aes, bit::test(auxv.hwcap2, 0)); |
| // SHA2 requires SHA1 & SHA2 features |
| enable_feature( |
| &mut value, |
| Feature::sha2, |
| bit::test(auxv.hwcap2, 2) && bit::test(auxv.hwcap2, 3), |
| ); |
| return value; |
| } |
| |
| #[cfg(feature = "std_detect_file_io")] |
| if let Ok(c) = super::cpuinfo::CpuInfo::new() { |
| enable_feature( |
| &mut value, |
| Feature::neon, |
| c.field("Features").has("neon") && !has_broken_neon(&c), |
| ); |
| enable_feature(&mut value, Feature::pmull, c.field("Features").has("pmull")); |
| enable_feature(&mut value, Feature::crc, c.field("Features").has("crc32")); |
| enable_feature( |
| &mut value, |
| Feature::crypto, |
| c.field("Features").has("aes") |
| && c.field("Features").has("pmull") |
| && c.field("Features").has("sha1") |
| && c.field("Features").has("sha2"), |
| ); |
| enable_feature(&mut value, Feature::aes, c.field("Features").has("aes")); |
| enable_feature( |
| &mut value, |
| Feature::sha2, |
| c.field("Features").has("sha1") && c.field("Features").has("sha2"), |
| ); |
| return value; |
| } |
| value |
| } |
| |
| /// Is the CPU known to have a broken NEON unit? |
| /// |
| /// See https://crbug.com/341598. |
| #[cfg(feature = "std_detect_file_io")] |
| fn has_broken_neon(cpuinfo: &super::cpuinfo::CpuInfo) -> bool { |
| cpuinfo.field("CPU implementer") == "0x51" |
| && cpuinfo.field("CPU architecture") == "7" |
| && cpuinfo.field("CPU variant") == "0x1" |
| && cpuinfo.field("CPU part") == "0x04d" |
| && cpuinfo.field("CPU revision") == "0" |
| } |