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.

478 lines
15KB

  1. /*
  2. * Carla Native Plugins
  3. * Copyright (C) 2012-2015 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 doc/GPL.txt file.
  16. */
  17. #include "CarlaNativeExtUI.hpp"
  18. #include "CarlaJuceUtils.hpp"
  19. #include "juce_audio_basics.h"
  20. #include "zita-rev1/jclient.cc"
  21. #include "zita-rev1/pareq.cc"
  22. #include "zita-rev1/reverb.cc"
  23. using juce::FloatVectorOperations;
  24. using juce::ScopedPointer;
  25. using namespace REV1;
  26. // -----------------------------------------------------------------------
  27. // REV1 Plugin
  28. class REV1Plugin : public NativePluginAndUiClass
  29. {
  30. public:
  31. enum Parameters {
  32. kParameterDELAY,
  33. kParameterXOVER,
  34. kParameterRTLOW,
  35. kParameterRTMID,
  36. kParameterFDAMP,
  37. kParameterEQ1FR,
  38. kParameterEQ1GN,
  39. kParameterEQ2FR,
  40. kParameterEQ2GN,
  41. kParameterOPMIXorRGXYZ,
  42. kParameterNROTARY
  43. };
  44. REV1Plugin(const NativeHostDescriptor* const host, const bool isAmbisonic)
  45. : NativePluginAndUiClass(host, "rev1-ui"),
  46. kIsAmbisonic(isAmbisonic),
  47. kNumInputs(2),
  48. kNumOutputs(isAmbisonic ? 4 : 2),
  49. fJackClient(),
  50. jclient(nullptr),
  51. leakDetector_REV1Plugin()
  52. {
  53. CARLA_SAFE_ASSERT(host != nullptr);
  54. carla_zeroStruct(fJackClient);
  55. fJackClient.clientName = "rev1";
  56. fJackClient.bufferSize = getBufferSize();
  57. fJackClient.sampleRate = getSampleRate();
  58. jclient = new Jclient(&fJackClient, isAmbisonic);
  59. // set initial values
  60. fParameters[kParameterDELAY] = 0.04f;
  61. fParameters[kParameterXOVER] = 200.0f;
  62. fParameters[kParameterRTLOW] = 3.0f;
  63. fParameters[kParameterRTMID] = 2.0f;
  64. fParameters[kParameterFDAMP] = 6.0e3;
  65. fParameters[kParameterEQ1FR] = 160.0f;
  66. fParameters[kParameterEQ1GN] = 0.0f;
  67. fParameters[kParameterEQ2FR] = 2.5e3;
  68. fParameters[kParameterEQ2GN] = 0.0f;
  69. if (isAmbisonic)
  70. fParameters[kParameterOPMIXorRGXYZ] = 0.0f;
  71. else
  72. fParameters[kParameterOPMIXorRGXYZ] = 0.5f;
  73. Reverb* const reverb(jclient->reverb());
  74. reverb->set_delay(fParameters[kParameterDELAY]);
  75. reverb->set_xover(fParameters[kParameterXOVER]);
  76. reverb->set_rtlow(fParameters[kParameterRTLOW]);
  77. reverb->set_rtmid(fParameters[kParameterRTMID]);
  78. reverb->set_fdamp(fParameters[kParameterFDAMP]);
  79. if (isAmbisonic)
  80. {
  81. reverb->set_opmix(0.5);
  82. reverb->set_rgxyz(fParameters[kParameterOPMIXorRGXYZ]);
  83. }
  84. else
  85. {
  86. reverb->set_opmix(fParameters[kParameterOPMIXorRGXYZ]);
  87. reverb->set_rgxyz(0.0);
  88. }
  89. reverb->set_eq1(fParameters[kParameterEQ1FR], fParameters[kParameterEQ1GN]);
  90. reverb->set_eq2(fParameters[kParameterEQ2FR], fParameters[kParameterEQ2GN]);
  91. }
  92. // -------------------------------------------------------------------
  93. // Plugin parameter calls
  94. uint32_t getParameterCount() const override
  95. {
  96. return kParameterNROTARY;
  97. }
  98. const NativeParameter* getParameterInfo(const uint32_t index) const override
  99. {
  100. CARLA_SAFE_ASSERT_RETURN(index < kParameterNROTARY, nullptr);
  101. static NativeParameter param;
  102. int hints = NATIVE_PARAMETER_IS_ENABLED|NATIVE_PARAMETER_IS_AUTOMABLE;
  103. // reset
  104. param.name = nullptr;
  105. param.unit = nullptr;
  106. param.ranges.def = 0.0f;
  107. param.ranges.min = 0.0f;
  108. param.ranges.max = 1.0f;
  109. param.ranges.step = 1.0f;
  110. param.ranges.stepSmall = 1.0f;
  111. param.ranges.stepLarge = 1.0f;
  112. param.scalePointCount = 0;
  113. param.scalePoints = nullptr;
  114. switch (index)
  115. {
  116. case kParameterDELAY:
  117. param.name = "Delay";
  118. param.ranges.def = 0.04f;
  119. param.ranges.min = 0.02f;
  120. param.ranges.max = 0.100f;
  121. break;
  122. case kParameterXOVER:
  123. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  124. param.name = "Crossover";
  125. param.ranges.def = 200.0f;
  126. param.ranges.min = 50.0f;
  127. param.ranges.max = 1000.0f;
  128. break;
  129. case kParameterRTLOW:
  130. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  131. param.name = "RT60 Low";
  132. param.ranges.def = 3.0f;
  133. param.ranges.min = 1.0f;
  134. param.ranges.max = 8.0f;
  135. break;
  136. case kParameterRTMID:
  137. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  138. param.name = "RT60 Mid";
  139. param.ranges.def = 2.0f;
  140. param.ranges.min = 1.0f;
  141. param.ranges.max = 8.0f;
  142. break;
  143. case kParameterFDAMP:
  144. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  145. param.name = "HF Damping";
  146. param.ranges.def = 6.0e3;
  147. param.ranges.min = 1.5e3;
  148. param.ranges.max = 24.0e3;
  149. break;
  150. case kParameterEQ1FR:
  151. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  152. param.name = "Eq1 Frequency";
  153. param.ranges.def = 160.0f;
  154. param.ranges.min = 40.0f;
  155. param.ranges.max = 2.5e3;
  156. break;
  157. case kParameterEQ1GN:
  158. param.name = "Eq1 Gain";
  159. param.ranges.def = 0.0f;
  160. param.ranges.min = -15.0;
  161. param.ranges.max = 15.0f;
  162. break;
  163. case kParameterEQ2FR:
  164. hints |= NATIVE_PARAMETER_IS_LOGARITHMIC;
  165. param.name = "Eq2 Frequency";
  166. param.ranges.def = 2.5e3;
  167. param.ranges.min = 160.0;
  168. param.ranges.max = 10e3;
  169. break;
  170. case kParameterEQ2GN:
  171. param.name = "Eq2 Gain";
  172. param.ranges.def = 0.0f;
  173. param.ranges.min = -15.0;
  174. param.ranges.max = 15.0f;
  175. break;
  176. case kParameterOPMIXorRGXYZ:
  177. if (kIsAmbisonic)
  178. {
  179. param.name = "XYZ gain";
  180. param.ranges.def = 0.0f;
  181. param.ranges.min = -9.0f;
  182. param.ranges.max = 9.0f;
  183. }
  184. else
  185. {
  186. param.name = "Dry/wet mix";
  187. param.ranges.def = 0.5f;
  188. param.ranges.min = 0.0f;
  189. param.ranges.max = 1.0f;
  190. }
  191. break;
  192. }
  193. param.hints = static_cast<NativeParameterHints>(hints);
  194. return &param;
  195. }
  196. float getParameterValue(const uint32_t index) const override
  197. {
  198. CARLA_SAFE_ASSERT_RETURN(index < kParameterNROTARY, 0.0f);
  199. return fParameters[index];
  200. }
  201. // -------------------------------------------------------------------
  202. // Plugin state calls
  203. void setParameterValue(const uint32_t index, const float value) override
  204. {
  205. CARLA_SAFE_ASSERT_RETURN(index < kParameterNROTARY,);
  206. Reverb* const reverb(jclient->reverb());
  207. fParameters[index] = value;
  208. switch (index)
  209. {
  210. case kParameterDELAY:
  211. reverb->set_delay(value);
  212. break;
  213. case kParameterXOVER:
  214. reverb->set_xover(value);
  215. break;
  216. case kParameterRTLOW:
  217. reverb->set_rtlow(value);
  218. break;
  219. case kParameterRTMID:
  220. reverb->set_rtmid(value);
  221. break;
  222. case kParameterFDAMP:
  223. reverb->set_fdamp(value);
  224. break;
  225. case kParameterEQ1FR:
  226. reverb->set_eq1(value, fParameters[kParameterEQ1GN]);
  227. break;
  228. case kParameterEQ1GN:
  229. reverb->set_eq1(fParameters[kParameterEQ1FR], value);
  230. break;
  231. case kParameterEQ2FR:
  232. reverb->set_eq2(value, fParameters[kParameterEQ2GN]);
  233. break;
  234. case kParameterEQ2GN:
  235. reverb->set_eq2(fParameters[kParameterEQ2FR], value);
  236. break;
  237. case kParameterOPMIXorRGXYZ:
  238. if (kIsAmbisonic)
  239. reverb->set_rgxyz(value);
  240. else
  241. reverb->set_opmix(value);
  242. break;
  243. }
  244. }
  245. // -------------------------------------------------------------------
  246. // Plugin process calls
  247. void process(float** const inBuffer, float** const outBuffer, const uint32_t frames, const NativeMidiEvent* const, const uint32_t) override
  248. {
  249. if (! fJackClient.active)
  250. {
  251. const int iframes(static_cast<int>(frames));
  252. for (uint32_t i=0; i<kNumInputs; ++i)
  253. FloatVectorOperations::clear(outBuffer[i], iframes);
  254. return;
  255. }
  256. for (uint32_t i=0; i<kNumInputs; ++i)
  257. fJackClient.portsAudioIn[i].buffer.audio = inBuffer[i];
  258. for (uint32_t i=0; i<kNumOutputs; ++i)
  259. fJackClient.portsAudioOut[i].buffer.audio = outBuffer[i];
  260. fJackClient.processCallback(frames, fJackClient.processPtr);
  261. }
  262. // -------------------------------------------------------------------
  263. // Plugin dispatcher calls
  264. void bufferSizeChanged(const uint32_t bufferSize) override
  265. {
  266. fJackClient.bufferSize = bufferSize;
  267. }
  268. void sampleRateChanged(const double sampleRate) override
  269. {
  270. fJackClient.sampleRate = sampleRate;
  271. }
  272. // -------------------------------------------------------------------
  273. // Plugin UI calls
  274. void uiShow(const bool show) override
  275. {
  276. if (show)
  277. {
  278. if (isPipeRunning())
  279. {
  280. const CarlaMutexLocker cml(getPipeLock());
  281. writeMessage("focus\n", 6);
  282. flushMessages();
  283. return;
  284. }
  285. carla_stdout("Trying to start UI using \"%s\"", getExtUiPath());
  286. CarlaExternalUI::setData(getExtUiPath(), kIsAmbisonic ? "true" : "false", getUiName());
  287. if (! CarlaExternalUI::startPipeServer(true))
  288. {
  289. uiClosed();
  290. hostUiUnavailable();
  291. }
  292. }
  293. else
  294. {
  295. CarlaExternalUI::stopPipeServer(2000);
  296. }
  297. }
  298. // -------------------------------------------------------------------
  299. private:
  300. const bool kIsAmbisonic;
  301. const uint32_t kNumInputs;
  302. const uint32_t kNumOutputs;
  303. // Fake jack client
  304. jack_client_t fJackClient;
  305. // Zita stuff (core)
  306. ScopedPointer<Jclient> jclient;
  307. float fParameters[kParameterNROTARY];
  308. public:
  309. static NativePluginHandle _instantiateAmbisonic(const NativeHostDescriptor* host)
  310. {
  311. return (host != nullptr) ? new REV1Plugin(host, true) : nullptr;
  312. }
  313. static NativePluginHandle _instantiateStereo(const NativeHostDescriptor* host)
  314. {
  315. return (host != nullptr) ? new REV1Plugin(host, false) : nullptr;
  316. }
  317. static void _cleanup(NativePluginHandle handle)
  318. {
  319. delete (REV1Plugin*)handle;
  320. }
  321. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(REV1Plugin)
  322. };
  323. // -----------------------------------------------------------------------
  324. static const NativePluginDescriptor rev1AmbisonicDesc = {
  325. /* category */ NATIVE_PLUGIN_CATEGORY_DELAY,
  326. /* hints */ static_cast<NativePluginHints>(NATIVE_PLUGIN_IS_RTSAFE
  327. |NATIVE_PLUGIN_HAS_UI
  328. |NATIVE_PLUGIN_NEEDS_FIXED_BUFFERS
  329. |NATIVE_PLUGIN_NEEDS_UI_MAIN_THREAD),
  330. /* supports */ static_cast<NativePluginSupports>(0x0),
  331. /* audioIns */ 2,
  332. /* audioOuts */ 4,
  333. /* midiIns */ 0,
  334. /* midiOuts */ 0,
  335. /* paramIns */ REV1Plugin::kParameterNROTARY,
  336. /* paramOuts */ 0,
  337. /* name */ "REV1 (Ambisonic)",
  338. /* label */ "rev1-ambisonic",
  339. /* maker */ "falkTX, Fons Adriaensen",
  340. /* copyright */ "GPL v2+",
  341. REV1Plugin::_instantiateAmbisonic,
  342. REV1Plugin::_cleanup,
  343. REV1Plugin::_get_parameter_count,
  344. REV1Plugin::_get_parameter_info,
  345. REV1Plugin::_get_parameter_value,
  346. REV1Plugin::_get_parameter_text,
  347. REV1Plugin::_get_midi_program_count,
  348. REV1Plugin::_get_midi_program_info,
  349. REV1Plugin::_set_parameter_value,
  350. REV1Plugin::_set_midi_program,
  351. REV1Plugin::_set_custom_data,
  352. REV1Plugin::_ui_show,
  353. REV1Plugin::_ui_idle,
  354. REV1Plugin::_ui_set_parameter_value,
  355. REV1Plugin::_ui_set_midi_program,
  356. REV1Plugin::_ui_set_custom_data,
  357. REV1Plugin::_activate,
  358. REV1Plugin::_deactivate,
  359. REV1Plugin::_process,
  360. REV1Plugin::_get_state,
  361. REV1Plugin::_set_state,
  362. REV1Plugin::_dispatcher
  363. };
  364. static const NativePluginDescriptor rev1StereoDesc = {
  365. /* category */ NATIVE_PLUGIN_CATEGORY_DELAY,
  366. /* hints */ static_cast<NativePluginHints>(NATIVE_PLUGIN_IS_RTSAFE
  367. |NATIVE_PLUGIN_HAS_UI
  368. |NATIVE_PLUGIN_NEEDS_FIXED_BUFFERS
  369. |NATIVE_PLUGIN_NEEDS_UI_MAIN_THREAD),
  370. /* supports */ static_cast<NativePluginSupports>(0x0),
  371. /* audioIns */ 2,
  372. /* audioOuts */ 2,
  373. /* midiIns */ 0,
  374. /* midiOuts */ 0,
  375. /* paramIns */ REV1Plugin::kParameterNROTARY,
  376. /* paramOuts */ 0,
  377. /* name */ "REV1 (Stereo)",
  378. /* label */ "rev1-stereo",
  379. /* maker */ "falkTX, Fons Adriaensen",
  380. /* copyright */ "GPL v2+",
  381. REV1Plugin::_instantiateStereo,
  382. REV1Plugin::_cleanup,
  383. REV1Plugin::_get_parameter_count,
  384. REV1Plugin::_get_parameter_info,
  385. REV1Plugin::_get_parameter_value,
  386. REV1Plugin::_get_parameter_text,
  387. REV1Plugin::_get_midi_program_count,
  388. REV1Plugin::_get_midi_program_info,
  389. REV1Plugin::_set_parameter_value,
  390. REV1Plugin::_set_midi_program,
  391. REV1Plugin::_set_custom_data,
  392. REV1Plugin::_ui_show,
  393. REV1Plugin::_ui_idle,
  394. REV1Plugin::_ui_set_parameter_value,
  395. REV1Plugin::_ui_set_midi_program,
  396. REV1Plugin::_ui_set_custom_data,
  397. REV1Plugin::_activate,
  398. REV1Plugin::_deactivate,
  399. REV1Plugin::_process,
  400. REV1Plugin::_get_state,
  401. REV1Plugin::_set_state,
  402. REV1Plugin::_dispatcher
  403. };
  404. // -----------------------------------------------------------------------
  405. CARLA_EXPORT
  406. void carla_register_native_plugin_zita_rev1();
  407. CARLA_EXPORT
  408. void carla_register_native_plugin_zita_rev1()
  409. {
  410. carla_register_native_plugin(&rev1AmbisonicDesc);
  411. carla_register_native_plugin(&rev1StereoDesc);
  412. }
  413. // -----------------------------------------------------------------------