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.

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