use num_integer::gcd;
use std::any::TypeId;
use std::collections::HashMap;
use std::sync::Arc;
use crate::{common::FftNum, fft_cache::FftCache, FftDirection};
use crate::algorithm::*;
use crate::sse::sse_butterflies::*;
use crate::sse::sse_prime_butterflies::*;
use crate::sse::sse_radix4::*;
use crate::Fft;
use crate::math_utils::{PrimeFactor, PrimeFactors};
const MIN_RADIX4_BITS: u32 = 6; const MAX_RADER_PRIME_FACTOR: usize = 23; const MIN_BLUESTEIN_MIXED_RADIX_LEN: usize = 90; #[derive(Debug, PartialEq, Clone)]
pub enum Recipe {
Dft(usize),
MixedRadix {
left_fft: Arc<Recipe>,
right_fft: Arc<Recipe>,
},
#[allow(dead_code)]
GoodThomasAlgorithm {
left_fft: Arc<Recipe>,
right_fft: Arc<Recipe>,
},
MixedRadixSmall {
left_fft: Arc<Recipe>,
right_fft: Arc<Recipe>,
},
GoodThomasAlgorithmSmall {
left_fft: Arc<Recipe>,
right_fft: Arc<Recipe>,
},
RadersAlgorithm {
inner_fft: Arc<Recipe>,
},
BluesteinsAlgorithm {
len: usize,
inner_fft: Arc<Recipe>,
},
Radix4(usize),
Butterfly1,
Butterfly2,
Butterfly3,
Butterfly4,
Butterfly5,
Butterfly6,
Butterfly7,
Butterfly8,
Butterfly9,
Butterfly10,
Butterfly11,
Butterfly12,
Butterfly13,
Butterfly15,
Butterfly16,
Butterfly17,
Butterfly19,
Butterfly23,
Butterfly29,
Butterfly31,
Butterfly32,
}
impl Recipe {
pub fn len(&self) -> usize {
match self {
Recipe::Dft(length) => *length,
Recipe::Radix4(length) => *length,
Recipe::Butterfly1 => 1,
Recipe::Butterfly2 => 2,
Recipe::Butterfly3 => 3,
Recipe::Butterfly4 => 4,
Recipe::Butterfly5 => 5,
Recipe::Butterfly6 => 6,
Recipe::Butterfly7 => 7,
Recipe::Butterfly8 => 8,
Recipe::Butterfly9 => 9,
Recipe::Butterfly10 => 10,
Recipe::Butterfly11 => 11,
Recipe::Butterfly12 => 12,
Recipe::Butterfly13 => 13,
Recipe::Butterfly15 => 15,
Recipe::Butterfly16 => 16,
Recipe::Butterfly17 => 17,
Recipe::Butterfly19 => 19,
Recipe::Butterfly23 => 23,
Recipe::Butterfly29 => 29,
Recipe::Butterfly31 => 31,
Recipe::Butterfly32 => 32,
Recipe::MixedRadix {
left_fft,
right_fft,
} => left_fft.len() * right_fft.len(),
Recipe::GoodThomasAlgorithm {
left_fft,
right_fft,
} => left_fft.len() * right_fft.len(),
Recipe::MixedRadixSmall {
left_fft,
right_fft,
} => left_fft.len() * right_fft.len(),
Recipe::GoodThomasAlgorithmSmall {
left_fft,
right_fft,
} => left_fft.len() * right_fft.len(),
Recipe::RadersAlgorithm { inner_fft } => inner_fft.len() + 1,
Recipe::BluesteinsAlgorithm { len, .. } => *len,
}
}
}
pub struct FftPlannerSse<T: FftNum> {
algorithm_cache: FftCache<T>,
recipe_cache: HashMap<usize, Arc<Recipe>>,
}
impl<T: FftNum> FftPlannerSse<T> {
pub fn new() -> Result<Self, ()> {
if is_x86_feature_detected!("sse4.1") {
let id_f32 = TypeId::of::<f32>();
let id_f64 = TypeId::of::<f64>();
let id_t = TypeId::of::<T>();
if id_t == id_f32 || id_t == id_f64 {
return Ok(Self {
algorithm_cache: FftCache::new(),
recipe_cache: HashMap::new(),
});
}
}
Err(())
}
pub fn plan_fft(&mut self, len: usize, direction: FftDirection) -> Arc<dyn Fft<T>> {
let recipe = self.design_fft_for_len(len);
self.build_fft(&recipe, direction)
}
pub fn plan_fft_forward(&mut self, len: usize) -> Arc<dyn Fft<T>> {
self.plan_fft(len, FftDirection::Forward)
}
pub fn plan_fft_inverse(&mut self, len: usize) -> Arc<dyn Fft<T>> {
self.plan_fft(len, FftDirection::Inverse)
}
fn design_fft_for_len(&mut self, len: usize) -> Arc<Recipe> {
if len < 1 {
Arc::new(Recipe::Dft(len))
} else if let Some(recipe) = self.recipe_cache.get(&len) {
Arc::clone(&recipe)
} else {
let factors = PrimeFactors::compute(len);
let recipe = self.design_fft_with_factors(len, factors);
self.recipe_cache.insert(len, Arc::clone(&recipe));
recipe
}
}
fn build_fft(&mut self, recipe: &Recipe, direction: FftDirection) -> Arc<dyn Fft<T>> {
let len = recipe.len();
if let Some(instance) = self.algorithm_cache.get(len, direction) {
instance
} else {
let fft = self.build_new_fft(recipe, direction);
self.algorithm_cache.insert(&fft);
fft
}
}
fn build_new_fft(&mut self, recipe: &Recipe, direction: FftDirection) -> Arc<dyn Fft<T>> {
let id_f32 = TypeId::of::<f32>();
let id_f64 = TypeId::of::<f64>();
let id_t = TypeId::of::<T>();
match recipe {
Recipe::Dft(len) => Arc::new(Dft::new(*len, direction)) as Arc<dyn Fft<T>>,
Recipe::Radix4(len) => {
if id_t == id_f32 {
Arc::new(Sse32Radix4::new(*len, direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(Sse64Radix4::new(*len, direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly1 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly1::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly1::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly2 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly2::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly2::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly3 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly3::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly3::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly4 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly4::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly4::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly5 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly5::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly5::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly6 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly6::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly6::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly7 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly7::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly7::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly8 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly8::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly8::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly9 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly9::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly9::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly10 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly10::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly10::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly11 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly11::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly11::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly12 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly12::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly12::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly13 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly13::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly13::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly15 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly15::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly15::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly16 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly16::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly16::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly17 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly17::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly17::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly19 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly19::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly19::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly23 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly23::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly23::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly29 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly29::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly29::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly31 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly31::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly31::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::Butterfly32 => {
if id_t == id_f32 {
Arc::new(SseF32Butterfly32::new(direction)) as Arc<dyn Fft<T>>
} else if id_t == id_f64 {
Arc::new(SseF64Butterfly32::new(direction)) as Arc<dyn Fft<T>>
} else {
panic!("Not f32 or f64");
}
}
Recipe::MixedRadix {
left_fft,
right_fft,
} => {
let left_fft = self.build_fft(&left_fft, direction);
let right_fft = self.build_fft(&right_fft, direction);
Arc::new(MixedRadix::new(left_fft, right_fft)) as Arc<dyn Fft<T>>
}
Recipe::GoodThomasAlgorithm {
left_fft,
right_fft,
} => {
let left_fft = self.build_fft(&left_fft, direction);
let right_fft = self.build_fft(&right_fft, direction);
Arc::new(GoodThomasAlgorithm::new(left_fft, right_fft)) as Arc<dyn Fft<T>>
}
Recipe::MixedRadixSmall {
left_fft,
right_fft,
} => {
let left_fft = self.build_fft(&left_fft, direction);
let right_fft = self.build_fft(&right_fft, direction);
Arc::new(MixedRadixSmall::new(left_fft, right_fft)) as Arc<dyn Fft<T>>
}
Recipe::GoodThomasAlgorithmSmall {
left_fft,
right_fft,
} => {
let left_fft = self.build_fft(&left_fft, direction);
let right_fft = self.build_fft(&right_fft, direction);
Arc::new(GoodThomasAlgorithmSmall::new(left_fft, right_fft)) as Arc<dyn Fft<T>>
}
Recipe::RadersAlgorithm { inner_fft } => {
let inner_fft = self.build_fft(&inner_fft, direction);
Arc::new(RadersAlgorithm::new(inner_fft)) as Arc<dyn Fft<T>>
}
Recipe::BluesteinsAlgorithm { len, inner_fft } => {
let inner_fft = self.build_fft(&inner_fft, direction);
Arc::new(BluesteinsAlgorithm::new(*len, inner_fft)) as Arc<dyn Fft<T>>
}
}
}
fn design_fft_with_factors(&mut self, len: usize, factors: PrimeFactors) -> Arc<Recipe> {
if let Some(fft_instance) = self.design_butterfly_algorithm(len) {
fft_instance
} else if factors.is_prime() {
self.design_prime(len)
} else if len.trailing_zeros() >= MIN_RADIX4_BITS {
if len.is_power_of_two() {
Arc::new(Recipe::Radix4(len))
} else {
let non_power_of_two = factors
.remove_factors(PrimeFactor {
value: 2,
count: len.trailing_zeros(),
})
.unwrap();
let power_of_two = PrimeFactors::compute(1 << len.trailing_zeros());
self.design_mixed_radix(power_of_two, non_power_of_two)
}
} else {
let butterflies: [usize; 20] = [
2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 15, 16, 17, 19, 23, 29, 31, 32,
];
let mut bf_left = 0;
let mut bf_right = 0;
if len > 13 && len <= 1024 {
for (n, bf_l) in butterflies.iter().enumerate() {
if len % bf_l == 0 {
let bf_r = len / bf_l;
if butterflies.iter().skip(n).any(|&m| m == bf_r) {
bf_left = *bf_l;
bf_right = bf_r;
}
}
}
if bf_left > 0 {
let fact_l = PrimeFactors::compute(bf_left);
let fact_r = PrimeFactors::compute(bf_right);
return self.design_mixed_radix(fact_l, fact_r);
}
}
let (left_factors, right_factors) = factors.partition_factors();
self.design_mixed_radix(left_factors, right_factors)
}
}
fn design_mixed_radix(
&mut self,
left_factors: PrimeFactors,
right_factors: PrimeFactors,
) -> Arc<Recipe> {
let left_len = left_factors.get_product();
let right_len = right_factors.get_product();
let left_fft = self.design_fft_with_factors(left_len, left_factors);
let right_fft = self.design_fft_with_factors(right_len, right_factors);
if left_len < 33 && right_len < 33 {
if gcd(left_len, right_len) == 1 {
Arc::new(Recipe::GoodThomasAlgorithmSmall {
left_fft,
right_fft,
})
} else {
Arc::new(Recipe::MixedRadixSmall {
left_fft,
right_fft,
})
}
} else {
Arc::new(Recipe::MixedRadix {
left_fft,
right_fft,
})
}
}
fn design_butterfly_algorithm(&mut self, len: usize) -> Option<Arc<Recipe>> {
match len {
1 => Some(Arc::new(Recipe::Butterfly1)),
2 => Some(Arc::new(Recipe::Butterfly2)),
3 => Some(Arc::new(Recipe::Butterfly3)),
4 => Some(Arc::new(Recipe::Butterfly4)),
5 => Some(Arc::new(Recipe::Butterfly5)),
6 => Some(Arc::new(Recipe::Butterfly6)),
7 => Some(Arc::new(Recipe::Butterfly7)),
8 => Some(Arc::new(Recipe::Butterfly8)),
9 => Some(Arc::new(Recipe::Butterfly9)),
10 => Some(Arc::new(Recipe::Butterfly10)),
11 => Some(Arc::new(Recipe::Butterfly11)),
12 => Some(Arc::new(Recipe::Butterfly12)),
13 => Some(Arc::new(Recipe::Butterfly13)),
15 => Some(Arc::new(Recipe::Butterfly15)),
16 => Some(Arc::new(Recipe::Butterfly16)),
17 => Some(Arc::new(Recipe::Butterfly17)),
19 => Some(Arc::new(Recipe::Butterfly19)),
23 => Some(Arc::new(Recipe::Butterfly23)),
29 => Some(Arc::new(Recipe::Butterfly29)),
31 => Some(Arc::new(Recipe::Butterfly31)),
32 => Some(Arc::new(Recipe::Butterfly32)),
_ => None,
}
}
fn design_prime(&mut self, len: usize) -> Arc<Recipe> {
let inner_fft_len_rader = len - 1;
let raders_factors = PrimeFactors::compute(inner_fft_len_rader);
if raders_factors
.get_other_factors()
.iter()
.any(|val| val.value > MAX_RADER_PRIME_FACTOR)
{
let inner_fft_len_pow2 = (2 * len - 1).checked_next_power_of_two().unwrap();
let min_inner_len = 2 * len - 1;
let mixed_radix_len = 3 * inner_fft_len_pow2 / 4;
let inner_fft =
if mixed_radix_len >= min_inner_len && len >= MIN_BLUESTEIN_MIXED_RADIX_LEN {
let mixed_radix_factors = PrimeFactors::compute(mixed_radix_len);
self.design_fft_with_factors(mixed_radix_len, mixed_radix_factors)
} else {
Arc::new(Recipe::Radix4(inner_fft_len_pow2))
};
Arc::new(Recipe::BluesteinsAlgorithm { len, inner_fft })
} else {
let inner_fft = self.design_fft_with_factors(inner_fft_len_rader, raders_factors);
Arc::new(Recipe::RadersAlgorithm { inner_fft })
}
}
}
#[cfg(test)]
mod unit_tests {
use super::*;
fn is_mixedradix(plan: &Recipe) -> bool {
match plan {
&Recipe::MixedRadix { .. } => true,
_ => false,
}
}
fn is_mixedradixsmall(plan: &Recipe) -> bool {
match plan {
&Recipe::MixedRadixSmall { .. } => true,
_ => false,
}
}
fn is_goodthomassmall(plan: &Recipe) -> bool {
match plan {
&Recipe::GoodThomasAlgorithmSmall { .. } => true,
_ => false,
}
}
fn is_raders(plan: &Recipe) -> bool {
match plan {
&Recipe::RadersAlgorithm { .. } => true,
_ => false,
}
}
fn is_bluesteins(plan: &Recipe) -> bool {
match plan {
&Recipe::BluesteinsAlgorithm { .. } => true,
_ => false,
}
}
#[test]
fn test_plan_sse_trivial() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for len in 0..1 {
let plan = planner.design_fft_for_len(len);
assert_eq!(*plan, Recipe::Dft(len));
assert_eq!(plan.len(), len, "Recipe reports wrong length");
}
}
#[test]
fn test_plan_sse_largepoweroftwo() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for pow in 6..32 {
let len = 1 << pow;
let plan = planner.design_fft_for_len(len);
assert_eq!(*plan, Recipe::Radix4(len));
assert_eq!(plan.len(), len, "Recipe reports wrong length");
}
}
#[test]
fn test_plan_sse_butterflies() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
assert_eq!(*planner.design_fft_for_len(2), Recipe::Butterfly2);
assert_eq!(*planner.design_fft_for_len(3), Recipe::Butterfly3);
assert_eq!(*planner.design_fft_for_len(4), Recipe::Butterfly4);
assert_eq!(*planner.design_fft_for_len(5), Recipe::Butterfly5);
assert_eq!(*planner.design_fft_for_len(6), Recipe::Butterfly6);
assert_eq!(*planner.design_fft_for_len(7), Recipe::Butterfly7);
assert_eq!(*planner.design_fft_for_len(8), Recipe::Butterfly8);
assert_eq!(*planner.design_fft_for_len(9), Recipe::Butterfly9);
assert_eq!(*planner.design_fft_for_len(10), Recipe::Butterfly10);
assert_eq!(*planner.design_fft_for_len(11), Recipe::Butterfly11);
assert_eq!(*planner.design_fft_for_len(12), Recipe::Butterfly12);
assert_eq!(*planner.design_fft_for_len(13), Recipe::Butterfly13);
assert_eq!(*planner.design_fft_for_len(15), Recipe::Butterfly15);
assert_eq!(*planner.design_fft_for_len(16), Recipe::Butterfly16);
assert_eq!(*planner.design_fft_for_len(17), Recipe::Butterfly17);
assert_eq!(*planner.design_fft_for_len(19), Recipe::Butterfly19);
assert_eq!(*planner.design_fft_for_len(23), Recipe::Butterfly23);
assert_eq!(*planner.design_fft_for_len(29), Recipe::Butterfly29);
assert_eq!(*planner.design_fft_for_len(31), Recipe::Butterfly31);
assert_eq!(*planner.design_fft_for_len(32), Recipe::Butterfly32);
}
#[test]
fn test_plan_sse_mixedradix() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for pow2 in 2..5 {
for pow3 in 2..5 {
for pow5 in 2..5 {
for pow7 in 2..5 {
let len = 2usize.pow(pow2)
* 3usize.pow(pow3)
* 5usize.pow(pow5)
* 7usize.pow(pow7);
let plan = planner.design_fft_for_len(len);
assert!(is_mixedradix(&plan), "Expected MixedRadix, got {:?}", plan);
assert_eq!(plan.len(), len, "Recipe reports wrong length");
}
}
}
}
}
#[test]
fn test_plan_sse_mixedradixsmall() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for len in [5 * 20, 5 * 25].iter() {
let plan = planner.design_fft_for_len(*len);
assert!(
is_mixedradixsmall(&plan),
"Expected MixedRadixSmall, got {:?}",
plan
);
assert_eq!(plan.len(), *len, "Recipe reports wrong length");
}
}
#[test]
fn test_plan_sse_goodthomasbutterfly() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for len in [3 * 7, 5 * 7, 11 * 13, 2 * 29].iter() {
let plan = planner.design_fft_for_len(*len);
assert!(
is_goodthomassmall(&plan),
"Expected GoodThomasAlgorithmSmall, got {:?}",
plan
);
assert_eq!(plan.len(), *len, "Recipe reports wrong length");
}
}
#[test]
fn test_plan_sse_bluestein_vs_rader() {
let difficultprimes: [usize; 11] = [59, 83, 107, 149, 167, 173, 179, 359, 719, 1439, 2879];
let easyprimes: [usize; 24] = [
53, 61, 67, 71, 73, 79, 89, 97, 101, 103, 109, 113, 127, 131, 137, 139, 151, 157, 163,
181, 191, 193, 197, 199,
];
let mut planner = FftPlannerSse::<f64>::new().unwrap();
for len in difficultprimes.iter() {
let plan = planner.design_fft_for_len(*len);
assert!(
is_bluesteins(&plan),
"Expected BluesteinsAlgorithm, got {:?}",
plan
);
assert_eq!(plan.len(), *len, "Recipe reports wrong length");
}
for len in easyprimes.iter() {
let plan = planner.design_fft_for_len(*len);
assert!(is_raders(&plan), "Expected RadersAlgorithm, got {:?}", plan);
assert_eq!(plan.len(), *len, "Recipe reports wrong length");
}
}
#[test]
fn test_sse_fft_cache() {
{
let mut planner = FftPlannerSse::<f64>::new().unwrap();
let fft_a = planner.plan_fft(1234, FftDirection::Forward);
let fft_b = planner.plan_fft(1234, FftDirection::Forward);
assert!(Arc::ptr_eq(&fft_a, &fft_b), "Existing fft was not reused");
}
{
let mut planner = FftPlannerSse::<f64>::new().unwrap();
let fft_a = planner.plan_fft(1234, FftDirection::Inverse);
let fft_b = planner.plan_fft(1234, FftDirection::Inverse);
assert!(Arc::ptr_eq(&fft_a, &fft_b), "Existing fft was not reused");
}
{
let mut planner = FftPlannerSse::<f64>::new().unwrap();
let fft_a = planner.plan_fft(1234, FftDirection::Forward);
let fft_b = planner.plan_fft(1234, FftDirection::Inverse);
assert!(
!Arc::ptr_eq(&fft_a, &fft_b),
"Existing fft was reused, even though directions don't match"
);
}
}
#[test]
fn test_sse_recipe_cache() {
let mut planner = FftPlannerSse::<f64>::new().unwrap();
let fft_a = planner.design_fft_for_len(1234);
let fft_b = planner.design_fft_for_len(1234);
assert!(
Arc::ptr_eq(&fft_a, &fft_b),
"Existing recipe was not reused"
);
}
}