summaryrefslogtreecommitdiff
path: root/src/loris/Synthesizer.h
diff options
context:
space:
mode:
authorJohn Glover <glover.john@gmail.com>2011-07-08 18:06:21 +0100
committerJohn Glover <glover.john@gmail.com>2011-07-08 18:06:21 +0100
commitd6073e01c933c77f1e2bc3c3fe1126d617003549 (patch)
tree695d23677c5b84bf3a0f88fbd4959b4f7cbc0e90 /src/loris/Synthesizer.h
parent641688b252da468eb374674a0dbaae1bbac70b2b (diff)
downloadsimpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.tar.gz
simpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.tar.bz2
simpl-d6073e01c933c77f1e2bc3c3fe1126d617003549.zip
Start adding Loris files
Diffstat (limited to 'src/loris/Synthesizer.h')
-rw-r--r--src/loris/Synthesizer.h391
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 */