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 }