/* * @file biquad.cpp * * @date Sep 10, 2016 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "biquad.h" #include "error_handling.h" #include Biquad::Biquad() { // Default to passthru a0 = 1; a1 = a2 = b1 = b2 = 0; z1 = z2 = 0; } static float getK(float samplingFrequency, float cutoff) { return tanf(3.14159f * cutoff / samplingFrequency); } static float getNorm(float K, float Q) { return 1 / (1 + K / Q + K * K); } void Biquad::configureBandpass(float samplingFrequency, float centerFrequency, float Q) { efiAssertVoid(OBD_PCM_Processor_Fault, samplingFrequency >= 2 * centerFrequency, "Invalid biquad parameters"); float K = getK(samplingFrequency, centerFrequency); float norm = getNorm(K, Q); a0 = K / Q * norm; a1 = 0; a2 = -a0; b1 = 2 * (K * K - 1) * norm; b2 = (1 - K / Q + K * K) * norm; } void Biquad::configureLowpass(float samplingFrequency, float cutoffFrequency, float Q) { efiAssertVoid(OBD_PCM_Processor_Fault, samplingFrequency >= 2 * cutoffFrequency, "Invalid biquad parameters"); float K = getK(samplingFrequency, cutoffFrequency); float norm = getNorm(K, Q); norm = 1 / (1 + K / Q + K * K); a0 = K * K * norm; a1 = 2 * a0; a2 = a0; b1 = 2 * (K * K - 1) * norm; b2 = (1 - K / Q + K * K) * norm; } float Biquad::filter(float input) { float result = input * a0 + z1; z1 = input * a1 + z2 - b1 * result; z2 = input * a2 - b2 * result; return result; }