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.

2183 lines
72KB

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