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
241
242
243
244
245
246
247
248
249
|
#ifndef INCLUDE_FILTER_H
#define INCLUDE_FILTER_H
/*
* This is the Loris C++ Class Library, implementing analysis,
* manipulation, and synthesis of digitized sounds using the Reassigned
* Bandwidth-Enhanced Additive Sound Model.
*
* Loris is Copyright (c) 1999-2010 by Kelly Fitz and Lippold Haken
*
* 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 2 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*
* Filter.h
*
* Definition of class Loris::Filter, a generic digital filter of
* arbitrary order having both feed-forward and feedback coefficients.
*
* Kelly Fitz, 1 Sept 1999
* revised 9 Oct 2009
* loris@cerlsoundgroup.org
*
* http://www.cerlsoundgroup.org/Loris/
*
*/
#include "LorisExceptions.h"
#include "Notifier.h"
#include <algorithm>
#include <deque>
#include <vector>
// begin namespace
namespace Loris {
// ---------------------------------------------------------------------------
// class Filter
//
//! Filter is an Direct Form II realization of a filter specified
//! by its difference equation coefficients and (optionally) gain,
//! applied to the filter output (defaults to 1.). Coefficients are
//! specified and stored in order of increasing delay.
//!
//! Implements the rational transfer function
//!
//! -1 -nb
//! b[0] + b[1]z + ... + b[nb] z
//! Y(z) = G ---------------------------------- X(z)
//! -1 -na
//! a[0] + a[1]z + ... + a[na] z
//!
//! where b[k] are the feed forward coefficients, and a[k] are the feedback
//! coefficients. If a[0] is not 1, then both a and b are normalized by a[0].
//! G is the additional filter gain, and is unity if unspecified.
//!
//!
//! Filter is implemented using a std::deque to store the filter state,
//! and relies on the efficiency of that class. If deque is not implemented
//! using some sort of circular buffer (as it should be -- deque is guaranteed
//! to be efficient for repeated insertion and removal at both ends), then
//! this filter class will be slow.
//
class Filter
{
public:
// --- lifecycle ---
// default construction
//! Construct a filter with an all-pass unity gain response.
Filter( void );
// initialized construction
//! Initialize a Filter having the specified coefficients, and
//! order equal to the larger of the two coefficient ranges.
//! Coefficients in the sequences are stored in increasing order
//! (lowest order coefficient first).
//!
//! If template members are allowed, then the coefficients
//! can be stored in any kind of iterator range, otherwise,
//! they must be in an array of doubles.
//!
//! \param ffwdbegin is the beginning of a sequence of feed-forward coefficients
//! \param ffwdend is the end of a sequence of feed-forward coefficients
//! \param fbackbegin is the beginning of a sequence of feedback coefficients
//! \param fbackend is the end of a sequence of feedback coefficients
//! \param gain is an optional gain scale applied to the filtered signal
//
#if !defined(NO_TEMPLATE_MEMBERS)
template<typename IterT1, typename IterT2>
Filter( IterT1 ffwdbegin, IterT1 ffwdend, // feed-forward coeffs
IterT2 fbackbegin, IterT2 fbackend, // feedback coeffs
double gain = 1. );
#else
Filter( const double * ffwdbegin, const double * ffwdend, // feed-forward coeffs
const double * fbackbegin, const double * fbackend, // feedback coeffs
double gain = 1. );
#endif
// copy constructor
//! Make a copy of another digital filter.
//! Do not copy the filter state (delay line).
Filter( const Filter & other );
// assignment operator
//! Make a copy of another digital filter.
//! Do not copy the filter state (delay line).
Filter & operator=( const Filter & rhs );
//! Destructor is virtual to enable subclassing. Subclasses may specialize
//! construction, and may add functionality, but for efficiency, the filtering
//! operation is non-virtual.
~Filter( void );
// --- filtering ---
//! Compute a filtered sample from the next input sample.
//!
//! \param input is the next input sample
//! \return the next output sample
double apply( double input );
//! Function call operator, same as sample().
//!
//! \sa apply
double operator() ( double input ) { return apply(input); }
// --- access/mutation ---
//! Provide access to the numerator (feed-forward) coefficients
//! of this filter. The coefficients are stored in order of increasing
//! delay (lowest order coefficient first).
std::vector< double > numerator( void );
//! Provide access to the numerator (feed-forward) coefficients
//! of this filter. The coefficients are stored in order of increasing
//! delay (lowest order coefficient first).
const std::vector< double > numerator( void ) const;
//! Provide access to the denominator (feedback) coefficients
//! of this filter. The coefficients are stored in order of increasing
//! delay (lowest order coefficient first).
std::vector< double > denominator( void );
//! Provide access to the denominator (feedback) coefficients
//! of this filter. The coefficients are stored in order of increasing
//! delay (lowest order coefficient first).
const std::vector< double > denominator( void ) const;
//! Clear the filter state.
void clear( void );
private:
// --- implementation ---
//! single delay line for Direct-Form II implementation
std::deque< double > m_delayline;
//! feed-forward coefficients
std::vector< double > m_ffwdcoefs;
//! feedback coefficients
std::vector< double > m_fbackcoefs;
//! filter gain (applied to output)
double m_gain;
}; // end of class Filter
// ---------------------------------------------------------------------------
// constructor
// ---------------------------------------------------------------------------
//! Initialize a Filter having the specified coefficients, and
//! order equal to the larger of the two coefficient ranges.
//! Coefficients in the sequences are stored in increasing order
//! (lowest order coefficient first).
//!
//! If template members are allowed, then the coefficients
//! can be stored in any kind of iterator range, otherwise,
//! they must be in an array of doubles.
//!
//! \param ffwdbegin is the beginning of a sequence of feed-forward coefficients
//! \param ffwdend is the end of a sequence of feed-forward coefficients
//! \param fbackbegin is the beginning of a sequence of feedback coefficients
//! \param fbackend is the end of a sequence of feedback coefficients
//! \param gain is an optional gain scale applied to the filtered signal
//
#if !defined(NO_TEMPLATE_MEMBERS)
template<typename IterT1, typename IterT2>
Filter::Filter( IterT1 ffwdbegin, IterT1 ffwdend, // feed-forward coeffs
IterT2 fbackbegin, IterT2 fbackend, // feedback coeffs
double gain ) :
#else
inline
Filter::Filter( const double * ffwdbegin, const double * ffwdend, // feed-forward coeffs
const double * fbackbegin, const double * fbackend, // feedback coeffs
double gain ) :
#endif
m_ffwdcoefs( ffwdbegin, ffwdend ),
m_fbackcoefs( fbackbegin, fbackend ),
m_delayline( std::max( ffwdend-ffwdbegin, fbackend-fbackbegin ) - 1, 0. ),
m_gain( gain )
{
if ( *fbackbegin == 0. )
{
Throw( InvalidObject,
"Tried to create a Filter with feeback coefficient at zero delay equal to 0.0" );
}
// normalize the coefficients by 1/a[0], if a[0] is not equal to 1.0
// (already checked for a[0] == 0 above)
if ( *fbackbegin != 1. )
{
// scale all filter coefficients by a[0]:
std::transform( m_ffwdcoefs.begin(), m_ffwdcoefs.end(), m_ffwdcoefs.begin(),
std::bind2nd( std::divides<double>(), *fbackbegin ) );
std::transform( m_fbackcoefs.begin(), m_fbackcoefs.end(), m_fbackcoefs.begin(),
std::bind2nd( std::divides<double>(), *fbackbegin ) );
m_fbackcoefs[0] = 1.;
}
}
} // end of namespace Loris
#endif /* ndef INCLUDE_FILTER_H */
|