2016-09-10 13:03:27 -07:00
|
|
|
/*
|
|
|
|
* @file biquad.cpp
|
|
|
|
*
|
|
|
|
* @date Sep 10, 2016
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2016-09-10 13:03:27 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "biquad.h"
|
2020-08-21 16:47:12 -07:00
|
|
|
#include "error_handling.h"
|
|
|
|
|
2020-11-22 05:26:17 -08:00
|
|
|
#include "efilib.h"
|
2016-09-10 13:03:27 -07:00
|
|
|
|
|
|
|
Biquad::Biquad() {
|
2020-08-28 18:13:50 -07:00
|
|
|
// Default to passthru
|
2020-08-21 16:47:12 -07:00
|
|
|
a0 = 1;
|
|
|
|
a1 = a2 = b1 = b2 = 0;
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Biquad::reset() {
|
2016-09-17 12:02:55 -07:00
|
|
|
z1 = z2 = 0;
|
|
|
|
}
|
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
static float getK(float samplingFrequency, float cutoff) {
|
2020-11-22 05:26:17 -08:00
|
|
|
return tanf_taylor(3.14159f * cutoff / samplingFrequency);
|
2020-08-21 16:47:12 -07:00
|
|
|
}
|
2016-09-10 13:03:27 -07:00
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
static float getNorm(float K, float Q) {
|
|
|
|
return 1 / (1 + K / Q + K * K);
|
2016-09-10 19:03:38 -07:00
|
|
|
}
|
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
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;
|
2016-09-10 13:03:27 -07:00
|
|
|
}
|
|
|
|
|
2020-08-21 16:47:12 -07:00
|
|
|
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);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2020-08-28 18:13:50 -07:00
|
|
|
|
|
|
|
void Biquad::cookSteadyState(float steadyStateInput) {
|
|
|
|
float Y = steadyStateInput * (a0 + a1 + a2) / (1 + b1 + b2);
|
|
|
|
|
|
|
|
float steady_z2 = steadyStateInput * a2 - Y * b2;
|
|
|
|
float steady_z1 = steady_z2 + steadyStateInput * a1 - Y * b1;
|
|
|
|
|
|
|
|
this->z1 = steady_z1;
|
|
|
|
this->z2 = steady_z2;
|
|
|
|
}
|