use arg_enum_proc_macro::ArgEnum;
use std::env;
use std::str::FromStr;
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, ArgEnum)]
pub enum CpuFeatureLevel {
RUST,
NEON,
}
impl CpuFeatureLevel {
#[cfg(test)]
pub(crate) const fn all() -> &'static [Self] {
use CpuFeatureLevel::*;
&[RUST, NEON]
}
pub const fn len() -> usize {
CpuFeatureLevel::NEON as usize + 1
}
#[inline(always)]
pub fn as_index(self) -> usize {
self as usize
}
}
impl Default for CpuFeatureLevel {
fn default() -> CpuFeatureLevel {
let detected = CpuFeatureLevel::NEON;
let manual: CpuFeatureLevel = match env::var("RAV1E_CPU_TARGET") {
Ok(feature) => CpuFeatureLevel::from_str(&feature).unwrap_or(detected),
Err(_e) => detected,
};
if manual > detected {
detected
} else {
manual
}
}
}
macro_rules! cpu_function_lookup_table {
($name:ident: [$type:ty], default: $empty:expr, [$(($key:ident, $value:expr)),*]) => {
static $name: [$type; crate::cpu_features::CpuFeatureLevel::len()] = {
use crate::cpu_features::CpuFeatureLevel;
#[allow(unused_mut)]
let mut out: [$type; CpuFeatureLevel::len()] = [$empty; CpuFeatureLevel::len()];
#[allow(unused_mut)]
let mut set: [bool; CpuFeatureLevel::len()] = [false; CpuFeatureLevel::len()];
#[allow(unused_imports)]
use CpuFeatureLevel::*;
$(
out[$key as usize] = $value;
set[$key as usize] = true;
)*
cpu_function_lookup_table!(waterfall_cpu_features(out, set, [NEON]));
out
};
};
($pub:vis, $name:ident: [$type:ty], default: $empty:expr, [$(($key:ident, $value:expr)),*]) => {
$pub cpu_function_lookup_table!($name: [$type], default: $empty, [$(($key, $value)),*]);
};
(waterfall_cpu_features($out:ident, $set:ident, [$($cpu:ident),*])) => {
#[allow(unused_assignments)]
let mut best = [$out[0], $out[0]];
$(
best[$set[$cpu as usize] as usize] = $out[$cpu as usize];
$out[$cpu as usize] = best[1];
)*
};
($pub:vis, $name:ident: [$type:ty], default: $empty:expr, [$($key:ident),*]) => {
paste::item!{
cpu_function_lookup_table!(
$pub, $name: [$type], default: $empty, [$(($key, [<$name _$key>])),*]
);
}
};
($name:ident: [$type:ty], default: $empty:expr, [$($key:ident),*]) => {
paste::item!{
cpu_function_lookup_table!(
$name: [$type], default: $empty, [$(($key, [<$name _$key>])),*]
);
}
};
}