/* Calf Box, an open source musical instrument. Copyright (C) 2010-2011 Krzysztof Foltman 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 3 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, see . */ #include "config.h" #include "config-api.h" #include "dspmath.h" #include "module.h" #include #include #include #include #include #include #include #define MAX_CHORUS_LENGTH 4096 #define MODULE_PARAMS chorus_params static float sine_table[2049]; struct chorus_params { float lfo_freq; float min_delay; float mod_depth; float wet_dry; float sphase; }; struct chorus_module { struct cbox_module module; float storage[MAX_CHORUS_LENGTH][2]; struct chorus_params *params; int pos; float tp32dsr; uint32_t phase; }; MODULE_PROCESSCMD_FUNCTION(chorus) { struct chorus_module *m = (struct chorus_module *)ct->user_data; EFFECT_PARAM("/min_delay", "f", min_delay, double, , 1, 20) else EFFECT_PARAM("/mod_depth", "f", mod_depth, double, , 1, 20) else EFFECT_PARAM("/lfo_freq", "f", lfo_freq, double, , 0, 20) else EFFECT_PARAM("/stereo_phase", "f", sphase, double, , 0, 360) else EFFECT_PARAM("/wet_dry", "f", wet_dry, double, , 0, 1) else if (!strcmp(cmd->command, "/status") && !strcmp(cmd->arg_types, "")) { if (!cbox_check_fb_channel(fb, cmd->command, error)) return FALSE; return cbox_execute_on(fb, NULL, "/min_delay", "f", error, m->params->min_delay) && cbox_execute_on(fb, NULL, "/mod_depth", "f", error, m->params->mod_depth) && cbox_execute_on(fb, NULL, "/lfo_freq", "f", error, m->params->lfo_freq) && cbox_execute_on(fb, NULL, "/stereo_phase", "f", error, m->params->sphase) && cbox_execute_on(fb, NULL, "/wet_dry", "f", error, m->params->wet_dry) && CBOX_OBJECT_DEFAULT_STATUS(&m->module, fb, error); } else return cbox_object_default_process_cmd(ct, fb, cmd, error); return TRUE; } void chorus_process_event(struct cbox_module *module, const uint8_t *data, uint32_t len) { // struct chorus_module *m = (struct chorus_module *)module; } void chorus_process_block(struct cbox_module *module, cbox_sample_t **inputs, cbox_sample_t **outputs) { struct chorus_module *m = (struct chorus_module *)module; struct chorus_params *p = m->params; float min_delay = p->min_delay; float mod_depth = p->mod_depth; float wet_dry = p->wet_dry; int i, c; int mask = MAX_CHORUS_LENGTH - 1; uint32_t sphase = (uint32_t)(p->sphase * 65536.0 * 65536.0 / 360); uint32_t dphase = (uint32_t)(p->lfo_freq * m->tp32dsr); const int fracbits = 32 - 11; const int fracscale = 1 << fracbits; for (c = 0; c < 2; c++) { int pos = m->pos; uint32_t phase = m->phase + c * sphase; for (i = 0; i < CBOX_BLOCK_SIZE; i++) { float dry = inputs[c][i]; float v0 = sine_table[phase >> fracbits]; float v1 = sine_table[1 + (phase >> fracbits)]; float lfo = v0 + (v1 - v0) * ((phase & (fracscale - 1)) * (1.0 / fracscale)); m->storage[pos & mask][c] = dry; float dva = min_delay + mod_depth * lfo; int dv = (int)dva; float frac = dva - dv; float smp0 = m->storage[(pos - dv) & mask][c]; float smp1 = m->storage[(pos - dv - 1) & mask][c]; float smp = smp0 + (smp1 - smp0) * frac; outputs[c][i] = sanef(dry + (smp - dry) * wet_dry); pos++; phase += dphase; } } m->phase += CBOX_BLOCK_SIZE * dphase; m->pos += CBOX_BLOCK_SIZE; } MODULE_SIMPLE_DESTROY_FUNCTION(chorus) MODULE_CREATE_FUNCTION(chorus) { static int inited = 0; int i; if (!inited) { inited = 1; for (i = 0; i < 2049; i++) sine_table[i] = 1 + sin(i * M_PI / 1024); } struct chorus_module *m = malloc(sizeof(struct chorus_module)); CALL_MODULE_INIT(m, 2, 2, chorus); m->module.process_event = chorus_process_event; m->module.process_block = chorus_process_block; m->pos = 0; m->phase = 0; m->tp32dsr = 65536.0 * 65536.0 * m->module.srate_inv; struct chorus_params *p = malloc(sizeof(struct chorus_params)); m->params = p; p->sphase = cbox_config_get_float(cfg_section, "stereo_phase", 90.f); p->lfo_freq = cbox_config_get_float(cfg_section, "lfo_freq", 1.f); p->min_delay = cbox_config_get_float(cfg_section, "min_delay", 20.f); p->mod_depth = cbox_config_get_float(cfg_section, "mod_depth", 15.f); p->wet_dry = cbox_config_get_float(cfg_section, "wet_dry", 0.5f); for (i = 0; i < MAX_CHORUS_LENGTH; i++) m->storage[i][0] = m->storage[i][1] = 0.f; return &m->module; } struct cbox_module_keyrange_metadata chorus_keyranges[] = { }; struct cbox_module_livecontroller_metadata chorus_controllers[] = { }; DEFINE_MODULE(chorus, 2, 2)