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"
|
|
|
|
|
|
|
|
#include <math.h>
|
2016-09-10 13:03:27 -07:00
|
|
|
|
|
|
|
Biquad::Biquad() {
|
2020-08-21 16:47:12 -07:00
|
|
|
// Default to passthru
|
|
|
|
a0 = 1;
|
|
|
|
a1 = a2 = b1 = b2 = 0;
|
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) {
|
|
|
|
return tanf(3.14159f * cutoff / samplingFrequency);
|
|
|
|
}
|
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);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|