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.

2389 lines
77KB

  1. /*
  2. * Carla Standalone
  3. * Copyright (C) 2011-2013 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License as
  7. * published by the Free Software Foundation; either version 2 of
  8. * the License, or any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * For a full copy of the GNU General Public License see the doc/GPL.txt file.
  16. */
  17. // TODO:
  18. // Check carla_stderr2("Engine is not running"); <= prepend func name and args
  19. #include "CarlaHost.h"
  20. #include "CarlaMIDI.h"
  21. #include "CarlaNative.h"
  22. #include "CarlaEngine.hpp"
  23. #include "CarlaPlugin.hpp"
  24. #include "CarlaBackendUtils.hpp"
  25. #include "CarlaThread.hpp"
  26. #include <QtCore/QByteArray>
  27. #ifdef BUILD_BRIDGE
  28. # undef HAVE_JUCE
  29. #endif
  30. #ifdef HAVE_JUCE
  31. # include "juce_gui_basics.h"
  32. using juce::initialiseJuce_GUI;
  33. using juce::shutdownJuce_GUI;
  34. using juce::MessageManager;
  35. #endif
  36. namespace CB = CarlaBackend;
  37. using CB::EngineOptions;
  38. #if defined(HAVE_JUCE) && defined(CARLA_OS_LINUX)
  39. using juce::Thread;
  40. // -----------------------------------------------------------------------
  41. // Juce Message Thread
  42. class JuceMessageThread : public Thread
  43. {
  44. public:
  45. JuceMessageThread()
  46. : Thread("JuceMessageThread"),
  47. fInitialised(false)
  48. {
  49. }
  50. ~JuceMessageThread() override
  51. {
  52. stop();
  53. }
  54. void start()
  55. {
  56. CARLA_SAFE_ASSERT_RETURN(! fInitialised,);
  57. fInitialised = false;
  58. startThread();
  59. while (! fInitialised)
  60. Thread::sleep(1);
  61. }
  62. void stop()
  63. {
  64. if (! fInitialised)
  65. return;
  66. stopThread(-1);
  67. }
  68. protected:
  69. void run() override
  70. {
  71. fInitialised = true;
  72. if (MessageManager* const msgMgr = MessageManager::getInstance())
  73. {
  74. msgMgr->setCurrentThreadAsMessageThread();
  75. for (; (! threadShouldExit()) && msgMgr->runDispatchLoopUntil(250);) {}
  76. }
  77. fInitialised = false;
  78. }
  79. private:
  80. volatile bool fInitialised;
  81. };
  82. #endif
  83. // -------------------------------------------------------------------------------------------------------------------
  84. // Single, standalone engine
  85. struct CarlaBackendStandalone {
  86. CarlaEngine* engine;
  87. EngineCallbackFunc engineCallback;
  88. void* engineCallbackPtr;
  89. EngineOptions engineOptions;
  90. FileCallbackFunc fileCallback;
  91. void* fileCallbackPtr;
  92. CarlaString lastError;
  93. #if defined(HAVE_JUCE) && defined(CARLA_OS_LINUX)
  94. JuceMessageThread juceMsgThread;
  95. #endif
  96. CarlaBackendStandalone()
  97. : engine(nullptr),
  98. engineCallback(nullptr),
  99. engineCallbackPtr(nullptr),
  100. fileCallback(nullptr),
  101. fileCallbackPtr(nullptr)
  102. {
  103. #ifdef BUILD_BRIDGE
  104. engineOptions.processMode = CB::ENGINE_PROCESS_MODE_BRIDGE;
  105. engineOptions.transportMode = CB::ENGINE_TRANSPORT_MODE_BRIDGE;
  106. engineOptions.forceStereo = false;
  107. engineOptions.preferPluginBridges = false;
  108. engineOptions.preferUiBridges = false;
  109. #endif
  110. }
  111. ~CarlaBackendStandalone()
  112. {
  113. CARLA_ASSERT(engine == nullptr);
  114. #ifdef HAVE_JUCE
  115. CARLA_ASSERT(MessageManager::getInstanceWithoutCreating() == nullptr);
  116. #endif
  117. }
  118. #ifdef HAVE_JUCE
  119. void init()
  120. {
  121. JUCE_AUTORELEASEPOOL
  122. {
  123. initialiseJuce_GUI();
  124. # ifdef CARLA_OS_LINUX
  125. juceMsgThread.start();
  126. # else
  127. if (MessageManager* const msgMgr = MessageManager::getInstance())
  128. msgMgr->setCurrentThreadAsMessageThread();
  129. # endif
  130. }
  131. }
  132. void close()
  133. {
  134. JUCE_AUTORELEASEPOOL
  135. {
  136. # ifdef CARLA_OS_LINUX
  137. juceMsgThread.stop();
  138. # else
  139. MessageManager::deleteInstance();
  140. # endif
  141. shutdownJuce_GUI();
  142. }
  143. }
  144. # ifndef CARLA_OS_LINUX
  145. void idle()
  146. {
  147. JUCE_AUTORELEASEPOOL
  148. {
  149. if (MessageManager* const msgMgr = MessageManager::getInstanceWithoutCreating())
  150. msgMgr->runDispatchLoopUntil(5);
  151. }
  152. }
  153. # endif
  154. #endif
  155. CARLA_DECLARE_NON_COPY_STRUCT(CarlaBackendStandalone)
  156. };
  157. static CarlaBackendStandalone gStandalone;
  158. // -------------------------------------------------------------------------------------------------------------------
  159. // Always return a valid string ptr
  160. static const char* const gNullCharPtr = "";
  161. static void checkStringPtr(const char*& charPtr)
  162. {
  163. if (charPtr == nullptr)
  164. charPtr = gNullCharPtr;
  165. }
  166. // -------------------------------------------------------------------------------------------------------------------
  167. // Constructors
  168. _CarlaPluginInfo::_CarlaPluginInfo() noexcept
  169. : type(CB::PLUGIN_NONE),
  170. category(CB::PLUGIN_CATEGORY_NONE),
  171. hints(0x0),
  172. optionsAvailable(0x0),
  173. optionsEnabled(0x0),
  174. filename(gNullCharPtr),
  175. name(gNullCharPtr),
  176. label(gNullCharPtr),
  177. maker(gNullCharPtr),
  178. copyright(gNullCharPtr),
  179. iconName(gNullCharPtr),
  180. uniqueId(0) {}
  181. _CarlaPluginInfo::~_CarlaPluginInfo() noexcept
  182. {
  183. if (label != gNullCharPtr)
  184. delete[] label;
  185. if (maker != gNullCharPtr)
  186. delete[] maker;
  187. if (copyright != gNullCharPtr)
  188. delete[] copyright;
  189. }
  190. _CarlaNativePluginInfo::_CarlaNativePluginInfo() noexcept
  191. : category(CB::PLUGIN_CATEGORY_NONE),
  192. hints(0x0),
  193. audioIns(0),
  194. audioOuts(0),
  195. midiIns(0),
  196. midiOuts(0),
  197. parameterIns(0),
  198. parameterOuts(0),
  199. name(gNullCharPtr),
  200. label(gNullCharPtr),
  201. maker(gNullCharPtr),
  202. copyright(gNullCharPtr) {}
  203. _CarlaParameterInfo::_CarlaParameterInfo() noexcept
  204. : name(gNullCharPtr),
  205. symbol(gNullCharPtr),
  206. unit(gNullCharPtr),
  207. scalePointCount(0) {}
  208. _CarlaParameterInfo::~_CarlaParameterInfo() noexcept
  209. {
  210. if (name != gNullCharPtr)
  211. delete[] name;
  212. if (symbol != gNullCharPtr)
  213. delete[] symbol;
  214. if (unit != gNullCharPtr)
  215. delete[] unit;
  216. }
  217. _CarlaScalePointInfo::_CarlaScalePointInfo() noexcept
  218. : value(0.0f),
  219. label(gNullCharPtr) {}
  220. _CarlaScalePointInfo::~_CarlaScalePointInfo() noexcept
  221. {
  222. if (label != gNullCharPtr)
  223. delete[] label;
  224. }
  225. _CarlaTransportInfo::_CarlaTransportInfo() noexcept
  226. : playing(false),
  227. frame(0),
  228. bar(0),
  229. beat(0),
  230. tick(0),
  231. bpm(0.0) {}
  232. // -------------------------------------------------------------------------------------------------------------------
  233. // API
  234. const char* carla_get_complete_license_text()
  235. {
  236. carla_debug("carla_get_complete_license_text()");
  237. static CarlaString retText;
  238. if (retText.isEmpty())
  239. {
  240. CarlaString text1, text2, text3, text4, text5;
  241. text1 += "<p>This current Carla build is using the following features and 3rd-party code:</p>";
  242. text1 += "<ul>";
  243. // Plugin formats
  244. #ifdef WANT_LADSPA
  245. text2 += "<li>LADSPA plugin support, http://www.ladspa.org/</li>";
  246. #endif
  247. #ifdef WANT_DSSI
  248. text2 += "<li>DSSI plugin support, http://dssi.sourceforge.net/</li>";
  249. #endif
  250. #ifdef WANT_LV2
  251. text2 += "<li>LV2 plugin support, http://lv2plug.in/</li>";
  252. #endif
  253. #ifdef WANT_VST
  254. # ifdef VESTIGE_HEADER
  255. text2 += "<li>VST plugin support, using VeSTige header by Javier Serrano Polo</li>";
  256. # else
  257. text2 += "<li>VST plugin support, using official VST SDK 2.4 (trademark of Steinberg Media Technologies GmbH)</li>";
  258. # endif
  259. #endif
  260. #ifdef WANT_AU
  261. text2 += "<li>AU plugin support</li>"; // FIXME
  262. #endif
  263. // Files
  264. #ifdef WANT_CSOUND
  265. text2 += "<li>CSound library for csd support</li>"; // FIXME
  266. #endif
  267. // Sample kit libraries
  268. #ifdef WANT_FLUIDSYNTH
  269. text2 += "<li>FluidSynth library for SF2 support, http://www.fluidsynth.org/</li>";
  270. #endif
  271. #ifdef WANT_LINUXSAMPLER
  272. text2 += "<li>LinuxSampler library for GIG and SFZ support*, http://www.linuxsampler.org/</li>";
  273. #endif
  274. #ifdef WANT_NATIVE
  275. // Internal plugins
  276. # ifdef HAVE_OPENGL
  277. text3 += "<li>DISTRHO Mini-Series plugin code, based on LOSER-dev suite by Michael Gruhn</li>";
  278. # endif
  279. text3 += "<li>NekoFilter plugin code, based on lv2fil by Nedko Arnaudov and Fons Adriaensen</li>";
  280. //text1 += "<li>SunVox library file support, http://www.warmplace.ru/soft/sunvox/</li>"; // unfinished
  281. # ifdef WANT_AUDIOFILE
  282. text3 += "<li>AudioDecoder library for Audio file support, by Robin Gareus</li>";
  283. # endif
  284. # ifdef WANT_MIDIFILE
  285. text3 += "<li>LibSMF library for MIDI file support, http://libsmf.sourceforge.net/</li>";
  286. # endif
  287. # ifdef WANT_ZYNADDSUBFX
  288. text3 += "<li>ZynAddSubFX plugin code, http://zynaddsubfx.sf.net/</li>";
  289. # ifdef WANT_ZYNADDSUBFX_UI
  290. text3 += "<li>ZynAddSubFX UI using NTK, http://non.tuxfamily.org/wiki/NTK</li>";
  291. # endif
  292. # endif
  293. #endif
  294. // misc libs
  295. text4 += "<li>liblo library for OSC support, http://liblo.sourceforge.net/</li>";
  296. #ifdef WANT_LV2
  297. text4 += "<li>serd, sord, sratom and lilv libraries for LV2 discovery, http://drobilla.net/software/lilv/</li>";
  298. #endif
  299. text4 += "<li>RtAudio+RtMidi libraries for extra Audio and MIDI support, http://www.music.mcgill.ca/~gary/rtaudio/</li>";
  300. // end
  301. text4 += "</ul>";
  302. // code snippets
  303. text5 += "<p>Additionally, Carla uses code snippets from the following projects:</p>";
  304. text5 += "<ul>";
  305. text5 += "<li>Pointer and data leak utils from JUCE, http://www.rawmaterialsoftware.com/juce.php</li>";
  306. text5 += "<li>Shared memory utils from dssi-vst, http://www.breakfastquay.com/dssi-vst/</li>";
  307. text5 += "<li>Real-time memory pool, by Nedko Arnaudov</li>";
  308. text5 += "</ul>";
  309. #ifdef WANT_LINUXSAMPLER
  310. // LinuxSampler GPL exception
  311. text5 += "<p>(*) Using LinuxSampler code in commercial hardware or software products is not allowed without prior written authorization by the authors.</p>";
  312. #endif
  313. retText = text1 + text2 + text3 + text4 + text5;
  314. }
  315. return retText;
  316. }
  317. const char* carla_get_supported_file_extensions()
  318. {
  319. carla_debug("carla_get_supported_file_extensions()");
  320. static CarlaString retText;
  321. if (retText.isEmpty())
  322. {
  323. // Base types
  324. retText += "*.carxp;*.carxs";
  325. // CSound
  326. #ifdef WANT_CSOUND
  327. retText += ";*.csd";
  328. #endif
  329. // Sample kits
  330. #ifdef WANT_FLUIDSYNTH
  331. retText += ";*.sf2";
  332. #endif
  333. #ifdef WANT_LINUXSAMPLER
  334. retText += ";*.gig;*.sfz";
  335. #endif
  336. // Files provided by internal plugins
  337. #ifdef WANT_AUDIOFILE
  338. retText += ";*.aiff;*.flac;*.oga;*.ogg;*.w64;*.wav";
  339. # ifdef HAVE_FFMPEG
  340. retText += ";*.3g2;*.3gp;*.aac;*.ac3;*.amr;*.ape;*.mp2;*.mp3;*.mpc;*.wma";
  341. # endif
  342. #endif
  343. #ifdef WANT_MIDIFILE
  344. retText += ";*.mid;*.midi";
  345. #endif
  346. // Plugin presets
  347. #ifdef WANT_ZYNADDSUBFX
  348. retText += ";*.xmz;*.xiz";
  349. #endif
  350. }
  351. return retText;
  352. }
  353. // -------------------------------------------------------------------------------------------------------------------
  354. unsigned int carla_get_engine_driver_count()
  355. {
  356. carla_debug("carla_get_engine_driver_count()");
  357. return CarlaEngine::getDriverCount();
  358. }
  359. const char* carla_get_engine_driver_name(unsigned int index)
  360. {
  361. carla_debug("carla_get_engine_driver_name(%i)", index);
  362. return CarlaEngine::getDriverName(index);
  363. }
  364. const char* const* carla_get_engine_driver_device_names(unsigned int index)
  365. {
  366. carla_debug("carla_get_engine_driver_device_names(%i)", index);
  367. return CarlaEngine::getDriverDeviceNames(index);
  368. }
  369. const EngineDriverDeviceInfo* carla_get_engine_driver_device_info(unsigned int index, const char* name)
  370. {
  371. CARLA_SAFE_ASSERT_RETURN(name != nullptr && name[0] != '\0', nullptr);
  372. carla_debug("carla_get_engine_driver_device_info(%i, \"%s\")", index, name);
  373. if (const EngineDriverDeviceInfo* const ret = CarlaEngine::getDriverDeviceInfo(index, name))
  374. {
  375. static EngineDriverDeviceInfo devInfo;
  376. static const uint32_t nullBufferSizes[] = { 0 };
  377. static const double nullSampleRates[] = { 0.0 };
  378. devInfo.hints = ret->hints;
  379. devInfo.bufferSizes = (ret->bufferSizes != nullptr) ? ret->bufferSizes : nullBufferSizes;
  380. devInfo.sampleRates = (ret->sampleRates != nullptr) ? ret->sampleRates : nullSampleRates;
  381. return &devInfo;
  382. }
  383. return nullptr;
  384. }
  385. // -------------------------------------------------------------------------------------------------------------------
  386. unsigned int carla_get_internal_plugin_count()
  387. {
  388. carla_debug("carla_get_internal_plugin_count()");
  389. #ifdef WANT_NATIVE
  390. return static_cast<unsigned int>(CarlaPlugin::getNativePluginCount());
  391. #else
  392. return 0;
  393. #endif
  394. }
  395. const CarlaNativePluginInfo* carla_get_internal_plugin_info(unsigned int index)
  396. {
  397. carla_debug("carla_get_internal_plugin_info(%i)", index);
  398. #ifdef WANT_NATIVE
  399. static CarlaNativePluginInfo info;
  400. const NativePluginDescriptor* const nativePlugin(CarlaPlugin::getNativePluginDescriptor(index));
  401. // as internal plugin, this must never fail
  402. CARLA_SAFE_ASSERT_RETURN(nativePlugin != nullptr, nullptr);
  403. info.category = static_cast<CB::PluginCategory>(nativePlugin->category);
  404. info.hints = 0x0;
  405. if (nativePlugin->hints & ::PLUGIN_IS_RTSAFE)
  406. info.hints |= CB::PLUGIN_IS_RTSAFE;
  407. if (nativePlugin->hints & ::PLUGIN_IS_SYNTH)
  408. info.hints |= CB::PLUGIN_IS_SYNTH;
  409. if (nativePlugin->hints & ::PLUGIN_HAS_UI)
  410. info.hints |= CB::PLUGIN_HAS_CUSTOM_UI;
  411. if (nativePlugin->hints & ::PLUGIN_NEEDS_FIXED_BUFFERS)
  412. info.hints |= CB::PLUGIN_NEEDS_FIXED_BUFFERS;
  413. if (nativePlugin->hints & ::PLUGIN_NEEDS_SINGLE_THREAD)
  414. info.hints |= CB::PLUGIN_NEEDS_SINGLE_THREAD;
  415. info.audioIns = nativePlugin->audioIns;
  416. info.audioOuts = nativePlugin->audioOuts;
  417. info.midiIns = nativePlugin->midiIns;
  418. info.midiOuts = nativePlugin->midiOuts;
  419. info.parameterIns = nativePlugin->paramIns;
  420. info.parameterOuts = nativePlugin->paramOuts;
  421. info.name = nativePlugin->name;
  422. info.label = nativePlugin->label;
  423. info.maker = nativePlugin->maker;
  424. info.copyright = nativePlugin->copyright;
  425. checkStringPtr(info.name);
  426. checkStringPtr(info.label);
  427. checkStringPtr(info.maker);
  428. checkStringPtr(info.copyright);
  429. return &info;
  430. #else
  431. return nullptr;
  432. // unused
  433. (void)index;
  434. #endif
  435. }
  436. // -------------------------------------------------------------------------------------------------------------------
  437. const CarlaEngine* carla_get_engine()
  438. {
  439. carla_debug("carla_get_engine()");
  440. return gStandalone.engine;
  441. }
  442. // -------------------------------------------------------------------------------------------------------------------
  443. bool carla_engine_init(const char* driverName, const char* clientName)
  444. {
  445. CARLA_SAFE_ASSERT_RETURN(driverName != nullptr && driverName[0] != '\0', false);
  446. CARLA_SAFE_ASSERT_RETURN(clientName != nullptr && clientName[0] != '\0', false);
  447. carla_debug("carla_engine_init(\"%s\", \"%s\")", driverName, clientName);
  448. if (gStandalone.engine != nullptr)
  449. {
  450. carla_stderr2("Engine is already running");
  451. gStandalone.lastError = "Engine is already running";
  452. return false;
  453. }
  454. #ifdef CARLA_OS_WIN
  455. carla_setenv("WINEASIO_CLIENT_NAME", clientName);
  456. #endif
  457. // TODO: make this an option, put somewhere else
  458. if (std::getenv("WINE_RT") == nullptr)
  459. {
  460. carla_setenv("WINE_RT", "15");
  461. carla_setenv("WINE_SVR_RT", "10");
  462. }
  463. gStandalone.engine = CarlaEngine::newDriverByName(driverName);
  464. if (gStandalone.engine == nullptr)
  465. {
  466. carla_stderr2("The seleted audio driver is not available");
  467. gStandalone.lastError = "The seleted audio driver is not available";
  468. return false;
  469. }
  470. gStandalone.engine->setCallback(gStandalone.engineCallback, gStandalone.engineCallbackPtr);
  471. gStandalone.engine->setFileCallback(gStandalone.fileCallback, gStandalone.fileCallbackPtr);
  472. #ifdef BUILD_BRIDGE
  473. gStandalone.engine->setOption(CB::ENGINE_OPTION_PROCESS_MODE, CB::ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS, nullptr);
  474. gStandalone.engine->setOption(CB::ENGINE_OPTION_TRANSPORT_MODE, CB::ENGINE_TRANSPORT_MODE_JACK, nullptr);
  475. #else
  476. gStandalone.engine->setOption(CB::ENGINE_OPTION_PROCESS_MODE, static_cast<int>(gStandalone.engineOptions.processMode), nullptr);
  477. gStandalone.engine->setOption(CB::ENGINE_OPTION_TRANSPORT_MODE, static_cast<int>(gStandalone.engineOptions.transportMode), nullptr);
  478. #endif
  479. gStandalone.engine->setOption(CB::ENGINE_OPTION_FORCE_STEREO, gStandalone.engineOptions.forceStereo ? 1 : 0, nullptr);
  480. gStandalone.engine->setOption(CB::ENGINE_OPTION_PREFER_PLUGIN_BRIDGES, gStandalone.engineOptions.preferPluginBridges ? 1 : 0, nullptr);
  481. gStandalone.engine->setOption(CB::ENGINE_OPTION_PREFER_UI_BRIDGES, gStandalone.engineOptions.preferUiBridges ? 1 : 0, nullptr);
  482. gStandalone.engine->setOption(CB::ENGINE_OPTION_UIS_ALWAYS_ON_TOP, gStandalone.engineOptions.uisAlwaysOnTop ? 1 : 0, nullptr);
  483. gStandalone.engine->setOption(CB::ENGINE_OPTION_MAX_PARAMETERS, static_cast<int>(gStandalone.engineOptions.maxParameters), nullptr);
  484. gStandalone.engine->setOption(CB::ENGINE_OPTION_UI_BRIDGES_TIMEOUT, static_cast<int>(gStandalone.engineOptions.uiBridgesTimeout), nullptr);
  485. gStandalone.engine->setOption(CB::ENGINE_OPTION_AUDIO_NUM_PERIODS, static_cast<int>(gStandalone.engineOptions.audioNumPeriods), nullptr);
  486. gStandalone.engine->setOption(CB::ENGINE_OPTION_AUDIO_BUFFER_SIZE, static_cast<int>(gStandalone.engineOptions.audioBufferSize), nullptr);
  487. gStandalone.engine->setOption(CB::ENGINE_OPTION_AUDIO_SAMPLE_RATE, static_cast<int>(gStandalone.engineOptions.audioSampleRate), nullptr);
  488. if (gStandalone.engineOptions.audioDevice != nullptr)
  489. gStandalone.engine->setOption(CB::ENGINE_OPTION_AUDIO_DEVICE, 0, gStandalone.engineOptions.audioDevice);
  490. if (gStandalone.engineOptions.binaryDir != nullptr && gStandalone.engineOptions.binaryDir[0] != '\0')
  491. gStandalone.engine->setOption(CB::ENGINE_OPTION_PATH_BINARIES, 0, gStandalone.engineOptions.binaryDir);
  492. if (gStandalone.engineOptions.resourceDir != nullptr && gStandalone.engineOptions.resourceDir[0] != '\0')
  493. gStandalone.engine->setOption(CB::ENGINE_OPTION_PATH_RESOURCES, 0, gStandalone.engineOptions.resourceDir);
  494. if (gStandalone.engineOptions.frontendWinId != 0)
  495. {
  496. char strBuf[STR_MAX+1];
  497. std::sprintf(strBuf, P_UINTPTR, gStandalone.engineOptions.frontendWinId);
  498. gStandalone.engine->setOption(CB::ENGINE_OPTION_FRONTEND_WIN_ID, 0, strBuf);
  499. }
  500. else
  501. gStandalone.engine->setOption(CB::ENGINE_OPTION_FRONTEND_WIN_ID, 0, "0");
  502. if (gStandalone.engine->init(clientName))
  503. {
  504. gStandalone.lastError = "No error";
  505. #ifdef HAVE_JUCE
  506. gStandalone.init();
  507. #endif
  508. return true;
  509. }
  510. else
  511. {
  512. gStandalone.lastError = gStandalone.engine->getLastError();
  513. delete gStandalone.engine;
  514. gStandalone.engine = nullptr;
  515. return false;
  516. }
  517. }
  518. #ifdef BUILD_BRIDGE
  519. bool carla_engine_init_bridge(const char audioBaseName[6+1], const char controlBaseName[6+1], const char timeBaseName[6+1], const char* clientName)
  520. {
  521. CARLA_SAFE_ASSERT_RETURN(audioBaseName != nullptr && audioBaseName[0] != '\0', false);
  522. CARLA_SAFE_ASSERT_RETURN(controlBaseName != nullptr && controlBaseName[0] != '\0', false);
  523. CARLA_SAFE_ASSERT_RETURN(timeBaseName != nullptr && timeBaseName[0] != '\0', false);
  524. CARLA_SAFE_ASSERT_RETURN(clientName != nullptr && clientName[0] != '\0', false);
  525. carla_debug("carla_engine_init_bridge(\"%s\", \"%s\", \"%s\", \"%s\")", audioBaseName, controlBaseName, timeBaseName, clientName);
  526. if (gStandalone.engine != nullptr)
  527. {
  528. carla_stderr2("Engine is already running");
  529. gStandalone.lastError = "Engine is already running";
  530. return false;
  531. }
  532. gStandalone.engine = CarlaEngine::newBridge(audioBaseName, controlBaseName, timeBaseName);
  533. if (gStandalone.engine == nullptr)
  534. {
  535. carla_stderr2("The seleted audio driver is not available!");
  536. gStandalone.lastError = "The seleted audio driver is not available!";
  537. return false;
  538. }
  539. gStandalone.engine->setCallback(gStandalone.engineCallback, gStandalone.engineCallbackPtr);
  540. gStandalone.engine->setFileCallback(gStandalone.fileCallback, gStandalone.fileCallbackPtr);
  541. // forced options for bridge mode
  542. gStandalone.engine->setOption(CB::ENGINE_OPTION_PROCESS_MODE, CB::ENGINE_PROCESS_MODE_BRIDGE, nullptr);
  543. gStandalone.engine->setOption(CB::ENGINE_OPTION_TRANSPORT_MODE, CB::ENGINE_TRANSPORT_MODE_BRIDGE, nullptr);
  544. gStandalone.engine->setOption(CB::ENGINE_OPTION_FORCE_STEREO, false, nullptr);
  545. gStandalone.engine->setOption(CB::ENGINE_OPTION_PREFER_PLUGIN_BRIDGES, false, nullptr);
  546. gStandalone.engine->setOption(CB::ENGINE_OPTION_PREFER_UI_BRIDGES, false, nullptr);
  547. if (const char* const uisAlwaysOnTop = std::getenv("ENGINE_OPTION_UIS_ALWAYS_ON_TOP"))
  548. gStandalone.engine->setOption(CB::ENGINE_OPTION_UIS_ALWAYS_ON_TOP, (std::strcmp(uisAlwaysOnTop, "true") == 0) ? 1 : 0, nullptr);
  549. if (const char* const maxParameters = std::getenv("ENGINE_OPTION_MAX_PARAMETERS"))
  550. gStandalone.engine->setOption(CB::ENGINE_OPTION_MAX_PARAMETERS, std::atoi(maxParameters), nullptr);
  551. if (const char* const uiBridgesTimeout = std::getenv("ENGINE_OPTION_UI_BRIDGES_TIMEOUT"))
  552. gStandalone.engine->setOption(CB::ENGINE_OPTION_UI_BRIDGES_TIMEOUT, std::atoi(uiBridgesTimeout), nullptr);
  553. if (const char* const binaryDir = std::getenv("ENGINE_OPTION_PATH_BINARIES"))
  554. gStandalone.engine->setOption(CB::ENGINE_OPTION_PATH_BINARIES, 0, binaryDir);
  555. if (const char* const resourceDir = std::getenv("ENGINE_OPTION_PATH_RESOURCES"))
  556. gStandalone.engine->setOption(CB::ENGINE_OPTION_PATH_RESOURCES, 0, resourceDir);
  557. if (const char* const frontendWinId = std::getenv("ENGINE_OPTION_FRONTEND_WIN_ID"))
  558. gStandalone.engine->setOption(CB::ENGINE_OPTION_FRONTEND_WIN_ID, 0, frontendWinId);
  559. if (gStandalone.engine->init(clientName))
  560. {
  561. gStandalone.lastError = "No error";
  562. #ifdef HAVE_JUCE
  563. gStandalone.init();
  564. #endif
  565. return true;
  566. }
  567. else
  568. {
  569. gStandalone.lastError = gStandalone.engine->getLastError();
  570. delete gStandalone.engine;
  571. gStandalone.engine = nullptr;
  572. return false;
  573. }
  574. }
  575. #endif
  576. bool carla_engine_close()
  577. {
  578. carla_debug("carla_engine_close()");
  579. if (gStandalone.engine == nullptr)
  580. {
  581. carla_stderr2("Engine is not running");
  582. gStandalone.lastError = "Engine is not running";
  583. return false;
  584. }
  585. gStandalone.engine->setAboutToClose();
  586. gStandalone.engine->removeAllPlugins();
  587. const bool closed(gStandalone.engine->close());
  588. if (! closed)
  589. gStandalone.lastError = gStandalone.engine->getLastError();
  590. #ifdef HAVE_JUCE
  591. gStandalone.close();
  592. #endif
  593. delete gStandalone.engine;
  594. gStandalone.engine = nullptr;
  595. return closed;
  596. }
  597. void carla_engine_idle()
  598. {
  599. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  600. #if defined(HAVE_JUCE) && ! defined(CARLA_OS_LINUX)
  601. gStandalone.idle();
  602. #endif
  603. gStandalone.engine->idle();
  604. }
  605. bool carla_is_engine_running()
  606. {
  607. return (gStandalone.engine != nullptr && gStandalone.engine->isRunning());
  608. }
  609. void carla_set_engine_about_to_close()
  610. {
  611. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  612. carla_debug("carla_set_engine_about_to_close()");
  613. gStandalone.engine->setAboutToClose();
  614. }
  615. void carla_set_engine_callback(EngineCallbackFunc func, void* ptr)
  616. {
  617. carla_debug("carla_set_engine_callback(%p, %p)", func, ptr);
  618. gStandalone.engineCallback = func;
  619. gStandalone.engineCallbackPtr = ptr;
  620. if (gStandalone.engine != nullptr)
  621. gStandalone.engine->setCallback(func, ptr);
  622. //#ifdef WANT_LOGS
  623. // gLogThread.setCallback(func, ptr);
  624. //#endif
  625. }
  626. #ifndef BUILD_BRIDGE
  627. void carla_set_engine_option(EngineOption option, int value, const char* valueStr)
  628. {
  629. carla_debug("carla_set_engine_option(%i:%s, %i, \"%s\")", option, CB::EngineOption2Str(option), value, valueStr);
  630. switch (option)
  631. {
  632. case CB::ENGINE_OPTION_DEBUG:
  633. break;
  634. case CB::ENGINE_OPTION_PROCESS_MODE:
  635. CARLA_SAFE_ASSERT_RETURN(value >= CB::ENGINE_PROCESS_MODE_SINGLE_CLIENT && value < CB::ENGINE_PROCESS_MODE_BRIDGE,);
  636. gStandalone.engineOptions.processMode = static_cast<CB::EngineProcessMode>(value);
  637. break;
  638. case CB::ENGINE_OPTION_TRANSPORT_MODE:
  639. CARLA_SAFE_ASSERT_RETURN(value >= CB::ENGINE_TRANSPORT_MODE_INTERNAL && value < CB::ENGINE_TRANSPORT_MODE_BRIDGE,);
  640. gStandalone.engineOptions.transportMode = static_cast<CB::EngineTransportMode>(value);
  641. break;
  642. case CB::ENGINE_OPTION_FORCE_STEREO:
  643. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  644. gStandalone.engineOptions.forceStereo = (value != 0);
  645. break;
  646. case CB::ENGINE_OPTION_PREFER_PLUGIN_BRIDGES:
  647. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  648. gStandalone.engineOptions.preferPluginBridges = (value != 0);
  649. break;
  650. case CB::ENGINE_OPTION_PREFER_UI_BRIDGES:
  651. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  652. gStandalone.engineOptions.preferUiBridges = (value != 0);
  653. break;
  654. case CB::ENGINE_OPTION_UIS_ALWAYS_ON_TOP:
  655. CARLA_SAFE_ASSERT_RETURN(value == 0 || value == 1,);
  656. gStandalone.engineOptions.uisAlwaysOnTop = (value != 0);
  657. break;
  658. case CB::ENGINE_OPTION_MAX_PARAMETERS:
  659. CARLA_SAFE_ASSERT_RETURN(value >= 0,);
  660. gStandalone.engineOptions.maxParameters = static_cast<unsigned int>(value);
  661. break;
  662. case CB::ENGINE_OPTION_UI_BRIDGES_TIMEOUT:
  663. CARLA_SAFE_ASSERT_RETURN(value >= 0,);
  664. gStandalone.engineOptions.uiBridgesTimeout = static_cast<unsigned int>(value);
  665. break;
  666. case CB::ENGINE_OPTION_AUDIO_NUM_PERIODS:
  667. CARLA_SAFE_ASSERT_RETURN(value >= 2 && value <= 3,);
  668. gStandalone.engineOptions.audioNumPeriods = static_cast<unsigned int>(value);
  669. break;
  670. case CB::ENGINE_OPTION_AUDIO_BUFFER_SIZE:
  671. CARLA_SAFE_ASSERT_RETURN(value >= 8,);
  672. gStandalone.engineOptions.audioBufferSize = static_cast<unsigned int>(value);
  673. break;
  674. case CB::ENGINE_OPTION_AUDIO_SAMPLE_RATE:
  675. CARLA_SAFE_ASSERT_RETURN(value >= 22050,);
  676. gStandalone.engineOptions.audioSampleRate = static_cast<unsigned int>(value);
  677. break;
  678. case CB::ENGINE_OPTION_AUDIO_DEVICE:
  679. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr,);
  680. if (gStandalone.engineOptions.audioDevice != nullptr)
  681. delete[] gStandalone.engineOptions.audioDevice;
  682. gStandalone.engineOptions.audioDevice = carla_strdup(valueStr);
  683. break;
  684. case CB::ENGINE_OPTION_PATH_BINARIES:
  685. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  686. if (gStandalone.engineOptions.binaryDir != nullptr)
  687. delete[] gStandalone.engineOptions.binaryDir;
  688. gStandalone.engineOptions.binaryDir = carla_strdup(valueStr);
  689. break;
  690. case CB::ENGINE_OPTION_PATH_RESOURCES:
  691. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  692. if (gStandalone.engineOptions.resourceDir != nullptr)
  693. delete[] gStandalone.engineOptions.resourceDir;
  694. gStandalone.engineOptions.resourceDir = carla_strdup(valueStr);
  695. break;
  696. case CB::ENGINE_OPTION_FRONTEND_WIN_ID:
  697. CARLA_SAFE_ASSERT_RETURN(valueStr != nullptr && valueStr[0] != '\0',);
  698. const long winId(std::atol(valueStr));
  699. CARLA_SAFE_ASSERT_RETURN(winId >= 0,);
  700. gStandalone.engineOptions.frontendWinId = static_cast<uintptr_t>(winId);
  701. break;
  702. }
  703. if (gStandalone.engine != nullptr)
  704. gStandalone.engine->setOption(option, value, valueStr);
  705. }
  706. #endif
  707. void carla_set_file_callback(FileCallbackFunc func, void* ptr)
  708. {
  709. carla_debug("carla_set_file_callback(%p, %p)", func, ptr);
  710. gStandalone.fileCallback = func;
  711. gStandalone.fileCallbackPtr = ptr;
  712. if (gStandalone.engine != nullptr)
  713. gStandalone.engine->setFileCallback(func, ptr);
  714. }
  715. // -------------------------------------------------------------------------------------------------------------------
  716. bool carla_load_file(const char* filename)
  717. {
  718. CARLA_SAFE_ASSERT_RETURN(filename != nullptr && filename[0] != '\0', false);
  719. carla_debug("carla_load_file(\"%s\")", filename);
  720. if (gStandalone.engine != nullptr)
  721. return gStandalone.engine->loadFile(filename);
  722. carla_stderr2("Engine is not running");
  723. gStandalone.lastError = "Engine is not running";
  724. return false;
  725. }
  726. bool carla_load_project(const char* filename)
  727. {
  728. CARLA_SAFE_ASSERT_RETURN(filename != nullptr && filename[0] != '\0', false);
  729. carla_debug("carla_load_project(\"%s\")", filename);
  730. if (gStandalone.engine != nullptr)
  731. return gStandalone.engine->loadProject(filename);
  732. carla_stderr2("Engine is not running");
  733. gStandalone.lastError = "Engine is not running";
  734. return false;
  735. }
  736. bool carla_save_project(const char* filename)
  737. {
  738. CARLA_SAFE_ASSERT_RETURN(filename != nullptr && filename[0] != '\0', false);
  739. carla_debug("carla_save_project(\"%s\")", filename);
  740. if (gStandalone.engine != nullptr)
  741. return gStandalone.engine->saveProject(filename);
  742. carla_stderr2("Engine was never initiated");
  743. gStandalone.lastError = "Engine was never initiated";
  744. return false;
  745. }
  746. #ifndef BUILD_BRIDGE
  747. // -------------------------------------------------------------------------------------------------------------------
  748. bool carla_patchbay_connect(int portIdA, int portIdB)
  749. {
  750. CARLA_SAFE_ASSERT_RETURN(portIdA != portIdB, false);
  751. carla_debug("carla_patchbay_connect(%i, %i)", portIdA, portIdB);
  752. if (gStandalone.engine != nullptr)
  753. return gStandalone.engine->patchbayConnect(portIdA, portIdB);
  754. carla_stderr2("Engine is not running");
  755. gStandalone.lastError = "Engine is not running";
  756. return false;
  757. }
  758. bool carla_patchbay_disconnect(uint connectionId)
  759. {
  760. carla_debug("carla_patchbay_disconnect(%i)", connectionId);
  761. if (gStandalone.engine != nullptr)
  762. return gStandalone.engine->patchbayDisconnect(connectionId);
  763. carla_stderr2("Engine is not running");
  764. gStandalone.lastError = "Engine is not running";
  765. return false;
  766. }
  767. bool carla_patchbay_refresh()
  768. {
  769. carla_debug("carla_patchbay_refresh()");
  770. if (gStandalone.engine != nullptr)
  771. return gStandalone.engine->patchbayRefresh();
  772. carla_stderr2("Engine is not running");
  773. gStandalone.lastError = "Engine is not running";
  774. return false;
  775. }
  776. #endif
  777. // -------------------------------------------------------------------------------------------------------------------
  778. void carla_transport_play()
  779. {
  780. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(),);
  781. carla_debug("carla_transport_play()");
  782. gStandalone.engine->transportPlay();
  783. }
  784. void carla_transport_pause()
  785. {
  786. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(),);
  787. carla_debug("carla_transport_pause()");
  788. gStandalone.engine->transportPause();
  789. }
  790. void carla_transport_relocate(uint64_t frame)
  791. {
  792. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(),);
  793. carla_debug("carla_transport_relocate(%i)", frame);
  794. gStandalone.engine->transportRelocate(frame);
  795. }
  796. uint64_t carla_get_current_transport_frame()
  797. {
  798. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(), 0);
  799. const CB::EngineTimeInfo& timeInfo(gStandalone.engine->getTimeInfo());
  800. return timeInfo.frame;
  801. }
  802. const CarlaTransportInfo* carla_get_transport_info()
  803. {
  804. static CarlaTransportInfo retInfo;
  805. // reset
  806. retInfo.playing = false;
  807. retInfo.frame = 0;
  808. retInfo.bar = 0;
  809. retInfo.beat = 0;
  810. retInfo.tick = 0;
  811. retInfo.bpm = 0.0;
  812. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(), &retInfo);
  813. const CB::EngineTimeInfo& timeInfo(gStandalone.engine->getTimeInfo());
  814. retInfo.playing = timeInfo.playing;
  815. retInfo.frame = timeInfo.frame;
  816. if (timeInfo.valid & CB::EngineTimeInfo::kValidBBT)
  817. {
  818. retInfo.bar = timeInfo.bbt.bar;
  819. retInfo.beat = timeInfo.bbt.beat;
  820. retInfo.tick = timeInfo.bbt.tick;
  821. retInfo.bpm = timeInfo.bbt.beatsPerMinute;
  822. }
  823. return &retInfo;
  824. }
  825. // -------------------------------------------------------------------------------------------------------------------
  826. bool carla_add_plugin(BinaryType btype, PluginType ptype, const char* filename, const char* name, const char* label, int64_t uniqueId, const void* extraPtr)
  827. {
  828. CARLA_SAFE_ASSERT_RETURN(label != nullptr && label[0] != '\0', false);
  829. carla_debug("carla_add_plugin(%i:%s, %i:%s, \"%s\", \"%s\", \"%s\", " P_INT64 ", %p)", btype, CB::BinaryType2Str(btype), ptype, CB::PluginType2Str(ptype), filename, name, label, uniqueId, extraPtr);
  830. if (gStandalone.engine != nullptr)
  831. return gStandalone.engine->addPlugin(btype, ptype, filename, name, label, uniqueId, extraPtr);
  832. carla_stderr2("Engine is not running");
  833. gStandalone.lastError = "Engine is not running";
  834. return false;
  835. }
  836. bool carla_remove_plugin(uint pluginId)
  837. {
  838. carla_debug("carla_remove_plugin(%i)", pluginId);
  839. if (gStandalone.engine != nullptr)
  840. return gStandalone.engine->removePlugin(pluginId);
  841. carla_stderr2("Engine is not running");
  842. gStandalone.lastError = "Engine is not running";
  843. return false;
  844. }
  845. bool carla_remove_all_plugins()
  846. {
  847. carla_debug("carla_remove_all_plugins()");
  848. if (gStandalone.engine != nullptr)
  849. return gStandalone.engine->removeAllPlugins();
  850. carla_stderr2("Engine is not running");
  851. gStandalone.lastError = "Engine is not running";
  852. return false;
  853. }
  854. const char* carla_rename_plugin(uint pluginId, const char* newName)
  855. {
  856. CARLA_SAFE_ASSERT_RETURN(newName != nullptr && newName[0] != '\0', nullptr);
  857. carla_debug("carla_rename_plugin(%i, \"%s\")", pluginId, newName);
  858. if (gStandalone.engine != nullptr)
  859. return gStandalone.engine->renamePlugin(pluginId, newName);
  860. carla_stderr2("Engine is not running");
  861. gStandalone.lastError = "Engine is not running";
  862. return nullptr;
  863. }
  864. bool carla_clone_plugin(uint pluginId)
  865. {
  866. carla_debug("carla_clone_plugin(%i)", pluginId);
  867. if (gStandalone.engine != nullptr)
  868. return gStandalone.engine->clonePlugin(pluginId);
  869. carla_stderr2("Engine is not running");
  870. gStandalone.lastError = "Engine is not running";
  871. return false;
  872. }
  873. bool carla_replace_plugin(uint pluginId)
  874. {
  875. carla_debug("carla_replace_plugin(%i)", pluginId);
  876. if (gStandalone.engine != nullptr)
  877. return gStandalone.engine->replacePlugin(pluginId);
  878. carla_stderr2("Engine is not running");
  879. gStandalone.lastError = "Engine is not running";
  880. return false;
  881. }
  882. bool carla_switch_plugins(uint pluginIdA, uint pluginIdB)
  883. {
  884. CARLA_SAFE_ASSERT_RETURN(pluginIdA != pluginIdB, false);
  885. carla_debug("carla_switch_plugins(%i, %i)", pluginIdA, pluginIdB);
  886. if (gStandalone.engine != nullptr)
  887. return gStandalone.engine->switchPlugins(pluginIdA, pluginIdB);
  888. carla_stderr2("Engine is not running");
  889. gStandalone.lastError = "Engine is not running";
  890. return false;
  891. }
  892. // -------------------------------------------------------------------------------------------------------------------
  893. bool carla_load_plugin_state(uint pluginId, const char* filename)
  894. {
  895. CARLA_SAFE_ASSERT_RETURN(filename != nullptr && filename[0] != '\0', false);
  896. carla_debug("carla_load_plugin_state(%i, \"%s\")", pluginId, filename);
  897. if (gStandalone.engine == nullptr || ! gStandalone.engine->isRunning())
  898. {
  899. carla_stderr2("Engine is not running");
  900. gStandalone.lastError = "Engine is not running";
  901. return false;
  902. }
  903. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  904. return plugin->loadStateFromFile(filename);
  905. carla_stderr2("carla_load_plugin_state(%i, \"%s\") - could not find plugin", pluginId, filename);
  906. return false;
  907. }
  908. bool carla_save_plugin_state(uint pluginId, const char* filename)
  909. {
  910. CARLA_SAFE_ASSERT_RETURN(filename != nullptr && filename[0] != '\0', false);
  911. carla_debug("carla_save_plugin_state(%i, \"%s\")", pluginId, filename);
  912. if (gStandalone.engine == nullptr)
  913. {
  914. carla_stderr2("Engine is not running");
  915. gStandalone.lastError = "Engine is not running";
  916. return false;
  917. }
  918. // allow to save even if engine isn't running
  919. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  920. return plugin->saveStateToFile(filename);
  921. carla_stderr2("carla_save_plugin_state(%i, \"%s\") - could not find plugin", pluginId, filename);
  922. return false;
  923. }
  924. // -------------------------------------------------------------------------------------------------------------------
  925. const CarlaPluginInfo* carla_get_plugin_info(uint pluginId)
  926. {
  927. carla_debug("carla_get_plugin_info(%i)", pluginId);
  928. static CarlaPluginInfo info;
  929. // reset
  930. info.type = CB::PLUGIN_NONE;
  931. info.category = CB::PLUGIN_CATEGORY_NONE;
  932. info.hints = 0x0;
  933. info.optionsAvailable = 0x0;
  934. info.optionsEnabled = 0x0;
  935. info.filename = gNullCharPtr;
  936. info.name = gNullCharPtr;
  937. info.iconName = gNullCharPtr;
  938. info.uniqueId = 0;
  939. // cleanup
  940. if (info.label != gNullCharPtr)
  941. {
  942. delete[] info.label;
  943. info.label = gNullCharPtr;
  944. }
  945. if (info.maker != gNullCharPtr)
  946. {
  947. delete[] info.maker;
  948. info.maker = gNullCharPtr;
  949. }
  950. if (info.copyright != gNullCharPtr)
  951. {
  952. delete[] info.copyright;
  953. info.copyright = gNullCharPtr;
  954. }
  955. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  956. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  957. {
  958. char strBufLabel[STR_MAX+1];
  959. char strBufMaker[STR_MAX+1];
  960. char strBufCopyright[STR_MAX+1];
  961. carla_zeroChar(strBufLabel, STR_MAX+1);
  962. carla_zeroChar(strBufMaker, STR_MAX+1);
  963. carla_zeroChar(strBufCopyright, STR_MAX+1);
  964. info.type = plugin->getType();
  965. info.category = plugin->getCategory();
  966. info.hints = plugin->getHints();
  967. info.filename = plugin->getFilename();
  968. info.name = plugin->getName();
  969. info.iconName = plugin->getIconName();
  970. info.uniqueId = plugin->getUniqueId();
  971. info.optionsAvailable = plugin->getOptionsAvailable();
  972. info.optionsEnabled = plugin->getOptionsEnabled();
  973. plugin->getLabel(strBufLabel);
  974. info.label = carla_strdup(strBufLabel);
  975. plugin->getMaker(strBufMaker);
  976. info.maker = carla_strdup(strBufMaker);
  977. plugin->getCopyright(strBufCopyright);
  978. info.copyright = carla_strdup(strBufCopyright);
  979. checkStringPtr(info.filename);
  980. checkStringPtr(info.name);
  981. checkStringPtr(info.iconName);
  982. return &info;
  983. }
  984. carla_stderr2("carla_get_plugin_info(%i) - could not find plugin", pluginId);
  985. return &info;
  986. }
  987. const CarlaPortCountInfo* carla_get_audio_port_count_info(uint pluginId)
  988. {
  989. carla_debug("carla_get_audio_port_count_info(%i)", pluginId);
  990. static CarlaPortCountInfo info;
  991. // reset
  992. info.ins = 0;
  993. info.outs = 0;
  994. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  995. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  996. {
  997. info.ins = plugin->getAudioInCount();
  998. info.outs = plugin->getAudioOutCount();
  999. return &info;
  1000. }
  1001. carla_stderr2("carla_get_audio_port_count_info(%i) - could not find plugin", pluginId);
  1002. return &info;
  1003. }
  1004. const CarlaPortCountInfo* carla_get_midi_port_count_info(uint pluginId)
  1005. {
  1006. carla_debug("carla_get_midi_port_count_info(%i)", pluginId);
  1007. static CarlaPortCountInfo info;
  1008. // reset
  1009. info.ins = 0;
  1010. info.outs = 0;
  1011. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  1012. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1013. {
  1014. info.ins = plugin->getMidiInCount();
  1015. info.outs = plugin->getMidiOutCount();
  1016. return &info;
  1017. }
  1018. carla_stderr2("carla_get_midi_port_count_info(%i) - could not find plugin", pluginId);
  1019. return &info;
  1020. }
  1021. const CarlaPortCountInfo* carla_get_parameter_count_info(uint pluginId)
  1022. {
  1023. carla_debug("carla_get_parameter_count_info(%i)", pluginId);
  1024. static CarlaPortCountInfo info;
  1025. // reset
  1026. info.ins = 0;
  1027. info.outs = 0;
  1028. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  1029. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1030. {
  1031. plugin->getParameterCountInfo(info.ins, info.outs);
  1032. return &info;
  1033. }
  1034. carla_stderr2("carla_get_parameter_count_info(%i) - could not find plugin", pluginId);
  1035. return &info;
  1036. }
  1037. const CarlaParameterInfo* carla_get_parameter_info(uint pluginId, uint32_t parameterId)
  1038. {
  1039. carla_debug("carla_get_parameter_info(%i, %i)", pluginId, parameterId);
  1040. static CarlaParameterInfo info;
  1041. // reset
  1042. info.scalePointCount = 0;
  1043. // cleanup
  1044. if (info.name != gNullCharPtr)
  1045. {
  1046. delete[] info.name;
  1047. info.name = gNullCharPtr;
  1048. }
  1049. if (info.symbol != gNullCharPtr)
  1050. {
  1051. delete[] info.symbol;
  1052. info.symbol = gNullCharPtr;
  1053. }
  1054. if (info.unit != gNullCharPtr)
  1055. {
  1056. delete[] info.unit;
  1057. info.unit = gNullCharPtr;
  1058. }
  1059. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  1060. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1061. {
  1062. if (parameterId < plugin->getParameterCount())
  1063. {
  1064. char strBufName[STR_MAX+1];
  1065. char strBufSymbol[STR_MAX+1];
  1066. char strBufUnit[STR_MAX+1];
  1067. carla_zeroChar(strBufName, STR_MAX+1);
  1068. carla_zeroChar(strBufSymbol, STR_MAX+1);
  1069. carla_zeroChar(strBufUnit, STR_MAX+1);
  1070. info.scalePointCount = plugin->getParameterScalePointCount(parameterId);
  1071. plugin->getParameterName(parameterId, strBufName);
  1072. info.name = carla_strdup(strBufName);
  1073. plugin->getParameterSymbol(parameterId, strBufSymbol);
  1074. info.symbol = carla_strdup(strBufSymbol);
  1075. plugin->getParameterUnit(parameterId, strBufUnit);
  1076. info.unit = carla_strdup(strBufUnit);
  1077. }
  1078. else
  1079. carla_stderr2("carla_get_parameter_info(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1080. return &info;
  1081. }
  1082. carla_stderr2("carla_get_parameter_info(%i, %i) - could not find plugin", pluginId, parameterId);
  1083. return &info;
  1084. }
  1085. const CarlaScalePointInfo* carla_get_parameter_scalepoint_info(uint pluginId, uint32_t parameterId, uint32_t scalePointId)
  1086. {
  1087. carla_debug("carla_get_parameter_scalepoint_info(%i, %i, %i)", pluginId, parameterId, scalePointId);
  1088. CARLA_ASSERT(gStandalone.engine != nullptr);
  1089. static CarlaScalePointInfo info;
  1090. // reset
  1091. info.value = 0.0f;
  1092. // cleanup
  1093. if (info.label != gNullCharPtr)
  1094. {
  1095. delete[] info.label;
  1096. info.label = gNullCharPtr;
  1097. }
  1098. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &info);
  1099. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1100. {
  1101. if (parameterId < plugin->getParameterCount())
  1102. {
  1103. if (scalePointId < plugin->getParameterScalePointCount(parameterId))
  1104. {
  1105. char strBufLabel[STR_MAX+1];
  1106. carla_zeroChar(strBufLabel, STR_MAX+1);
  1107. info.value = plugin->getParameterScalePointValue(parameterId, scalePointId);
  1108. plugin->getParameterScalePointLabel(parameterId, scalePointId, strBufLabel);
  1109. info.label = carla_strdup(strBufLabel);
  1110. }
  1111. else
  1112. carla_stderr2("carla_get_parameter_scalepoint_info(%i, %i, %i) - scalePointId out of bounds", pluginId, parameterId, scalePointId);
  1113. }
  1114. else
  1115. carla_stderr2("carla_get_parameter_scalepoint_info(%i, %i, %i) - parameterId out of bounds", pluginId, parameterId, scalePointId);
  1116. return &info;
  1117. }
  1118. carla_stderr2("carla_get_parameter_scalepoint_info(%i, %i, %i) - could not find plugin", pluginId, parameterId, scalePointId);
  1119. return &info;
  1120. }
  1121. // -------------------------------------------------------------------------------------------------------------------
  1122. const ParameterData* carla_get_parameter_data(uint pluginId, uint32_t parameterId)
  1123. {
  1124. carla_debug("carla_get_parameter_data(%i, %i)", pluginId, parameterId);
  1125. static const ParameterData fallbackParameterData = { CB::PARAMETER_UNKNOWN, 0x0, CB::PARAMETER_NULL, -1, -1, 0 };
  1126. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &fallbackParameterData);
  1127. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1128. {
  1129. if (parameterId < plugin->getParameterCount())
  1130. return &plugin->getParameterData(parameterId);
  1131. carla_stderr2("carla_get_parameter_data(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1132. return &fallbackParameterData;
  1133. }
  1134. carla_stderr2("carla_get_parameter_data(%i, %i) - could not find plugin", pluginId, parameterId);
  1135. return &fallbackParameterData;
  1136. }
  1137. const ParameterRanges* carla_get_parameter_ranges(uint pluginId, uint32_t parameterId)
  1138. {
  1139. carla_debug("carla_get_parameter_ranges(%i, %i)", pluginId, parameterId);
  1140. static const ParameterRanges fallbackParamRanges = { 0.0f, 0.0f, 1.0f, 0.01f, 0.0001f, 0.1f };
  1141. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &fallbackParamRanges);
  1142. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1143. {
  1144. if (parameterId < plugin->getParameterCount())
  1145. return &plugin->getParameterRanges(parameterId);
  1146. carla_stderr2("carla_get_parameter_ranges(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1147. return &fallbackParamRanges;
  1148. }
  1149. carla_stderr2("carla_get_parameter_ranges(%i, %i) - could not find plugin", pluginId, parameterId);
  1150. return &fallbackParamRanges;
  1151. }
  1152. const MidiProgramData* carla_get_midi_program_data(uint pluginId, uint32_t midiProgramId)
  1153. {
  1154. carla_debug("carla_get_midi_program_data(%i, %i)", pluginId, midiProgramId);
  1155. static MidiProgramData midiProgData;
  1156. // reset
  1157. midiProgData.bank = 0;
  1158. midiProgData.program = 0;
  1159. midiProgData.name = gNullCharPtr;
  1160. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &midiProgData);
  1161. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1162. {
  1163. if (midiProgramId < plugin->getMidiProgramCount())
  1164. {
  1165. const MidiProgramData& ret(plugin->getMidiProgramData(midiProgramId));
  1166. carla_copyStruct<MidiProgramData>(midiProgData, ret);
  1167. checkStringPtr(midiProgData.name);
  1168. return &midiProgData;
  1169. }
  1170. carla_stderr2("carla_get_midi_program_data(%i, %i) - midiProgramId out of bounds", pluginId, midiProgramId);
  1171. return &midiProgData;
  1172. }
  1173. carla_stderr2("carla_get_midi_program_data(%i, %i) - could not find plugin", pluginId, midiProgramId);
  1174. return &midiProgData;
  1175. }
  1176. const CustomData* carla_get_custom_data(uint pluginId, uint32_t customDataId)
  1177. {
  1178. carla_debug("carla_get_custom_data(%i, %i)", pluginId, customDataId);
  1179. static CustomData customData;
  1180. // reset
  1181. customData.type = gNullCharPtr;
  1182. customData.key = gNullCharPtr;
  1183. customData.value = gNullCharPtr;
  1184. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, &customData);
  1185. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1186. {
  1187. if (customDataId < plugin->getCustomDataCount())
  1188. {
  1189. const CustomData& ret(plugin->getCustomData(customDataId));
  1190. carla_copyStruct<CustomData>(customData, ret);
  1191. checkStringPtr(customData.type);
  1192. checkStringPtr(customData.key);
  1193. checkStringPtr(customData.value);
  1194. return &customData;
  1195. }
  1196. carla_stderr2("carla_get_custom_data(%i, %i) - customDataId out of bounds", pluginId, customDataId);
  1197. return &customData;
  1198. }
  1199. carla_stderr2("carla_get_custom_data(%i, %i) - could not find plugin", pluginId, customDataId);
  1200. return &customData;
  1201. }
  1202. const char* carla_get_chunk_data(uint pluginId)
  1203. {
  1204. carla_debug("carla_get_chunk_data(%i)", pluginId);
  1205. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, nullptr);
  1206. static CarlaString chunkData;
  1207. // cleanup
  1208. chunkData.clear();
  1209. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1210. {
  1211. if (plugin->getOptionsEnabled() & CB::PLUGIN_OPTION_USE_CHUNKS)
  1212. {
  1213. void* data = nullptr;
  1214. const int32_t dataSize(plugin->getChunkData(&data));
  1215. if (data != nullptr && dataSize > 0)
  1216. {
  1217. chunkData = QByteArray((char*)data, dataSize).toBase64().constData();
  1218. return chunkData;
  1219. }
  1220. else
  1221. carla_stderr2("carla_get_chunk_data(%i) - got invalid chunk data", pluginId);
  1222. }
  1223. else
  1224. carla_stderr2("carla_get_chunk_data(%i) - plugin does not use chunks", pluginId);
  1225. return nullptr;
  1226. }
  1227. carla_stderr2("carla_get_chunk_data(%i) - could not find plugin", pluginId);
  1228. return nullptr;
  1229. }
  1230. // -------------------------------------------------------------------------------------------------------------------
  1231. uint32_t carla_get_parameter_count(uint pluginId)
  1232. {
  1233. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0);
  1234. carla_debug("carla_get_parameter_count(%i)", pluginId);
  1235. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1236. return plugin->getParameterCount();
  1237. carla_stderr2("carla_get_parameter_count(%i) - could not find plugin", pluginId);
  1238. return 0;
  1239. }
  1240. uint32_t carla_get_program_count(uint pluginId)
  1241. {
  1242. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0);
  1243. carla_debug("carla_get_program_count(%i)", pluginId);
  1244. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1245. return plugin->getProgramCount();
  1246. carla_stderr2("carla_get_program_count(%i) - could not find plugin", pluginId);
  1247. return 0;
  1248. }
  1249. uint32_t carla_get_midi_program_count(uint pluginId)
  1250. {
  1251. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0);
  1252. carla_debug("carla_get_midi_program_count(%i)", pluginId);
  1253. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1254. return plugin->getMidiProgramCount();
  1255. carla_stderr2("carla_get_midi_program_count(%i) - could not find plugin", pluginId);
  1256. return 0;
  1257. }
  1258. uint32_t carla_get_custom_data_count(uint pluginId)
  1259. {
  1260. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0);
  1261. carla_debug("carla_get_custom_data_count(%i)", pluginId);
  1262. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1263. return plugin->getCustomDataCount();
  1264. carla_stderr2("carla_get_custom_data_count(%i) - could not find plugin", pluginId);
  1265. return 0;
  1266. }
  1267. // -------------------------------------------------------------------------------------------------------------------
  1268. const char* carla_get_parameter_text(uint pluginId, uint32_t parameterId, float value)
  1269. {
  1270. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, nullptr);
  1271. carla_debug("carla_get_parameter_text(%i, %i)", pluginId, parameterId);
  1272. static char textBuf[STR_MAX+1];
  1273. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1274. {
  1275. if (parameterId < plugin->getParameterCount())
  1276. {
  1277. carla_zeroChar(textBuf, STR_MAX+1);
  1278. plugin->getParameterText(parameterId, value, textBuf);
  1279. return textBuf;
  1280. }
  1281. carla_stderr2("carla_get_parameter_text(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1282. return nullptr;
  1283. }
  1284. carla_stderr2("carla_get_parameter_text(%i, %i) - could not find plugin", pluginId, parameterId);
  1285. return nullptr;
  1286. }
  1287. const char* carla_get_program_name(uint pluginId, uint32_t programId)
  1288. {
  1289. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, nullptr);
  1290. carla_debug("carla_get_program_name(%i, %i)", pluginId, programId);
  1291. static char programName[STR_MAX+1];
  1292. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1293. {
  1294. if (programId < plugin->getProgramCount())
  1295. {
  1296. carla_zeroChar(programName, STR_MAX+1);
  1297. plugin->getProgramName(programId, programName);
  1298. return programName;
  1299. }
  1300. carla_stderr2("carla_get_program_name(%i, %i) - programId out of bounds", pluginId, programId);
  1301. return nullptr;
  1302. }
  1303. carla_stderr2("carla_get_program_name(%i, %i) - could not find plugin", pluginId, programId);
  1304. return nullptr;
  1305. }
  1306. const char* carla_get_midi_program_name(uint pluginId, uint32_t midiProgramId)
  1307. {
  1308. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, nullptr);
  1309. carla_debug("carla_get_midi_program_name(%i, %i)", pluginId, midiProgramId);
  1310. static char midiProgramName[STR_MAX+1];
  1311. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1312. {
  1313. if (midiProgramId < plugin->getMidiProgramCount())
  1314. {
  1315. carla_zeroChar(midiProgramName, STR_MAX+1);
  1316. plugin->getMidiProgramName(midiProgramId, midiProgramName);
  1317. return midiProgramName;
  1318. }
  1319. carla_stderr2("carla_get_midi_program_name(%i, %i) - midiProgramId out of bounds", pluginId, midiProgramId);
  1320. return nullptr;
  1321. }
  1322. carla_stderr2("carla_get_midi_program_name(%i, %i) - could not find plugin", pluginId, midiProgramId);
  1323. return nullptr;
  1324. }
  1325. const char* carla_get_real_plugin_name(uint pluginId)
  1326. {
  1327. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, nullptr);
  1328. carla_debug("carla_get_real_plugin_name(%i)", pluginId);
  1329. static char realPluginName[STR_MAX+1];
  1330. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1331. {
  1332. carla_zeroChar(realPluginName, STR_MAX+1);
  1333. plugin->getRealName(realPluginName);
  1334. return realPluginName;
  1335. }
  1336. carla_stderr2("carla_get_real_plugin_name(%i) - could not find plugin", pluginId);
  1337. return nullptr;
  1338. }
  1339. // -------------------------------------------------------------------------------------------------------------------
  1340. int32_t carla_get_current_program_index(uint pluginId)
  1341. {
  1342. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, -1);
  1343. carla_debug("carla_get_current_program_index(%i)", pluginId);
  1344. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1345. return plugin->getCurrentProgram();
  1346. carla_stderr2("carla_get_current_program_index(%i) - could not find plugin", pluginId);
  1347. return -1;
  1348. }
  1349. int32_t carla_get_current_midi_program_index(uint pluginId)
  1350. {
  1351. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, -1);
  1352. carla_debug("carla_get_current_midi_program_index(%i)", pluginId);
  1353. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1354. return plugin->getCurrentMidiProgram();
  1355. carla_stderr2("carla_get_current_midi_program_index(%i) - could not find plugin", pluginId);
  1356. return -1;
  1357. }
  1358. // -------------------------------------------------------------------------------------------------------------------
  1359. float carla_get_default_parameter_value(uint pluginId, uint32_t parameterId)
  1360. {
  1361. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0.0f);
  1362. carla_debug("carla_get_default_parameter_value(%i, %i)", pluginId, parameterId);
  1363. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1364. {
  1365. if (parameterId < plugin->getParameterCount())
  1366. return plugin->getParameterRanges(parameterId).def;
  1367. carla_stderr2("carla_get_default_parameter_value(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1368. return 0.0f;
  1369. }
  1370. carla_stderr2("carla_get_default_parameter_value(%i, %i) - could not find plugin", pluginId, parameterId);
  1371. return 0.0f;
  1372. }
  1373. float carla_get_current_parameter_value(uint pluginId, uint32_t parameterId)
  1374. {
  1375. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0.0f);
  1376. carla_debug("carla_get_current_parameter_value(%i, %i)", pluginId, parameterId);
  1377. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1378. {
  1379. if (parameterId < plugin->getParameterCount())
  1380. return plugin->getParameterValue(parameterId);
  1381. carla_stderr2("carla_get_current_parameter_value(%i, %i) - parameterId out of bounds", pluginId, parameterId);
  1382. return 0.0f;
  1383. }
  1384. carla_stderr2("carla_get_current_parameter_value(%i, %i) - could not find plugin", pluginId, parameterId);
  1385. return 0.0f;
  1386. }
  1387. // -------------------------------------------------------------------------------------------------------------------
  1388. float carla_get_input_peak_value(uint pluginId, bool isLeft)
  1389. {
  1390. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0.0f);
  1391. return gStandalone.engine->getInputPeak(pluginId, isLeft);
  1392. }
  1393. float carla_get_output_peak_value(uint pluginId, bool isLeft)
  1394. {
  1395. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0.0f);
  1396. return gStandalone.engine->getOutputPeak(pluginId, isLeft);
  1397. }
  1398. // -------------------------------------------------------------------------------------------------------------------
  1399. void carla_set_option(uint pluginId, uint option, bool yesNo)
  1400. {
  1401. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1402. carla_debug("carla_set_option(%i, %i, %s)", pluginId, option, bool2str(yesNo));
  1403. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1404. return plugin->setOption(option, yesNo);
  1405. carla_stderr2("carla_set_option(%i, %i, %s) - could not find plugin", pluginId, option, bool2str(yesNo));
  1406. }
  1407. void carla_set_active(uint pluginId, bool onOff)
  1408. {
  1409. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1410. carla_debug("carla_set_active(%i, %s)", pluginId, bool2str(onOff));
  1411. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1412. return plugin->setActive(onOff, true, false);
  1413. carla_stderr2("carla_set_active(%i, %s) - could not find plugin", pluginId, bool2str(onOff));
  1414. }
  1415. #ifndef BUILD_BRIDGE
  1416. void carla_set_drywet(uint pluginId, float value)
  1417. {
  1418. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1419. carla_debug("carla_set_drywet(%i, %f)", pluginId, value);
  1420. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1421. return plugin->setDryWet(value, true, false);
  1422. carla_stderr2("carla_set_drywet(%i, %f) - could not find plugin", pluginId, value);
  1423. }
  1424. void carla_set_volume(uint pluginId, float value)
  1425. {
  1426. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1427. carla_debug("carla_set_volume(%i, %f)", pluginId, value);
  1428. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1429. return plugin->setVolume(value, true, false);
  1430. carla_stderr2("carla_set_volume(%i, %f) - could not find plugin", pluginId, value);
  1431. }
  1432. void carla_set_balance_left(uint pluginId, float value)
  1433. {
  1434. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1435. carla_debug("carla_set_balance_left(%i, %f)", pluginId, value);
  1436. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1437. return plugin->setBalanceLeft(value, true, false);
  1438. carla_stderr2("carla_set_balance_left(%i, %f) - could not find plugin", pluginId, value);
  1439. }
  1440. void carla_set_balance_right(uint pluginId, float value)
  1441. {
  1442. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1443. carla_debug("carla_set_balance_right(%i, %f)", pluginId, value);
  1444. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1445. return plugin->setBalanceRight(value, true, false);
  1446. carla_stderr2("carla_set_balance_right(%i, %f) - could not find plugin", pluginId, value);
  1447. }
  1448. void carla_set_panning(uint pluginId, float value)
  1449. {
  1450. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1451. carla_debug("carla_set_panning(%i, %f)", pluginId, value);
  1452. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1453. return plugin->setPanning(value, true, false);
  1454. carla_stderr2("carla_set_panning(%i, %f) - could not find plugin", pluginId, value);
  1455. }
  1456. #endif
  1457. void carla_set_ctrl_channel(uint pluginId, int8_t channel)
  1458. {
  1459. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1460. carla_debug("carla_set_ctrl_channel(%i, %i)", pluginId, channel);
  1461. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1462. return plugin->setCtrlChannel(channel, true, false);
  1463. carla_stderr2("carla_set_ctrl_channel(%i, %i) - could not find plugin", pluginId, channel);
  1464. }
  1465. // -------------------------------------------------------------------------------------------------------------------
  1466. void carla_set_parameter_value(uint pluginId, uint32_t parameterId, float value)
  1467. {
  1468. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1469. carla_debug("carla_set_parameter_value(%i, %i, %f)", pluginId, parameterId, value);
  1470. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1471. {
  1472. if (parameterId < plugin->getParameterCount())
  1473. return plugin->setParameterValue(parameterId, value, true, true, false);
  1474. carla_stderr2("carla_set_parameter_value(%i, %i, %f) - parameterId out of bounds", pluginId, parameterId, value);
  1475. return;
  1476. }
  1477. carla_stderr2("carla_set_parameter_value(%i, %i, %f) - could not find plugin", pluginId, parameterId, value);
  1478. }
  1479. #ifndef BUILD_BRIDGE
  1480. void carla_set_parameter_midi_channel(uint pluginId, uint32_t parameterId, uint8_t channel)
  1481. {
  1482. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1483. CARLA_SAFE_ASSERT_RETURN(channel >= MAX_MIDI_CHANNELS,);
  1484. carla_debug("carla_set_parameter_midi_channel(%i, %i, %i)", pluginId, parameterId, channel);
  1485. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1486. {
  1487. if (parameterId < plugin->getParameterCount())
  1488. return plugin->setParameterMidiChannel(parameterId, channel, true, false);
  1489. carla_stderr2("carla_set_parameter_midi_channel(%i, %i, %i) - parameterId out of bounds", pluginId, parameterId, channel);
  1490. return;
  1491. }
  1492. carla_stderr2("carla_set_parameter_midi_channel(%i, %i, %i) - could not find plugin", pluginId, parameterId, channel);
  1493. }
  1494. void carla_set_parameter_midi_cc(uint pluginId, uint32_t parameterId, int16_t cc)
  1495. {
  1496. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1497. CARLA_SAFE_ASSERT_RETURN(cc >= -1 && cc <= 0x5F,);
  1498. carla_debug("carla_set_parameter_midi_cc(%i, %i, %i)", pluginId, parameterId, cc);
  1499. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1500. {
  1501. if (parameterId < plugin->getParameterCount())
  1502. return plugin->setParameterMidiCC(parameterId, cc, true, false);
  1503. carla_stderr2("carla_set_parameter_midi_cc(%i, %i, %i) - parameterId out of bounds", pluginId, parameterId, cc);
  1504. return;
  1505. }
  1506. carla_stderr2("carla_set_parameter_midi_cc(%i, %i, %i) - could not find plugin", pluginId, parameterId, cc);
  1507. }
  1508. #endif
  1509. // -------------------------------------------------------------------------------------------------------------------
  1510. void carla_set_program(uint pluginId, uint32_t programId)
  1511. {
  1512. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1513. carla_debug("carla_set_program(%i, %i)", pluginId, programId);
  1514. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1515. {
  1516. if (programId < plugin->getProgramCount())
  1517. return plugin->setProgram(static_cast<int32_t>(programId), true, true, false);
  1518. carla_stderr2("carla_set_program(%i, %i) - programId out of bounds", pluginId, programId);
  1519. return;
  1520. }
  1521. carla_stderr2("carla_set_program(%i, %i) - could not find plugin", pluginId, programId);
  1522. }
  1523. void carla_set_midi_program(uint pluginId, uint32_t midiProgramId)
  1524. {
  1525. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1526. carla_debug("carla_set_midi_program(%i, %i)", pluginId, midiProgramId);
  1527. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1528. {
  1529. if (midiProgramId < plugin->getMidiProgramCount())
  1530. return plugin->setMidiProgram(static_cast<int32_t>(midiProgramId), true, true, false);
  1531. carla_stderr2("carla_set_midi_program(%i, %i) - midiProgramId out of bounds", pluginId, midiProgramId);
  1532. return;
  1533. }
  1534. carla_stderr2("carla_set_midi_program(%i, %i) - could not find plugin", pluginId, midiProgramId);
  1535. }
  1536. // -------------------------------------------------------------------------------------------------------------------
  1537. void carla_set_custom_data(uint pluginId, const char* type, const char* key, const char* value)
  1538. {
  1539. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1540. CARLA_SAFE_ASSERT_RETURN(type != nullptr && type[0] != '\0',);
  1541. CARLA_SAFE_ASSERT_RETURN(key != nullptr && key[0] != '\0',);
  1542. CARLA_SAFE_ASSERT_RETURN(value != nullptr,);
  1543. carla_debug("carla_set_custom_data(%i, \"%s\", \"%s\", \"%s\")", pluginId, type, key, value);
  1544. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1545. return plugin->setCustomData(type, key, value, true);
  1546. carla_stderr2("carla_set_custom_data(%i, \"%s\", \"%s\", \"%s\") - could not find plugin", pluginId, type, key, value);
  1547. }
  1548. void carla_set_chunk_data(uint pluginId, const char* chunkData)
  1549. {
  1550. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1551. CARLA_SAFE_ASSERT_RETURN(chunkData != nullptr && chunkData[0] != '\0',);
  1552. carla_debug("carla_set_chunk_data(%i, \"%s\")", pluginId, chunkData);
  1553. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1554. {
  1555. if (plugin->getOptionsEnabled() & CB::PLUGIN_OPTION_USE_CHUNKS)
  1556. return plugin->setChunkData(chunkData);
  1557. carla_stderr2("carla_set_chunk_data(%i, \"%s\") - plugin does not use chunks", pluginId, chunkData);
  1558. return;
  1559. }
  1560. carla_stderr2("carla_set_chunk_data(%i, \"%s\") - could not find plugin", pluginId, chunkData);
  1561. }
  1562. // -------------------------------------------------------------------------------------------------------------------
  1563. void carla_prepare_for_save(uint pluginId)
  1564. {
  1565. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1566. carla_debug("carla_prepare_for_save(%i)", pluginId);
  1567. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1568. return plugin->prepareForSave();
  1569. carla_stderr2("carla_prepare_for_save(%i) - could not find plugin", pluginId);
  1570. }
  1571. #ifndef BUILD_BRIDGE
  1572. void carla_send_midi_note(uint pluginId, uint8_t channel, uint8_t note, uint8_t velocity)
  1573. {
  1574. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr && gStandalone.engine->isRunning(),);
  1575. carla_debug("carla_send_midi_note(%i, %i, %i, %i)", pluginId, channel, note, velocity);
  1576. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1577. return plugin->sendMidiSingleNote(channel, note, velocity, true, true, false);
  1578. carla_stderr2("carla_send_midi_note(%i, %i, %i, %i) - could not find plugin", pluginId, channel, note, velocity);
  1579. }
  1580. #endif
  1581. void carla_show_custom_ui(uint pluginId, bool yesNo)
  1582. {
  1583. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr,);
  1584. carla_debug("carla_show_custom_ui(%i, %s)", pluginId, bool2str(yesNo));
  1585. if (CarlaPlugin* const plugin = gStandalone.engine->getPlugin(pluginId))
  1586. return plugin->showCustomUI(yesNo);
  1587. carla_stderr2("carla_show_custom_ui(%i, %s) - could not find plugin", pluginId, bool2str(yesNo));
  1588. }
  1589. // -------------------------------------------------------------------------------------------------------------------
  1590. uint32_t carla_get_buffer_size()
  1591. {
  1592. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0);
  1593. carla_debug("carla_get_buffer_size()");
  1594. return gStandalone.engine->getBufferSize();
  1595. }
  1596. double carla_get_sample_rate()
  1597. {
  1598. CARLA_SAFE_ASSERT_RETURN(gStandalone.engine != nullptr, 0.0);
  1599. carla_debug("carla_get_sample_rate()");
  1600. return gStandalone.engine->getSampleRate();
  1601. }
  1602. // -------------------------------------------------------------------------------------------------------------------
  1603. const char* carla_get_last_error()
  1604. {
  1605. carla_debug("carla_get_last_error()");
  1606. if (gStandalone.engine != nullptr)
  1607. return gStandalone.engine->getLastError();
  1608. return gStandalone.lastError;
  1609. }
  1610. const char* carla_get_host_osc_url_tcp()
  1611. {
  1612. carla_debug("carla_get_host_osc_url_tcp()");
  1613. if (gStandalone.engine == nullptr)
  1614. {
  1615. carla_stderr2("Engine is not running");
  1616. gStandalone.lastError = "Engine is not running";
  1617. return nullptr;
  1618. }
  1619. return gStandalone.engine->getOscServerPathTCP();
  1620. }
  1621. const char* carla_get_host_osc_url_udp()
  1622. {
  1623. carla_debug("carla_get_host_osc_url_udp()");
  1624. if (gStandalone.engine == nullptr)
  1625. {
  1626. carla_stderr2("Engine is not running");
  1627. gStandalone.lastError = "Engine is not running";
  1628. return nullptr;
  1629. }
  1630. return gStandalone.engine->getOscServerPathUDP();
  1631. }
  1632. // -------------------------------------------------------------------------------------------------------------------
  1633. #ifdef WANT_DSSI
  1634. # include "CarlaDssiUtils.cpp"
  1635. #endif
  1636. #include "CarlaStateUtils.cpp"
  1637. #if 0
  1638. int main(int argc, char* argv[])
  1639. {
  1640. return 0;
  1641. }
  1642. #endif
  1643. #if 0
  1644. //#include "CarlaOscUtils.hpp"
  1645. // -------------------------------------------------------------------------------------------------------------------
  1646. #define NSM_API_VERSION_MAJOR 1
  1647. #define NSM_API_VERSION_MINOR 2
  1648. class CarlaNSM
  1649. {
  1650. public:
  1651. CarlaNSM()
  1652. : fServerThread(nullptr),
  1653. fReplyAddr(nullptr),
  1654. fIsReady(false),
  1655. fIsOpened(false),
  1656. fIsSaved(false)
  1657. {
  1658. }
  1659. ~CarlaNSM()
  1660. {
  1661. if (fReplyAddr != nullptr)
  1662. lo_address_free(fReplyAddr);
  1663. if (fServerThread != nullptr)
  1664. {
  1665. lo_server_thread_stop(fServerThread);
  1666. lo_server_thread_del_method(fServerThread, "/reply", "ssss");
  1667. lo_server_thread_del_method(fServerThread, "/nsm/client/open", "sss");
  1668. lo_server_thread_del_method(fServerThread, "/nsm/client/save", "");
  1669. lo_server_thread_free(fServerThread);
  1670. }
  1671. }
  1672. void announce(const char* const url, const char* appName, const int pid)
  1673. {
  1674. lo_address const addr = lo_address_new_from_url(url);
  1675. if (addr == nullptr)
  1676. return;
  1677. const int proto = lo_address_get_protocol(addr);
  1678. if (fServerThread == nullptr)
  1679. {
  1680. // create new OSC thread
  1681. fServerThread = lo_server_thread_new_with_proto(nullptr, proto, error_handler);
  1682. // register message handlers and start OSC thread
  1683. lo_server_thread_add_method(fServerThread, "/reply", "ssss", _reply_handler, this);
  1684. lo_server_thread_add_method(fServerThread, "/nsm/client/open", "sss", _open_handler, this);
  1685. lo_server_thread_add_method(fServerThread, "/nsm/client/save", "", _save_handler, this);
  1686. lo_server_thread_start(fServerThread);
  1687. }
  1688. #ifndef BUILD_ANSI_TEST
  1689. lo_send_from(addr, lo_server_thread_get_server(fServerThread), LO_TT_IMMEDIATE, "/nsm/server/announce", "sssiii",
  1690. "Carla", ":switch:", appName, NSM_API_VERSION_MAJOR, NSM_API_VERSION_MINOR, pid);
  1691. #endif
  1692. lo_address_free(addr);
  1693. }
  1694. void ready()
  1695. {
  1696. fIsReady = true;
  1697. }
  1698. void replyOpen()
  1699. {
  1700. fIsOpened = true;
  1701. }
  1702. void replySave()
  1703. {
  1704. fIsSaved = true;
  1705. }
  1706. protected:
  1707. int handleReply(const char* const path, const char* const types, lo_arg** const argv, const int argc, const lo_message msg)
  1708. {
  1709. carla_debug("CarlaNSM::handleReply(%s, %i, %p, %s, %p)", path, argc, argv, types, msg);
  1710. if (fReplyAddr != nullptr)
  1711. lo_address_free(fReplyAddr);
  1712. fIsOpened = false;
  1713. fIsSaved = false;
  1714. char* const url = lo_address_get_url(lo_message_get_source(msg));
  1715. fReplyAddr = lo_address_new_from_url(url);
  1716. std::free(url);
  1717. const char* const method = &argv[0]->s;
  1718. const char* const smName = &argv[2]->s;
  1719. // wait max 6 secs for host to init
  1720. for (int i=0; i < 60 && ! fIsReady; ++i)
  1721. carla_msleep(100);
  1722. if (std::strcmp(method, "/nsm/server/announce") == 0 && gStandalone.callback != nullptr)
  1723. gStandalone.callback(gStandalone.callbackPtr, CB::ENGINE_CALLBACK_NSM_ANNOUNCE, 0, 0, 0, 0.0f, smName);
  1724. return 0;
  1725. #ifndef DEBUG
  1726. // unused
  1727. (void)path;
  1728. (void)types;
  1729. (void)argc;
  1730. #endif
  1731. }
  1732. int handleOpen(const char* const path, const char* const types, lo_arg** const argv, const int argc, const lo_message msg)
  1733. {
  1734. carla_debug("CarlaNSM::handleOpen(\"%s\", \"%s\", %p, %i, %p)", path, types, argv, argc, msg);
  1735. if (gStandalone.callback == nullptr)
  1736. return 1;
  1737. if (fServerThread == nullptr)
  1738. return 1;
  1739. if (fReplyAddr == nullptr)
  1740. return 1;
  1741. const char* const projectPath = &argv[0]->s;
  1742. const char* const clientId = &argv[2]->s;
  1743. char data[std::strlen(projectPath)+std::strlen(clientId)+2];
  1744. std::strcpy(data, projectPath);
  1745. std::strcat(data, ":");
  1746. std::strcat(data, clientId);
  1747. fIsOpened = false;
  1748. gStandalone.callback(nullptr, CB::ENGINE_CALLBACK_NSM_OPEN, 0, 0, 0, 0.0f, data);
  1749. // wait max 10 secs to open
  1750. for (int i=0; i < 100 && ! fIsOpened; ++i)
  1751. carla_msleep(100);
  1752. #ifndef BUILD_ANSI_TEST
  1753. if (fIsOpened)
  1754. lo_send_from(fReplyAddr, lo_server_thread_get_server(fServerThread), LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/open", "OK");
  1755. #endif
  1756. return 0;
  1757. #ifndef DEBUG
  1758. // unused
  1759. (void)path;
  1760. (void)types;
  1761. (void)argc;
  1762. (void)msg;
  1763. #endif
  1764. }
  1765. int handleSave(const char* const path, const char* const types, lo_arg** const argv, const int argc, const lo_message msg)
  1766. {
  1767. carla_debug("CarlaNSM::handleSave(\"%s\", \"%s\", %p, %i, %p)", path, types, argv, argc, msg);
  1768. if (gStandalone.callback == nullptr)
  1769. return 1;
  1770. if (fServerThread == nullptr)
  1771. return 1;
  1772. if (fReplyAddr == nullptr)
  1773. return 1;
  1774. fIsSaved = false;
  1775. gStandalone.callback(nullptr, CB::ENGINE_CALLBACK_NSM_SAVE, 0, 0, 0, 0.0f, nullptr);
  1776. // wait max 10 secs to save
  1777. for (int i=0; i < 100 && ! fIsSaved; ++i)
  1778. carla_msleep(100);
  1779. #ifndef BUILD_ANSI_TEST
  1780. if (fIsSaved)
  1781. lo_send_from(fReplyAddr, lo_server_thread_get_server(fServerThread), LO_TT_IMMEDIATE, "/reply", "ss", "/nsm/client/save", "OK");
  1782. #endif
  1783. return 0;
  1784. #ifndef DEBUG
  1785. // unused
  1786. (void)path;
  1787. (void)types;
  1788. (void)argv;
  1789. (void)argc;
  1790. (void)msg;
  1791. #endif
  1792. }
  1793. private:
  1794. lo_server_thread fServerThread;
  1795. lo_address fReplyAddr;
  1796. bool fIsReady; // used to startup, only once
  1797. bool fIsOpened;
  1798. bool fIsSaved;
  1799. #define handlePtr ((CarlaNSM*)data)
  1800. static int _reply_handler(const char* path, const char* types, lo_arg** argv, int argc, lo_message msg, void* data)
  1801. {
  1802. return handlePtr->handleReply(path, types, argv, argc, msg);
  1803. }
  1804. static int _open_handler(const char* path, const char* types, lo_arg** argv, int argc, lo_message msg, void* data)
  1805. {
  1806. return handlePtr->handleOpen(path, types, argv, argc, msg);
  1807. }
  1808. static int _save_handler(const char* path, const char* types, lo_arg** argv, int argc, lo_message msg, void* data)
  1809. {
  1810. return handlePtr->handleSave(path, types, argv, argc, msg);
  1811. }
  1812. #undef handlePtr
  1813. static void error_handler(int num, const char* msg, const char* path)
  1814. {
  1815. carla_stderr2("CarlaNSM::error_handler(%i, \"%s\", \"%s\")", num, msg, path);
  1816. }
  1817. };
  1818. static CarlaNSM gCarlaNSM;
  1819. void carla_nsm_announce(const char* url, const char* appName, int pid)
  1820. {
  1821. CARLA_SAFE_ASSERT_RETURN(url != nullptr && url[0] != '\0',);
  1822. CARLA_SAFE_ASSERT_RETURN(appName != nullptr && appName[0] != '\0',);
  1823. CARLA_SAFE_ASSERT_RETURN(pid != 0,);
  1824. carla_debug("carla_nsm_announce(\"%s\", \"%s\", %i)", url, appName, pid);
  1825. gCarlaNSM.announce(url, appName, pid);
  1826. }
  1827. void carla_nsm_ready()
  1828. {
  1829. carla_debug("carla_nsm_ready()");
  1830. gCarlaNSM.ready();
  1831. }
  1832. void carla_nsm_reply_open()
  1833. {
  1834. carla_debug("carla_nsm_reply_open()");
  1835. gCarlaNSM.replyOpen();
  1836. }
  1837. void carla_nsm_reply_save()
  1838. {
  1839. carla_debug("carla_nsm_reply_save()");
  1840. gCarlaNSM.replySave();
  1841. }
  1842. #endif
  1843. // -------------------------------------------------------------------------------------------------------------------
  1844. //#include "CarlaLogThread.hpp"
  1845. //#if ! (defined(DEBUG) || defined(WANT_LOGS) || defined(BUILD_ANSI_TEST))
  1846. //# define WANT_LOGS
  1847. //#endif
  1848. //#ifdef WANT_LOGS
  1849. //static CarlaLogThread gLogThread;
  1850. //#endif
  1851. // -------------------------------------------------------------------------------------------------------------------