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 }