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.

1986 lines
65KB

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