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.

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