diff options
Diffstat (limited to 'src/loris/Synthesizer.h')
-rw-r--r-- | src/loris/Synthesizer.h | 391 |
1 files changed, 391 insertions, 0 deletions
diff --git a/src/loris/Synthesizer.h b/src/loris/Synthesizer.h new file mode 100644 index 0000000..e0268ba --- /dev/null +++ b/src/loris/Synthesizer.h @@ -0,0 +1,391 @@ +#ifndef INCLUDE_SYNTHESIZER_H +#define INCLUDE_SYNTHESIZER_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 + * + * + * Synthesizer.h + * + * Definition of class Loris::Synthesizer, a renderer of + * bandwidth-enhanced Partials. + * + * Kelly Fitz, 16 Aug 1999 + * loris@cerlsoundgroup.org + * + * http://www.cerlsoundgroup.org/Loris/ + * + */ + + + // TODO: + // Passing the sample vector in to Synthesizer no longer makes any sense, + // if the Synthesizer can (must) resize the buffer anyway, it may as well + // own it. In a future version, the Synthesizer owns a vector (need to add + // a method to clear it), and provides access to it (as it currently does). + // When this change is made, it probably makes sense to remove the function + // call operators. + + +#include "Oscillator.h" +#include "PartialList.h" +#include "PartialUtils.h" + +#include <vector> + +// begin namespace +namespace Loris { + +// --------------------------------------------------------------------------- +// class Synthesizer +// +//! A Synthesizer renders bandwidth-enhanced Partials into a buffer +//! of samples. +//! +//! Class Synthesizer represents an algorithm for rendering +//! bandwidth-enhanced Partials as floating point (double) samples at a +//! specified sampling rate, and accumulating them into a buffer. +//! +//! The Synthesizer does not own the sample buffer, the client is responsible +//! for its construction and destruction, and many Synthesizers may share +//! a buffer. +// +class Synthesizer +{ +// -- public interface -- +public: +// -- construction -- + + struct Parameters; // defined below + + //! Construct a Synthesizer using the default parameters and sample + //! buffer (a standard library vector). Since Partials generated by + //! the Loris Analyzer generally begin and end at non-zero amplitude, + //! zero-amplitude Breakpoints are inserted at either end of the Partial, + //! at a temporal distance equal to the fade time, to reduce turn-on and + //! turn-off artifacts. + //! + //! \sa Synthesizer::Parameters + //! + //! \param buffer The vector (of doubles) into which rendered samples + //! should be accumulated. + //! \throw InvalidArgument if any of the parameters is invalid. + Synthesizer( std::vector<double> & buffer ); + + + //! Construct a Synthesizer using the specified parameters and sample + //! buffer (a standard library vector). Since Partials generated by the + //! Loris Analyzer generally begin and end at non-zero amplitude, zero-amplitude + //! Breakpoints are inserted at either end of the Partial, at a temporal + //! distance equal to the fade time, to reduce turn-on and turn-off + //! artifacts. + //! + //! \param params A Parameters struct storing the configuration of + //! Synthesizer parameters. + //! \param buffer The vector (of doubles) into which rendered samples + //! should be accumulated. + //! \throw InvalidArgument if any of the parameters is invalid. + Synthesizer( Parameters params, std::vector<double> & buffer ); + + //! Construct a Synthesizer using the specified sampling rate, sample + //! buffer (a standard library vector), and the default fade time + //! stored in the DefaultParameters. Since Partials generated by the Loris + //! Analyzer generally begin and end at non-zero amplitude, zero-amplitude + //! Breakpoints are inserted at either end of the Partial, at a temporal + //! distance equal to the fade time, to reduce turn-on and turn-off + //! artifacts. + //! + //! \param srate The rate (Hz) at which to synthesize samples + //! (must be positive). + //! \param buffer The vector (of doubles) into which rendered samples + //! should be accumulated. + //! \throw InvalidArgument if the specfied sample rate is non-positive. + Synthesizer( double srate, std::vector<double> & buffer ); + + //! Construct a Synthesizer using the specified sampling rate, sample + //! buffer (a standard library vector), and Partial fade time + //! (in seconds). Since Partials generated by the Loris Analyzer + //! generally begin and end at non-zero amplitude, zero-amplitude + //! Breakpoints are inserted at either end of the Partial, at a temporal + //! distance equal to the fade time, to reduce turn-on and turn-off + //! artifacts. + //! + //! \param srate The rate (Hz) at which to synthesize samples + //! (must be positive). + //! \param buffer The vector (of doubles) into which rendered samples + //! should be accumulated. + //! \param fadeTime The Partial fade time in seconds (must be non-negative). + //! \throw InvalidArgument if the specfied sample rate is non-positive. + //! \throw InvalidArgument if the specified fade time is negative. + Synthesizer( double srate, std::vector<double> & buffer, double fadeTime ); + + // Compiler can generate copy, assign, and destroy. + // Synthesizer( const Synthesizer & other ); + // ~Synthesizer( void ); + // Synthesizer & operator= ( const Synthesizer & other ); + +// -- synthesis -- + + //! Synthesize a bandwidth-enhanced sinusoidal Partial. Zero-amplitude + //! Breakpoints are inserted at either end of the Partial to reduce + //! turn-on and turn-off artifacts, as described above. The synthesizer + //! will resize the buffer as necessary to accommodate all the samples, + //! including the fade out. Previous contents of the buffer are not + //! overwritten. Partials with start times earlier than the Partial fade + //! time will have shorter onset fades. Partials are not rendered at + //! frequencies above the half-sample rate. + //! + //! \param p The Partial to synthesize. + //! \return Nothing. + //! \pre The partial must have non-negative start time. + //! \post This Synthesizer's sample buffer (vector) has been + //! resized to accommodate the entire duration of the + //! Partial, p, including fade out at the end. + //! \throw InvalidPartial if the Partial has negative start time. + void synthesize( Partial p ); + + //! Function call operator: same as synthesize( p ). + void operator() ( const Partial & p ) { synthesize( p ) ; } + + + //! Synthesize all Partials on the specified half-open (STL-style) range. + //! Null Breakpoints are inserted at either end of the Partial to reduce + //! turn-on and turn-off artifacts, as described above. The synthesizer + //! will resize the buffer as necessary to accommodate all the samples, + //! including the fade outs. Previous contents of the buffer are not + //! overwritten. Partials with start times earlier than the Partial fade + //! time will have shorter onset fades. Partials are not rendered at + //! frequencies above the half-sample rate. + //! + //! \param begin_partials The beginning of the range of Partials + //! to synthesize. + //! \param end_partials The end of the range of Partials + //! to synthesize. + //! \return Nothing. + //! \pre The partials must have non-negative start times. + //! \post This Synthesizer's sample buffer (vector) has been + //! resized to accommodate the entire duration of all the + //! Partials including fade out at the ends. + //! \throw InvalidPartial if any Partial has negative start time. +#if ! defined(NO_TEMPLATE_MEMBERS) + template< typename Iter > + void synthesize( Iter begin_partials, Iter end_partials ); +#else + inline + void synthesize( PartialList::iterator begin_partials, + PartialList::iterator end_partials ); +#endif + + //! Function call operator: same as + //! synthesize( begin_partials, end_partials ). +#if ! defined(NO_TEMPLATE_MEMBERS) + template< typename Iter > + void operator() ( Iter begin_partials, Iter end_partials ); +#else + inline + void operator() ( PartialList::iterator begin_partials, + PartialList::iterator end_partials ); +#endif + +// -- sample access -- + + //! Return a const reference to the sample buffer used (not + //! owned) by this Synthesizer. + const std::vector<double> & samples( void ) const; + + //! Return a reference to the sample buffer used (not + //! owned) by this Synthesizer. + std::vector<double> & samples( void ); + + +// -- parameter access and mutation -- + + + //! Return this Synthesizer's Partial fade time, in seconds. + double fadeTime( void ) const; + + //! Return the sampling rate (in Hz) for this Synthesizer. + double sampleRate( void ) const; + + //! Set this Synthesizer's fade time to the specified value + //! (in seconds, must be non-negative). + //! + //! \param partialFadeTime The new Partial fade time. + //! \throw InvalidArgument if the specified fade time is negative. + void setFadeTime( double partialFadeTime ); + + //! Set this Synthesizer's sample rate to the specified value + //! (in Hz, must be positive). + //! + //! \param rate The new synthesis sample rate. + //! \throw InvalidArgument if the specified rate is nonpositive. + void setSampleRate( double rate ); + + //! Return access to the Filter used by this Synthesizer's + //! Oscillator to implement bandwidth-enhanced sinusoidal + //! synthesis. (Can use this access to make changes to the + //! filter coefficients.) + Filter & filter( void ); + + +// -- parameters structure -- + + enum + { + Default_FadeTime_Ms = 1, + Default_SampleRate_Hz = 44100 + }; + + // enum EnhancementFlag { Sinusoidal = 0, BwEnhanced = 1 }; + + //! Structure storing a configuration of Synthesizer parameters. + //! Elements in this struct can be freely modified, the configuration + //! is validated before it can be used to construct a Synthesizer. + //! + //! \sa IsValidParameters + struct Parameters + { + double fadeTime; + double sampleRate; + // EnhancementFlag enhancement; + + Filter filter; + + // default constructor + // + //! Assign default initial values to the Synthesizer parameters, Filter + //! defaults are defined in Filter.C, others in Synthesizer.C. + Parameters( void ); + + // copy, assign, and destroy are free + }; + + //! Default configuration of a Loris::Synthesizer. Modify the values + //! in this structure to alter the configuration of all Synthesizers, + //! including those used by the AiffFile class to render Partials. + static const Parameters & DefaultParameters( void ); + + //! Assign a new default configuration of a Loris::Synthesizer + //! to alter the configuration of all Synthesizers, + //! including those used by the AiffFile class to render Partials. + //! + //! \param params A Parameters struct describing the new default + //! configuration for Synthesizers. + //! \throw InvalidArgument if params reprsents an invalid configuration. + static void SetDefaultParameters( const Parameters & params ); + + //! Check the validty of a Parameters structure. Returns true if the + //! struct represents a valid Synthesizer configuration, otherwise + //! raise InvalidArgument reporting the specific error. + //! + //! \param params A Parameters struct describing a configuration for + //! Synthesizers. + //! \throw InvalidArgument if params reprsents an invalid configuration. + static bool IsValidParameters( const Parameters & params ); + + +// -- implementation -- +private: + + Oscillator m_osc; // the Synthesizer has-a Oscillator that it uses to render + // all the Partials one by one. + + std::vector< double > * m_sampleBuffer; // samples are computed and stored here, BUT + // Synthesizer does NOT own this buffer. + + double m_fadeTimeSec; // Partial fade in/out time in seconds + double m_srateHz; // sample rate in Hz + +}; // end of class Synthesizer + + +// --------------------------------------------------------------------------- +// synthesize +// --------------------------------------------------------------------------- +//! Synthesize all Partials on the specified half-open (STL-style) range. +//! Null Breakpoints are inserted at either end of the Partial to reduce +//! turn-on and turn-off artifacts, as described above. The synthesizer +//! will resize the buffer as necessary to accommodate all the samples, +//! including the fade outs. Previous contents of the buffer are not +//! overwritten. Partials with start times earlier than the Partial fade +//! time will have shorter onset fades. +//! +//! \param begin_partials The beginning of the range of Partials +//! to synthesize. +//! \param end_partials The end of the range of Partials +//! to synthesize. +//! \return Nothing. +//! \pre The partials must have non-negative start times. +//! \post This Synthesizer's sample buffer (vector) has been +//! resized to accommodate the entire duration of all the +//! Partials including fade out at the ends. +//! \throw InvalidPartial if any Partial has negative start time. +// +#if ! defined(NO_TEMPLATE_MEMBERS) +template<typename Iter> +void +Synthesizer::synthesize( Iter begin_partials, Iter end_partials ) +#else +inline void +Synthesizer::synthesize( PartialList::iterator begin_partials, + PartialList::iterator end_partials ) +#endif +{ + // grow the sample buffer, if necessary, to accommodate the latest + // Partial, with the fade time tacked on the end + double duration = + PartialUtils::timeSpan( begin_partials, end_partials ).second + + m_fadeTimeSec; + + typedef std::vector< double >::size_type Sz_Type; + Sz_Type Nsamps = 1 + Sz_Type( duration * m_srateHz ); + if ( m_sampleBuffer->size() < Nsamps ) + { + m_sampleBuffer->resize( Nsamps ); + } + + while ( begin_partials != end_partials ) + { + synthesize( *(begin_partials++) ); + } +} + +// --------------------------------------------------------------------------- +// operator() +// --------------------------------------------------------------------------- +//! Function call operator: same as +//! synthesize( begin_partials, end_partials, timeShift ). +// +#if ! defined(NO_TEMPLATE_MEMBERS) +template<typename Iter> +void +Synthesizer::operator() ( Iter begin_partials, Iter end_partials ) +#else +inline void +Synthesizer::operator() ( PartialList::iterator begin_partials, + PartialList::iterator end_partials ) +#endif +{ + synthesize( begin_partials, end_partials ); +} + +} // end of namespace Loris + +#endif /* ndef INCLUDE_SYNTHESIZER_H */ |