zynaddsubfx

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

Controller.cpp (17246B)


      1 /*
      2   ZynAddSubFX - a software synthesizer
      3 
      4   Controller.cpp - (Midi) Controllers implementation
      5   Copyright (C) 2002-2005 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 "Controller.h"
     15 #include "../Misc/Util.h"
     16 #include "../Misc/Time.h"
     17 #include "../Misc/XMLwrapper.h"
     18 #include <cmath>
     19 #include <cstdio>
     20 
     21 #include <rtosc/ports.h>
     22 #include <rtosc/port-sugar.h>
     23 using namespace rtosc;
     24 
     25 namespace zyn {
     26 
     27 #define rObject Controller
     28 
     29 #define rChangeCbBase if (obj->time) { obj->last_update_timestamp = obj->time->time(); }
     30 
     31 #undef rChangeCb
     32 #define rChangeCb rChangeCbBase
     33 const rtosc::Ports Controller::ports = {
     34 #undef rChangeCb
     35 #define rChangeCb obj->setpanning(); rChangeCbBase
     36     rParamZyn(panning.depth,       rShort("pan.d"), rDefault(64),
     37         "Depth of Panning MIDI Control"),
     38 #undef rChangeCb
     39 #define rChangeCb obj->setfiltercutoff(); rChangeCbBase
     40     rParamZyn(filtercutoff.depth,  rShort("fc.d"), rDefault(64),
     41         "Depth of Filter Cutoff MIDI Control"),
     42 #undef rChangeCb
     43 #define rChangeCb obj->setfilterq(); rChangeCbBase
     44     rParamZyn(filterq.depth,       rShort("fq.d"), rDefault(64),
     45         "Depth of Filter Q MIDI Control"),
     46 #undef rChangeCb
     47 #define rChangeCb obj->setbandwidth(); rChangeCbBase
     48     rParamZyn(bandwidth.depth,     rShort("bw.d"), rDefault(64),
     49         "Depth of Bandwidth MIDI Control"),
     50     rToggle(bandwidth.exponential, rShort("bw.exp"), rDefault(false),
     51         "Bandwidth Exponential Mode"),
     52 #undef rChangeCb
     53 #define rChangeCb obj->setmodwheel(); rChangeCbBase
     54     rParamZyn(modwheel.depth,      rShort("mdw.d"), rDefault(80),
     55         "Depth of Modwheel MIDI Control"),
     56     rToggle(modwheel.exponential,  rShort("mdw.exp"), rDefault(false),
     57         "Modwheel Exponential Mode"),
     58 #undef rChangeCb
     59 #define rChangeCb obj->setpitchwheel(); rChangeCbBase
     60     rToggle(pitchwheel.is_split,   rDefault(false),
     61         "If PitchWheel has unified bendrange or not"),
     62     rParamI(pitchwheel.bendrange,  rShort("pch.d"), rDefault(200),
     63             rLinear(-6400, 6400), rUnit(% of semitone),
     64         "Range of MIDI Pitch Wheel"),
     65     rParamI(pitchwheel.bendrange_down, rDefault(0),
     66         "Lower Range of MIDI Pitch Wheel"),
     67 #undef rChangeCb
     68 #define rChangeCb obj->setexpression(); rChangeCbBase
     69     rToggle(expression.receive, rShort("exp.rcv"), rDefault(true),
     70         "Expression MIDI Receive"),
     71 #undef rChangeCb
     72 #define rChangeCb obj->setfmamp(); rChangeCbBase
     73     rToggle(fmamp.receive,      rShort("fma.rcv"), rDefault(true),
     74         "FM amplitude MIDI Receive"),
     75 #undef rChangeCb
     76 #define rChangeCb obj->setvolume(); rChangeCbBase
     77     rToggle(volume.receive,     rShort("vol.rcv"), rDefault(true),
     78         "Volume MIDI Receive"),
     79 #undef rChangeCb
     80 #define rChangeCb obj->setsustain(); rChangeCbBase
     81     rToggle(sustain.receive,    rShort("sus.rcv"), rDefault(true),
     82         "Sustain MIDI Receive"),
     83 #undef rChangeCb
     84 #define rChangeCb rChangeCbBase
     85     rToggle(portamento.receive, rShort("prt.rcv"), rDefault(true),
     86         "Portamento MIDI Receive"),
     87     rToggle(portamento.portamento, rDefault(false),
     88         "Portamento Enable"),
     89     rToggle(portamento.automode, rDefault(true),
     90         "Portamento Auto mode"),
     91     rParamZyn(portamento.time,          rShort("time"), rDefault(64),
     92         "Portamento Length"),
     93     rToggle(portamento.proportional,    rShort("propt."), rDefault(false),
     94             "Whether the portamento time is proportional"
     95             "to the size of the interval between two notes."),
     96     rParamZyn(portamento.propRate,      rShort("scale"), rDefault(80),
     97         "Portamento proportional scale"),
     98     rParamZyn(portamento.propDepth,     rShort("depth"), rDefault(90),
     99         "Portamento proportional depth"),
    100     rParamZyn(portamento.pitchthresh,   rShort("thresh"), rDefault(3),
    101         "Threshold for portamento"),
    102     rToggle(portamento.pitchthreshtype, rShort("tr.type"), rDefault(true),
    103         "Type of threshold"),
    104     rParamZyn(portamento.updowntimestretch, rShort("up/dwn"), rDefault(64),
    105         "Relative length of glide up vs glide down"),
    106 #undef rChangeCb
    107 #define rChangeCb obj->setresonancecenter(); rChangeCbBase
    108     rParamZyn(resonancecenter.depth,    rShort("rfc.d"), rDefault(64),
    109         "Resonance Center MIDI Depth"),
    110 #undef rChangeCb
    111 #define rChangeCb obj->setresonancebw(); rChangeCbBase
    112     rParamZyn(resonancebandwidth.depth, rShort("rbw.d"), rDefault(64),
    113         "Resonance Bandwidth MIDI Depth"),
    114 #undef rChangeCb
    115 #define rChangeCb rChangeCbBase
    116     rToggle(NRPN.receive, rDefault(true), "NRPN MIDI Enable"),
    117     rAction(defaults, "Reset Controller to defaults"),
    118 };
    119 #undef rChangeCb
    120 
    121 Controller::Controller(const SYNTH_T &synth_, const AbsTime *time_)
    122     :time(time_), last_update_timestamp(0), synth(synth_)
    123 {
    124     defaults();
    125     resetall();
    126 }
    127 
    128 void Controller::defaults()
    129 {
    130     pitchwheel.bendrange         = 200; //2 halftones
    131     pitchwheel.bendrange_down    = 0; //Unused by default
    132     pitchwheel.is_split          = false;
    133     expression.receive           = 1;
    134     panning.depth                = 64;
    135     filtercutoff.depth           = 64;
    136     filterq.depth                = 64;
    137     bandwidth.depth              = 64;
    138     bandwidth.exponential        = 0;
    139     modwheel.depth               = 80;
    140     modwheel.exponential         = 0;
    141     fmamp.receive                = 1;
    142     volume.receive               = 1;
    143     sustain.receive              = 1;
    144     NRPN.receive                 = 1;
    145 
    146     portamento.portamento        = 0;
    147     portamento.automode          = 1;
    148     portamento.proportional      = 0;
    149     portamento.propRate          = 80;
    150     portamento.propDepth         = 90;
    151     portamento.receive           = 1;
    152     portamento.time              = 64;
    153     portamento.updowntimestretch = 64;
    154     portamento.pitchthresh       = 3;
    155     portamento.pitchthreshtype   = 1;
    156 
    157     resonancecenter.depth        = 64;
    158     resonancebandwidth.depth     = 64;
    159 
    160     setportamento(0);
    161 }
    162 
    163 void Controller::resetall()
    164 {
    165     setpitchwheel(0); //center
    166     setexpression(127);
    167     setpanning(64);
    168     setfiltercutoff(64);
    169     setfilterq(64);
    170     setbandwidth(64);
    171     setmodwheel(64);
    172     setfmamp(127);
    173     setvolume(127);
    174     setsustain(0);
    175     setresonancecenter(64);
    176     setresonancebw(64);
    177 
    178     //reset the NRPN
    179     NRPN.parhi = -1;
    180     NRPN.parlo = -1;
    181     NRPN.valhi = -1;
    182     NRPN.vallo = -1;
    183 }
    184 
    185 void Controller::setpitchwheel(int value)
    186 {
    187     pitchwheel.data = value;
    188     setpitchwheel();
    189 }
    190 
    191 void Controller::setpitchwheel()
    192 {
    193     int value = pitchwheel.data;
    194     float cents = value / 8192.0f;
    195     if(pitchwheel.is_split && cents < 0)
    196         cents *= pitchwheel.bendrange_down;
    197     else
    198         cents *= pitchwheel.bendrange;
    199     pitchwheel.relfreq = powf(2, cents / 1200.0f);
    200     //fprintf(stderr,"%ld %ld -> %.3f\n",pitchwheel.bendrange,pitchwheel.data,pitchwheel.relfreq);fflush(stderr);
    201 }
    202 
    203 void Controller::setexpression(int value)
    204 {
    205     expression.data = value;
    206     setexpression();
    207 }
    208 
    209 void Controller::setexpression(void)
    210 {
    211     int value = expression.data;
    212     if(expression.receive != 0)
    213     {
    214         assert( value <= 127 ); /* to protect what's left of JML's hearing */
    215 
    216         expression.relvolume = value / 127.0f;
    217     }
    218     else
    219         expression.relvolume = 1.0f;
    220 }
    221 
    222 void Controller::setpanning(int value)
    223 {
    224     panning.data = value;
    225     setpanning();
    226 }
    227 
    228 void Controller::setpanning(void)
    229 {
    230     int value = panning.data;
    231     panning.pan  = (value / 128.0f - 0.5f) * (panning.depth / 64.0f);
    232 }
    233 
    234 void Controller::setfiltercutoff(int value)
    235 {
    236     filtercutoff.data  = value;
    237     setfiltercutoff();
    238 }
    239 
    240 void Controller::setfiltercutoff(void)
    241 {
    242     int value = filtercutoff.data;
    243     filtercutoff.relfreq =
    244         (value - 64.0f) * filtercutoff.depth / 4096.0f * 3.321928f;         //3.3219f..=ln2(10)
    245 }
    246 
    247 void Controller::setfilterq(int value)
    248 {
    249     filterq.data = value;
    250     setfilterq();
    251 }
    252 
    253 void Controller::setfilterq(void)
    254 {
    255     int value = filterq.data;
    256     filterq.relq = powf(30.0f, (value - 64.0f) / 64.0f * (filterq.depth / 64.0f));
    257 }
    258 
    259 void Controller::setbandwidth(int value)
    260 {
    261     bandwidth.data = value;
    262     setbandwidth();
    263 }
    264 
    265 void Controller::setbandwidth(void)
    266 {
    267     int value = bandwidth.data;
    268     if(bandwidth.exponential == 0) {
    269         float tmp = powf(25.0f, powf(bandwidth.depth / 127.0f, 1.5f)) - 1.0f;
    270         if((value < 64) && (bandwidth.depth >= 64))
    271             tmp = 1.0f;
    272         bandwidth.relbw = (value / 64.0f - 1.0f) * tmp + 1.0f;
    273         if(bandwidth.relbw < 0.01f)
    274             bandwidth.relbw = 0.01f;
    275     }
    276     else
    277         bandwidth.relbw =
    278             powf(25.0f, (value - 64.0f) / 64.0f * (bandwidth.depth / 64.0f));
    279     ;
    280 }
    281 
    282 void Controller::setmodwheel(int value)
    283 {
    284     modwheel.data = value;
    285     setmodwheel();
    286 }
    287 
    288 void Controller::setmodwheel(void)
    289 {
    290     int value = modwheel.data;
    291     if(modwheel.exponential == 0) {
    292         float tmp =
    293             powf(25.0f, powf(modwheel.depth / 127.0f, 1.5f) * 2.0f) / 25.0f;
    294         if((value < 64) && (modwheel.depth >= 64))
    295             tmp = 1.0f;
    296         modwheel.relmod = (value / 64.0f - 1.0f) * tmp + 1.0f;
    297         if(modwheel.relmod < 0.0f)
    298             modwheel.relmod = 0.0f;
    299     }
    300     else
    301         modwheel.relmod =
    302             powf(25.0f, (value - 64.0f) / 64.0f * (modwheel.depth / 80.0f));
    303 }
    304 
    305 void Controller::setfmamp(int value)
    306 {
    307     fmamp.data = value;
    308     setfmamp();
    309 }
    310 
    311 void Controller::setfmamp(void)
    312 {
    313     int value = fmamp.data;
    314     fmamp.relamp = value / 127.0f;
    315     if(fmamp.receive != 0)
    316         fmamp.relamp = value / 127.0f;
    317     else
    318         fmamp.relamp = 1.0f;
    319 }
    320 
    321 void Controller::setvolume(int value)
    322 {
    323     volume.data = value;
    324     setvolume();
    325 }
    326 
    327 void Controller::setvolume(void)
    328 {
    329     int value = volume.data;
    330     if(volume.receive != 0)
    331     {
    332         /* volume.volume = powf(0.1f, (127 - value) / 127.0f * 2.0f); */
    333         /* rather than doing something fancy that results in a value
    334          * of 0 not completely muting the output, do the reasonable
    335          * thing and just give the value the user ordered. */
    336         assert( value <= 127 );
    337 
    338         volume.volume = value / 127.0f;
    339     }
    340     else
    341         volume.volume = 1.0f;
    342 }
    343 
    344 void Controller::setsustain(int value)
    345 {
    346     sustain.data = value;
    347     setsustain();
    348 }
    349 
    350 void Controller::setsustain(void)
    351 {
    352     int value = sustain.data;
    353     if(sustain.receive != 0)
    354         sustain.sustain = ((value < 64) ? 0 : 1);
    355     else
    356         sustain.sustain = 0;
    357 }
    358 
    359 void Controller::setportamento(int value)
    360 {
    361     portamento.data = value;
    362     if(portamento.receive != 0)
    363         portamento.portamento = ((value < 64) ? 0 : 1);
    364 }
    365 
    366 void Controller::setresonancecenter(int value)
    367 {
    368     resonancecenter.data = value;
    369     setresonancecenter();
    370 }
    371 
    372 void Controller::setresonancecenter(void)
    373 {
    374     int value = resonancecenter.data;
    375     resonancecenter.relcenter =
    376         powf(3.0f, (value - 64.0f) / 64.0f * (resonancecenter.depth / 64.0f));
    377 }
    378 
    379 void Controller::setresonancebw(int value)
    380 {
    381     resonancebandwidth.data  = value;
    382     setresonancebw();
    383 }
    384 
    385 void Controller::setresonancebw(void)
    386 {
    387     int value = resonancebandwidth.data;
    388     resonancebandwidth.relbw =
    389         powf(1.5f, (value - 64.0f) / 64.0f * (resonancebandwidth.depth / 127.0f));
    390 }
    391 
    392 
    393 //Returns 0 if there is NRPN or 1 if there is not
    394 int Controller::getnrpn(int *parhi, int *parlo, int *valhi, int *vallo)
    395 {
    396     if(NRPN.receive == 0)
    397         return 1;
    398     if((NRPN.parhi < 0) || (NRPN.parlo < 0) || (NRPN.valhi < 0)
    399        || (NRPN.vallo < 0))
    400         return 1;
    401 
    402     *parhi = NRPN.parhi;
    403     *parlo = NRPN.parlo;
    404     *valhi = NRPN.valhi;
    405     *vallo = NRPN.vallo;
    406     return 0;
    407 }
    408 
    409 
    410 void Controller::setparameternumber(unsigned int type, int value)
    411 {
    412     switch(type) {
    413         case C_nrpnhi:
    414             NRPN.parhi = value;
    415             NRPN.valhi = -1;
    416             NRPN.vallo = -1; //clear the values
    417             break;
    418         case C_nrpnlo:
    419             NRPN.parlo = value;
    420             NRPN.valhi = -1;
    421             NRPN.vallo = -1; //clear the values
    422             break;
    423         case C_dataentryhi:
    424             if((NRPN.parhi >= 0) && (NRPN.parlo >= 0))
    425                 NRPN.valhi = value;
    426             break;
    427         case C_dataentrylo:
    428             if((NRPN.parhi >= 0) && (NRPN.parlo >= 0))
    429                 NRPN.vallo = value;
    430             break;
    431     }
    432 }
    433 
    434 
    435 
    436 void Controller::add2XML(XMLwrapper& xml)
    437 {
    438     xml.addpar("pitchwheel_bendrange", pitchwheel.bendrange);
    439     xml.addpar("pitchwheel_bendrange_down", pitchwheel.bendrange_down);
    440     xml.addparbool("pitchwheel_split", pitchwheel.is_split);
    441 
    442     xml.addparbool("expression_receive", expression.receive);
    443     xml.addpar("panning_depth", panning.depth);
    444     xml.addpar("filter_cutoff_depth", filtercutoff.depth);
    445     xml.addpar("filter_q_depth", filterq.depth);
    446     xml.addpar("bandwidth_depth", bandwidth.depth);
    447     xml.addpar("mod_wheel_depth", modwheel.depth);
    448     xml.addparbool("mod_wheel_exponential", modwheel.exponential);
    449     xml.addparbool("fm_amp_receive", fmamp.receive);
    450     xml.addparbool("volume_receive", volume.receive);
    451     xml.addparbool("sustain_receive", sustain.receive);
    452 
    453     xml.addparbool("portamento_receive", portamento.receive);
    454     xml.addpar("portamento_time", portamento.time);
    455     xml.addpar("portamento_pitchthresh", portamento.pitchthresh);
    456     xml.addpar("portamento_pitchthreshtype", portamento.pitchthreshtype);
    457     xml.addpar("portamento_portamento", portamento.portamento);
    458     xml.addparbool("portamento_auto", portamento.automode);
    459     xml.addpar("portamento_updowntimestretch", portamento.updowntimestretch);
    460     xml.addpar("portamento_proportional", portamento.proportional);
    461     xml.addpar("portamento_proprate", portamento.propRate);
    462     xml.addpar("portamento_propdepth", portamento.propDepth);
    463 
    464     xml.addpar("resonance_center_depth", resonancecenter.depth);
    465     xml.addpar("resonance_bandwidth_depth", resonancebandwidth.depth);
    466 }
    467 
    468 void Controller::getfromXML(XMLwrapper& xml)
    469 {
    470     pitchwheel.bendrange = xml.getpar("pitchwheel_bendrange",
    471                                        pitchwheel.bendrange,
    472                                        -6400,
    473                                        6400);
    474     pitchwheel.bendrange_down = xml.getpar("pitchwheel_bendrange_down",
    475                                        pitchwheel.bendrange_down,
    476                                        -6400,
    477                                        6400);
    478     pitchwheel.is_split = xml.getparbool("pitchwheel_split",
    479                                           pitchwheel.is_split);
    480 
    481     expression.receive = xml.getparbool("expression_receive",
    482                                          expression.receive);
    483     panning.depth      = xml.getpar127("panning_depth", panning.depth);
    484     filtercutoff.depth = xml.getpar127("filter_cutoff_depth",
    485                                         filtercutoff.depth);
    486     filterq.depth        = xml.getpar127("filter_q_depth", filterq.depth);
    487     bandwidth.depth      = xml.getpar127("bandwidth_depth", bandwidth.depth);
    488     modwheel.depth       = xml.getpar127("mod_wheel_depth", modwheel.depth);
    489     modwheel.exponential = xml.getparbool("mod_wheel_exponential",
    490                                            modwheel.exponential);
    491     fmamp.receive = xml.getparbool("fm_amp_receive",
    492                                     fmamp.receive);
    493     volume.receive = xml.getparbool("volume_receive",
    494                                      volume.receive);
    495     sustain.receive = xml.getparbool("sustain_receive",
    496                                       sustain.receive);
    497 
    498     portamento.receive = xml.getparbool("portamento_receive",
    499                                          portamento.receive);
    500     portamento.automode = xml.getparbool("portamento_auto",
    501                                          portamento.automode);
    502     portamento.time = xml.getpar127("portamento_time",
    503                                      portamento.time);
    504     portamento.pitchthresh = xml.getpar127("portamento_pitchthresh",
    505                                             portamento.pitchthresh);
    506     portamento.pitchthreshtype = xml.getpar127("portamento_pitchthreshtype",
    507                                                 portamento.pitchthreshtype);
    508     portamento.portamento = xml.getpar127("portamento_portamento",
    509                                            portamento.portamento);
    510     portamento.updowntimestretch = xml.getpar127(
    511         "portamento_updowntimestretch",
    512         portamento.updowntimestretch);
    513     portamento.proportional = xml.getpar127("portamento_proportional",
    514                                              portamento.proportional);
    515     portamento.propRate = xml.getpar127("portamento_proprate",
    516                                          portamento.propRate);
    517     portamento.propDepth = xml.getpar127("portamento_propdepth",
    518                                           portamento.propDepth);
    519 
    520 
    521     resonancecenter.depth = xml.getpar127("resonance_center_depth",
    522                                            resonancecenter.depth);
    523     resonancebandwidth.depth = xml.getpar127("resonance_bandwidth_depth",
    524                                               resonancebandwidth.depth);
    525 }
    526 
    527 }