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.

1238 lines
43KB

  1. /*
  2. * Carla Bridge UI
  3. * Copyright (C) 2011-2019 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. #include "CarlaBridgeFormat.hpp"
  18. #include "CarlaBridgeToolkit.hpp"
  19. #include "CarlaLibUtils.hpp"
  20. #include "CarlaLv2Utils.hpp"
  21. #include "CarlaMIDI.h"
  22. #include "LinkedList.hpp"
  23. #include "water/files/File.h"
  24. #include <string>
  25. #include <vector>
  26. #define URI_CARLA_ATOM_WORKER_IN "http://kxstudio.sf.net/ns/carla/atomWorkerIn"
  27. #define URI_CARLA_ATOM_WORKER_RESP "http://kxstudio.sf.net/ns/carla/atomWorkerResp"
  28. using water::File;
  29. CARLA_BRIDGE_UI_START_NAMESPACE
  30. // --------------------------------------------------------------------------------------------------------------------
  31. static double gInitialSampleRate = 44100.0;
  32. static const char* const kNullWindowTitle = "TestUI";
  33. static const uint32_t kNullWindowTitleSize = 6;
  34. // LV2 URI Map Ids
  35. enum CarlaLv2URIDs {
  36. kUridNull = 0,
  37. kUridAtomBlank,
  38. kUridAtomBool,
  39. kUridAtomChunk,
  40. kUridAtomDouble,
  41. kUridAtomEvent,
  42. kUridAtomFloat,
  43. kUridAtomInt,
  44. kUridAtomLiteral,
  45. kUridAtomLong,
  46. kUridAtomNumber,
  47. kUridAtomObject,
  48. kUridAtomPath,
  49. kUridAtomProperty,
  50. kUridAtomResource,
  51. kUridAtomSequence,
  52. kUridAtomSound,
  53. kUridAtomString,
  54. kUridAtomTuple,
  55. kUridAtomURI,
  56. kUridAtomURID,
  57. kUridAtomVector,
  58. kUridAtomTransferAtom,
  59. kUridAtomTransferEvent,
  60. kUridBufMaxLength,
  61. kUridBufMinLength,
  62. kUridBufNominalLength,
  63. kUridBufSequenceSize,
  64. kUridLogError,
  65. kUridLogNote,
  66. kUridLogTrace,
  67. kUridLogWarning,
  68. // time base type
  69. kUridTimePosition,
  70. // time values
  71. kUridTimeBar,
  72. kUridTimeBarBeat,
  73. kUridTimeBeat,
  74. kUridTimeBeatUnit,
  75. kUridTimeBeatsPerBar,
  76. kUridTimeBeatsPerMinute,
  77. kUridTimeFrame,
  78. kUridTimeFramesPerSecond,
  79. kUridTimeSpeed,
  80. kUridTimeTicksPerBeat,
  81. kUridMidiEvent,
  82. kUridParamSampleRate,
  83. kUridWindowTitle,
  84. kUridCarlaAtomWorkerIn,
  85. kUridCarlaAtomWorkerResp,
  86. kUridCarlaTransientWindowId,
  87. kUridCount
  88. };
  89. // LV2 Feature Ids
  90. enum CarlaLv2Features {
  91. // DSP features
  92. kFeatureIdLogs = 0,
  93. kFeatureIdOptions,
  94. kFeatureIdPrograms,
  95. kFeatureIdStateMakePath,
  96. kFeatureIdStateMapPath,
  97. kFeatureIdUriMap,
  98. kFeatureIdUridMap,
  99. kFeatureIdUridUnmap,
  100. kFeatureIdUiIdleInterface,
  101. kFeatureIdUiFixedSize,
  102. kFeatureIdUiMakeResident,
  103. kFeatureIdUiMakeResident2,
  104. kFeatureIdUiNoUserResize,
  105. kFeatureIdUiParent,
  106. kFeatureIdUiPortMap,
  107. kFeatureIdUiPortSubscribe,
  108. kFeatureIdUiResize,
  109. kFeatureIdUiTouch,
  110. kFeatureCount
  111. };
  112. // --------------------------------------------------------------------------------------------------------------------
  113. struct Lv2PluginOptions {
  114. enum OptIndex {
  115. SampleRate,
  116. TransientWinId,
  117. WindowTitle,
  118. Null,
  119. Count
  120. };
  121. float sampleRate;
  122. int64_t transientWinId;
  123. const char* windowTitle;
  124. LV2_Options_Option opts[Count];
  125. Lv2PluginOptions() noexcept
  126. : sampleRate(static_cast<float>(gInitialSampleRate)),
  127. transientWinId(0),
  128. windowTitle(kNullWindowTitle)
  129. {
  130. LV2_Options_Option& optSampleRate(opts[SampleRate]);
  131. optSampleRate.context = LV2_OPTIONS_INSTANCE;
  132. optSampleRate.subject = 0;
  133. optSampleRate.key = kUridParamSampleRate;
  134. optSampleRate.size = sizeof(float);
  135. optSampleRate.type = kUridAtomFloat;
  136. optSampleRate.value = &sampleRate;
  137. LV2_Options_Option& optTransientWinId(opts[TransientWinId]);
  138. optTransientWinId.context = LV2_OPTIONS_INSTANCE;
  139. optTransientWinId.subject = 0;
  140. optTransientWinId.key = kUridCarlaTransientWindowId;
  141. optTransientWinId.size = sizeof(int64_t);
  142. optTransientWinId.type = kUridAtomLong;
  143. optTransientWinId.value = &transientWinId;
  144. LV2_Options_Option& optWindowTitle(opts[WindowTitle]);
  145. optWindowTitle.context = LV2_OPTIONS_INSTANCE;
  146. optWindowTitle.subject = 0;
  147. optWindowTitle.key = kUridWindowTitle;
  148. optWindowTitle.size = kNullWindowTitleSize;
  149. optWindowTitle.type = kUridAtomString;
  150. optWindowTitle.value = kNullWindowTitle;
  151. LV2_Options_Option& optNull(opts[Null]);
  152. optNull.context = LV2_OPTIONS_INSTANCE;
  153. optNull.subject = 0;
  154. optNull.key = kUridNull;
  155. optNull.size = 0;
  156. optNull.type = kUridNull;
  157. optNull.value = nullptr;
  158. }
  159. };
  160. // --------------------------------------------------------------------------------------------------------------------
  161. class CarlaLv2Client : public CarlaBridgeFormat
  162. {
  163. public:
  164. CarlaLv2Client()
  165. : CarlaBridgeFormat(),
  166. fHandle(nullptr),
  167. fWidget(nullptr),
  168. fDescriptor(nullptr),
  169. fRdfDescriptor(nullptr),
  170. fRdfUiDescriptor(nullptr),
  171. fControlDesignatedPort(0),
  172. fLv2Options(),
  173. fUiOptions(),
  174. fCustomURIDs(kUridCount, std::string("urn:null")),
  175. fExt()
  176. {
  177. CARLA_SAFE_ASSERT(fCustomURIDs.size() == kUridCount);
  178. carla_zeroPointers(fFeatures, kFeatureCount+1);
  179. // ------------------------------------------------------------------------------------------------------------
  180. // initialize features (part 1)
  181. LV2_Log_Log* const logFt = new LV2_Log_Log;
  182. logFt->handle = this;
  183. logFt->printf = carla_lv2_log_printf;
  184. logFt->vprintf = carla_lv2_log_vprintf;
  185. LV2_State_Make_Path* const stateMakePathFt = new LV2_State_Make_Path;
  186. stateMakePathFt->handle = this;
  187. stateMakePathFt->path = carla_lv2_state_make_path;
  188. LV2_State_Map_Path* const stateMapPathFt = new LV2_State_Map_Path;
  189. stateMapPathFt->handle = this;
  190. stateMapPathFt->abstract_path = carla_lv2_state_map_abstract_path;
  191. stateMapPathFt->absolute_path = carla_lv2_state_map_absolute_path;
  192. LV2_Programs_Host* const programsFt = new LV2_Programs_Host;
  193. programsFt->handle = this;
  194. programsFt->program_changed = carla_lv2_program_changed;
  195. LV2_URI_Map_Feature* const uriMapFt = new LV2_URI_Map_Feature;
  196. uriMapFt->callback_data = this;
  197. uriMapFt->uri_to_id = carla_lv2_uri_to_id;
  198. LV2_URID_Map* const uridMapFt = new LV2_URID_Map;
  199. uridMapFt->handle = this;
  200. uridMapFt->map = carla_lv2_urid_map;
  201. LV2_URID_Unmap* const uridUnmapFt = new LV2_URID_Unmap;
  202. uridUnmapFt->handle = this;
  203. uridUnmapFt->unmap = carla_lv2_urid_unmap;
  204. LV2UI_Port_Map* const uiPortMapFt = new LV2UI_Port_Map;
  205. uiPortMapFt->handle = this;
  206. uiPortMapFt->port_index = carla_lv2_ui_port_map;
  207. LV2UI_Resize* const uiResizeFt = new LV2UI_Resize;
  208. uiResizeFt->handle = this;
  209. uiResizeFt->ui_resize = carla_lv2_ui_resize;
  210. // ------------------------------------------------------------------------------------------------------------
  211. // initialize features (part 2)
  212. for (uint32_t i=0; i < kFeatureCount; ++i)
  213. fFeatures[i] = new LV2_Feature;
  214. fFeatures[kFeatureIdLogs]->URI = LV2_LOG__log;
  215. fFeatures[kFeatureIdLogs]->data = logFt;
  216. fFeatures[kFeatureIdOptions]->URI = LV2_OPTIONS__options;
  217. fFeatures[kFeatureIdOptions]->data = fLv2Options.opts;
  218. fFeatures[kFeatureIdPrograms]->URI = LV2_PROGRAMS__Host;
  219. fFeatures[kFeatureIdPrograms]->data = programsFt;
  220. fFeatures[kFeatureIdStateMakePath]->URI = LV2_STATE__makePath;
  221. fFeatures[kFeatureIdStateMakePath]->data = stateMakePathFt;
  222. fFeatures[kFeatureIdStateMapPath]->URI = LV2_STATE__mapPath;
  223. fFeatures[kFeatureIdStateMapPath]->data = stateMapPathFt;
  224. fFeatures[kFeatureIdUriMap]->URI = LV2_URI_MAP_URI;
  225. fFeatures[kFeatureIdUriMap]->data = uriMapFt;
  226. fFeatures[kFeatureIdUridMap]->URI = LV2_URID__map;
  227. fFeatures[kFeatureIdUridMap]->data = uridMapFt;
  228. fFeatures[kFeatureIdUridUnmap]->URI = LV2_URID__unmap;
  229. fFeatures[kFeatureIdUridUnmap]->data = uridUnmapFt;
  230. fFeatures[kFeatureIdUiIdleInterface]->URI = LV2_UI__idleInterface;
  231. fFeatures[kFeatureIdUiIdleInterface]->data = nullptr;
  232. fFeatures[kFeatureIdUiFixedSize]->URI = LV2_UI__fixedSize;
  233. fFeatures[kFeatureIdUiFixedSize]->data = nullptr;
  234. fFeatures[kFeatureIdUiMakeResident]->URI = LV2_UI__makeResident;
  235. fFeatures[kFeatureIdUiMakeResident]->data = nullptr;
  236. fFeatures[kFeatureIdUiMakeResident2]->URI = LV2_UI__makeSONameResident;
  237. fFeatures[kFeatureIdUiMakeResident2]->data = nullptr;
  238. fFeatures[kFeatureIdUiNoUserResize]->URI = LV2_UI__noUserResize;
  239. fFeatures[kFeatureIdUiNoUserResize]->data = nullptr;
  240. fFeatures[kFeatureIdUiParent]->URI = LV2_UI__parent;
  241. fFeatures[kFeatureIdUiParent]->data = nullptr;
  242. fFeatures[kFeatureIdUiPortMap]->URI = LV2_UI__portMap;
  243. fFeatures[kFeatureIdUiPortMap]->data = uiPortMapFt;
  244. fFeatures[kFeatureIdUiPortSubscribe]->URI = LV2_UI__portSubscribe;
  245. fFeatures[kFeatureIdUiPortSubscribe]->data = nullptr;
  246. fFeatures[kFeatureIdUiResize]->URI = LV2_UI__resize;
  247. fFeatures[kFeatureIdUiResize]->data = uiResizeFt;
  248. fFeatures[kFeatureIdUiTouch]->URI = LV2_UI__touch;
  249. fFeatures[kFeatureIdUiTouch]->data = nullptr;
  250. }
  251. ~CarlaLv2Client() override
  252. {
  253. if (fHandle != nullptr && fDescriptor != nullptr && fDescriptor->cleanup != nullptr)
  254. {
  255. fDescriptor->cleanup(fHandle);
  256. fHandle = nullptr;
  257. }
  258. if (fRdfDescriptor != nullptr)
  259. {
  260. delete fRdfDescriptor;
  261. fRdfDescriptor = nullptr;
  262. }
  263. fRdfUiDescriptor = nullptr;
  264. delete (LV2_Log_Log*)fFeatures[kFeatureIdLogs]->data;
  265. delete (LV2_State_Make_Path*)fFeatures[kFeatureIdStateMakePath]->data;
  266. delete (LV2_State_Map_Path*)fFeatures[kFeatureIdStateMapPath]->data;
  267. delete (LV2_Programs_Host*)fFeatures[kFeatureIdPrograms]->data;
  268. delete (LV2_URI_Map_Feature*)fFeatures[kFeatureIdUriMap]->data;
  269. delete (LV2_URID_Map*)fFeatures[kFeatureIdUridMap]->data;
  270. delete (LV2_URID_Unmap*)fFeatures[kFeatureIdUridUnmap]->data;
  271. delete (LV2UI_Port_Map*)fFeatures[kFeatureIdUiPortMap]->data;
  272. delete (LV2UI_Resize*)fFeatures[kFeatureIdUiResize]->data;
  273. for (uint32_t i=0; i < kFeatureCount; ++i)
  274. {
  275. if (fFeatures[i] != nullptr)
  276. {
  277. delete fFeatures[i];
  278. fFeatures[i] = nullptr;
  279. }
  280. }
  281. }
  282. // ----------------------------------------------------------------------------------------------------------------
  283. // UI initialization
  284. bool init(const int argc, const char* argv[]) override
  285. {
  286. const char* pluginURI = argv[1];
  287. const char* uiURI = argc > 2 ? argv[2] : nullptr;
  288. // ------------------------------------------------------------------------------------------------------------
  289. // load plugin
  290. Lv2WorldClass& lv2World(Lv2WorldClass::getInstance());
  291. lv2World.initIfNeeded(std::getenv("LV2_PATH"));
  292. #if 0
  293. Lilv::Node bundleNode(lv2World.new_file_uri(nullptr, uiBundle));
  294. CARLA_SAFE_ASSERT_RETURN(bundleNode.is_uri(), false);
  295. CarlaString sBundle(bundleNode.as_uri());
  296. if (! sBundle.endsWith("/"))
  297. sBundle += "/";
  298. lv2World.load_bundle(sBundle);
  299. #endif
  300. // ------------------------------------------------------------------------------------------------------------
  301. // get plugin from lv2_rdf (lilv)
  302. fRdfDescriptor = lv2_rdf_new(pluginURI, false);
  303. CARLA_SAFE_ASSERT_RETURN(fRdfDescriptor != nullptr, false);
  304. // ------------------------------------------------------------------------------------------------------------
  305. // find requested UI
  306. if (uiURI == nullptr)
  307. {
  308. CARLA_SAFE_ASSERT_RETURN(fRdfDescriptor->UICount > 0, false);
  309. fRdfUiDescriptor = &fRdfDescriptor->UIs[0];
  310. uiURI = fRdfUiDescriptor->URI;
  311. }
  312. else
  313. {
  314. for (uint32_t i=0; i < fRdfDescriptor->UICount; ++i)
  315. {
  316. if (std::strcmp(fRdfDescriptor->UIs[i].URI, uiURI) == 0)
  317. {
  318. fRdfUiDescriptor = &fRdfDescriptor->UIs[i];
  319. break;
  320. }
  321. }
  322. }
  323. CARLA_SAFE_ASSERT_RETURN(fRdfUiDescriptor != nullptr, false);
  324. // ------------------------------------------------------------------------------------------------------------
  325. // check if not resizable
  326. for (uint32_t i=0; i < fRdfUiDescriptor->FeatureCount; ++i)
  327. {
  328. if (std::strcmp(fRdfUiDescriptor->Features[i].URI, LV2_UI__fixedSize ) == 0 ||
  329. std::strcmp(fRdfUiDescriptor->Features[i].URI, LV2_UI__noUserResize) == 0)
  330. {
  331. fUiOptions.isResizable = false;
  332. break;
  333. }
  334. }
  335. // ------------------------------------------------------------------------------------------------------------
  336. // init UI
  337. if (! CarlaBridgeFormat::init(argc, argv))
  338. return false;
  339. // ------------------------------------------------------------------------------------------------------------
  340. // open DLL
  341. if (! libOpen(fRdfUiDescriptor->Binary))
  342. {
  343. carla_stderr("Failed to load UI binary, error was:\n%s", libError());
  344. return false;
  345. }
  346. // ------------------------------------------------------------------------------------------------------------
  347. // get DLL main entry
  348. const LV2UI_DescriptorFunction ui_descFn = (LV2UI_DescriptorFunction)libSymbol("lv2ui_descriptor");
  349. if (ui_descFn == nullptr)
  350. return false;
  351. // ------------------------------------------------------------------------------------------------------------
  352. // get descriptor that matches URI
  353. for (uint32_t i=0; (fDescriptor = ui_descFn(i++)) != nullptr;)
  354. {
  355. if (std::strcmp(fDescriptor->URI, uiURI) == 0)
  356. break;
  357. }
  358. if (fDescriptor == nullptr)
  359. {
  360. carla_stderr("Failed to find UI descriptor");
  361. return false;
  362. }
  363. // ------------------------------------------------------------------------------------------------------------
  364. // initialize UI
  365. #if defined(BRIDGE_COCOA) || defined(BRIDGE_HWND) || defined(BRIDGE_X11)
  366. fFeatures[kFeatureIdUiParent]->data = fToolkit->getContainerId();
  367. #endif
  368. fHandle = fDescriptor->instantiate(fDescriptor, fRdfDescriptor->URI, fRdfUiDescriptor->Bundle,
  369. carla_lv2_ui_write_function, this, &fWidget, fFeatures);
  370. CARLA_SAFE_ASSERT_RETURN(fHandle != nullptr, false);
  371. #if defined(BRIDGE_COCOA) || defined(BRIDGE_HWND) || defined(BRIDGE_X11)
  372. if (fWidget != nullptr)
  373. fToolkit->setChildWindow(fWidget);
  374. #endif
  375. // ------------------------------------------------------------------------------------------------------------
  376. // check for known extensions
  377. if (fDescriptor->extension_data != nullptr)
  378. {
  379. fExt.options = (const LV2_Options_Interface*)fDescriptor->extension_data(LV2_OPTIONS__interface);
  380. fExt.programs = (const LV2_Programs_UI_Interface*)fDescriptor->extension_data(LV2_PROGRAMS__UIInterface);
  381. fExt.idle = (const LV2UI_Idle_Interface*)fDescriptor->extension_data(LV2_UI__idleInterface);
  382. fExt.resize = (const LV2UI_Resize*)fDescriptor->extension_data(LV2_UI__resize);
  383. // check if invalid
  384. if (fExt.programs != nullptr && fExt.programs->select_program == nullptr)
  385. fExt.programs = nullptr;
  386. if (fExt.idle != nullptr && fExt.idle->idle == nullptr)
  387. fExt.idle = nullptr;
  388. if (fExt.resize != nullptr && fExt.resize->ui_resize == nullptr)
  389. fExt.resize = nullptr;
  390. }
  391. for (uint32_t i=0; i<fRdfDescriptor->PortCount; ++i)
  392. {
  393. if (LV2_IS_PORT_DESIGNATION_CONTROL(fRdfDescriptor->Ports[i].Designation))
  394. {
  395. fControlDesignatedPort = i;
  396. break;
  397. }
  398. }
  399. return true;
  400. }
  401. void idleUI() override
  402. {
  403. #if defined(BRIDGE_COCOA) || defined(BRIDGE_HWND) || defined(BRIDGE_X11)
  404. if (fHandle != nullptr && fExt.idle != nullptr && fExt.idle->idle(fHandle) != 0)
  405. {
  406. if (isPipeRunning() && ! fQuitReceived)
  407. writeExitingMessageAndWait();
  408. }
  409. #endif
  410. }
  411. // ----------------------------------------------------------------------------------------------------------------
  412. // UI management
  413. void* getWidget() const noexcept override
  414. {
  415. return fWidget;
  416. }
  417. const Options& getOptions() const noexcept override
  418. {
  419. return fUiOptions;
  420. }
  421. // ----------------------------------------------------------------------------------------------------------------
  422. // DSP Callbacks
  423. void dspParameterChanged(const uint32_t index, const float value) override
  424. {
  425. CARLA_SAFE_ASSERT_RETURN(fHandle != nullptr,)
  426. CARLA_SAFE_ASSERT_RETURN(fDescriptor != nullptr,);
  427. if (fDescriptor->port_event == nullptr)
  428. return;
  429. fDescriptor->port_event(fHandle, index, sizeof(float), kUridNull, &value);
  430. }
  431. void dspProgramChanged(const uint32_t index) override
  432. {
  433. carla_stderr2("dspProgramChanged(%i) - not handled", index);
  434. }
  435. void dspMidiProgramChanged(const uint32_t bank, const uint32_t program) override
  436. {
  437. CARLA_SAFE_ASSERT_RETURN(fHandle != nullptr,)
  438. if (fExt.programs == nullptr)
  439. return;
  440. fExt.programs->select_program(fHandle, bank, program);
  441. }
  442. void dspStateChanged(const char* const, const char* const) override
  443. {
  444. }
  445. void dspNoteReceived(const bool onOff, const uint8_t channel, const uint8_t note, const uint8_t velocity) override
  446. {
  447. CARLA_SAFE_ASSERT_RETURN(fHandle != nullptr,)
  448. CARLA_SAFE_ASSERT_RETURN(fDescriptor != nullptr,);
  449. if (fDescriptor->port_event == nullptr)
  450. return;
  451. LV2_Atom_MidiEvent midiEv;
  452. midiEv.atom.type = kUridMidiEvent;
  453. midiEv.atom.size = 3;
  454. midiEv.data[0] = uint8_t((onOff ? MIDI_STATUS_NOTE_ON : MIDI_STATUS_NOTE_OFF) | (channel & MIDI_CHANNEL_BIT));
  455. midiEv.data[1] = note;
  456. midiEv.data[2] = velocity;
  457. fDescriptor->port_event(fHandle, fControlDesignatedPort, lv2_atom_total_size(midiEv), kUridAtomTransferEvent, &midiEv);
  458. }
  459. void dspAtomReceived(const uint32_t portIndex, const LV2_Atom* const atom) override
  460. {
  461. CARLA_SAFE_ASSERT_RETURN(fHandle != nullptr,);
  462. CARLA_SAFE_ASSERT_RETURN(fDescriptor != nullptr,);
  463. CARLA_SAFE_ASSERT_RETURN(atom != nullptr,);
  464. if (fDescriptor->port_event == nullptr)
  465. return;
  466. fDescriptor->port_event(fHandle, portIndex, lv2_atom_total_size(atom), kUridAtomTransferEvent, atom);
  467. }
  468. void dspURIDReceived(const LV2_URID urid, const char* const uri) override
  469. {
  470. CARLA_SAFE_ASSERT_RETURN(urid == fCustomURIDs.size(),);
  471. CARLA_SAFE_ASSERT_RETURN(uri != nullptr && uri[0] != '\0',);
  472. fCustomURIDs.push_back(uri);
  473. }
  474. void uiOptionsChanged(const double sampleRate, const bool useTheme, const bool useThemeColors, const char* const windowTitle, uintptr_t transientWindowId) override
  475. {
  476. carla_debug("CarlaLv2Client::uiOptionsChanged(%g, %s, %s, \"%s\", " P_UINTPTR ")", sampleRate, bool2str(useTheme), bool2str(useThemeColors), windowTitle, transientWindowId);
  477. const float sampleRatef = static_cast<float>(sampleRate);
  478. if (carla_isNotEqual(fLv2Options.sampleRate, sampleRatef))
  479. {
  480. fLv2Options.sampleRate = sampleRatef;
  481. if (fExt.options != nullptr && fExt.options->set != nullptr)
  482. {
  483. LV2_Options_Option options[2];
  484. carla_zeroStructs(options, 2);
  485. LV2_Options_Option& optSampleRate(options[0]);
  486. optSampleRate.context = LV2_OPTIONS_INSTANCE;
  487. optSampleRate.subject = 0;
  488. optSampleRate.key = kUridParamSampleRate;
  489. optSampleRate.size = sizeof(float);
  490. optSampleRate.type = kUridAtomFloat;
  491. optSampleRate.value = &fLv2Options.sampleRate;
  492. fExt.options->set(fHandle, options);
  493. }
  494. }
  495. if (fLv2Options.windowTitle != nullptr && fLv2Options.windowTitle != kNullWindowTitle)
  496. delete[] fLv2Options.windowTitle;
  497. fLv2Options.transientWinId = static_cast<int64_t>(transientWindowId);
  498. fLv2Options.windowTitle = carla_strdup_safe(windowTitle);
  499. if (fLv2Options.windowTitle == nullptr)
  500. fLv2Options.windowTitle = kNullWindowTitle;
  501. else
  502. fUiOptions.windowTitle = fLv2Options.windowTitle;
  503. fLv2Options.opts[Lv2PluginOptions::WindowTitle].size = (uint32_t)std::strlen(fLv2Options.windowTitle);
  504. fLv2Options.opts[Lv2PluginOptions::WindowTitle].value = fLv2Options.windowTitle;
  505. fUiOptions.useTheme = useTheme;
  506. fUiOptions.useThemeColors = useThemeColors;
  507. fUiOptions.transientWindowId = transientWindowId;
  508. }
  509. void uiResized(const uint width, const uint height) override
  510. {
  511. if (fHandle != nullptr && fExt.resize != nullptr)
  512. fExt.resize->ui_resize(fHandle, static_cast<int>(width), static_cast<int>(height));
  513. }
  514. // ----------------------------------------------------------------------------------------------------------------
  515. LV2_URID getCustomURID(const char* const uri)
  516. {
  517. CARLA_SAFE_ASSERT_RETURN(uri != nullptr && uri[0] != '\0', kUridNull);
  518. carla_debug("CarlaLv2Client::getCustomURID(\"%s\")", uri);
  519. const std::string s_uri(uri);
  520. const std::ptrdiff_t s_pos(std::find(fCustomURIDs.begin(), fCustomURIDs.end(), s_uri) - fCustomURIDs.begin());
  521. if (s_pos <= 0 || s_pos >= INT32_MAX)
  522. return kUridNull;
  523. const LV2_URID urid = static_cast<LV2_URID>(s_pos);
  524. const LV2_URID uriCount = static_cast<LV2_URID>(fCustomURIDs.size());
  525. if (urid < uriCount)
  526. return urid;
  527. CARLA_SAFE_ASSERT(urid == uriCount);
  528. fCustomURIDs.push_back(uri);
  529. if (isPipeRunning())
  530. writeLv2UridMessage(urid, uri);
  531. return urid;
  532. }
  533. const char* getCustomURIDString(const LV2_URID urid) const noexcept
  534. {
  535. static const char* const sFallback = "urn:null";
  536. CARLA_SAFE_ASSERT_RETURN(urid != kUridNull, sFallback);
  537. CARLA_SAFE_ASSERT_RETURN(urid < fCustomURIDs.size(), sFallback);
  538. carla_debug("CarlaLv2Client::getCustomURIDString(%i)", urid);
  539. return fCustomURIDs[urid].c_str();
  540. }
  541. // ----------------------------------------------------------------------------------------------------------------
  542. void handleProgramChanged(const int32_t index)
  543. {
  544. if (isPipeRunning())
  545. writeReloadProgramsMessage(index);
  546. }
  547. uint32_t handleUiPortMap(const char* const symbol)
  548. {
  549. CARLA_SAFE_ASSERT_RETURN(symbol != nullptr && symbol[0] != '\0', LV2UI_INVALID_PORT_INDEX);
  550. carla_debug("CarlaLv2Client::handleUiPortMap(\"%s\")", symbol);
  551. for (uint32_t i=0; i < fRdfDescriptor->PortCount; ++i)
  552. {
  553. if (std::strcmp(fRdfDescriptor->Ports[i].Symbol, symbol) == 0)
  554. return i;
  555. }
  556. return LV2UI_INVALID_PORT_INDEX;
  557. }
  558. int handleUiResize(const int width, const int height)
  559. {
  560. CARLA_SAFE_ASSERT_RETURN(fToolkit != nullptr, 1);
  561. CARLA_SAFE_ASSERT_RETURN(width > 0, 1);
  562. CARLA_SAFE_ASSERT_RETURN(height > 0, 1);
  563. carla_debug("CarlaLv2Client::handleUiResize(%i, %i)", width, height);
  564. fToolkit->setSize(static_cast<uint>(width), static_cast<uint>(height));
  565. return 0;
  566. }
  567. void handleUiWrite(uint32_t rindex, uint32_t bufferSize, uint32_t format, const void* buffer)
  568. {
  569. CARLA_SAFE_ASSERT_RETURN(buffer != nullptr,);
  570. CARLA_SAFE_ASSERT_RETURN(bufferSize > 0,);
  571. carla_debug("CarlaLv2Client::handleUiWrite(%i, %i, %i, %p)", rindex, bufferSize, format, buffer);
  572. switch (format)
  573. {
  574. case kUridNull:
  575. CARLA_SAFE_ASSERT_RETURN(bufferSize == sizeof(float),);
  576. if (isPipeRunning())
  577. {
  578. const float value(*(const float*)buffer);
  579. writeControlMessage(rindex, value);
  580. }
  581. break;
  582. case kUridAtomTransferAtom:
  583. case kUridAtomTransferEvent:
  584. CARLA_SAFE_ASSERT_RETURN(bufferSize >= sizeof(LV2_Atom),);
  585. if (isPipeRunning())
  586. {
  587. const LV2_Atom* const atom((const LV2_Atom*)buffer);
  588. // plugins sometimes fail on this, not good...
  589. const uint32_t totalSize = lv2_atom_total_size(atom);
  590. const uint32_t paddedSize = lv2_atom_pad_size(totalSize);
  591. if (bufferSize != totalSize && bufferSize != paddedSize)
  592. carla_stderr2("Warning: LV2 UI sending atom with invalid size! size: %u, padded-size: %u", totalSize, paddedSize);
  593. writeLv2AtomMessage(rindex, atom);
  594. }
  595. break;
  596. default:
  597. carla_stderr("CarlaLv2Client::handleUiWrite(%i, %i, %i:\"%s\", %p) - unknown format",
  598. rindex, bufferSize, format, carla_lv2_urid_unmap(this, format), buffer);
  599. break;
  600. }
  601. }
  602. // ----------------------------------------------------------------------------------------------------------------
  603. private:
  604. LV2UI_Handle fHandle;
  605. LV2UI_Widget fWidget;
  606. LV2_Feature* fFeatures[kFeatureCount+1];
  607. const LV2UI_Descriptor* fDescriptor;
  608. const LV2_RDF_Descriptor* fRdfDescriptor;
  609. const LV2_RDF_UI* fRdfUiDescriptor;
  610. uint32_t fControlDesignatedPort;
  611. Lv2PluginOptions fLv2Options;
  612. Options fUiOptions;
  613. std::vector<std::string> fCustomURIDs;
  614. struct Extensions {
  615. const LV2_Options_Interface* options;
  616. const LV2_Programs_UI_Interface* programs;
  617. const LV2UI_Idle_Interface* idle;
  618. const LV2UI_Resize* resize;
  619. Extensions()
  620. : options(nullptr),
  621. programs(nullptr),
  622. idle(nullptr),
  623. resize(nullptr) {}
  624. } fExt;
  625. // ----------------------------------------------------------------------------------------------------------------
  626. // Logs Feature
  627. static int carla_lv2_log_printf(LV2_Log_Handle handle, LV2_URID type, const char* fmt, ...)
  628. {
  629. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, 0);
  630. CARLA_SAFE_ASSERT_RETURN(type != kUridNull, 0);
  631. CARLA_SAFE_ASSERT_RETURN(fmt != nullptr, 0);
  632. #ifndef DEBUG
  633. if (type == kUridLogTrace)
  634. return 0;
  635. #endif
  636. va_list args;
  637. va_start(args, fmt);
  638. const int ret(carla_lv2_log_vprintf(handle, type, fmt, args));
  639. va_end(args);
  640. return ret;
  641. }
  642. static int carla_lv2_log_vprintf(LV2_Log_Handle handle, LV2_URID type, const char* fmt, va_list ap)
  643. {
  644. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, 0);
  645. CARLA_SAFE_ASSERT_RETURN(type != kUridNull, 0);
  646. CARLA_SAFE_ASSERT_RETURN(fmt != nullptr, 0);
  647. int ret = 0;
  648. switch (type)
  649. {
  650. case kUridLogError:
  651. std::fprintf(stderr, "\x1b[31m");
  652. ret = std::vfprintf(stderr, fmt, ap);
  653. std::fprintf(stderr, "\x1b[0m");
  654. break;
  655. case kUridLogNote:
  656. ret = std::vfprintf(stdout, fmt, ap);
  657. break;
  658. case kUridLogTrace:
  659. #ifdef DEBUG
  660. std::fprintf(stdout, "\x1b[30;1m");
  661. ret = std::vfprintf(stdout, fmt, ap);
  662. std::fprintf(stdout, "\x1b[0m");
  663. #endif
  664. break;
  665. case kUridLogWarning:
  666. ret = std::vfprintf(stderr, fmt, ap);
  667. break;
  668. default:
  669. break;
  670. }
  671. return ret;
  672. }
  673. // ----------------------------------------------------------------------------------------------------------------
  674. // Programs Feature
  675. static void carla_lv2_program_changed(LV2_Programs_Handle handle, int32_t index)
  676. {
  677. CARLA_SAFE_ASSERT_RETURN(handle != nullptr,);
  678. carla_debug("carla_lv2_program_changed(%p, %i)", handle, index);
  679. ((CarlaLv2Client*)handle)->handleProgramChanged(index);
  680. }
  681. // ----------------------------------------------------------------------------------------------------------------
  682. // State Feature
  683. static char* carla_lv2_state_make_path(LV2_State_Make_Path_Handle handle, const char* path)
  684. {
  685. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, nullptr);
  686. CARLA_SAFE_ASSERT_RETURN(path != nullptr && path[0] != '\0', nullptr);
  687. carla_debug("carla_lv2_state_make_path(%p, \"%s\")", handle, path);
  688. File file;
  689. if (File::isAbsolutePath(path))
  690. file = File(path);
  691. else
  692. file = File::getCurrentWorkingDirectory().getChildFile(path);
  693. file.getParentDirectory().createDirectory();
  694. return strdup(file.getFullPathName().toRawUTF8());
  695. }
  696. static char* carla_lv2_state_map_abstract_path(LV2_State_Map_Path_Handle handle, const char* absolute_path)
  697. {
  698. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, strdup(""));
  699. CARLA_SAFE_ASSERT_RETURN(absolute_path != nullptr && absolute_path[0] != '\0', strdup(""));
  700. carla_debug("carla_lv2_state_map_abstract_path(%p, \"%s\")", handle, absolute_path);
  701. // may already be an abstract path
  702. if (! File::isAbsolutePath(absolute_path))
  703. return strdup(absolute_path);
  704. return strdup(File(absolute_path).getRelativePathFrom(File::getCurrentWorkingDirectory()).toRawUTF8());
  705. }
  706. static char* carla_lv2_state_map_absolute_path(LV2_State_Map_Path_Handle handle, const char* abstract_path)
  707. {
  708. const char* const cwd(File::getCurrentWorkingDirectory().getFullPathName().toRawUTF8());
  709. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, strdup(cwd));
  710. CARLA_SAFE_ASSERT_RETURN(abstract_path != nullptr && abstract_path[0] != '\0', strdup(cwd));
  711. carla_debug("carla_lv2_state_map_absolute_path(%p, \"%s\")", handle, abstract_path);
  712. // may already be an absolute path
  713. if (File::isAbsolutePath(abstract_path))
  714. return strdup(abstract_path);
  715. return strdup(File::getCurrentWorkingDirectory().getChildFile(abstract_path).getFullPathName().toRawUTF8());
  716. }
  717. // ----------------------------------------------------------------------------------------------------------------
  718. // URI-Map Feature
  719. static uint32_t carla_lv2_uri_to_id(LV2_URI_Map_Callback_Data data, const char* map, const char* uri)
  720. {
  721. carla_debug("carla_lv2_uri_to_id(%p, \"%s\", \"%s\")", data, map, uri);
  722. return carla_lv2_urid_map((LV2_URID_Map_Handle*)data, uri);
  723. // unused
  724. (void)map;
  725. }
  726. // ----------------------------------------------------------------------------------------------------------------
  727. // URID Feature
  728. static LV2_URID carla_lv2_urid_map(LV2_URID_Map_Handle handle, const char* uri)
  729. {
  730. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, kUridNull);
  731. CARLA_SAFE_ASSERT_RETURN(uri != nullptr && uri[0] != '\0', kUridNull);
  732. carla_debug("carla_lv2_urid_map(%p, \"%s\")", handle, uri);
  733. // Atom types
  734. if (std::strcmp(uri, LV2_ATOM__Blank) == 0)
  735. return kUridAtomBlank;
  736. if (std::strcmp(uri, LV2_ATOM__Bool) == 0)
  737. return kUridAtomBool;
  738. if (std::strcmp(uri, LV2_ATOM__Chunk) == 0)
  739. return kUridAtomChunk;
  740. if (std::strcmp(uri, LV2_ATOM__Double) == 0)
  741. return kUridAtomDouble;
  742. if (std::strcmp(uri, LV2_ATOM__Event) == 0)
  743. return kUridAtomEvent;
  744. if (std::strcmp(uri, LV2_ATOM__Float) == 0)
  745. return kUridAtomFloat;
  746. if (std::strcmp(uri, LV2_ATOM__Int) == 0)
  747. return kUridAtomInt;
  748. if (std::strcmp(uri, LV2_ATOM__Literal) == 0)
  749. return kUridAtomLiteral;
  750. if (std::strcmp(uri, LV2_ATOM__Long) == 0)
  751. return kUridAtomLong;
  752. if (std::strcmp(uri, LV2_ATOM__Number) == 0)
  753. return kUridAtomNumber;
  754. if (std::strcmp(uri, LV2_ATOM__Object) == 0)
  755. return kUridAtomObject;
  756. if (std::strcmp(uri, LV2_ATOM__Path) == 0)
  757. return kUridAtomPath;
  758. if (std::strcmp(uri, LV2_ATOM__Property) == 0)
  759. return kUridAtomProperty;
  760. if (std::strcmp(uri, LV2_ATOM__Resource) == 0)
  761. return kUridAtomResource;
  762. if (std::strcmp(uri, LV2_ATOM__Sequence) == 0)
  763. return kUridAtomSequence;
  764. if (std::strcmp(uri, LV2_ATOM__Sound) == 0)
  765. return kUridAtomSound;
  766. if (std::strcmp(uri, LV2_ATOM__String) == 0)
  767. return kUridAtomString;
  768. if (std::strcmp(uri, LV2_ATOM__Tuple) == 0)
  769. return kUridAtomTuple;
  770. if (std::strcmp(uri, LV2_ATOM__URI) == 0)
  771. return kUridAtomURI;
  772. if (std::strcmp(uri, LV2_ATOM__URID) == 0)
  773. return kUridAtomURID;
  774. if (std::strcmp(uri, LV2_ATOM__Vector) == 0)
  775. return kUridAtomVector;
  776. if (std::strcmp(uri, LV2_ATOM__atomTransfer) == 0)
  777. return kUridAtomTransferAtom;
  778. if (std::strcmp(uri, LV2_ATOM__eventTransfer) == 0)
  779. return kUridAtomTransferEvent;
  780. // BufSize types
  781. if (std::strcmp(uri, LV2_BUF_SIZE__maxBlockLength) == 0)
  782. return kUridBufMaxLength;
  783. if (std::strcmp(uri, LV2_BUF_SIZE__minBlockLength) == 0)
  784. return kUridBufMinLength;
  785. if (std::strcmp(uri, LV2_BUF_SIZE__nominalBlockLength) == 0)
  786. return kUridBufNominalLength;
  787. if (std::strcmp(uri, LV2_BUF_SIZE__sequenceSize) == 0)
  788. return kUridBufSequenceSize;
  789. // Log types
  790. if (std::strcmp(uri, LV2_LOG__Error) == 0)
  791. return kUridLogError;
  792. if (std::strcmp(uri, LV2_LOG__Note) == 0)
  793. return kUridLogNote;
  794. if (std::strcmp(uri, LV2_LOG__Trace) == 0)
  795. return kUridLogTrace;
  796. if (std::strcmp(uri, LV2_LOG__Warning) == 0)
  797. return kUridLogWarning;
  798. // Time types
  799. if (std::strcmp(uri, LV2_TIME__Position) == 0)
  800. return kUridTimePosition;
  801. if (std::strcmp(uri, LV2_TIME__bar) == 0)
  802. return kUridTimeBar;
  803. if (std::strcmp(uri, LV2_TIME__barBeat) == 0)
  804. return kUridTimeBarBeat;
  805. if (std::strcmp(uri, LV2_TIME__beat) == 0)
  806. return kUridTimeBeat;
  807. if (std::strcmp(uri, LV2_TIME__beatUnit) == 0)
  808. return kUridTimeBeatUnit;
  809. if (std::strcmp(uri, LV2_TIME__beatsPerBar) == 0)
  810. return kUridTimeBeatsPerBar;
  811. if (std::strcmp(uri, LV2_TIME__beatsPerMinute) == 0)
  812. return kUridTimeBeatsPerMinute;
  813. if (std::strcmp(uri, LV2_TIME__frame) == 0)
  814. return kUridTimeFrame;
  815. if (std::strcmp(uri, LV2_TIME__framesPerSecond) == 0)
  816. return kUridTimeFramesPerSecond;
  817. if (std::strcmp(uri, LV2_TIME__speed) == 0)
  818. return kUridTimeSpeed;
  819. if (std::strcmp(uri, LV2_KXSTUDIO_PROPERTIES__TimePositionTicksPerBeat) == 0)
  820. return kUridTimeTicksPerBeat;
  821. // Others
  822. if (std::strcmp(uri, LV2_MIDI__MidiEvent) == 0)
  823. return kUridMidiEvent;
  824. if (std::strcmp(uri, LV2_PARAMETERS__sampleRate) == 0)
  825. return kUridParamSampleRate;
  826. if (std::strcmp(uri, LV2_UI__windowTitle) == 0)
  827. return kUridWindowTitle;
  828. // Custom Carla types
  829. if (std::strcmp(uri, URI_CARLA_ATOM_WORKER_IN) == 0)
  830. return kUridCarlaAtomWorkerIn;
  831. if (std::strcmp(uri, URI_CARLA_ATOM_WORKER_RESP) == 0)
  832. return kUridCarlaAtomWorkerResp;
  833. if (std::strcmp(uri, LV2_KXSTUDIO_PROPERTIES__TransientWindowId) == 0)
  834. return kUridCarlaTransientWindowId;
  835. // Custom plugin types
  836. return ((CarlaLv2Client*)handle)->getCustomURID(uri);
  837. }
  838. static const char* carla_lv2_urid_unmap(LV2_URID_Map_Handle handle, LV2_URID urid)
  839. {
  840. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, nullptr);
  841. CARLA_SAFE_ASSERT_RETURN(urid != kUridNull, nullptr);
  842. carla_debug("carla_lv2_urid_unmap(%p, %i)", handle, urid);
  843. switch (urid)
  844. {
  845. // Atom types
  846. case kUridAtomBlank:
  847. return LV2_ATOM__Blank;
  848. case kUridAtomBool:
  849. return LV2_ATOM__Bool;
  850. case kUridAtomChunk:
  851. return LV2_ATOM__Chunk;
  852. case kUridAtomDouble:
  853. return LV2_ATOM__Double;
  854. case kUridAtomEvent:
  855. return LV2_ATOM__Event;
  856. case kUridAtomFloat:
  857. return LV2_ATOM__Float;
  858. case kUridAtomInt:
  859. return LV2_ATOM__Int;
  860. case kUridAtomLiteral:
  861. return LV2_ATOM__Literal;
  862. case kUridAtomLong:
  863. return LV2_ATOM__Long;
  864. case kUridAtomNumber:
  865. return LV2_ATOM__Number;
  866. case kUridAtomObject:
  867. return LV2_ATOM__Object;
  868. case kUridAtomPath:
  869. return LV2_ATOM__Path;
  870. case kUridAtomProperty:
  871. return LV2_ATOM__Property;
  872. case kUridAtomResource:
  873. return LV2_ATOM__Resource;
  874. case kUridAtomSequence:
  875. return LV2_ATOM__Sequence;
  876. case kUridAtomSound:
  877. return LV2_ATOM__Sound;
  878. case kUridAtomString:
  879. return LV2_ATOM__String;
  880. case kUridAtomTuple:
  881. return LV2_ATOM__Tuple;
  882. case kUridAtomURI:
  883. return LV2_ATOM__URI;
  884. case kUridAtomURID:
  885. return LV2_ATOM__URID;
  886. case kUridAtomVector:
  887. return LV2_ATOM__Vector;
  888. case kUridAtomTransferAtom:
  889. return LV2_ATOM__atomTransfer;
  890. case kUridAtomTransferEvent:
  891. return LV2_ATOM__eventTransfer;
  892. // BufSize types
  893. case kUridBufMaxLength:
  894. return LV2_BUF_SIZE__maxBlockLength;
  895. case kUridBufMinLength:
  896. return LV2_BUF_SIZE__minBlockLength;
  897. case kUridBufNominalLength:
  898. return LV2_BUF_SIZE__nominalBlockLength;
  899. case kUridBufSequenceSize:
  900. return LV2_BUF_SIZE__sequenceSize;
  901. // Log types
  902. case kUridLogError:
  903. return LV2_LOG__Error;
  904. case kUridLogNote:
  905. return LV2_LOG__Note;
  906. case kUridLogTrace:
  907. return LV2_LOG__Trace;
  908. case kUridLogWarning:
  909. return LV2_LOG__Warning;
  910. // Time types
  911. case kUridTimePosition:
  912. return LV2_TIME__Position;
  913. case kUridTimeBar:
  914. return LV2_TIME__bar;
  915. case kUridTimeBarBeat:
  916. return LV2_TIME__barBeat;
  917. case kUridTimeBeat:
  918. return LV2_TIME__beat;
  919. case kUridTimeBeatUnit:
  920. return LV2_TIME__beatUnit;
  921. case kUridTimeBeatsPerBar:
  922. return LV2_TIME__beatsPerBar;
  923. case kUridTimeBeatsPerMinute:
  924. return LV2_TIME__beatsPerMinute;
  925. case kUridTimeFrame:
  926. return LV2_TIME__frame;
  927. case kUridTimeFramesPerSecond:
  928. return LV2_TIME__framesPerSecond;
  929. case kUridTimeSpeed:
  930. return LV2_TIME__speed;
  931. case kUridTimeTicksPerBeat:
  932. return LV2_KXSTUDIO_PROPERTIES__TimePositionTicksPerBeat;
  933. // Others
  934. case kUridMidiEvent:
  935. return LV2_MIDI__MidiEvent;
  936. case kUridParamSampleRate:
  937. return LV2_PARAMETERS__sampleRate;
  938. case kUridWindowTitle:
  939. return LV2_UI__windowTitle;
  940. // Custom Carla types
  941. case kUridCarlaAtomWorkerIn:
  942. return URI_CARLA_ATOM_WORKER_IN;
  943. case kUridCarlaAtomWorkerResp:
  944. return URI_CARLA_ATOM_WORKER_RESP;
  945. case kUridCarlaTransientWindowId:
  946. return LV2_KXSTUDIO_PROPERTIES__TransientWindowId;
  947. }
  948. // Custom types
  949. return ((CarlaLv2Client*)handle)->getCustomURIDString(urid);
  950. }
  951. // ----------------------------------------------------------------------------------------------------------------
  952. // UI Port-Map Feature
  953. static uint32_t carla_lv2_ui_port_map(LV2UI_Feature_Handle handle, const char* symbol)
  954. {
  955. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, LV2UI_INVALID_PORT_INDEX);
  956. carla_debug("carla_lv2_ui_port_map(%p, \"%s\")", handle, symbol);
  957. return ((CarlaLv2Client*)handle)->handleUiPortMap(symbol);
  958. }
  959. // ----------------------------------------------------------------------------------------------------------------
  960. // UI Resize Feature
  961. static int carla_lv2_ui_resize(LV2UI_Feature_Handle handle, int width, int height)
  962. {
  963. CARLA_SAFE_ASSERT_RETURN(handle != nullptr, 1);
  964. carla_debug("carla_lv2_ui_resize(%p, %i, %i)", handle, width, height);
  965. return ((CarlaLv2Client*)handle)->handleUiResize(width, height);
  966. }
  967. // ----------------------------------------------------------------------------------------------------------------
  968. // UI Extension
  969. static void carla_lv2_ui_write_function(LV2UI_Controller controller, uint32_t port_index, uint32_t buffer_size, uint32_t format, const void* buffer)
  970. {
  971. CARLA_SAFE_ASSERT_RETURN(controller != nullptr,);
  972. carla_debug("carla_lv2_ui_write_function(%p, %i, %i, %i, %p)", controller, port_index, buffer_size, format, buffer);
  973. ((CarlaLv2Client*)controller)->handleUiWrite(port_index, buffer_size, format, buffer);
  974. }
  975. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaLv2Client)
  976. };
  977. // --------------------------------------------------------------------------------------------------------------------
  978. CARLA_BRIDGE_UI_END_NAMESPACE
  979. // --------------------------------------------------------------------------------------------------------------------
  980. int main(int argc, const char* argv[])
  981. {
  982. CARLA_BRIDGE_UI_USE_NAMESPACE
  983. if (argc < 2)
  984. {
  985. carla_stderr("usage: %s <plugin-uri> [ui-uri]", argv[0]);
  986. return 1;
  987. }
  988. const bool testingModeOnly = (argc != 7);
  989. // try to get sampleRate value
  990. if (const char* const sampleRateStr = std::getenv("CARLA_SAMPLE_RATE"))
  991. gInitialSampleRate = std::atof(sampleRateStr);
  992. // Init LV2 client
  993. CarlaLv2Client client;
  994. // Load UI
  995. int ret;
  996. if (client.init(argc, argv))
  997. {
  998. client.exec(testingModeOnly);
  999. ret = 0;
  1000. }
  1001. else
  1002. {
  1003. ret = 1;
  1004. }
  1005. return ret;
  1006. }
  1007. // --------------------------------------------------------------------------------------------------------------------