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.

2415 lines
83KB

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