diff options
-rw-r--r-- | simpl/synthesis.pxd | 3 | ||||
-rw-r--r-- | simpl/synthesis.pyx | 12 | ||||
-rw-r--r-- | src/simpl/synthesis.cpp | 109 | ||||
-rw-r--r-- | src/simpl/synthesis.h | 21 | ||||
-rw-r--r-- | tests/test_synthesis.cpp | 52 |
5 files changed, 197 insertions, 0 deletions
diff --git a/simpl/synthesis.pxd b/simpl/synthesis.pxd index 6e87fe1..201a545 100644 --- a/simpl/synthesis.pxd +++ b/simpl/synthesis.pxd @@ -25,6 +25,9 @@ cdef extern from "../src/simpl/synthesis.h" namespace "simpl": void synth_frame(c_Frame* frame) vector[c_Frame*] synth(vector[c_Frame*] frames) + cdef cppclass c_MQSynthesis "simpl::MQSynthesis"(c_Synthesis): + c_MQSynthesis() + cdef cppclass c_SMSSynthesis "simpl::SMSSynthesis"(c_Synthesis): c_SMSSynthesis() int num_stochastic_coeffs() diff --git a/simpl/synthesis.pyx b/simpl/synthesis.pyx index 708c210..b86a7c0 100644 --- a/simpl/synthesis.pyx +++ b/simpl/synthesis.pyx @@ -50,6 +50,18 @@ cdef class Synthesis: return output +cdef class MQSynthesis(Synthesis): + def __cinit__(self): + if self.thisptr: + del self.thisptr + self.thisptr = new c_MQSynthesis() + + def __dealloc__(self): + if self.thisptr: + del self.thisptr + self.thisptr = <c_Synthesis*>0 + + cdef class SMSSynthesis(Synthesis): SMS_DET_IFFT = 0 SMS_DET_SIN = 1 diff --git a/src/simpl/synthesis.cpp b/src/simpl/synthesis.cpp index 3f48c78..c5d3570 100644 --- a/src/simpl/synthesis.cpp +++ b/src/simpl/synthesis.cpp @@ -59,6 +59,115 @@ Frames Synthesis::synth(Frames frames) { // --------------------------------------------------------------------------- +// MQSynthesis +// --------------------------------------------------------------------------- +MQSynthesis::MQSynthesis() { + _prev_amps = NULL; + _prev_freqs = NULL; + _prev_phases = NULL; + reset(); +} + +MQSynthesis::~MQSynthesis() { + if(_prev_amps) delete [] _prev_amps; + if(_prev_freqs) delete [] _prev_freqs; + if(_prev_phases) delete [] _prev_phases; + + _prev_amps = NULL; + _prev_freqs = NULL; + _prev_phases = NULL; +} + +void MQSynthesis::reset() { + if(_prev_amps) delete [] _prev_amps; + if(_prev_freqs) delete [] _prev_freqs; + if(_prev_phases) delete [] _prev_phases; + + _prev_amps = new sample[_max_partials]; + _prev_freqs = new sample[_max_partials]; + _prev_phases = new sample[_max_partials]; + + memset(_prev_amps, 0.0, sizeof(sample) * _max_partials); + memset(_prev_freqs, 0.0, sizeof(sample) * _max_partials); + memset(_prev_phases, 0.0, sizeof(sample) * _max_partials); +} + +sample MQSynthesis::hz_to_radians(sample f) { + return (f * 2 * M_PI) / _sampling_rate; +} + +void MQSynthesis::max_partials(int new_max_partials) { + _max_partials = new_max_partials; + reset(); +} + +void MQSynthesis::synth_frame(Frame* frame) { + int num_partials = frame->num_partials(); + if(num_partials > _max_partials) { + num_partials = _max_partials; + } + + for(int n = 0; n < _hop_size; n++) { + frame->synth()[n] = 0.f; + } + + for(int i = 0; i < num_partials; i++) { + sample amp = frame->partial(i)->amplitude; + sample freq = hz_to_radians(frame->partial(i)->frequency); + sample phase = frame->partial(i)->phase; + + // get values for last amplitude, frequency and phase + // these are the initial values of the instantaneous + // amplitude/frequency/phase + sample prev_amp = _prev_amps[i]; + sample prev_freq = _prev_freqs[i]; + sample prev_phase = _prev_phases[i]; + + if(prev_amp == 0) { + prev_freq = freq; + prev_phase = frame->partial(i)->phase - (freq * _hop_size); + while(prev_phase >= M_PI) { + prev_phase -= (2.0 * M_PI); + } + while(prev_phase < -M_PI) { + prev_phase += (2.0 * M_PI); + } + } + + // amplitudes are linearly interpolated between frames + sample inst_amp = prev_amp; + sample amp_inc = (frame->partial(i)->amplitude - prev_amp) / _hop_size; + + // freqs/phases are calculated by cubic interpolation + sample freq_diff = freq - prev_freq; + sample x = (prev_phase + (prev_freq * _hop_size) - phase) + + (freq_diff * (_hop_size / 2.0)); + x /= (2.0 * M_PI); + int m = floor(x + 0.5); + sample phase_diff = phase - prev_phase - (prev_freq * _hop_size) + + (2.0 * M_PI * m); + sample alpha = ((3.0 / pow(_hop_size, 2.0)) * phase_diff) - + (freq_diff / _hop_size); + sample beta = ((-2.0 / pow(_hop_size, 3.0)) * phase_diff) + + (freq_diff / pow(_hop_size, 2.0)); + + // calculate output samples + sample inst_phase = 0.f; + for(int n = 0; n < _hop_size; n++) { + inst_amp += amp_inc; + inst_phase = prev_phase + (prev_freq * n) + + (alpha * pow((sample)n, 2.0)) + + (beta * pow((sample)n, 3.0)); + frame->synth()[n] += (2.f * inst_amp) * cos(inst_phase); + } + + _prev_amps[i] = amp; + _prev_freqs[i] = freq; + _prev_phases[i] = phase; + } +} + +// --------------------------------------------------------------------------- // SMSSynthesis // --------------------------------------------------------------------------- diff --git a/src/simpl/synthesis.h b/src/simpl/synthesis.h index e93669f..64f6f7b 100644 --- a/src/simpl/synthesis.h +++ b/src/simpl/synthesis.h @@ -1,6 +1,8 @@ #ifndef SYNTHESIS_H #define SYNTHESIS_H +#include <math.h> + #include "base.h" extern "C" { @@ -51,6 +53,25 @@ class Synthesis { // --------------------------------------------------------------------------- +// MQSynthesis +// --------------------------------------------------------------------------- +class MQSynthesis : public Synthesis { + private: + sample* _prev_amps; + sample* _prev_freqs; + sample* _prev_phases; + void reset(); + sample hz_to_radians(sample f); + + public: + MQSynthesis(); + ~MQSynthesis(); + void max_partials(int new_max_partials); + void synth_frame(Frame* frame); +}; + + +// --------------------------------------------------------------------------- // SMSSynthesis // --------------------------------------------------------------------------- class SMSSynthesis : public Synthesis { diff --git a/tests/test_synthesis.cpp b/tests/test_synthesis.cpp index 50ca6ed..fa49016 100644 --- a/tests/test_synthesis.cpp +++ b/tests/test_synthesis.cpp @@ -16,6 +16,57 @@ namespace simpl { // --------------------------------------------------------------------------- +// TestMQSynthesis +// --------------------------------------------------------------------------- +class TestMQSynthesis : public CPPUNIT_NS::TestCase { + CPPUNIT_TEST_SUITE(TestMQSynthesis); + CPPUNIT_TEST(test_basic); + CPPUNIT_TEST_SUITE_END(); + +protected: + static const double PRECISION = 0.001; + MQPeakDetection* pd; + MQPartialTracking* pt; + MQSynthesis* synth; + SndfileHandle sf; + int num_samples; + + void test_basic() { + sample* audio = new sample[(int)sf.frames()]; + sf.read(audio, (int)sf.frames()); + Frames frames = pd->find_peaks(num_samples, &(audio[(int)sf.frames() / 2])); + frames = pt->find_partials(frames); + frames = synth->synth(frames); + + for(int i = 0; i < frames.size(); i++) { + CPPUNIT_ASSERT(frames[i]->num_peaks() > 0); + CPPUNIT_ASSERT(frames[i]->num_partials() > 0); + + double energy = 0.f; + for(int j = 0; j < synth->hop_size(); j++) { + energy += frames[i]->synth()[j] * frames[i]->synth()[j]; + } + CPPUNIT_ASSERT(energy > 0.f); + } + } + +public: + void setUp() { + pd = new MQPeakDetection(); + pt = new MQPartialTracking(); + synth = new MQSynthesis(); + sf = SndfileHandle("../tests/audio/flute.wav"); + num_samples = 4096; + } + + void tearDown() { + delete pd; + delete pt; + delete synth; + } +}; + +// --------------------------------------------------------------------------- // TestLorisSynthesis // --------------------------------------------------------------------------- class TestLorisSynthesis : public CPPUNIT_NS::TestCase { @@ -72,6 +123,7 @@ public: } // end of namespace simpl +CPPUNIT_TEST_SUITE_REGISTRATION(simpl::TestMQSynthesis); CPPUNIT_TEST_SUITE_REGISTRATION(simpl::TestLorisSynthesis); int main(int arg, char **argv) { |