2015-08-19 09:15:14 -03:00
|
|
|
//
|
|
|
|
/// @file LowPassFilter.cpp
|
2024-07-27 09:16:20 -03:00
|
|
|
/// @brief A class to implement a low pass filter
|
2015-08-19 09:15:14 -03:00
|
|
|
|
2021-11-29 16:15:49 -04:00
|
|
|
#ifndef HAL_DEBUG_BUILD
|
|
|
|
#define AP_INLINE_VECTOR_OPS
|
|
|
|
#pragma GCC optimize("O2")
|
|
|
|
#endif
|
2015-08-19 09:15:14 -03:00
|
|
|
#include "LowPassFilter.h"
|
2022-12-05 02:18:51 -04:00
|
|
|
#include <AP_InternalError/AP_InternalError.h>
|
2015-08-19 09:15:14 -03:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
2024-07-27 09:16:20 -03:00
|
|
|
// DigitalLPF, base class
|
2015-08-19 09:15:14 -03:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
DigitalLPF<T>::DigitalLPF() {
|
|
|
|
// built in initialization
|
2024-07-27 09:16:20 -03:00
|
|
|
output = T();
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// add a new raw value to the filter, retrieve the filtered result
|
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
T DigitalLPF<T>::_apply(const T &sample, const float &alpha) {
|
|
|
|
output += (sample - output) * alpha;
|
2021-04-03 20:14:57 -03:00
|
|
|
if (!initialised) {
|
|
|
|
initialised = true;
|
2024-07-27 09:16:20 -03:00
|
|
|
output = sample;
|
2016-11-09 20:42:58 -04:00
|
|
|
}
|
2024-07-27 09:16:20 -03:00
|
|
|
return output;
|
2016-11-09 20:42:58 -04:00
|
|
|
}
|
|
|
|
|
2015-08-19 09:15:14 -03:00
|
|
|
// get latest filtered value from filter (equal to the value returned by latest call to apply method)
|
|
|
|
template <class T>
|
|
|
|
const T &DigitalLPF<T>::get() const {
|
2024-07-27 09:16:20 -03:00
|
|
|
return output;
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// Reset filter to given value
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
void DigitalLPF<T>::reset(const T &value) {
|
|
|
|
output = value;
|
2021-04-03 20:14:57 -03:00
|
|
|
initialised = true;
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
2016-11-21 01:48:10 -04:00
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// Set reset flag such that the filter will be reset to the next value applied
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
void DigitalLPF<T>::reset() {
|
|
|
|
initialised = false;
|
|
|
|
}
|
2016-11-21 01:48:10 -04:00
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
template class DigitalLPF<float>;
|
|
|
|
template class DigitalLPF<Vector2f>;
|
|
|
|
template class DigitalLPF<Vector3f>;
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Low pass filter with constant time step
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
2016-11-21 01:48:10 -04:00
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// constructor
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
LowPassFilterConstDt<T>::LowPassFilterConstDt(const float &sample_freq, const float &new_cutoff_freq)
|
2016-11-21 01:48:10 -04:00
|
|
|
{
|
2024-07-27 09:16:20 -03:00
|
|
|
set_cutoff_frequency(sample_freq, new_cutoff_freq);
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// change parameters
|
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
void LowPassFilterConstDt<T>::set_cutoff_frequency(const float &sample_freq, const float &new_cutoff_freq) {
|
|
|
|
cutoff_freq = new_cutoff_freq;
|
|
|
|
|
|
|
|
if (sample_freq <= 0) {
|
|
|
|
alpha = 1;
|
|
|
|
} else {
|
|
|
|
alpha = calc_lowpass_alpha_dt(1.0/sample_freq, cutoff_freq);
|
|
|
|
}
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// return the cutoff frequency
|
2016-11-09 20:42:58 -04:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
float LowPassFilterConstDt<T>::get_cutoff_freq(void) const {
|
|
|
|
return cutoff_freq;
|
2016-11-09 20:42:58 -04:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// add a new raw value to the filter, retrieve the filtered result
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
T LowPassFilterConstDt<T>::apply(const T &sample) {
|
|
|
|
return this->_apply(sample, alpha);
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
template class LowPassFilterConstDt<float>;
|
|
|
|
template class LowPassFilterConstDt<Vector2f>;
|
|
|
|
template class LowPassFilterConstDt<Vector3f>;
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Low pass filter with variable time step
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
LowPassFilter<T>::LowPassFilter(const float &new_cutoff_freq)
|
|
|
|
{
|
|
|
|
set_cutoff_frequency(new_cutoff_freq);
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// change parameters
|
2016-11-09 20:42:58 -04:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
void LowPassFilter<T>::set_cutoff_frequency(const float &new_cutoff_freq) {
|
|
|
|
cutoff_freq = new_cutoff_freq;
|
2016-11-09 20:42:58 -04:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// return the cutoff frequency
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
float LowPassFilter<T>::get_cutoff_freq() const {
|
|
|
|
return cutoff_freq;
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
2024-07-27 09:16:20 -03:00
|
|
|
// add a new raw value to the filter, retrieve the filtered result
|
2015-08-19 09:15:14 -03:00
|
|
|
template <class T>
|
2024-07-27 09:16:20 -03:00
|
|
|
T LowPassFilter<T>::apply(const T &sample, const float &dt) {
|
2024-08-26 07:17:14 -03:00
|
|
|
const float alpha = calc_lowpass_alpha_dt(dt, cutoff_freq);
|
2024-07-27 09:16:20 -03:00
|
|
|
return this->_apply(sample, alpha);
|
2015-08-19 09:15:14 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make an instances
|
|
|
|
* Otherwise we have to move the constructor implementations to the header file :P
|
|
|
|
*/
|
|
|
|
template class LowPassFilter<float>;
|
|
|
|
template class LowPassFilter<Vector2f>;
|
2015-11-24 15:33:52 -04:00
|
|
|
template class LowPassFilter<Vector3f>;
|
|
|
|
|