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.

2421 lines
79KB

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