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.

2151 lines
71KB

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