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.

2201 lines
72KB

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