2017-08-16 02:27:39 -03:00
|
|
|
/*
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2021-11-29 16:15:49 -04:00
|
|
|
#ifndef HAL_DEBUG_BUILD
|
|
|
|
#define AP_INLINE_VECTOR_OPS
|
|
|
|
#pragma GCC optimize("O2")
|
|
|
|
#endif
|
|
|
|
|
2017-08-16 02:27:39 -03:00
|
|
|
#include "NotchFilter.h"
|
|
|
|
|
2022-09-17 11:53:41 -03:00
|
|
|
const static float NOTCH_MAX_SLEW = 0.05f;
|
|
|
|
const static float NOTCH_MAX_SLEW_LOWER = 1.0f - NOTCH_MAX_SLEW;
|
|
|
|
const static float NOTCH_MAX_SLEW_UPPER = 1.0f / NOTCH_MAX_SLEW_LOWER;
|
|
|
|
|
2019-06-17 05:43:08 -03:00
|
|
|
/*
|
|
|
|
calculate the attenuation and quality factors of the filter
|
|
|
|
*/
|
|
|
|
template <class T>
|
|
|
|
void NotchFilter<T>::calculate_A_and_Q(float center_freq_hz, float bandwidth_hz, float attenuation_dB, float& A, float& Q) {
|
|
|
|
A = powf(10, -attenuation_dB / 40.0f);
|
2019-08-31 22:25:36 -03:00
|
|
|
if (center_freq_hz > 0.5 * bandwidth_hz) {
|
|
|
|
const float octaves = log2f(center_freq_hz / (center_freq_hz - bandwidth_hz / 2.0f)) * 2.0f;
|
|
|
|
Q = sqrtf(powf(2, octaves)) / (powf(2, octaves) - 1.0f);
|
|
|
|
} else {
|
|
|
|
Q = 0.0;
|
|
|
|
}
|
2019-06-17 05:43:08 -03:00
|
|
|
}
|
|
|
|
|
2017-08-16 02:27:39 -03:00
|
|
|
/*
|
|
|
|
initialise filter
|
|
|
|
*/
|
|
|
|
template <class T>
|
|
|
|
void NotchFilter<T>::init(float sample_freq_hz, float center_freq_hz, float bandwidth_hz, float attenuation_dB)
|
2019-06-17 05:43:08 -03:00
|
|
|
{
|
2019-08-31 22:25:36 -03:00
|
|
|
// check center frequency is in the allowable range
|
|
|
|
if ((center_freq_hz > 0.5 * bandwidth_hz) && (center_freq_hz < 0.5 * sample_freq_hz)) {
|
|
|
|
float A, Q;
|
2022-09-21 06:45:42 -03:00
|
|
|
initialised = false; // force center frequency change
|
2019-08-31 22:25:36 -03:00
|
|
|
calculate_A_and_Q(center_freq_hz, bandwidth_hz, attenuation_dB, A, Q);
|
|
|
|
init_with_A_and_Q(sample_freq_hz, center_freq_hz, A, Q);
|
|
|
|
} else {
|
|
|
|
initialised = false;
|
|
|
|
}
|
2019-06-17 05:43:08 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void NotchFilter<T>::init_with_A_and_Q(float sample_freq_hz, float center_freq_hz, float A, float Q)
|
2017-08-16 02:27:39 -03:00
|
|
|
{
|
2022-09-21 06:45:42 -03:00
|
|
|
// don't update if no updates required
|
|
|
|
if (initialised && is_equal(center_freq_hz, _center_freq_hz) && is_equal(sample_freq_hz, _sample_freq_hz)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-09-17 11:53:41 -03:00
|
|
|
float new_center_freq = center_freq_hz;
|
|
|
|
|
|
|
|
// constrain the new center frequency by a percentage of the old frequency
|
|
|
|
if (initialised && !need_reset && !is_zero(_center_freq_hz)) {
|
|
|
|
new_center_freq = constrain_float(new_center_freq, _center_freq_hz * NOTCH_MAX_SLEW_LOWER,
|
|
|
|
_center_freq_hz * NOTCH_MAX_SLEW_UPPER);
|
|
|
|
}
|
|
|
|
|
2023-11-03 14:05:23 -03:00
|
|
|
if (is_positive(new_center_freq) && (new_center_freq < 0.5 * sample_freq_hz) && (Q > 0.0)) {
|
2022-09-17 11:53:41 -03:00
|
|
|
float omega = 2.0 * M_PI * new_center_freq / sample_freq_hz;
|
2019-08-31 22:25:36 -03:00
|
|
|
float alpha = sinf(omega) / (2 * Q);
|
|
|
|
b0 = 1.0 + alpha*sq(A);
|
|
|
|
b1 = -2.0 * cosf(omega);
|
|
|
|
b2 = 1.0 - alpha*sq(A);
|
|
|
|
a1 = b1;
|
|
|
|
a2 = 1.0 - alpha;
|
2023-07-07 22:28:32 -03:00
|
|
|
|
|
|
|
const float a0_inv = 1.0/(1.0 + alpha);
|
|
|
|
|
|
|
|
// Pre-multiply to save runtime calc
|
|
|
|
b0 *= a0_inv;
|
|
|
|
b1 *= a0_inv;
|
|
|
|
b2 *= a0_inv;
|
|
|
|
a1 *= a0_inv;
|
|
|
|
a2 *= a0_inv;
|
|
|
|
|
2022-09-17 11:53:41 -03:00
|
|
|
_center_freq_hz = new_center_freq;
|
2022-09-21 06:45:42 -03:00
|
|
|
_sample_freq_hz = sample_freq_hz;
|
2019-08-31 22:25:36 -03:00
|
|
|
initialised = true;
|
|
|
|
} else {
|
2022-09-17 11:53:41 -03:00
|
|
|
// leave center_freq_hz at last value
|
2019-08-31 22:25:36 -03:00
|
|
|
initialised = false;
|
|
|
|
}
|
2017-08-16 02:27:39 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
apply a new input sample, returning new output
|
|
|
|
*/
|
|
|
|
template <class T>
|
|
|
|
T NotchFilter<T>::apply(const T &sample)
|
|
|
|
{
|
2022-06-05 20:40:34 -03:00
|
|
|
if (!initialised || need_reset) {
|
2017-08-16 02:27:39 -03:00
|
|
|
// if we have not been initialised when return the input
|
2019-08-31 22:25:36 -03:00
|
|
|
// sample as output and update delayed samples
|
|
|
|
signal1 = sample;
|
2022-06-11 07:36:01 -03:00
|
|
|
signal2 = sample;
|
|
|
|
ntchsig1 = sample;
|
|
|
|
ntchsig2 = sample;
|
2022-06-05 20:40:34 -03:00
|
|
|
need_reset = false;
|
2017-08-16 02:27:39 -03:00
|
|
|
return sample;
|
|
|
|
}
|
2023-07-07 22:28:32 -03:00
|
|
|
|
|
|
|
T output = sample*b0 + ntchsig1*b1 + ntchsig2*b2 - signal1*a1 - signal2*a2;
|
|
|
|
|
2017-08-16 02:27:39 -03:00
|
|
|
ntchsig2 = ntchsig1;
|
2023-07-07 22:28:32 -03:00
|
|
|
ntchsig1 = sample;
|
|
|
|
|
2017-08-16 02:27:39 -03:00
|
|
|
signal2 = signal1;
|
|
|
|
signal1 = output;
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2019-05-17 12:59:54 -03:00
|
|
|
template <class T>
|
|
|
|
void NotchFilter<T>::reset()
|
|
|
|
{
|
2022-06-05 20:40:34 -03:00
|
|
|
need_reset = true;
|
2019-05-17 12:59:54 -03:00
|
|
|
}
|
|
|
|
|
2017-08-16 02:27:39 -03:00
|
|
|
/*
|
|
|
|
instantiate template classes
|
|
|
|
*/
|
|
|
|
template class NotchFilter<float>;
|
2021-06-17 03:36:03 -03:00
|
|
|
template class NotchFilter<Vector2f>;
|
2017-08-16 02:27:39 -03:00
|
|
|
template class NotchFilter<Vector3f>;
|