#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 // 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 & 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 & 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 & 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 & 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 & samples( void ) const; //! Return a reference to the sample buffer used (not //! owned) by this Synthesizer. std::vector & 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 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 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 */