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.

1989 lines
65KB

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