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.

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