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.

2367 lines
91KB

  1. /*
  2. * Carla Plugin Host
  3. * Copyright (C) 2011-2014 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License as
  7. * published by the Free Software Foundation; either version 2 of
  8. * the License, or any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * For a full copy of the GNU General Public License see the doc/GPL.txt file.
  16. */
  17. /* TODO:
  18. * - complete processRack(): carefully add to input, sorted events
  19. * - implement processPatchbay()
  20. * - implement oscSend_control_switch_plugins()
  21. * - proper find&load plugins
  22. * - something about the peaks?
  23. * - patchbayDisconnect should return false sometimes
  24. */
  25. #include "CarlaEngineInternal.hpp"
  26. #include "CarlaPlugin.hpp"
  27. #include "CarlaBackendUtils.hpp"
  28. #include "CarlaEngineUtils.hpp"
  29. #include "CarlaMathUtils.hpp"
  30. #include "CarlaStateUtils.hpp"
  31. #include "CarlaMIDI.h"
  32. #include "jackbridge/JackBridge.hpp"
  33. #include "juce_core.h"
  34. using juce::File;
  35. using juce::MemoryOutputStream;
  36. using juce::ScopedPointer;
  37. using juce::String;
  38. using juce::XmlDocument;
  39. using juce::XmlElement;
  40. // -----------------------------------------------------------------------
  41. CARLA_BACKEND_START_NAMESPACE
  42. #if 0
  43. } // Fix editor indentation
  44. #endif
  45. // -----------------------------------------------------------------------
  46. // Carla Engine
  47. CarlaEngine::CarlaEngine()
  48. : pData(new ProtectedData(this))
  49. {
  50. carla_debug("CarlaEngine::CarlaEngine()");
  51. }
  52. CarlaEngine::~CarlaEngine()
  53. {
  54. carla_debug("CarlaEngine::~CarlaEngine()");
  55. delete pData;
  56. }
  57. // -----------------------------------------------------------------------
  58. // Static calls
  59. uint CarlaEngine::getDriverCount()
  60. {
  61. carla_debug("CarlaEngine::getDriverCount()");
  62. uint count = 0;
  63. if (jackbridge_is_ok())
  64. count += 1;
  65. #ifndef BUILD_BRIDGE
  66. count += getRtAudioApiCount();
  67. # ifdef HAVE_JUCE
  68. count += getJuceApiCount();
  69. # endif
  70. #endif
  71. return count;
  72. }
  73. const char* CarlaEngine::getDriverName(const uint index2)
  74. {
  75. carla_debug("CarlaEngine::getDriverName(%i)", index2);
  76. uint index(index2);
  77. if (jackbridge_is_ok() && index-- == 0)
  78. return "JACK";
  79. #ifndef BUILD_BRIDGE
  80. if (index < getRtAudioApiCount())
  81. return getRtAudioApiName(index);
  82. index -= getRtAudioApiCount();
  83. # ifdef HAVE_JUCE
  84. if (index < getJuceApiCount())
  85. return getJuceApiName(index);
  86. # endif
  87. #endif
  88. carla_stderr("CarlaEngine::getDriverName(%i) - invalid index", index2);
  89. return nullptr;
  90. }
  91. const char* const* CarlaEngine::getDriverDeviceNames(const uint index2)
  92. {
  93. carla_debug("CarlaEngine::getDriverDeviceNames(%i)", index2);
  94. uint index(index2);
  95. if (jackbridge_is_ok() && index-- == 0)
  96. {
  97. static const char* ret[3] = { "Auto-Connect OFF", "Auto-Connect ON", nullptr };
  98. return ret;
  99. }
  100. #ifndef BUILD_BRIDGE
  101. if (index < getRtAudioApiCount())
  102. return getRtAudioApiDeviceNames(index);
  103. index -= getRtAudioApiCount();
  104. # ifdef HAVE_JUCE
  105. if (index < getJuceApiCount())
  106. return getJuceApiDeviceNames(index);
  107. # endif
  108. #endif
  109. carla_stderr("CarlaEngine::getDriverDeviceNames(%i) - invalid index", index2);
  110. return nullptr;
  111. }
  112. const EngineDriverDeviceInfo* CarlaEngine::getDriverDeviceInfo(const uint index2, const char* const deviceName)
  113. {
  114. carla_debug("CarlaEngine::getDriverDeviceInfo(%i, \"%s\")", index2, deviceName);
  115. uint index(index2);
  116. if (jackbridge_is_ok() && index-- == 0)
  117. {
  118. static uint32_t bufSizes[11] = { 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 0 };
  119. static EngineDriverDeviceInfo devInfo;
  120. devInfo.hints = ENGINE_DRIVER_DEVICE_VARIABLE_BUFFER_SIZE;
  121. devInfo.bufferSizes = bufSizes;
  122. devInfo.sampleRates = nullptr;
  123. return &devInfo;
  124. }
  125. #ifndef BUILD_BRIDGE
  126. if (index < getRtAudioApiCount())
  127. return getRtAudioDeviceInfo(index, deviceName);
  128. index -= getRtAudioApiCount();
  129. # ifdef HAVE_JUCE
  130. if (index < getJuceApiCount())
  131. return getJuceDeviceInfo(index, deviceName);
  132. # endif
  133. #endif
  134. carla_stderr("CarlaEngine::getDriverDeviceNames(%i, \"%s\") - invalid index", index2, deviceName);
  135. return nullptr;
  136. }
  137. CarlaEngine* CarlaEngine::newDriverByName(const char* const driverName)
  138. {
  139. CARLA_SAFE_ASSERT_RETURN(driverName != nullptr && driverName[0] != '\0', nullptr);
  140. carla_debug("CarlaEngine::newDriverByName(\"%s\")", driverName);
  141. if (std::strcmp(driverName, "JACK") == 0)
  142. return newJack();
  143. #ifndef BUILD_BRIDGE
  144. // -------------------------------------------------------------------
  145. // common
  146. if (std::strncmp(driverName, "JACK ", 5) == 0)
  147. return newRtAudio(AUDIO_API_JACK);
  148. // -------------------------------------------------------------------
  149. // linux
  150. if (std::strcmp(driverName, "ALSA") == 0)
  151. {
  152. # ifdef HAVE_JUCE
  153. return newJuce(AUDIO_API_ALSA);
  154. # else
  155. return newRtAudio(AUDIO_API_ALSA);
  156. # endif
  157. }
  158. if (std::strcmp(driverName, "OSS") == 0)
  159. return newRtAudio(AUDIO_API_OSS);
  160. if (std::strcmp(driverName, "PulseAudio") == 0)
  161. return newRtAudio(AUDIO_API_PULSE);
  162. // -------------------------------------------------------------------
  163. // macos
  164. if (std::strcmp(driverName, "CoreAudio") == 0)
  165. {
  166. # ifdef HAVE_JUCE
  167. return newJuce(AUDIO_API_CORE);
  168. # else
  169. return newRtAudio(AUDIO_API_CORE);
  170. # endif
  171. }
  172. // -------------------------------------------------------------------
  173. // windows
  174. if (std::strcmp(driverName, "ASIO") == 0)
  175. {
  176. # ifdef HAVE_JUCE
  177. return newJuce(AUDIO_API_ASIO);
  178. # else
  179. return newRtAudio(AUDIO_API_ASIO);
  180. # endif
  181. }
  182. if (std::strcmp(driverName, "DirectSound") == 0)
  183. {
  184. # ifdef HAVE_JUCE
  185. return newJuce(AUDIO_API_DS);
  186. # else
  187. return newRtAudio(AUDIO_API_DS);
  188. # endif
  189. }
  190. #endif
  191. carla_stderr("CarlaEngine::newDriverByName(\"%s\") - invalid driver name", driverName);
  192. return nullptr;
  193. }
  194. // -----------------------------------------------------------------------
  195. // Maximum values
  196. uint CarlaEngine::getMaxClientNameSize() const noexcept
  197. {
  198. return STR_MAX/2;
  199. }
  200. uint CarlaEngine::getMaxPortNameSize() const noexcept
  201. {
  202. return STR_MAX;
  203. }
  204. uint CarlaEngine::getCurrentPluginCount() const noexcept
  205. {
  206. return pData->curPluginCount;
  207. }
  208. uint CarlaEngine::getMaxPluginNumber() const noexcept
  209. {
  210. return pData->maxPluginNumber;
  211. }
  212. // -----------------------------------------------------------------------
  213. // Virtual, per-engine type calls
  214. bool CarlaEngine::init(const char* const clientName)
  215. {
  216. carla_debug("CarlaEngine::init(\"%s\")", clientName);
  217. if (! pData->init(clientName))
  218. return false;
  219. callback(ENGINE_CALLBACK_ENGINE_STARTED, 0, pData->options.processMode, pData->options.transportMode, 0.0f, getCurrentDriverName());
  220. return true;
  221. }
  222. bool CarlaEngine::close()
  223. {
  224. carla_debug("CarlaEngine::close()");
  225. if (pData->curPluginCount != 0)
  226. {
  227. pData->aboutToClose = true;
  228. removeAllPlugins();
  229. }
  230. #ifndef BUILD_BRIDGE
  231. if (pData->osc.isControlRegistered())
  232. oscSend_control_exit();
  233. #endif
  234. pData->close();
  235. callback(ENGINE_CALLBACK_ENGINE_STOPPED, 0, 0, 0, 0.0f, nullptr);
  236. return true;
  237. }
  238. void CarlaEngine::idle() noexcept
  239. {
  240. CARLA_SAFE_ASSERT_RETURN(pData->nextAction.opcode == kEnginePostActionNull,); // TESTING, remove later
  241. CARLA_SAFE_ASSERT_RETURN(pData->nextPluginId == pData->maxPluginNumber,); // TESTING, remove later
  242. CARLA_SAFE_ASSERT_RETURN(pData->plugins != nullptr,); // this one too maybe
  243. for (uint i=0; i < pData->curPluginCount; ++i)
  244. {
  245. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  246. if (plugin != nullptr && plugin->isEnabled())
  247. {
  248. try {
  249. plugin->idle();
  250. } CARLA_SAFE_EXCEPTION_CONTINUE("Plugin idle");
  251. }
  252. }
  253. pData->osc.idle();
  254. }
  255. CarlaEngineClient* CarlaEngine::addClient(CarlaPlugin* const)
  256. {
  257. return new CarlaEngineClient(*this);
  258. }
  259. // -----------------------------------------------------------------------
  260. // Plugin management
  261. bool CarlaEngine::addPlugin(const BinaryType btype, const PluginType ptype, const char* const filename, const char* const name, const char* const label, const int64_t uniqueId, const void* const extra)
  262. {
  263. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #10)");
  264. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextPluginId <= pData->maxPluginNumber, "Invalid engine internal data (err #11)");
  265. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #12)");
  266. CARLA_SAFE_ASSERT_RETURN_ERR(btype != BINARY_NONE, "Invalid plugin params (err #1)");
  267. CARLA_SAFE_ASSERT_RETURN_ERR(ptype != PLUGIN_NONE, "Invalid plugin params (err #2)");
  268. CARLA_SAFE_ASSERT_RETURN_ERR((filename != nullptr && filename[0] != '\0') || (label != nullptr && label[0] != '\0'), "Invalid plugin params (err #3)");
  269. carla_debug("CarlaEngine::addPlugin(%i:%s, %i:%s, \"%s\", \"%s\", \"%s\", " P_INT64 ", %p)", btype, BinaryType2Str(btype), ptype, PluginType2Str(ptype), filename, name, label, uniqueId, extra);
  270. uint id;
  271. #ifndef BUILD_BRIDGE
  272. CarlaPlugin* oldPlugin = nullptr;
  273. if (pData->nextPluginId < pData->curPluginCount)
  274. {
  275. id = pData->nextPluginId;
  276. pData->nextPluginId = pData->maxPluginNumber;
  277. oldPlugin = pData->plugins[id].plugin;
  278. CARLA_SAFE_ASSERT_RETURN_ERR(oldPlugin != nullptr, "Invalid replace plugin Id");
  279. }
  280. else
  281. #endif
  282. {
  283. id = pData->curPluginCount;
  284. if (id == pData->maxPluginNumber)
  285. {
  286. setLastError("Maximum number of plugins reached");
  287. return false;
  288. }
  289. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins[id].plugin == nullptr, "Invalid engine internal data (err #13)");
  290. }
  291. CarlaPlugin::Initializer initializer = {
  292. this,
  293. id,
  294. filename,
  295. name,
  296. label,
  297. uniqueId
  298. };
  299. CarlaPlugin* plugin = nullptr;
  300. #ifndef BUILD_BRIDGE
  301. CarlaString bridgeBinary(pData->options.binaryDir);
  302. if (bridgeBinary.isNotEmpty())
  303. {
  304. # ifndef CARLA_OS_WIN
  305. if (btype == BINARY_NATIVE)
  306. {
  307. bridgeBinary += OS_SEP_STR "carla-bridge-native";
  308. }
  309. else
  310. # endif
  311. {
  312. switch (btype)
  313. {
  314. case BINARY_POSIX32:
  315. bridgeBinary += OS_SEP_STR "carla-bridge-posix32";
  316. break;
  317. case BINARY_POSIX64:
  318. bridgeBinary += OS_SEP_STR "carla-bridge-posix64";
  319. break;
  320. case BINARY_WIN32:
  321. bridgeBinary += OS_SEP_STR "carla-bridge-win32.exe";
  322. break;
  323. case BINARY_WIN64:
  324. bridgeBinary += OS_SEP_STR "carla-bridge-win64.exe";
  325. break;
  326. default:
  327. bridgeBinary.clear();
  328. break;
  329. }
  330. }
  331. File file(bridgeBinary.buffer());
  332. if (! file.existsAsFile())
  333. bridgeBinary.clear();
  334. }
  335. if (ptype != PLUGIN_INTERNAL && ptype != PLUGIN_JACK && (btype != BINARY_NATIVE || (pData->options.preferPluginBridges && bridgeBinary.isNotEmpty())))
  336. {
  337. if (bridgeBinary.isNotEmpty())
  338. {
  339. plugin = CarlaPlugin::newBridge(initializer, btype, ptype, bridgeBinary);
  340. }
  341. # ifdef CARLA_OS_LINUX
  342. else if (btype == BINARY_WIN32)
  343. {
  344. // fallback to dssi-vst
  345. File file(filename);
  346. CarlaString label2(file.getFullPathName().toRawUTF8());
  347. label2.replace(' ', '*');
  348. CarlaPlugin::Initializer init2 = {
  349. this,
  350. id,
  351. "/usr/lib/dssi/dssi-vst.so",
  352. name,
  353. label2,
  354. uniqueId
  355. };
  356. char* const oldVstPath(getenv("VST_PATH"));
  357. carla_setenv("VST_PATH", file.getParentDirectory().getFullPathName().toRawUTF8());
  358. plugin = CarlaPlugin::newDSSI(init2);
  359. if (oldVstPath != nullptr)
  360. carla_setenv("VST_PATH", oldVstPath);
  361. }
  362. # endif
  363. else
  364. {
  365. setLastError("This Carla build cannot handle this binary");
  366. return false;
  367. }
  368. }
  369. else
  370. #endif // ! BUILD_BRIDGE
  371. {
  372. bool use16Outs;
  373. setLastError("Invalid or unsupported plugin type");
  374. switch (ptype)
  375. {
  376. case PLUGIN_NONE:
  377. break;
  378. case PLUGIN_INTERNAL:
  379. if (std::strcmp(label, "Csound") == 0)
  380. {
  381. plugin = CarlaPlugin::newCsound(initializer);
  382. }
  383. else if (std::strcmp(label, "FluidSynth") == 0)
  384. {
  385. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  386. plugin = CarlaPlugin::newFluidSynth(initializer, use16Outs);
  387. }
  388. else if (std::strcmp(label, "LinuxSampler (GIG)") == 0)
  389. {
  390. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  391. plugin = CarlaPlugin::newLinuxSampler(initializer, "GIG", use16Outs);
  392. }
  393. else if (std::strcmp(label, "LinuxSampler (SF2)") == 0)
  394. {
  395. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  396. plugin = CarlaPlugin::newLinuxSampler(initializer, "SF2", use16Outs);
  397. }
  398. else if (std::strcmp(label, "LinuxSampler (SFZ)") == 0)
  399. {
  400. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  401. plugin = CarlaPlugin::newLinuxSampler(initializer, "SFZ", use16Outs);
  402. }
  403. else
  404. {
  405. plugin = CarlaPlugin::newNative(initializer);
  406. }
  407. break;
  408. case PLUGIN_LADSPA:
  409. plugin = CarlaPlugin::newLADSPA(initializer, (const LADSPA_RDF_Descriptor*)extra);
  410. break;
  411. case PLUGIN_DSSI:
  412. plugin = CarlaPlugin::newDSSI(initializer);
  413. break;
  414. case PLUGIN_LV2:
  415. plugin = CarlaPlugin::newLV2(initializer);
  416. break;
  417. case PLUGIN_VST:
  418. plugin = CarlaPlugin::newVST(initializer);
  419. break;
  420. case PLUGIN_VST3:
  421. plugin = CarlaPlugin::newVST3(initializer);
  422. break;
  423. case PLUGIN_AU:
  424. plugin = CarlaPlugin::newAU(initializer);
  425. break;
  426. case PLUGIN_JACK:
  427. plugin = CarlaPlugin::newJACK(initializer);
  428. break;
  429. case PLUGIN_REWIRE:
  430. plugin = CarlaPlugin::newReWire(initializer);
  431. break;
  432. case PLUGIN_FILE_CSD:
  433. plugin = CarlaPlugin::newFileCSD(initializer);
  434. break;
  435. case PLUGIN_FILE_GIG:
  436. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  437. plugin = CarlaPlugin::newFileGIG(initializer, use16Outs);
  438. break;
  439. case PLUGIN_FILE_SF2:
  440. use16Outs = (extra != nullptr && std::strcmp((const char*)extra, "true") == 0);
  441. plugin = CarlaPlugin::newFileSF2(initializer, use16Outs);
  442. break;
  443. case PLUGIN_FILE_SFZ:
  444. plugin = CarlaPlugin::newFileSFZ(initializer);
  445. break;
  446. }
  447. }
  448. if (plugin == nullptr)
  449. {
  450. #ifndef BUILD_BRIDGE
  451. pData->plugins[id].plugin = oldPlugin;
  452. #endif
  453. return false;
  454. }
  455. plugin->registerToOscClient();
  456. EnginePluginData& pluginData(pData->plugins[id]);
  457. pluginData.plugin = plugin;
  458. pluginData.insPeak[0] = 0.0f;
  459. pluginData.insPeak[1] = 0.0f;
  460. pluginData.outsPeak[0] = 0.0f;
  461. pluginData.outsPeak[1] = 0.0f;
  462. #ifndef BUILD_BRIDGE
  463. if (oldPlugin != nullptr)
  464. {
  465. bool wasActive = (oldPlugin->getInternalParameterValue(PARAMETER_ACTIVE) >= 0.5f);
  466. float oldDryWet = oldPlugin->getInternalParameterValue(PARAMETER_DRYWET);
  467. float oldVolume = oldPlugin->getInternalParameterValue(PARAMETER_VOLUME);
  468. delete oldPlugin;
  469. callback(ENGINE_CALLBACK_RELOAD_ALL, id, 0, 0, 0.0f, plugin->getName());
  470. if (wasActive)
  471. plugin->setActive(true, true, true);
  472. if (plugin->getHints() & PLUGIN_CAN_DRYWET)
  473. plugin->setDryWet(oldDryWet, true, true);
  474. if (plugin->getHints() & PLUGIN_CAN_VOLUME)
  475. plugin->setVolume(oldVolume, true, true);
  476. }
  477. else
  478. #endif
  479. {
  480. ++pData->curPluginCount;
  481. callback(ENGINE_CALLBACK_PLUGIN_ADDED, id, 0, 0, 0.0f, plugin->getName());
  482. //if (pData->curPluginCount == 1 && pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  483. // callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_DATA_CHANGED, 0, PATCHBAY_ICON_CARLA, 0, 0.0f, nullptr);
  484. }
  485. return true;
  486. }
  487. bool CarlaEngine::addPlugin(const PluginType ptype, const char* const filename, const char* const name, const char* const label, const int64_t uniqueId, const void* const extra)
  488. {
  489. return addPlugin(BINARY_NATIVE, ptype, filename, name, label, uniqueId, extra);
  490. }
  491. bool CarlaEngine::removePlugin(const uint id)
  492. {
  493. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #14)");
  494. CARLA_SAFE_ASSERT_RETURN_ERR(pData->curPluginCount != 0, "Invalid engine internal data (err #15)");
  495. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #16)");
  496. CARLA_SAFE_ASSERT_RETURN_ERR(id < pData->curPluginCount, "Invalid plugin Id (err #1)");
  497. carla_debug("CarlaEngine::removePlugin(%i)", id);
  498. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  499. CARLA_SAFE_ASSERT_RETURN_ERR(plugin != nullptr, "Could not find plugin to remove");
  500. CARLA_SAFE_ASSERT_RETURN_ERR(plugin->getId() == id, "Invalid engine internal data (err #17)");
  501. pData->thread.stopThread(500);
  502. const bool lockWait(isRunning() && pData->options.processMode != ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS);
  503. const ScopedActionLock sal(pData, kEnginePostActionRemovePlugin, id, 0, lockWait);
  504. #ifndef BUILD_BRIDGE
  505. if (isOscControlRegistered())
  506. oscSend_control_remove_plugin(id);
  507. #endif
  508. delete plugin;
  509. if (isRunning() && ! pData->aboutToClose)
  510. pData->thread.startThread();
  511. callback(ENGINE_CALLBACK_PLUGIN_REMOVED, id, 0, 0, 0.0f, nullptr);
  512. return true;
  513. }
  514. bool CarlaEngine::removeAllPlugins()
  515. {
  516. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #18)");
  517. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextPluginId == pData->maxPluginNumber, "Invalid engine internal data (err #19)");
  518. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #20)");
  519. carla_debug("CarlaEngine::removeAllPlugins()");
  520. if (pData->curPluginCount == 0)
  521. return true;
  522. pData->thread.stopThread(500);
  523. const bool lockWait(isRunning());
  524. const ScopedActionLock sal(pData, kEnginePostActionZeroCount, 0, 0, lockWait);
  525. callback(ENGINE_CALLBACK_IDLE, 0, 0, 0, 0.0f, nullptr);
  526. for (uint i=0; i < pData->maxPluginNumber; ++i)
  527. {
  528. EnginePluginData& pluginData(pData->plugins[i]);
  529. if (pluginData.plugin != nullptr)
  530. {
  531. delete pluginData.plugin;
  532. pluginData.plugin = nullptr;
  533. }
  534. pluginData.insPeak[0] = 0.0f;
  535. pluginData.insPeak[1] = 0.0f;
  536. pluginData.outsPeak[0] = 0.0f;
  537. pluginData.outsPeak[1] = 0.0f;
  538. callback(ENGINE_CALLBACK_IDLE, 0, 0, 0, 0.0f, nullptr);
  539. }
  540. if (isRunning() && ! pData->aboutToClose)
  541. pData->thread.startThread();
  542. return true;
  543. }
  544. const char* CarlaEngine::renamePlugin(const uint id, const char* const newName)
  545. {
  546. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->plugins != nullptr, "Invalid engine internal data (err #21)");
  547. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->curPluginCount != 0, "Invalid engine internal data (err #22)");
  548. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #23)");
  549. CARLA_SAFE_ASSERT_RETURN_ERRN(id < pData->curPluginCount, "Invalid plugin Id (err #2)");
  550. CARLA_SAFE_ASSERT_RETURN_ERRN(newName != nullptr && newName[0] != '\0', "Invalid plugin name");
  551. carla_debug("CarlaEngine::renamePlugin(%i, \"%s\")", id, newName);
  552. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  553. CARLA_SAFE_ASSERT_RETURN_ERRN(plugin != nullptr, "Could not find plugin to rename");
  554. CARLA_SAFE_ASSERT_RETURN_ERRN(plugin->getId() == id, "Invalid engine internal data (err #24)");
  555. if (const char* const name = getUniquePluginName(newName))
  556. {
  557. plugin->setName(name);
  558. return name;
  559. }
  560. setLastError("Unable to get new unique plugin name");
  561. return nullptr;
  562. }
  563. bool CarlaEngine::clonePlugin(const uint id)
  564. {
  565. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #25)");
  566. CARLA_SAFE_ASSERT_RETURN_ERR(pData->curPluginCount != 0, "Invalid engine internal data (err #26)");
  567. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #27)");
  568. CARLA_SAFE_ASSERT_RETURN_ERR(id < pData->curPluginCount, "Invalid plugin Id (err #3)");
  569. carla_debug("CarlaEngine::clonePlugin(%i)", id);
  570. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  571. CARLA_SAFE_ASSERT_RETURN_ERR(plugin != nullptr, "Could not find plugin to clone");
  572. CARLA_SAFE_ASSERT_RETURN_ERR(plugin->getId() == id, "Invalid engine internal data (err #28)");
  573. char label[STR_MAX+1];
  574. carla_zeroChar(label, STR_MAX+1);
  575. plugin->getLabel(label);
  576. const uint pluginCountBefore(pData->curPluginCount);
  577. if (! addPlugin(plugin->getBinaryType(), plugin->getType(), plugin->getFilename(), plugin->getName(), label, plugin->getUniqueId(), plugin->getExtraStuff()))
  578. return false;
  579. CARLA_SAFE_ASSERT_RETURN_ERR(pluginCountBefore+1 == pData->curPluginCount, "No new plugin found");
  580. if (CarlaPlugin* const newPlugin = pData->plugins[pluginCountBefore].plugin)
  581. newPlugin->loadStateSave(plugin->getStateSave());
  582. return true;
  583. }
  584. bool CarlaEngine::replacePlugin(const uint id)
  585. {
  586. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #29)");
  587. CARLA_SAFE_ASSERT_RETURN_ERR(pData->curPluginCount != 0, "Invalid engine internal data (err #30)");
  588. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #31)");
  589. carla_debug("CarlaEngine::replacePlugin(%i)", id);
  590. // might use this to reset
  591. if (id == pData->curPluginCount || id == pData->maxPluginNumber)
  592. {
  593. pData->nextPluginId = pData->maxPluginNumber;
  594. return true;
  595. }
  596. CARLA_SAFE_ASSERT_RETURN_ERR(id < pData->curPluginCount, "Invalid plugin Id (err #4)");
  597. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  598. CARLA_SAFE_ASSERT_RETURN_ERR(plugin != nullptr, "Could not find plugin to replace");
  599. CARLA_SAFE_ASSERT_RETURN_ERR(plugin->getId() == id, "Invalid engine internal data (err #32)");
  600. pData->nextPluginId = id;
  601. return true;
  602. }
  603. bool CarlaEngine::switchPlugins(const uint idA, const uint idB)
  604. {
  605. CARLA_SAFE_ASSERT_RETURN_ERR(pData->plugins != nullptr, "Invalid engine internal data (err #33)");
  606. CARLA_SAFE_ASSERT_RETURN_ERR(pData->curPluginCount >= 2, "Invalid engine internal data (err #34)");
  607. CARLA_SAFE_ASSERT_RETURN_ERR(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #35)");
  608. CARLA_SAFE_ASSERT_RETURN_ERR(idA != idB, "Invalid operation, cannot switch plugin with itself");
  609. CARLA_SAFE_ASSERT_RETURN_ERR(idA < pData->curPluginCount, "Invalid plugin Id (err #5)");
  610. CARLA_SAFE_ASSERT_RETURN_ERR(idB < pData->curPluginCount, "Invalid plugin Id (err #6)");
  611. carla_debug("CarlaEngine::switchPlugins(%i)", idA, idB);
  612. CarlaPlugin* const pluginA(pData->plugins[idA].plugin);
  613. CarlaPlugin* const pluginB(pData->plugins[idB].plugin);
  614. CARLA_SAFE_ASSERT_RETURN_ERR(pluginA != nullptr, "Could not find plugin to switch (err #1)");
  615. CARLA_SAFE_ASSERT_RETURN_ERR(pluginA != nullptr, "Could not find plugin to switch (err #2)");
  616. CARLA_SAFE_ASSERT_RETURN_ERR(pluginA->getId() == idA, "Invalid engine internal data (err #36)");
  617. CARLA_SAFE_ASSERT_RETURN_ERR(pluginB->getId() == idB, "Invalid engine internal data (err #37)");
  618. pData->thread.stopThread(500);
  619. const bool lockWait(isRunning() && pData->options.processMode != ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS);
  620. const ScopedActionLock sal(pData, kEnginePostActionSwitchPlugins, idA, idB, lockWait);
  621. #ifndef BUILD_BRIDGE // TODO
  622. //if (isOscControlRegistered())
  623. // oscSend_control_switch_plugins(idA, idB);
  624. #endif
  625. if (isRunning() && ! pData->aboutToClose)
  626. pData->thread.startThread();
  627. return true;
  628. }
  629. CarlaPlugin* CarlaEngine::getPlugin(const uint id) const
  630. {
  631. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->plugins != nullptr, "Invalid engine internal data (err #38)");
  632. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->curPluginCount != 0, "Invalid engine internal data (err #39)");
  633. CARLA_SAFE_ASSERT_RETURN_ERRN(pData->nextAction.opcode == kEnginePostActionNull, "Invalid engine internal data (err #40)");
  634. CARLA_SAFE_ASSERT_RETURN_ERRN(id < pData->curPluginCount, "Invalid plugin Id (err #7)");
  635. return pData->plugins[id].plugin;
  636. }
  637. CarlaPlugin* CarlaEngine::getPluginUnchecked(const uint id) const noexcept
  638. {
  639. return pData->plugins[id].plugin;
  640. }
  641. const char* CarlaEngine::getUniquePluginName(const char* const name) const
  642. {
  643. CARLA_SAFE_ASSERT_RETURN(pData->nextAction.opcode == kEnginePostActionNull, nullptr);
  644. CARLA_SAFE_ASSERT_RETURN(name != nullptr && name[0] != '\0', nullptr);
  645. carla_debug("CarlaEngine::getUniquePluginName(\"%s\")", name);
  646. CarlaString sname;
  647. sname = name;
  648. if (sname.isEmpty())
  649. {
  650. sname = "(No name)";
  651. return sname.dup();
  652. }
  653. const size_t maxNameSize(carla_min<uint>(getMaxClientNameSize(), 0xff, 6) - 6); // 6 = strlen(" (10)") + 1
  654. if (maxNameSize == 0 || ! isRunning())
  655. return sname.dup();
  656. sname.truncate(maxNameSize);
  657. sname.replace(':', '.'); // ':' is used in JACK1 to split client/port names
  658. for (uint i=0; i < pData->curPluginCount; ++i)
  659. {
  660. CARLA_SAFE_ASSERT_BREAK(pData->plugins[i].plugin != nullptr);
  661. // Check if unique name doesn't exist
  662. if (const char* const pluginName = pData->plugins[i].plugin->getName())
  663. {
  664. if (sname != pluginName)
  665. continue;
  666. }
  667. // Check if string has already been modified
  668. {
  669. const size_t len(sname.length());
  670. // 1 digit, ex: " (2)"
  671. if (sname[len-4] == ' ' && sname[len-3] == '(' && sname.isDigit(len-2) && sname[len-1] == ')')
  672. {
  673. int number = sname[len-2] - '0';
  674. if (number == 9)
  675. {
  676. // next number is 10, 2 digits
  677. sname.truncate(len-4);
  678. sname += " (10)";
  679. //sname.replace(" (9)", " (10)");
  680. }
  681. else
  682. sname[len-2] = char('0' + number + 1);
  683. continue;
  684. }
  685. // 2 digits, ex: " (11)"
  686. if (sname[len-5] == ' ' && sname[len-4] == '(' && sname.isDigit(len-3) && sname.isDigit(len-2) && sname[len-1] == ')')
  687. {
  688. char n2 = sname[len-2];
  689. char n3 = sname[len-3];
  690. if (n2 == '9')
  691. {
  692. n2 = '0';
  693. n3 = static_cast<char>(n3 + 1);
  694. }
  695. else
  696. n2 = static_cast<char>(n2 + 1);
  697. sname[len-2] = n2;
  698. sname[len-3] = n3;
  699. continue;
  700. }
  701. }
  702. // Modify string if not
  703. sname += " (2)";
  704. }
  705. return sname.dup();
  706. }
  707. // -----------------------------------------------------------------------
  708. // Project management
  709. bool CarlaEngine::loadFile(const char* const filename)
  710. {
  711. CARLA_SAFE_ASSERT_RETURN_ERR(filename != nullptr && filename[0] != '\0', "Invalid filename (err #1)");
  712. carla_debug("CarlaEngine::loadFile(\"%s\")", filename);
  713. File file(filename);
  714. CARLA_SAFE_ASSERT_RETURN_ERR(file.existsAsFile(), "Requested file does not exist or is not a readable file");
  715. CarlaString baseName(file.getFileName().toRawUTF8());
  716. CarlaString extension(file.getFileExtension().toRawUTF8());
  717. extension.toLower();
  718. // -------------------------------------------------------------------
  719. if (extension == "carxp" || extension == "carxs")
  720. return loadProject(filename);
  721. // -------------------------------------------------------------------
  722. if (extension == "csd")
  723. return addPlugin(PLUGIN_FILE_CSD, filename, baseName, baseName, 0, nullptr);
  724. if (extension == "gig")
  725. return addPlugin(PLUGIN_FILE_GIG, filename, baseName, baseName, 0, nullptr);
  726. if (extension == "sf2")
  727. return addPlugin(PLUGIN_FILE_SF2, filename, baseName, baseName, 0, nullptr);
  728. if (extension == "sfz")
  729. return addPlugin(PLUGIN_FILE_SFZ, filename, baseName, baseName, 0, nullptr);
  730. // -------------------------------------------------------------------
  731. if (extension == "aiff" || extension == "flac" || extension == "oga" || extension == "ogg" || extension == "w64" || extension == "wav")
  732. {
  733. #ifdef WANT_AUDIOFILE
  734. if (addPlugin(PLUGIN_INTERNAL, nullptr, baseName, "audiofile", 0, nullptr))
  735. {
  736. if (CarlaPlugin* const plugin = getPlugin(pData->curPluginCount-1))
  737. plugin->setCustomData(CUSTOM_DATA_TYPE_STRING, "file", filename, true);
  738. return true;
  739. }
  740. return false;
  741. #else
  742. setLastError("This Carla build does not have Audio file support");
  743. return false;
  744. #endif
  745. }
  746. if (extension == "3g2" || extension == "3gp" || extension == "aac" || extension == "ac3" || extension == "amr" || extension == "ape" ||
  747. extension == "mp2" || extension == "mp3" || extension == "mpc" || extension == "wma")
  748. {
  749. #ifdef WANT_AUDIOFILE
  750. # ifdef HAVE_FFMPEG
  751. if (addPlugin(PLUGIN_INTERNAL, nullptr, baseName, "audiofile", 0, nullptr))
  752. {
  753. if (CarlaPlugin* const plugin = getPlugin(pData->curPluginCount-1))
  754. plugin->setCustomData(CUSTOM_DATA_TYPE_STRING, "file", filename, true);
  755. return true;
  756. }
  757. return false;
  758. # else
  759. setLastError("This Carla build has Audio file support, but not libav/ffmpeg");
  760. return false;
  761. # endif
  762. #else
  763. setLastError("This Carla build does not have Audio file support");
  764. return false;
  765. #endif
  766. }
  767. // -------------------------------------------------------------------
  768. if (extension == "mid" || extension == "midi")
  769. {
  770. #ifdef WANT_MIDIFILE
  771. if (addPlugin(PLUGIN_INTERNAL, nullptr, baseName, "midifile", 0, nullptr))
  772. {
  773. if (CarlaPlugin* const plugin = getPlugin(pData->curPluginCount-1))
  774. plugin->setCustomData(CUSTOM_DATA_TYPE_STRING, "file", filename, true);
  775. return true;
  776. }
  777. return false;
  778. #else
  779. setLastError("This Carla build does not have MIDI file support");
  780. return false;
  781. #endif
  782. }
  783. // -------------------------------------------------------------------
  784. // ZynAddSubFX
  785. if (extension == "xmz" || extension == "xiz")
  786. {
  787. #ifdef WANT_ZYNADDSUBFX
  788. if (addPlugin(PLUGIN_INTERNAL, nullptr, baseName, "zynaddsubfx", 0, nullptr))
  789. {
  790. if (CarlaPlugin* const plugin = getPlugin(pData->curPluginCount-1))
  791. plugin->setCustomData(CUSTOM_DATA_TYPE_STRING, (extension == "xmz") ? "CarlaAlternateFile1" : "CarlaAlternateFile2", filename, true);
  792. return true;
  793. }
  794. return false;
  795. #else
  796. setLastError("This Carla build does not have ZynAddSubFX support");
  797. return false;
  798. #endif
  799. }
  800. // -------------------------------------------------------------------
  801. setLastError("Unknown file extension");
  802. return false;
  803. }
  804. bool CarlaEngine::loadProject(const char* const filename)
  805. {
  806. CARLA_SAFE_ASSERT_RETURN_ERR(filename != nullptr && filename[0] != '\0', "Invalid filename (err #2)");
  807. carla_debug("CarlaEngine::loadProject(\"%s\")", filename);
  808. File file(filename);
  809. CARLA_SAFE_ASSERT_RETURN_ERR(file.existsAsFile(), "Requested file does not exist or is not a readable file");
  810. XmlDocument xml(file);
  811. ScopedPointer<XmlElement> xmlElement(xml.getDocumentElement(true));
  812. CARLA_SAFE_ASSERT_RETURN_ERR(xmlElement != nullptr, "Failed to parse project file");
  813. const String& xmlType(xmlElement->getTagName());
  814. const bool isPreset(xmlType.equalsIgnoreCase("carla-preset"));
  815. if (! (xmlType.equalsIgnoreCase("carla-project") || isPreset))
  816. {
  817. setLastError("Not a valid Carla project or preset file");
  818. return false;
  819. }
  820. // completely load file
  821. xmlElement = xml.getDocumentElement(false);
  822. CARLA_SAFE_ASSERT_RETURN_ERR(xmlElement != nullptr, "Failed to completely parse project file");
  823. // handle plugins first
  824. for (XmlElement* elem = xmlElement->getFirstChildElement(); elem != nullptr; elem = elem->getNextElement())
  825. {
  826. const String& tagName(elem->getTagName());
  827. if (isPreset || tagName.equalsIgnoreCase("plugin"))
  828. {
  829. StateSave stateSave;
  830. stateSave.fillFromXmlElement(isPreset ? xmlElement.get() : elem);
  831. callback(ENGINE_CALLBACK_IDLE, 0, 0, 0, 0.0f, nullptr);
  832. CARLA_SAFE_ASSERT_CONTINUE(stateSave.type != nullptr);
  833. const void* extraStuff = nullptr;
  834. // check if using GIG, SF2 or SFZ 16outs
  835. static const char kUse16OutsSuffix[] = " (16 outs)";
  836. const PluginType ptype(getPluginTypeFromString(stateSave.type));
  837. if (CarlaString(stateSave.label).endsWith(kUse16OutsSuffix))
  838. {
  839. if (ptype == PLUGIN_FILE_GIG || ptype == PLUGIN_FILE_SF2)
  840. extraStuff = "true";
  841. }
  842. // TODO - proper find&load plugins
  843. if (addPlugin(ptype, stateSave.binary, stateSave.name, stateSave.label, stateSave.uniqueId, extraStuff))
  844. {
  845. if (CarlaPlugin* const plugin = getPlugin(pData->curPluginCount-1))
  846. plugin->loadStateSave(stateSave);
  847. }
  848. else
  849. carla_stderr2("Failed to load a plugin, error was:%s\n", getLastError());
  850. }
  851. if (isPreset)
  852. return true;
  853. }
  854. #ifndef BUILD_BRIDGE
  855. callback(ENGINE_CALLBACK_IDLE, 0, 0, 0, 0.0f, nullptr);
  856. // if we're running inside some session-manager, let them handle the connections
  857. if (pData->options.processMode != ENGINE_PROCESS_MODE_PATCHBAY)
  858. {
  859. if (std::getenv("CARLA_DONT_MANAGE_CONNECTIONS") != nullptr || std::getenv("LADISH_APP_NAME") != nullptr || std::getenv("NSM_URL") != nullptr)
  860. return true;
  861. }
  862. // now handle connections
  863. for (XmlElement* elem = xmlElement->getFirstChildElement(); elem != nullptr; elem = elem->getNextElement())
  864. {
  865. const String& tagName(elem->getTagName());
  866. if (tagName.equalsIgnoreCase("patchbay"))
  867. {
  868. CarlaString sourcePort, targetPort;
  869. for (XmlElement* patchElem = elem->getFirstChildElement(); patchElem != nullptr; patchElem = patchElem->getNextElement())
  870. {
  871. const String& patchTag(patchElem->getTagName());
  872. sourcePort.clear();
  873. targetPort.clear();
  874. if (! patchTag.equalsIgnoreCase("connection"))
  875. continue;
  876. for (XmlElement* connElem = patchElem->getFirstChildElement(); connElem != nullptr; connElem = connElem->getNextElement())
  877. {
  878. const String& tag(connElem->getTagName());
  879. const String text(connElem->getAllSubText().trim());
  880. if (tag.equalsIgnoreCase("source"))
  881. sourcePort = text.toRawUTF8();
  882. else if (tag.equalsIgnoreCase("target"))
  883. targetPort = text.toRawUTF8();
  884. }
  885. if (sourcePort.isNotEmpty() && targetPort.isNotEmpty())
  886. restorePatchbayConnection(sourcePort, targetPort);
  887. }
  888. break;
  889. }
  890. }
  891. #endif
  892. return true;
  893. }
  894. bool CarlaEngine::saveProject(const char* const filename)
  895. {
  896. CARLA_SAFE_ASSERT_RETURN_ERR(filename != nullptr && filename[0] != '\0', "Invalid filename (err #3)");
  897. carla_debug("CarlaEngine::saveProject(\"%s\")", filename);
  898. MemoryOutputStream out;
  899. out << "<?xml version='1.0' encoding='UTF-8'?>\n";
  900. out << "<!DOCTYPE CARLA-PROJECT>\n";
  901. out << "<CARLA-PROJECT VERSION='2.0'>\n";
  902. bool firstPlugin = true;
  903. char strBuf[STR_MAX+1];
  904. for (uint i=0; i < pData->curPluginCount; ++i)
  905. {
  906. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  907. if (plugin != nullptr && plugin->isEnabled())
  908. {
  909. if (! firstPlugin)
  910. out << "\n";
  911. strBuf[0] = '\0';
  912. plugin->getRealName(strBuf);
  913. //if (strBuf[0] != '\0')
  914. // out << QString(" <!-- %1 -->\n").arg(xmlSafeString(strBuf, true));
  915. out << " <Plugin>\n";
  916. out << plugin->getStateSave().toString();
  917. out << " </Plugin>\n";
  918. firstPlugin = false;
  919. }
  920. }
  921. #ifndef BUILD_BRIDGE
  922. // if we're running inside some session-manager, let them handle the connections
  923. if (pData->options.processMode != ENGINE_PROCESS_MODE_PATCHBAY)
  924. {
  925. if (std::getenv("CARLA_DONT_MANAGE_CONNECTIONS") != nullptr || std::getenv("LADISH_APP_NAME") != nullptr || std::getenv("NSM_URL") != nullptr)
  926. return true;
  927. }
  928. if (const char* const* patchbayConns = getPatchbayConnections())
  929. {
  930. if (! firstPlugin)
  931. out << "\n";
  932. out << " <Patchbay>\n";
  933. for (int i=0; patchbayConns[i] != nullptr && patchbayConns[i+1] != nullptr; ++i, ++i )
  934. {
  935. const char* const connSource(patchbayConns[i]);
  936. const char* const connTarget(patchbayConns[i+1]);
  937. CARLA_SAFE_ASSERT_CONTINUE(connSource != nullptr && connSource[0] != '\0');
  938. CARLA_SAFE_ASSERT_CONTINUE(connTarget != nullptr && connTarget[0] != '\0');
  939. out << " <Connection>\n";
  940. out << " <Source>" << connSource << "</Source>\n";
  941. out << " <Target>" << connTarget << "</Target>\n";
  942. out << " </Connection>\n";
  943. delete[] connSource;
  944. delete[] connTarget;
  945. }
  946. out << " </Patchbay>\n";
  947. }
  948. #endif
  949. out << "</CARLA-PROJECT>\n";
  950. File file(filename);
  951. if (file.replaceWithData(out.getData(), out.getDataSize()))
  952. return true;
  953. setLastError("Failed to write file");
  954. return false;
  955. }
  956. // -----------------------------------------------------------------------
  957. // Information (base)
  958. uint CarlaEngine::getHints() const noexcept
  959. {
  960. return pData->hints;
  961. }
  962. uint32_t CarlaEngine::getBufferSize() const noexcept
  963. {
  964. return pData->bufferSize;
  965. }
  966. double CarlaEngine::getSampleRate() const noexcept
  967. {
  968. return pData->sampleRate;
  969. }
  970. const char* CarlaEngine::getName() const noexcept
  971. {
  972. return pData->name;
  973. }
  974. EngineProcessMode CarlaEngine::getProccessMode() const noexcept
  975. {
  976. return pData->options.processMode;
  977. }
  978. const EngineOptions& CarlaEngine::getOptions() const noexcept
  979. {
  980. return pData->options;
  981. }
  982. const EngineTimeInfo& CarlaEngine::getTimeInfo() const noexcept
  983. {
  984. return pData->timeInfo;
  985. }
  986. // -----------------------------------------------------------------------
  987. // Information (peaks)
  988. float CarlaEngine::getInputPeak(const uint pluginId, const bool isLeft) const noexcept
  989. {
  990. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount, 0.0f);
  991. return pData->plugins[pluginId].insPeak[isLeft ? 0 : 1];
  992. }
  993. float CarlaEngine::getOutputPeak(const uint pluginId, const bool isLeft) const noexcept
  994. {
  995. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount, 0.0f);
  996. return pData->plugins[pluginId].outsPeak[isLeft ? 0 : 1];
  997. }
  998. // -----------------------------------------------------------------------
  999. // Callback
  1000. void CarlaEngine::callback(const EngineCallbackOpcode action, const uint pluginId, const int value1, const int value2, const float value3, const char* const valueStr) noexcept
  1001. {
  1002. carla_debug("CarlaEngine::callback(%s, %i, %i, %i, %f, \"%s\")", EngineCallbackOpcode2Str(action), pluginId, value1, value2, value3, valueStr);
  1003. if (pData->callback != nullptr)
  1004. {
  1005. try {
  1006. pData->callback(pData->callbackPtr, action, pluginId, value1, value2, value3, valueStr);
  1007. } catch(...) {}
  1008. }
  1009. }
  1010. void CarlaEngine::setCallback(const EngineCallbackFunc func, void* const ptr) noexcept
  1011. {
  1012. carla_debug("CarlaEngine::setCallback(%p, %p)", func, ptr);
  1013. pData->callback = func;
  1014. pData->callbackPtr = ptr;
  1015. }
  1016. // -----------------------------------------------------------------------
  1017. // File Callback
  1018. const char* CarlaEngine::runFileCallback(const FileCallbackOpcode action, const bool isDir, const char* const title, const char* const filter) noexcept
  1019. {
  1020. CARLA_SAFE_ASSERT_RETURN(title != nullptr && title[0] != '\0', nullptr);
  1021. CARLA_SAFE_ASSERT_RETURN(filter != nullptr, nullptr);
  1022. carla_debug("CarlaEngine::runFileCallback(%i:%s, %s, \"%s\", \"%s\")", action, FileCallbackOpcode2Str(action), bool2str(isDir), title, filter);
  1023. const char* ret = nullptr;
  1024. if (pData->fileCallback != nullptr)
  1025. {
  1026. try {
  1027. ret = pData->fileCallback(pData->fileCallbackPtr, action, isDir, title, filter);
  1028. } catch(...) {}
  1029. }
  1030. return ret;
  1031. }
  1032. void CarlaEngine::setFileCallback(const FileCallbackFunc func, void* const ptr) noexcept
  1033. {
  1034. carla_debug("CarlaEngine::setFileCallback(%p, %p)", func, ptr);
  1035. pData->fileCallback = func;
  1036. pData->fileCallbackPtr = ptr;
  1037. }
  1038. #ifndef BUILD_BRIDGE
  1039. // -----------------------------------------------------------------------
  1040. // Patchbay
  1041. bool CarlaEngine::patchbayConnect(const uint groupA, const uint portA, const uint groupB, const uint portB)
  1042. {
  1043. CARLA_SAFE_ASSERT_RETURN(pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK || pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY, false);
  1044. CARLA_SAFE_ASSERT_RETURN(pData->audio.isReady, false);
  1045. carla_debug("CarlaEngine::patchbayConnect(%u, %u, %u, %u)", groupA, portA, groupB, portB);
  1046. if (pData->graph.isRack)
  1047. {
  1048. CARLA_SAFE_ASSERT_RETURN(pData->graph.rack != nullptr, nullptr);
  1049. return pData->graph.rack->connect(this, groupA, portA, groupB, portB);
  1050. }
  1051. else
  1052. {
  1053. CARLA_SAFE_ASSERT_RETURN(pData->graph.patchbay != nullptr, nullptr);
  1054. return pData->graph.patchbay->connect(this, groupA, portA, groupB, portB);
  1055. }
  1056. }
  1057. bool CarlaEngine::patchbayDisconnect(const uint connectionId)
  1058. {
  1059. CARLA_SAFE_ASSERT_RETURN(pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK || pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY, false);
  1060. CARLA_SAFE_ASSERT_RETURN(pData->audio.isReady, false);
  1061. carla_debug("CarlaEngine::patchbayDisconnect(%u)", connectionId);
  1062. if (pData->graph.isRack)
  1063. {
  1064. CARLA_SAFE_ASSERT_RETURN(pData->graph.rack != nullptr, nullptr);
  1065. return pData->graph.rack->disconnect(this, connectionId);
  1066. }
  1067. else
  1068. {
  1069. CARLA_SAFE_ASSERT_RETURN(pData->graph.patchbay != nullptr, nullptr);
  1070. return pData->graph.patchbay->disconnect(this, connectionId);
  1071. }
  1072. }
  1073. bool CarlaEngine::patchbayRefresh()
  1074. {
  1075. setLastError("Unsupported operation");
  1076. return false;
  1077. }
  1078. #endif
  1079. // -----------------------------------------------------------------------
  1080. // Transport
  1081. void CarlaEngine::transportPlay() noexcept
  1082. {
  1083. pData->time.playing = true;
  1084. }
  1085. void CarlaEngine::transportPause() noexcept
  1086. {
  1087. pData->time.playing = false;
  1088. }
  1089. void CarlaEngine::transportRelocate(const uint64_t frame) noexcept
  1090. {
  1091. pData->time.frame = frame;
  1092. }
  1093. // -----------------------------------------------------------------------
  1094. // Error handling
  1095. const char* CarlaEngine::getLastError() const noexcept
  1096. {
  1097. return pData->lastError;
  1098. }
  1099. void CarlaEngine::setLastError(const char* const error) const noexcept
  1100. {
  1101. pData->lastError = error;
  1102. }
  1103. void CarlaEngine::setAboutToClose() noexcept
  1104. {
  1105. carla_debug("CarlaEngine::setAboutToClose()");
  1106. pData->aboutToClose = true;
  1107. }
  1108. // -----------------------------------------------------------------------
  1109. // Global options
  1110. void CarlaEngine::setOption(const EngineOption option, const int value, const char* const valueStr)
  1111. {
  1112. carla_debug("CarlaEngine::setOption(%i:%s, %i, \"%s\")", option, EngineOption2Str(option), value, valueStr);
  1113. if (isRunning() && (option == ENGINE_OPTION_PROCESS_MODE || option == ENGINE_OPTION_AUDIO_NUM_PERIODS || option == ENGINE_OPTION_AUDIO_DEVICE))
  1114. return carla_stderr("CarlaEngine::setOption(%i:%s, %i, \"%s\") - Cannot set this option while engine is running!", option, EngineOption2Str(option), value, valueStr);
  1115. switch (option)
  1116. {
  1117. case ENGINE_OPTION_DEBUG:
  1118. case ENGINE_OPTION_NSM_INIT:
  1119. break;
  1120. case ENGINE_OPTION_PROCESS_MODE:
  1121. CARLA_SAFE_ASSERT_RETURN(value >= ENGINE_PROCESS_MODE_SINGLE_CLIENT && value <= ENGINE_PROCESS_MODE_BRIDGE,);
  1122. pData->options.processMode = static_cast<EngineProcessMode>(value);
  1123. break;
  1124. case ENGINE_OPTION_TRANSPORT_MODE:
  1125. CARLA_SAFE_ASSERT_RETURN(value >= ENGINE_TRANSPORT_MODE_INTERNAL && value <= ENGINE_TRANSPORT_MODE_BRIDGE,);
  1126. pData->options.transportMode = static_cast<EngineTransportMode>(value);
  1127. break;
  1128. case ENGINE_OPTION_FORCE_STEREO:
  1129. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  1130. pData->options.forceStereo = (value != 0);
  1131. break;
  1132. case ENGINE_OPTION_PREFER_PLUGIN_BRIDGES:
  1133. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  1134. pData->options.preferPluginBridges = (value != 0);
  1135. break;
  1136. case ENGINE_OPTION_PREFER_UI_BRIDGES:
  1137. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  1138. pData->options.preferUiBridges = (value != 0);
  1139. break;
  1140. case ENGINE_OPTION_UIS_ALWAYS_ON_TOP:
  1141. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  1142. pData->options.uisAlwaysOnTop = (value != 0);
  1143. break;
  1144. case ENGINE_OPTION_MAX_PARAMETERS:
  1145. CARLA_SAFE_ASSERT_RETURN(value >= 0,);
  1146. pData->options.maxParameters = static_cast<uint>(value);
  1147. break;
  1148. case ENGINE_OPTION_UI_BRIDGES_TIMEOUT:
  1149. CARLA_SAFE_ASSERT_RETURN(value >= 0,);
  1150. pData->options.uiBridgesTimeout = static_cast<uint>(value);
  1151. break;
  1152. case ENGINE_OPTION_AUDIO_NUM_PERIODS:
  1153. CARLA_SAFE_ASSERT_RETURN(value >= 2 && value <= 3,);
  1154. pData->options.audioNumPeriods = static_cast<uint>(value);
  1155. break;
  1156. case ENGINE_OPTION_AUDIO_BUFFER_SIZE:
  1157. CARLA_SAFE_ASSERT_RETURN(value >= 8,);
  1158. pData->options.audioBufferSize = static_cast<uint>(value);
  1159. break;
  1160. case ENGINE_OPTION_AUDIO_SAMPLE_RATE:
  1161. CARLA_SAFE_ASSERT_RETURN(value >= 22050,);
  1162. pData->options.audioSampleRate = static_cast<uint>(value);
  1163. break;
  1164. case ENGINE_OPTION_AUDIO_DEVICE:
  1165. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr,);
  1166. if (pData->options.audioDevice != nullptr)
  1167. delete[] pData->options.audioDevice;
  1168. pData->options.audioDevice = carla_strdup(valueStr);
  1169. break;
  1170. case ENGINE_OPTION_PATH_BINARIES:
  1171. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  1172. if (pData->options.binaryDir != nullptr)
  1173. delete[] pData->options.binaryDir;
  1174. pData->options.binaryDir = carla_strdup(valueStr);
  1175. break;
  1176. case ENGINE_OPTION_PATH_RESOURCES:
  1177. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  1178. if (pData->options.resourceDir != nullptr)
  1179. delete[] pData->options.resourceDir;
  1180. pData->options.resourceDir = carla_strdup(valueStr);
  1181. break;
  1182. case ENGINE_OPTION_FRONTEND_WIN_ID:
  1183. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  1184. const long long winId(std::strtoll(valueStr, nullptr, 16));
  1185. CARLA_SAFE_ASSERT_RETURN(winId >= 0,);
  1186. pData->options.frontendWinId = static_cast<uintptr_t>(winId);
  1187. break;
  1188. }
  1189. }
  1190. // -----------------------------------------------------------------------
  1191. // OSC Stuff
  1192. #ifdef BUILD_BRIDGE
  1193. bool CarlaEngine::isOscBridgeRegistered() const noexcept
  1194. {
  1195. return (pData->oscData != nullptr);
  1196. }
  1197. #else
  1198. bool CarlaEngine::isOscControlRegistered() const noexcept
  1199. {
  1200. return pData->osc.isControlRegistered();
  1201. }
  1202. #endif
  1203. void CarlaEngine::idleOsc() const noexcept
  1204. {
  1205. pData->osc.idle();
  1206. }
  1207. const char* CarlaEngine::getOscServerPathTCP() const noexcept
  1208. {
  1209. return pData->osc.getServerPathTCP();
  1210. }
  1211. const char* CarlaEngine::getOscServerPathUDP() const noexcept
  1212. {
  1213. return pData->osc.getServerPathUDP();
  1214. }
  1215. #ifdef BUILD_BRIDGE
  1216. void CarlaEngine::setOscBridgeData(const CarlaOscData* const oscData) const noexcept
  1217. {
  1218. pData->oscData = oscData;
  1219. }
  1220. #endif
  1221. // -----------------------------------------------------------------------
  1222. // Helper functions
  1223. EngineEvent* CarlaEngine::getInternalEventBuffer(const bool isInput) const noexcept
  1224. {
  1225. return isInput ? pData->events.in : pData->events.out;
  1226. }
  1227. void CarlaEngine::registerEnginePlugin(const uint id, CarlaPlugin* const plugin) noexcept
  1228. {
  1229. CARLA_SAFE_ASSERT_RETURN(id == pData->curPluginCount,);
  1230. carla_debug("CarlaEngine::registerEnginePlugin(%i, %p)", id, plugin);
  1231. pData->plugins[id].plugin = plugin;
  1232. }
  1233. // -----------------------------------------------------------------------
  1234. // Internal stuff
  1235. void CarlaEngine::bufferSizeChanged(const uint32_t newBufferSize)
  1236. {
  1237. carla_debug("CarlaEngine::bufferSizeChanged(%i)", newBufferSize);
  1238. for (uint i=0; i < pData->curPluginCount; ++i)
  1239. {
  1240. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1241. if (plugin != nullptr && plugin->isEnabled())
  1242. plugin->bufferSizeChanged(newBufferSize);
  1243. }
  1244. callback(ENGINE_CALLBACK_BUFFER_SIZE_CHANGED, 0, static_cast<int>(newBufferSize), 0, 0.0f, nullptr);
  1245. }
  1246. void CarlaEngine::sampleRateChanged(const double newSampleRate)
  1247. {
  1248. carla_debug("CarlaEngine::sampleRateChanged(%g)", newSampleRate);
  1249. for (uint i=0; i < pData->curPluginCount; ++i)
  1250. {
  1251. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1252. if (plugin != nullptr && plugin->isEnabled())
  1253. plugin->sampleRateChanged(newSampleRate);
  1254. }
  1255. callback(ENGINE_CALLBACK_SAMPLE_RATE_CHANGED, 0, 0, 0, static_cast<float>(newSampleRate), nullptr);
  1256. }
  1257. void CarlaEngine::offlineModeChanged(const bool isOfflineNow)
  1258. {
  1259. carla_debug("CarlaEngine::offlineModeChanged(%s)", bool2str(isOfflineNow));
  1260. for (uint i=0; i < pData->curPluginCount; ++i)
  1261. {
  1262. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1263. if (plugin != nullptr && plugin->isEnabled())
  1264. plugin->offlineModeChanged(isOfflineNow);
  1265. }
  1266. }
  1267. void CarlaEngine::runPendingRtEvents() noexcept
  1268. {
  1269. pData->doNextPluginAction(true);
  1270. if (pData->time.playing)
  1271. pData->time.frame += pData->bufferSize;
  1272. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_INTERNAL)
  1273. {
  1274. pData->timeInfo.playing = pData->time.playing;
  1275. pData->timeInfo.frame = pData->time.frame;
  1276. }
  1277. }
  1278. void CarlaEngine::setPluginPeaks(const uint pluginId, float const inPeaks[2], float const outPeaks[2]) noexcept
  1279. {
  1280. EnginePluginData& pluginData(pData->plugins[pluginId]);
  1281. pluginData.insPeak[0] = inPeaks[0];
  1282. pluginData.insPeak[1] = inPeaks[1];
  1283. pluginData.outsPeak[0] = outPeaks[0];
  1284. pluginData.outsPeak[1] = outPeaks[1];
  1285. }
  1286. #ifndef BUILD_BRIDGE
  1287. // -----------------------------------------------------------------------
  1288. // Patchbay stuff
  1289. const char* const* CarlaEngine::getPatchbayConnections() const
  1290. {
  1291. carla_debug("CarlaEngine::getPatchbayConnections()");
  1292. if (pData->graph.isRack)
  1293. {
  1294. CARLA_SAFE_ASSERT_RETURN(pData->graph.rack != nullptr, nullptr);
  1295. return pData->graph.rack->getConnections();
  1296. }
  1297. else
  1298. {
  1299. CARLA_SAFE_ASSERT_RETURN(pData->graph.patchbay != nullptr, nullptr);
  1300. return pData->graph.patchbay->getConnections();
  1301. }
  1302. }
  1303. void CarlaEngine::restorePatchbayConnection(const char* const connSource, const char* const connTarget)
  1304. {
  1305. CARLA_SAFE_ASSERT_RETURN(connSource != nullptr && connSource[0] != '\0',);
  1306. CARLA_SAFE_ASSERT_RETURN(connTarget != nullptr && connTarget[0] != '\0',);
  1307. carla_debug("CarlaEngine::restorePatchbayConnection(\"%s\", \"%s\")", connSource, connTarget);
  1308. uint groupA, portA;
  1309. uint groupB, portB;
  1310. if (pData->graph.isRack)
  1311. {
  1312. CARLA_SAFE_ASSERT_RETURN(pData->graph.rack != nullptr,);
  1313. if (! pData->graph.rack->getPortIdFromFullName(connSource, groupA, portA))
  1314. return;
  1315. if (! pData->graph.rack->getPortIdFromFullName(connTarget, groupB, portB))
  1316. return;
  1317. }
  1318. else
  1319. {
  1320. CARLA_SAFE_ASSERT_RETURN(pData->graph.patchbay != nullptr,);
  1321. if (! pData->graph.patchbay->getPortIdFromFullName(connSource, groupA, portA))
  1322. return;
  1323. if (! pData->graph.patchbay->getPortIdFromFullName(connTarget, groupB, portB))
  1324. return;
  1325. }
  1326. patchbayConnect(groupA, portA, groupB, portB);
  1327. }
  1328. #endif
  1329. // -----------------------------------------------------------------------
  1330. // Bridge/Controller OSC stuff
  1331. #ifdef BUILD_BRIDGE
  1332. void CarlaEngine::oscSend_bridge_plugin_info1(const PluginCategory category, const uint hints, const int64_t uniqueId) const noexcept
  1333. {
  1334. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1335. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1336. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1337. carla_debug("CarlaEngine::oscSend_bridge_plugin_info1(%i:%s, %X, " P_INT64 ")", category, PluginCategory2Str(category), hints, uniqueId);
  1338. char targetPath[std::strlen(pData->oscData->path)+21];
  1339. std::strcpy(targetPath, pData->oscData->path);
  1340. std::strcat(targetPath, "/bridge_plugin_info1");
  1341. try_lo_send(pData->oscData->target, targetPath, "iih", static_cast<int32_t>(category), static_cast<int32_t>(hints), uniqueId);
  1342. }
  1343. void CarlaEngine::oscSend_bridge_plugin_info2(const char* const realName, const char* const label, const char* const maker, const char* const copyright) const noexcept
  1344. {
  1345. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1346. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1347. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1348. CARLA_SAFE_ASSERT_RETURN(realName != nullptr && realName[0] != '\0',);
  1349. CARLA_SAFE_ASSERT_RETURN(label != nullptr && label[0] != '\0',);
  1350. CARLA_SAFE_ASSERT_RETURN(maker != nullptr,);
  1351. CARLA_SAFE_ASSERT_RETURN(copyright != nullptr,);
  1352. carla_debug("CarlaEngine::oscSend_bridge_plugin_info2(\"%s\", \"%s\", \"%s\", \"%s\")", realName, label, maker, copyright);
  1353. char targetPath[std::strlen(pData->oscData->path)+21];
  1354. std::strcpy(targetPath, pData->oscData->path);
  1355. std::strcat(targetPath, "/bridge_plugin_info2");
  1356. try_lo_send(pData->oscData->target, targetPath, "ssss", realName, label, maker, copyright);
  1357. }
  1358. void CarlaEngine::oscSend_bridge_audio_count(const uint32_t ins, const uint32_t outs) const noexcept
  1359. {
  1360. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1361. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1362. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1363. carla_debug("CarlaEngine::oscSend_bridge_audio_count(%i, %i)", ins, outs);
  1364. char targetPath[std::strlen(pData->oscData->path)+20];
  1365. std::strcpy(targetPath, pData->oscData->path);
  1366. std::strcat(targetPath, "/bridge_audio_count");
  1367. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1368. }
  1369. void CarlaEngine::oscSend_bridge_midi_count(const uint32_t ins, const uint32_t outs) const noexcept
  1370. {
  1371. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1372. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1373. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1374. carla_debug("CarlaEngine::oscSend_bridge_midi_count(%i, %i)", ins, outs);
  1375. char targetPath[std::strlen(pData->oscData->path)+19];
  1376. std::strcpy(targetPath, pData->oscData->path);
  1377. std::strcat(targetPath, "/bridge_midi_count");
  1378. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1379. }
  1380. void CarlaEngine::oscSend_bridge_parameter_count(const uint32_t ins, const uint32_t outs) const noexcept
  1381. {
  1382. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1383. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1384. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1385. carla_debug("CarlaEngine::oscSend_bridge_parameter_count(%i, %i)", ins, outs);
  1386. char targetPath[std::strlen(pData->oscData->path)+24];
  1387. std::strcpy(targetPath, pData->oscData->path);
  1388. std::strcat(targetPath, "/bridge_parameter_count");
  1389. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1390. }
  1391. void CarlaEngine::oscSend_bridge_program_count(const uint32_t count) const noexcept
  1392. {
  1393. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1394. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1395. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1396. carla_debug("CarlaEngine::oscSend_bridge_program_count(%i)", count);
  1397. char targetPath[std::strlen(pData->oscData->path)+23];
  1398. std::strcpy(targetPath, pData->oscData->path);
  1399. std::strcat(targetPath, "/bridge_program_count");
  1400. try_lo_send(pData->oscData->target, targetPath, "i", static_cast<int32_t>(count));
  1401. }
  1402. void CarlaEngine::oscSend_bridge_midi_program_count(const uint32_t count) const noexcept
  1403. {
  1404. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1405. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1406. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1407. carla_debug("CarlaEngine::oscSend_bridge_midi_program_count(%i)", count);
  1408. char targetPath[std::strlen(pData->oscData->path)+27];
  1409. std::strcpy(targetPath, pData->oscData->path);
  1410. std::strcat(targetPath, "/bridge_midi_program_count");
  1411. try_lo_send(pData->oscData->target, targetPath, "i", static_cast<int32_t>(count));
  1412. }
  1413. void CarlaEngine::oscSend_bridge_parameter_data(const uint32_t index, const int32_t rindex, const ParameterType type, const uint hints, const char* const name, const char* const unit) const noexcept
  1414. {
  1415. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1416. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1417. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1418. CARLA_SAFE_ASSERT_RETURN(name != nullptr,);
  1419. CARLA_SAFE_ASSERT_RETURN(unit != nullptr,);
  1420. carla_debug("CarlaEngine::oscSend_bridge_parameter_data(%i, %i, %i:%s, %X, \"%s\", \"%s\")", index, rindex, type, ParameterType2Str(type), hints, name, unit);
  1421. char targetPath[std::strlen(pData->oscData->path)+24];
  1422. std::strcpy(targetPath, pData->oscData->path);
  1423. std::strcat(targetPath, "/bridge_parameter_data");
  1424. try_lo_send(pData->oscData->target, targetPath, "iiiiss", static_cast<int32_t>(index), static_cast<int32_t>(rindex), static_cast<int32_t>(type), static_cast<int32_t>(hints), name, unit);
  1425. }
  1426. void CarlaEngine::oscSend_bridge_parameter_ranges1(const uint32_t index, const float def, const float min, const float max) const noexcept
  1427. {
  1428. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1429. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1430. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1431. carla_debug("CarlaEngine::oscSend_bridge_parameter_ranges(%i, %f, %f, %f)", index, def, min, max);
  1432. char targetPath[std::strlen(pData->oscData->path)+26];
  1433. std::strcpy(targetPath, pData->oscData->path);
  1434. std::strcat(targetPath, "/bridge_parameter_ranges1");
  1435. try_lo_send(pData->oscData->target, targetPath, "ifff", static_cast<int32_t>(index), def, min, max);
  1436. }
  1437. void CarlaEngine::oscSend_bridge_parameter_ranges2(const uint32_t index, const float step, const float stepSmall, const float stepLarge) const noexcept
  1438. {
  1439. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1440. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1441. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1442. carla_debug("CarlaEngine::oscSend_bridge_parameter_ranges(%i, %f, %f, %f)", index, step, stepSmall, stepLarge);
  1443. char targetPath[std::strlen(pData->oscData->path)+26];
  1444. std::strcpy(targetPath, pData->oscData->path);
  1445. std::strcat(targetPath, "/bridge_parameter_ranges2");
  1446. try_lo_send(pData->oscData->target, targetPath, "ifff", static_cast<int32_t>(index), step, stepSmall, stepLarge);
  1447. }
  1448. void CarlaEngine::oscSend_bridge_parameter_midi_cc(const uint32_t index, const int16_t cc) const noexcept
  1449. {
  1450. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1451. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1452. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1453. carla_debug("CarlaEngine::oscSend_bridge_parameter_midi_cc(%i, %i)", index, cc);
  1454. char targetPath[std::strlen(pData->oscData->path)+26];
  1455. std::strcpy(targetPath, pData->oscData->path);
  1456. std::strcat(targetPath, "/bridge_parameter_midi_cc");
  1457. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(index), static_cast<int32_t>(cc));
  1458. }
  1459. void CarlaEngine::oscSend_bridge_parameter_midi_channel(const uint32_t index, const uint8_t channel) const noexcept
  1460. {
  1461. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1462. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1463. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1464. carla_debug("CarlaEngine::oscSend_bridge_parameter_midi_channel(%i, %i)", index, channel);
  1465. char targetPath[std::strlen(pData->oscData->path)+31];
  1466. std::strcpy(targetPath, pData->oscData->path);
  1467. std::strcat(targetPath, "/bridge_parameter_midi_channel");
  1468. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(index), static_cast<int32_t>(channel));
  1469. }
  1470. void CarlaEngine::oscSend_bridge_parameter_value(const uint32_t index, const float value) const noexcept
  1471. {
  1472. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1473. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1474. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1475. carla_debug("CarlaEngine::oscSend_bridge_parameter_value(%i, %f)", index, value);
  1476. char targetPath[std::strlen(pData->oscData->path)+24];
  1477. std::strcpy(targetPath, pData->oscData->path);
  1478. std::strcat(targetPath, "/bridge_parameter_value");
  1479. try_lo_send(pData->oscData->target, targetPath, "if", static_cast<int32_t>(index), value);
  1480. }
  1481. void CarlaEngine::oscSend_bridge_default_value(const uint32_t index, const float value) const noexcept
  1482. {
  1483. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1484. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1485. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1486. carla_debug("CarlaEngine::oscSend_bridge_default_value(%i, %f)", index, value);
  1487. char targetPath[std::strlen(pData->oscData->path)+22];
  1488. std::strcpy(targetPath, pData->oscData->path);
  1489. std::strcat(targetPath, "/bridge_default_value");
  1490. try_lo_send(pData->oscData->target, targetPath, "if", static_cast<int32_t>(index), value);
  1491. }
  1492. void CarlaEngine::oscSend_bridge_current_program(const int32_t index) const noexcept
  1493. {
  1494. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1495. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1496. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1497. carla_debug("CarlaEngine::oscSend_bridge_current_program(%i)", index);
  1498. char targetPath[std::strlen(pData->oscData->path)+24];
  1499. std::strcpy(targetPath, pData->oscData->path);
  1500. std::strcat(targetPath, "/bridge_current_program");
  1501. try_lo_send(pData->oscData->target, targetPath, "i", index);
  1502. }
  1503. void CarlaEngine::oscSend_bridge_current_midi_program(const int32_t index) const noexcept
  1504. {
  1505. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1506. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1507. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1508. carla_debug("CarlaEngine::oscSend_bridge_current_midi_program(%i)", index);
  1509. char targetPath[std::strlen(pData->oscData->path)+30];
  1510. std::strcpy(targetPath, pData->oscData->path);
  1511. std::strcat(targetPath, "/bridge_current_midi_program");
  1512. try_lo_send(pData->oscData->target, targetPath, "i", index);
  1513. }
  1514. void CarlaEngine::oscSend_bridge_program_name(const uint32_t index, const char* const name) const noexcept
  1515. {
  1516. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1517. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1518. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1519. CARLA_SAFE_ASSERT_RETURN(name != nullptr,);
  1520. carla_debug("CarlaEngine::oscSend_bridge_program_name(%i, \"%s\")", index, name);
  1521. char targetPath[std::strlen(pData->oscData->path)+21];
  1522. std::strcpy(targetPath, pData->oscData->path);
  1523. std::strcat(targetPath, "/bridge_program_name");
  1524. try_lo_send(pData->oscData->target, targetPath, "is", static_cast<int32_t>(index), name);
  1525. }
  1526. void CarlaEngine::oscSend_bridge_midi_program_data(const uint32_t index, const uint32_t bank, const uint32_t program, const char* const name) const noexcept
  1527. {
  1528. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1529. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1530. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1531. CARLA_SAFE_ASSERT_RETURN(name != nullptr,);
  1532. carla_debug("CarlaEngine::oscSend_bridge_midi_program_data(%i, %i, %i, \"%s\")", index, bank, program, name);
  1533. char targetPath[std::strlen(pData->oscData->path)+26];
  1534. std::strcpy(targetPath, pData->oscData->path);
  1535. std::strcat(targetPath, "/bridge_midi_program_data");
  1536. try_lo_send(pData->oscData->target, targetPath, "iiis", static_cast<int32_t>(index), static_cast<int32_t>(bank), static_cast<int32_t>(program), name);
  1537. }
  1538. void CarlaEngine::oscSend_bridge_configure(const char* const key, const char* const value) const noexcept
  1539. {
  1540. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1541. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1542. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1543. CARLA_SAFE_ASSERT_RETURN(key != nullptr && key[0] != '\0',);
  1544. CARLA_SAFE_ASSERT_RETURN(value != nullptr,);
  1545. carla_debug("CarlaEngine::oscSend_bridge_configure(\"%s\", \"%s\")", key, value);
  1546. char targetPath[std::strlen(pData->oscData->path)+18];
  1547. std::strcpy(targetPath, pData->oscData->path);
  1548. std::strcat(targetPath, "/bridge_configure");
  1549. try_lo_send(pData->oscData->target, targetPath, "ss", key, value);
  1550. }
  1551. void CarlaEngine::oscSend_bridge_set_custom_data(const char* const type, const char* const key, const char* const value) const noexcept
  1552. {
  1553. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1554. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1555. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1556. CARLA_SAFE_ASSERT_RETURN(type != nullptr && type[0] != '\0',);
  1557. CARLA_SAFE_ASSERT_RETURN(key != nullptr && key[0] != '\0',);
  1558. CARLA_SAFE_ASSERT_RETURN(value != nullptr,);
  1559. carla_debug("CarlaEngine::oscSend_bridge_set_custom_data(\"%s\", \"%s\", \"%s\")", type, key, value);
  1560. char targetPath[std::strlen(pData->oscData->path)+24];
  1561. std::strcpy(targetPath, pData->oscData->path);
  1562. std::strcat(targetPath, "/bridge_set_custom_data");
  1563. try_lo_send(pData->oscData->target, targetPath, "sss", type, key, value);
  1564. }
  1565. void CarlaEngine::oscSend_bridge_set_chunk_data(const char* const chunkFile) const noexcept
  1566. {
  1567. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1568. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1569. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1570. CARLA_SAFE_ASSERT_RETURN(chunkFile != nullptr && chunkFile[0] != '\0',);
  1571. carla_debug("CarlaEngine::oscSend_bridge_set_chunk_data(\"%s\")", chunkFile);
  1572. char targetPath[std::strlen(pData->oscData->path)+23];
  1573. std::strcpy(targetPath, pData->oscData->path);
  1574. std::strcat(targetPath, "/bridge_set_chunk_data");
  1575. try_lo_send(pData->oscData->target, targetPath, "s", chunkFile);
  1576. }
  1577. void CarlaEngine::oscSend_bridge_pong() const noexcept
  1578. {
  1579. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1580. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1581. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1582. //carla_debug("CarlaEngine::oscSend_pong()");
  1583. char targetPath[std::strlen(pData->oscData->path)+13];
  1584. std::strcpy(targetPath, pData->oscData->path);
  1585. std::strcat(targetPath, "/bridge_pong");
  1586. try_lo_send(pData->oscData->target, targetPath, "");
  1587. }
  1588. #else
  1589. void CarlaEngine::oscSend_control_add_plugin_start(const uint pluginId, const char* const pluginName) const noexcept
  1590. {
  1591. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1592. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1593. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1594. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1595. CARLA_SAFE_ASSERT_RETURN(pluginName != nullptr && pluginName[0] != '\0',);
  1596. carla_debug("CarlaEngine::oscSend_control_add_plugin_start(%i, \"%s\")", pluginId, pluginName);
  1597. char targetPath[std::strlen(pData->oscData->path)+18];
  1598. std::strcpy(targetPath, pData->oscData->path);
  1599. std::strcat(targetPath, "/add_plugin_start");
  1600. try_lo_send(pData->oscData->target, targetPath, "is", static_cast<int32_t>(pluginId), pluginName);
  1601. }
  1602. void CarlaEngine::oscSend_control_add_plugin_end(const uint pluginId) const noexcept
  1603. {
  1604. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1605. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1606. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1607. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1608. carla_debug("CarlaEngine::oscSend_control_add_plugin_end(%i)", pluginId);
  1609. char targetPath[std::strlen(pData->oscData->path)+16];
  1610. std::strcpy(targetPath, pData->oscData->path);
  1611. std::strcat(targetPath, "/add_plugin_end");
  1612. try_lo_send(pData->oscData->target, targetPath, "i", static_cast<int32_t>(pluginId));
  1613. }
  1614. void CarlaEngine::oscSend_control_remove_plugin(const uint pluginId) const noexcept
  1615. {
  1616. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1617. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1618. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1619. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1620. carla_debug("CarlaEngine::oscSend_control_remove_plugin(%i)", pluginId);
  1621. char targetPath[std::strlen(pData->oscData->path)+15];
  1622. std::strcpy(targetPath, pData->oscData->path);
  1623. std::strcat(targetPath, "/remove_plugin");
  1624. try_lo_send(pData->oscData->target, targetPath, "i", static_cast<int32_t>(pluginId));
  1625. }
  1626. void CarlaEngine::oscSend_control_set_plugin_info1(const uint pluginId, const PluginType type, const PluginCategory category, const uint hints, const int64_t uniqueId) const noexcept
  1627. {
  1628. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1629. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1630. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1631. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1632. CARLA_SAFE_ASSERT_RETURN(type != PLUGIN_NONE,);
  1633. carla_debug("CarlaEngine::oscSend_control_set_plugin_data(%i, %i:%s, %i:%s, %X, " P_INT64 ")", pluginId, type, PluginType2Str(type), category, PluginCategory2Str(category), hints, uniqueId);
  1634. char targetPath[std::strlen(pData->oscData->path)+18];
  1635. std::strcpy(targetPath, pData->oscData->path);
  1636. std::strcat(targetPath, "/set_plugin_info1");
  1637. try_lo_send(pData->oscData->target, targetPath, "iiiih", static_cast<int32_t>(pluginId), static_cast<int32_t>(type), static_cast<int32_t>(category), static_cast<int32_t>(hints), static_cast<int64_t>(uniqueId));
  1638. }
  1639. void CarlaEngine::oscSend_control_set_plugin_info2(const uint pluginId, const char* const realName, const char* const label, const char* const maker, const char* const copyright) const noexcept
  1640. {
  1641. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1642. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1643. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1644. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1645. CARLA_SAFE_ASSERT_RETURN(realName != nullptr && realName[0] != '\0',);
  1646. CARLA_SAFE_ASSERT_RETURN(label != nullptr && label[0] != '\0',);
  1647. CARLA_SAFE_ASSERT_RETURN(maker != nullptr,);
  1648. CARLA_SAFE_ASSERT_RETURN(copyright != nullptr,);
  1649. carla_debug("CarlaEngine::oscSend_control_set_plugin_data(%i, \"%s\", \"%s\", \"%s\", \"%s\")", pluginId, realName, label, maker, copyright);
  1650. char targetPath[std::strlen(pData->oscData->path)+18];
  1651. std::strcpy(targetPath, pData->oscData->path);
  1652. std::strcat(targetPath, "/set_plugin_info2");
  1653. try_lo_send(pData->oscData->target, targetPath, "issss", static_cast<int32_t>(pluginId), realName, label, maker, copyright);
  1654. }
  1655. void CarlaEngine::oscSend_control_set_audio_count(const uint pluginId, const uint32_t ins, const uint32_t outs) const noexcept
  1656. {
  1657. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1658. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1659. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1660. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1661. carla_debug("CarlaEngine::oscSend_control_set_audio_count(%i, %i, %i)", pluginId, ins, outs);
  1662. char targetPath[std::strlen(pData->oscData->path)+18];
  1663. std::strcpy(targetPath, pData->oscData->path);
  1664. std::strcat(targetPath, "/set_audio_count");
  1665. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1666. }
  1667. void CarlaEngine::oscSend_control_set_midi_count(const uint pluginId, const uint32_t ins, const uint32_t outs) const noexcept
  1668. {
  1669. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1670. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1671. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1672. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1673. carla_debug("CarlaEngine::oscSend_control_set_midi_count(%i, %i, %i)", pluginId, ins, outs);
  1674. char targetPath[std::strlen(pData->oscData->path)+18];
  1675. std::strcpy(targetPath, pData->oscData->path);
  1676. std::strcat(targetPath, "/set_midi_count");
  1677. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1678. }
  1679. void CarlaEngine::oscSend_control_set_parameter_count(const uint pluginId, const uint32_t ins, const uint32_t outs) const noexcept
  1680. {
  1681. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1682. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1683. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1684. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1685. carla_debug("CarlaEngine::oscSend_control_set_parameter_count(%i, %i, %i)", pluginId, ins, outs);
  1686. char targetPath[std::strlen(pData->oscData->path)+18];
  1687. std::strcpy(targetPath, pData->oscData->path);
  1688. std::strcat(targetPath, "/set_parameter_count");
  1689. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(ins), static_cast<int32_t>(outs));
  1690. }
  1691. void CarlaEngine::oscSend_control_set_program_count(const uint pluginId, const uint32_t count) const noexcept
  1692. {
  1693. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1694. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1695. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1696. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1697. carla_debug("CarlaEngine::oscSend_control_set_program_count(%i, %i)", pluginId, count);
  1698. char targetPath[std::strlen(pData->oscData->path)+19];
  1699. std::strcpy(targetPath, pData->oscData->path);
  1700. std::strcat(targetPath, "/set_program_count");
  1701. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(pluginId), static_cast<int32_t>(count));
  1702. }
  1703. void CarlaEngine::oscSend_control_set_midi_program_count(const uint pluginId, const uint32_t count) const noexcept
  1704. {
  1705. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1706. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1707. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1708. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1709. carla_debug("CarlaEngine::oscSend_control_set_midi_program_count(%i, %i)", pluginId, count);
  1710. char targetPath[std::strlen(pData->oscData->path)+24];
  1711. std::strcpy(targetPath, pData->oscData->path);
  1712. std::strcat(targetPath, "/set_midi_program_count");
  1713. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(pluginId), static_cast<int32_t>(count));
  1714. }
  1715. void CarlaEngine::oscSend_control_set_parameter_data(const uint pluginId, const uint32_t index, const ParameterType type, const uint hints, const char* const name, const char* const unit) const noexcept
  1716. {
  1717. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1718. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1719. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1720. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1721. CARLA_SAFE_ASSERT_RETURN(name != nullptr && name[0] != '\0',);
  1722. CARLA_SAFE_ASSERT_RETURN(unit != nullptr,);
  1723. carla_debug("CarlaEngine::oscSend_control_set_parameter_data(%i, %i, %i:%s, %X, \"%s\", \"%s\")", pluginId, index, type, ParameterType2Str(type), hints, name, unit);
  1724. char targetPath[std::strlen(pData->oscData->path)+20];
  1725. std::strcpy(targetPath, pData->oscData->path);
  1726. std::strcat(targetPath, "/set_parameter_data");
  1727. try_lo_send(pData->oscData->target, targetPath, "iiiiss", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), static_cast<int32_t>(type), static_cast<int32_t>(hints), name, unit);
  1728. }
  1729. void CarlaEngine::oscSend_control_set_parameter_ranges1(const uint pluginId, const uint32_t index, const float def, const float min, const float max) const noexcept
  1730. {
  1731. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1732. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1733. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1734. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1735. CARLA_SAFE_ASSERT_RETURN(def <= min && def >= max,);
  1736. CARLA_SAFE_ASSERT_RETURN(min < max,);
  1737. carla_debug("CarlaEngine::oscSend_control_set_parameter_ranges1(%i, %i, %f, %f, %f)", pluginId, index, def, min, max, def);
  1738. char targetPath[std::strlen(pData->oscData->path)+23];
  1739. std::strcpy(targetPath, pData->oscData->path);
  1740. std::strcat(targetPath, "/set_parameter_ranges1");
  1741. try_lo_send(pData->oscData->target, targetPath, "iifff", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), def, min, max);
  1742. }
  1743. void CarlaEngine::oscSend_control_set_parameter_ranges2(const uint pluginId, const uint32_t index, const float step, const float stepSmall, const float stepLarge) const noexcept
  1744. {
  1745. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1746. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1747. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1748. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1749. CARLA_SAFE_ASSERT_RETURN(step <= stepSmall && step >= stepLarge,);
  1750. CARLA_SAFE_ASSERT_RETURN(stepSmall <= stepLarge,);
  1751. carla_debug("CarlaEngine::oscSend_control_set_parameter_ranges2(%i, %i, %f, %f, %f)", pluginId, index, step, stepSmall, stepLarge);
  1752. char targetPath[std::strlen(pData->oscData->path)+23];
  1753. std::strcpy(targetPath, pData->oscData->path);
  1754. std::strcat(targetPath, "/set_parameter_ranges");
  1755. try_lo_send(pData->oscData->target, targetPath, "iifff", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), step, stepSmall, stepLarge);
  1756. }
  1757. void CarlaEngine::oscSend_control_set_parameter_midi_cc(const uint pluginId, const uint32_t index, const int16_t cc) const noexcept
  1758. {
  1759. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1760. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1761. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1762. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1763. CARLA_SAFE_ASSERT_RETURN(cc <= 0x5F,);
  1764. carla_debug("CarlaEngine::oscSend_control_set_parameter_midi_cc(%i, %i, %i)", pluginId, index, cc);
  1765. char targetPath[std::strlen(pData->oscData->path)+23];
  1766. std::strcpy(targetPath, pData->oscData->path);
  1767. std::strcat(targetPath, "/set_parameter_midi_cc");
  1768. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), static_cast<int32_t>(cc));
  1769. }
  1770. void CarlaEngine::oscSend_control_set_parameter_midi_channel(const uint pluginId, const uint32_t index, const uint8_t channel) const noexcept
  1771. {
  1772. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1773. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1774. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1775. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1776. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS,);
  1777. carla_debug("CarlaEngine::oscSend_control_set_parameter_midi_channel(%i, %i, %i)", pluginId, index, channel);
  1778. char targetPath[std::strlen(pData->oscData->path)+28];
  1779. std::strcpy(targetPath, pData->oscData->path);
  1780. std::strcat(targetPath, "/set_parameter_midi_channel");
  1781. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), static_cast<int32_t>(channel));
  1782. }
  1783. void CarlaEngine::oscSend_control_set_parameter_value(const uint pluginId, const int32_t index, const float value) const noexcept
  1784. {
  1785. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1786. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1787. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1788. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1789. CARLA_SAFE_ASSERT_RETURN(index != PARAMETER_NULL,);
  1790. carla_debug("CarlaEngine::oscSend_control_set_parameter_value(%i, %i:%s, %f)", pluginId, index, (index < 0) ? InternalParameterIndex2Str(static_cast<InternalParameterIndex>(index)) : "(none)", value);
  1791. char targetPath[std::strlen(pData->oscData->path)+21];
  1792. std::strcpy(targetPath, pData->oscData->path);
  1793. std::strcat(targetPath, "/set_parameter_value");
  1794. try_lo_send(pData->oscData->target, targetPath, "iif", static_cast<int32_t>(pluginId), index, value);
  1795. }
  1796. void CarlaEngine::oscSend_control_set_default_value(const uint pluginId, const uint32_t index, const float value) const noexcept
  1797. {
  1798. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1799. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1800. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1801. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1802. carla_debug("CarlaEngine::oscSend_control_set_default_value(%i, %i, %f)", pluginId, index, value);
  1803. char targetPath[std::strlen(pData->oscData->path)+19];
  1804. std::strcpy(targetPath, pData->oscData->path);
  1805. std::strcat(targetPath, "/set_default_value");
  1806. try_lo_send(pData->oscData->target, targetPath, "iif", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), value);
  1807. }
  1808. void CarlaEngine::oscSend_control_set_current_program(const uint pluginId, const int32_t index) const noexcept
  1809. {
  1810. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1811. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1812. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1813. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1814. carla_debug("CarlaEngine::oscSend_control_set_current_program(%i, %i)", pluginId, index);
  1815. char targetPath[std::strlen(pData->oscData->path)+21];
  1816. std::strcpy(targetPath, pData->oscData->path);
  1817. std::strcat(targetPath, "/set_current_program");
  1818. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(pluginId), index);
  1819. }
  1820. void CarlaEngine::oscSend_control_set_current_midi_program(const uint pluginId, const int32_t index) const noexcept
  1821. {
  1822. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1823. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1824. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1825. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1826. carla_debug("CarlaEngine::oscSend_control_set_current_midi_program(%i, %i)", pluginId, index);
  1827. char targetPath[std::strlen(pData->oscData->path)+26];
  1828. std::strcpy(targetPath, pData->oscData->path);
  1829. std::strcat(targetPath, "/set_current_midi_program");
  1830. try_lo_send(pData->oscData->target, targetPath, "ii", static_cast<int32_t>(pluginId), index);
  1831. }
  1832. void CarlaEngine::oscSend_control_set_program_name(const uint pluginId, const uint32_t index, const char* const name) const noexcept
  1833. {
  1834. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1835. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1836. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1837. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1838. CARLA_SAFE_ASSERT_RETURN(name != nullptr,);
  1839. carla_debug("CarlaEngine::oscSend_control_set_program_name(%i, %i, \"%s\")", pluginId, index, name);
  1840. char targetPath[std::strlen(pData->oscData->path)+18];
  1841. std::strcpy(targetPath, pData->oscData->path);
  1842. std::strcat(targetPath, "/set_program_name");
  1843. try_lo_send(pData->oscData->target, targetPath, "iis", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), name);
  1844. }
  1845. void CarlaEngine::oscSend_control_set_midi_program_data(const uint pluginId, const uint32_t index, const uint32_t bank, const uint32_t program, const char* const name) const noexcept
  1846. {
  1847. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1848. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1849. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1850. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1851. CARLA_SAFE_ASSERT_RETURN(name != nullptr,);
  1852. carla_debug("CarlaEngine::oscSend_control_set_midi_program_data(%i, %i, %i, %i, \"%s\")", pluginId, index, bank, program, name);
  1853. char targetPath[std::strlen(pData->oscData->path)+23];
  1854. std::strcpy(targetPath, pData->oscData->path);
  1855. std::strcat(targetPath, "/set_midi_program_data");
  1856. try_lo_send(pData->oscData->target, targetPath, "iiiis", static_cast<int32_t>(pluginId), static_cast<int32_t>(index), static_cast<int32_t>(bank), static_cast<int32_t>(program), name);
  1857. }
  1858. void CarlaEngine::oscSend_control_note_on(const uint pluginId, const uint8_t channel, const uint8_t note, const uint8_t velo) const noexcept
  1859. {
  1860. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1861. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1862. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1863. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1864. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS,);
  1865. CARLA_SAFE_ASSERT_RETURN(note < MAX_MIDI_NOTE,);
  1866. CARLA_SAFE_ASSERT_RETURN(velo < MAX_MIDI_VALUE,);
  1867. carla_debug("CarlaEngine::oscSend_control_note_on(%i, %i, %i, %i)", pluginId, channel, note, velo);
  1868. char targetPath[std::strlen(pData->oscData->path)+9];
  1869. std::strcpy(targetPath, pData->oscData->path);
  1870. std::strcat(targetPath, "/note_on");
  1871. try_lo_send(pData->oscData->target, targetPath, "iiii", static_cast<int32_t>(pluginId), static_cast<int32_t>(channel), static_cast<int32_t>(note), static_cast<int32_t>(velo));
  1872. }
  1873. void CarlaEngine::oscSend_control_note_off(const uint pluginId, const uint8_t channel, const uint8_t note) const noexcept
  1874. {
  1875. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1876. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1877. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1878. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1879. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS,);
  1880. CARLA_SAFE_ASSERT_RETURN(note < MAX_MIDI_NOTE,);
  1881. carla_debug("CarlaEngine::oscSend_control_note_off(%i, %i, %i)", pluginId, channel, note);
  1882. char targetPath[std::strlen(pData->oscData->path)+10];
  1883. std::strcpy(targetPath, pData->oscData->path);
  1884. std::strcat(targetPath, "/note_off");
  1885. try_lo_send(pData->oscData->target, targetPath, "iii", static_cast<int32_t>(pluginId), static_cast<int32_t>(channel), static_cast<int32_t>(note));
  1886. }
  1887. void CarlaEngine::oscSend_control_set_peaks(const uint pluginId) const noexcept
  1888. {
  1889. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1890. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1891. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1892. CARLA_SAFE_ASSERT_RETURN(pluginId < pData->curPluginCount,);
  1893. // TODO - try and see if we can get peaks[4] ref
  1894. const EnginePluginData& epData(pData->plugins[pluginId]);
  1895. char targetPath[std::strlen(pData->oscData->path)+11];
  1896. std::strcpy(targetPath, pData->oscData->path);
  1897. std::strcat(targetPath, "/set_peaks");
  1898. try_lo_send(pData->oscData->target, targetPath, "iffff", static_cast<int32_t>(pluginId), epData.insPeak[0], epData.insPeak[1], epData.outsPeak[0], epData.outsPeak[1]);
  1899. }
  1900. void CarlaEngine::oscSend_control_exit() const noexcept
  1901. {
  1902. CARLA_SAFE_ASSERT_RETURN(pData->oscData != nullptr,);
  1903. CARLA_SAFE_ASSERT_RETURN(pData->oscData->path != nullptr && pData->oscData->path[0] != '\0',);
  1904. CARLA_SAFE_ASSERT_RETURN(pData->oscData->target != nullptr,);
  1905. carla_debug("CarlaEngine::oscSend_control_exit()");
  1906. char targetPath[std::strlen(pData->oscData->path)+6];
  1907. std::strcpy(targetPath, pData->oscData->path);
  1908. std::strcat(targetPath, "/exit");
  1909. try_lo_send(pData->oscData->target, targetPath, "");
  1910. }
  1911. #endif
  1912. // -----------------------------------------------------------------------
  1913. CARLA_BACKEND_END_NAMESPACE