-
Notifications
You must be signed in to change notification settings - Fork 190
/
ResonantFilter.h
240 lines (193 loc) · 9.53 KB
/
ResonantFilter.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
/*
* ResonantFilter.h
*
* This file is part of Mozzi.
*
* Copyright 2012-2024 Tim Barrass and the Mozzi Team
*
* Mozzi is licensed under the GNU Lesser General Public Licence (LGPL) Version 2.1 or later.
*
*/
#ifndef RESONANTFILTER_H_
#define RESONANTFILTER_H_
#include "IntegerType.h"
#include "AudioOutput.h"
#include "meta.h"
/*
simple resonant filter posted to musicdsp.org by Paul Kellett
http://www.musicdsp.org/archive.php?classid=3#259, applying the
modification from Peter Schoffhauzer to make it able to output
all filter types (LowPass, HighPass, Notch and BandPass).
The generic filter is ResonantFilter<unsigned_t type, FILTER_TYPE>.
- type specifies the type expected for the cutoff and resonance. Only uint8_t and uint16_t have been tested. These are denoted 8bits and 16bits versions of the filter in the following.
- FILTER_TYPE specifies the filter type. LOWPASS, BANDPASS, HIGHPASS and NOTCH are available types.
Two versions are available: the 8bits and 16bits versions (see above).
The 8bits version is an optimized version that uses 8bits values to set
the resonance and the cutoff_freq. It can works on 8bits samples only
on 8bits platforms.
The 16bits version consumes more CPU ressources but uses 16bits values
for resonance and cutoff_freq and can work on samples up to 16bits on
8bits platforms and up to 32 on 32bits platforms.
The filter can be instanciated using the template version ResonantFilter<unsigned_t type, FILTER_TYPE>. For ease of use, the following types are also accepted:
8bits versions: LowPassFilter, HighPassFilter, BandPassFilter, NotchFilter
16bits versions: LowPassFilter16, HighPassFilter16, BandPassFilter16, NotchFilter16
//// ALGORITHM ////
// set feedback amount given f and q between 0 and 1
fb = q + q/(1.0 - f);
In order to avoid a slow division we use the use a Taylor expansion to approximate 1/(1.0 - f):
Close to f=0: 1/(1.0-f) approx 1.0+f.
Hence: fb = q + q * (1.0 + f)
This approximation is less and less valid with an increasing cutoff, leading to a reduction of the resonance of the filter at high cutoff frequencies.
// for each sample...
buf0 = buf0 + f * (in - buf0 + fb * (buf0 - buf1));
buf1 = buf1 + f * (buf0 - buf1);
out = buf1; // LowPass
out = in - buf0; // HighPass
out = buf0 - buf1; // BandPass
out = in - buf0 + buf1; // Notch
fixed point version of the filter
"dave's blog of art and programming" http://www.pawfal.org/dave/blog/2011/09/
*/
enum filter_types { LOWPASS, BANDPASS, HIGHPASS, NOTCH };
/** A generic resonant filter for audio signals.
*/
template<int8_t FILTER_TYPE, typename su=uint8_t>
class ResonantFilter
{
public:
/** Constructor.
*/
ResonantFilter() { ; }
/** deprecated. Use setCutoffFreqAndResonance(su cutoff, su
resonance).
Set the cut off frequency,
@param cutoff use the range 0-255 to represent 0-8191 Hz (MOZZI_AUDIO_RATE/2) for ResonantFilter, cutoff use the range 0-65535 to represent 0-MOZZI_AUDIO_RATE/2.
Be careful of distortion at the lower end, especially with high resonance.
*/
void setCutoffFreq(su cutoff)
{
f = cutoff;
fb = q + ucfxmul(q, (typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type) SHIFTED_1 + cutoff);
}
/** deprecated. Use setCutoffFreqAndResonance(su cutoff, su
resonance).
Set the resonance. If you hear unwanted distortion, back off the resonance.
After setting resonance, you need to call setCuttoffFreq() to hear the change!
@param resonance in the range 0-255 for ResonantFilter, 0-65535 for ResonantFilter<FILTER_TYPE, uint16_t>, with 255/65535 being most resonant
@note Remember to call setCuttoffFreq() after resonance is changed!
*/
void setResonance(su resonance) { q = resonance; }
/**
Set the cut off frequency and resonance. Replaces setCutoffFreq() and
setResonance(). (Because the internal calculations need to be done whenever either parameter changes.)
@param cutoff range 0-255 represents 0-8191 Hz (MOZZI_AUDIO_RATE/2) for ResonantFilter, range 0-65535 for ResonantFilter16
Be careful of distortion at the lower end, especially with high resonance.
@param resonance range 0-255 for ResonantFilter, 0-65535 for ResonantFilter<FILTER_TYPE, uint16_t>, 255/65535 is most resonant.
*/
void setCutoffFreqAndResonance(su cutoff, su resonance)
{
f = cutoff;
q = resonance; // hopefully optimised away when compiled, just here for
// backwards compatibility
fb = q + ucfxmul(q,(typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type) SHIFTED_1 + cutoff);
}
/** Calculate the next sample, given an input signal.
@param in the signal input. Should not be more than 8bits on 8bits platforms (Arduino) if using the 8bits version and not 16bits version.
@return the signal output.
@note Timing: about 11us.
*/
// 10.5 to 12.5 us, mostly 10.5 us (was 14us)
inline AudioOutputStorage_t next(AudioOutputStorage_t in)
{
advanceBuffers(in);
return current(in, Int2Type<FILTER_TYPE>());
}
protected:
su q;
su f;
typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type fb;
AudioOutputStorage_t buf0, buf1;
const uint8_t FX_SHIFT = sizeof(su) << 3;
const uint8_t FX_SHIFT_M_1 = FX_SHIFT-1;
const su SHIFTED_1 = (1<<FX_SHIFT)-1;
// // multiply two fixed point numbers (returns fixed point)
// inline
// long fxmul(long a, long b)
// {
// return (a*b)>>FX_SHIFT;
// }
inline void advanceBuffers(AudioOutputStorage_t in)
{
buf0 += fxmul(((in - buf0) + fxmul(fb, buf0 - buf1)), f);
buf1 += ifxmul(buf0 - buf1, f); // could overflow if input changes fast
}
inline AudioOutputStorage_t current(AudioOutputStorage_t in, Int2Type<LOWPASS>) {return buf1;}
inline AudioOutputStorage_t current(AudioOutputStorage_t in, Int2Type<HIGHPASS>) {return in - buf0;}
inline AudioOutputStorage_t current(AudioOutputStorage_t in, Int2Type<BANDPASS>) {return buf0-buf1;}
inline AudioOutputStorage_t current(AudioOutputStorage_t in, Int2Type<NOTCH>) {return in - buf0 + buf1;}
// multiply two fixed point numbers (returns fixed point)
inline typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type ucfxmul(su a, typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type b)
{
return (((typename IntegerType<sizeof(su)+sizeof(su)>::unsigned_type)a * (b >> 1)) >> (FX_SHIFT_M_1));
}
// multiply two fixed point numbers (returns fixed point)
inline typename IntegerType<sizeof(AudioOutputStorage_t)+sizeof(su)-1>::signed_type ifxmul(typename IntegerType<sizeof(AudioOutputStorage_t )+sizeof(su)-1>::signed_type a, su b) { return ((a * b) >> FX_SHIFT); }
// multiply two fixed point numbers (returns fixed point)
inline typename IntegerType<sizeof(AudioOutputStorage_t)+sizeof(AudioOutputStorage_t)>::signed_type fxmul(typename IntegerType<sizeof(AudioOutputStorage_t)+sizeof(AudioOutputStorage_t)>::signed_type a, typename IntegerType<sizeof(AudioOutputStorage_t)+sizeof(su)-1>::signed_type b) { return ((a * b) >> FX_SHIFT); }
};
/** A generic filter for audio signals that can produce lowpass, highpass, bandpass and notch outputs at runtime.
Behaves like ResonantFilter for setting the resonance and cutoff frequency.
Like ResonantFilter, it can be used on different sample sizes: MultiResonantFilter<uint8_t> and MultiResonantFilter<uint16_t> have been tested.
For the former, both cutoff and resonance are uint8_t, hence between 0-255.
For the later, both cutoff and resonance are uint16_t, hence between 0-65535.
*/
template<typename su=uint8_t>
class MultiResonantFilter: public ResonantFilter<LOWPASS,su>
{
public:
/** Compute the filters, given an input signal.
@param in the signal input. Should not be more than 8bits on 8bits platforms (Arduino) if using the 8bits version and not 16bits version.
*/
inline void next (AudioOutputStorage_t in)
{
last_in = in;
ResonantFilter<LOWPASS,su>::advanceBuffers(in);
}
/** Return the input filtered with a lowpass filter
@return the filtered signal output.
*/
inline AudioOutputStorage_t low() {return ResonantFilter<LOWPASS,su>::current(last_in,Int2Type<LOWPASS>());}
/** Return the input filtered with a highpass filter
@return the filtered signal output.
*/
inline AudioOutputStorage_t high() {return ResonantFilter<LOWPASS,su>::current(last_in,Int2Type<HIGHPASS>());}
/** Return the input filtered with a bandpass filter
@return the filtered signal output.
*/
inline AudioOutputStorage_t band() {return ResonantFilter<LOWPASS,su>::current(last_in,Int2Type<BANDPASS>());}
/** Return the input filtered with a notch filter
@return the filtered signal output.
*/
inline AudioOutputStorage_t notch() {return ResonantFilter<LOWPASS,su>::current(last_in,Int2Type<NOTCH>());}
private:
AudioOutputStorage_t last_in;
};
typedef ResonantFilter<LOWPASS> LowPassFilter;
typedef ResonantFilter<LOWPASS, uint16_t> LowPassFilter16;
/*
typedef ResonantFilter<uint8_t, HIGHPASS> HighPassFilter;
typedef ResonantFilter<uint16_t, HIGHPASS> HighPassFilter16;
typedef ResonantFilter<uint8_t, BANDPASS> BandPassFilter;
typedef ResonantFilter<uint16_t, BANDPASS> BandPassFilter16;
typedef ResonantFilter<uint8_t, NOTCH> NotchFilter;
typedef ResonantFilter<uint16_t, NOTCH> NotchFilter16;
*/
/**
@example 10.Audio_Filters/ResonantFilter/ResonantFilter.ino
This example demonstrates the ResonantFilter specification of this class.
@example 10.Audio_Filters/ResonantFilter16/ResonantFilter16.ino
This example demonstrates the ResonantFilter16 specification of this class.
@example 10.Audio_Filters/MultiResonantFilter/MultiResonantFilter.ino
This example demonstrates the MultiResonantFilter specification of this class.
*/
#endif /* RESONANTFILTER_H_ */