diff options
author | John Glover <glover.john@gmail.com> | 2011-07-08 18:06:21 +0100 |
---|---|---|
committer | John Glover <glover.john@gmail.com> | 2011-07-08 18:06:21 +0100 |
commit | d6073e01c933c77f1e2bc3c3fe1126d617003549 (patch) | |
tree | 695d23677c5b84bf3a0f88fbd4959b4f7cbc0e90 /src/loris/Filter.h | |
parent | 641688b252da468eb374674a0dbaae1bbac70b2b (diff) | |
download | simpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.tar.gz simpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.tar.bz2 simpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.zip |
Start adding Loris files
Diffstat (limited to 'src/loris/Filter.h')
-rw-r--r-- | src/loris/Filter.h | 249 |
1 files changed, 249 insertions, 0 deletions
diff --git a/src/loris/Filter.h b/src/loris/Filter.h new file mode 100644 index 0000000..5a23a3e --- /dev/null +++ b/src/loris/Filter.h @@ -0,0 +1,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 */ |