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.

2382 lines
78KB

  1. /*
  2. * Carla Plugin Host
  3. * Copyright (C) 2011-2014 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 "CarlaEngineInternal.hpp"
  18. #include "CarlaPlugin.hpp"
  19. #include "CarlaBackendUtils.hpp"
  20. #include "CarlaEngineUtils.hpp"
  21. #include "CarlaMathUtils.hpp"
  22. #include "CarlaMIDI.h"
  23. #include "jackbridge/JackBridge.hpp"
  24. #include <QtCore/QStringList>
  25. #define URI_CANVAS_CV "http://kxstudio.sf.net/ns/canvas/cv"
  26. #define URI_CANVAS_ICON "http://kxstudio.sf.net/ns/canvas/icon"
  27. CARLA_BACKEND_START_NAMESPACE
  28. #if 0
  29. } // Fix editor indentation
  30. #endif
  31. class CarlaEngineJack;
  32. class CarlaEngineJackClient;
  33. // -----------------------------------------------------------------------
  34. // Fallback data
  35. static const EngineEvent kFallbackJackEngineEvent = { kEngineEventTypeNull, 0, 0, {{ kEngineControlEventTypeNull, 0, 0.0f }} };
  36. // -----------------------------------------------------------------------
  37. // Carla Engine Port removal helper
  38. class CarlaEngineJackAudioPort;
  39. class CarlaEngineJackCVPort;
  40. class CarlaEngineJackEventPort;
  41. struct JackPortDeletionCallback {
  42. virtual ~JackPortDeletionCallback() {}
  43. virtual void jackAudioPortDeleted(CarlaEngineJackAudioPort* const) = 0;
  44. virtual void jackCVPortDeleted(CarlaEngineJackCVPort* const) = 0;
  45. virtual void jackEventPortDeleted(CarlaEngineJackEventPort* const) = 0;
  46. };
  47. // -----------------------------------------------------------------------
  48. // Carla Engine JACK-Audio port
  49. class CarlaEngineJackAudioPort : public CarlaEngineAudioPort
  50. {
  51. public:
  52. CarlaEngineJackAudioPort(const CarlaEngineClient& client, const bool isInputPort, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  53. : CarlaEngineAudioPort(client, isInputPort),
  54. fJackClient(jackClient),
  55. fJackPort(jackPort),
  56. fDeletionCallback(delCallback)
  57. {
  58. carla_debug("CarlaEngineJackAudioPort::CarlaEngineJackAudioPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  59. switch (fClient.getEngine().getProccessMode())
  60. {
  61. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  62. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  63. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  64. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  65. jackbridge_set_property(jackClient, uuid, URI_CANVAS_CV, "NO", "text/plain");
  66. break;
  67. default:
  68. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  69. break;
  70. }
  71. }
  72. ~CarlaEngineJackAudioPort() noexcept override
  73. {
  74. carla_debug("CarlaEngineJackAudioPort::~CarlaEngineJackAudioPort()");
  75. if (fJackClient != nullptr && fJackPort != nullptr)
  76. {
  77. if (const jack_uuid_t uuid = jackbridge_port_uuid(fJackPort))
  78. jackbridge_remove_property(fJackClient, uuid, URI_CANVAS_CV);
  79. try {
  80. jackbridge_port_unregister(fJackClient, fJackPort);
  81. } catch(...) {}
  82. fJackClient = nullptr;
  83. fJackPort = nullptr;
  84. }
  85. if (fDeletionCallback != nullptr)
  86. fDeletionCallback->jackAudioPortDeleted(this);
  87. }
  88. void initBuffer() noexcept override
  89. {
  90. if (fJackPort == nullptr)
  91. return CarlaEngineAudioPort::initBuffer();
  92. const uint32_t bufferSize(fClient.getEngine().getBufferSize());
  93. try {
  94. fBuffer = (float*)jackbridge_port_get_buffer(fJackPort, bufferSize);
  95. }
  96. catch(...) {
  97. fBuffer = nullptr;
  98. return;
  99. }
  100. if (! fIsInput)
  101. FLOAT_CLEAR(fBuffer, bufferSize);
  102. }
  103. void invalidate() noexcept
  104. {
  105. fJackClient = nullptr;
  106. fJackPort = nullptr;
  107. }
  108. private:
  109. jack_client_t* fJackClient;
  110. jack_port_t* fJackPort;
  111. JackPortDeletionCallback* const fDeletionCallback;
  112. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackAudioPort)
  113. };
  114. // -----------------------------------------------------------------------
  115. // Carla Engine JACK-CV port
  116. class CarlaEngineJackCVPort : public CarlaEngineCVPort
  117. {
  118. public:
  119. CarlaEngineJackCVPort(const CarlaEngineClient& client, const bool isInputPort, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  120. : CarlaEngineCVPort(client, isInputPort),
  121. fJackClient(jackClient),
  122. fJackPort(jackPort),
  123. fDeletionCallback(delCallback)
  124. {
  125. carla_debug("CarlaEngineJackCVPort::CarlaEngineJackCVPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  126. switch (fClient.getEngine().getProccessMode())
  127. {
  128. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  129. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  130. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  131. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  132. jackbridge_set_property(jackClient, uuid, URI_CANVAS_CV, "YES", "text/plain");
  133. break;
  134. default:
  135. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  136. break;
  137. }
  138. }
  139. ~CarlaEngineJackCVPort() noexcept override
  140. {
  141. carla_debug("CarlaEngineJackCVPort::~CarlaEngineJackCVPort()");
  142. if (fJackClient != nullptr && fJackPort != nullptr)
  143. {
  144. if (const jack_uuid_t uuid = jackbridge_port_uuid(fJackPort))
  145. jackbridge_remove_property(fJackClient, uuid, URI_CANVAS_CV);
  146. try {
  147. jackbridge_port_unregister(fJackClient, fJackPort);
  148. } catch(...) {}
  149. fJackClient = nullptr;
  150. fJackPort = nullptr;
  151. }
  152. if (fDeletionCallback != nullptr)
  153. fDeletionCallback->jackCVPortDeleted(this);
  154. }
  155. void initBuffer() noexcept override
  156. {
  157. if (fJackPort == nullptr)
  158. return CarlaEngineCVPort::initBuffer();
  159. const uint32_t bufferSize(fClient.getEngine().getBufferSize());
  160. try {
  161. fBuffer = (float*)jackbridge_port_get_buffer(fJackPort, bufferSize);
  162. }
  163. catch(...) {
  164. fBuffer = nullptr;
  165. return;
  166. }
  167. if (! fIsInput)
  168. FLOAT_CLEAR(fBuffer, bufferSize);
  169. }
  170. void invalidate() noexcept
  171. {
  172. fJackClient = nullptr;
  173. fJackPort = nullptr;
  174. }
  175. private:
  176. jack_client_t* fJackClient;
  177. jack_port_t* fJackPort;
  178. JackPortDeletionCallback* const fDeletionCallback;
  179. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackCVPort)
  180. };
  181. // -----------------------------------------------------------------------
  182. // Carla Engine JACK-Event port
  183. class CarlaEngineJackEventPort : public CarlaEngineEventPort
  184. {
  185. public:
  186. CarlaEngineJackEventPort(const CarlaEngineClient& client, const bool isInputPort, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  187. : CarlaEngineEventPort(client, isInputPort),
  188. fJackClient(jackClient),
  189. fJackPort(jackPort),
  190. fJackBuffer(nullptr),
  191. fDeletionCallback(delCallback)
  192. {
  193. carla_debug("CarlaEngineJackEventPort::CarlaEngineJackEventPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  194. switch (fClient.getEngine().getProccessMode())
  195. {
  196. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  197. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  198. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  199. break;
  200. default:
  201. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  202. break;
  203. }
  204. }
  205. ~CarlaEngineJackEventPort() noexcept override
  206. {
  207. carla_debug("CarlaEngineJackEventPort::~CarlaEngineJackEventPort()");
  208. if (fJackClient != nullptr && fJackPort != nullptr)
  209. {
  210. try {
  211. jackbridge_port_unregister(fJackClient, fJackPort);
  212. } catch(...) {}
  213. fJackClient = nullptr;
  214. fJackPort = nullptr;
  215. }
  216. if (fDeletionCallback != nullptr)
  217. fDeletionCallback->jackEventPortDeleted(this);
  218. }
  219. void initBuffer() noexcept override
  220. {
  221. if (fJackPort == nullptr)
  222. return CarlaEngineEventPort::initBuffer();
  223. try {
  224. fJackBuffer = jackbridge_port_get_buffer(fJackPort, fClient.getEngine().getBufferSize());
  225. }
  226. catch(...) {
  227. fJackBuffer = nullptr;
  228. return;
  229. }
  230. if (! fIsInput)
  231. jackbridge_midi_clear_buffer(fJackBuffer);
  232. }
  233. uint32_t getEventCount() const noexcept override
  234. {
  235. if (fJackPort == nullptr)
  236. return CarlaEngineEventPort::getEventCount();
  237. CARLA_SAFE_ASSERT_RETURN(fIsInput, 0);
  238. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, 0);
  239. uint32_t count = 0;
  240. try {
  241. count = jackbridge_midi_get_event_count(fJackBuffer);
  242. } catch(...) {}
  243. return count;
  244. }
  245. const EngineEvent& getEvent(const uint32_t index) const noexcept override
  246. {
  247. if (fJackPort == nullptr)
  248. return CarlaEngineEventPort::getEvent(index);
  249. CARLA_SAFE_ASSERT_RETURN(fIsInput, kFallbackJackEngineEvent);
  250. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, kFallbackJackEngineEvent);
  251. return getEventUnchecked(index);
  252. }
  253. const EngineEvent& getEventUnchecked(const uint32_t index) const noexcept override
  254. {
  255. jack_midi_event_t jackEvent;
  256. bool test = false;
  257. try {
  258. test = jackbridge_midi_event_get(&jackEvent, fJackBuffer, index);
  259. } catch(...) {}
  260. if (! test)
  261. return kFallbackJackEngineEvent;
  262. CARLA_SAFE_ASSERT_RETURN(jackEvent.size <= 0xFF /* uint8_t max */, kFallbackJackEngineEvent);
  263. fRetEvent.time = jackEvent.time;
  264. fRetEvent.fillFromMidiData(static_cast<uint8_t>(jackEvent.size), jackEvent.buffer);
  265. return fRetEvent;
  266. }
  267. bool writeControlEvent(const uint32_t time, const uint8_t channel, const EngineControlEventType type, const uint16_t param, const float value) noexcept override
  268. {
  269. if (fJackPort == nullptr)
  270. return CarlaEngineEventPort::writeControlEvent(time, channel, type, param, value);
  271. CARLA_SAFE_ASSERT_RETURN(! fIsInput, false);
  272. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, false);
  273. CARLA_SAFE_ASSERT_RETURN(type != kEngineControlEventTypeNull, false);
  274. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS, false);
  275. CARLA_SAFE_ASSERT_RETURN(param < 0x5F, false);
  276. CARLA_SAFE_ASSERT(value >= 0.0f && value <= 1.0f);
  277. if (type == kEngineControlEventTypeParameter) {
  278. CARLA_SAFE_ASSERT(! MIDI_IS_CONTROL_BANK_SELECT(param));
  279. }
  280. uint8_t size = 0;
  281. uint8_t data[3] = { 0, 0, 0 };
  282. EngineControlEvent ctrlEvent = { type, param, value };
  283. ctrlEvent.convertToMidiData(channel, size, data);
  284. if (size == 0)
  285. return false;
  286. bool ret = false;
  287. try {
  288. ret = jackbridge_midi_event_write(fJackBuffer, time, data, size);
  289. } catch(...) {}
  290. return ret;
  291. }
  292. bool writeMidiEvent(const uint32_t time, const uint8_t channel, const uint8_t port, const uint8_t size, const uint8_t* const data) noexcept override
  293. {
  294. if (fJackPort == nullptr)
  295. return CarlaEngineEventPort::writeMidiEvent(time, channel, port, size, data);
  296. CARLA_SAFE_ASSERT_RETURN(! fIsInput, false);
  297. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, false);
  298. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS, false);
  299. CARLA_SAFE_ASSERT_RETURN(size > 0, false);
  300. CARLA_SAFE_ASSERT_RETURN(data != nullptr, false);
  301. jack_midi_data_t jdata[size];
  302. jdata[0] = static_cast<jack_midi_data_t>(MIDI_GET_STATUS_FROM_DATA(data) + channel);
  303. for (uint8_t i=1; i < size; ++i)
  304. jdata[i] = data[i];
  305. bool ret = false;
  306. try {
  307. ret = jackbridge_midi_event_write(fJackBuffer, time, jdata, size);
  308. } catch(...) {}
  309. return ret;
  310. }
  311. void invalidate() noexcept
  312. {
  313. fJackClient = nullptr;
  314. fJackPort = nullptr;
  315. }
  316. private:
  317. jack_client_t* fJackClient;
  318. jack_port_t* fJackPort;
  319. void* fJackBuffer;
  320. mutable EngineEvent fRetEvent;
  321. JackPortDeletionCallback* const fDeletionCallback;
  322. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackEventPort)
  323. };
  324. // -----------------------------------------------------------------------
  325. // Jack Engine client
  326. class CarlaEngineJackClient : public CarlaEngineClient,
  327. JackPortDeletionCallback
  328. {
  329. public:
  330. CarlaEngineJackClient(const CarlaEngine& engine, jack_client_t* const jackClient)
  331. : CarlaEngineClient(engine),
  332. fJackClient(jackClient),
  333. fUseClient(engine.getProccessMode() == ENGINE_PROCESS_MODE_SINGLE_CLIENT || engine.getProccessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  334. {
  335. carla_debug("CarlaEngineJackClient::CarlaEngineJackClient(%p)", jackClient);
  336. if (fUseClient)
  337. {
  338. CARLA_SAFE_ASSERT(jackClient != nullptr);
  339. }
  340. else
  341. {
  342. CARLA_SAFE_ASSERT(jackClient == nullptr);
  343. }
  344. }
  345. ~CarlaEngineJackClient() override
  346. {
  347. carla_debug("CarlaEngineClient::~CarlaEngineClient()");
  348. if (fEngine.getProccessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS && fJackClient != nullptr)
  349. jackbridge_client_close(fJackClient);
  350. // ports must have been deleted by now!
  351. //fAudioPorts.clear();
  352. //fCVPorts.clear();
  353. //fEventPorts.clear();
  354. }
  355. void activate() noexcept override
  356. {
  357. carla_debug("CarlaEngineJackClient::activate()");
  358. if (fEngine.getProccessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  359. {
  360. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr && ! fActive,);
  361. try {
  362. jackbridge_activate(fJackClient);
  363. } catch(...) {}
  364. }
  365. CarlaEngineClient::activate();
  366. }
  367. void deactivate() noexcept override
  368. {
  369. carla_debug("CarlaEngineJackClient::deactivate()");
  370. if (fEngine.getProccessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  371. {
  372. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr && fActive,);
  373. try {
  374. jackbridge_deactivate(fJackClient);
  375. } catch(...) {}
  376. }
  377. CarlaEngineClient::deactivate();
  378. }
  379. bool isOk() const noexcept override
  380. {
  381. if (fUseClient)
  382. return (fJackClient != nullptr);
  383. return CarlaEngineClient::isOk();
  384. }
  385. #if 0
  386. void setLatency(const uint32_t samples) noexcept override
  387. {
  388. CarlaEngineClient::setLatency(samples);
  389. if (fUseClient && fJackClient != nullptr)
  390. {
  391. // try etc
  392. jackbridge_recompute_total_latencies(fJackClient);
  393. }
  394. }
  395. #endif
  396. CarlaEnginePort* addPort(const EnginePortType portType, const char* const name, const bool isInput) override
  397. {
  398. carla_debug("CarlaEngineJackClient::addPort(%i:%s, \"%s\", %s)", portType, EnginePortType2Str(portType), name, bool2str(isInput));
  399. jack_port_t* jackPort = nullptr;
  400. // Create JACK port first, if needed
  401. if (fUseClient && fJackClient != nullptr)
  402. {
  403. switch (portType)
  404. {
  405. case kEnginePortTypeNull:
  406. break;
  407. case kEnginePortTypeAudio:
  408. jackPort = jackbridge_port_register(fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  409. break;
  410. case kEnginePortTypeCV:
  411. jackPort = jackbridge_port_register(fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  412. break;
  413. case kEnginePortTypeEvent:
  414. jackPort = jackbridge_port_register(fJackClient, name, JACK_DEFAULT_MIDI_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  415. break;
  416. }
  417. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr, nullptr);
  418. }
  419. // Create Engine port
  420. switch (portType)
  421. {
  422. case kEnginePortTypeNull:
  423. break;
  424. case kEnginePortTypeAudio: {
  425. CarlaEngineJackAudioPort* const enginePort(new CarlaEngineJackAudioPort(*this, isInput, fJackClient, jackPort, this));
  426. fAudioPorts.append(enginePort);
  427. return enginePort;
  428. }
  429. case kEnginePortTypeCV: {
  430. CarlaEngineJackCVPort* const enginePort(new CarlaEngineJackCVPort(*this, isInput, fJackClient, jackPort, this));
  431. fCVPorts.append(enginePort);
  432. return enginePort;
  433. }
  434. case kEnginePortTypeEvent: {
  435. CarlaEngineJackEventPort* const enginePort(new CarlaEngineJackEventPort(*this, isInput, fJackClient, jackPort, this));
  436. fEventPorts.append(enginePort);
  437. return enginePort;
  438. }
  439. }
  440. carla_stderr("CarlaEngineJackClient::addPort(%i, \"%s\", %s) - invalid type", portType, name, bool2str(isInput));
  441. return nullptr;
  442. }
  443. void invalidate() noexcept
  444. {
  445. for (LinkedList<CarlaEngineJackAudioPort*>::Itenerator it = fAudioPorts.begin(); it.valid(); it.next())
  446. {
  447. CarlaEngineJackAudioPort* const port(it.getValue());
  448. port->invalidate();
  449. }
  450. for (LinkedList<CarlaEngineJackCVPort*>::Itenerator it = fCVPorts.begin(); it.valid(); it.next())
  451. {
  452. CarlaEngineJackCVPort* const port(it.getValue());
  453. port->invalidate();
  454. }
  455. for (LinkedList<CarlaEngineJackEventPort*>::Itenerator it = fEventPorts.begin(); it.valid(); it.next())
  456. {
  457. CarlaEngineJackEventPort* const port(it.getValue());
  458. port->invalidate();
  459. }
  460. fJackClient = nullptr;
  461. }
  462. const char* getJackClientName() const noexcept
  463. {
  464. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr, nullptr);
  465. try {
  466. return jackbridge_get_client_name(fJackClient);
  467. } CARLA_SAFE_EXCEPTION("JACK client get name");
  468. return nullptr;
  469. }
  470. void jackAudioPortDeleted(CarlaEngineJackAudioPort* const port) override
  471. {
  472. fAudioPorts.removeAll(port);
  473. }
  474. void jackCVPortDeleted(CarlaEngineJackCVPort* const port) override
  475. {
  476. fCVPorts.removeAll(port);
  477. }
  478. void jackEventPortDeleted(CarlaEngineJackEventPort* const port) override
  479. {
  480. fEventPorts.removeAll(port);
  481. }
  482. private:
  483. jack_client_t* fJackClient;
  484. const bool fUseClient;
  485. LinkedList<CarlaEngineJackAudioPort*> fAudioPorts;
  486. LinkedList<CarlaEngineJackCVPort*> fCVPorts;
  487. LinkedList<CarlaEngineJackEventPort*> fEventPorts;
  488. friend class CarlaEngineJack;
  489. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackClient)
  490. };
  491. // -----------------------------------------------------------------------
  492. // Jack Engine
  493. class CarlaEngineJack : public CarlaEngine
  494. {
  495. public:
  496. CarlaEngineJack()
  497. : CarlaEngine(),
  498. fClient(nullptr),
  499. fTransportState(JackTransportStopped),
  500. fFreewheel(false),
  501. #ifdef BUILD_BRIDGE
  502. fIsRunning(false)
  503. #else
  504. fLastGroupId(0),
  505. fLastPortId(0),
  506. fLastConnectionId(0)
  507. #endif
  508. {
  509. carla_debug("CarlaEngineJack::CarlaEngineJack()");
  510. #ifdef BUILD_BRIDGE
  511. pData->options.processMode = ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS;
  512. #else
  513. carla_fill<jack_port_t*>(fRackPorts, kRackPortCount, nullptr);
  514. #endif
  515. // FIXME: Always enable JACK transport for now
  516. pData->options.transportMode = ENGINE_TRANSPORT_MODE_JACK;
  517. carla_zeroStruct<jack_position_t>(fTransportPos);
  518. }
  519. ~CarlaEngineJack() override
  520. {
  521. carla_debug("CarlaEngineJack::~CarlaEngineJack()");
  522. CARLA_SAFE_ASSERT(fClient == nullptr);
  523. #ifndef BUILD_BRIDGE
  524. fUsedGroupNames.clear();
  525. fUsedPortNames.clear();
  526. fUsedConnections.clear();
  527. fNewGroups.clear();
  528. #endif
  529. }
  530. // -------------------------------------------------------------------
  531. // Maximum values
  532. uint getMaxClientNameSize() const noexcept override
  533. {
  534. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT || pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  535. {
  536. uint ret = 0;
  537. try {
  538. ret = static_cast<uint>(jackbridge_client_name_size());
  539. }
  540. catch (...) {}
  541. return ret;
  542. }
  543. return CarlaEngine::getMaxClientNameSize();
  544. }
  545. uint getMaxPortNameSize() const noexcept override
  546. {
  547. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT || pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  548. {
  549. uint ret = 0;
  550. try {
  551. ret = static_cast<uint>(jackbridge_port_name_size());
  552. }
  553. catch (...) {}
  554. return ret;
  555. }
  556. return CarlaEngine::getMaxPortNameSize();
  557. }
  558. // -------------------------------------------------------------------
  559. // Virtual, per-engine type calls
  560. bool init(const char* const clientName) override
  561. {
  562. carla_debug("CarlaEngineJack::init(\"%s\")", clientName);
  563. fFreewheel = false;
  564. fTransportState = JackTransportStopped;
  565. carla_zeroStruct<jack_position_t>(fTransportPos);
  566. #ifdef BUILD_BRIDGE
  567. if (pData->bufferSize == 0 || pData->sampleRate == 0.0)
  568. {
  569. // open temp client to get initial buffer-size and sample-rate values
  570. if (jack_client_t* tmpClient = jackbridge_client_open(clientName, JackNullOption, nullptr))
  571. {
  572. pData->bufferSize = jackbridge_get_buffer_size(tmpClient);
  573. pData->sampleRate = jackbridge_get_sample_rate(tmpClient);
  574. jackbridge_client_close(tmpClient);
  575. }
  576. }
  577. fIsRunning = true;
  578. return CarlaEngine::init(clientName);
  579. #else
  580. fLastGroupId = 0;
  581. fLastPortId = 0;
  582. fLastConnectionId = 0;
  583. fUsedGroupNames.clear();
  584. fUsedPortNames.clear();
  585. fUsedConnections.clear();
  586. fNewGroups.clear();
  587. fClient = jackbridge_client_open(clientName, JackNullOption, nullptr);
  588. if (fClient != nullptr)
  589. {
  590. pData->bufferSize = jackbridge_get_buffer_size(fClient);
  591. pData->sampleRate = jackbridge_get_sample_rate(fClient);
  592. jackbridge_set_buffer_size_callback(fClient, carla_jack_bufsize_callback, this);
  593. jackbridge_set_sample_rate_callback(fClient, carla_jack_srate_callback, this);
  594. jackbridge_set_freewheel_callback(fClient, carla_jack_freewheel_callback, this);
  595. //jackbridge_set_latency_callback(fClient, carla_jack_latency_callback, this);
  596. jackbridge_set_process_callback(fClient, carla_jack_process_callback, this);
  597. jackbridge_on_shutdown(fClient, carla_jack_shutdown_callback, this);
  598. const char* const jackClientName(jackbridge_get_client_name(fClient));
  599. initJackPatchbay(jackClientName);
  600. jackbridge_set_client_registration_callback(fClient, carla_jack_client_registration_callback, this);
  601. jackbridge_set_port_registration_callback(fClient, carla_jack_port_registration_callback, this);
  602. jackbridge_set_port_connect_callback(fClient, carla_jack_port_connect_callback, this);
  603. jackbridge_set_port_rename_callback(fClient, carla_jack_port_rename_callback, this);
  604. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  605. {
  606. fRackPorts[kRackPortAudioIn1] = jackbridge_port_register(fClient, "audio-in1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  607. fRackPorts[kRackPortAudioIn2] = jackbridge_port_register(fClient, "audio-in2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  608. fRackPorts[kRackPortAudioOut1] = jackbridge_port_register(fClient, "audio-out1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  609. fRackPorts[kRackPortAudioOut2] = jackbridge_port_register(fClient, "audio-out2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  610. fRackPorts[kRackPortEventIn] = jackbridge_port_register(fClient, "events-in", JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0);
  611. fRackPorts[kRackPortEventOut] = jackbridge_port_register(fClient, "events-out", JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput, 0);
  612. }
  613. if (jackbridge_activate(fClient))
  614. {
  615. CarlaEngine::init(jackClientName);
  616. return true;
  617. }
  618. else
  619. {
  620. setLastError("Failed to activate the JACK client");
  621. jackbridge_client_close(fClient);
  622. fClient = nullptr;
  623. }
  624. }
  625. else
  626. setLastError("Failed to create new JACK client");
  627. return false;
  628. #endif
  629. }
  630. bool close() override
  631. {
  632. carla_debug("CarlaEngineJack::close()");
  633. CarlaEngine::close();
  634. #ifdef BUILD_BRIDGE
  635. fClient = nullptr;
  636. fIsRunning = false;
  637. return true;
  638. #else
  639. if (jackbridge_deactivate(fClient))
  640. {
  641. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK && fRackPorts[0] != nullptr)
  642. {
  643. jackbridge_port_unregister(fClient, fRackPorts[kRackPortAudioIn1]);
  644. jackbridge_port_unregister(fClient, fRackPorts[kRackPortAudioIn2]);
  645. jackbridge_port_unregister(fClient, fRackPorts[kRackPortAudioOut1]);
  646. jackbridge_port_unregister(fClient, fRackPorts[kRackPortAudioOut2]);
  647. jackbridge_port_unregister(fClient, fRackPorts[kRackPortEventIn]);
  648. jackbridge_port_unregister(fClient, fRackPorts[kRackPortEventOut]);
  649. carla_fill<jack_port_t*>(fRackPorts, kRackPortCount, nullptr);
  650. }
  651. if (jackbridge_client_close(fClient))
  652. {
  653. fClient = nullptr;
  654. return true;
  655. }
  656. else
  657. setLastError("Failed to close the JACK client");
  658. }
  659. else
  660. setLastError("Failed to deactivate the JACK client");
  661. fClient = nullptr;
  662. fUsedGroupNames.clear();
  663. fUsedPortNames.clear();
  664. fUsedConnections.clear();
  665. fNewGroups.clear();
  666. return false;
  667. #endif
  668. }
  669. #ifndef BUILD_BRIDGE
  670. void idle() override
  671. {
  672. CarlaEngine::idle();
  673. if (fNewGroups.count() == 0)
  674. return;
  675. LinkedList<int> newPlugins;
  676. fNewGroups.spliceInsert(newPlugins, true);
  677. for (LinkedList<int>::Itenerator it = newPlugins.begin(); it.valid(); it.next())
  678. {
  679. const int groupId(it.getValue());
  680. const char* const groupName(getGroupName(groupId));
  681. CARLA_SAFE_ASSERT_CONTINUE(groupId > 0 && groupName != nullptr && groupName[0] != '\0');
  682. int pluginId = -1;
  683. PatchbayIcon icon = PATCHBAY_ICON_PLUGIN;
  684. if (findPluginIdAndIcon(groupName, pluginId, icon))
  685. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_DATA_CHANGED, static_cast<uint>(groupId), icon, pluginId, 0.0f, nullptr);
  686. }
  687. newPlugins.clear();
  688. }
  689. #endif
  690. bool isRunning() const noexcept override
  691. {
  692. #ifdef BUILD_BRIDGE
  693. return (fClient != nullptr || fIsRunning);
  694. #else
  695. return (fClient != nullptr);
  696. #endif
  697. }
  698. bool isOffline() const noexcept override
  699. {
  700. return fFreewheel;
  701. }
  702. EngineType getType() const noexcept override
  703. {
  704. return kEngineTypeJack;
  705. }
  706. const char* getCurrentDriverName() const noexcept override
  707. {
  708. return "JACK";
  709. }
  710. CarlaEngineClient* addClient(CarlaPlugin* const plugin) override
  711. {
  712. jack_client_t* client = nullptr;
  713. #ifdef BUILD_BRIDGE
  714. client = fClient = jackbridge_client_open(plugin->getName(), JackNullOption, nullptr);
  715. CARLA_SAFE_ASSERT_RETURN(client != nullptr, nullptr);
  716. pData->bufferSize = jackbridge_get_buffer_size(client);
  717. pData->sampleRate = jackbridge_get_sample_rate(client);
  718. jackbridge_set_buffer_size_callback(client, carla_jack_bufsize_callback, this);
  719. jackbridge_set_sample_rate_callback(client, carla_jack_srate_callback, this);
  720. jackbridge_set_freewheel_callback(client, carla_jack_freewheel_callback, this);
  721. //jackbridge_set_latency_callback(client, carla_jack_latency_callback, this);
  722. jackbridge_set_process_callback(client, carla_jack_process_callback, this);
  723. jackbridge_on_shutdown(client, carla_jack_shutdown_callback, this);
  724. #else
  725. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  726. {
  727. client = fClient;
  728. }
  729. else if (pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  730. {
  731. client = jackbridge_client_open(plugin->getName(), JackNullOption, nullptr);
  732. CARLA_SAFE_ASSERT_RETURN(client != nullptr, nullptr);
  733. //jackbridge_set_latency_callback(client, carla_jack_latency_callback_plugin, plugin);
  734. jackbridge_set_process_callback(client, carla_jack_process_callback_plugin, plugin);
  735. jackbridge_on_shutdown(fClient, carla_jack_shutdown_callback_plugin, plugin);
  736. }
  737. #endif
  738. return new CarlaEngineJackClient(*this, client);
  739. }
  740. #ifndef BUILD_BRIDGE
  741. const char* renamePlugin(const uint id, const char* const newName) override
  742. {
  743. CARLA_SAFE_ASSERT_RETURN(pData->curPluginCount > 0, nullptr);
  744. CARLA_SAFE_ASSERT_RETURN(id < pData->curPluginCount, nullptr);
  745. CARLA_SAFE_ASSERT_RETURN(pData->plugins != nullptr, nullptr);
  746. CARLA_SAFE_ASSERT_RETURN(newName != nullptr && newName[0] != '\0', nullptr);
  747. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  748. if (plugin == nullptr)
  749. {
  750. carla_stderr("CarlaEngine::clonePlugin(%i) - could not find plugin", id);
  751. return nullptr;
  752. }
  753. CARLA_SAFE_ASSERT(plugin->getId() == id);
  754. CarlaString uniqueName;
  755. try {
  756. const char* const tmpName = getUniquePluginName(newName);
  757. uniqueName = tmpName;
  758. delete[] tmpName;
  759. } CARLA_SAFE_EXCEPTION("JACK renamePlugin");
  760. if (uniqueName.isEmpty())
  761. {
  762. setLastError("Failed to request new unique plugin name");
  763. return nullptr;
  764. }
  765. // single client always re-inits
  766. bool needsReinit = (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT);
  767. // rename in multiple client mode
  768. if (pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  769. {
  770. CarlaEngineJackClient* const client((CarlaEngineJackClient*)plugin->getEngineClient());
  771. #if 0
  772. if (bridge.client_rename_ptr != nullptr)
  773. {
  774. newName = jackbridge_client_rename(client->fClient, newName);
  775. }
  776. else
  777. #endif
  778. {
  779. // we should not be able to do this, jack really needs to allow client rename
  780. if (jack_client_t* const jackClient = jackbridge_client_open(uniqueName, JackNullOption, nullptr))
  781. {
  782. // close old client
  783. plugin->setEnabled(false);
  784. if (client->isActive())
  785. client->deactivate();
  786. plugin->clearBuffers();
  787. jackbridge_client_close(client->fJackClient);
  788. // set new client data
  789. uniqueName = jackbridge_get_client_name(jackClient);
  790. jackbridge_set_process_callback(jackClient, carla_jack_process_callback_plugin, plugin);
  791. //jackbridge_set_latency_callback(jackClient, carla_jack_latency_callback_plugin, plugin);
  792. jackbridge_on_shutdown(jackClient, carla_jack_shutdown_callback_plugin, plugin);
  793. client->fJackClient = jackClient;
  794. needsReinit = true;
  795. }
  796. else
  797. {
  798. setLastError("Failed to create new JACK client");
  799. return nullptr;
  800. }
  801. }
  802. }
  803. // Rename
  804. plugin->setName(uniqueName);
  805. if (needsReinit)
  806. {
  807. // reload plugin to recreate its ports
  808. const SaveState& saveState(plugin->getSaveState());
  809. plugin->reload();
  810. plugin->loadSaveState(saveState);
  811. }
  812. return plugin->getName();
  813. }
  814. // -------------------------------------------------------------------
  815. // Patchbay
  816. bool patchbayConnect(const int groupA, const int portA, const int groupB, const int portB) override
  817. {
  818. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  819. if (fClient == nullptr)
  820. {
  821. setLastError("Invalid JACK client");
  822. return false;
  823. }
  824. if (portA < 0 || portB < 0)
  825. {
  826. // both must be < 0
  827. CARLA_SAFE_ASSERT_RETURN(portA < 0 && portB < 0, false);
  828. ConnectionToId connectionToId;
  829. connectionToId.setData(fLastConnectionId++, groupA, portA, groupB, portB);
  830. fUsedConnections.append(connectionToId);
  831. char strBuf[STR_MAX+1];
  832. strBuf[STR_MAX] = '\0';
  833. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i", groupA, portA, groupB, portB);
  834. callback(ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED, connectionToId.id, 0, 0, 0.0f, strBuf);
  835. return true;
  836. }
  837. char portNameA[STR_MAX+1];
  838. char portNameB[STR_MAX+1];
  839. getFullPortName(portA, portNameA);
  840. getFullPortName(portB, portNameB);
  841. if (! jackbridge_connect(fClient, portNameA, portNameB))
  842. {
  843. setLastError("JACK operation failed");
  844. return false;
  845. }
  846. return true;
  847. }
  848. bool patchbayDisconnect(const uint connectionId) override
  849. {
  850. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  851. for (LinkedList<ConnectionToId>::Itenerator it = fUsedConnections.begin(); it.valid(); it.next())
  852. {
  853. const ConnectionToId& connectionToId(it.getValue());
  854. if (connectionToId.id == connectionId)
  855. {
  856. char portNameOut[STR_MAX+1];
  857. char portNameIn[STR_MAX+1];
  858. getFullPortName(connectionToId.portOut, portNameOut);
  859. getFullPortName(connectionToId.portIn, portNameIn);
  860. if (! jackbridge_disconnect(fClient, portNameOut, portNameIn))
  861. {
  862. setLastError("JACK operation failed");
  863. return false;
  864. }
  865. return true;
  866. }
  867. }
  868. setLastError("Failed to find the requested connection");
  869. return false;
  870. }
  871. bool patchbayRefresh() override
  872. {
  873. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  874. fLastGroupId = 0;
  875. fLastPortId = 0;
  876. fLastConnectionId = 0;
  877. fUsedGroupNames.clear();
  878. fUsedPortNames.clear();
  879. fUsedConnections.clear();
  880. fNewGroups.clear();
  881. initJackPatchbay(jackbridge_get_client_name(fClient));
  882. return true;
  883. }
  884. // -------------------------------------------------------------------
  885. // Transport
  886. void transportPlay() noexcept override
  887. {
  888. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_INTERNAL)
  889. return CarlaEngine::transportPlay();
  890. if (fClient != nullptr)
  891. {
  892. try {
  893. jackbridge_transport_start(fClient);
  894. } catch(...) {}
  895. }
  896. }
  897. void transportPause() noexcept override
  898. {
  899. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_INTERNAL)
  900. return CarlaEngine::transportPause();
  901. if (fClient != nullptr)
  902. {
  903. try {
  904. jackbridge_transport_stop(fClient);
  905. } catch(...) {}
  906. }
  907. }
  908. void transportRelocate(const uint64_t frame) noexcept override
  909. {
  910. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_INTERNAL)
  911. return CarlaEngine::transportRelocate(frame);
  912. if (fClient != nullptr)
  913. {
  914. try {
  915. jackbridge_transport_locate(fClient, static_cast<jack_nframes_t>(frame));
  916. } catch(...) {}
  917. }
  918. }
  919. // -------------------------------------------------------------------
  920. // Patchbay stuff
  921. const char* const* getPatchbayConnections() const override
  922. {
  923. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, nullptr);
  924. carla_debug("CarlaEngineJack::getPatchbayConnections()");
  925. LinkedList<const char*> connList;
  926. if (const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, JackPortIsOutput))
  927. {
  928. for (int i=0; ports[i] != nullptr; ++i)
  929. {
  930. jack_port_t* const jackPort(jackbridge_port_by_name(fClient, ports[i]));
  931. const char* const fullPortName(ports[i]);
  932. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  933. if (const char** connections = jackbridge_port_get_all_connections(fClient, jackPort))
  934. {
  935. for (int j=0; connections[j] != nullptr; ++j)
  936. {
  937. connList.append(carla_strdup(fullPortName));
  938. connList.append(carla_strdup(connections[j]));
  939. }
  940. jackbridge_free(connections);
  941. }
  942. }
  943. jackbridge_free(ports);
  944. }
  945. const size_t connCount(connList.count());
  946. if (connCount == 0)
  947. return nullptr;
  948. const char** const retConns = new const char*[connCount+1];
  949. for (size_t i=0; i < connCount; ++i)
  950. retConns[i] = connList.getAt(i);
  951. retConns[connCount] = nullptr;
  952. connList.clear();
  953. return retConns;
  954. }
  955. void restorePatchbayConnection(const char* const connSource, const char* const connTarget) override
  956. {
  957. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr,);
  958. CARLA_SAFE_ASSERT_RETURN(connSource != nullptr && connSource[0] != '\0',);
  959. CARLA_SAFE_ASSERT_RETURN(connTarget != nullptr && connTarget[0] != '\0',);
  960. carla_debug("CarlaEngineJack::restorePatchbayConnection(\"%s\", \"%s\")", connSource, connTarget);
  961. if (jack_port_t* const port = jackbridge_port_by_name(fClient, connSource))
  962. {
  963. if (jackbridge_port_by_name(fClient, connTarget) == nullptr)
  964. return;
  965. if (! jackbridge_port_connected_to(port, connTarget))
  966. jackbridge_connect(fClient, connSource, connTarget);
  967. }
  968. }
  969. #endif
  970. // -------------------------------------------------------------------
  971. protected:
  972. void handleJackBufferSizeCallback(const uint32_t newBufferSize)
  973. {
  974. if (pData->bufferSize == newBufferSize)
  975. return;
  976. pData->bufferSize = newBufferSize;
  977. bufferSizeChanged(newBufferSize);
  978. }
  979. void handleJackSampleRateCallback(const double newSampleRate)
  980. {
  981. if (pData->sampleRate == newSampleRate)
  982. return;
  983. pData->sampleRate = newSampleRate;
  984. sampleRateChanged(newSampleRate);
  985. }
  986. void handleJackFreewheelCallback(const bool isFreewheel)
  987. {
  988. if (fFreewheel == isFreewheel)
  989. return;
  990. fFreewheel = isFreewheel;
  991. offlineModeChanged(isFreewheel);
  992. }
  993. void saveTransportInfo()
  994. {
  995. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK)
  996. return;
  997. fTransportPos.unique_1 = fTransportPos.unique_2 + 1; // invalidate
  998. fTransportState = jackbridge_transport_query(fClient, &fTransportPos);
  999. pData->timeInfo.playing = (fTransportState == JackTransportRolling);
  1000. if (fTransportPos.unique_1 == fTransportPos.unique_2)
  1001. {
  1002. pData->timeInfo.frame = fTransportPos.frame;
  1003. pData->timeInfo.usecs = fTransportPos.usecs;
  1004. if (fTransportPos.valid & JackPositionBBT)
  1005. {
  1006. pData->timeInfo.valid = EngineTimeInfo::kValidBBT;
  1007. pData->timeInfo.bbt.bar = fTransportPos.bar;
  1008. pData->timeInfo.bbt.beat = fTransportPos.beat;
  1009. pData->timeInfo.bbt.tick = fTransportPos.tick;
  1010. pData->timeInfo.bbt.barStartTick = fTransportPos.bar_start_tick;
  1011. pData->timeInfo.bbt.beatsPerBar = fTransportPos.beats_per_bar;
  1012. pData->timeInfo.bbt.beatType = fTransportPos.beat_type;
  1013. pData->timeInfo.bbt.ticksPerBeat = fTransportPos.ticks_per_beat;
  1014. pData->timeInfo.bbt.beatsPerMinute = fTransportPos.beats_per_minute;
  1015. }
  1016. else
  1017. pData->timeInfo.valid = 0x0;
  1018. }
  1019. else
  1020. {
  1021. pData->timeInfo.frame = 0;
  1022. pData->timeInfo.valid = 0x0;
  1023. }
  1024. }
  1025. void handleJackProcessCallback(const uint32_t nframes)
  1026. {
  1027. saveTransportInfo();
  1028. if (pData->curPluginCount == 0)
  1029. {
  1030. #ifndef BUILD_BRIDGE
  1031. // pass-through
  1032. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  1033. {
  1034. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn1], nframes);
  1035. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn2], nframes);
  1036. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut1], nframes);
  1037. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut2], nframes);
  1038. void* const eventOut = jackbridge_port_get_buffer(fRackPorts[kRackPortEventOut], nframes);
  1039. FLOAT_COPY(audioOut1, audioIn1, nframes);
  1040. FLOAT_COPY(audioOut2, audioIn2, nframes);
  1041. jackbridge_midi_clear_buffer(eventOut);
  1042. }
  1043. #endif
  1044. return runPendingRtEvents();
  1045. }
  1046. #ifdef BUILD_BRIDGE
  1047. CarlaPlugin* const plugin(pData->plugins[0].plugin);
  1048. if (plugin != nullptr && plugin->isEnabled() && plugin->tryLock(fFreewheel))
  1049. {
  1050. plugin->initBuffers();
  1051. processPlugin(plugin, nframes);
  1052. plugin->unlock();
  1053. }
  1054. return runPendingRtEvents();
  1055. #else
  1056. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  1057. {
  1058. for (uint i=0; i < pData->curPluginCount; ++i)
  1059. {
  1060. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1061. if (plugin != nullptr && plugin->isEnabled() && plugin->tryLock(fFreewheel))
  1062. {
  1063. plugin->initBuffers();
  1064. processPlugin(plugin, nframes);
  1065. plugin->unlock();
  1066. }
  1067. }
  1068. return runPendingRtEvents();
  1069. }
  1070. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  1071. {
  1072. // get buffers from jack
  1073. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn1], nframes);
  1074. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn2], nframes);
  1075. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut1], nframes);
  1076. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut2], nframes);
  1077. void* const eventIn = jackbridge_port_get_buffer(fRackPorts[kRackPortEventIn], nframes);
  1078. void* const eventOut = jackbridge_port_get_buffer(fRackPorts[kRackPortEventOut], nframes);
  1079. #if 0
  1080. // assert buffers
  1081. CARLA_SAFE_ASSERT(audioIn1 != nullptr);
  1082. CARLA_SAFE_ASSERT(audioIn2 != nullptr);
  1083. CARLA_SAFE_ASSERT(audioOut1 != nullptr);
  1084. CARLA_SAFE_ASSERT(audioOut2 != nullptr);
  1085. CARLA_SAFE_ASSERT(eventIn != nullptr);
  1086. CARLA_SAFE_ASSERT(eventOut != nullptr);
  1087. #endif
  1088. // create audio buffers
  1089. const float* inBuf[2] = { audioIn1, audioIn2 };
  1090. float* outBuf[2] = { audioOut1, audioOut2 };
  1091. // initialize input events
  1092. carla_zeroStruct<EngineEvent>(pData->events.in, kMaxEngineEventInternalCount);
  1093. {
  1094. uint32_t engineEventIndex = 0;
  1095. jack_midi_event_t jackEvent;
  1096. const uint32_t jackEventCount(jackbridge_midi_get_event_count(eventIn));
  1097. for (uint32_t jackEventIndex=0; jackEventIndex < jackEventCount; ++jackEventIndex)
  1098. {
  1099. if (! jackbridge_midi_event_get(&jackEvent, eventIn, jackEventIndex))
  1100. continue;
  1101. CARLA_SAFE_ASSERT_CONTINUE(jackEvent.size <= 0xFF /* uint8_t max */);
  1102. EngineEvent& engineEvent(pData->events.in[engineEventIndex++]);
  1103. engineEvent.time = jackEvent.time;
  1104. engineEvent.fillFromMidiData(static_cast<uint8_t>(jackEvent.size), jackEvent.buffer);
  1105. if (engineEventIndex >= kMaxEngineEventInternalCount)
  1106. break;
  1107. }
  1108. }
  1109. // process rack
  1110. pData->processRack(inBuf, outBuf, nframes, fFreewheel);
  1111. // output control
  1112. {
  1113. jackbridge_midi_clear_buffer(eventOut);
  1114. for (ushort i=0; i < kMaxEngineEventInternalCount; ++i)
  1115. {
  1116. const EngineEvent& engineEvent(pData->events.out[i]);
  1117. uint8_t size = 0;
  1118. uint8_t data[3] = { 0, 0, 0 };
  1119. const uint8_t* dataPtr = data;
  1120. switch (engineEvent.type)
  1121. {
  1122. case kEngineEventTypeNull:
  1123. break;
  1124. case kEngineEventTypeControl:
  1125. {
  1126. const EngineControlEvent& ctrlEvent(engineEvent.ctrl);
  1127. ctrlEvent.convertToMidiData(engineEvent.channel, size, data);
  1128. break;
  1129. }
  1130. case kEngineEventTypeMidi:
  1131. {
  1132. const EngineMidiEvent& midiEvent(engineEvent.midi);
  1133. size = midiEvent.size;
  1134. if (size > EngineMidiEvent::kDataSize && midiEvent.dataExt != nullptr)
  1135. dataPtr = midiEvent.dataExt;
  1136. else
  1137. dataPtr = midiEvent.dataExt;
  1138. break;
  1139. }
  1140. }
  1141. if (size > 0)
  1142. jackbridge_midi_event_write(eventOut, engineEvent.time, dataPtr, size);
  1143. }
  1144. }
  1145. return runPendingRtEvents();
  1146. }
  1147. #endif // ! BUILD_BRIDGE
  1148. runPendingRtEvents();
  1149. }
  1150. #if 0
  1151. void handleJackLatencyCallback(const jack_latency_callback_mode_t mode)
  1152. {
  1153. if (pData->options.processMode != ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  1154. return;
  1155. for (uint i=0; i < pData->curPluginCount; ++i)
  1156. {
  1157. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1158. if (plugin != nullptr && plugin->isEnabled())
  1159. latencyPlugin(plugin, mode);
  1160. }
  1161. }
  1162. #endif
  1163. #ifndef BUILD_BRIDGE
  1164. void handleJackClientRegistrationCallback(const char* const name, const bool reg)
  1165. {
  1166. // do nothing on client registration, wait for first port
  1167. if (reg) return;
  1168. const int id(getGroupId(name)); // also checks name nullness
  1169. if (id < 0)
  1170. return;
  1171. GroupNameToId groupNameToId;
  1172. groupNameToId.setData(id, name);
  1173. fUsedGroupNames.removeAll(groupNameToId);
  1174. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_REMOVED, static_cast<uint>(id), 0, 0, 0.0f, nullptr);
  1175. }
  1176. void handleJackPortRegistrationCallback(const jack_port_id_t port, const bool reg)
  1177. {
  1178. jack_port_t* const jackPort(jackbridge_port_by_id(fClient, port));
  1179. const char* const portName(jackbridge_port_short_name(jackPort));
  1180. const char* const fullPortName(jackbridge_port_name(jackPort));
  1181. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr,);
  1182. CARLA_SAFE_ASSERT_RETURN(portName != nullptr,);
  1183. CARLA_SAFE_ASSERT_RETURN(fullPortName != nullptr,);
  1184. CarlaString groupName(fullPortName);
  1185. groupName.truncate(groupName.rfind(portName)-1);
  1186. int groupId = getGroupId(groupName);
  1187. if (reg)
  1188. {
  1189. const int jackPortFlags(jackbridge_port_flags(jackPort));
  1190. if (groupId < 0)
  1191. {
  1192. groupId = fLastGroupId++;
  1193. CARLA_SAFE_ASSERT_RETURN(groupId >= 0,);
  1194. GroupNameToId groupNameToId;
  1195. groupNameToId.setData(groupId, groupName);
  1196. fUsedGroupNames.append(groupNameToId);
  1197. PatchbayIcon icon = (jackPortFlags & JackPortIsPhysical) ? PATCHBAY_ICON_HARDWARE : PATCHBAY_ICON_APPLICATION;
  1198. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED, static_cast<uint>(groupId), icon, -1, 0.0f, groupName);
  1199. fNewGroups.append(groupId);
  1200. }
  1201. bool portIsInput = (jackPortFlags & JackPortIsInput);
  1202. bool portIsAudio = (std::strcmp(jackbridge_port_type(jackPort), JACK_DEFAULT_AUDIO_TYPE) == 0);
  1203. bool portIsCV = false;
  1204. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  1205. {
  1206. char* value = nullptr;
  1207. char* type = nullptr;
  1208. if (jackbridge_get_property(uuid, URI_CANVAS_CV, &value, &type) && value != nullptr && type != nullptr && std::strcmp(type, "text/plain") == 0)
  1209. portIsCV = (std::strcmp(value, "YES") == 0);
  1210. }
  1211. uint canvasPortFlags = 0x0;
  1212. canvasPortFlags |= portIsInput ? PATCHBAY_PORT_IS_INPUT : 0x0;
  1213. if (portIsCV)
  1214. canvasPortFlags |= PATCHBAY_PORT_TYPE_CV;
  1215. else if (portIsAudio)
  1216. canvasPortFlags |= PATCHBAY_PORT_TYPE_AUDIO;
  1217. else
  1218. canvasPortFlags |= PATCHBAY_PORT_TYPE_MIDI;
  1219. PortNameToId portNameToId;
  1220. portNameToId.setData(groupId, fLastPortId++, portName, fullPortName);
  1221. fUsedPortNames.append(portNameToId);
  1222. callback(ENGINE_CALLBACK_PATCHBAY_PORT_ADDED, static_cast<uint>(groupId), portNameToId.portId, static_cast<int>(canvasPortFlags), 0.0f, portName);
  1223. }
  1224. else
  1225. {
  1226. const int portId(getPortId(fullPortName));
  1227. CARLA_SAFE_ASSERT_RETURN(groupId >= 0,);
  1228. CARLA_SAFE_ASSERT_RETURN(portId >= 0,);
  1229. PortNameToId portNameToId;
  1230. portNameToId.setData(groupId, portId, portName, fullPortName);
  1231. fUsedPortNames.removeOne(portNameToId);
  1232. callback(ENGINE_CALLBACK_PATCHBAY_PORT_REMOVED, static_cast<uint>(groupId), portId, 0, 0.0f, nullptr);
  1233. }
  1234. }
  1235. void handleJackPortConnectCallback(const jack_port_id_t a, const jack_port_id_t b, const bool connect)
  1236. {
  1237. jack_port_t* const jackPortA(jackbridge_port_by_id(fClient, a));
  1238. jack_port_t* const jackPortB(jackbridge_port_by_id(fClient, b));
  1239. const char* const fullPortNameA(jackbridge_port_name(jackPortA));
  1240. const char* const fullPortNameB(jackbridge_port_name(jackPortB));
  1241. CARLA_SAFE_ASSERT_RETURN(jackPortA != nullptr,);
  1242. CARLA_SAFE_ASSERT_RETURN(jackPortB != nullptr,);
  1243. CARLA_SAFE_ASSERT_RETURN(fullPortNameA != nullptr,);
  1244. CARLA_SAFE_ASSERT_RETURN(fullPortNameB != nullptr,);
  1245. const PortNameToId& portIdA(getPortNameToId(fullPortNameA));
  1246. const PortNameToId& portIdB(getPortNameToId(fullPortNameB));
  1247. if (portIdA.portId == -1 || portIdB.portId == -1)
  1248. return;
  1249. if (connect)
  1250. {
  1251. ConnectionToId connectionToId;
  1252. connectionToId.setData(fLastConnectionId++, portIdA.groupId, portIdA.portId, portIdB.groupId, portIdB.portId);
  1253. fUsedConnections.append(connectionToId);
  1254. char strBuf[STR_MAX+1];
  1255. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i", portIdA.groupId, portIdA.portId, portIdB.groupId, portIdB.portId);
  1256. callback(ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED, connectionToId.id, 0, 0, 0.0f, strBuf);
  1257. }
  1258. else
  1259. {
  1260. for (LinkedList<ConnectionToId>::Itenerator it = fUsedConnections.begin(); it.valid(); it.next())
  1261. {
  1262. const ConnectionToId& connectionToId(it.getValue());
  1263. if (connectionToId.groupOut == portIdA.groupId && connectionToId.portOut == portIdA.portId &&
  1264. connectionToId.groupIn == portIdB.groupId && connectionToId.portIn == portIdB.portId)
  1265. {
  1266. callback(ENGINE_CALLBACK_PATCHBAY_CONNECTION_REMOVED, static_cast<uint>(connectionToId.id), 0, 0, 0.0f, nullptr);
  1267. fUsedConnections.remove(it);
  1268. break;
  1269. }
  1270. }
  1271. }
  1272. }
  1273. void handleJackClientRenameCallback(const char* const oldName, const char* const newName)
  1274. {
  1275. for (LinkedList<GroupNameToId>::Itenerator it = fUsedGroupNames.begin(); it.valid(); it.next())
  1276. {
  1277. GroupNameToId& groupNameToId(it.getValue());
  1278. if (std::strcmp(groupNameToId.name, oldName) == 0)
  1279. {
  1280. groupNameToId.rename(newName);
  1281. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_RENAMED, static_cast<uint>(groupNameToId.id), 0, 0, 0.0f, newName);
  1282. break;
  1283. }
  1284. }
  1285. }
  1286. void handleJackPortRenameCallback(const jack_port_id_t port, const char* const oldName, const char* const newName)
  1287. {
  1288. jack_port_t* const jackPort(jackbridge_port_by_id(fClient, port));
  1289. const char* const portName(jackbridge_port_short_name(jackPort));
  1290. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr,);
  1291. CARLA_SAFE_ASSERT_RETURN(portName != nullptr,);
  1292. CarlaString groupName(newName);
  1293. groupName.truncate(groupName.rfind(portName)-1);
  1294. const int groupId(getGroupId(groupName));
  1295. CARLA_SAFE_ASSERT_RETURN(groupId >= 0,);
  1296. for (LinkedList<PortNameToId>::Itenerator it = fUsedPortNames.begin(); it.valid(); it.next())
  1297. {
  1298. PortNameToId& portNameId(it.getValue());
  1299. if (std::strcmp(portNameId.fullName, oldName) == 0)
  1300. {
  1301. CARLA_SAFE_ASSERT(portNameId.groupId == groupId);
  1302. portNameId.rename(portName, newName);
  1303. callback(ENGINE_CALLBACK_PATCHBAY_PORT_RENAMED, static_cast<uint>(groupId), portNameId.portId, 0, 0.0f, newName);
  1304. break;
  1305. }
  1306. }
  1307. }
  1308. #endif
  1309. void handleJackShutdownCallback()
  1310. {
  1311. for (uint i=0; i < pData->curPluginCount; ++i)
  1312. {
  1313. if (CarlaPlugin* const plugin = pData->plugins[i].plugin)
  1314. {
  1315. plugin->tryLock(true);
  1316. if (CarlaEngineJackClient* const client = (CarlaEngineJackClient*)plugin->getEngineClient())
  1317. client->invalidate();
  1318. plugin->unlock();
  1319. }
  1320. }
  1321. #ifndef BUILD_BRIDGE
  1322. carla_fill<jack_port_t*>(fRackPorts, kRackPortCount, nullptr);
  1323. #endif
  1324. fClient = nullptr;
  1325. callback(ENGINE_CALLBACK_QUIT, 0, 0, 0, 0.0f, nullptr);
  1326. }
  1327. // -------------------------------------------------------------------
  1328. private:
  1329. jack_client_t* fClient;
  1330. jack_position_t fTransportPos;
  1331. jack_transport_state_t fTransportState;
  1332. bool fFreewheel;
  1333. // -------------------------------------------------------------------
  1334. #ifdef BUILD_BRIDGE
  1335. bool fIsRunning;
  1336. #else
  1337. enum RackPorts {
  1338. kRackPortAudioIn1 = 0,
  1339. kRackPortAudioIn2 = 1,
  1340. kRackPortAudioOut1 = 2,
  1341. kRackPortAudioOut2 = 3,
  1342. kRackPortEventIn = 4,
  1343. kRackPortEventOut = 5,
  1344. kRackPortCount = 6
  1345. };
  1346. jack_port_t* fRackPorts[kRackPortCount];
  1347. struct GroupNameToId {
  1348. int id;
  1349. char name[STR_MAX+1];
  1350. void clear() noexcept
  1351. {
  1352. id = -1;
  1353. name[0] = '\0';
  1354. }
  1355. void setData(const int i, const char n[]) noexcept
  1356. {
  1357. id = i;
  1358. rename(n);
  1359. }
  1360. void rename(const char n[]) noexcept
  1361. {
  1362. std::strncpy(name, n, STR_MAX);
  1363. name[STR_MAX] = '\0';
  1364. }
  1365. bool operator==(const GroupNameToId& groupNameId) const noexcept
  1366. {
  1367. if (groupNameId.id != id)
  1368. return false;
  1369. if (std::strcmp(groupNameId.name, name) != 0)
  1370. return false;
  1371. return true;
  1372. }
  1373. bool operator!=(const GroupNameToId& groupNameId) const noexcept
  1374. {
  1375. return !operator==(groupNameId);
  1376. }
  1377. };
  1378. struct PortNameToId {
  1379. int groupId;
  1380. int portId;
  1381. char name[STR_MAX+1];
  1382. char fullName[STR_MAX+1]; // unique
  1383. void clear() noexcept
  1384. {
  1385. groupId = -1;
  1386. portId = -1;
  1387. name[0] = '\0';
  1388. fullName[0] = '\0';
  1389. }
  1390. void setData(const int gId, const int pId, const char n[], const char fn[]) noexcept
  1391. {
  1392. groupId = gId;
  1393. portId = pId;
  1394. rename(n, fn);
  1395. }
  1396. void rename(const char n[], const char fn[]) noexcept
  1397. {
  1398. std::strncpy(name, n, STR_MAX);
  1399. name[STR_MAX] = '\0';
  1400. std::strncpy(fullName, fn, STR_MAX);
  1401. fullName[STR_MAX] = '\0';
  1402. }
  1403. bool operator==(const PortNameToId& portNameId) noexcept
  1404. {
  1405. if (portNameId.groupId != groupId || portNameId.portId != portId)
  1406. return false;
  1407. if (std::strcmp(portNameId.name, name) != 0)
  1408. return false;
  1409. if (std::strcmp(portNameId.fullName, fullName) != 0)
  1410. return false;
  1411. return true;
  1412. }
  1413. bool operator!=(const PortNameToId& portNameId) noexcept
  1414. {
  1415. return !operator==(portNameId);
  1416. }
  1417. };
  1418. struct ConnectionToId {
  1419. uint id;
  1420. int groupOut;
  1421. int portOut;
  1422. int groupIn;
  1423. int portIn;
  1424. void clear() noexcept
  1425. {
  1426. id = 0;
  1427. groupOut = -1;
  1428. portOut = -1;
  1429. groupIn = -1;
  1430. portIn = -1;
  1431. }
  1432. void setData(const uint i, const int gout, const int pout, const int gin, const int pin) noexcept
  1433. {
  1434. id = i;
  1435. groupOut = gout;
  1436. portOut = pout;
  1437. groupIn = gin;
  1438. portIn = pin;
  1439. }
  1440. bool operator==(const ConnectionToId& connectionId) const noexcept
  1441. {
  1442. if (connectionId.id != id)
  1443. return false;
  1444. if (connectionId.groupOut != groupOut || connectionId.portOut != portOut)
  1445. return false;
  1446. if (connectionId.groupIn != groupIn || connectionId.portIn != portIn)
  1447. return false;
  1448. return true;
  1449. }
  1450. bool operator!=(const ConnectionToId& connectionId) const noexcept
  1451. {
  1452. return !operator==(connectionId);
  1453. }
  1454. };
  1455. int fLastGroupId;
  1456. int fLastPortId;
  1457. uint fLastConnectionId;
  1458. LinkedList<GroupNameToId> fUsedGroupNames;
  1459. LinkedList<PortNameToId> fUsedPortNames;
  1460. LinkedList<ConnectionToId> fUsedConnections;
  1461. LinkedList<int> fNewGroups;
  1462. int getGroupId(const char* const name)
  1463. {
  1464. CARLA_SAFE_ASSERT_RETURN(name != nullptr, -1);
  1465. for (LinkedList<GroupNameToId>::Itenerator it = fUsedGroupNames.begin(); it.valid(); it.next())
  1466. {
  1467. const GroupNameToId& groupNameId(it.getValue());
  1468. if (std::strcmp(groupNameId.name, name) == 0)
  1469. return groupNameId.id;
  1470. }
  1471. return -1;
  1472. }
  1473. const char* getGroupName(const int groupId)
  1474. {
  1475. static const char fallback[1] = { '\0' };
  1476. CARLA_SAFE_ASSERT_RETURN(groupId >= 0, fallback);
  1477. for (LinkedList<GroupNameToId>::Itenerator it = fUsedGroupNames.begin(); it.valid(); it.next())
  1478. {
  1479. const GroupNameToId& groupNameId(it.getValue());
  1480. if (groupNameId.id == groupId)
  1481. return groupNameId.name;
  1482. }
  1483. return fallback;
  1484. }
  1485. int getPortId(const char* const fullName)
  1486. {
  1487. CARLA_SAFE_ASSERT_RETURN(fullName != nullptr, -1);
  1488. for (LinkedList<PortNameToId>::Itenerator it = fUsedPortNames.begin(); it.valid(); it.next())
  1489. {
  1490. const PortNameToId& portNameId(it.getValue());
  1491. if (std::strcmp(portNameId.fullName, fullName) == 0)
  1492. return portNameId.portId;
  1493. }
  1494. return -1;
  1495. }
  1496. const PortNameToId& getPortNameToId(const char* const fullName)
  1497. {
  1498. static const PortNameToId fallback = { -1, -1, { '\0' }, { '\0' } };
  1499. CARLA_SAFE_ASSERT_RETURN(fullName != nullptr, fallback);
  1500. for (LinkedList<PortNameToId>::Itenerator it = fUsedPortNames.begin(); it.valid(); it.next())
  1501. {
  1502. const PortNameToId& portNameId(it.getValue());
  1503. if (std::strcmp(portNameId.fullName, fullName) == 0)
  1504. return portNameId;
  1505. }
  1506. return fallback;
  1507. }
  1508. void getFullPortName(const int portId, char nameBuf[STR_MAX+1])
  1509. {
  1510. for (LinkedList<PortNameToId>::Itenerator it = fUsedPortNames.begin(); it.valid(); it.next())
  1511. {
  1512. const PortNameToId& portNameId(it.getValue());
  1513. if (portNameId.portId == portId)
  1514. {
  1515. std::strncpy(nameBuf, portNameId.fullName, STR_MAX);
  1516. nameBuf[STR_MAX] = '\0';
  1517. return;
  1518. }
  1519. }
  1520. nameBuf[0] = '\0';
  1521. }
  1522. bool findPluginIdAndIcon(const char* const clientName, int& pluginId, PatchbayIcon& icon)
  1523. {
  1524. carla_debug("CarlaEngineJack::findPluginIdAndIcon(\"%s\", ...)", clientName);
  1525. for (uint i=0; i < pData->curPluginCount; ++i)
  1526. {
  1527. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1528. if (plugin == nullptr || ! plugin->isEnabled())
  1529. continue;
  1530. const CarlaEngineJackClient* const engClient((const CarlaEngineJackClient*)plugin->getEngineClient());
  1531. CARLA_SAFE_ASSERT_CONTINUE(engClient != nullptr);
  1532. const char* const thisClientName(engClient->getJackClientName());
  1533. CARLA_SAFE_ASSERT_CONTINUE(thisClientName != nullptr && thisClientName[0] != '\0');
  1534. if (std::strcmp(clientName, thisClientName) == 0)
  1535. {
  1536. pluginId = static_cast<int>(i);
  1537. icon = PATCHBAY_ICON_PLUGIN;
  1538. if (const char* const pluginIcon = plugin->getIconName())
  1539. {
  1540. if (std::strcmp(pluginIcon, "app") == 0 || std::strcmp(pluginIcon, "application") == 0)
  1541. icon = PATCHBAY_ICON_APPLICATION;
  1542. else if (std::strcmp(pluginIcon, "plugin") == 0)
  1543. icon = PATCHBAY_ICON_PLUGIN;
  1544. else if (std::strcmp(pluginIcon, "hardware") == 0)
  1545. icon = PATCHBAY_ICON_HARDWARE;
  1546. else if (std::strcmp(pluginIcon, "carla") == 0)
  1547. icon = PATCHBAY_ICON_CARLA;
  1548. else if (std::strcmp(pluginIcon, "distrho") == 0)
  1549. icon = PATCHBAY_ICON_DISTRHO;
  1550. else if (std::strcmp(pluginIcon, "file") == 0)
  1551. icon = PATCHBAY_ICON_FILE;
  1552. }
  1553. return true;
  1554. }
  1555. }
  1556. return false;
  1557. }
  1558. void initJackPatchbay(const char* const ourName)
  1559. {
  1560. CARLA_SAFE_ASSERT_RETURN(fLastGroupId == 0,);
  1561. CARLA_SAFE_ASSERT_RETURN(fLastPortId == 0,);
  1562. CARLA_SAFE_ASSERT_RETURN(fLastConnectionId == 0,);
  1563. CARLA_SAFE_ASSERT_RETURN(ourName != nullptr,);
  1564. // query initial jack ports
  1565. QStringList parsedGroups;
  1566. // our client
  1567. {
  1568. parsedGroups.append(QString(ourName));
  1569. GroupNameToId groupNameToId;
  1570. groupNameToId.setData(fLastGroupId++, ourName);
  1571. fUsedGroupNames.append(groupNameToId);
  1572. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED, 0 /* our client */, PATCHBAY_ICON_CARLA, -1, 0.0f, ourName);
  1573. }
  1574. if (const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, 0))
  1575. {
  1576. for (int i=0; ports[i] != nullptr; ++i)
  1577. {
  1578. jack_port_t* const jackPort(jackbridge_port_by_name(fClient, ports[i]));
  1579. const char* const portName(jackbridge_port_short_name(jackPort));
  1580. const char* const fullPortName(ports[i]);
  1581. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  1582. CARLA_SAFE_ASSERT_CONTINUE(portName != nullptr);
  1583. const int jackPortFlags(jackbridge_port_flags(jackPort));
  1584. int groupId = -1;
  1585. bool found;
  1586. CarlaString groupName(fullPortName);
  1587. groupName.truncate(groupName.rfind(portName, &found)-1);
  1588. CARLA_SAFE_ASSERT(found);
  1589. QString qGroupName(groupName.buffer());
  1590. if (parsedGroups.contains(qGroupName))
  1591. {
  1592. groupId = getGroupId(groupName);
  1593. CARLA_SAFE_ASSERT(groupId >= 0);
  1594. }
  1595. else
  1596. {
  1597. groupId = fLastGroupId++;
  1598. parsedGroups.append(qGroupName);
  1599. GroupNameToId groupNameToId;
  1600. groupNameToId.setData(groupId, groupName);
  1601. fUsedGroupNames.append(groupNameToId);
  1602. int pluginId = -1;
  1603. PatchbayIcon icon = (jackPortFlags & JackPortIsPhysical) ? PATCHBAY_ICON_HARDWARE : PATCHBAY_ICON_APPLICATION;
  1604. findPluginIdAndIcon(groupName, pluginId, icon);
  1605. callback(ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED, static_cast<uint>(groupId), icon, pluginId, 0.0f, groupName);
  1606. #if 0
  1607. if (pluginId >= 0)
  1608. {
  1609. CarlaPlugin* const plugin(getPlugin(static_cast<uint32_t>(pluginId)));
  1610. if (plugin != nullptr && plugin->isEnabled())
  1611. {
  1612. uint canvasPortFlags;
  1613. char strBuf[STR_MAX+1];
  1614. for (uint32_t j=0, count=plugin->getParameterCount(); j < count; ++j)
  1615. {
  1616. if ((plugin->getParameterData(j).hints & PARAMETER_IS_AUTOMABLE) == 0)
  1617. continue;
  1618. canvasPortFlags = PATCHBAY_PORT_TYPE_PARAMETER;
  1619. if (! plugin->isParameterOutput(j))
  1620. canvasPortFlags |= PATCHBAY_PORT_IS_INPUT;
  1621. plugin->getParameterName(j, strBuf);
  1622. const int pluginPortId = -static_cast<int>(j)-1;
  1623. const float pluginValue = plugin->getParameterRanges(j).getNormalizedValue(plugin->getParameterValue(j));
  1624. callback(ENGINE_CALLBACK_PATCHBAY_PORT_ADDED, static_cast<uint>(groupId), pluginPortId, static_cast<int>(canvasPortFlags), 0.0f, strBuf);
  1625. callback(ENGINE_CALLBACK_PATCHBAY_PORT_VALUE_CHANGED, static_cast<uint>(groupId), pluginPortId, 0, pluginValue, nullptr);
  1626. }
  1627. }
  1628. }
  1629. #endif
  1630. }
  1631. bool portIsInput = (jackPortFlags & JackPortIsInput);
  1632. bool portIsAudio = (std::strcmp(jackbridge_port_type(jackPort), JACK_DEFAULT_AUDIO_TYPE) == 0);
  1633. bool portIsCV = false;
  1634. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  1635. {
  1636. char* value = nullptr;
  1637. char* type = nullptr;
  1638. if (jackbridge_get_property(uuid, URI_CANVAS_CV, &value, &type) && value != nullptr && type != nullptr && std::strcmp(type, "text/plain") == 0)
  1639. portIsCV = (std::strcmp(value, "YES") == 0);
  1640. }
  1641. uint canvasPortFlags = 0x0;
  1642. canvasPortFlags |= portIsInput ? PATCHBAY_PORT_IS_INPUT : 0x0;
  1643. if (portIsCV)
  1644. canvasPortFlags |= PATCHBAY_PORT_TYPE_CV;
  1645. else if (portIsAudio)
  1646. canvasPortFlags |= PATCHBAY_PORT_TYPE_AUDIO;
  1647. else
  1648. canvasPortFlags |= PATCHBAY_PORT_TYPE_MIDI;
  1649. PortNameToId portNameToId;
  1650. portNameToId.setData(groupId, fLastPortId++, portName, fullPortName);
  1651. fUsedPortNames.append(portNameToId);
  1652. callback(ENGINE_CALLBACK_PATCHBAY_PORT_ADDED, static_cast<uint>(groupId), portNameToId.portId, static_cast<int>(canvasPortFlags), 0.0f, portName);
  1653. }
  1654. jackbridge_free(ports);
  1655. }
  1656. // query connections, after all ports are in place
  1657. if (const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, JackPortIsOutput))
  1658. {
  1659. for (int i=0; ports[i] != nullptr; ++i)
  1660. {
  1661. jack_port_t* const jackPort(jackbridge_port_by_name(fClient, ports[i]));
  1662. const char* const fullPortName(ports[i]);
  1663. const PortNameToId& thisPort(getPortNameToId(fullPortName));
  1664. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  1665. CARLA_SAFE_ASSERT_CONTINUE(thisPort.portId != -1);
  1666. if (const char** connections = jackbridge_port_get_all_connections(fClient, jackPort))
  1667. {
  1668. for (int j=0; connections[j] != nullptr; ++j)
  1669. {
  1670. const PortNameToId& targetPort(getPortNameToId(connections[j]));
  1671. ConnectionToId connectionToId;
  1672. connectionToId.setData(fLastConnectionId++, thisPort.groupId, thisPort.portId, targetPort.groupId, targetPort.portId);
  1673. fUsedConnections.append(connectionToId);
  1674. char strBuf[STR_MAX+1];
  1675. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i", thisPort.groupId, thisPort.portId, targetPort.groupId, targetPort.portId);
  1676. callback(ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED, connectionToId.id, 0, 0, 0.0f, strBuf);
  1677. }
  1678. jackbridge_free(connections);
  1679. }
  1680. }
  1681. jackbridge_free(ports);
  1682. }
  1683. }
  1684. #endif
  1685. // -------------------------------------
  1686. void processPlugin(CarlaPlugin* const plugin, const uint32_t nframes)
  1687. {
  1688. const uint32_t inCount(plugin->getAudioInCount());
  1689. const uint32_t outCount(plugin->getAudioOutCount());
  1690. float* inBuffer[inCount];
  1691. float* outBuffer[outCount];
  1692. float inPeaks[2] = { 0.0f };
  1693. float outPeaks[2] = { 0.0f };
  1694. for (uint32_t i=0; i < inCount; ++i)
  1695. {
  1696. CarlaEngineAudioPort* const port(plugin->getAudioInPort(i));
  1697. inBuffer[i] = port->getBuffer();
  1698. }
  1699. for (uint32_t i=0; i < outCount; ++i)
  1700. {
  1701. CarlaEngineAudioPort* const port(plugin->getAudioOutPort(i));
  1702. outBuffer[i] = port->getBuffer();
  1703. }
  1704. for (uint32_t i=0; i < inCount && i < 2; ++i)
  1705. {
  1706. for (uint32_t j=0; j < nframes; ++j)
  1707. {
  1708. const float absV(std::abs(inBuffer[i][j]));
  1709. if (absV > inPeaks[i])
  1710. inPeaks[i] = absV;
  1711. }
  1712. }
  1713. plugin->process(inBuffer, outBuffer, nframes);
  1714. for (uint32_t i=0; i < outCount && i < 2; ++i)
  1715. {
  1716. for (uint32_t j=0; j < nframes; ++j)
  1717. {
  1718. const float absV(std::abs(outBuffer[i][j]));
  1719. if (absV > outPeaks[i])
  1720. outPeaks[i] = absV;
  1721. }
  1722. }
  1723. setPluginPeaks(plugin->getId(), inPeaks, outPeaks);
  1724. }
  1725. #if 0
  1726. void latencyPlugin(CarlaPlugin* const plugin, jack_latency_callback_mode_t mode)
  1727. {
  1728. //const uint32_t inCount(plugin->audioInCount());
  1729. //const uint32_t outCount(plugin->audioOutCount());
  1730. const uint32_t latency(plugin->getLatencyInFrames());
  1731. if (latency == 0)
  1732. return;
  1733. //jack_latency_range_t range;
  1734. // TODO
  1735. if (mode == JackCaptureLatency)
  1736. {
  1737. for (uint32_t i=0; i < inCount; ++i)
  1738. {
  1739. uint32_t aOutI = (i >= outCount) ? outCount : i;
  1740. jack_port_t* const portIn = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioInPort(plugin, i))->kPort;
  1741. jack_port_t* const portOut = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioOutPort(plugin, aOutI))->kPort;
  1742. jackbridge_port_get_latency_range(portIn, mode, &range);
  1743. range.min += latency;
  1744. range.max += latency;
  1745. jackbridge_port_set_latency_range(portOut, mode, &range);
  1746. }
  1747. }
  1748. else
  1749. {
  1750. for (uint32_t i=0; i < outCount; ++i)
  1751. {
  1752. uint32_t aInI = (i >= inCount) ? inCount : i;
  1753. jack_port_t* const portIn = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioInPort(plugin, aInI))->kPort;
  1754. jack_port_t* const portOut = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioOutPort(plugin, i))->kPort;
  1755. jackbridge_port_get_latency_range(portOut, mode, &range);
  1756. range.min += latency;
  1757. range.max += latency;
  1758. jackbridge_port_set_latency_range(portIn, mode, &range);
  1759. }
  1760. }
  1761. }
  1762. #endif
  1763. // -------------------------------------
  1764. #define handlePtr ((CarlaEngineJack*)arg)
  1765. static int carla_jack_bufsize_callback(jack_nframes_t newBufferSize, void* arg)
  1766. {
  1767. handlePtr->handleJackBufferSizeCallback(newBufferSize);
  1768. return 0;
  1769. }
  1770. static int carla_jack_srate_callback(jack_nframes_t newSampleRate, void* arg)
  1771. {
  1772. handlePtr->handleJackSampleRateCallback(newSampleRate);
  1773. return 0;
  1774. }
  1775. static void carla_jack_freewheel_callback(int starting, void* arg)
  1776. {
  1777. handlePtr->handleJackFreewheelCallback(bool(starting));
  1778. }
  1779. static int carla_jack_process_callback(jack_nframes_t nframes, void* arg)
  1780. {
  1781. handlePtr->handleJackProcessCallback(nframes);
  1782. return 0;
  1783. }
  1784. #if 0
  1785. static void carla_jack_latency_callback(jack_latency_callback_mode_t mode, void* arg)
  1786. {
  1787. handlePtr->handleJackLatencyCallback(mode);
  1788. }
  1789. #endif
  1790. #ifndef BUILD_BRIDGE
  1791. static void carla_jack_client_registration_callback(const char* name, int reg, void* arg)
  1792. {
  1793. handlePtr->handleJackClientRegistrationCallback(name, (reg != 0));
  1794. }
  1795. static void carla_jack_port_registration_callback(jack_port_id_t port, int reg, void* arg)
  1796. {
  1797. handlePtr->handleJackPortRegistrationCallback(port, (reg != 0));
  1798. }
  1799. static void carla_jack_port_connect_callback(jack_port_id_t a, jack_port_id_t b, int connect, void* arg)
  1800. {
  1801. handlePtr->handleJackPortConnectCallback(a, b, (connect != 0));
  1802. }
  1803. static int carla_jack_client_rename_callback(const char* oldName, const char* newName, void* arg)
  1804. {
  1805. handlePtr->handleJackClientRenameCallback(oldName, newName);
  1806. return 0;
  1807. }
  1808. static int carla_jack_port_rename_callback(jack_port_id_t port, const char* oldName, const char* newName, void* arg)
  1809. {
  1810. handlePtr->handleJackPortRenameCallback(port, oldName, newName);
  1811. return 0;
  1812. }
  1813. #endif
  1814. static void carla_jack_shutdown_callback(void* arg)
  1815. {
  1816. handlePtr->handleJackShutdownCallback();
  1817. }
  1818. #undef handlePtr
  1819. // -------------------------------------------------------------------
  1820. #ifndef BUILD_BRIDGE
  1821. static int carla_jack_process_callback_plugin(jack_nframes_t nframes, void* arg)
  1822. {
  1823. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  1824. if (plugin != nullptr && plugin->isEnabled())
  1825. {
  1826. CarlaEngineJack* const engine((CarlaEngineJack*)plugin->getEngine());
  1827. CARLA_SAFE_ASSERT_RETURN(engine != nullptr,0);
  1828. if (plugin->tryLock(engine->fFreewheel))
  1829. {
  1830. plugin->initBuffers();
  1831. engine->saveTransportInfo();
  1832. engine->processPlugin(plugin, nframes);
  1833. plugin->unlock();
  1834. }
  1835. }
  1836. return 0;
  1837. }
  1838. # if 0
  1839. static void carla_jack_latency_callback_plugin(jack_latency_callback_mode_t mode, void* arg)
  1840. {
  1841. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  1842. if (plugin != nullptr && plugin->isEnabled())
  1843. {
  1844. CarlaEngineJack* const engine((CarlaEngineJack*)plugin->getEngine());
  1845. CARLA_SAFE_ASSERT_RETURN(engine != nullptr,);
  1846. engine->latencyPlugin(plugin, mode);
  1847. }
  1848. }
  1849. # endif
  1850. static void carla_jack_shutdown_callback_plugin(void* arg)
  1851. {
  1852. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  1853. if (plugin != nullptr /*&& plugin->isEnabled()*/)
  1854. {
  1855. CarlaEngine* const engine(plugin->getEngine());
  1856. CARLA_SAFE_ASSERT_RETURN(engine != nullptr,);
  1857. CarlaEngineJackClient* const engineClient((CarlaEngineJackClient*)plugin->getEngineClient());
  1858. CARLA_SAFE_ASSERT_RETURN(engineClient != nullptr,);
  1859. plugin->tryLock(true);
  1860. engineClient->invalidate();
  1861. plugin->unlock();
  1862. engine->callback(ENGINE_CALLBACK_PLUGIN_UNAVAILABLE, plugin->getId(), 0, 0, 0.0f, "Killed by JACK");
  1863. }
  1864. }
  1865. #endif
  1866. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJack)
  1867. };
  1868. // -----------------------------------------------------------------------
  1869. CarlaEngine* CarlaEngine::newJack()
  1870. {
  1871. carla_debug("CarlaEngine::newJack()");
  1872. return new CarlaEngineJack();
  1873. }
  1874. // -----------------------------------------------------------------------
  1875. CARLA_BACKEND_END_NAMESPACE