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.

2088 lines
58KB

  1. /*
  2. * Carla Plugin
  3. * Copyright (C) 2011-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. #include "CarlaPluginInternal.hpp"
  18. #include "CarlaLibUtils.hpp"
  19. #include "CarlaMIDI.h"
  20. #include <QtCore/QFile>
  21. #include <QtCore/QTextStream>
  22. //#include <QtGui/QtEvents>
  23. CARLA_BACKEND_START_NAMESPACE
  24. // -------------------------------------------------------------------
  25. // fallback data
  26. static const ParameterData kParameterDataNull;
  27. static const ParameterRanges kParameterRangesNull;
  28. static const MidiProgramData kMidiProgramDataNull;
  29. static const CustomData kCustomDataNull;
  30. // -------------------------------------------------------------------
  31. // Helpers
  32. CarlaEngine* CarlaPluginGetEngine(CarlaPlugin* const plugin)
  33. {
  34. return CarlaPluginProtectedData::getEngine(plugin);
  35. }
  36. CarlaEngineAudioPort* CarlaPluginGetAudioInPort(CarlaPlugin* const plugin, const uint32_t index)
  37. {
  38. return CarlaPluginProtectedData::getAudioInPort(plugin, index);
  39. }
  40. CarlaEngineAudioPort* CarlaPluginGetAudioOutPort(CarlaPlugin* const plugin, const uint32_t index)
  41. {
  42. return CarlaPluginProtectedData::getAudioOutPort(plugin, index);
  43. }
  44. // -------------------------------------------------------------------
  45. // Constructor and destructor
  46. CarlaPlugin::CarlaPlugin(CarlaEngine* const engine, const unsigned int id)
  47. : fId(id),
  48. fHints(0x0),
  49. fOptions(0x0),
  50. fEnabled(false),
  51. kData(new CarlaPluginProtectedData(engine, this))
  52. {
  53. CARLA_ASSERT(kData != nullptr);
  54. CARLA_ASSERT(engine != nullptr);
  55. CARLA_ASSERT(id < engine->maxPluginNumber());
  56. carla_debug("CarlaPlugin::CarlaPlugin(%p, %i)", engine, id);
  57. switch (engine->getProccessMode())
  58. {
  59. case PROCESS_MODE_SINGLE_CLIENT:
  60. case PROCESS_MODE_MULTIPLE_CLIENTS:
  61. CARLA_ASSERT(id < MAX_DEFAULT_PLUGINS);
  62. break;
  63. case PROCESS_MODE_CONTINUOUS_RACK:
  64. CARLA_ASSERT(id < MAX_RACK_PLUGINS);
  65. break;
  66. case PROCESS_MODE_PATCHBAY:
  67. CARLA_ASSERT(id < MAX_PATCHBAY_PLUGINS);
  68. break;
  69. case PROCESS_MODE_BRIDGE:
  70. CARLA_ASSERT(id == 0);
  71. break;
  72. }
  73. if (engine->getOptions().forceStereo)
  74. fOptions |= PLUGIN_OPTION_FORCE_STEREO;
  75. }
  76. CarlaPlugin::~CarlaPlugin()
  77. {
  78. carla_debug("CarlaPlugin::~CarlaPlugin()");
  79. // Remove client and ports
  80. if (kData->client != nullptr)
  81. {
  82. if (kData->client->isActive())
  83. kData->client->deactivate();
  84. // can't call virtual functions in destructor
  85. CarlaPlugin::deleteBuffers();
  86. delete kData->client;
  87. }
  88. if (kData->latencyBuffers != nullptr)
  89. {
  90. for (uint32_t i=0; i < kData->audioIn.count; i++)
  91. delete[] kData->latencyBuffers[i];
  92. delete[] kData->latencyBuffers;
  93. }
  94. kData->prog.clear();
  95. kData->midiprog.clear();
  96. kData->custom.clear();
  97. // MUST have been unlocked before
  98. kData->masterMutex.unlock();
  99. kData->singleMutex.unlock();
  100. libClose();
  101. delete kData;
  102. }
  103. // -------------------------------------------------------------------
  104. // Information (base)
  105. uint32_t CarlaPlugin::latency() const
  106. {
  107. return kData->latency;
  108. }
  109. // -------------------------------------------------------------------
  110. // Information (count)
  111. uint32_t CarlaPlugin::audioInCount() const
  112. {
  113. return kData->audioIn.count;
  114. }
  115. uint32_t CarlaPlugin::audioOutCount() const
  116. {
  117. return kData->audioOut.count;
  118. }
  119. uint32_t CarlaPlugin::midiInCount() const
  120. {
  121. return (kData->extraHints & PLUGIN_HINT_HAS_MIDI_IN) ? 1 : 0;
  122. }
  123. uint32_t CarlaPlugin::midiOutCount() const
  124. {
  125. return (kData->extraHints & PLUGIN_HINT_HAS_MIDI_OUT) ? 1 : 0;
  126. }
  127. uint32_t CarlaPlugin::parameterCount() const
  128. {
  129. return kData->param.count;
  130. }
  131. uint32_t CarlaPlugin::parameterScalePointCount(const uint32_t parameterId) const
  132. {
  133. CARLA_ASSERT(parameterId < kData->param.count);
  134. return 0;
  135. // unused
  136. (void)parameterId;
  137. }
  138. uint32_t CarlaPlugin::programCount() const
  139. {
  140. return kData->prog.count;
  141. }
  142. uint32_t CarlaPlugin::midiProgramCount() const
  143. {
  144. return kData->midiprog.count;
  145. }
  146. uint32_t CarlaPlugin::customDataCount() const
  147. {
  148. return kData->custom.count();
  149. }
  150. // -------------------------------------------------------------------
  151. // Information (current data)
  152. int32_t CarlaPlugin::currentProgram() const
  153. {
  154. return kData->prog.current;
  155. }
  156. int32_t CarlaPlugin::currentMidiProgram() const
  157. {
  158. return kData->midiprog.current;
  159. }
  160. const ParameterData& CarlaPlugin::parameterData(const uint32_t parameterId) const
  161. {
  162. CARLA_ASSERT(parameterId < kData->param.count);
  163. return (parameterId < kData->param.count) ? kData->param.data[parameterId] : kParameterDataNull;
  164. }
  165. const ParameterRanges& CarlaPlugin::parameterRanges(const uint32_t parameterId) const
  166. {
  167. CARLA_ASSERT(parameterId < kData->param.count);
  168. return (parameterId < kData->param.count) ? kData->param.ranges[parameterId] : kParameterRangesNull;
  169. }
  170. bool CarlaPlugin::parameterIsOutput(const uint32_t parameterId) const
  171. {
  172. CARLA_ASSERT(parameterId < kData->param.count);
  173. return (parameterId < kData->param.count) ? (kData->param.data[parameterId].type == PARAMETER_OUTPUT) : false;
  174. }
  175. const MidiProgramData& CarlaPlugin::midiProgramData(const uint32_t index) const
  176. {
  177. CARLA_ASSERT(index < kData->midiprog.count);
  178. return (index < kData->midiprog.count) ? kData->midiprog.data[index] : kMidiProgramDataNull;
  179. }
  180. const CustomData& CarlaPlugin::customData(const size_t index) const
  181. {
  182. CARLA_ASSERT(index < kData->custom.count());
  183. return (index < kData->custom.count()) ? kData->custom.getAt(index) : kCustomDataNull;
  184. }
  185. int32_t CarlaPlugin::chunkData(void** const dataPtr)
  186. {
  187. CARLA_ASSERT(dataPtr != nullptr);
  188. return 0;
  189. // unused
  190. (void)dataPtr;
  191. }
  192. // -------------------------------------------------------------------
  193. // Information (per-plugin data)
  194. unsigned int CarlaPlugin::availableOptions()
  195. {
  196. return 0x0;
  197. }
  198. float CarlaPlugin::getParameterValue(const uint32_t parameterId)
  199. {
  200. CARLA_ASSERT(parameterId < parameterCount());
  201. return 0.0f;
  202. // unused
  203. (void)parameterId;
  204. }
  205. float CarlaPlugin::getParameterScalePointValue(const uint32_t parameterId, const uint32_t scalePointId)
  206. {
  207. CARLA_ASSERT(parameterId < parameterCount());
  208. CARLA_ASSERT(scalePointId < parameterScalePointCount(parameterId));
  209. return 0.0f;
  210. // unused
  211. (void)parameterId;
  212. (void)scalePointId;
  213. }
  214. void CarlaPlugin::getLabel(char* const strBuf)
  215. {
  216. *strBuf = 0;
  217. }
  218. void CarlaPlugin::getMaker(char* const strBuf)
  219. {
  220. *strBuf = 0;
  221. }
  222. void CarlaPlugin::getCopyright(char* const strBuf)
  223. {
  224. *strBuf = 0;
  225. }
  226. void CarlaPlugin::getRealName(char* const strBuf)
  227. {
  228. *strBuf = 0;
  229. }
  230. void CarlaPlugin::getParameterName(const uint32_t parameterId, char* const strBuf)
  231. {
  232. CARLA_ASSERT(parameterId < parameterCount());
  233. *strBuf = 0;
  234. return;
  235. // unused
  236. (void)parameterId;
  237. }
  238. void CarlaPlugin::getParameterSymbol(const uint32_t parameterId, char* const strBuf)
  239. {
  240. CARLA_ASSERT(parameterId < parameterCount());
  241. *strBuf = 0;
  242. return;
  243. // unused
  244. (void)parameterId;
  245. }
  246. void CarlaPlugin::getParameterText(const uint32_t parameterId, char* const strBuf)
  247. {
  248. CARLA_ASSERT(parameterId < parameterCount());
  249. *strBuf = 0;
  250. return;
  251. // unused
  252. (void)parameterId;
  253. }
  254. void CarlaPlugin::getParameterUnit(const uint32_t parameterId, char* const strBuf)
  255. {
  256. CARLA_ASSERT(parameterId < parameterCount());
  257. *strBuf = 0;
  258. return;
  259. // unused
  260. (void)parameterId;
  261. }
  262. void CarlaPlugin::getParameterScalePointLabel(const uint32_t parameterId, const uint32_t scalePointId, char* const strBuf)
  263. {
  264. CARLA_ASSERT(parameterId < parameterCount());
  265. CARLA_ASSERT(scalePointId < parameterScalePointCount(parameterId));
  266. *strBuf = 0;
  267. return;
  268. // unused
  269. (void)parameterId;
  270. (void)scalePointId;
  271. }
  272. void CarlaPlugin::getProgramName(const uint32_t index, char* const strBuf)
  273. {
  274. CARLA_ASSERT(index < kData->prog.count);
  275. CARLA_ASSERT(kData->prog.names[index] != nullptr);
  276. if (index < kData->prog.count && kData->prog.names[index])
  277. std::strncpy(strBuf, kData->prog.names[index], STR_MAX);
  278. else
  279. *strBuf = 0;
  280. }
  281. void CarlaPlugin::getMidiProgramName(const uint32_t index, char* const strBuf)
  282. {
  283. CARLA_ASSERT(index < kData->midiprog.count);
  284. CARLA_ASSERT(kData->midiprog.data[index].name != nullptr);
  285. if (index < kData->midiprog.count && kData->midiprog.data[index].name)
  286. std::strncpy(strBuf, kData->midiprog.data[index].name, STR_MAX);
  287. else
  288. *strBuf = 0;
  289. }
  290. void CarlaPlugin::getParameterCountInfo(uint32_t* const ins, uint32_t* const outs, uint32_t* const total)
  291. {
  292. CARLA_ASSERT(ins != nullptr);
  293. CARLA_ASSERT(outs != nullptr);
  294. CARLA_ASSERT(total != nullptr);
  295. if (ins == nullptr || outs == nullptr || total == nullptr)
  296. return;
  297. *ins = 0;
  298. *outs = 0;
  299. *total = kData->param.count;
  300. for (uint32_t i=0; i < kData->param.count; i++)
  301. {
  302. if (kData->param.data[i].type == PARAMETER_INPUT)
  303. *ins += 1;
  304. else if (kData->param.data[i].type == PARAMETER_OUTPUT)
  305. *outs += 1;
  306. }
  307. }
  308. // -------------------------------------------------------------------
  309. // Set data (state)
  310. void CarlaPlugin::prepareForSave()
  311. {
  312. }
  313. const SaveState& CarlaPlugin::getSaveState()
  314. {
  315. static SaveState saveState;
  316. saveState.reset();
  317. prepareForSave();
  318. char strBuf[STR_MAX];
  319. // ----------------------------
  320. // Basic info
  321. switch (type())
  322. {
  323. case PLUGIN_INTERNAL:
  324. saveState.type = carla_strdup("Internal");
  325. break;
  326. case PLUGIN_LADSPA:
  327. saveState.type = carla_strdup("LADSPA");
  328. break;
  329. case PLUGIN_DSSI:
  330. saveState.type = carla_strdup("DSSI");
  331. break;
  332. case PLUGIN_LV2:
  333. saveState.type = carla_strdup("LV2");
  334. break;
  335. case PLUGIN_VST:
  336. saveState.type = carla_strdup("VST");
  337. break;
  338. case PLUGIN_GIG:
  339. saveState.type = carla_strdup("GIG");
  340. break;
  341. case PLUGIN_SF2:
  342. saveState.type = carla_strdup("SF2");
  343. break;
  344. case PLUGIN_SFZ:
  345. saveState.type = carla_strdup("SFZ");
  346. break;
  347. default:
  348. saveState.type = carla_strdup("Unknown");
  349. break;
  350. }
  351. getLabel(strBuf);
  352. saveState.name = carla_strdup(fName);
  353. saveState.label = carla_strdup(strBuf);
  354. saveState.binary = carla_strdup(fFilename);
  355. saveState.uniqueID = uniqueId();
  356. // ----------------------------
  357. // Internals
  358. saveState.active = kData->active;
  359. saveState.dryWet = kData->postProc.dryWet;
  360. saveState.volume = kData->postProc.volume;
  361. saveState.balanceLeft = kData->postProc.balanceLeft;
  362. saveState.balanceRight = kData->postProc.balanceRight;
  363. saveState.panning = kData->postProc.panning;
  364. saveState.ctrlChannel = kData->ctrlChannel;
  365. // ----------------------------
  366. // Current Program
  367. if (kData->prog.current >= 0)
  368. {
  369. saveState.currentProgramIndex = kData->prog.current;
  370. saveState.currentProgramName = carla_strdup(kData->prog.names[kData->prog.current]);
  371. }
  372. // ----------------------------
  373. // Current MIDI Program
  374. if (kData->midiprog.current >= 0)
  375. {
  376. const MidiProgramData& mpData = kData->midiprog.getCurrent();
  377. saveState.currentMidiBank = mpData.bank;
  378. saveState.currentMidiProgram = mpData.program;
  379. }
  380. // ----------------------------
  381. // Parameters
  382. float sampleRate = kData->engine->getSampleRate();
  383. for (uint32_t i=0, count=kData->param.count; i < count; i++)
  384. {
  385. const ParameterData& paramData = kData->param.data[i];
  386. if (paramData.type != PARAMETER_INPUT)
  387. continue;
  388. StateParameter* stateParameter(new StateParameter);
  389. stateParameter->index = paramData.index;
  390. stateParameter->midiCC = paramData.midiCC;
  391. stateParameter->midiChannel = paramData.midiChannel + 1;
  392. getParameterName(i, strBuf);
  393. stateParameter->name = carla_strdup(strBuf);
  394. getParameterSymbol(i, strBuf);
  395. stateParameter->symbol = carla_strdup(strBuf);;
  396. stateParameter->value = getParameterValue(i);
  397. if (paramData.hints & PARAMETER_USES_SAMPLERATE)
  398. stateParameter->value /= sampleRate;
  399. saveState.parameters.push_back(stateParameter);
  400. }
  401. // ----------------------------
  402. // Custom Data
  403. for (auto it = kData->custom.begin(); it.valid(); it.next())
  404. {
  405. const CustomData& cData(*it);
  406. if (cData.type == nullptr)
  407. continue;
  408. StateCustomData* stateCustomData(new StateCustomData);
  409. stateCustomData->type = carla_strdup(cData.type);
  410. stateCustomData->key = carla_strdup(cData.key);
  411. stateCustomData->value = carla_strdup(cData.value);
  412. saveState.customData.push_back(stateCustomData);
  413. }
  414. // ----------------------------
  415. // Chunk
  416. if (fOptions & PLUGIN_OPTION_USE_CHUNKS)
  417. {
  418. void* data = nullptr;
  419. const int32_t dataSize = chunkData(&data);
  420. if (data != nullptr && dataSize >= 4)
  421. {
  422. CarlaString chunkStr;
  423. chunkStr.importBinaryAsBase64((const uint8_t*)data, static_cast<size_t>(dataSize));
  424. saveState.chunk = carla_strdup(chunkStr);
  425. }
  426. }
  427. return saveState;
  428. }
  429. void CarlaPlugin::loadSaveState(const SaveState& saveState)
  430. {
  431. char strBuf[STR_MAX];
  432. // ---------------------------------------------------------------------
  433. // Part 1 - set custom data (except binary/chunks)
  434. for (auto it = saveState.customData.begin(); it != saveState.customData.end(); ++it)
  435. {
  436. const StateCustomData* const stateCustomData(*it);
  437. if (std::strcmp(stateCustomData->type, CUSTOM_DATA_CHUNK) != 0)
  438. setCustomData(stateCustomData->type, stateCustomData->key, stateCustomData->value, true);
  439. }
  440. // ---------------------------------------------------------------------
  441. // Part 2 - set program
  442. int32_t programId = -1;
  443. if (saveState.currentProgramName != nullptr)
  444. {
  445. getProgramName(saveState.currentProgramIndex, strBuf);
  446. // Program name matches
  447. if (std::strcmp(saveState.currentProgramName, strBuf) == 0)
  448. {
  449. programId = saveState.currentProgramIndex;
  450. }
  451. // index < count
  452. else if (saveState.currentProgramIndex < static_cast<int32_t>(kData->prog.count))
  453. {
  454. programId = saveState.currentProgramIndex;
  455. }
  456. // index not valid, try to find by name
  457. else
  458. {
  459. for (uint32_t i=0; i < kData->prog.count; i++)
  460. {
  461. getProgramName(i, strBuf);
  462. if (std::strcmp(saveState.currentProgramName, strBuf) == 0)
  463. {
  464. programId = i;
  465. break;
  466. }
  467. }
  468. }
  469. }
  470. // set program now, if valid
  471. if (programId >= 0)
  472. setProgram(programId, true, true, true);
  473. // ---------------------------------------------------------------------
  474. // Part 3 - set midi program
  475. if (saveState.currentMidiBank >= 0 && saveState.currentMidiProgram)
  476. setMidiProgramById(saveState.currentMidiBank, saveState.currentMidiProgram, true, true, true);
  477. // ---------------------------------------------------------------------
  478. // Part 4a - get plugin parameter symbols
  479. struct ParamSymbol {
  480. uint32_t index;
  481. const char* symbol;
  482. ParamSymbol()
  483. : index(0),
  484. symbol(nullptr) {}
  485. ParamSymbol(uint32_t index_, const char* symbol_)
  486. : index(index_),
  487. symbol(carla_strdup(symbol_)) {}
  488. void free()
  489. {
  490. if (symbol != nullptr)
  491. {
  492. delete[] symbol;
  493. symbol = nullptr;
  494. }
  495. }
  496. };
  497. QVector<ParamSymbol> paramSymbols;
  498. if (std::strcmp(saveState.type, "LADSPA") == 0 || std::strcmp(saveState.type, "LV2") == 0)
  499. {
  500. for (uint32_t i=0; i < kData->param.count; i++)
  501. {
  502. getParameterSymbol(i, strBuf);
  503. if (*strBuf != 0)
  504. {
  505. ParamSymbol paramSymbol(i, strBuf);
  506. paramSymbols.append(paramSymbol);
  507. }
  508. }
  509. }
  510. // ---------------------------------------------------------------------
  511. // Part 4b - set parameter values (carefully)
  512. float sampleRate = kData->engine->getSampleRate();
  513. for (auto it = saveState.parameters.begin(); it != saveState.parameters.end(); ++it)
  514. {
  515. StateParameter* stateParameter = *it;
  516. int32_t index = -1;
  517. if (std::strcmp(saveState.type, "LADSPA") == 0)
  518. {
  519. // Try to set by symbol, otherwise use index
  520. if (stateParameter->symbol != nullptr && *stateParameter->symbol != 0)
  521. {
  522. foreach (const ParamSymbol& paramSymbol, paramSymbols)
  523. {
  524. if (std::strcmp(stateParameter->symbol, paramSymbol.symbol) == 0)
  525. {
  526. index = paramSymbol.index;
  527. break;
  528. }
  529. }
  530. if (index == -1)
  531. index = stateParameter->index;
  532. }
  533. else
  534. index = stateParameter->index;
  535. }
  536. else if (std::strcmp(saveState.type, "LV2") == 0)
  537. {
  538. // Symbol only
  539. if (stateParameter->symbol != nullptr && *stateParameter->symbol != 0)
  540. {
  541. foreach (const ParamSymbol& paramSymbol, paramSymbols)
  542. {
  543. if (std::strcmp(stateParameter->symbol, paramSymbol.symbol) == 0)
  544. {
  545. index = paramSymbol.index;
  546. break;
  547. }
  548. }
  549. if (index == -1)
  550. carla_stderr("Failed to find LV2 parameter symbol for '%s')", stateParameter->symbol);
  551. }
  552. else
  553. carla_stderr("LV2 Plugin parameter '%s' has no symbol", stateParameter->name);
  554. }
  555. else
  556. {
  557. // Index only
  558. index = stateParameter->index;
  559. }
  560. // Now set parameter
  561. if (index >= 0 && index < static_cast<int32_t>(kData->param.count))
  562. {
  563. if (kData->param.data[index].hints & PARAMETER_USES_SAMPLERATE)
  564. stateParameter->value *= sampleRate;
  565. setParameterValue(index, stateParameter->value, true, true, true);
  566. setParameterMidiCC(index, stateParameter->midiCC, true, true);
  567. setParameterMidiChannel(index, stateParameter->midiChannel-1, true, true);
  568. }
  569. else
  570. carla_stderr("Could not set parameter data for '%s'", stateParameter->name);
  571. }
  572. // clear
  573. foreach (ParamSymbol paramSymbol, paramSymbols)
  574. paramSymbol.free();
  575. // ---------------------------------------------------------------------
  576. // Part 5 - set chunk data
  577. for (auto it = saveState.customData.begin(); it != saveState.customData.end(); ++it)
  578. {
  579. const StateCustomData* const stateCustomData(*it);
  580. if (std::strcmp(stateCustomData->type, CUSTOM_DATA_CHUNK) == 0)
  581. setCustomData(stateCustomData->type, stateCustomData->key, stateCustomData->value, true);
  582. }
  583. if (saveState.chunk != nullptr && (fOptions & PLUGIN_OPTION_USE_CHUNKS) != 0)
  584. setChunkData(saveState.chunk);
  585. // ---------------------------------------------------------------------
  586. // Part 6 - set internal stuff
  587. setDryWet(saveState.dryWet, true, true);
  588. setVolume(saveState.volume, true, true);
  589. setBalanceLeft(saveState.balanceLeft, true, true);
  590. setBalanceRight(saveState.balanceRight, true, true);
  591. setPanning(saveState.panning, true, true);
  592. setCtrlChannel(saveState.ctrlChannel, true, true);
  593. setActive(saveState.active, true, true);
  594. }
  595. bool CarlaPlugin::saveStateToFile(const char* const filename)
  596. {
  597. carla_debug("CarlaPlugin::saveStateToFile(\"%s\")", filename);
  598. CARLA_ASSERT(filename != nullptr);
  599. QFile file(filename);
  600. if (! file.open(QIODevice::WriteOnly | QIODevice::Text))
  601. return false;
  602. QTextStream out(&file);
  603. out << "<?xml version='1.0' encoding='UTF-8'?>\n";
  604. out << "<!DOCTYPE CARLA-PRESET>\n";
  605. out << "<CARLA-PRESET VERSION='1.0'>\n";
  606. out << getXMLFromSaveState(getSaveState());
  607. out << "</CARLA-PRESET>\n";
  608. file.close();
  609. return true;
  610. }
  611. bool CarlaPlugin::loadStateFromFile(const char* const filename)
  612. {
  613. carla_debug("CarlaPlugin::loadStateFromFile(\"%s\")", filename);
  614. CARLA_ASSERT(filename != nullptr);
  615. QFile file(filename);
  616. if (! file.open(QIODevice::ReadOnly | QIODevice::Text))
  617. return false;
  618. QDomDocument xml;
  619. xml.setContent(file.readAll());
  620. file.close();
  621. QDomNode xmlNode(xml.documentElement());
  622. if (xmlNode.toElement().tagName() != "CARLA-PRESET")
  623. {
  624. carla_stderr2("Not a valid Carla preset file");
  625. return false;
  626. }
  627. loadSaveState(getSaveStateDictFromXML(xmlNode));
  628. return true;
  629. }
  630. // -------------------------------------------------------------------
  631. // Set data (internal stuff)
  632. void CarlaPlugin::setId(const unsigned int id)
  633. {
  634. fId = id;
  635. if (kData->engine->getProccessMode() == PROCESS_MODE_CONTINUOUS_RACK)
  636. {
  637. CARLA_ASSERT(id < MAX_RACK_PLUGINS);
  638. if (id >= MAX_RACK_PLUGINS || kData->ctrlChannel == static_cast<int8_t>(id))
  639. return;
  640. kData->ctrlChannel = id;
  641. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_CTRL_CHANNEL, 0, id, nullptr);
  642. }
  643. }
  644. void CarlaPlugin::setOption(const unsigned int option, const bool yesNo)
  645. {
  646. if (yesNo)
  647. fOptions |= option;
  648. else
  649. fOptions &= ~option;
  650. }
  651. void CarlaPlugin::setEnabled(const bool yesNo)
  652. {
  653. fEnabled = yesNo;
  654. }
  655. void CarlaPlugin::setActive(const bool active, const bool sendOsc, const bool sendCallback)
  656. {
  657. if (kData->active == active)
  658. return;
  659. kData->active = active;
  660. const float value = active ? 1.0f : 0.0f;
  661. #ifndef BUILD_BRIDGE
  662. if (sendOsc)
  663. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_ACTIVE, value);
  664. #else
  665. // unused
  666. (void)sendOsc;
  667. #endif
  668. if (sendCallback)
  669. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_ACTIVE, 0, value, nullptr);
  670. #ifndef BUILD_BRIDGE
  671. else if (fHints & PLUGIN_IS_BRIDGE)
  672. osc_send_control(&kData->osc.data, PARAMETER_ACTIVE, value);
  673. #endif
  674. }
  675. void CarlaPlugin::setDryWet(const float value, const bool sendOsc, const bool sendCallback)
  676. {
  677. CARLA_ASSERT(value >= 0.0f && value <= 1.0f);
  678. const float fixedValue = carla_fixValue<float>(0.0f, 1.0f, value);
  679. if (kData->postProc.dryWet == fixedValue)
  680. return;
  681. kData->postProc.dryWet = fixedValue;
  682. #ifndef BUILD_BRIDGE
  683. if (sendOsc)
  684. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_DRYWET, fixedValue);
  685. #else
  686. // unused
  687. (void)sendOsc;
  688. #endif
  689. if (sendCallback)
  690. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_DRYWET, 0, fixedValue, nullptr);
  691. #ifndef BUILD_BRIDGE
  692. else if (fHints & PLUGIN_IS_BRIDGE)
  693. osc_send_control(&kData->osc.data, PARAMETER_DRYWET, fixedValue);
  694. #endif
  695. }
  696. void CarlaPlugin::setVolume(const float value, const bool sendOsc, const bool sendCallback)
  697. {
  698. CARLA_ASSERT(value >= 0.0f && value <= 1.27f);
  699. const float fixedValue = carla_fixValue<float>(0.0f, 1.27f, value);
  700. if (kData->postProc.volume == fixedValue)
  701. return;
  702. kData->postProc.volume = fixedValue;
  703. #ifndef BUILD_BRIDGE
  704. if (sendOsc)
  705. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_VOLUME, fixedValue);
  706. #else
  707. // unused
  708. (void)sendOsc;
  709. #endif
  710. if (sendCallback)
  711. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_VOLUME, 0, fixedValue, nullptr);
  712. #ifndef BUILD_BRIDGE
  713. else if (fHints & PLUGIN_IS_BRIDGE)
  714. osc_send_control(&kData->osc.data, PARAMETER_VOLUME, fixedValue);
  715. #endif
  716. }
  717. void CarlaPlugin::setBalanceLeft(const float value, const bool sendOsc, const bool sendCallback)
  718. {
  719. CARLA_ASSERT(value >= -1.0f && value <= 1.0f);
  720. const float fixedValue = carla_fixValue<float>(-1.0f, 1.0f, value);
  721. if (kData->postProc.balanceLeft == fixedValue)
  722. return;
  723. kData->postProc.balanceLeft = fixedValue;
  724. #ifndef BUILD_BRIDGE
  725. if (sendOsc)
  726. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_BALANCE_LEFT, fixedValue);
  727. #else
  728. // unused
  729. (void)sendOsc;
  730. #endif
  731. if (sendCallback)
  732. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_BALANCE_LEFT, 0, fixedValue, nullptr);
  733. #ifndef BUILD_BRIDGE
  734. else if (fHints & PLUGIN_IS_BRIDGE)
  735. osc_send_control(&kData->osc.data, PARAMETER_BALANCE_LEFT, fixedValue);
  736. #endif
  737. }
  738. void CarlaPlugin::setBalanceRight(const float value, const bool sendOsc, const bool sendCallback)
  739. {
  740. CARLA_ASSERT(value >= -1.0f && value <= 1.0f);
  741. const float fixedValue = carla_fixValue<float>(-1.0f, 1.0f, value);
  742. if (kData->postProc.balanceRight == fixedValue)
  743. return;
  744. kData->postProc.balanceRight = fixedValue;
  745. #ifndef BUILD_BRIDGE
  746. if (sendOsc)
  747. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_BALANCE_RIGHT, fixedValue);
  748. #else
  749. // unused
  750. (void)sendOsc;
  751. #endif
  752. if (sendCallback)
  753. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_BALANCE_RIGHT, 0, fixedValue, nullptr);
  754. #ifndef BUILD_BRIDGE
  755. else if (fHints & PLUGIN_IS_BRIDGE)
  756. osc_send_control(&kData->osc.data, PARAMETER_BALANCE_RIGHT, fixedValue);
  757. #endif
  758. }
  759. void CarlaPlugin::setPanning(const float value, const bool sendOsc, const bool sendCallback)
  760. {
  761. CARLA_ASSERT(value >= -1.0f && value <= 1.0f);
  762. const float fixedValue = carla_fixValue<float>(-1.0f, 1.0f, value);
  763. if (kData->postProc.panning == fixedValue)
  764. return;
  765. kData->postProc.panning = fixedValue;
  766. #ifndef BUILD_BRIDGE
  767. if (sendOsc)
  768. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_PANNING, fixedValue);
  769. #else
  770. // unused
  771. (void)sendOsc;
  772. #endif
  773. if (sendCallback)
  774. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_PANNING, 0, fixedValue, nullptr);
  775. #ifndef BUILD_BRIDGE
  776. else if (fHints & PLUGIN_IS_BRIDGE)
  777. osc_send_control(&kData->osc.data, PARAMETER_PANNING, fixedValue);
  778. #endif
  779. }
  780. void CarlaPlugin::setCtrlChannel(const int8_t channel, const bool sendOsc, const bool sendCallback)
  781. {
  782. if (kData->ctrlChannel == channel)
  783. return;
  784. kData->ctrlChannel = channel;
  785. #ifndef BUILD_BRIDGE
  786. const float ctrlf = channel;
  787. if (sendOsc)
  788. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_CTRL_CHANNEL, ctrlf);
  789. #else
  790. // unused
  791. (void)sendOsc;
  792. #endif
  793. if (sendCallback)
  794. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, PARAMETER_CTRL_CHANNEL, 0, channel, nullptr);
  795. #ifndef BUILD_BRIDGE
  796. else if (fHints & PLUGIN_IS_BRIDGE)
  797. osc_send_control(&kData->osc.data, PARAMETER_CTRL_CHANNEL, ctrlf);
  798. #endif
  799. }
  800. // -------------------------------------------------------------------
  801. // Set data (plugin-specific stuff)
  802. void CarlaPlugin::setParameterValue(const uint32_t parameterId, const float value, const bool sendGui, const bool sendOsc, const bool sendCallback)
  803. {
  804. CARLA_ASSERT(parameterId < kData->param.count);
  805. if (sendGui)
  806. uiParameterChange(parameterId, value);
  807. #ifndef BUILD_BRIDGE
  808. if (sendOsc)
  809. kData->engine->osc_send_control_set_parameter_value(fId, parameterId, value);
  810. #else
  811. // unused
  812. (void)sendOsc;
  813. #endif
  814. if (sendCallback)
  815. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, parameterId, 0, value, nullptr);
  816. #ifndef BUILD_BRIDGE
  817. else if (fHints & PLUGIN_IS_BRIDGE)
  818. osc_send_control(&kData->osc.data, parameterId, value);
  819. #endif
  820. }
  821. void CarlaPlugin::setParameterValueByRIndex(const int32_t rindex, const float value, const bool sendGui, const bool sendOsc, const bool sendCallback)
  822. {
  823. CARLA_ASSERT(rindex > PARAMETER_MAX && rindex != PARAMETER_NULL);
  824. if (rindex <= PARAMETER_MAX)
  825. return;
  826. if (rindex == PARAMETER_NULL)
  827. return;
  828. if (rindex == PARAMETER_ACTIVE)
  829. return setActive((value > 0.0f), sendOsc, sendCallback);
  830. if (rindex == PARAMETER_DRYWET)
  831. return setDryWet(value, sendOsc, sendCallback);
  832. if (rindex == PARAMETER_VOLUME)
  833. return setVolume(value, sendOsc, sendCallback);
  834. if (rindex == PARAMETER_BALANCE_LEFT)
  835. return setBalanceLeft(value, sendOsc, sendCallback);
  836. if (rindex == PARAMETER_BALANCE_RIGHT)
  837. return setBalanceRight(value, sendOsc, sendCallback);
  838. if (rindex == PARAMETER_PANNING)
  839. return setPanning(value, sendOsc, sendCallback);
  840. if (rindex == PARAMETER_CTRL_CHANNEL)
  841. return setCtrlChannel(int8_t(value), sendOsc, sendCallback);
  842. for (uint32_t i=0; i < kData->param.count; i++)
  843. {
  844. if (kData->param.data[i].rindex == rindex)
  845. return setParameterValue(i, value, sendGui, sendOsc, sendCallback);
  846. }
  847. }
  848. void CarlaPlugin::setParameterMidiChannel(const uint32_t parameterId, uint8_t channel, const bool sendOsc, const bool sendCallback)
  849. {
  850. CARLA_ASSERT(parameterId < kData->param.count);
  851. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  852. if (channel >= MAX_MIDI_CHANNELS)
  853. channel = MAX_MIDI_CHANNELS;
  854. kData->param.data[parameterId].midiChannel = channel;
  855. #ifndef BUILD_BRIDGE
  856. if (sendOsc)
  857. kData->engine->osc_send_control_set_parameter_midi_channel(fId, parameterId, channel);
  858. #else
  859. // unused
  860. (void)sendOsc;
  861. #endif
  862. if (sendCallback)
  863. kData->engine->callback(CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED, fId, parameterId, channel, 0.0f, nullptr);
  864. #ifndef BUILD_BRIDGE
  865. else if (fHints & PLUGIN_IS_BRIDGE)
  866. {} // TODO
  867. #endif
  868. }
  869. void CarlaPlugin::setParameterMidiCC(const uint32_t parameterId, int16_t cc, const bool sendOsc, const bool sendCallback)
  870. {
  871. CARLA_ASSERT(parameterId < kData->param.count);
  872. CARLA_ASSERT(cc >= -1);
  873. if (cc < -1 || cc > 0x5F)
  874. cc = -1;
  875. kData->param.data[parameterId].midiCC = cc;
  876. #ifndef BUILD_BRIDGE
  877. if (sendOsc)
  878. kData->engine->osc_send_control_set_parameter_midi_cc(fId, parameterId, cc);
  879. #else
  880. // unused
  881. (void)sendOsc;
  882. #endif
  883. if (sendCallback)
  884. kData->engine->callback(CALLBACK_PARAMETER_MIDI_CC_CHANGED, fId, parameterId, cc, 0.0f, nullptr);
  885. #ifndef BUILD_BRIDGE
  886. else if (fHints & PLUGIN_IS_BRIDGE)
  887. {} // TODO
  888. #endif
  889. }
  890. void CarlaPlugin::setCustomData(const char* const type, const char* const key, const char* const value, const bool sendGui)
  891. {
  892. CARLA_ASSERT(type != nullptr);
  893. CARLA_ASSERT(key != nullptr);
  894. CARLA_ASSERT(value != nullptr);
  895. if (type == nullptr)
  896. return carla_stderr2("CarlaPlugin::setCustomData(\"%s\", \"%s\", \"%s\", %s) - type is null", type, key, value, bool2str(sendGui));
  897. if (key == nullptr)
  898. return carla_stderr2("CarlaPlugin::setCustomData(\"%s\", \"%s\", \"%s\", %s) - key is null", type, key, value, bool2str(sendGui));
  899. if (value == nullptr)
  900. return carla_stderr2("CarlaPlugin::setCustomData(\"%s\", \"%s\", \"%s\", %s) - value is null", type, key, value, bool2str(sendGui));
  901. bool saveData = true;
  902. if (std::strcmp(type, CUSTOM_DATA_STRING) == 0)
  903. {
  904. // Ignore some keys
  905. if (std::strncmp(key, "OSC:", 4) == 0 || std::strcmp(key, "guiVisible") == 0)
  906. saveData = false;
  907. //else if (std::strcmp(key, CARLA_BRIDGE_MSG_SAVE_NOW) == 0 || std::strcmp(key, CARLA_BRIDGE_MSG_SET_CHUNK) == 0 || std::strcmp(key, CARLA_BRIDGE_MSG_SET_CUSTOM) == 0)
  908. // saveData = false;
  909. }
  910. if (saveData)
  911. {
  912. // Check if we already have this key
  913. for (auto it = kData->custom.begin(); it.valid(); it.next())
  914. {
  915. CustomData& cData(*it);
  916. CARLA_ASSERT(cData.type != nullptr);
  917. CARLA_ASSERT(cData.key != nullptr);
  918. CARLA_ASSERT(cData.value != nullptr);
  919. if (std::strcmp(cData.key, key) == 0)
  920. {
  921. if (cData.value != nullptr)
  922. delete[] cData.value;
  923. cData.value = carla_strdup(value);
  924. return;
  925. }
  926. }
  927. // Otherwise store it
  928. CustomData newData;
  929. newData.type = carla_strdup(type);
  930. newData.key = carla_strdup(key);
  931. newData.value = carla_strdup(value);
  932. kData->custom.append(newData);
  933. }
  934. }
  935. void CarlaPlugin::setChunkData(const char* const stringData)
  936. {
  937. CARLA_ASSERT(stringData != nullptr);
  938. return;
  939. // unused
  940. (void)stringData;
  941. }
  942. void CarlaPlugin::setProgram(const int32_t index, const bool sendGui, const bool sendOsc, const bool sendCallback)
  943. {
  944. CARLA_ASSERT(index >= -1 && index < static_cast<int32_t>(kData->prog.count));
  945. if (index > static_cast<int32_t>(kData->prog.count))
  946. return;
  947. const int32_t fixedIndex = carla_fixValue<int32_t>(-1, kData->prog.count, index);
  948. kData->prog.current = fixedIndex;
  949. // Change default parameter values
  950. if (fixedIndex >= 0)
  951. {
  952. if (sendGui)
  953. uiProgramChange(fixedIndex);
  954. for (uint32_t i=0; i < kData->param.count; i++)
  955. {
  956. // FIXME?
  957. kData->param.ranges[i].def = getParameterValue(i);
  958. kData->param.ranges[i].fixDefault();
  959. if (sendOsc)
  960. {
  961. #ifndef BUILD_BRIDGE
  962. kData->engine->osc_send_control_set_default_value(fId, i, kData->param.ranges[i].def);
  963. kData->engine->osc_send_control_set_parameter_value(fId, i, kData->param.ranges[i].def);
  964. #endif
  965. }
  966. }
  967. }
  968. #ifndef BUILD_BRIDGE
  969. if (sendOsc)
  970. kData->engine->osc_send_control_set_program(fId, fixedIndex);
  971. #endif
  972. if (sendCallback)
  973. kData->engine->callback(CALLBACK_PROGRAM_CHANGED, fId, fixedIndex, 0, 0.0f, nullptr);
  974. }
  975. void CarlaPlugin::setMidiProgram(int32_t index, const bool sendGui, const bool sendOsc, const bool sendCallback)
  976. {
  977. CARLA_ASSERT(index >= -1 && index < static_cast<int32_t>(kData->midiprog.count));
  978. if (index > static_cast<int32_t>(kData->midiprog.count))
  979. return;
  980. const int32_t fixedIndex = carla_fixValue<int32_t>(-1, kData->midiprog.count, index);
  981. kData->midiprog.current = fixedIndex;
  982. if (fixedIndex >= 0)
  983. {
  984. if (sendGui)
  985. uiMidiProgramChange(fixedIndex);
  986. // Change default parameter values (sound banks never change defaults)
  987. #ifndef BUILD_BRIDGE // FIXME
  988. if (type() != PLUGIN_GIG && type() != PLUGIN_SF2 && type() != PLUGIN_SFZ)
  989. #endif
  990. {
  991. for (uint32_t i=0; i < kData->param.count; i++)
  992. {
  993. // FIXME?
  994. kData->param.ranges[i].def = getParameterValue(i);
  995. kData->param.ranges[i].fixDefault();
  996. if (sendOsc)
  997. {
  998. #ifndef BUILD_BRIDGE
  999. kData->engine->osc_send_control_set_default_value(fId, i, kData->param.ranges[i].def);
  1000. kData->engine->osc_send_control_set_parameter_value(fId, i, kData->param.ranges[i].def);
  1001. #endif
  1002. }
  1003. }
  1004. }
  1005. }
  1006. #ifndef BUILD_BRIDGE
  1007. if (sendOsc)
  1008. kData->engine->osc_send_control_set_midi_program(fId, fixedIndex);
  1009. #endif
  1010. if (sendCallback)
  1011. kData->engine->callback(CALLBACK_MIDI_PROGRAM_CHANGED, fId, fixedIndex, 0, 0.0f, nullptr);
  1012. }
  1013. void CarlaPlugin::setMidiProgramById(const uint32_t bank, const uint32_t program, const bool sendGui, const bool sendOsc, const bool sendCallback)
  1014. {
  1015. for (uint32_t i=0; i < kData->midiprog.count; i++)
  1016. {
  1017. if (kData->midiprog.data[i].bank == bank && kData->midiprog.data[i].program == program)
  1018. return setMidiProgram(i, sendGui, sendOsc, sendCallback);
  1019. }
  1020. }
  1021. // -------------------------------------------------------------------
  1022. // Set gui stuff
  1023. void CarlaPlugin::showGui(const bool yesNo)
  1024. {
  1025. return;
  1026. // unused
  1027. (void)yesNo;
  1028. }
  1029. void CarlaPlugin::idleGui()
  1030. {
  1031. if (! fEnabled)
  1032. return;
  1033. if (fHints & PLUGIN_HAS_SINGLE_THREAD)
  1034. {
  1035. // Process postponed events
  1036. postRtEventsRun();
  1037. // Update parameter outputs
  1038. for (uint32_t i=0; i < kData->param.count; i++)
  1039. {
  1040. if (kData->param.data[i].type == PARAMETER_OUTPUT)
  1041. uiParameterChange(i, getParameterValue(i));
  1042. }
  1043. }
  1044. }
  1045. // -------------------------------------------------------------------
  1046. // Plugin state
  1047. void CarlaPlugin::reload()
  1048. {
  1049. }
  1050. void CarlaPlugin::reloadPrograms(const bool)
  1051. {
  1052. }
  1053. // -------------------------------------------------------------------
  1054. // Plugin processing
  1055. void CarlaPlugin::process(float** const, float** const, const uint32_t)
  1056. {
  1057. }
  1058. void CarlaPlugin::bufferSizeChanged(const uint32_t)
  1059. {
  1060. }
  1061. void CarlaPlugin::sampleRateChanged(const double)
  1062. {
  1063. }
  1064. void CarlaPlugin::recreateLatencyBuffers()
  1065. {
  1066. if (kData->latencyBuffers != nullptr)
  1067. {
  1068. for (uint32_t i=0; i < kData->audioIn.count; i++)
  1069. {
  1070. CARLA_ASSERT(kData->latencyBuffers[i] != nullptr);
  1071. if (kData->latencyBuffers[i] != nullptr)
  1072. delete[] kData->latencyBuffers[i];
  1073. }
  1074. delete[] kData->latencyBuffers;
  1075. kData->latencyBuffers = nullptr;
  1076. }
  1077. if (kData->audioIn.count > 0 && kData->latency > 0)
  1078. {
  1079. kData->latencyBuffers = new float*[kData->audioIn.count];
  1080. for (uint32_t i=0; i < kData->audioIn.count; i++)
  1081. {
  1082. kData->latencyBuffers[i] = new float[kData->latency];
  1083. carla_zeroFloat(kData->latencyBuffers[i], kData->latency);
  1084. }
  1085. }
  1086. }
  1087. bool CarlaPlugin::tryLock()
  1088. {
  1089. return kData->masterMutex.tryLock();
  1090. }
  1091. void CarlaPlugin::unlock()
  1092. {
  1093. kData->masterMutex.unlock();
  1094. }
  1095. // -------------------------------------------------------------------
  1096. // OSC stuff
  1097. void CarlaPlugin::registerToOscClient()
  1098. {
  1099. #ifdef BUILD_BRIDGE
  1100. if (! kData->engine->isOscBridgeRegistered())
  1101. return;
  1102. #else
  1103. if (! kData->engine->isOscControlRegistered())
  1104. return;
  1105. #endif
  1106. #ifndef BUILD_BRIDGE
  1107. kData->engine->osc_send_control_add_plugin_start(fId, fName);
  1108. #endif
  1109. // Base data
  1110. {
  1111. char bufName[STR_MAX] = { 0 };
  1112. char bufLabel[STR_MAX] = { 0 };
  1113. char bufMaker[STR_MAX] = { 0 };
  1114. char bufCopyright[STR_MAX] = { 0 };
  1115. getRealName(bufName);
  1116. getLabel(bufLabel);
  1117. getMaker(bufMaker);
  1118. getCopyright(bufCopyright);
  1119. #ifdef BUILD_BRIDGE
  1120. kData->engine->osc_send_bridge_plugin_info(category(), fHints, bufName, bufLabel, bufMaker, bufCopyright, uniqueId());
  1121. #else
  1122. kData->engine->osc_send_control_set_plugin_data(fId, type(), category(), fHints, bufName, bufLabel, bufMaker, bufCopyright, uniqueId());
  1123. #endif
  1124. }
  1125. // Base count
  1126. {
  1127. uint32_t cIns, cOuts, cTotals;
  1128. getParameterCountInfo(&cIns, &cOuts, &cTotals);
  1129. #ifdef BUILD_BRIDGE
  1130. kData->engine->osc_send_bridge_audio_count(audioInCount(), audioOutCount(), audioInCount() + audioOutCount());
  1131. kData->engine->osc_send_bridge_midi_count(midiInCount(), midiOutCount(), midiInCount() + midiOutCount());
  1132. kData->engine->osc_send_bridge_parameter_count(cIns, cOuts, cTotals);
  1133. #else
  1134. kData->engine->osc_send_control_set_plugin_ports(fId, audioInCount(), audioOutCount(), midiInCount(), midiOutCount(), cIns, cOuts, cTotals);
  1135. #endif
  1136. }
  1137. // Plugin Parameters
  1138. if (kData->param.count > 0 && kData->param.count < kData->engine->getOptions().maxParameters)
  1139. {
  1140. char bufName[STR_MAX], bufUnit[STR_MAX];
  1141. for (uint32_t i=0; i < kData->param.count; i++)
  1142. {
  1143. getParameterName(i, bufName);
  1144. getParameterUnit(i, bufUnit);
  1145. const ParameterData& paramData(kData->param.data[i]);
  1146. const ParameterRanges& paramRanges(kData->param.ranges[i]);
  1147. #ifdef BUILD_BRIDGE
  1148. kData->engine->osc_send_bridge_parameter_info(i, bufName, bufUnit);
  1149. kData->engine->osc_send_bridge_parameter_data(i, paramData.type, paramData.rindex, paramData.hints, paramData.midiChannel, paramData.midiCC);
  1150. kData->engine->osc_send_bridge_parameter_ranges(i, paramRanges.def, paramRanges.min, paramRanges.max, paramRanges.step, paramRanges.stepSmall, paramRanges.stepLarge);
  1151. kData->engine->osc_send_bridge_set_parameter_value(i, getParameterValue(i));
  1152. #else
  1153. kData->engine->osc_send_control_set_parameter_data(fId, i, paramData.type, paramData.hints, bufName, bufUnit, getParameterValue(i));
  1154. kData->engine->osc_send_control_set_parameter_ranges(fId, i, paramRanges.min, paramRanges.max, paramRanges.def, paramRanges.step, paramRanges.stepSmall, paramRanges.stepLarge);
  1155. kData->engine->osc_send_control_set_parameter_midi_cc(fId, i, paramData.midiCC);
  1156. kData->engine->osc_send_control_set_parameter_midi_channel(fId, i, paramData.midiChannel);
  1157. kData->engine->osc_send_control_set_parameter_value(fId, i, getParameterValue(i));
  1158. #endif
  1159. }
  1160. }
  1161. // Programs
  1162. if (kData->prog.count > 0)
  1163. {
  1164. #ifdef BUILD_BRIDGE
  1165. kData->engine->osc_send_bridge_program_count(kData->prog.count);
  1166. for (uint32_t i=0; i < kData->prog.count; i++)
  1167. kData->engine->osc_send_bridge_program_info(i, kData->prog.names[i]);
  1168. kData->engine->osc_send_bridge_set_program(kData->prog.current);
  1169. #else
  1170. kData->engine->osc_send_control_set_program_count(fId, kData->prog.count);
  1171. for (uint32_t i=0; i < kData->prog.count; i++)
  1172. kData->engine->osc_send_control_set_program_name(fId, i, kData->prog.names[i]);
  1173. kData->engine->osc_send_control_set_program(fId, kData->prog.current);
  1174. #endif
  1175. }
  1176. // MIDI Programs
  1177. if (kData->midiprog.count > 0)
  1178. {
  1179. #ifdef BUILD_BRIDGE
  1180. kData->engine->osc_send_bridge_midi_program_count(kData->midiprog.count);
  1181. for (uint32_t i=0; i < kData->midiprog.count; i++)
  1182. {
  1183. const MidiProgramData& mpData(kData->midiprog.data[i]);
  1184. kData->engine->osc_send_bridge_midi_program_info(i, mpData.bank, mpData.program, mpData.name);
  1185. }
  1186. kData->engine->osc_send_bridge_set_midi_program(kData->midiprog.current);
  1187. #else
  1188. kData->engine->osc_send_control_set_midi_program_count(fId, kData->midiprog.count);
  1189. for (uint32_t i=0; i < kData->midiprog.count; i++)
  1190. {
  1191. const MidiProgramData& mpData(kData->midiprog.data[i]);
  1192. kData->engine->osc_send_control_set_midi_program_data(fId, i, mpData.bank, mpData.program, mpData.name);
  1193. }
  1194. kData->engine->osc_send_control_set_midi_program(fId, kData->midiprog.current);
  1195. #endif
  1196. }
  1197. #ifndef BUILD_BRIDGE
  1198. kData->engine->osc_send_control_add_plugin_end(fId);
  1199. // Internal Parameters
  1200. {
  1201. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_ACTIVE, kData->active ? 1.0 : 0.0);
  1202. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_DRYWET, kData->postProc.dryWet);
  1203. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_VOLUME, kData->postProc.volume);
  1204. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_BALANCE_LEFT, kData->postProc.balanceLeft);
  1205. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_BALANCE_RIGHT, kData->postProc.balanceRight);
  1206. kData->engine->osc_send_control_set_parameter_value(fId, PARAMETER_PANNING, kData->postProc.panning);
  1207. }
  1208. #endif
  1209. }
  1210. void CarlaPlugin::updateOscData(const lo_address& source, const char* const url)
  1211. {
  1212. // FIXME - remove debug prints later
  1213. carla_stdout("CarlaPlugin::updateOscData(%p, \"%s\")", source, url);
  1214. kData->osc.data.free();
  1215. const int proto = lo_address_get_protocol(source);
  1216. {
  1217. const char* host = lo_address_get_hostname(source);
  1218. const char* port = lo_address_get_port(source);
  1219. kData->osc.data.source = lo_address_new_with_proto(proto, host, port);
  1220. carla_stdout("CarlaPlugin::updateOscData() - source: host \"%s\", port \"%s\"", host, port);
  1221. }
  1222. {
  1223. char* host = lo_url_get_hostname(url);
  1224. char* port = lo_url_get_port(url);
  1225. kData->osc.data.path = carla_strdup_free(lo_url_get_path(url));
  1226. kData->osc.data.target = lo_address_new_with_proto(proto, host, port);
  1227. carla_stdout("CarlaPlugin::updateOscData() - target: host \"%s\", port \"%s\", path \"%s\"", host, port, kData->osc.data.path);
  1228. std::free(host);
  1229. std::free(port);
  1230. }
  1231. #ifndef BUILD_BRIDGE
  1232. if (fHints & PLUGIN_IS_BRIDGE)
  1233. return;
  1234. #endif
  1235. osc_send_sample_rate(&kData->osc.data, kData->engine->getSampleRate());
  1236. for (auto it = kData->custom.begin(); it.valid(); it.next())
  1237. {
  1238. const CustomData& cData(*it);
  1239. CARLA_ASSERT(cData.type != nullptr);
  1240. CARLA_ASSERT(cData.key != nullptr);
  1241. CARLA_ASSERT(cData.value != nullptr);
  1242. #ifdef WANT_LV2
  1243. if (type() == PLUGIN_LV2)
  1244. osc_send_lv2_transfer_event(&kData->osc.data, 0, cData.type, cData.value);
  1245. else
  1246. #endif
  1247. if (std::strcmp(cData.type, CUSTOM_DATA_STRING) == 0)
  1248. osc_send_configure(&kData->osc.data, cData.key, cData.value);
  1249. }
  1250. if (kData->prog.current >= 0)
  1251. osc_send_program(&kData->osc.data, kData->prog.current);
  1252. if (kData->midiprog.current >= 0)
  1253. {
  1254. const MidiProgramData& curMidiProg(kData->midiprog.getCurrent());
  1255. if (type() == PLUGIN_DSSI)
  1256. osc_send_program(&kData->osc.data, curMidiProg.bank, curMidiProg.program);
  1257. else
  1258. osc_send_midi_program(&kData->osc.data, curMidiProg.bank, curMidiProg.program);
  1259. }
  1260. for (uint32_t i=0; i < kData->param.count; i++)
  1261. osc_send_control(&kData->osc.data, kData->param.data[i].rindex, getParameterValue(i));
  1262. carla_stdout("CarlaPlugin::updateOscData() - done");
  1263. }
  1264. void CarlaPlugin::freeOscData()
  1265. {
  1266. kData->osc.data.free();
  1267. }
  1268. bool CarlaPlugin::waitForOscGuiShow()
  1269. {
  1270. carla_stdout("CarlaPlugin::waitForOscGuiShow()");
  1271. uint i=0, oscUiTimeout = kData->engine->getOptions().oscUiTimeout;
  1272. // wait for UI 'update' call
  1273. for (; i < oscUiTimeout; i++)
  1274. {
  1275. if (kData->osc.data.target != nullptr)
  1276. {
  1277. carla_stdout("CarlaPlugin::waitForOscGuiShow() - got response, asking UI to show itself now");
  1278. osc_send_show(&kData->osc.data);
  1279. return true;
  1280. }
  1281. else
  1282. carla_msleep(100);
  1283. }
  1284. carla_stdout("CarlaPlugin::waitForOscGuiShow() - Timeout while waiting for UI to respond (waited %u msecs)", oscUiTimeout);
  1285. return false;
  1286. }
  1287. // -------------------------------------------------------------------
  1288. // MIDI events
  1289. void CarlaPlugin::sendMidiSingleNote(const uint8_t channel, const uint8_t note, const uint8_t velo, const bool sendGui, const bool sendOsc, const bool sendCallback)
  1290. {
  1291. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  1292. CARLA_ASSERT(note < MAX_MIDI_NOTE);
  1293. CARLA_ASSERT(velo < MAX_MIDI_VALUE);
  1294. if (! kData->active)
  1295. return;
  1296. ExternalMidiNote extNote;
  1297. extNote.channel = channel;
  1298. extNote.note = note;
  1299. extNote.velo = velo;
  1300. kData->extNotes.append(extNote);
  1301. if (sendGui)
  1302. {
  1303. if (velo > 0)
  1304. uiNoteOn(channel, note, velo);
  1305. else
  1306. uiNoteOff(channel, note);
  1307. }
  1308. #ifndef BUILD_BRIDGE
  1309. if (sendOsc)
  1310. {
  1311. if (velo > 0)
  1312. kData->engine->osc_send_control_note_on(fId, channel, note, velo);
  1313. else
  1314. kData->engine->osc_send_control_note_off(fId, channel, note);
  1315. }
  1316. #else
  1317. // unused
  1318. (void)sendOsc;
  1319. #endif
  1320. if (sendCallback)
  1321. kData->engine->callback((velo > 0) ? CALLBACK_NOTE_ON : CALLBACK_NOTE_OFF, fId, channel, note, velo, nullptr);
  1322. }
  1323. void CarlaPlugin::sendMidiAllNotesOff()
  1324. {
  1325. if (kData->ctrlChannel < 0 || kData->ctrlChannel >= MAX_MIDI_CHANNELS)
  1326. return;
  1327. PluginPostRtEvent postEvent;
  1328. postEvent.type = kPluginPostRtEventNoteOff;
  1329. postEvent.value1 = kData->ctrlChannel;
  1330. postEvent.value2 = 0;
  1331. postEvent.value3 = 0.0f;
  1332. for (unsigned short i=0; i < MAX_MIDI_NOTE; i++)
  1333. {
  1334. postEvent.value2 = i;
  1335. kData->postRtEvents.appendRT(postEvent);
  1336. }
  1337. }
  1338. // -------------------------------------------------------------------
  1339. // Post-poned events
  1340. void CarlaPlugin::postponeRtEvent(const PluginPostRtEventType type, const int32_t value1, const int32_t value2, const float value3)
  1341. {
  1342. PluginPostRtEvent event;
  1343. event.type = type;
  1344. event.value1 = value1;
  1345. event.value2 = value2;
  1346. event.value3 = value3;
  1347. kData->postRtEvents.appendRT(event);
  1348. }
  1349. void CarlaPlugin::postRtEventsRun()
  1350. {
  1351. const CarlaMutex::ScopedLocker sl(&kData->postRtEvents.mutex);
  1352. while (! kData->postRtEvents.data.isEmpty())
  1353. {
  1354. const PluginPostRtEvent& event = kData->postRtEvents.data.getFirst(true);
  1355. switch (event.type)
  1356. {
  1357. case kPluginPostRtEventNull:
  1358. break;
  1359. case kPluginPostRtEventDebug:
  1360. kData->engine->callback(CALLBACK_DEBUG, fId, event.value1, event.value2, event.value3, nullptr);
  1361. break;
  1362. case kPluginPostRtEventParameterChange:
  1363. // Update UI
  1364. if (event.value1 >= 0)
  1365. uiParameterChange(event.value1, event.value3);
  1366. #ifndef BUILD_BRIDGE
  1367. // Update OSC control client
  1368. if (kData->engine->isOscControlRegistered())
  1369. kData->engine->osc_send_control_set_parameter_value(fId, event.value1, event.value3);
  1370. #endif
  1371. // Update Host
  1372. kData->engine->callback(CALLBACK_PARAMETER_VALUE_CHANGED, fId, event.value1, 0, event.value3, nullptr);
  1373. break;
  1374. case kPluginPostRtEventProgramChange:
  1375. // Update UI
  1376. if (event.value1 >= 0)
  1377. uiProgramChange(event.value1);
  1378. #ifndef BUILD_BRIDGE
  1379. // Update OSC control client
  1380. if (kData->engine->isOscControlRegistered())
  1381. {
  1382. kData->engine->osc_send_control_set_program(fId, event.value1);
  1383. for (uint32_t j=0; j < kData->param.count; j++)
  1384. kData->engine->osc_send_control_set_default_value(fId, j, kData->param.ranges[j].def);
  1385. }
  1386. #endif
  1387. // Update Host
  1388. kData->engine->callback(CALLBACK_PROGRAM_CHANGED, fId, event.value1, 0, 0.0, nullptr);
  1389. break;
  1390. case kPluginPostRtEventMidiProgramChange:
  1391. // Update UI
  1392. if (event.value1 >= 0)
  1393. uiMidiProgramChange(event.value1);
  1394. #ifndef BUILD_BRIDGE
  1395. // Update OSC control client
  1396. if (kData->engine->isOscControlRegistered())
  1397. {
  1398. kData->engine->osc_send_control_set_midi_program(fId, event.value1);
  1399. for (uint32_t j=0; j < kData->param.count; j++)
  1400. kData->engine->osc_send_control_set_default_value(fId, j, kData->param.ranges[j].def);
  1401. }
  1402. #endif
  1403. // Update Host
  1404. kData->engine->callback(CALLBACK_MIDI_PROGRAM_CHANGED, fId, event.value1, 0, 0.0, nullptr);
  1405. break;
  1406. case kPluginPostRtEventNoteOn:
  1407. // Update UI
  1408. uiNoteOn(event.value1, event.value2, int(event.value3));
  1409. #ifndef BUILD_BRIDGE
  1410. // Update OSC control client
  1411. if (kData->engine->isOscControlRegistered())
  1412. kData->engine->osc_send_control_note_on(fId, event.value1, event.value2, int(event.value3));
  1413. #endif
  1414. // Update Host
  1415. kData->engine->callback(CALLBACK_NOTE_ON, fId, event.value1, event.value2, int(event.value3), nullptr);
  1416. break;
  1417. case kPluginPostRtEventNoteOff:
  1418. // Update UI
  1419. uiNoteOff(event.value1, event.value2);
  1420. #ifndef BUILD_BRIDGE
  1421. // Update OSC control client
  1422. if (kData->engine->isOscControlRegistered())
  1423. kData->engine->osc_send_control_note_off(fId, event.value1, event.value2);
  1424. #endif
  1425. // Update Host
  1426. kData->engine->callback(CALLBACK_NOTE_OFF, fId, event.value1, event.value2, 0.0, nullptr);
  1427. break;
  1428. }
  1429. }
  1430. }
  1431. void CarlaPlugin::uiParameterChange(const uint32_t index, const float value)
  1432. {
  1433. CARLA_ASSERT(index < parameterCount());
  1434. return;
  1435. // unused
  1436. (void)index;
  1437. (void)value;
  1438. }
  1439. void CarlaPlugin::uiProgramChange(const uint32_t index)
  1440. {
  1441. CARLA_ASSERT(index < programCount());
  1442. return;
  1443. // unused
  1444. (void)index;
  1445. }
  1446. void CarlaPlugin::uiMidiProgramChange(const uint32_t index)
  1447. {
  1448. CARLA_ASSERT(index < midiProgramCount());
  1449. return;
  1450. // unused
  1451. (void)index;
  1452. }
  1453. void CarlaPlugin::uiNoteOn(const uint8_t channel, const uint8_t note, const uint8_t velo)
  1454. {
  1455. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  1456. CARLA_ASSERT(note < MAX_MIDI_NOTE);
  1457. CARLA_ASSERT(velo > 0 && velo < MAX_MIDI_VALUE);
  1458. return;
  1459. // unused
  1460. (void)channel;
  1461. (void)note;
  1462. (void)velo;
  1463. }
  1464. void CarlaPlugin::uiNoteOff(const uint8_t channel, const uint8_t note)
  1465. {
  1466. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  1467. CARLA_ASSERT(note < MAX_MIDI_NOTE);
  1468. return;
  1469. // unused
  1470. (void)channel;
  1471. (void)note;
  1472. }
  1473. // -------------------------------------------------------------------
  1474. // Cleanup
  1475. void CarlaPlugin::initBuffers()
  1476. {
  1477. kData->audioIn.initBuffers(kData->engine);
  1478. kData->audioOut.initBuffers(kData->engine);
  1479. kData->event.initBuffers(kData->engine);
  1480. }
  1481. void CarlaPlugin::deleteBuffers()
  1482. {
  1483. carla_debug("CarlaPlugin::deleteBuffers() - start");
  1484. kData->audioIn.clear();
  1485. kData->audioOut.clear();
  1486. kData->param.clear();
  1487. kData->event.clear();
  1488. carla_debug("CarlaPlugin::deleteBuffers() - end");
  1489. }
  1490. // -------------------------------------------------------------------
  1491. // Library functions
  1492. bool CarlaPlugin::libOpen(const char* const filename)
  1493. {
  1494. kData->lib = lib_open(filename);
  1495. return bool(kData->lib);
  1496. }
  1497. bool CarlaPlugin::libClose()
  1498. {
  1499. if (kData->lib == nullptr)
  1500. return false;
  1501. const bool ret = lib_close(kData->lib);
  1502. kData->lib = nullptr;
  1503. return ret;
  1504. }
  1505. void* CarlaPlugin::libSymbol(const char* const symbol)
  1506. {
  1507. return lib_symbol(kData->lib, symbol);
  1508. }
  1509. const char* CarlaPlugin::libError(const char* const filename)
  1510. {
  1511. return lib_error(filename);
  1512. }
  1513. // -------------------------------------------------------------------
  1514. // Scoped Disabler
  1515. CarlaPlugin::ScopedDisabler::ScopedDisabler(CarlaPlugin* const plugin)
  1516. : kPlugin(plugin)
  1517. {
  1518. carla_debug("CarlaPlugin::ScopedDisabler(%p)", plugin);
  1519. CARLA_ASSERT(plugin != nullptr);
  1520. plugin->kData->masterMutex.lock();
  1521. if (plugin->fEnabled)
  1522. plugin->fEnabled = false;
  1523. if (plugin->kData->client->isActive())
  1524. plugin->kData->client->deactivate();
  1525. }
  1526. CarlaPlugin::ScopedDisabler::~ScopedDisabler()
  1527. {
  1528. carla_debug("CarlaPlugin::~ScopedDisabler()");
  1529. kPlugin->fEnabled = true;
  1530. kPlugin->kData->client->activate();
  1531. kPlugin->kData->masterMutex.unlock();
  1532. }
  1533. // -------------------------------------------------------------------
  1534. // Scoped Process Locker
  1535. CarlaPlugin::ScopedProcessLocker::ScopedProcessLocker(CarlaPlugin* const plugin, const bool block)
  1536. : kPlugin(plugin),
  1537. kBlock(block)
  1538. {
  1539. carla_debug("CarlaPlugin::ScopedProcessLocker(%p, %s)", plugin, bool2str(block));
  1540. CARLA_ASSERT(plugin != nullptr);
  1541. if (block)
  1542. plugin->kData->singleMutex.lock();
  1543. }
  1544. CarlaPlugin::ScopedProcessLocker::~ScopedProcessLocker()
  1545. {
  1546. carla_debug("CarlaPlugin::~ScopedProcessLocker()");
  1547. if (kBlock)
  1548. {
  1549. if (kPlugin->kData->singleMutex.wasTryLockCalled())
  1550. kPlugin->kData->needsReset = true;
  1551. kPlugin->kData->singleMutex.unlock();
  1552. }
  1553. }
  1554. // -------------------------------------------------------------------
  1555. // CarlaPluginGUI
  1556. #if 0
  1557. CarlaPluginGUI::CarlaPluginGUI(QWidget* const parent, Callback* const callback)
  1558. : QMainWindow(parent),
  1559. kCallback(callback)
  1560. {
  1561. carla_debug("CarlaPluginGUI::CarlaPluginGUI(%p)", parent);
  1562. //CARLA_ASSERT(callback);
  1563. //m_container = new GuiContainer(this);
  1564. //setCentralWidget(m_container);
  1565. //adjustSize();
  1566. //m_container->setParent(this);
  1567. //m_container->show();
  1568. //m_resizable = true;
  1569. //setNewSize(50, 50);
  1570. QMainWindow::setVisible(false);
  1571. }
  1572. CarlaPluginGUI::~CarlaPluginGUI()
  1573. {
  1574. carla_debug("CarlaPluginGUI::~CarlaPluginGUI()");
  1575. //CARLA_ASSERT(m_container);
  1576. // FIXME, automatically deleted by parent ?
  1577. //delete m_container;
  1578. }
  1579. #endif
  1580. #if 0
  1581. // -------------------------------------------------------------------
  1582. GuiContainer* CarlaPluginGUI::getContainer() const
  1583. {
  1584. return m_container;
  1585. }
  1586. WId CarlaPluginGUI::getWinId() const
  1587. {
  1588. return m_container->winId();
  1589. }
  1590. // -------------------------------------------------------------------
  1591. void CarlaPluginGUI::setNewSize(int width, int height)
  1592. {
  1593. carla_debug("CarlaPluginGUI::setNewSize(%i, %i)", width, height);
  1594. if (width < 30)
  1595. width = 30;
  1596. if (height < 30)
  1597. height = 30;
  1598. if (m_resizable)
  1599. {
  1600. resize(width, height);
  1601. }
  1602. else
  1603. {
  1604. setFixedSize(width, height);
  1605. m_container->setFixedSize(width, height);
  1606. }
  1607. }
  1608. void CarlaPluginGUI::setResizable(bool resizable)
  1609. {
  1610. m_resizable = resizable;
  1611. setNewSize(width(), height());
  1612. #ifdef Q_OS_WIN
  1613. if (! resizable)
  1614. setWindowFlags(windowFlags() | Qt::MSWindowsFixedSizeDialogHint);
  1615. #endif
  1616. }
  1617. void CarlaPluginGUI::setTitle(const char* const title)
  1618. {
  1619. CARLA_ASSERT(title);
  1620. setWindowTitle(QString("%1 (GUI)").arg(title));
  1621. }
  1622. void CarlaPluginGUI::setVisible(const bool yesNo)
  1623. {
  1624. carla_debug("CarlaPluginGUI::setVisible(%s)", bool2str(yesNo));
  1625. if (yesNo)
  1626. {
  1627. if (! m_geometry.isNull())
  1628. restoreGeometry(m_geometry);
  1629. }
  1630. else
  1631. m_geometry = saveGeometry();
  1632. QMainWindow::setVisible(yesNo);
  1633. }
  1634. // -------------------------------------------------------------------
  1635. void CarlaPluginGUI::hideEvent(QHideEvent* const event)
  1636. {
  1637. carla_debug("CarlaPluginGUI::hideEvent(%p)", event);
  1638. CARLA_ASSERT(event);
  1639. event->accept();
  1640. close();
  1641. }
  1642. void CarlaPluginGUI::closeEvent(QCloseEvent* const event)
  1643. {
  1644. carla_debug("CarlaPluginGUI::closeEvent(%p)", event);
  1645. CARLA_ASSERT(event);
  1646. if (event->spontaneous())
  1647. {
  1648. if (m_callback)
  1649. m_callback->guiClosedCallback();
  1650. QMainWindow::closeEvent(event);
  1651. return;
  1652. }
  1653. event->ignore();
  1654. }
  1655. #endif
  1656. // -------------------------------------------------------------------
  1657. CARLA_BACKEND_END_NAMESPACE