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.

2110 lines
59KB

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