mirror of
https://github.com/rust-lang/rust.git
synced 2025-10-24 14:08:17 +00:00

Arm's FEAT_FP and Feat_AdvSIMD describe the same thing on AArch64: The Neon unit, which handles both floating point and SIMD instructions. Moreover, a configuration for AArch64 must include both or neither. Arm says "entirely proprietary" toolchains may omit floating point: https://developer.arm.com/documentation/102374/0101/Data-processing---floating-point In the Programmer's Guide for Armv8-A, Arm says AArch64 can have both FP and Neon or neither in custom implementations: https://developer.arm.com/documentation/den0024/a/AArch64-Floating-point-and-NEON In "Bare metal boot code for Armv8-A", enabling Neon and FP is just disabling the same trap flag: https://developer.arm.com/documentation/dai0527/a In an unlikely future where "Neon and FP" become unrelated, we can add "[+-]fp" as its own feature flag. Until then, we can simplify programming with Rust on AArch64 by folding both into "[+-]neon", which is valid as it supersets both. "[+-]neon" is retained for niche uses such as firmware, kernels, "I just hate floats", and so on.
154 lines
8.2 KiB
Rust
154 lines
8.2 KiB
Rust
//! These tests just check that the macros are available in libstd.
|
|
|
|
#![cfg_attr(
|
|
any(
|
|
all(target_arch = "arm", any(target_os = "linux", target_os = "android")),
|
|
all(bootstrap, target_arch = "aarch64", any(target_os = "linux", target_os = "android")),
|
|
all(target_arch = "powerpc", target_os = "linux"),
|
|
all(target_arch = "powerpc64", target_os = "linux"),
|
|
),
|
|
feature(stdsimd)
|
|
)]
|
|
|
|
#[test]
|
|
#[cfg(all(target_arch = "arm", any(target_os = "linux", target_os = "android")))]
|
|
fn arm_linux() {
|
|
use std::arch::is_arm_feature_detected;
|
|
println!("neon: {}", is_arm_feature_detected!("neon"));
|
|
println!("pmull: {}", is_arm_feature_detected!("pmull"));
|
|
println!("crypto: {}", is_arm_feature_detected!("crypto"));
|
|
println!("crc: {}", is_arm_feature_detected!("crc"));
|
|
println!("aes: {}", is_arm_feature_detected!("aes"));
|
|
println!("sha2: {}", is_arm_feature_detected!("sha2"));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(all(target_arch = "aarch64", any(target_os = "linux", target_os = "android")))]
|
|
fn aarch64_linux() {
|
|
use std::arch::is_aarch64_feature_detected;
|
|
println!("neon: {}", is_aarch64_feature_detected!("neon"));
|
|
println!("asimd: {}", is_aarch64_feature_detected!("asimd"));
|
|
println!("pmull: {}", is_aarch64_feature_detected!("pmull"));
|
|
println!("fp16: {}", is_aarch64_feature_detected!("fp16"));
|
|
println!("sve: {}", is_aarch64_feature_detected!("sve"));
|
|
println!("crc: {}", is_aarch64_feature_detected!("crc"));
|
|
println!("lse: {}", is_aarch64_feature_detected!("lse"));
|
|
println!("lse2: {}", is_aarch64_feature_detected!("lse2"));
|
|
println!("rdm: {}", is_aarch64_feature_detected!("rdm"));
|
|
println!("rcpc: {}", is_aarch64_feature_detected!("rcpc"));
|
|
println!("rcpc2: {}", is_aarch64_feature_detected!("rcpc2"));
|
|
println!("dotprod: {}", is_aarch64_feature_detected!("dotprod"));
|
|
println!("tme: {}", is_aarch64_feature_detected!("tme"));
|
|
println!("fhm: {}", is_aarch64_feature_detected!("fhm"));
|
|
println!("dit: {}", is_aarch64_feature_detected!("dit"));
|
|
println!("flagm: {}", is_aarch64_feature_detected!("flagm"));
|
|
println!("ssbs: {}", is_aarch64_feature_detected!("ssbs"));
|
|
println!("sb: {}", is_aarch64_feature_detected!("sb"));
|
|
println!("paca: {}", is_aarch64_feature_detected!("paca"));
|
|
println!("pacg: {}", is_aarch64_feature_detected!("pacg"));
|
|
println!("dpb: {}", is_aarch64_feature_detected!("dpb"));
|
|
println!("dpb2: {}", is_aarch64_feature_detected!("dpb2"));
|
|
println!("sve2: {}", is_aarch64_feature_detected!("sve2"));
|
|
println!("sve2-aes: {}", is_aarch64_feature_detected!("sve2-aes"));
|
|
println!("sve2-sm4: {}", is_aarch64_feature_detected!("sve2-sm4"));
|
|
println!("sve2-sha3: {}", is_aarch64_feature_detected!("sve2-sha3"));
|
|
println!("sve2-bitperm: {}", is_aarch64_feature_detected!("sve2-bitperm"));
|
|
println!("frintts: {}", is_aarch64_feature_detected!("frintts"));
|
|
println!("i8mm: {}", is_aarch64_feature_detected!("i8mm"));
|
|
println!("f32mm: {}", is_aarch64_feature_detected!("f32mm"));
|
|
println!("f64mm: {}", is_aarch64_feature_detected!("f64mm"));
|
|
println!("bf16: {}", is_aarch64_feature_detected!("bf16"));
|
|
println!("rand: {}", is_aarch64_feature_detected!("rand"));
|
|
println!("bti: {}", is_aarch64_feature_detected!("bti"));
|
|
println!("mte: {}", is_aarch64_feature_detected!("mte"));
|
|
println!("jsconv: {}", is_aarch64_feature_detected!("jsconv"));
|
|
println!("fcma: {}", is_aarch64_feature_detected!("fcma"));
|
|
println!("aes: {}", is_aarch64_feature_detected!("aes"));
|
|
println!("sha2: {}", is_aarch64_feature_detected!("sha2"));
|
|
println!("sha3: {}", is_aarch64_feature_detected!("sha3"));
|
|
println!("sm4: {}", is_aarch64_feature_detected!("sm4"));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(all(target_arch = "powerpc", target_os = "linux"))]
|
|
fn powerpc_linux() {
|
|
use std::arch::is_powerpc_feature_detected;
|
|
println!("altivec: {}", is_powerpc_feature_detected!("altivec"));
|
|
println!("vsx: {}", is_powerpc_feature_detected!("vsx"));
|
|
println!("power8: {}", is_powerpc_feature_detected!("power8"));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(all(target_arch = "powerpc64", target_os = "linux"))]
|
|
fn powerpc64_linux() {
|
|
use std::arch::is_powerpc64_feature_detected;
|
|
println!("altivec: {}", is_powerpc64_feature_detected!("altivec"));
|
|
println!("vsx: {}", is_powerpc64_feature_detected!("vsx"));
|
|
println!("power8: {}", is_powerpc64_feature_detected!("power8"));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
|
|
fn x86_all() {
|
|
use std::arch::is_x86_feature_detected;
|
|
|
|
// the below is the set of features we can test at runtime, but don't actually
|
|
// use to gate anything and are thus not part of the X86_ALLOWED_FEATURES list
|
|
|
|
println!("abm: {:?}", is_x86_feature_detected!("abm")); // this is a synonym for lzcnt but we test it anyways
|
|
println!("mmx: {:?}", is_x86_feature_detected!("mmx"));
|
|
println!("tsc: {:?}", is_x86_feature_detected!("tsc"));
|
|
|
|
// the below is in alphabetical order and matches
|
|
// the order of X86_ALLOWED_FEATURES in rustc_codegen_ssa's target_features.rs
|
|
|
|
println!("adx: {:?}", is_x86_feature_detected!("adx"));
|
|
println!("aes: {:?}", is_x86_feature_detected!("aes"));
|
|
println!("avx: {:?}", is_x86_feature_detected!("avx"));
|
|
println!("avx2: {:?}", is_x86_feature_detected!("avx2"));
|
|
println!("avx512bf16: {:?}", is_x86_feature_detected!("avx512bf16"));
|
|
println!("avx512bitalg: {:?}", is_x86_feature_detected!("avx512bitalg"));
|
|
println!("avx512bw: {:?}", is_x86_feature_detected!("avx512bw"));
|
|
println!("avx512cd: {:?}", is_x86_feature_detected!("avx512cd"));
|
|
println!("avx512dq: {:?}", is_x86_feature_detected!("avx512dq"));
|
|
println!("avx512er: {:?}", is_x86_feature_detected!("avx512er"));
|
|
println!("avx512f: {:?}", is_x86_feature_detected!("avx512f"));
|
|
println!("avx512gfni: {:?}", is_x86_feature_detected!("avx512gfni"));
|
|
println!("avx512ifma: {:?}", is_x86_feature_detected!("avx512ifma"));
|
|
println!("avx512pf: {:?}", is_x86_feature_detected!("avx512pf"));
|
|
println!("avx512vaes: {:?}", is_x86_feature_detected!("avx512vaes"));
|
|
println!("avx512vbmi: {:?}", is_x86_feature_detected!("avx512vbmi"));
|
|
println!("avx512vbmi2: {:?}", is_x86_feature_detected!("avx512vbmi2"));
|
|
println!("avx512vl: {:?}", is_x86_feature_detected!("avx512vl"));
|
|
println!("avx512vnni: {:?}", is_x86_feature_detected!("avx512vnni"));
|
|
println!("avx512vp2intersect: {:?}", is_x86_feature_detected!("avx512vp2intersect"));
|
|
println!("avx512vpclmulqdq: {:?}", is_x86_feature_detected!("avx512vpclmulqdq"));
|
|
println!("avx512vpopcntdq: {:?}", is_x86_feature_detected!("avx512vpopcntdq"));
|
|
println!("bmi1: {:?}", is_x86_feature_detected!("bmi1"));
|
|
println!("bmi2: {:?}", is_x86_feature_detected!("bmi2"));
|
|
println!("cmpxchg16b: {:?}", is_x86_feature_detected!("cmpxchg16b"));
|
|
println!("f16c: {:?}", is_x86_feature_detected!("f16c"));
|
|
println!("fma: {:?}", is_x86_feature_detected!("fma"));
|
|
println!("fxsr: {:?}", is_x86_feature_detected!("fxsr"));
|
|
println!("lzcnt: {:?}", is_x86_feature_detected!("lzcnt"));
|
|
//println!("movbe: {:?}", is_x86_feature_detected!("movbe")); // movbe is unsupported as a target feature
|
|
println!("pclmulqdq: {:?}", is_x86_feature_detected!("pclmulqdq"));
|
|
println!("popcnt: {:?}", is_x86_feature_detected!("popcnt"));
|
|
println!("rdrand: {:?}", is_x86_feature_detected!("rdrand"));
|
|
println!("rdseed: {:?}", is_x86_feature_detected!("rdseed"));
|
|
println!("rtm: {:?}", is_x86_feature_detected!("rtm"));
|
|
println!("sha: {:?}", is_x86_feature_detected!("sha"));
|
|
println!("sse: {:?}", is_x86_feature_detected!("sse"));
|
|
println!("sse2: {:?}", is_x86_feature_detected!("sse2"));
|
|
println!("sse3: {:?}", is_x86_feature_detected!("sse3"));
|
|
println!("sse4.1: {:?}", is_x86_feature_detected!("sse4.1"));
|
|
println!("sse4.2: {:?}", is_x86_feature_detected!("sse4.2"));
|
|
println!("sse4a: {:?}", is_x86_feature_detected!("sse4a"));
|
|
println!("ssse3: {:?}", is_x86_feature_detected!("ssse3"));
|
|
println!("tbm: {:?}", is_x86_feature_detected!("tbm"));
|
|
println!("xsave: {:?}", is_x86_feature_detected!("xsave"));
|
|
println!("xsavec: {:?}", is_x86_feature_detected!("xsavec"));
|
|
println!("xsaveopt: {:?}", is_x86_feature_detected!("xsaveopt"));
|
|
println!("xsaves: {:?}", is_x86_feature_detected!("xsaves"));
|
|
}
|