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.

2479 lines
85KB

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