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.

2296 lines
76KB

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