2019-06-17 05:43:08 -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/>.
*/
# include "HarmonicNotchFilter.h"
2019-08-29 04:13:59 -03:00
# include <GCS_MAVLink/GCS.h>
2019-06-17 05:43:08 -03:00
2019-12-27 08:06:21 -04:00
# define HNF_MAX_FILTERS 6 // must be even for double-notch filters
# define HNF_MAX_HARMONICS 8
2019-06-17 05:43:08 -03:00
// table of user settable parameters
const AP_Param : : GroupInfo HarmonicNotchFilterParams : : var_info [ ] = {
// @Param: ENABLE
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter enable
// @Description: Harmonic Notch Filter enable
2019-06-17 05:43:08 -03:00
// @Values: 0:Disabled,1:Enabled
// @User: Advanced
AP_GROUPINFO_FLAGS ( " ENABLE " , 1 , HarmonicNotchFilterParams , _enable , 0 , AP_PARAM_FLAG_ENABLE ) ,
// @Param: FREQ
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter base frequency
2020-11-21 16:36:11 -04:00
// @Description: Harmonic Notch Filter base center frequency in Hz. This should be set at most half the backend gyro rate (which is typically 1Khz). For helicopters using RPM sensor to dynamically set the notch frequency, use this parameter to provide a lower limit to the dynamic notch filter. Recommend setting it to half the operating rotor speed in Hz.
// @Range: 10 495
2019-06-17 05:43:08 -03:00
// @Units: Hz
// @User: Advanced
AP_GROUPINFO ( " FREQ " , 2 , HarmonicNotchFilterParams , _center_freq_hz , 80 ) ,
// @Param: BW
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter bandwidth
2020-11-21 16:36:11 -04:00
// @Description: Harmonic Notch Filter bandwidth in Hz. This is typically set to half the base frequency. The ratio of base frequency to bandwidth determines the notch quality factor and is fixed across harmonics.
// @Range: 5 250
2019-06-17 05:43:08 -03:00
// @Units: Hz
// @User: Advanced
2020-11-21 16:36:11 -04:00
AP_GROUPINFO ( " BW " , 3 , HarmonicNotchFilterParams , _bandwidth_hz , 40 ) ,
2019-06-17 05:43:08 -03:00
// @Param: ATT
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter attenuation
2020-11-23 13:11:39 -04:00
// @Description: Harmonic Notch Filter attenuation in dB. Values greater than 40dB will typically produce a hard notch rather than a modest attenuation of motor noise.
2020-11-21 16:36:11 -04:00
// @Range: 5 50
2019-06-17 05:43:08 -03:00
// @Units: dB
// @User: Advanced
2020-11-21 16:36:11 -04:00
AP_GROUPINFO ( " ATT " , 4 , HarmonicNotchFilterParams , _attenuation_dB , 40 ) ,
2019-06-17 05:43:08 -03:00
// @Param: HMNCS
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter harmonics
2019-09-11 07:46:19 -03:00
// @Description: Bitmask of harmonic frequencies to apply Harmonic Notch Filter to. This option takes effect on the next reboot. A maximum of 3 harmonics can be used at any one time.
2019-06-17 05:43:08 -03:00
// @Bitmask: 0:1st harmonic,1:2nd harmonic,2:3rd harmonic,3:4th hamronic,4:5th harmonic,5:6th harmonic,6:7th harmonic,7:8th harmonic
// @User: Advanced
// @RebootRequired: True
AP_GROUPINFO ( " HMNCS " , 5 , HarmonicNotchFilterParams , _harmonics , 3 ) ,
// @Param: REF
2019-08-28 17:18:00 -03:00
// @DisplayName: Harmonic Notch Filter reference value
2019-10-10 14:36:09 -03:00
// @Description: A reference value of zero disables dynamic updates on the Harmonic Notch Filter and a positive value enables dynamic updates on the Harmonic Notch Filter. For throttle-based scaling, this parameter is the reference value associated with the specified frequency to facilitate frequency scaling of the Harmonic Notch Filter. For RPM and ESC telemetry based tracking, this parameter is set to 1 to enable the Harmonic Notch Filter using the RPM sensor or ESC telemetry set to measure rotor speed. The sensor data is converted to Hz automatically for use in the Harmonic Notch Filter. This reference value may also be used to scale the sensor data, if required. For example, rpm sensor data is required to measure heli motor RPM. Therefore the reference value can be used to scale the RPM sensor to the rotor RPM.
2019-06-17 05:43:08 -03:00
// @User: Advanced
2019-10-10 14:36:09 -03:00
// @Range: 0.0 1.0
2019-06-17 05:43:08 -03:00
// @RebootRequired: True
2019-09-11 07:46:19 -03:00
AP_GROUPINFO ( " REF " , 6 , HarmonicNotchFilterParams , _reference , 0 ) ,
2019-06-17 05:43:08 -03:00
2019-10-10 14:36:09 -03:00
// @Param: MODE
// @DisplayName: Harmonic Notch Filter dynamic frequency tracking mode
2019-10-10 14:36:09 -03:00
// @Description: Harmonic Notch Filter dynamic frequency tracking mode. Dynamic updates can be throttle, RPM sensor, ESC telemetry or dynamic FFT based. Throttle-based updates should only be used with multicopters.
// @Range: 0 4
// @Values: 0:Disabled,1:Throttle,2:RPM Sensor,3:ESC Telemetry,4:Dynamic FFT
2019-10-10 14:36:09 -03:00
// @User: Advanced
AP_GROUPINFO ( " MODE " , 7 , HarmonicNotchFilterParams , _tracking_mode , 1 ) ,
2019-12-27 08:06:21 -04:00
// @Param: OPTS
// @DisplayName: Harmonic Notch Filter options
2021-05-12 13:33:11 -03:00
// @Description: Harmonic Notch Filter options. Double-notches can provide deeper attenuation across a wider bandwidth than single notches and are suitable for larger aircraft. Dynamic harmonics attaches a harmonic notch to each detected noise frequency instead of simply being multiples of the base frequency, in the case of FFT it will attach notches to each of three detected noise peaks, in the case of ESC it will attach notches to each of four motor RPM values. Loop rate update changes the notch center frequency at the scheduler loop rate rather than at the default of 200Hz.
// @Bitmask: 0:Double notch,1:Dynamic harmonic,2:Update at loop rate
2019-12-27 08:06:21 -04:00
// @User: Advanced
// @RebootRequired: True
AP_GROUPINFO ( " OPTS " , 8 , HarmonicNotchFilterParams , _options , 0 ) ,
2019-06-17 05:43:08 -03:00
AP_GROUPEND
} ;
2019-08-28 17:18:00 -03:00
/*
destroy all of the associated notch filters
*/
template < class T >
HarmonicNotchFilter < T > : : ~ HarmonicNotchFilter ( ) {
delete [ ] _filters ;
_num_filters = 0 ;
2019-08-30 04:34:36 -03:00
_num_enabled_filters = 0 ;
2019-08-28 17:18:00 -03:00
}
2019-06-17 05:43:08 -03:00
/*
2019-08-28 17:18:00 -03:00
initialise the associated filters with the provided shaping constraints
the constraints are used to determine attenuation ( A ) and quality ( Q ) factors for the filter
2019-06-17 05:43:08 -03:00
*/
template < class T >
void HarmonicNotchFilter < T > : : init ( float sample_freq_hz , float center_freq_hz , float bandwidth_hz , float attenuation_dB )
{
2019-08-30 04:34:36 -03:00
// sanity check the input
if ( _filters = = nullptr | | is_zero ( sample_freq_hz ) | | isnan ( sample_freq_hz ) ) {
2019-06-17 05:43:08 -03:00
return ;
}
_sample_freq_hz = sample_freq_hz ;
2019-12-27 08:06:21 -04:00
const float nyquist_limit = sample_freq_hz * 0.48f ;
const float bandwidth_limit = bandwidth_hz * 0.52f ;
2019-08-30 04:34:36 -03:00
// adjust the fundamental center frequency to be in the allowable range
2019-12-27 08:06:21 -04:00
center_freq_hz = constrain_float ( center_freq_hz , bandwidth_limit , nyquist_limit ) ;
2020-04-14 05:51:58 -03:00
// Calculate spread required to achieve an equivalent single notch using two notches with Bandwidth/2
_notch_spread = bandwidth_hz / ( 32 * center_freq_hz ) ;
2019-12-27 08:06:21 -04:00
if ( _double_notch ) {
// position the individual notches so that the attenuation is no worse than a single notch
// calculate attenuation and quality from the shaping constraints
NotchFilter < T > : : calculate_A_and_Q ( center_freq_hz , bandwidth_hz * 0.5 , attenuation_dB , _A , _Q ) ;
} else {
// calculate attenuation and quality from the shaping constraints
NotchFilter < T > : : calculate_A_and_Q ( center_freq_hz , bandwidth_hz , attenuation_dB , _A , _Q ) ;
2019-06-17 05:43:08 -03:00
}
2019-12-27 08:06:21 -04:00
2019-06-17 05:43:08 -03:00
_initialised = true ;
2019-12-27 08:06:21 -04:00
update ( center_freq_hz ) ;
2019-06-17 05:43:08 -03:00
}
/*
2019-08-28 17:18:00 -03:00
allocate a collection of , at most HNF_MAX_FILTERS , notch filters to be managed by this harmonic notch filter
2019-06-17 05:43:08 -03:00
*/
template < class T >
2019-12-27 08:06:21 -04:00
void HarmonicNotchFilter < T > : : allocate_filters ( uint8_t harmonics , bool double_notch )
2019-06-17 05:43:08 -03:00
{
2019-12-27 08:06:21 -04:00
_double_notch = double_notch ;
2019-06-17 05:43:08 -03:00
for ( uint8_t i = 0 ; i < HNF_MAX_HARMONICS & & _num_filters < HNF_MAX_FILTERS ; i + + ) {
if ( ( 1U < < i ) & harmonics ) {
_num_filters + + ;
2019-12-27 08:06:21 -04:00
if ( _double_notch ) {
_num_filters + + ;
}
2019-06-17 05:43:08 -03:00
}
}
if ( _num_filters > 0 ) {
_filters = new NotchFilter < T > [ _num_filters ] ;
2019-08-29 04:13:59 -03:00
if ( _filters = = nullptr ) {
2020-03-29 19:49:54 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " Failed to allocate %u bytes for HarmonicNotchFilter " , ( unsigned int ) ( _num_filters * sizeof ( NotchFilter < T > ) ) ) ;
2019-08-29 04:13:59 -03:00
_num_filters = 0 ;
}
2019-06-17 05:43:08 -03:00
}
_harmonics = harmonics ;
}
2019-08-28 17:18:00 -03:00
/*
update the underlying filters ' center frequency using the current attenuation and quality
this function is cheaper than init ( ) because A & Q do not need to be recalculated
*/
2019-06-17 05:43:08 -03:00
template < class T >
void HarmonicNotchFilter < T > : : update ( float center_freq_hz )
{
if ( ! _initialised ) {
return ;
}
2019-08-30 04:34:36 -03:00
// adjust the fundamental center frequency to be in the allowable range
const float nyquist_limit = _sample_freq_hz * 0.48f ;
center_freq_hz = constrain_float ( center_freq_hz , 1.0f , nyquist_limit ) ;
2019-06-17 05:43:08 -03:00
2019-08-30 04:34:36 -03:00
_num_enabled_filters = 0 ;
2019-08-28 17:18:00 -03:00
// update all of the filters using the new center frequency and existing A & Q
2020-05-29 13:28:43 -03:00
for ( uint8_t i = 0 ; i < HNF_MAX_HARMONICS & & _num_enabled_filters < _num_filters ; i + + ) {
2019-06-17 05:43:08 -03:00
if ( ( 1U < < i ) & _harmonics ) {
2019-12-27 08:06:21 -04:00
const float notch_center = center_freq_hz * ( i + 1 ) ;
if ( ! _double_notch ) {
// only enable the filter if its center frequency is below the nyquist frequency
if ( notch_center < nyquist_limit ) {
2020-05-29 13:28:43 -03:00
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center , _A , _Q ) ;
2019-12-27 08:06:21 -04:00
}
} else {
float notch_center_double ;
// only enable the filter if its center frequency is below the nyquist frequency
notch_center_double = notch_center * ( 1.0 - _notch_spread ) ;
if ( notch_center_double < nyquist_limit ) {
2020-05-29 13:28:43 -03:00
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center_double , _A , _Q ) ;
2019-12-27 08:06:21 -04:00
}
// only enable the filter if its center frequency is below the nyquist frequency
notch_center_double = notch_center * ( 1.0 + _notch_spread ) ;
if ( notch_center_double < nyquist_limit ) {
2020-05-29 13:28:43 -03:00
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center_double , _A , _Q ) ;
2019-12-27 08:06:21 -04:00
}
2019-08-30 04:34:36 -03:00
}
2020-05-29 13:28:43 -03:00
}
}
}
/*
update the underlying filters ' center frequency using the current attenuation and quality
this function is cheaper than init ( ) because A & Q do not need to be recalculated
*/
template < class T >
void HarmonicNotchFilter < T > : : update ( uint8_t num_centers , const float center_freq_hz [ ] )
{
if ( ! _initialised ) {
return ;
}
// adjust the frequencies to be in the allowable range
const float nyquist_limit = _sample_freq_hz * 0.48f ;
_num_enabled_filters = 0 ;
// update all of the filters using the new center frequencies and existing A & Q
for ( uint8_t i = 0 ; i < HNF_MAX_HARMONICS & & i < num_centers & & _num_enabled_filters < _num_filters ; i + + ) {
const float notch_center = constrain_float ( center_freq_hz [ i ] , 1.0f , nyquist_limit ) ;
if ( ! _double_notch ) {
// only enable the filter if its center frequency is below the nyquist frequency
if ( notch_center < nyquist_limit ) {
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center , _A , _Q ) ;
}
} else {
float notch_center_double ;
// only enable the filter if its center frequency is below the nyquist frequency
notch_center_double = notch_center * ( 1.0 - _notch_spread ) ;
if ( notch_center_double < nyquist_limit ) {
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center_double , _A , _Q ) ;
}
// only enable the filter if its center frequency is below the nyquist frequency
notch_center_double = notch_center * ( 1.0 + _notch_spread ) ;
if ( notch_center_double < nyquist_limit ) {
_filters [ _num_enabled_filters + + ] . init_with_A_and_Q ( _sample_freq_hz , notch_center_double , _A , _Q ) ;
}
2019-06-17 05:43:08 -03:00
}
}
}
2019-08-28 17:18:00 -03:00
/*
apply a sample to each of the underlying filters in turn and return the output
*/
2019-06-17 05:43:08 -03:00
template < class T >
T HarmonicNotchFilter < T > : : apply ( const T & sample )
{
if ( ! _initialised ) {
return sample ;
}
T output = sample ;
2019-08-30 04:34:36 -03:00
for ( uint8_t i = 0 ; i < _num_enabled_filters ; i + + ) {
2019-06-17 05:43:08 -03:00
output = _filters [ i ] . apply ( output ) ;
}
return output ;
}
2019-08-28 17:18:00 -03:00
/*
reset all of the underlying filters
*/
2019-06-17 05:43:08 -03:00
template < class T >
void HarmonicNotchFilter < T > : : reset ( )
{
if ( ! _initialised ) {
return ;
}
for ( uint8_t i = 0 ; i < _num_filters ; i + + ) {
_filters [ i ] . reset ( ) ;
}
}
/*
2019-08-28 17:18:00 -03:00
create parameters for the harmonic notch filter and initialise defaults
2019-06-17 05:43:08 -03:00
*/
HarmonicNotchFilterParams : : HarmonicNotchFilterParams ( void )
{
AP_Param : : setup_object_defaults ( this , var_info ) ;
}
/*
2019-08-28 17:18:00 -03:00
instantiate template classes
2019-06-17 05:43:08 -03:00
*/
template class HarmonicNotchFilter < Vector3f > ;