pounder_test/dsp/src/lowpass.rs

38 lines
1.1 KiB
Rust
Raw Normal View History

use generic_array::{ArrayLength, GenericArray};
/// Arbitrary order, high dynamic range, wide coefficient range,
/// lowpass filter implementation.
///
/// Type argument N is the filter order + 1.
#[derive(Clone, Default)]
pub struct Lowpass<N: ArrayLength<i32>> {
// IIR state storage
xy: GenericArray<i32, N>,
}
impl<N: ArrayLength<i32>> Lowpass<N> {
/// Update the filter with a new sample.
///
/// # Args
/// * `x`: Input data
2021-02-10 20:44:10 +08:00
/// * `k`: Log2 time constant, 1..32
///
2021-02-10 16:46:49 +08:00
/// # Return
/// Filtered output y
2021-02-10 16:46:49 +08:00
pub fn update(&mut self, x: i32, k: u8) -> i32 {
2021-02-10 20:44:10 +08:00
debug_assert!((1..32).contains(&k));
// This is an unrolled and optimized first-order IIR loop
// that works for all possible time constants.
2021-02-10 20:44:10 +08:00
// Note the zero(s) at Nyquist.
2021-02-10 16:46:49 +08:00
let mut x0 = x;
let mut x1 = self.xy[0];
2021-02-10 20:31:41 +08:00
self.xy[0] = x0;
for y1 in self.xy[1..].iter_mut() {
2021-02-10 20:44:10 +08:00
x0 = *y1 + (((x0 >> 2) + (x1 >> 2) - (*y1 >> 1) + (1 << (k - 1))) >> k);
x1 = *y1;
*y1 = x0;
}
x0
}
}