Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2184 lines
60KB

  1. /*
  2. * Carla Native Plugins
  3. * Copyright (C) 2012-2013 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License as
  7. * published by the Free Software Foundation; either version 2 of
  8. * the License, or any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * For a full copy of the GNU General Public License see the GPL.txt file
  16. */
  17. // for UINT32_MAX
  18. #define __STDC_LIMIT_MACROS
  19. #include <cstdint>
  20. #include "CarlaNative.hpp"
  21. #include "CarlaMIDI.h"
  22. #include "CarlaString.hpp"
  23. #include "RtList.hpp"
  24. #include <QtCore/QThread>
  25. #include "zynaddsubfx/Misc/Master.h"
  26. #include "zynaddsubfx/Misc/Part.h"
  27. #include "zynaddsubfx/Misc/Util.h"
  28. #include "zynaddsubfx/Effects/Alienwah.h"
  29. #include "zynaddsubfx/Effects/Chorus.h"
  30. #include "zynaddsubfx/Effects/Distorsion.h"
  31. #include "zynaddsubfx/Effects/DynamicFilter.h"
  32. #include "zynaddsubfx/Effects/Echo.h"
  33. #include "zynaddsubfx/Effects/Phaser.h"
  34. #include "zynaddsubfx/Effects/Reverb.h"
  35. #ifdef WANT_ZYNADDSUBFX_UI
  36. # ifdef override
  37. # define override_hack
  38. # undef override
  39. # endif
  40. # include "zynaddsubfx/UI/common.H"
  41. # include "zynaddsubfx/UI/MasterUI.h"
  42. # include <FL/Fl_Shared_Image.H>
  43. # include <FL/Fl_Tiled_Image.H>
  44. # include <FL/Fl_Theme.H>
  45. # ifdef override_hack
  46. # define override
  47. # undef override_hack
  48. # endif
  49. #endif
  50. #include <ctime>
  51. #include <set>
  52. #include <string>
  53. // Dummy variables and functions for linking purposes
  54. const char* instance_name = nullptr;
  55. class WavFile;
  56. namespace Nio {
  57. bool start(void){return 1;}
  58. void stop(void){}
  59. bool setSource(std::string){return true;}
  60. bool setSink(std::string){return true;}
  61. std::set<std::string> getSources(void){return std::set<std::string>();}
  62. std::set<std::string> getSinks(void){return std::set<std::string>();}
  63. std::string getSource(void){return "";}
  64. std::string getSink(void){return "";}
  65. void waveNew(WavFile*){}
  66. void waveStart(void){}
  67. void waveStop(void){}
  68. void waveEnd(void){}
  69. }
  70. SYNTH_T* synth = nullptr;
  71. #ifdef WANT_ZYNADDSUBFX_UI
  72. #define PIXMAP_PATH "/resources/zynaddsubfx/"
  73. static Fl_Tiled_Image* gModuleBackdrop = nullptr;
  74. static CarlaString gPixmapPath;
  75. extern CarlaString gUiPixmapPath;
  76. void set_module_parameters(Fl_Widget* o)
  77. {
  78. CARLA_ASSERT(gModuleBackdrop != nullptr);
  79. o->box(FL_DOWN_FRAME);
  80. o->align(o->align() | FL_ALIGN_IMAGE_BACKDROP);
  81. o->color(FL_BLACK);
  82. o->labeltype(FL_SHADOW_LABEL);
  83. if (gModuleBackdrop != nullptr)
  84. o->image(gModuleBackdrop);
  85. }
  86. #endif
  87. // -----------------------------------------------------------------------
  88. class ZynAddSubFxPrograms
  89. {
  90. public:
  91. ZynAddSubFxPrograms()
  92. : fInitiated(false)
  93. {
  94. }
  95. ~ZynAddSubFxPrograms()
  96. {
  97. if (! fInitiated)
  98. return;
  99. for (auto it = fPrograms.begin(); it.valid(); it.next())
  100. {
  101. const ProgramInfo*& pInfo(*it);
  102. delete pInfo;
  103. }
  104. fPrograms.clear();
  105. }
  106. void init()
  107. {
  108. if (fInitiated)
  109. return;
  110. fInitiated = true;
  111. fPrograms.append(new ProgramInfo(0, 0, "default"));
  112. Master& master(Master::getInstance());
  113. pthread_mutex_lock(&master.mutex);
  114. // refresh banks
  115. master.bank.rescanforbanks();
  116. for (uint32_t i=0, size = master.bank.banks.size(); i < size; ++i)
  117. {
  118. if (master.bank.banks[i].dir.empty())
  119. continue;
  120. master.bank.loadbank(master.bank.banks[i].dir);
  121. for (unsigned int instrument = 0; instrument < BANK_SIZE; ++instrument)
  122. {
  123. const std::string insName(master.bank.getname(instrument));
  124. if (insName.empty() || insName[0] == '\0' || insName[0] == ' ')
  125. continue;
  126. fPrograms.append(new ProgramInfo(i+1, instrument, insName.c_str()));
  127. }
  128. }
  129. pthread_mutex_unlock(&master.mutex);
  130. }
  131. void load(Master* const master, const uint8_t channel, const uint32_t bank, const uint32_t program)
  132. {
  133. if (bank == 0)
  134. {
  135. pthread_mutex_lock(&master->mutex);
  136. master->partonoff(channel, 1);
  137. master->part[channel]->defaults();
  138. master->part[channel]->applyparameters(false);
  139. pthread_mutex_unlock(&master->mutex);
  140. return;
  141. }
  142. const std::string& bankdir(master->bank.banks[bank-1].dir);
  143. if (! bankdir.empty())
  144. {
  145. pthread_mutex_lock(&master->mutex);
  146. master->partonoff(channel, 1);
  147. master->bank.loadbank(bankdir);
  148. master->bank.loadfromslot(program, master->part[channel]);
  149. master->part[channel]->applyparameters(false);
  150. pthread_mutex_unlock(&master->mutex);
  151. }
  152. }
  153. uint32_t count()
  154. {
  155. return fPrograms.count();
  156. }
  157. const MidiProgram* getInfo(const uint32_t index)
  158. {
  159. if (index >= fPrograms.count())
  160. return nullptr;
  161. const ProgramInfo*& pInfo(fPrograms.getAt(index));
  162. fRetProgram.bank = pInfo->bank;
  163. fRetProgram.program = pInfo->prog;
  164. fRetProgram.name = pInfo->name;
  165. return &fRetProgram;
  166. }
  167. private:
  168. struct ProgramInfo {
  169. uint32_t bank;
  170. uint32_t prog;
  171. const char* name;
  172. ProgramInfo(uint32_t bank_, uint32_t prog_, const char* name_)
  173. : bank(bank_),
  174. prog(prog_),
  175. name(carla_strdup(name_)) {}
  176. ~ProgramInfo()
  177. {
  178. if (name != nullptr)
  179. {
  180. delete[] name;
  181. name = nullptr;
  182. }
  183. }
  184. #ifdef CARLA_PROPER_CPP11_SUPPORT
  185. ProgramInfo() = delete;
  186. ProgramInfo(ProgramInfo&) = delete;
  187. ProgramInfo(const ProgramInfo&) = delete;
  188. #endif
  189. };
  190. bool fInitiated;
  191. MidiProgram fRetProgram;
  192. NonRtList<const ProgramInfo*> fPrograms;
  193. CARLA_DECLARE_NON_COPYABLE(ZynAddSubFxPrograms)
  194. };
  195. static ZynAddSubFxPrograms sPrograms;
  196. // -----------------------------------------------------------------------
  197. class ZynAddSubFxInstanceCount
  198. {
  199. public:
  200. ZynAddSubFxInstanceCount()
  201. : fCount(0)
  202. {
  203. }
  204. ~ZynAddSubFxInstanceCount()
  205. {
  206. CARLA_ASSERT(fCount == 0);
  207. }
  208. void addOne(const HostDescriptor* const host)
  209. {
  210. if (fCount++ == 0)
  211. {
  212. CARLA_ASSERT(synth == nullptr);
  213. CARLA_ASSERT(denormalkillbuf == nullptr);
  214. reinit(host);
  215. #ifdef WANT_ZYNADDSUBFX_UI
  216. if (gPixmapPath.isEmpty())
  217. {
  218. gPixmapPath = host->resource_dir;
  219. gPixmapPath += PIXMAP_PATH;
  220. gUiPixmapPath = gPixmapPath;
  221. }
  222. #endif
  223. }
  224. }
  225. void removeOne()
  226. {
  227. if (--fCount == 0)
  228. {
  229. CARLA_ASSERT(synth != nullptr);
  230. CARLA_ASSERT(denormalkillbuf != nullptr);
  231. Master::deleteInstance();
  232. delete[] denormalkillbuf;
  233. denormalkillbuf = nullptr;
  234. delete synth;
  235. synth = nullptr;
  236. }
  237. }
  238. void reinit(const HostDescriptor* const host)
  239. {
  240. Master::deleteInstance();
  241. if (denormalkillbuf != nullptr)
  242. {
  243. delete[] denormalkillbuf;
  244. denormalkillbuf = nullptr;
  245. }
  246. if (synth != nullptr)
  247. {
  248. delete synth;
  249. synth = nullptr;
  250. }
  251. synth = new SYNTH_T();
  252. synth->buffersize = host->get_buffer_size(host->handle);
  253. synth->samplerate = host->get_sample_rate(host->handle);
  254. synth->alias();
  255. config.init();
  256. config.cfg.SoundBufferSize = synth->buffersize;
  257. config.cfg.SampleRate = synth->samplerate;
  258. config.cfg.GzipCompression = 0;
  259. sprng(std::time(nullptr));
  260. denormalkillbuf = new float[synth->buffersize];
  261. for (int i=0; i < synth->buffersize; ++i)
  262. denormalkillbuf[i] = (RND - 0.5f) * 1e-16;
  263. Master::getInstance();
  264. }
  265. void maybeReinit(const HostDescriptor* const host)
  266. {
  267. if (host->get_buffer_size(host->handle) == static_cast<uint32_t>(synth->buffersize) &&
  268. host->get_sample_rate(host->handle) == static_cast<double>(synth->samplerate))
  269. return;
  270. reinit(host);
  271. }
  272. private:
  273. int fCount;
  274. CARLA_DECLARE_NON_COPYABLE(ZynAddSubFxInstanceCount)
  275. };
  276. static ZynAddSubFxInstanceCount sInstanceCount;
  277. // -----------------------------------------------------------------------
  278. class ZynAddSubFxThread : public QThread
  279. {
  280. public:
  281. ZynAddSubFxThread(Master* const master, const HostDescriptor* const host)
  282. : fMaster(master),
  283. kHost(host),
  284. #ifdef WANT_ZYNADDSUBFX_UI
  285. fUi(nullptr),
  286. fUiClosed(0),
  287. fNextUiAction(-1),
  288. #endif
  289. fQuit(false),
  290. fChangeProgram(false),
  291. fNextChannel(0),
  292. fNextBank(0),
  293. fNextProgram(0)
  294. {
  295. }
  296. ~ZynAddSubFxThread()
  297. {
  298. // must be closed by now
  299. #ifdef WANT_ZYNADDSUBFX_UI
  300. CARLA_ASSERT(fUi == nullptr);
  301. #endif
  302. CARLA_ASSERT(fQuit);
  303. }
  304. void loadProgramLater(const uint8_t channel, const uint32_t bank, const uint32_t program)
  305. {
  306. fNextChannel = channel;
  307. fNextBank = bank;
  308. fNextProgram = program;
  309. fChangeProgram = true;
  310. }
  311. void stopLoadProgramLater()
  312. {
  313. fChangeProgram = false;
  314. fNextChannel = 0;
  315. fNextBank = 0;
  316. fNextProgram = 0;
  317. }
  318. void setMaster(Master* const master)
  319. {
  320. fMaster = master;
  321. }
  322. void stop()
  323. {
  324. fQuit = true;
  325. quit();
  326. }
  327. #ifdef WANT_ZYNADDSUBFX_UI
  328. void uiHide()
  329. {
  330. fNextUiAction = 0;
  331. }
  332. void uiShow()
  333. {
  334. fNextUiAction = 1;
  335. }
  336. void uiRepaint()
  337. {
  338. if (fUi != nullptr)
  339. fNextUiAction = 2;
  340. }
  341. #endif
  342. protected:
  343. void run() override
  344. {
  345. while (! fQuit)
  346. {
  347. #ifdef WANT_ZYNADDSUBFX_UI
  348. Fl::lock();
  349. if (fNextUiAction == 2) // repaint
  350. {
  351. CARLA_ASSERT(fUi != nullptr);
  352. if (fUi != nullptr)
  353. fUi->refresh_master_ui();
  354. }
  355. else if (fNextUiAction == 1) // init/show
  356. {
  357. static bool initialized = false;
  358. if (! initialized)
  359. {
  360. initialized = true;
  361. fl_register_images();
  362. Fl_Dial::default_style(Fl_Dial::PIXMAP_DIAL);
  363. if (Fl_Shared_Image* const img = Fl_Shared_Image::get(gPixmapPath + "knob.png"))
  364. Fl_Dial::default_image(img);
  365. if (Fl_Shared_Image* const img = Fl_Shared_Image::get(gPixmapPath + "window_backdrop.png"))
  366. Fl::scheme_bg(new Fl_Tiled_Image(img));
  367. if(Fl_Shared_Image* const img = Fl_Shared_Image::get(gPixmapPath + "module_backdrop.png"))
  368. gModuleBackdrop = new Fl_Tiled_Image(img);
  369. Fl::background(50, 50, 50);
  370. Fl::background2(70, 70, 70);
  371. Fl::foreground(255, 255, 255);
  372. Fl_Theme::set("Cairo");
  373. }
  374. CARLA_ASSERT(fUi == nullptr);
  375. if (fUi == nullptr)
  376. {
  377. fUiClosed = 0;
  378. fUi = new MasterUI(fMaster, &fUiClosed);
  379. fUi->showUI();
  380. }
  381. }
  382. else if (fNextUiAction == 0) // close
  383. {
  384. CARLA_ASSERT(fUi != nullptr);
  385. if (fUi != nullptr)
  386. {
  387. delete fUi;
  388. fUi = nullptr;
  389. }
  390. }
  391. fNextUiAction = -1;
  392. if (fUiClosed != 0)
  393. {
  394. fUiClosed = 0;
  395. fNextUiAction = 0;
  396. kHost->ui_closed(kHost->handle);
  397. }
  398. Fl::check();
  399. Fl::unlock();
  400. #endif
  401. if (fChangeProgram)
  402. {
  403. fChangeProgram = false;
  404. sPrograms.load(fMaster, fNextChannel, fNextBank, fNextProgram);
  405. fNextChannel = 0;
  406. fNextBank = 0;
  407. fNextProgram = 0;
  408. #ifdef WANT_ZYNADDSUBFX_UI
  409. if (fUi != nullptr)
  410. {
  411. Fl::lock();
  412. fUi->refresh_master_ui();
  413. Fl::unlock();
  414. }
  415. #endif
  416. carla_msleep(15);
  417. }
  418. else
  419. {
  420. carla_msleep(30);
  421. }
  422. }
  423. #ifdef WANT_ZYNADDSUBFX_UI
  424. if (fQuit && fUi != nullptr)
  425. {
  426. Fl::lock();
  427. delete fUi;
  428. fUi = nullptr;
  429. Fl::check();
  430. Fl::unlock();
  431. }
  432. #endif
  433. }
  434. private:
  435. Master* fMaster;
  436. const HostDescriptor* const kHost;
  437. #ifdef WANT_ZYNADDSUBFX_UI
  438. MasterUI* fUi;
  439. int fUiClosed;
  440. int fNextUiAction;
  441. #endif
  442. bool fQuit;
  443. bool fChangeProgram;
  444. uint8_t fNextChannel;
  445. uint32_t fNextBank;
  446. uint32_t fNextProgram;
  447. };
  448. // -----------------------------------------------------------------------
  449. #define ZynPluginDescriptorClassEND(ClassName) \
  450. public: \
  451. static PluginHandle _instantiate(const PluginDescriptor*, HostDescriptor* host) \
  452. { \
  453. sInstanceCount.addOne(host); \
  454. return new ClassName(host); \
  455. } \
  456. static void _cleanup(PluginHandle handle) \
  457. { \
  458. delete (ClassName*)handle; \
  459. sInstanceCount.removeOne(); \
  460. }
  461. // -----------------------------------------------------------------------
  462. class FxAbstractPlugin : public PluginDescriptorClass
  463. {
  464. protected:
  465. FxAbstractPlugin(const HostDescriptor* const host, const uint32_t paramCount, const uint32_t programCount)
  466. : PluginDescriptorClass(host),
  467. fEffect(nullptr),
  468. efxoutl(new float[synth->buffersize]),
  469. efxoutr(new float[synth->buffersize]),
  470. fFirstInit(true),
  471. kParamCount(paramCount-2), // volume and pan handled by host
  472. kProgramCount(programCount)
  473. {
  474. carla_zeroFloat(efxoutl, synth->buffersize);
  475. carla_zeroFloat(efxoutr, synth->buffersize);
  476. }
  477. ~FxAbstractPlugin() override
  478. {
  479. CARLA_ASSERT(fEffect != nullptr);
  480. if (efxoutl != nullptr)
  481. {
  482. delete[] efxoutl;
  483. efxoutl = nullptr;
  484. }
  485. if (efxoutr != nullptr)
  486. {
  487. delete[] efxoutr;
  488. efxoutr = nullptr;
  489. }
  490. if (fEffect != nullptr)
  491. {
  492. delete fEffect;
  493. fEffect = nullptr;
  494. }
  495. }
  496. // -------------------------------------------------------------------
  497. // Plugin parameter calls
  498. uint32_t getParameterCount() final
  499. {
  500. return kParamCount;
  501. }
  502. float getParameterValue(const uint32_t index) final
  503. {
  504. return fEffect->getpar(index+2);
  505. }
  506. // -------------------------------------------------------------------
  507. // Plugin midi-program calls
  508. uint32_t getMidiProgramCount() final
  509. {
  510. return kProgramCount;
  511. }
  512. // -------------------------------------------------------------------
  513. // Plugin state calls
  514. void setParameterValue(const uint32_t index, const float value) final
  515. {
  516. fEffect->changepar(index+2, value);
  517. fFirstInit = false;
  518. }
  519. void setMidiProgram(const uint8_t, const uint32_t, const uint32_t program) final
  520. {
  521. fEffect->setpreset(program);
  522. const float volume(float(fEffect->getpar(0))/127.0f);
  523. hostDispatcher(HOST_OPCODE_SET_VOLUME, 0, 0, nullptr, volume);
  524. const unsigned char pan(fEffect->getpar(1));
  525. const float panning(float(pan)/63.5f-1.0f);
  526. hostDispatcher(HOST_OPCODE_SET_PANNING, 0, 0, nullptr, (pan == 64) ? 0.0f : panning);
  527. fEffect->changepar(0, 127);
  528. fEffect->changepar(1, 64);
  529. }
  530. // -------------------------------------------------------------------
  531. // Plugin process calls
  532. void activate() final
  533. {
  534. fEffect->cleanup();
  535. if (fFirstInit)
  536. {
  537. fFirstInit = false;
  538. hostDispatcher(HOST_OPCODE_SET_DRYWET, 0, 0, nullptr, 0.5f);
  539. }
  540. }
  541. void process(float** const inBuffer, float** const outBuffer, const uint32_t frames, const uint32_t, const MidiEvent* const) final
  542. {
  543. CARLA_ASSERT(synth->buffersize == static_cast<int>(frames));
  544. fEffect->out(Stereo<float*>(inBuffer[0], inBuffer[1]));
  545. carla_copyFloat(outBuffer[0], efxoutl, frames);
  546. carla_copyFloat(outBuffer[1], efxoutr, frames);
  547. }
  548. // -------------------------------------------------------------------
  549. // Plugin dispatcher
  550. intptr_t pluginDispatcher(const PluginDispatcherOpcode opcode, const int32_t, const intptr_t, void* const, const float) final
  551. {
  552. switch (opcode)
  553. {
  554. case PLUGIN_OPCODE_BUFFER_SIZE_CHANGED:
  555. {
  556. const uint32_t bufferSize(getBufferSize());
  557. delete[] efxoutl;
  558. delete[] efxoutr;
  559. efxoutl = new float[bufferSize];
  560. efxoutr = new float[bufferSize];
  561. carla_zeroFloat(efxoutl, synth->buffersize);
  562. carla_zeroFloat(efxoutr, synth->buffersize);
  563. *((float**)&fEffect->efxoutl) = efxoutl;
  564. *((float**)&fEffect->efxoutr) = efxoutr;
  565. // no break
  566. }
  567. case PLUGIN_OPCODE_SAMPLE_RATE_CHANGED:
  568. sInstanceCount.maybeReinit(hostHandle());
  569. break;
  570. default:
  571. break;
  572. }
  573. return 0;
  574. }
  575. Effect* fEffect;
  576. float* efxoutl;
  577. float* efxoutr;
  578. bool fFirstInit;
  579. const uint32_t kParamCount;
  580. const uint32_t kProgramCount;
  581. };
  582. // -----------------------------------------------------------------------
  583. class FxAlienWahPlugin : public FxAbstractPlugin
  584. {
  585. public:
  586. FxAlienWahPlugin(const HostDescriptor* const host)
  587. : FxAbstractPlugin(host, 11, 4)
  588. {
  589. fEffect = new Alienwah(false, efxoutl, efxoutr);
  590. }
  591. protected:
  592. // -------------------------------------------------------------------
  593. // Plugin parameter calls
  594. const Parameter* getParameterInfo(const uint32_t index) override
  595. {
  596. if (index >= kParamCount)
  597. return nullptr;
  598. static Parameter param;
  599. static ParameterScalePoint scalePoints[2];
  600. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  601. param.name = nullptr;
  602. param.unit = nullptr;
  603. param.ranges.def = 1.0f;
  604. param.ranges.min = 0.0f;
  605. param.ranges.max = 127.0f;
  606. param.ranges.step = 1.0f;
  607. param.ranges.stepSmall = 1.0f;
  608. param.ranges.stepLarge = 20.0f;
  609. param.scalePointCount = 0;
  610. param.scalePoints = nullptr;
  611. switch (index)
  612. {
  613. case 0:
  614. hints |= PARAMETER_IS_AUTOMABLE;
  615. param.name = "LFO Frequency";
  616. param.ranges.def = 70.0f;
  617. break;
  618. case 1:
  619. hints |= PARAMETER_IS_AUTOMABLE;
  620. param.name = "LFO Randomness";
  621. param.ranges.def = 0.0f;
  622. break;
  623. case 2:
  624. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN|PARAMETER_USES_SCALEPOINTS;
  625. param.name = "LFO Type";
  626. param.ranges.def = 0.0f;
  627. param.ranges.max = 1.0f;
  628. param.scalePointCount = 2;
  629. param.scalePoints = scalePoints;
  630. scalePoints[0].label = "Sine";
  631. scalePoints[1].label = "Triangle";
  632. scalePoints[0].value = 0.0f;
  633. scalePoints[1].value = 1.0f;
  634. break;
  635. case 3:
  636. hints |= PARAMETER_IS_AUTOMABLE;
  637. param.name = "LFO Stereo";
  638. param.ranges.def = 62.0f;
  639. break;
  640. case 4:
  641. hints |= PARAMETER_IS_AUTOMABLE;
  642. param.name = "Depth";
  643. param.ranges.def = 60.0f;
  644. break;
  645. case 5:
  646. hints |= PARAMETER_IS_AUTOMABLE;
  647. param.name = "Feedback";
  648. param.ranges.def = 105.0f;
  649. break;
  650. case 6:
  651. param.name = "Delay";
  652. param.ranges.def = 25.0f;
  653. param.ranges.min = 1.0f;
  654. param.ranges.max = 100.0f;
  655. break;
  656. case 7:
  657. hints |= PARAMETER_IS_AUTOMABLE;
  658. param.name = "L/R Cross";
  659. param.ranges.def = 0.0f;
  660. break;
  661. case 8:
  662. hints |= PARAMETER_IS_AUTOMABLE;
  663. param.name = "Phase";
  664. param.ranges.def = 64.0f;
  665. break;
  666. }
  667. param.hints = static_cast<ParameterHints>(hints);
  668. return &param;
  669. }
  670. // -------------------------------------------------------------------
  671. // Plugin midi-program calls
  672. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  673. {
  674. if (index >= kProgramCount)
  675. return nullptr;
  676. static MidiProgram midiProg;
  677. midiProg.bank = 0;
  678. midiProg.program = index;
  679. switch (index)
  680. {
  681. case 0:
  682. midiProg.name = "AlienWah1";
  683. break;
  684. case 1:
  685. midiProg.name = "AlienWah2";
  686. break;
  687. case 2:
  688. midiProg.name = "AlienWah3";
  689. break;
  690. case 3:
  691. midiProg.name = "AlienWah4";
  692. break;
  693. default:
  694. midiProg.name = nullptr;
  695. break;
  696. }
  697. return &midiProg;
  698. }
  699. ZynPluginDescriptorClassEND(FxAlienWahPlugin)
  700. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxAlienWahPlugin)
  701. };
  702. // -----------------------------------------------------------------------
  703. class FxChorusPlugin : public FxAbstractPlugin
  704. {
  705. public:
  706. FxChorusPlugin(const HostDescriptor* const host)
  707. : FxAbstractPlugin(host, 12, 10)
  708. {
  709. fEffect = new Chorus(false, efxoutl, efxoutr);
  710. }
  711. protected:
  712. // -------------------------------------------------------------------
  713. // Plugin parameter calls
  714. const Parameter* getParameterInfo(const uint32_t index) override
  715. {
  716. if (index >= kParamCount)
  717. return nullptr;
  718. static Parameter param;
  719. static ParameterScalePoint scalePoints[2];
  720. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  721. param.name = nullptr;
  722. param.unit = nullptr;
  723. param.ranges.def = 1.0f;
  724. param.ranges.min = 0.0f;
  725. param.ranges.max = 127.0f;
  726. param.ranges.step = 1.0f;
  727. param.ranges.stepSmall = 1.0f;
  728. param.ranges.stepLarge = 20.0f;
  729. param.scalePointCount = 0;
  730. param.scalePoints = nullptr;
  731. switch (index)
  732. {
  733. case 0:
  734. hints |= PARAMETER_IS_AUTOMABLE;
  735. param.name = "LFO Frequency";
  736. param.ranges.def = 50.0f;
  737. break;
  738. case 1:
  739. hints |= PARAMETER_IS_AUTOMABLE;
  740. param.name = "LFO Randomness";
  741. param.ranges.def = 0.0f;
  742. break;
  743. case 2:
  744. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN|PARAMETER_USES_SCALEPOINTS;
  745. param.name = "LFO Type";
  746. param.ranges.def = 0.0f;
  747. param.ranges.max = 1.0f;
  748. param.scalePointCount = 2;
  749. param.scalePoints = scalePoints;
  750. scalePoints[0].label = "Sine";
  751. scalePoints[1].label = "Triangle";
  752. scalePoints[0].value = 0.0f;
  753. scalePoints[1].value = 1.0f;
  754. break;
  755. case 3:
  756. hints |= PARAMETER_IS_AUTOMABLE;
  757. param.name = "LFO Stereo";
  758. param.ranges.def = 90.0f;
  759. break;
  760. case 4:
  761. hints |= PARAMETER_IS_AUTOMABLE;
  762. param.name = "Depth";
  763. param.ranges.def = 40.0f;
  764. break;
  765. case 5:
  766. hints |= PARAMETER_IS_AUTOMABLE;
  767. param.name = "Delay";
  768. param.ranges.def = 85.0f;
  769. break;
  770. case 6:
  771. hints |= PARAMETER_IS_AUTOMABLE;
  772. param.name = "Feedback";
  773. param.ranges.def = 64.0f;
  774. break;
  775. case 7:
  776. hints |= PARAMETER_IS_AUTOMABLE;
  777. param.name = "L/R Cross";
  778. param.ranges.def = 119.0f;
  779. break;
  780. case 8:
  781. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  782. param.name = "Flange Mode";
  783. param.ranges.def = 0.0f;
  784. param.ranges.max = 1.0f;
  785. break;
  786. case 9:
  787. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  788. param.name = "Subtract Output";
  789. param.ranges.def = 0.0f;
  790. param.ranges.max = 1.0f;
  791. break;
  792. }
  793. param.hints = static_cast<ParameterHints>(hints);
  794. return &param;
  795. }
  796. // -------------------------------------------------------------------
  797. // Plugin midi-program calls
  798. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  799. {
  800. if (index >= kProgramCount)
  801. return nullptr;
  802. static MidiProgram midiProg;
  803. midiProg.bank = 0;
  804. midiProg.program = index;
  805. switch (index)
  806. {
  807. case 0:
  808. midiProg.name = "Chorus1";
  809. break;
  810. case 1:
  811. midiProg.name = "Chorus2";
  812. break;
  813. case 2:
  814. midiProg.name = "Chorus3";
  815. break;
  816. case 3:
  817. midiProg.name = "Celeste1";
  818. break;
  819. case 4:
  820. midiProg.name = "Celeste2";
  821. break;
  822. case 5:
  823. midiProg.name = "Flange1";
  824. break;
  825. case 6:
  826. midiProg.name = "Flange2";
  827. break;
  828. case 7:
  829. midiProg.name = "Flange3";
  830. break;
  831. case 8:
  832. midiProg.name = "Flange4";
  833. break;
  834. case 9:
  835. midiProg.name = "Flange5";
  836. break;
  837. default:
  838. midiProg.name = nullptr;
  839. break;
  840. }
  841. return &midiProg;
  842. }
  843. ZynPluginDescriptorClassEND(FxChorusPlugin)
  844. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxChorusPlugin)
  845. };
  846. // -----------------------------------------------------------------------
  847. class FxDistortionPlugin : public FxAbstractPlugin
  848. {
  849. public:
  850. FxDistortionPlugin(const HostDescriptor* const host)
  851. : FxAbstractPlugin(host, 11, 6)
  852. {
  853. fEffect = new Distorsion(false, efxoutl, efxoutr);
  854. }
  855. protected:
  856. // -------------------------------------------------------------------
  857. // Plugin parameter calls
  858. const Parameter* getParameterInfo(const uint32_t index) override
  859. {
  860. if (index >= kParamCount)
  861. return nullptr;
  862. static Parameter param;
  863. static ParameterScalePoint scalePoints[14];
  864. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  865. param.name = nullptr;
  866. param.unit = nullptr;
  867. param.ranges.def = 1.0f;
  868. param.ranges.min = 0.0f;
  869. param.ranges.max = 127.0f;
  870. param.ranges.step = 1.0f;
  871. param.ranges.stepSmall = 1.0f;
  872. param.ranges.stepLarge = 20.0f;
  873. param.scalePointCount = 0;
  874. param.scalePoints = nullptr;
  875. switch (index)
  876. {
  877. case 0:
  878. hints |= PARAMETER_IS_AUTOMABLE;
  879. param.name = "L/R Cross";
  880. param.ranges.def = 35.0f;
  881. break;
  882. case 1:
  883. hints |= PARAMETER_IS_AUTOMABLE;
  884. param.name = "Drive";
  885. param.ranges.def = 56.0f;
  886. break;
  887. case 2:
  888. hints |= PARAMETER_IS_AUTOMABLE;
  889. param.name = "Level";
  890. param.ranges.def = 70.0f;
  891. break;
  892. case 3:
  893. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_USES_SCALEPOINTS;
  894. param.name = "Type";
  895. param.ranges.def = 0.0f;
  896. param.ranges.max = 13.0f;
  897. param.scalePointCount = 14;
  898. param.scalePoints = scalePoints;
  899. scalePoints[ 0].label = "Arctangent";
  900. scalePoints[ 1].label = "Asymmetric";
  901. scalePoints[ 2].label = "Pow";
  902. scalePoints[ 3].label = "Sine";
  903. scalePoints[ 4].label = "Quantisize";
  904. scalePoints[ 5].label = "Zigzag";
  905. scalePoints[ 6].label = "Limiter";
  906. scalePoints[ 7].label = "Upper Limiter";
  907. scalePoints[ 8].label = "Lower Limiter";
  908. scalePoints[ 9].label = "Inverse Limiter";
  909. scalePoints[10].label = "Clip";
  910. scalePoints[11].label = "Asym2";
  911. scalePoints[12].label = "Pow2";
  912. scalePoints[13].label = "Sigmoid";
  913. scalePoints[ 0].value = 0.0f;
  914. scalePoints[ 1].value = 1.0f;
  915. scalePoints[ 2].value = 2.0f;
  916. scalePoints[ 3].value = 3.0f;
  917. scalePoints[ 4].value = 4.0f;
  918. scalePoints[ 5].value = 5.0f;
  919. scalePoints[ 6].value = 6.0f;
  920. scalePoints[ 7].value = 7.0f;
  921. scalePoints[ 8].value = 8.0f;
  922. scalePoints[ 9].value = 9.0f;
  923. scalePoints[10].value = 10.0f;
  924. scalePoints[11].value = 11.0f;
  925. scalePoints[12].value = 12.0f;
  926. scalePoints[13].value = 13.0f;
  927. break;
  928. case 4:
  929. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  930. param.name = "Negate";
  931. param.ranges.def = 0.0f;
  932. param.ranges.max = 1.0f;
  933. break;
  934. case 5:
  935. hints |= PARAMETER_IS_AUTOMABLE;
  936. param.name = "Low-Pass Filter";
  937. param.ranges.def = 96.0f;
  938. break;
  939. case 6:
  940. hints |= PARAMETER_IS_AUTOMABLE;
  941. param.name = "High-Pass Filter";
  942. param.ranges.def = 0.0f;
  943. break;
  944. case 7:
  945. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  946. param.name = "Stereo";
  947. param.ranges.def = 0.0f;
  948. param.ranges.max = 1.0f;
  949. break;
  950. case 8:
  951. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  952. param.name = "Pre-Filtering";
  953. param.ranges.def = 0.0f;
  954. param.ranges.max = 1.0f;
  955. break;
  956. }
  957. param.hints = static_cast<ParameterHints>(hints);
  958. return &param;
  959. }
  960. // -------------------------------------------------------------------
  961. // Plugin midi-program calls
  962. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  963. {
  964. if (index >= kProgramCount)
  965. return nullptr;
  966. static MidiProgram midiProg;
  967. midiProg.bank = 0;
  968. midiProg.program = index;
  969. switch (index)
  970. {
  971. case 0:
  972. midiProg.name = "Overdrive 1";
  973. break;
  974. case 1:
  975. midiProg.name = "Overdrive 2";
  976. break;
  977. case 2:
  978. midiProg.name = "A. Exciter 1";
  979. break;
  980. case 3:
  981. midiProg.name = "A. Exciter 2";
  982. break;
  983. case 4:
  984. midiProg.name = "Guitar Amp";
  985. break;
  986. case 5:
  987. midiProg.name = "Quantisize";
  988. break;
  989. default:
  990. midiProg.name = nullptr;
  991. break;
  992. }
  993. return &midiProg;
  994. }
  995. ZynPluginDescriptorClassEND(FxDistortionPlugin)
  996. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxDistortionPlugin)
  997. };
  998. // -----------------------------------------------------------------------
  999. class FxDynamicFilterPlugin : public FxAbstractPlugin
  1000. {
  1001. public:
  1002. FxDynamicFilterPlugin(const HostDescriptor* const host)
  1003. : FxAbstractPlugin(host, 10, 5)
  1004. {
  1005. fEffect = new DynamicFilter(false, efxoutl, efxoutr);
  1006. }
  1007. protected:
  1008. // -------------------------------------------------------------------
  1009. // Plugin parameter calls
  1010. const Parameter* getParameterInfo(const uint32_t index) override
  1011. {
  1012. if (index >= kParamCount)
  1013. return nullptr;
  1014. static Parameter param;
  1015. static ParameterScalePoint scalePoints[2];
  1016. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  1017. param.name = nullptr;
  1018. param.unit = nullptr;
  1019. param.ranges.def = 1.0f;
  1020. param.ranges.min = 0.0f;
  1021. param.ranges.max = 127.0f;
  1022. param.ranges.step = 1.0f;
  1023. param.ranges.stepSmall = 1.0f;
  1024. param.ranges.stepLarge = 20.0f;
  1025. param.scalePointCount = 0;
  1026. param.scalePoints = nullptr;
  1027. switch (index)
  1028. {
  1029. case 0:
  1030. hints |= PARAMETER_IS_AUTOMABLE;
  1031. param.name = "LFO Frequency";
  1032. param.ranges.def = 80.0f;
  1033. break;
  1034. case 1:
  1035. hints |= PARAMETER_IS_AUTOMABLE;
  1036. param.name = "LFO Randomness";
  1037. param.ranges.def = 0.0f;
  1038. break;
  1039. case 2:
  1040. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN|PARAMETER_USES_SCALEPOINTS;
  1041. param.name = "LFO Type";
  1042. param.ranges.def = 0.0f;
  1043. param.ranges.max = 1.0f;
  1044. param.scalePointCount = 2;
  1045. param.scalePoints = scalePoints;
  1046. scalePoints[0].label = "Sine";
  1047. scalePoints[1].label = "Triangle";
  1048. scalePoints[0].value = 0.0f;
  1049. scalePoints[1].value = 1.0f;
  1050. break;
  1051. case 3:
  1052. hints |= PARAMETER_IS_AUTOMABLE;
  1053. param.name = "LFO Stereo";
  1054. param.ranges.def = 64.0f;
  1055. break;
  1056. case 4:
  1057. hints |= PARAMETER_IS_AUTOMABLE;
  1058. param.name = "LFO Depth";
  1059. param.ranges.def = 0.0f;
  1060. break;
  1061. case 5:
  1062. hints |= PARAMETER_IS_AUTOMABLE;
  1063. param.name = "Amp sns";
  1064. param.ranges.def = 90.0f;
  1065. break;
  1066. case 6:
  1067. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  1068. param.name = "Amp sns inv";
  1069. param.ranges.def = 0.0f;
  1070. param.ranges.max = 1.0f;
  1071. break;
  1072. case 7:
  1073. hints |= PARAMETER_IS_AUTOMABLE;
  1074. param.name = "Amp Smooth";
  1075. param.ranges.def = 60.0f;
  1076. break;
  1077. }
  1078. param.hints = static_cast<ParameterHints>(hints);
  1079. return &param;
  1080. }
  1081. // -------------------------------------------------------------------
  1082. // Plugin midi-program calls
  1083. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  1084. {
  1085. if (index >= kProgramCount)
  1086. return nullptr;
  1087. static MidiProgram midiProg;
  1088. midiProg.bank = 0;
  1089. midiProg.program = index;
  1090. switch (index)
  1091. {
  1092. case 0:
  1093. midiProg.name = "WahWah";
  1094. break;
  1095. case 1:
  1096. midiProg.name = "AutoWah";
  1097. break;
  1098. case 2:
  1099. midiProg.name = "Sweep";
  1100. break;
  1101. case 3:
  1102. midiProg.name = "VocalMorph1";
  1103. break;
  1104. case 4:
  1105. midiProg.name = "VocalMorph2";
  1106. break;
  1107. default:
  1108. midiProg.name = nullptr;
  1109. break;
  1110. }
  1111. return &midiProg;
  1112. }
  1113. ZynPluginDescriptorClassEND(FxDynamicFilterPlugin)
  1114. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxDynamicFilterPlugin)
  1115. };
  1116. // -----------------------------------------------------------------------
  1117. class FxEchoPlugin : public FxAbstractPlugin
  1118. {
  1119. public:
  1120. FxEchoPlugin(const HostDescriptor* const host)
  1121. : FxAbstractPlugin(host, 7, 9)
  1122. {
  1123. fEffect = new Echo(false, efxoutl, efxoutr);
  1124. }
  1125. protected:
  1126. // -------------------------------------------------------------------
  1127. // Plugin parameter calls
  1128. const Parameter* getParameterInfo(const uint32_t index) override
  1129. {
  1130. if (index >= kParamCount)
  1131. return nullptr;
  1132. static Parameter param;
  1133. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  1134. param.name = nullptr;
  1135. param.unit = nullptr;
  1136. param.ranges.def = 1.0f;
  1137. param.ranges.min = 0.0f;
  1138. param.ranges.max = 127.0f;
  1139. param.ranges.step = 1.0f;
  1140. param.ranges.stepSmall = 1.0f;
  1141. param.ranges.stepLarge = 20.0f;
  1142. param.scalePointCount = 0;
  1143. param.scalePoints = nullptr;
  1144. switch (index)
  1145. {
  1146. case 0:
  1147. hints |= PARAMETER_IS_AUTOMABLE;
  1148. param.name = "Delay";
  1149. param.ranges.def = 35.0f;
  1150. break;
  1151. case 1:
  1152. hints |= PARAMETER_IS_AUTOMABLE;
  1153. param.name = "L/R Delay";
  1154. param.ranges.def = 64.0f;
  1155. break;
  1156. case 2:
  1157. hints |= PARAMETER_IS_AUTOMABLE;
  1158. param.name = "L/R Cross";
  1159. param.ranges.def = 30.0f;
  1160. break;
  1161. case 3:
  1162. hints |= PARAMETER_IS_AUTOMABLE;
  1163. param.name = "Feedback";
  1164. param.ranges.def = 59.0f;
  1165. break;
  1166. case 4:
  1167. hints |= PARAMETER_IS_AUTOMABLE;
  1168. param.name = "High Damp";
  1169. param.ranges.def = 0.0f;
  1170. break;
  1171. }
  1172. param.hints = static_cast<ParameterHints>(hints);
  1173. return &param;
  1174. }
  1175. // -------------------------------------------------------------------
  1176. // Plugin midi-program calls
  1177. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  1178. {
  1179. if (index >= kProgramCount)
  1180. return nullptr;
  1181. static MidiProgram midiProg;
  1182. midiProg.bank = 0;
  1183. midiProg.program = index;
  1184. switch (index)
  1185. {
  1186. case 0:
  1187. midiProg.name = "Echo 1";
  1188. break;
  1189. case 1:
  1190. midiProg.name = "Echo 2";
  1191. break;
  1192. case 2:
  1193. midiProg.name = "Echo 3";
  1194. break;
  1195. case 3:
  1196. midiProg.name = "Simple Echo";
  1197. break;
  1198. case 4:
  1199. midiProg.name = "Canyon";
  1200. break;
  1201. case 5:
  1202. midiProg.name = "Panning Echo 1";
  1203. break;
  1204. case 6:
  1205. midiProg.name = "Panning Echo 2";
  1206. break;
  1207. case 7:
  1208. midiProg.name = "Panning Echo 3";
  1209. break;
  1210. case 8:
  1211. midiProg.name = "Feedback Echo";
  1212. break;
  1213. default:
  1214. midiProg.name = nullptr;
  1215. break;
  1216. }
  1217. return &midiProg;
  1218. }
  1219. ZynPluginDescriptorClassEND(FxEchoPlugin)
  1220. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxEchoPlugin)
  1221. };
  1222. // -----------------------------------------------------------------------
  1223. class FxPhaserPlugin : public FxAbstractPlugin
  1224. {
  1225. public:
  1226. FxPhaserPlugin(const HostDescriptor* const host)
  1227. : FxAbstractPlugin(host, 15, 12)
  1228. {
  1229. fEffect = new Phaser(false, efxoutl, efxoutr);
  1230. }
  1231. protected:
  1232. // -------------------------------------------------------------------
  1233. // Plugin parameter calls
  1234. const Parameter* getParameterInfo(const uint32_t index) override
  1235. {
  1236. if (index >= kParamCount)
  1237. return nullptr;
  1238. static Parameter param;
  1239. static ParameterScalePoint scalePoints[2];
  1240. int hints = PARAMETER_IS_ENABLED|PARAMETER_IS_INTEGER;
  1241. param.name = nullptr;
  1242. param.unit = nullptr;
  1243. param.ranges.def = 1.0f;
  1244. param.ranges.min = 0.0f;
  1245. param.ranges.max = 127.0f;
  1246. param.ranges.step = 1.0f;
  1247. param.ranges.stepSmall = 1.0f;
  1248. param.ranges.stepLarge = 20.0f;
  1249. param.scalePointCount = 0;
  1250. param.scalePoints = nullptr;
  1251. switch (index)
  1252. {
  1253. case 0:
  1254. hints |= PARAMETER_IS_AUTOMABLE;
  1255. param.name = "LFO Frequency";
  1256. param.ranges.def = 36.0f;
  1257. break;
  1258. case 1:
  1259. hints |= PARAMETER_IS_AUTOMABLE;
  1260. param.name = "LFO Randomness";
  1261. param.ranges.def = 0.0f;
  1262. break;
  1263. case 2:
  1264. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN|PARAMETER_USES_SCALEPOINTS;
  1265. param.name = "LFO Type";
  1266. param.ranges.def = 0.0f;
  1267. param.ranges.max = 1.0f;
  1268. param.scalePointCount = 2;
  1269. param.scalePoints = scalePoints;
  1270. scalePoints[0].label = "Sine";
  1271. scalePoints[1].label = "Triangle";
  1272. scalePoints[0].value = 0.0f;
  1273. scalePoints[1].value = 1.0f;
  1274. break;
  1275. case 3:
  1276. hints |= PARAMETER_IS_AUTOMABLE;
  1277. param.name = "LFO Stereo";
  1278. param.ranges.def = 64.0f;
  1279. break;
  1280. case 4:
  1281. hints |= PARAMETER_IS_AUTOMABLE;
  1282. param.name = "Depth";
  1283. param.ranges.def = 110.0f;
  1284. break;
  1285. case 5:
  1286. hints |= PARAMETER_IS_AUTOMABLE;
  1287. param.name = "Feedback";
  1288. param.ranges.def = 64.0f;
  1289. break;
  1290. case 6:
  1291. param.name = "Stages";
  1292. param.ranges.def = 1.0f;
  1293. param.ranges.min = 1.0f;
  1294. param.ranges.max = 12.0f;
  1295. break;
  1296. case 7:
  1297. hints |= PARAMETER_IS_AUTOMABLE;
  1298. param.name = "L/R Cross|Offset";
  1299. param.ranges.def = 0.0f;
  1300. break;
  1301. case 8:
  1302. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  1303. param.name = "Subtract Output";
  1304. param.ranges.def = 0.0f;
  1305. param.ranges.max = 1.0f;
  1306. break;
  1307. case 9:
  1308. hints |= PARAMETER_IS_AUTOMABLE;
  1309. param.name = "Phase|Width";
  1310. param.ranges.def = 20.0f;
  1311. break;
  1312. case 10:
  1313. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  1314. param.name = "Hyper";
  1315. param.ranges.def = 0.0f;
  1316. param.ranges.max = 1.0f;
  1317. break;
  1318. case 11:
  1319. hints |= PARAMETER_IS_AUTOMABLE;
  1320. param.name = "Distortion";
  1321. param.ranges.def = 0.0f;
  1322. break;
  1323. case 12:
  1324. hints |= PARAMETER_IS_AUTOMABLE|PARAMETER_IS_BOOLEAN;
  1325. param.name = "Analog";
  1326. param.ranges.def = 0.0f;
  1327. param.ranges.max = 1.0f;
  1328. break;
  1329. }
  1330. param.hints = static_cast<ParameterHints>(hints);
  1331. return &param;
  1332. }
  1333. // -------------------------------------------------------------------
  1334. // Plugin midi-program calls
  1335. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  1336. {
  1337. if (index >= kProgramCount)
  1338. return nullptr;
  1339. static MidiProgram midiProg;
  1340. midiProg.bank = 0;
  1341. midiProg.program = index;
  1342. switch (index)
  1343. {
  1344. case 0:
  1345. midiProg.name = "Phaser 1";
  1346. break;
  1347. case 1:
  1348. midiProg.name = "Phaser 2";
  1349. break;
  1350. case 2:
  1351. midiProg.name = "Phaser 3";
  1352. break;
  1353. case 3:
  1354. midiProg.name = "Phaser 4";
  1355. break;
  1356. case 4:
  1357. midiProg.name = "Phaser 5";
  1358. break;
  1359. case 5:
  1360. midiProg.name = "Phaser 6";
  1361. break;
  1362. case 6:
  1363. midiProg.name = "APhaser 1";
  1364. break;
  1365. case 7:
  1366. midiProg.name = "APhaser 2";
  1367. break;
  1368. case 8:
  1369. midiProg.name = "APhaser 3";
  1370. break;
  1371. case 9:
  1372. midiProg.name = "APhaser 4";
  1373. break;
  1374. case 10:
  1375. midiProg.name = "APhaser 5";
  1376. break;
  1377. case 11:
  1378. midiProg.name = "APhaser 6";
  1379. break;
  1380. default:
  1381. midiProg.name = nullptr;
  1382. break;
  1383. }
  1384. return &midiProg;
  1385. }
  1386. ZynPluginDescriptorClassEND(FxPhaserPlugin)
  1387. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxPhaserPlugin)
  1388. };
  1389. // -----------------------------------------------------------------------
  1390. class FxReverbPlugin : public FxAbstractPlugin
  1391. {
  1392. public:
  1393. FxReverbPlugin(const HostDescriptor* const host)
  1394. : FxAbstractPlugin(host, 13, 13)
  1395. {
  1396. fEffect = new Reverb(false, efxoutl, efxoutr);
  1397. }
  1398. protected:
  1399. // -------------------------------------------------------------------
  1400. // Plugin parameter calls
  1401. const Parameter* getParameterInfo(const uint32_t index) override
  1402. {
  1403. if (index >= kParamCount)
  1404. return nullptr;
  1405. static Parameter param;
  1406. static ParameterScalePoint scalePoints[3];
  1407. int hints = PARAMETER_IS_ENABLED | PARAMETER_IS_INTEGER;
  1408. param.name = nullptr;
  1409. param.unit = nullptr;
  1410. param.ranges.def = 1.0f;
  1411. param.ranges.min = 0.0f;
  1412. param.ranges.max = 127.0f;
  1413. param.ranges.step = 1.0f;
  1414. param.ranges.stepSmall = 1.0f;
  1415. param.ranges.stepLarge = 20.0f;
  1416. param.scalePointCount = 0;
  1417. param.scalePoints = nullptr;
  1418. switch (index)
  1419. {
  1420. case 0:
  1421. hints |= PARAMETER_IS_AUTOMABLE;
  1422. param.name = "Time";
  1423. param.ranges.def = 63.0f;
  1424. break;
  1425. case 1:
  1426. param.name = "Delay";
  1427. param.ranges.def = 24.0f;
  1428. break;
  1429. case 2:
  1430. hints |= PARAMETER_IS_AUTOMABLE;
  1431. param.name = "Feedback";
  1432. param.ranges.def = 0.0f;
  1433. break;
  1434. case 3:
  1435. hints = 0x0;
  1436. param.name = "bw";
  1437. break;
  1438. case 4:
  1439. hints = 0x0;
  1440. param.name = "E/R";
  1441. break;
  1442. case 5:
  1443. param.name = "Low-Pass Filter";
  1444. param.ranges.def = 85.0f;
  1445. break;
  1446. case 6:
  1447. param.name = "High-Pass Filter";
  1448. param.ranges.def = 5.0f;
  1449. break;
  1450. case 7:
  1451. hints |= PARAMETER_IS_AUTOMABLE;
  1452. param.name = "Damp";
  1453. param.ranges.def = 83.0f;
  1454. param.ranges.min = 64.0f;
  1455. break;
  1456. case 8:
  1457. hints |= PARAMETER_USES_SCALEPOINTS;
  1458. param.name = "Type";
  1459. param.ranges.def = 1.0f;
  1460. param.ranges.max = 2.0f;
  1461. param.scalePointCount = 3;
  1462. param.scalePoints = scalePoints;
  1463. scalePoints[0].label = "Random";
  1464. scalePoints[1].label = "Freeverb";
  1465. scalePoints[2].label = "Bandwidth";
  1466. scalePoints[0].value = 0.0f;
  1467. scalePoints[1].value = 1.0f;
  1468. scalePoints[2].value = 2.0f;
  1469. break;
  1470. case 9:
  1471. param.name = "Room size";
  1472. param.ranges.def = 64.0f;
  1473. param.ranges.min = 1.0f;
  1474. break;
  1475. case 10:
  1476. param.name = "Bandwidth";
  1477. param.ranges.def = 20.0f;
  1478. break;
  1479. }
  1480. param.hints = static_cast<ParameterHints>(hints);
  1481. return &param;
  1482. }
  1483. // -------------------------------------------------------------------
  1484. // Plugin midi-program calls
  1485. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  1486. {
  1487. if (index >= kProgramCount)
  1488. return nullptr;
  1489. static MidiProgram midiProg;
  1490. midiProg.bank = 0;
  1491. midiProg.program = index;
  1492. switch (index)
  1493. {
  1494. case 0:
  1495. midiProg.name = "Cathedral1";
  1496. break;
  1497. case 1:
  1498. midiProg.name = "Cathedral2";
  1499. break;
  1500. case 2:
  1501. midiProg.name = "Cathedral3";
  1502. break;
  1503. case 3:
  1504. midiProg.name = "Hall1";
  1505. break;
  1506. case 4:
  1507. midiProg.name = "Hall2";
  1508. break;
  1509. case 5:
  1510. midiProg.name = "Room1";
  1511. break;
  1512. case 6:
  1513. midiProg.name = "Room2";
  1514. break;
  1515. case 7:
  1516. midiProg.name = "Basement";
  1517. break;
  1518. case 8:
  1519. midiProg.name = "Tunnel";
  1520. break;
  1521. case 9:
  1522. midiProg.name = "Echoed1";
  1523. break;
  1524. case 10:
  1525. midiProg.name = "Echoed2";
  1526. break;
  1527. case 11:
  1528. midiProg.name = "VeryLong1";
  1529. break;
  1530. case 12:
  1531. midiProg.name = "VeryLong2";
  1532. break;
  1533. default:
  1534. midiProg.name = nullptr;
  1535. break;
  1536. }
  1537. return &midiProg;
  1538. }
  1539. ZynPluginDescriptorClassEND(FxReverbPlugin)
  1540. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(FxReverbPlugin)
  1541. };
  1542. // -----------------------------------------------------------------------
  1543. class ZynAddSubFxPlugin : public PluginDescriptorClass
  1544. {
  1545. public:
  1546. ZynAddSubFxPlugin(const HostDescriptor* const host)
  1547. : PluginDescriptorClass(host),
  1548. fMaster(new Master()),
  1549. fSampleRate(getSampleRate()),
  1550. fIsActive(false),
  1551. fThread(fMaster, host)
  1552. {
  1553. fThread.start();
  1554. for (int i = 0; i < NUM_MIDI_PARTS; ++i)
  1555. fMaster->partonoff(i, 1);
  1556. sPrograms.init();
  1557. }
  1558. ~ZynAddSubFxPlugin() override
  1559. {
  1560. //ensure that everything has stopped
  1561. pthread_mutex_lock(&fMaster->mutex);
  1562. pthread_mutex_unlock(&fMaster->mutex);
  1563. fThread.stop();
  1564. fThread.wait();
  1565. delete fMaster;
  1566. fMaster = nullptr;
  1567. }
  1568. protected:
  1569. // -------------------------------------------------------------------
  1570. // Plugin midi-program calls
  1571. uint32_t getMidiProgramCount() override
  1572. {
  1573. return sPrograms.count();
  1574. }
  1575. const MidiProgram* getMidiProgramInfo(const uint32_t index) override
  1576. {
  1577. return sPrograms.getInfo(index);
  1578. }
  1579. // -------------------------------------------------------------------
  1580. // Plugin state calls
  1581. void setMidiProgram(const uint8_t channel, const uint32_t bank, const uint32_t program) override
  1582. {
  1583. if (bank >= fMaster->bank.banks.size())
  1584. return;
  1585. if (program >= BANK_SIZE)
  1586. return;
  1587. if (isOffline() || ! fIsActive)
  1588. {
  1589. sPrograms.load(fMaster, channel, bank, program);
  1590. #ifdef WANT_ZYNADDSUBFX_UI
  1591. fThread.uiRepaint();
  1592. #endif
  1593. }
  1594. else
  1595. fThread.loadProgramLater(channel, bank, program);
  1596. }
  1597. void setCustomData(const char* const key, const char* const value) override
  1598. {
  1599. CARLA_ASSERT(key != nullptr);
  1600. CARLA_ASSERT(value != nullptr);
  1601. if (std::strcmp(key, "CarlaAlternateFile1") == 0) // xmz
  1602. fMaster->loadXML(value);
  1603. else if (std::strcmp(key, "CarlaAlternateFile2") == 0) // xiz
  1604. fMaster->part[0]->loadXMLinstrument(value);
  1605. }
  1606. // -------------------------------------------------------------------
  1607. // Plugin process calls
  1608. void activate() override
  1609. {
  1610. fIsActive = true;
  1611. }
  1612. void deactivate() override
  1613. {
  1614. fIsActive = false;
  1615. }
  1616. void process(float**, float** const outBuffer, const uint32_t frames, const uint32_t midiEventCount, const MidiEvent* const midiEvents) override
  1617. {
  1618. if (pthread_mutex_trylock(&fMaster->mutex) != 0)
  1619. {
  1620. carla_zeroFloat(outBuffer[0], frames);
  1621. carla_zeroFloat(outBuffer[1], frames);
  1622. return;
  1623. }
  1624. for (uint32_t i=0; i < midiEventCount; ++i)
  1625. {
  1626. const MidiEvent* const midiEvent(&midiEvents[i]);
  1627. const uint8_t status = MIDI_GET_STATUS_FROM_DATA(midiEvent->data);
  1628. const uint8_t channel = MIDI_GET_CHANNEL_FROM_DATA(midiEvent->data);
  1629. if (MIDI_IS_STATUS_NOTE_OFF(status))
  1630. {
  1631. const uint8_t note = midiEvent->data[1];
  1632. fMaster->noteOff(channel, note);
  1633. }
  1634. else if (MIDI_IS_STATUS_NOTE_ON(status))
  1635. {
  1636. const uint8_t note = midiEvent->data[1];
  1637. const uint8_t velo = midiEvent->data[2];
  1638. fMaster->noteOn(channel, note, velo);
  1639. }
  1640. else if (MIDI_IS_STATUS_POLYPHONIC_AFTERTOUCH(status))
  1641. {
  1642. const uint8_t note = midiEvent->data[1];
  1643. const uint8_t pressure = midiEvent->data[2];
  1644. fMaster->polyphonicAftertouch(channel, note, pressure);
  1645. }
  1646. else if (MIDI_IS_STATUS_CONTROL_CHANGE(status))
  1647. {
  1648. const uint8_t control = midiEvent->data[1];
  1649. const uint8_t value = midiEvent->data[2];
  1650. fMaster->setController(channel, control, value);
  1651. }
  1652. else if (MIDI_IS_STATUS_PITCH_WHEEL_CONTROL(status))
  1653. {
  1654. const uint8_t lsb = midiEvent->data[1];
  1655. const uint8_t msb = midiEvent->data[2];
  1656. const int value = ((msb << 7) | lsb) - 8192;
  1657. fMaster->setController(channel, C_pitchwheel, value);
  1658. }
  1659. }
  1660. fMaster->GetAudioOutSamples(frames, fSampleRate, outBuffer[0], outBuffer[1]);
  1661. pthread_mutex_unlock(&fMaster->mutex);
  1662. }
  1663. #ifdef WANT_ZYNADDSUBFX_UI
  1664. // -------------------------------------------------------------------
  1665. // Plugin UI calls
  1666. void uiShow(const bool show) override
  1667. {
  1668. if (show)
  1669. fThread.uiShow();
  1670. else
  1671. fThread.uiHide();
  1672. }
  1673. #endif
  1674. // -------------------------------------------------------------------
  1675. // Plugin state calls
  1676. char* getState() override
  1677. {
  1678. config.save();
  1679. char* data = nullptr;
  1680. fMaster->getalldata(&data);
  1681. return data;
  1682. }
  1683. void setState(const char* const data) override
  1684. {
  1685. fThread.stopLoadProgramLater();
  1686. fMaster->putalldata((char*)data, 0);
  1687. fMaster->applyparameters(true);
  1688. }
  1689. // -------------------------------------------------------------------
  1690. // Plugin dispatcher
  1691. intptr_t pluginDispatcher(const PluginDispatcherOpcode opcode, const int32_t index, const intptr_t value, void* const ptr, const float) override
  1692. {
  1693. switch (opcode)
  1694. {
  1695. case PLUGIN_OPCODE_NULL:
  1696. break;
  1697. case PLUGIN_OPCODE_BUFFER_SIZE_CHANGED:
  1698. // TODO
  1699. break;
  1700. case PLUGIN_OPCODE_SAMPLE_RATE_CHANGED:
  1701. // TODO
  1702. break;
  1703. case PLUGIN_OPCODE_OFFLINE_CHANGED:
  1704. break;
  1705. case PLUGIN_OPCODE_UI_NAME_CHANGED:
  1706. #ifdef WANT_ZYNADDSUBFX_UI
  1707. // TODO
  1708. #endif
  1709. break;
  1710. }
  1711. return 0;
  1712. // unused
  1713. (void)index;
  1714. (void)value;
  1715. (void)ptr;
  1716. }
  1717. // -------------------------------------------------------------------
  1718. private:
  1719. Master* fMaster;
  1720. unsigned fSampleRate;
  1721. bool fIsActive;
  1722. ZynAddSubFxThread fThread;
  1723. ZynPluginDescriptorClassEND(ZynAddSubFxPlugin)
  1724. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(ZynAddSubFxPlugin)
  1725. };
  1726. // -----------------------------------------------------------------------
  1727. static const PluginDescriptor fxAlienWahDesc = {
  1728. /* category */ PLUGIN_CATEGORY_MODULATOR,
  1729. /* hints */ static_cast<PluginHints>(PLUGIN_IS_RTSAFE|PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1730. /* supports */ static_cast<PluginSupports>(0x0),
  1731. /* audioIns */ 2,
  1732. /* audioOuts */ 2,
  1733. /* midiIns */ 0,
  1734. /* midiOuts */ 0,
  1735. /* paramIns */ 11-2,
  1736. /* paramOuts */ 0,
  1737. /* name */ "ZynAlienWah",
  1738. /* label */ "zynAlienWah",
  1739. /* maker */ "falkTX",
  1740. /* copyright */ "GNU GPL v2+",
  1741. PluginDescriptorFILL(FxAlienWahPlugin)
  1742. };
  1743. static const PluginDescriptor fxChorusDesc = {
  1744. /* category */ PLUGIN_CATEGORY_MODULATOR,
  1745. /* hints */ static_cast<PluginHints>(PLUGIN_IS_RTSAFE|PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1746. /* supports */ static_cast<PluginSupports>(0x0),
  1747. /* audioIns */ 2,
  1748. /* audioOuts */ 2,
  1749. /* midiIns */ 0,
  1750. /* midiOuts */ 0,
  1751. /* paramIns */ 12-2,
  1752. /* paramOuts */ 0,
  1753. /* name */ "ZynChorus",
  1754. /* label */ "zynChorus",
  1755. /* maker */ "falkTX",
  1756. /* copyright */ "GNU GPL v2+",
  1757. PluginDescriptorFILL(FxChorusPlugin)
  1758. };
  1759. static const PluginDescriptor fxDistortionDesc = {
  1760. /* category */ PLUGIN_CATEGORY_MODULATOR,
  1761. /* hints */ static_cast<PluginHints>(PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1762. /* supports */ static_cast<PluginSupports>(0x0),
  1763. /* audioIns */ 2,
  1764. /* audioOuts */ 2,
  1765. /* midiIns */ 0,
  1766. /* midiOuts */ 0,
  1767. /* paramIns */ 11-2,
  1768. /* paramOuts */ 0,
  1769. /* name */ "ZynDistortion",
  1770. /* label */ "zynDistortion",
  1771. /* maker */ "falkTX",
  1772. /* copyright */ "GNU GPL v2+",
  1773. PluginDescriptorFILL(FxDistortionPlugin)
  1774. };
  1775. static const PluginDescriptor fxDynamicFilterDesc = {
  1776. /* category */ PLUGIN_CATEGORY_FILTER,
  1777. /* hints */ static_cast<PluginHints>(PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1778. /* supports */ static_cast<PluginSupports>(0x0),
  1779. /* audioIns */ 2,
  1780. /* audioOuts */ 2,
  1781. /* midiIns */ 0,
  1782. /* midiOuts */ 0,
  1783. /* paramIns */ 10-2,
  1784. /* paramOuts */ 0,
  1785. /* name */ "ZynDynamicFilter",
  1786. /* label */ "zynDynamicFilter",
  1787. /* maker */ "falkTX",
  1788. /* copyright */ "GNU GPL v2+",
  1789. PluginDescriptorFILL(FxDynamicFilterPlugin)
  1790. };
  1791. static const PluginDescriptor fxEchoDesc = {
  1792. /* category */ PLUGIN_CATEGORY_DELAY,
  1793. /* hints */ static_cast<PluginHints>(PLUGIN_IS_RTSAFE|PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1794. /* supports */ static_cast<PluginSupports>(0x0),
  1795. /* audioIns */ 2,
  1796. /* audioOuts */ 2,
  1797. /* midiIns */ 0,
  1798. /* midiOuts */ 0,
  1799. /* paramIns */ 7-2,
  1800. /* paramOuts */ 0,
  1801. /* name */ "ZynEcho",
  1802. /* label */ "zynEcho",
  1803. /* maker */ "falkTX",
  1804. /* copyright */ "GNU GPL v2+",
  1805. PluginDescriptorFILL(FxEchoPlugin)
  1806. };
  1807. static const PluginDescriptor fxPhaserDesc = {
  1808. /* category */ PLUGIN_CATEGORY_MODULATOR,
  1809. /* hints */ static_cast<PluginHints>(PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1810. /* supports */ static_cast<PluginSupports>(0x0),
  1811. /* audioIns */ 2,
  1812. /* audioOuts */ 2,
  1813. /* midiIns */ 0,
  1814. /* midiOuts */ 0,
  1815. /* paramIns */ 15-2,
  1816. /* paramOuts */ 0,
  1817. /* name */ "ZynPhaser",
  1818. /* label */ "zynPhaser",
  1819. /* maker */ "falkTX",
  1820. /* copyright */ "GNU GPL v2+",
  1821. PluginDescriptorFILL(FxPhaserPlugin)
  1822. };
  1823. static const PluginDescriptor fxReverbDesc = {
  1824. /* category */ PLUGIN_CATEGORY_DELAY,
  1825. /* hints */ static_cast<PluginHints>(PLUGIN_USES_PANNING|PLUGIN_USES_STATIC_BUFFERS),
  1826. /* supports */ static_cast<PluginSupports>(0x0),
  1827. /* audioIns */ 2,
  1828. /* audioOuts */ 2,
  1829. /* midiIns */ 0,
  1830. /* midiOuts */ 0,
  1831. /* paramIns */ 13-2,
  1832. /* paramOuts */ 0,
  1833. /* name */ "ZynReverb",
  1834. /* label */ "zynReverb",
  1835. /* maker */ "falkTX",
  1836. /* copyright */ "GNU GPL v2+",
  1837. PluginDescriptorFILL(FxReverbPlugin)
  1838. };
  1839. static const PluginDescriptor zynaddsubfxDesc = {
  1840. /* category */ PLUGIN_CATEGORY_SYNTH,
  1841. #ifdef WANT_ZYNADDSUBFX_UI
  1842. /* hints */ static_cast<PluginHints>(PLUGIN_IS_SYNTH|PLUGIN_HAS_GUI|PLUGIN_USES_STATE),
  1843. #else
  1844. /* hints */ static_cast<PluginHints>(PLUGIN_IS_SYNTH|PLUGIN_USES_STATE),
  1845. #endif
  1846. /* supports */ static_cast<PluginSupports>(PLUGIN_SUPPORTS_CONTROL_CHANGES|PLUGIN_SUPPORTS_NOTE_AFTERTOUCH|PLUGIN_SUPPORTS_PITCHBEND|PLUGIN_SUPPORTS_ALL_SOUND_OFF),
  1847. /* audioIns */ 0,
  1848. /* audioOuts */ 2,
  1849. /* midiIns */ 1,
  1850. /* midiOuts */ 0,
  1851. /* paramIns */ 0,
  1852. /* paramOuts */ 0,
  1853. /* name */ "ZynAddSubFX",
  1854. /* label */ "zynaddsubfx",
  1855. /* maker */ "falkTX",
  1856. /* copyright */ "GNU GPL v2+",
  1857. PluginDescriptorFILL(ZynAddSubFxPlugin)
  1858. };
  1859. // -----------------------------------------------------------------------
  1860. void carla_register_native_plugin_zynaddsubfx()
  1861. {
  1862. carla_register_native_plugin(&fxAlienWahDesc);
  1863. carla_register_native_plugin(&fxChorusDesc);
  1864. carla_register_native_plugin(&fxDistortionDesc);
  1865. carla_register_native_plugin(&fxDynamicFilterDesc);
  1866. carla_register_native_plugin(&fxEchoDesc);
  1867. carla_register_native_plugin(&fxPhaserDesc);
  1868. carla_register_native_plugin(&fxReverbDesc);
  1869. carla_register_native_plugin(&zynaddsubfxDesc);
  1870. }
  1871. // -----------------------------------------------------------------------