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.

2157 lines
71KB

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