zynaddsubfx

ZynAddSubFX open source synthesizer
Log | Files | Refs | Submodules | LICENSE

Unison.cpp (5863B)


      1 /*
      2   ZynAddSubFX - a software synthesizer
      3 
      4   Unison.cpp - Unison effect (multivoice chorus)
      5   Copyright (C) 2002-2009 Nasca Octavian Paul
      6   Author: Nasca Octavian Paul
      7 
      8   This program is free software; you can redistribute it and/or
      9   modify it under the terms of the GNU General Public License
     10   as published by the Free Software Foundation; either version 2
     11   of the License, or (at your option) any later version.
     12 */
     13 
     14 #include <cmath>
     15 #include <cstring>
     16 
     17 #include "../Misc/Allocator.h"
     18 #include "Unison.h"
     19 #include "globals.h"
     20 
     21 #define errx(...) {}
     22 #define warnx(...) {}
     23 #ifndef errx
     24 #include <err.h>
     25 #endif
     26 
     27 namespace zyn {
     28 
     29 Unison::Unison(Allocator *alloc_, int update_period_samples_, float max_delay_sec_, float srate_f)
     30     :unison_size(0),
     31       base_freq(1.0f),
     32       uv(NULL),
     33       update_period_samples(update_period_samples_),
     34       update_period_sample_k(0),
     35       max_delay((int)(srate_f * max_delay_sec_) + 1),
     36       delay_k(0),
     37       first_time(false),
     38       delay_buffer(NULL),
     39       unison_amplitude_samples(0.0f),
     40       unison_bandwidth_cents(10.0f),
     41       samplerate_f(srate_f),
     42       alloc(*alloc_)
     43 {
     44     if(max_delay < 10)
     45         max_delay = 10;
     46     delay_buffer = alloc.valloc<float>(max_delay);
     47     memset(delay_buffer, 0, max_delay * sizeof(float));
     48     setSize(1);
     49 }
     50 
     51 Unison::~Unison() {
     52     alloc.devalloc(delay_buffer);
     53     alloc.devalloc(uv);
     54 }
     55 
     56 void Unison::setSize(int new_size)
     57 {
     58     if(new_size < 1)
     59         new_size = 1;
     60     unison_size = new_size;
     61     alloc.devalloc(uv);
     62     uv = alloc.valloc<UnisonVoice>(unison_size);
     63     first_time = true;
     64     updateParameters();
     65 }
     66 
     67 void Unison::setBaseFrequency(float freq)
     68 {
     69     base_freq = freq;
     70     updateParameters();
     71 }
     72 
     73 void Unison::setBandwidth(float bandwidth)
     74 {
     75     if(bandwidth < 0)
     76         bandwidth = 0.0f;
     77     if(bandwidth > 1200.0f)
     78         bandwidth = 1200.0f;
     79 
     80     /* If the bandwidth is too small, the audio may cancel itself out
     81      * (due to the sign change of the outputs)
     82      * TODO figure out the acceptable lower bound and codify it
     83      */
     84     unison_bandwidth_cents = bandwidth;
     85     updateParameters();
     86 }
     87 
     88 void Unison::updateParameters(void)
     89 {
     90     if(!uv)
     91         return;
     92     float increments_per_second = samplerate_f
     93                                   / (float) update_period_samples;
     94 //	printf("#%g, %g\n",increments_per_second,base_freq);
     95     for(int i = 0; i < unison_size; ++i) {
     96         float base = powf(UNISON_FREQ_SPAN, SYNTH_T::numRandom() * 2.0f - 1.0f);
     97         uv[i].relative_amplitude = base;
     98         float period = base / base_freq;
     99         float m      = 4.0f / (period * increments_per_second);
    100         if(SYNTH_T::numRandom() < 0.5f)
    101             m = -m;
    102         uv[i].step = m;
    103 //		printf("%g %g\n",uv[i].relative_amplitude,period);
    104     }
    105 
    106     float max_speed = powf(2.0f, unison_bandwidth_cents / 1200.0f);
    107     unison_amplitude_samples = 0.125f * (max_speed - 1.0f)
    108                                * samplerate_f / base_freq;
    109 
    110     //If functions exceed this limit, they should have requested a bigguer delay
    111     //and thus are buggy
    112     if(unison_amplitude_samples >= max_delay - 1) {
    113         warnx("BUG: Unison amplitude samples too big");
    114         warnx("Unision max_delay should be larger");
    115         unison_amplitude_samples = max_delay - 2;
    116     }
    117 
    118     updateUnisonData();
    119 }
    120 
    121 void Unison::process(int bufsize, float *inbuf, float *outbuf)
    122 {
    123     if(!uv)
    124         return;
    125     if(!outbuf)
    126         outbuf = inbuf;
    127 
    128     float volume    = 1.0f / sqrtf(unison_size);
    129     float xpos_step = 1.0f / (float) update_period_samples;
    130     float xpos      = (float) update_period_sample_k * xpos_step;
    131     for(int i = 0; i < bufsize; ++i) {
    132         if(update_period_sample_k++ >= update_period_samples) {
    133             updateUnisonData();
    134             update_period_sample_k = 0;
    135             xpos = 0.0f;
    136         }
    137         xpos += xpos_step;
    138         float in   = inbuf[i], out = 0.0f;
    139         float sign = 1.0f;
    140         for(int k = 0; k < unison_size; ++k) {
    141             float vpos = uv[k].realpos1 * (1.0f - xpos) + uv[k].realpos2 * xpos;        //optimize
    142             float pos  = (float)(delay_k + max_delay) - vpos - 1.0f;
    143             int   posi;
    144             F2I(pos, posi); //optimize!
    145             int posi_next = posi + 1;
    146             if(posi >= max_delay)
    147                 posi -= max_delay;
    148             if(posi_next >= max_delay)
    149                 posi_next -= max_delay;
    150             float posf = pos - floorf(pos);
    151             out += ((1.0f - posf) * delay_buffer[posi] + posf
    152                  * delay_buffer[posi_next]) * sign;
    153             sign = -sign;
    154         }
    155         outbuf[i] = out * volume;
    156 //		printf("%d %g\n",i,outbuf[i]);
    157         delay_buffer[delay_k] = in;
    158         delay_k = (++delay_k < max_delay) ? delay_k : 0;
    159     }
    160 }
    161 
    162 void Unison::updateUnisonData()
    163 {
    164     if(!uv)
    165         return;
    166 
    167     for(int k = 0; k < unison_size; ++k) {
    168         float pos  = uv[k].position;
    169         float step = uv[k].step;
    170         pos += step;
    171         if(pos <= -1.0f) {
    172             pos  = -1.0f;
    173             step = -step;
    174         }
    175         else
    176         if(pos >= 1.0f) {
    177             pos  = 1.0f;
    178             step = -step;
    179         }
    180         float vibratto_val = (pos - 0.333333333f * pos * pos * pos) * 1.5f; //make the vibratto lfo smoother
    181 
    182         //Relative amplitude is utilized, so the delay may be larger than the
    183         //whole buffer, if the buffer is too small, this indicates a buggy call
    184         //to Unison()
    185         float newval = 1.0f + 0.5f
    186                        * (vibratto_val + 1.0f) * unison_amplitude_samples
    187                        * uv[k].relative_amplitude;
    188 
    189         if(first_time)
    190             uv[k].realpos1 = uv[k].realpos2 = newval;
    191         else {
    192             uv[k].realpos1 = uv[k].realpos2;
    193             uv[k].realpos2 = newval;
    194         }
    195 
    196         uv[k].position = pos;
    197         uv[k].step     = step;
    198     }
    199     first_time = false;
    200 }
    201 
    202 }