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.

2284 lines
77KB

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