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.

2188 lines
76KB

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