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.

3064 lines
107KB

  1. /*
  2. * Carla Plugin Host
  3. * Copyright (C) 2011-2019 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. #ifdef USING_JUCE
  27. # if defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))
  28. # pragma GCC diagnostic push
  29. # pragma GCC diagnostic ignored "-Wconversion"
  30. # pragma GCC diagnostic ignored "-Weffc++"
  31. # pragma GCC diagnostic ignored "-Wsign-conversion"
  32. # pragma GCC diagnostic ignored "-Wundef"
  33. # endif
  34. # include "AppConfig.h"
  35. # include "juce_events/juce_events.h"
  36. # if defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))
  37. # pragma GCC diagnostic pop
  38. # endif
  39. #endif
  40. #ifdef __SSE2_MATH__
  41. # include <xmmintrin.h>
  42. #endif
  43. // must be last
  44. #include "jackbridge/JackBridge.hpp"
  45. #define URI_CANVAS_ICON "http://kxstudio.sf.net/ns/canvas/icon"
  46. CARLA_BACKEND_START_NAMESPACE
  47. class CarlaEngineJack;
  48. class CarlaEngineJackClient;
  49. #ifndef BUILD_BRIDGE
  50. // FIXME temporary function while jack2 client uuids are broken
  51. static int buggy_jack2_uuid_parse(const char* b, jack_uuid_t* u)
  52. {
  53. if (sscanf (b, P_UINT64, u) == 1) {
  54. if (*u < (0x1LLU << 32)) {
  55. // FIXME: bug in jack2, client bit not set
  56. // *u = (0x2LLU << 32) | *u;
  57. return -1;
  58. }
  59. return 0;
  60. }
  61. return -1;
  62. }
  63. #endif
  64. // -----------------------------------------------------------------------
  65. // Fallback data
  66. static const EngineEvent kFallbackJackEngineEvent = { kEngineEventTypeNull, 0, 0, {{ kEngineControlEventTypeNull, 0, 0.0f }} };
  67. // -----------------------------------------------------------------------
  68. // Carla Engine Port removal helper
  69. class CarlaEngineJackAudioPort;
  70. class CarlaEngineJackCVPort;
  71. class CarlaEngineJackEventPort;
  72. struct JackPortDeletionCallback {
  73. virtual ~JackPortDeletionCallback() noexcept {}
  74. virtual void jackAudioPortDeleted(CarlaEngineJackAudioPort* const) noexcept = 0;
  75. virtual void jackCVPortDeleted(CarlaEngineJackCVPort* const) noexcept = 0;
  76. virtual void jackEventPortDeleted(CarlaEngineJackEventPort* const) noexcept = 0;
  77. };
  78. // -----------------------------------------------------------------------
  79. // Carla Engine JACK-Audio port
  80. class CarlaEngineJackAudioPort : public CarlaEngineAudioPort
  81. {
  82. public:
  83. CarlaEngineJackAudioPort(const CarlaEngineClient& client, const bool isInputPort, const uint32_t indexOffset, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  84. : CarlaEngineAudioPort(client, isInputPort, indexOffset),
  85. fJackClient(jackClient),
  86. fJackPort(jackPort),
  87. kDeletionCallback(delCallback)
  88. {
  89. carla_debug("CarlaEngineJackAudioPort::CarlaEngineJackAudioPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  90. switch (kClient.getEngine().getProccessMode())
  91. {
  92. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  93. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  94. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  95. #ifndef BUILD_BRIDGE
  96. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  97. jackbridge_set_property(jackClient, uuid, JACKEY_SIGNAL_TYPE, "AUDIO", "text/plain");
  98. #endif
  99. break;
  100. default:
  101. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  102. break;
  103. }
  104. }
  105. ~CarlaEngineJackAudioPort() noexcept override
  106. {
  107. carla_debug("CarlaEngineJackAudioPort::~CarlaEngineJackAudioPort()");
  108. if (fJackClient != nullptr && fJackPort != nullptr)
  109. {
  110. #ifndef BUILD_BRIDGE
  111. try {
  112. if (const jack_uuid_t uuid = jackbridge_port_uuid(fJackPort))
  113. jackbridge_remove_property(fJackClient, uuid, JACKEY_SIGNAL_TYPE);
  114. } CARLA_SAFE_EXCEPTION("Audio port remove meta type");
  115. #endif
  116. try {
  117. jackbridge_port_unregister(fJackClient, fJackPort);
  118. } CARLA_SAFE_EXCEPTION("Audio port unregister");
  119. fJackClient = nullptr;
  120. fJackPort = nullptr;
  121. }
  122. if (kDeletionCallback != nullptr)
  123. kDeletionCallback->jackAudioPortDeleted(this);
  124. }
  125. void initBuffer() noexcept override
  126. {
  127. if (fJackPort == nullptr)
  128. return CarlaEngineAudioPort::initBuffer();
  129. const uint32_t bufferSize(kClient.getEngine().getBufferSize());
  130. try {
  131. fBuffer = (float*)jackbridge_port_get_buffer(fJackPort, bufferSize);
  132. }
  133. catch(...) {
  134. fBuffer = nullptr;
  135. return;
  136. }
  137. if (! kIsInput)
  138. carla_zeroFloats(fBuffer, bufferSize);
  139. }
  140. void invalidate() noexcept
  141. {
  142. fJackClient = nullptr;
  143. fJackPort = nullptr;
  144. }
  145. private:
  146. jack_client_t* fJackClient;
  147. jack_port_t* fJackPort;
  148. JackPortDeletionCallback* const kDeletionCallback;
  149. friend class CarlaEngineJackClient;
  150. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackAudioPort)
  151. };
  152. // -----------------------------------------------------------------------
  153. // Carla Engine JACK-CV port
  154. class CarlaEngineJackCVPort : public CarlaEngineCVPort
  155. {
  156. public:
  157. CarlaEngineJackCVPort(const CarlaEngineClient& client, const bool isInputPort, const uint32_t indexOffset, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  158. : CarlaEngineCVPort(client, isInputPort, indexOffset),
  159. fJackClient(jackClient),
  160. fJackPort(jackPort),
  161. kDeletionCallback(delCallback)
  162. {
  163. carla_debug("CarlaEngineJackCVPort::CarlaEngineJackCVPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  164. switch (kClient.getEngine().getProccessMode())
  165. {
  166. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  167. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  168. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  169. #ifndef BUILD_BRIDGE
  170. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  171. jackbridge_set_property(jackClient, uuid, JACKEY_SIGNAL_TYPE, "CV", "text/plain");
  172. #endif
  173. break;
  174. default:
  175. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  176. break;
  177. }
  178. }
  179. ~CarlaEngineJackCVPort() noexcept override
  180. {
  181. carla_debug("CarlaEngineJackCVPort::~CarlaEngineJackCVPort()");
  182. if (fJackClient != nullptr && fJackPort != nullptr)
  183. {
  184. #ifndef BUILD_BRIDGE
  185. try {
  186. if (const jack_uuid_t uuid = jackbridge_port_uuid(fJackPort))
  187. jackbridge_remove_property(fJackClient, uuid, JACKEY_SIGNAL_TYPE);
  188. } CARLA_SAFE_EXCEPTION("CV port remove meta type");
  189. #endif
  190. try {
  191. jackbridge_port_unregister(fJackClient, fJackPort);
  192. } CARLA_SAFE_EXCEPTION("CV port unregister");
  193. fJackClient = nullptr;
  194. fJackPort = nullptr;
  195. }
  196. if (kDeletionCallback != nullptr)
  197. kDeletionCallback->jackCVPortDeleted(this);
  198. }
  199. void initBuffer() noexcept override
  200. {
  201. if (fJackPort == nullptr)
  202. return CarlaEngineCVPort::initBuffer();
  203. const uint32_t bufferSize(kClient.getEngine().getBufferSize());
  204. try {
  205. fBuffer = (float*)jackbridge_port_get_buffer(fJackPort, bufferSize);
  206. }
  207. catch(...) {
  208. fBuffer = nullptr;
  209. return;
  210. }
  211. if (! kIsInput)
  212. carla_zeroFloats(fBuffer, bufferSize);
  213. }
  214. void invalidate() noexcept
  215. {
  216. fJackClient = nullptr;
  217. fJackPort = nullptr;
  218. }
  219. private:
  220. jack_client_t* fJackClient;
  221. jack_port_t* fJackPort;
  222. JackPortDeletionCallback* const kDeletionCallback;
  223. friend class CarlaEngineJackClient;
  224. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackCVPort)
  225. };
  226. // -----------------------------------------------------------------------
  227. // Carla Engine JACK-Event port
  228. class CarlaEngineJackEventPort : public CarlaEngineEventPort
  229. {
  230. public:
  231. CarlaEngineJackEventPort(const CarlaEngineClient& client, const bool isInputPort, const uint32_t indexOffset, jack_client_t* const jackClient, jack_port_t* const jackPort, JackPortDeletionCallback* const delCallback) noexcept
  232. : CarlaEngineEventPort(client, isInputPort, indexOffset),
  233. fJackClient(jackClient),
  234. fJackPort(jackPort),
  235. fJackBuffer(nullptr),
  236. fRetEvent(kFallbackJackEngineEvent),
  237. kDeletionCallback(delCallback)
  238. {
  239. carla_debug("CarlaEngineJackEventPort::CarlaEngineJackEventPort(%s, %p, %p)", bool2str(isInputPort), jackClient, jackPort);
  240. switch (kClient.getEngine().getProccessMode())
  241. {
  242. case ENGINE_PROCESS_MODE_SINGLE_CLIENT:
  243. case ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS:
  244. CARLA_SAFE_ASSERT_RETURN(jackClient != nullptr && jackPort != nullptr,);
  245. break;
  246. default:
  247. CARLA_SAFE_ASSERT(jackClient == nullptr && jackPort == nullptr);
  248. break;
  249. }
  250. }
  251. ~CarlaEngineJackEventPort() noexcept override
  252. {
  253. carla_debug("CarlaEngineJackEventPort::~CarlaEngineJackEventPort()");
  254. if (fJackClient != nullptr && fJackPort != nullptr)
  255. {
  256. try {
  257. jackbridge_port_unregister(fJackClient, fJackPort);
  258. } CARLA_SAFE_EXCEPTION("Event port unregister");
  259. fJackClient = nullptr;
  260. fJackPort = nullptr;
  261. }
  262. if (kDeletionCallback != nullptr)
  263. kDeletionCallback->jackEventPortDeleted(this);
  264. }
  265. void initBuffer() noexcept override
  266. {
  267. if (fJackPort == nullptr)
  268. return CarlaEngineEventPort::initBuffer();
  269. try {
  270. fJackBuffer = jackbridge_port_get_buffer(fJackPort, kClient.getEngine().getBufferSize());
  271. }
  272. catch(...) {
  273. fJackBuffer = nullptr;
  274. return;
  275. }
  276. if (! kIsInput)
  277. jackbridge_midi_clear_buffer(fJackBuffer);
  278. }
  279. uint32_t getEventCount() const noexcept override
  280. {
  281. if (fJackPort == nullptr)
  282. return CarlaEngineEventPort::getEventCount();
  283. CARLA_SAFE_ASSERT_RETURN(kIsInput, 0);
  284. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, 0);
  285. try {
  286. return jackbridge_midi_get_event_count(fJackBuffer);
  287. } CARLA_SAFE_EXCEPTION_RETURN("jack_midi_get_event_count", 0);
  288. }
  289. const EngineEvent& getEvent(const uint32_t index) const noexcept override
  290. {
  291. if (fJackPort == nullptr)
  292. return CarlaEngineEventPort::getEvent(index);
  293. CARLA_SAFE_ASSERT_RETURN(kIsInput, kFallbackJackEngineEvent);
  294. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, kFallbackJackEngineEvent);
  295. return getEventUnchecked(index);
  296. }
  297. const EngineEvent& getEventUnchecked(const uint32_t index) const noexcept override
  298. {
  299. jack_midi_event_t jackEvent;
  300. bool test = false;
  301. try {
  302. test = jackbridge_midi_event_get(&jackEvent, fJackBuffer, index);
  303. } CARLA_SAFE_EXCEPTION_RETURN("jack_midi_event_get", kFallbackJackEngineEvent);
  304. if (! test)
  305. return kFallbackJackEngineEvent;
  306. CARLA_SAFE_ASSERT_RETURN(jackEvent.size < 0xFF /* uint8_t max */, kFallbackJackEngineEvent);
  307. uint8_t port;
  308. if (kIndexOffset < 0xFF /* uint8_t max */)
  309. {
  310. port = static_cast<uint8_t>(kIndexOffset);
  311. }
  312. else
  313. {
  314. port = 0;
  315. carla_safe_assert_uint("kIndexOffset < 0xFF", __FILE__, __LINE__, kIndexOffset);
  316. }
  317. fRetEvent.time = jackEvent.time;
  318. fRetEvent.fillFromMidiData(static_cast<uint8_t>(jackEvent.size), jackEvent.buffer, port);
  319. return fRetEvent;
  320. }
  321. bool writeControlEvent(const uint32_t time, const uint8_t channel, const EngineControlEventType type, const uint16_t param, const float value) noexcept override
  322. {
  323. if (fJackPort == nullptr)
  324. return CarlaEngineEventPort::writeControlEvent(time, channel, type, param, value);
  325. CARLA_SAFE_ASSERT_RETURN(! kIsInput, false);
  326. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, false);
  327. CARLA_SAFE_ASSERT_RETURN(type != kEngineControlEventTypeNull, false);
  328. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS, false);
  329. CARLA_SAFE_ASSERT_RETURN(param < MAX_MIDI_CONTROL, false);
  330. CARLA_SAFE_ASSERT(value >= 0.0f && value <= 1.0f);
  331. if (type == kEngineControlEventTypeParameter) {
  332. CARLA_SAFE_ASSERT(! MIDI_IS_CONTROL_BANK_SELECT(param));
  333. }
  334. uint8_t data[3] = { 0, 0, 0 };
  335. EngineControlEvent ctrlEvent = { type, param, value };
  336. const uint8_t size = ctrlEvent.convertToMidiData(channel, data);
  337. if (size == 0)
  338. return false;
  339. try {
  340. return jackbridge_midi_event_write(fJackBuffer, time, data, size);
  341. } CARLA_SAFE_EXCEPTION_RETURN("jack_midi_event_write", false);
  342. }
  343. bool writeMidiEvent(const uint32_t time, const uint8_t channel, const uint8_t size, const uint8_t* const data) noexcept override
  344. {
  345. if (fJackPort == nullptr)
  346. return CarlaEngineEventPort::writeMidiEvent(time, channel, size, data);
  347. CARLA_SAFE_ASSERT_RETURN(! kIsInput, false);
  348. CARLA_SAFE_ASSERT_RETURN(fJackBuffer != nullptr, false);
  349. CARLA_SAFE_ASSERT_RETURN(channel < MAX_MIDI_CHANNELS, false);
  350. CARLA_SAFE_ASSERT_RETURN(size > 0, false);
  351. CARLA_SAFE_ASSERT_RETURN(data != nullptr, false);
  352. jack_midi_data_t jdata[size];
  353. jdata[0] = static_cast<jack_midi_data_t>(MIDI_GET_STATUS_FROM_DATA(data) + channel);
  354. for (uint8_t i=1; i < size; ++i)
  355. jdata[i] = data[i];
  356. try {
  357. return jackbridge_midi_event_write(fJackBuffer, time, jdata, size);
  358. } CARLA_SAFE_EXCEPTION_RETURN("jack_midi_event_write", false);
  359. }
  360. void invalidate() noexcept
  361. {
  362. fJackClient = nullptr;
  363. fJackPort = nullptr;
  364. }
  365. private:
  366. jack_client_t* fJackClient;
  367. jack_port_t* fJackPort;
  368. void* fJackBuffer;
  369. mutable EngineEvent fRetEvent;
  370. JackPortDeletionCallback* const kDeletionCallback;
  371. friend class CarlaEngineJackClient;
  372. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackEventPort)
  373. };
  374. // -----------------------------------------------------------------------
  375. // Jack Engine client
  376. class CarlaEngineJackClient : public CarlaEngineClient,
  377. private JackPortDeletionCallback
  378. {
  379. public:
  380. CarlaEngineJackClient(const CarlaEngine& engine, jack_client_t* const jackClient)
  381. : CarlaEngineClient(engine),
  382. fJackClient(jackClient),
  383. fUseClient(engine.getProccessMode() == ENGINE_PROCESS_MODE_SINGLE_CLIENT || engine.getProccessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS),
  384. fAudioPorts(),
  385. fCVPorts(),
  386. fEventPorts(),
  387. fPreRenameMutex(),
  388. fPreRenameConnections()
  389. {
  390. carla_debug("CarlaEngineJackClient::CarlaEngineJackClient(%p)", jackClient);
  391. if (fUseClient)
  392. {
  393. CARLA_SAFE_ASSERT(jackClient != nullptr);
  394. }
  395. else
  396. {
  397. CARLA_SAFE_ASSERT(jackClient == nullptr);
  398. }
  399. }
  400. ~CarlaEngineJackClient() noexcept override
  401. {
  402. carla_debug("CarlaEngineJackClient::~CarlaEngineJackClient()");
  403. if (getProcessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS && fJackClient != nullptr) // FIXME
  404. jackbridge_client_close(fJackClient);
  405. // ports must have been deleted by now!
  406. //fAudioPorts.clear();
  407. //fCVPorts.clear();
  408. //fEventPorts.clear();
  409. }
  410. void activate() noexcept override
  411. {
  412. carla_debug("CarlaEngineJackClient::activate()");
  413. if (getProcessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  414. {
  415. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr && ! isActive(),);
  416. try {
  417. jackbridge_activate(fJackClient);
  418. } catch(...) {}
  419. }
  420. CarlaEngineClient::activate();
  421. const CarlaMutexLocker cml(fPreRenameMutex);
  422. if (fJackClient != nullptr)
  423. {
  424. // restore pre-rename connections
  425. const char* portNameA = nullptr;
  426. const char* portNameB = nullptr;
  427. bool doConnection = false;
  428. for (CarlaStringList::Itenerator it = fPreRenameConnections.begin2(); it.valid(); it.next())
  429. {
  430. const bool connectNow = doConnection;
  431. doConnection = !doConnection;
  432. if (connectNow)
  433. portNameB = it.getValue(nullptr);
  434. else
  435. portNameA = it.getValue(nullptr);
  436. if (! connectNow)
  437. continue;
  438. CARLA_SAFE_ASSERT_CONTINUE(portNameA != nullptr && portNameA[0] != '\0');
  439. CARLA_SAFE_ASSERT_CONTINUE(portNameB != nullptr && portNameB[0] != '\0');
  440. jackbridge_connect(fJackClient, portNameA, portNameB);
  441. }
  442. }
  443. fPreRenameConnections.clear();
  444. }
  445. void deactivate() noexcept override
  446. {
  447. carla_debug("CarlaEngineJackClient::deactivate()");
  448. if (getProcessMode() == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  449. {
  450. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr && isActive(),);
  451. try {
  452. jackbridge_deactivate(fJackClient);
  453. } catch(...) {}
  454. }
  455. CarlaEngineClient::deactivate();
  456. }
  457. bool isOk() const noexcept override
  458. {
  459. if (fUseClient)
  460. return (fJackClient != nullptr);
  461. return CarlaEngineClient::isOk();
  462. }
  463. CarlaEnginePort* addPort(const EnginePortType portType, const char* const name, const bool isInput, const uint32_t indexOffset) override
  464. {
  465. carla_debug("CarlaEngineJackClient::addPort(%i:%s, \"%s\", %s)", portType, EnginePortType2Str(portType), name, bool2str(isInput));
  466. jack_port_t* jackPort = nullptr;
  467. const char* realName = name;
  468. // Create JACK port first, if needed
  469. if (fUseClient)
  470. {
  471. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr, nullptr);
  472. realName = _getUniquePortName(name);
  473. switch (portType)
  474. {
  475. case kEnginePortTypeNull:
  476. break;
  477. case kEnginePortTypeAudio:
  478. jackPort = jackbridge_port_register(fJackClient, realName, JACK_DEFAULT_AUDIO_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  479. break;
  480. case kEnginePortTypeCV:
  481. jackPort = jackbridge_port_register(fJackClient, realName, JACK_DEFAULT_AUDIO_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  482. break;
  483. case kEnginePortTypeEvent:
  484. jackPort = jackbridge_port_register(fJackClient, realName, JACK_DEFAULT_MIDI_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  485. break;
  486. }
  487. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr, nullptr);
  488. }
  489. // Create Engine port
  490. switch (portType)
  491. {
  492. case kEnginePortTypeNull:
  493. break;
  494. case kEnginePortTypeAudio: {
  495. _addAudioPortName(isInput, realName);
  496. if (realName != name) delete[] realName;
  497. CarlaEngineJackAudioPort* const enginePort(new CarlaEngineJackAudioPort(*this, isInput, indexOffset, fJackClient, jackPort, this));
  498. fAudioPorts.append(enginePort);
  499. return enginePort;
  500. }
  501. case kEnginePortTypeCV: {
  502. _addCVPortName(isInput, realName);
  503. if (realName != name) delete[] realName;
  504. CarlaEngineJackCVPort* const enginePort(new CarlaEngineJackCVPort(*this, isInput, indexOffset, fJackClient, jackPort, this));
  505. fCVPorts.append(enginePort);
  506. return enginePort;
  507. }
  508. case kEnginePortTypeEvent: {
  509. _addEventPortName(isInput, realName);
  510. if (realName != name) delete[] realName;
  511. CarlaEngineJackEventPort* const enginePort(new CarlaEngineJackEventPort(*this, isInput, indexOffset, fJackClient, jackPort, this));
  512. fEventPorts.append(enginePort);
  513. return enginePort;
  514. }
  515. }
  516. carla_stderr("CarlaEngineJackClient::addPort(%i, \"%s\", %s) - invalid type", portType, name, bool2str(isInput));
  517. return nullptr;
  518. }
  519. void invalidate() noexcept
  520. {
  521. for (LinkedList<CarlaEngineJackAudioPort*>::Itenerator it = fAudioPorts.begin2(); it.valid(); it.next())
  522. {
  523. CarlaEngineJackAudioPort* const port(it.getValue(nullptr));
  524. CARLA_SAFE_ASSERT_CONTINUE(port != nullptr);
  525. port->invalidate();
  526. }
  527. for (LinkedList<CarlaEngineJackCVPort*>::Itenerator it = fCVPorts.begin2(); it.valid(); it.next())
  528. {
  529. CarlaEngineJackCVPort* const port(it.getValue(nullptr));
  530. CARLA_SAFE_ASSERT_CONTINUE(port != nullptr);
  531. port->invalidate();
  532. }
  533. for (LinkedList<CarlaEngineJackEventPort*>::Itenerator it = fEventPorts.begin2(); it.valid(); it.next())
  534. {
  535. CarlaEngineJackEventPort* const port(it.getValue(nullptr));
  536. CARLA_SAFE_ASSERT_CONTINUE(port != nullptr);
  537. port->invalidate();
  538. }
  539. fJackClient = nullptr;
  540. CarlaEngineClient::deactivate();
  541. }
  542. const char* getJackClientName() const noexcept
  543. {
  544. CARLA_SAFE_ASSERT_RETURN(fJackClient != nullptr, nullptr);
  545. try {
  546. return jackbridge_get_client_name(fJackClient);
  547. } CARLA_SAFE_EXCEPTION_RETURN("jack_get_client_name", nullptr);
  548. }
  549. void jackAudioPortDeleted(CarlaEngineJackAudioPort* const port) noexcept override
  550. {
  551. fAudioPorts.removeAll(port);
  552. }
  553. void jackCVPortDeleted(CarlaEngineJackCVPort* const port) noexcept override
  554. {
  555. fCVPorts.removeAll(port);
  556. }
  557. void jackEventPortDeleted(CarlaEngineJackEventPort* const port) noexcept override
  558. {
  559. fEventPorts.removeAll(port);
  560. }
  561. bool renameInSingleClient(const CarlaString& newClientName)
  562. {
  563. const CarlaString clientNamePrefix(newClientName + ":");
  564. return _renamePorts(fAudioPorts, clientNamePrefix) &&
  565. _renamePorts(fCVPorts, clientNamePrefix) &&
  566. _renamePorts(fEventPorts, clientNamePrefix);
  567. }
  568. void closeForRename(jack_client_t* const newClient, const CarlaString& newClientName) noexcept
  569. {
  570. if (fJackClient != nullptr)
  571. {
  572. if (isActive())
  573. {
  574. {
  575. const CarlaString clientNamePrefix(newClientName + ":");
  576. // store current client connections
  577. const CarlaMutexLocker cml(fPreRenameMutex);
  578. fPreRenameConnections.clear();
  579. _savePortsConnections(fAudioPorts, clientNamePrefix);
  580. _savePortsConnections(fCVPorts, clientNamePrefix);
  581. _savePortsConnections(fEventPorts, clientNamePrefix);
  582. }
  583. try {
  584. jackbridge_deactivate(fJackClient);
  585. } catch(...) {}
  586. }
  587. try {
  588. jackbridge_client_close(fJackClient);
  589. } catch(...) {}
  590. invalidate();
  591. }
  592. fAudioPorts.clear();
  593. fCVPorts.clear();
  594. fEventPorts.clear();
  595. _clearPorts();
  596. fJackClient = newClient;
  597. }
  598. private:
  599. jack_client_t* fJackClient;
  600. const bool fUseClient;
  601. LinkedList<CarlaEngineJackAudioPort*> fAudioPorts;
  602. LinkedList<CarlaEngineJackCVPort*> fCVPorts;
  603. LinkedList<CarlaEngineJackEventPort*> fEventPorts;
  604. CarlaMutex fPreRenameMutex;
  605. CarlaStringList fPreRenameConnections;
  606. template<typename T>
  607. bool _renamePorts(const LinkedList<T*>& t, const CarlaString& clientNamePrefix)
  608. {
  609. for (typename LinkedList<T*>::Itenerator it = t.begin2(); it.valid(); it.next())
  610. {
  611. T* const port(it.getValue(nullptr));
  612. CARLA_SAFE_ASSERT_CONTINUE(port != nullptr);
  613. CARLA_SAFE_ASSERT_CONTINUE(port->fJackPort != nullptr);
  614. const char* shortPortName(jackbridge_port_short_name(port->fJackPort));
  615. CARLA_SAFE_ASSERT_CONTINUE(shortPortName != nullptr && shortPortName[0] != '\0');
  616. const char* const oldClientNameSep(std::strstr(shortPortName, ":"));
  617. CARLA_SAFE_ASSERT_CONTINUE(oldClientNameSep != nullptr && oldClientNameSep[0] != '\0' && oldClientNameSep[1] != '\0');
  618. shortPortName += oldClientNameSep-shortPortName + 1;
  619. const CarlaString newPortName(clientNamePrefix + shortPortName);
  620. if (! jackbridge_port_rename(fJackClient, port->fJackPort, newPortName))
  621. return false;
  622. }
  623. return true;
  624. }
  625. template<typename T>
  626. void _savePortsConnections(const LinkedList<T*>& t, const CarlaString& clientNamePrefix)
  627. {
  628. for (typename LinkedList<T*>::Itenerator it = t.begin2(); it.valid(); it.next())
  629. {
  630. T* const port(it.getValue(nullptr));
  631. CARLA_SAFE_ASSERT_CONTINUE(port != nullptr);
  632. CARLA_SAFE_ASSERT_CONTINUE(port->fJackPort != nullptr);
  633. const char* const shortPortName(jackbridge_port_short_name(port->fJackPort));
  634. CARLA_SAFE_ASSERT_CONTINUE(shortPortName != nullptr && shortPortName[0] != '\0');
  635. const CarlaString portName(clientNamePrefix + shortPortName);
  636. if (const char** const connections = jackbridge_port_get_all_connections(fJackClient, port->fJackPort))
  637. {
  638. for (int i=0; connections[i] != nullptr; ++i)
  639. {
  640. if (port->kIsInput)
  641. {
  642. fPreRenameConnections.append(connections[i]);
  643. fPreRenameConnections.append(portName);
  644. }
  645. else
  646. {
  647. fPreRenameConnections.append(portName);
  648. fPreRenameConnections.append(connections[i]);
  649. }
  650. }
  651. jackbridge_free(connections);
  652. }
  653. }
  654. }
  655. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackClient)
  656. };
  657. // -----------------------------------------------------------------------
  658. // Jack Engine
  659. class CarlaEngineJack : public CarlaEngine
  660. #ifndef BUILD_BRIDGE
  661. , private CarlaThread
  662. #endif
  663. {
  664. public:
  665. CarlaEngineJack()
  666. : CarlaEngine(),
  667. #ifndef BUILD_BRIDGE
  668. CarlaThread("CarlaEngineJackCallbacks"),
  669. #endif
  670. fClient(nullptr),
  671. fExternalPatchbayHost(true),
  672. fExternalPatchbayOsc(true),
  673. fFreewheel(false),
  674. #ifdef BUILD_BRIDGE
  675. fIsRunning(false)
  676. #else
  677. fTimebaseMaster(false),
  678. fTimebaseRolling(false),
  679. fTimebaseUsecs(0),
  680. fUsedGroups(),
  681. fUsedPorts(),
  682. fUsedConnections(),
  683. fPatchbayProcThreadProtectionMutex(),
  684. fRetConns(),
  685. fPostPonedEvents(),
  686. fPostPonedEventsMutex(),
  687. fIsInternalClient(false)
  688. #endif
  689. {
  690. carla_debug("CarlaEngineJack::CarlaEngineJack()");
  691. #ifdef BUILD_BRIDGE
  692. pData->options.processMode = ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS;
  693. #else
  694. carla_zeroPointers(fRackPorts, kRackPortCount);
  695. #endif
  696. }
  697. ~CarlaEngineJack() noexcept override
  698. {
  699. carla_debug("CarlaEngineJack::~CarlaEngineJack()");
  700. CARLA_SAFE_ASSERT(fClient == nullptr);
  701. #ifndef BUILD_BRIDGE
  702. fUsedGroups.clear();
  703. fUsedPorts.clear();
  704. fUsedConnections.clear();
  705. CARLA_SAFE_ASSERT(fPostPonedEvents.count() == 0);
  706. #endif
  707. }
  708. // -------------------------------------------------------------------
  709. // Maximum values
  710. uint getMaxClientNameSize() const noexcept override
  711. {
  712. #ifndef BUILD_BRIDGE
  713. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT || pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  714. #endif
  715. {
  716. try {
  717. return static_cast<uint>(jackbridge_client_name_size()-1);
  718. } CARLA_SAFE_EXCEPTION_RETURN("jack_client_name_size", 32);
  719. }
  720. return CarlaEngine::getMaxClientNameSize();
  721. }
  722. uint getMaxPortNameSize() const noexcept override
  723. {
  724. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT || pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  725. {
  726. try {
  727. return static_cast<uint>(jackbridge_port_name_size()-1);
  728. } CARLA_SAFE_EXCEPTION_RETURN("jack_port_name_size", 255);
  729. }
  730. return CarlaEngine::getMaxPortNameSize();
  731. }
  732. // -------------------------------------------------------------------
  733. // Virtual, per-engine type calls
  734. bool init(const char* const clientName) override
  735. {
  736. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr || (clientName != nullptr && clientName[0] != '\0'), false);
  737. CARLA_SAFE_ASSERT_RETURN(jackbridge_is_ok(), false);
  738. carla_debug("CarlaEngineJack::init(\"%s\")", clientName);
  739. fFreewheel = false;
  740. fExternalPatchbayHost = true;
  741. fExternalPatchbayOsc = true;
  742. CarlaString truncatedClientName;
  743. if (fClient == nullptr && clientName != nullptr)
  744. {
  745. truncatedClientName = clientName;
  746. truncatedClientName.truncate(getMaxClientNameSize());
  747. }
  748. #ifdef BUILD_BRIDGE
  749. fIsRunning = true;
  750. if (! pData->init(truncatedClientName))
  751. {
  752. close();
  753. setLastError("Failed to init internal data");
  754. return false;
  755. }
  756. if (pData->bufferSize == 0 || carla_isEqual(pData->sampleRate, 0.0))
  757. {
  758. // open temp client to get initial buffer-size and sample-rate values
  759. if (jack_client_t* const tmpClient = jackbridge_client_open(truncatedClientName, JackNoStartServer, nullptr))
  760. {
  761. pData->bufferSize = jackbridge_get_buffer_size(tmpClient);
  762. pData->sampleRate = jackbridge_get_sample_rate(tmpClient);
  763. jackbridge_client_close(tmpClient);
  764. }
  765. else
  766. {
  767. close();
  768. setLastError("Failed to init temporary jack client");
  769. return false;
  770. }
  771. }
  772. return true;
  773. #else
  774. if (fClient == nullptr && clientName != nullptr)
  775. fClient = jackbridge_client_open(truncatedClientName, JackNullOption, nullptr);
  776. if (fClient == nullptr)
  777. {
  778. setLastError("Failed to create new JACK client");
  779. return false;
  780. }
  781. const char* const jackClientName = jackbridge_get_client_name(fClient);
  782. if (! pData->init(jackClientName))
  783. {
  784. jackbridge_client_close(fClient);
  785. fClient = nullptr;
  786. setLastError("Failed to init internal data");
  787. return false;
  788. }
  789. const EngineOptions& opts(pData->options);
  790. pData->bufferSize = jackbridge_get_buffer_size(fClient);
  791. pData->sampleRate = jackbridge_get_sample_rate(fClient);
  792. pData->initTime(opts.transportExtra);
  793. jackbridge_set_thread_init_callback(fClient, carla_jack_thread_init_callback, nullptr);
  794. jackbridge_set_buffer_size_callback(fClient, carla_jack_bufsize_callback, this);
  795. jackbridge_set_sample_rate_callback(fClient, carla_jack_srate_callback, this);
  796. jackbridge_set_freewheel_callback(fClient, carla_jack_freewheel_callback, this);
  797. jackbridge_set_latency_callback(fClient, carla_jack_latency_callback, this);
  798. jackbridge_set_process_callback(fClient, carla_jack_process_callback, this);
  799. jackbridge_on_shutdown(fClient, carla_jack_shutdown_callback, this);
  800. fTimebaseRolling = false;
  801. if (opts.transportMode == ENGINE_TRANSPORT_MODE_JACK)
  802. fTimebaseMaster = jackbridge_set_timebase_callback(fClient, true, carla_jack_timebase_callback, this);
  803. else
  804. fTimebaseMaster = false;
  805. if (opts.processMode != ENGINE_PROCESS_MODE_PATCHBAY)
  806. initJackPatchbay(true, false, jackClientName);
  807. jackbridge_set_client_registration_callback(fClient, carla_jack_client_registration_callback, this);
  808. jackbridge_set_port_registration_callback(fClient, carla_jack_port_registration_callback, this);
  809. jackbridge_set_port_connect_callback(fClient, carla_jack_port_connect_callback, this);
  810. jackbridge_set_port_rename_callback(fClient, carla_jack_port_rename_callback, this);
  811. jackbridge_set_xrun_callback(fClient, carla_jack_xrun_callback, this);
  812. if (opts.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK || opts.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  813. {
  814. fRackPorts[kRackPortAudioIn1] = jackbridge_port_register(fClient, "audio-in1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  815. fRackPorts[kRackPortAudioIn2] = jackbridge_port_register(fClient, "audio-in2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  816. fRackPorts[kRackPortAudioOut1] = jackbridge_port_register(fClient, "audio-out1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  817. fRackPorts[kRackPortAudioOut2] = jackbridge_port_register(fClient, "audio-out2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  818. fRackPorts[kRackPortEventIn] = jackbridge_port_register(fClient, "events-in", JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0);
  819. fRackPorts[kRackPortEventOut] = jackbridge_port_register(fClient, "events-out", JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput, 0);
  820. if (opts.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  821. {
  822. // FIXME?
  823. pData->graph.create(0, 0);
  824. }
  825. else
  826. {
  827. pData->graph.create(2, 2);
  828. // pData->graph.setUsingExternalHost(true);
  829. // pData->graph.setUsingExternalOSC(true);
  830. patchbayRefresh(true, false, false);
  831. }
  832. }
  833. if (const char* const uuidchar = jackbridge_client_get_uuid(fClient))
  834. {
  835. jack_uuid_t uuid;
  836. if (buggy_jack2_uuid_parse(uuidchar, &uuid) == 0)
  837. {
  838. #if defined(HAVE_LIBLO) && !defined(BUILD_BRIDGE)
  839. const CarlaString& tcp(pData->osc.getServerPathTCP());
  840. const CarlaString& udp(pData->osc.getServerPathUDP());
  841. if (tcp.isNotEmpty())
  842. jackbridge_set_property(fClient, uuid,
  843. "https://kx.studio/ns/carla/osc-tcp", tcp.buffer(), "text/plain");
  844. if (tcp.isNotEmpty())
  845. jackbridge_set_property(fClient, uuid,
  846. "https://kx.studio/ns/carla/osc-udp", udp.buffer(), "text/plain");
  847. #endif
  848. }
  849. }
  850. if (jackbridge_activate(fClient))
  851. {
  852. if (opts.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK ||
  853. opts.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  854. {
  855. if (pData->options.audioDevice != nullptr &&
  856. std::strcmp(pData->options.audioDevice, "Auto-Connect ON") == 0 &&
  857. std::getenv("LADISH_APP_NAME") == nullptr &&
  858. std::getenv("NSM_URL") == nullptr)
  859. {
  860. char strBuf[STR_MAX];
  861. if (jackbridge_port_by_name(fClient, "system:capture_1") != nullptr)
  862. {
  863. std::snprintf(strBuf, STR_MAX, "%s:audio-in1", jackClientName);
  864. strBuf[STR_MAX-1] = '\0';
  865. jackbridge_connect(fClient, "system:capture_1", strBuf);
  866. std::snprintf(strBuf, STR_MAX, "%s:audio-in2", jackClientName);
  867. strBuf[STR_MAX-1] = '\0';
  868. if (jackbridge_port_by_name(fClient, "system:capture_2") != nullptr)
  869. jackbridge_connect(fClient, "system:capture_2", strBuf);
  870. else
  871. jackbridge_connect(fClient, "system:capture_1", strBuf);
  872. }
  873. if (jackbridge_port_by_name(fClient, "system:playback_1") != nullptr)
  874. {
  875. std::snprintf(strBuf, STR_MAX, "%s:audio-out1", jackClientName);
  876. strBuf[STR_MAX-1] = '\0';
  877. jackbridge_connect(fClient, strBuf, "system:playback_1");
  878. std::snprintf(strBuf, STR_MAX, "%s:audio-out2", jackClientName);
  879. strBuf[STR_MAX-1] = '\0';
  880. if (jackbridge_port_by_name(fClient, "system:playback_2") != nullptr)
  881. jackbridge_connect(fClient, strBuf, "system:playback_2");
  882. else
  883. jackbridge_connect(fClient, strBuf, "system:playback_1");
  884. }
  885. }
  886. }
  887. startThread();
  888. callback(true, true,
  889. ENGINE_CALLBACK_ENGINE_STARTED, 0,
  890. opts.processMode,
  891. opts.transportMode,
  892. static_cast<int>(pData->bufferSize),
  893. static_cast<float>(pData->sampleRate),
  894. getCurrentDriverName());
  895. return true;
  896. }
  897. if (opts.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK ||
  898. opts.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  899. {
  900. pData->graph.destroy();
  901. }
  902. pData->close();
  903. jackbridge_client_close(fClient);
  904. fClient = nullptr;
  905. setLastError("Failed to activate the JACK client");
  906. return false;
  907. #endif
  908. }
  909. #ifndef BUILD_BRIDGE
  910. bool initInternal(jack_client_t* const client)
  911. {
  912. fClient = client;
  913. fIsInternalClient = true;
  914. return init(nullptr);
  915. }
  916. #endif
  917. bool close() override
  918. {
  919. carla_debug("CarlaEngineJack::close()");
  920. #ifdef BUILD_BRIDGE
  921. fClient = nullptr;
  922. fIsRunning = false;
  923. CarlaEngine::close();
  924. return true;
  925. #else
  926. stopThread(-1);
  927. fPostPonedEvents.clear();
  928. CARLA_SAFE_ASSERT_RETURN_ERR(fClient != nullptr, "JACK Client is null");
  929. // deactivate and close client
  930. jackbridge_deactivate(fClient);
  931. jackbridge_client_close(fClient);
  932. // clear engine data
  933. CarlaEngine::close();
  934. fUsedGroups.clear();
  935. fUsedPorts.clear();
  936. fUsedConnections.clear();
  937. fPostPonedEvents.clear();
  938. // clear rack/patchbay stuff
  939. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK ||
  940. pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  941. {
  942. carla_zeroPointers(fRackPorts, kRackPortCount);
  943. pData->graph.destroy();
  944. }
  945. fClient = nullptr;
  946. return true;
  947. #endif
  948. }
  949. bool isRunning() const noexcept override
  950. {
  951. #ifdef BUILD_BRIDGE
  952. return (fClient != nullptr || fIsRunning);
  953. #else
  954. return (fClient != nullptr);
  955. #endif
  956. }
  957. bool isOffline() const noexcept override
  958. {
  959. return fFreewheel;
  960. }
  961. EngineType getType() const noexcept override
  962. {
  963. return kEngineTypeJack;
  964. }
  965. const char* getCurrentDriverName() const noexcept override
  966. {
  967. return "JACK";
  968. }
  969. #ifndef BUILD_BRIDGE
  970. float getDSPLoad() const noexcept override
  971. {
  972. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, 0.0f);
  973. return jackbridge_cpu_load(fClient);
  974. }
  975. #endif
  976. EngineTimeInfo getTimeInfo() const noexcept override
  977. {
  978. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK)
  979. return CarlaEngine::getTimeInfo();
  980. if (pData->options.processMode != ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  981. return CarlaEngine::getTimeInfo();
  982. jack_position_t jpos;
  983. // invalidate
  984. jpos.unique_1 = 1;
  985. jpos.unique_2 = 2;
  986. EngineTimeInfo timeInfo;
  987. const bool playing = jackbridge_transport_query(fClient, &jpos) == JackTransportRolling;
  988. if (jpos.unique_1 != jpos.unique_2)
  989. {
  990. timeInfo.playing = false;
  991. timeInfo.frame = 0;
  992. timeInfo.usecs = 0;
  993. timeInfo.bbt.valid = false;
  994. return timeInfo;
  995. }
  996. timeInfo.playing = playing;
  997. timeInfo.frame = jpos.frame;
  998. timeInfo.usecs = jpos.usecs;
  999. if (jpos.valid & JackPositionBBT)
  1000. {
  1001. timeInfo.bbt.valid = true;
  1002. timeInfo.bbt.bar = jpos.bar;
  1003. timeInfo.bbt.beat = jpos.beat;
  1004. timeInfo.bbt.tick = jpos.tick;
  1005. timeInfo.bbt.barStartTick = jpos.bar_start_tick;
  1006. timeInfo.bbt.beatsPerBar = jpos.beats_per_bar;
  1007. timeInfo.bbt.beatType = jpos.beat_type;
  1008. timeInfo.bbt.ticksPerBeat = jpos.ticks_per_beat;
  1009. timeInfo.bbt.beatsPerMinute = jpos.beats_per_minute;
  1010. }
  1011. else
  1012. {
  1013. timeInfo.bbt.valid = false;
  1014. }
  1015. return timeInfo;
  1016. }
  1017. #ifndef BUILD_BRIDGE
  1018. void setOption(const EngineOption option, const int value, const char* const valueStr) noexcept override
  1019. {
  1020. if (option == ENGINE_OPTION_TRANSPORT_MODE && fClient != nullptr)
  1021. {
  1022. CARLA_SAFE_ASSERT_RETURN(value >= ENGINE_TRANSPORT_MODE_DISABLED && value <= ENGINE_TRANSPORT_MODE_JACK,);
  1023. if (value == ENGINE_TRANSPORT_MODE_JACK)
  1024. {
  1025. fTimebaseMaster = jackbridge_set_timebase_callback(fClient, true, carla_jack_timebase_callback, this);
  1026. }
  1027. else
  1028. {
  1029. // jack transport cannot be disabled in multi-client
  1030. callback(true, true,
  1031. ENGINE_CALLBACK_TRANSPORT_MODE_CHANGED, 0,
  1032. ENGINE_TRANSPORT_MODE_JACK,
  1033. 0, 0, 0.0f,
  1034. pData->options.transportExtra);
  1035. CARLA_SAFE_ASSERT_RETURN(pData->options.processMode != ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS,);
  1036. jackbridge_release_timebase(fClient);
  1037. fTimebaseMaster = false;
  1038. }
  1039. }
  1040. CarlaEngine::setOption(option, value, valueStr);
  1041. }
  1042. #endif
  1043. CarlaEngineClient* addClient(CarlaPlugin* const plugin) override
  1044. {
  1045. jack_client_t* client = nullptr;
  1046. #ifndef BUILD_BRIDGE
  1047. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  1048. {
  1049. client = fClient;
  1050. }
  1051. else if (pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  1052. #endif
  1053. {
  1054. client = jackbridge_client_open(plugin->getName(), JackNullOption, nullptr);
  1055. CARLA_SAFE_ASSERT_RETURN(client != nullptr, nullptr);
  1056. jackbridge_set_thread_init_callback(client, carla_jack_thread_init_callback, nullptr);
  1057. #ifndef BUILD_BRIDGE
  1058. /*
  1059. jackbridge_set_buffer_size_callback(fClient, carla_jack_bufsize_callback_plugin, plugin);
  1060. jackbridge_set_sample_rate_callback(fClient, carla_jack_srate_callback_plugin, plugin);
  1061. */
  1062. jackbridge_set_latency_callback(client, carla_jack_latency_callback_plugin, plugin);
  1063. jackbridge_set_process_callback(client, carla_jack_process_callback_plugin, plugin);
  1064. jackbridge_on_shutdown(client, carla_jack_shutdown_callback_plugin, plugin);
  1065. if (const char* const uuidchar = jackbridge_client_get_uuid(client))
  1066. {
  1067. jack_uuid_t uuid;
  1068. if (buggy_jack2_uuid_parse(uuidchar, &uuid) == 0)
  1069. {
  1070. char strBufId[24];
  1071. std::snprintf(strBufId, 24, "%u", plugin->getId());
  1072. strBufId[23] = '\0';
  1073. jackbridge_set_property(client, uuid,
  1074. "https://kx.studio/ns/carla/plugin-id",
  1075. strBufId,
  1076. "http://www.w3.org/2001/XMLSchema#integer");
  1077. if (const char* const pluginIcon = plugin->getIconName())
  1078. jackbridge_set_property(client, uuid,
  1079. "https://kx.studio/ns/carla/plugin-icon",
  1080. pluginIcon,
  1081. "text/plain");
  1082. }
  1083. }
  1084. #else
  1085. fClient = client;
  1086. pData->bufferSize = jackbridge_get_buffer_size(client);
  1087. pData->sampleRate = jackbridge_get_sample_rate(client);
  1088. pData->initTime(nullptr);
  1089. jackbridge_set_buffer_size_callback(client, carla_jack_bufsize_callback, this);
  1090. jackbridge_set_sample_rate_callback(client, carla_jack_srate_callback, this);
  1091. jackbridge_set_freewheel_callback(client, carla_jack_freewheel_callback, this);
  1092. jackbridge_set_latency_callback(client, carla_jack_latency_callback, this);
  1093. jackbridge_set_process_callback(client, carla_jack_process_callback, this);
  1094. jackbridge_on_shutdown(client, carla_jack_shutdown_callback, this);
  1095. #endif
  1096. }
  1097. return new CarlaEngineJackClient(*this, client);
  1098. }
  1099. #ifndef BUILD_BRIDGE
  1100. const char* renamePlugin(const uint id, const char* const newName) override
  1101. {
  1102. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK ||
  1103. pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  1104. {
  1105. return CarlaEngine::renamePlugin(id, newName);
  1106. }
  1107. CARLA_SAFE_ASSERT_RETURN(pData->plugins != nullptr, nullptr);
  1108. CARLA_SAFE_ASSERT_RETURN(pData->curPluginCount != 0, nullptr);
  1109. CARLA_SAFE_ASSERT_RETURN(id < pData->curPluginCount, nullptr);
  1110. CARLA_SAFE_ASSERT_RETURN(newName != nullptr && newName[0] != '\0', nullptr);
  1111. CarlaPlugin* const plugin(pData->plugins[id].plugin);
  1112. CARLA_SAFE_ASSERT_RETURN_ERRN(plugin != nullptr, "Could not find plugin to rename");
  1113. CARLA_SAFE_ASSERT_RETURN_ERRN(plugin->getId() == id, "Invalid engine internal data");
  1114. // before we stop the engine thread we might need to get the plugin data
  1115. const bool needsReinit = (pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS);
  1116. const CarlaStateSave* saveStatePtr = nullptr;
  1117. if (needsReinit)
  1118. {
  1119. const CarlaStateSave& saveState(plugin->getStateSave());
  1120. saveStatePtr = &saveState;
  1121. }
  1122. CarlaString uniqueName;
  1123. try {
  1124. const char* const tmpName = getUniquePluginName(newName);
  1125. uniqueName = tmpName;
  1126. delete[] tmpName;
  1127. } CARLA_SAFE_EXCEPTION("JACK renamePlugin getUniquePluginName");
  1128. if (uniqueName.isEmpty())
  1129. {
  1130. setLastError("Failed to request new unique plugin name");
  1131. return nullptr;
  1132. }
  1133. const ScopedThreadStopper sts(this);
  1134. // rename on client client mode, just rename the ports
  1135. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  1136. {
  1137. CarlaEngineJackClient* const client((CarlaEngineJackClient*)plugin->getEngineClient());
  1138. if (! client->renameInSingleClient(uniqueName))
  1139. {
  1140. setLastError("Failed to rename some JACK ports, does your JACK version support proper port renaming?");
  1141. return nullptr;
  1142. }
  1143. }
  1144. // rename in multiple client mode
  1145. else if (pData->options.processMode == ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  1146. {
  1147. CarlaEngineJackClient* const client((CarlaEngineJackClient*)plugin->getEngineClient());
  1148. // we should not be able to do this, jack really needs to allow client rename
  1149. if (jack_client_t* const jackClient = jackbridge_client_open(uniqueName, JackNullOption, nullptr))
  1150. {
  1151. // get new client name
  1152. uniqueName = jackbridge_get_client_name(jackClient);
  1153. // close client
  1154. client->closeForRename(jackClient, uniqueName);
  1155. // disable plugin
  1156. plugin->setEnabled(false);
  1157. // set new client data
  1158. jackbridge_set_latency_callback(jackClient, carla_jack_latency_callback_plugin, plugin);
  1159. jackbridge_set_process_callback(jackClient, carla_jack_process_callback_plugin, plugin);
  1160. jackbridge_on_shutdown(jackClient, carla_jack_shutdown_callback_plugin, plugin);
  1161. // NOTE: jack1 locks up here
  1162. if (jackbridge_get_version_string() != nullptr)
  1163. jackbridge_set_thread_init_callback(jackClient, carla_jack_thread_init_callback, nullptr);
  1164. /* The following code is because of a tricky situation.
  1165. We cannot lock or do jack operations during jack callbacks on jack1. jack2 events are asynchronous.
  1166. When we close the client jack will trigger unregister-port callbacks, which we handle on a separate thread ASAP.
  1167. But before that happens we already registered a new client with the same ports (the "renamed" one),
  1168. and at this point the port we receive during that callback is actually the new one from the new client..
  1169. JACK2 seems to be reusing ports to save space, which is understandable.
  1170. Anyway, this means we have to remove all our port-related data before the new client ports are created.
  1171. (we also stop the separate jack-events thread to avoid any race conditions while modying our port data) */
  1172. stopThread(-1);
  1173. const uint groupId(fUsedGroups.getGroupId(plugin->getName()));
  1174. if (groupId > 0)
  1175. {
  1176. for (LinkedList<PortNameToId>::Itenerator it = fUsedPorts.list.begin2(); it.valid(); it.next())
  1177. {
  1178. for (LinkedList<ConnectionToId>::Itenerator it2 = fUsedConnections.list.begin2(); it2.valid(); it2.next())
  1179. {
  1180. static ConnectionToId connectionFallback = { 0, 0, 0, 0, 0 };
  1181. ConnectionToId& connectionToId = it2.getValue(connectionFallback);
  1182. CARLA_SAFE_ASSERT_CONTINUE(connectionToId.id != 0);
  1183. if (connectionToId.groupA != groupId && connectionToId.groupB != groupId)
  1184. continue;
  1185. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1186. ENGINE_CALLBACK_PATCHBAY_CONNECTION_REMOVED,
  1187. connectionToId.id,
  1188. 0, 0, 0, 0.0f, nullptr);
  1189. fUsedConnections.list.remove(it2);
  1190. }
  1191. static PortNameToId portNameFallback = { 0, 0, { '\0' }, { '\0' } };
  1192. PortNameToId& portNameToId(it.getValue(portNameFallback));
  1193. CARLA_SAFE_ASSERT_CONTINUE(portNameToId.group != 0);
  1194. if (portNameToId.group != groupId)
  1195. continue;
  1196. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1197. ENGINE_CALLBACK_PATCHBAY_PORT_REMOVED,
  1198. portNameToId.group,
  1199. static_cast<int>(portNameToId.port),
  1200. 0, 0, 0.0f, nullptr);
  1201. fUsedPorts.list.remove(it);
  1202. }
  1203. }
  1204. startThread();
  1205. }
  1206. else
  1207. {
  1208. setLastError("Failed to create new JACK client");
  1209. return nullptr;
  1210. }
  1211. }
  1212. // Rename
  1213. plugin->setName(uniqueName);
  1214. if (needsReinit)
  1215. {
  1216. // reload plugin to recreate its ports
  1217. plugin->reload();
  1218. plugin->loadStateSave(*saveStatePtr);
  1219. plugin->setEnabled(true);
  1220. }
  1221. return plugin->getName();
  1222. }
  1223. // -------------------------------------------------------------------
  1224. // Patchbay
  1225. bool patchbayConnect(const bool external,
  1226. const uint groupA, const uint portA, const uint groupB, const uint portB) override
  1227. {
  1228. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  1229. if (pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY && ! external)
  1230. return CarlaEngine::patchbayConnect(false, groupA, portA, groupB, portB);
  1231. const char* const fullPortNameA = fUsedPorts.getFullPortName(groupA, portA);
  1232. CARLA_SAFE_ASSERT_RETURN(fullPortNameA != nullptr && fullPortNameA[0] != '\0', false);
  1233. const char* const fullPortNameB = fUsedPorts.getFullPortName(groupB, portB);
  1234. CARLA_SAFE_ASSERT_RETURN(fullPortNameB != nullptr && fullPortNameB[0] != '\0', false);
  1235. if (! jackbridge_connect(fClient, fullPortNameA, fullPortNameB))
  1236. {
  1237. setLastError("JACK operation failed");
  1238. return false;
  1239. }
  1240. return true;
  1241. }
  1242. bool patchbayDisconnect(const bool external, const uint connectionId) override
  1243. {
  1244. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  1245. if (pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY && ! external)
  1246. return CarlaEngine::patchbayDisconnect(false, connectionId);
  1247. ConnectionToId connectionToId = { 0, 0, 0, 0, 0 };
  1248. {
  1249. const CarlaMutexLocker cml(fUsedConnections.mutex);
  1250. for (LinkedList<ConnectionToId>::Itenerator it = fUsedConnections.list.begin2(); it.valid(); it.next())
  1251. {
  1252. connectionToId = it.getValue(connectionToId);
  1253. CARLA_SAFE_ASSERT_CONTINUE(connectionToId.id != 0);
  1254. if (connectionToId.id == connectionId)
  1255. break;
  1256. }
  1257. }
  1258. if (connectionToId.id == 0 || connectionToId.id != connectionId)
  1259. {
  1260. setLastError("Failed to find the requested connection");
  1261. return false;
  1262. }
  1263. const char* const fullPortNameA = fUsedPorts.getFullPortName(connectionToId.groupA, connectionToId.portA);
  1264. CARLA_SAFE_ASSERT_RETURN(fullPortNameA != nullptr && fullPortNameA[0] != '\0', false);
  1265. const char* const fullPortNameB = fUsedPorts.getFullPortName(connectionToId.groupB, connectionToId.portB);
  1266. CARLA_SAFE_ASSERT_RETURN(fullPortNameB != nullptr && fullPortNameB[0] != '\0', false);
  1267. if (! jackbridge_disconnect(fClient, fullPortNameA, fullPortNameB))
  1268. {
  1269. setLastError("JACK operation failed");
  1270. return false;
  1271. }
  1272. return true;
  1273. }
  1274. bool patchbayRefresh(const bool sendHost, const bool sendOSC, const bool external) override
  1275. {
  1276. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, false);
  1277. carla_debug("patchbayRefresh(%s, %s, %s)", bool2str(sendHost), bool2str(sendOSC), bool2str(external));
  1278. if (pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  1279. {
  1280. if (sendHost)
  1281. {
  1282. fExternalPatchbayHost = external;
  1283. pData->graph.setUsingExternalHost(external);
  1284. }
  1285. if (sendOSC)
  1286. {
  1287. fExternalPatchbayOsc = external;
  1288. pData->graph.setUsingExternalOSC(external);
  1289. }
  1290. if (! external)
  1291. return CarlaEngine::patchbayRefresh(sendHost, sendOSC, false);
  1292. }
  1293. fUsedGroups.clear();
  1294. fUsedPorts.clear();
  1295. fUsedConnections.clear();
  1296. initJackPatchbay(sendHost, sendOSC, jackbridge_get_client_name(fClient));
  1297. return true;
  1298. }
  1299. // -------------------------------------------------------------------
  1300. // Transport
  1301. void transportPlay() noexcept override
  1302. {
  1303. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK)
  1304. return CarlaEngine::transportPlay();
  1305. if (fClient != nullptr)
  1306. {
  1307. if (! pData->timeInfo.bbt.valid)
  1308. {
  1309. // old timebase master no longer active, make ourselves master again
  1310. pData->time.setNeedsReset();
  1311. fTimebaseMaster = jackbridge_set_timebase_callback(fClient, true, carla_jack_timebase_callback, this);
  1312. }
  1313. try {
  1314. jackbridge_transport_start(fClient);
  1315. } catch(...) {}
  1316. }
  1317. }
  1318. void transportPause() noexcept override
  1319. {
  1320. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK)
  1321. return CarlaEngine::transportPause();
  1322. if (fClient != nullptr)
  1323. {
  1324. try {
  1325. jackbridge_transport_stop(fClient);
  1326. } catch(...) {}
  1327. }
  1328. }
  1329. void transportBPM(const double bpm) noexcept override
  1330. {
  1331. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK || fTimebaseMaster)
  1332. return CarlaEngine::transportBPM(bpm);
  1333. if (fClient == nullptr)
  1334. return;
  1335. jack_position_t jpos;
  1336. // invalidate
  1337. jpos.unique_1 = 1;
  1338. jpos.unique_2 = 2;
  1339. jackbridge_transport_query(fClient, &jpos);
  1340. if (jpos.unique_1 == jpos.unique_2 && (jpos.valid & JackPositionBBT) != 0)
  1341. {
  1342. carla_stdout("NOTE: Changing BPM without being JACK timebase master");
  1343. jpos.beats_per_minute = bpm;
  1344. try {
  1345. jackbridge_transport_reposition(fClient, &jpos);
  1346. } catch(...) {}
  1347. }
  1348. }
  1349. void transportRelocate(const uint64_t frame) noexcept override
  1350. {
  1351. if (pData->options.transportMode != ENGINE_TRANSPORT_MODE_JACK)
  1352. return CarlaEngine::transportRelocate(frame);
  1353. if (fClient != nullptr)
  1354. {
  1355. try {
  1356. jackbridge_transport_locate(fClient, static_cast<jack_nframes_t>(frame));
  1357. } catch(...) {}
  1358. }
  1359. }
  1360. // -------------------------------------------------------------------
  1361. // Patchbay stuff
  1362. const char* const* getPatchbayConnections(const bool external) const override
  1363. {
  1364. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr, nullptr);
  1365. carla_debug("CarlaEngineJack::getPatchbayConnections(%s)", bool2str(external));
  1366. if (pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY && ! external)
  1367. return CarlaEngine::getPatchbayConnections(external);
  1368. CarlaStringList connList;
  1369. if (const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, JackPortIsOutput))
  1370. {
  1371. for (int i=0; ports[i] != nullptr; ++i)
  1372. {
  1373. const jack_port_t* const jackPort(jackbridge_port_by_name(fClient, ports[i]));
  1374. const char* const fullPortName(ports[i]);
  1375. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  1376. if (const char** const connections = jackbridge_port_get_all_connections(fClient, jackPort))
  1377. {
  1378. for (int j=0; connections[j] != nullptr; ++j)
  1379. {
  1380. connList.append(fullPortName);
  1381. connList.append(connections[j]);
  1382. }
  1383. jackbridge_free(connections);
  1384. }
  1385. }
  1386. jackbridge_free(ports);
  1387. }
  1388. if (connList.count() == 0)
  1389. return nullptr;
  1390. fRetConns = connList.toCharStringListPtr();
  1391. return fRetConns;
  1392. }
  1393. void restorePatchbayConnection(const bool external, const char* const connSource, const char* const connTarget) override
  1394. {
  1395. CARLA_SAFE_ASSERT_RETURN(fClient != nullptr,);
  1396. CARLA_SAFE_ASSERT_RETURN(connSource != nullptr && connSource[0] != '\0',);
  1397. CARLA_SAFE_ASSERT_RETURN(connTarget != nullptr && connTarget[0] != '\0',);
  1398. carla_debug("CarlaEngineJack::restorePatchbayConnection(%s, \"%s\", \"%s\")",
  1399. bool2str(external), connSource, connTarget);
  1400. if (pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY && ! external)
  1401. return CarlaEngine::restorePatchbayConnection(external, connSource, connTarget);
  1402. if (const jack_port_t* const port = jackbridge_port_by_name(fClient, connSource))
  1403. {
  1404. if (jackbridge_port_by_name(fClient, connTarget) == nullptr)
  1405. return;
  1406. if (! jackbridge_port_connected_to(port, connTarget))
  1407. jackbridge_connect(fClient, connSource, connTarget);
  1408. }
  1409. }
  1410. #endif
  1411. // -------------------------------------------------------------------
  1412. protected:
  1413. void handleJackBufferSizeCallback(const uint32_t newBufferSize)
  1414. {
  1415. if (pData->bufferSize == newBufferSize)
  1416. return;
  1417. #ifndef BUILD_BRIDGE
  1418. const CarlaMutexLocker cml(fPatchbayProcThreadProtectionMutex);
  1419. #endif
  1420. pData->bufferSize = newBufferSize;
  1421. bufferSizeChanged(newBufferSize);
  1422. }
  1423. void handleJackSampleRateCallback(const double newSampleRate)
  1424. {
  1425. if (carla_isEqual(pData->sampleRate, newSampleRate))
  1426. return;
  1427. #ifndef BUILD_BRIDGE
  1428. const CarlaMutexLocker cml(fPatchbayProcThreadProtectionMutex);
  1429. #endif
  1430. pData->sampleRate = newSampleRate;
  1431. sampleRateChanged(newSampleRate);
  1432. }
  1433. void handleJackFreewheelCallback(const bool isFreewheel)
  1434. {
  1435. if (fFreewheel == isFreewheel)
  1436. return;
  1437. #ifndef BUILD_BRIDGE
  1438. const CarlaMutexLocker cml(fPatchbayProcThreadProtectionMutex);
  1439. #endif
  1440. fFreewheel = isFreewheel;
  1441. offlineModeChanged(isFreewheel);
  1442. }
  1443. void handleJackProcessCallback(const uint32_t nframes)
  1444. {
  1445. const PendingRtEventsRunner prt(this, nframes);
  1446. CARLA_SAFE_ASSERT_INT2_RETURN(nframes == pData->bufferSize, nframes, pData->bufferSize,);
  1447. #ifdef BUILD_BRIDGE
  1448. CarlaPlugin* const plugin(pData->plugins[0].plugin);
  1449. if (plugin != nullptr && plugin->isEnabled() && plugin->tryLock(fFreewheel))
  1450. {
  1451. plugin->initBuffers();
  1452. processPlugin(plugin, nframes);
  1453. plugin->unlock();
  1454. }
  1455. #else
  1456. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_JACK && !fTimebaseMaster)
  1457. {
  1458. jack_position_t jpos;
  1459. // invalidate
  1460. jpos.unique_1 = 1;
  1461. jpos.unique_2 = 2;
  1462. EngineTimeInfo timeInfo;
  1463. const bool playing = jackbridge_transport_query(fClient, &jpos) == JackTransportRolling;
  1464. if (jpos.unique_1 != jpos.unique_2)
  1465. {
  1466. timeInfo.playing = false;
  1467. timeInfo.frame = 0;
  1468. timeInfo.usecs = 0;
  1469. timeInfo.bbt.valid = false;
  1470. }
  1471. else
  1472. {
  1473. timeInfo.playing = playing;
  1474. timeInfo.frame = jpos.frame;
  1475. timeInfo.usecs = jpos.usecs;
  1476. if (jpos.valid & JackPositionBBT)
  1477. {
  1478. timeInfo.bbt.valid = true;
  1479. timeInfo.bbt.bar = jpos.bar;
  1480. timeInfo.bbt.beat = jpos.beat;
  1481. timeInfo.bbt.tick = jpos.tick;
  1482. timeInfo.bbt.barStartTick = jpos.bar_start_tick;
  1483. timeInfo.bbt.beatsPerBar = jpos.beats_per_bar;
  1484. timeInfo.bbt.beatType = jpos.beat_type;
  1485. timeInfo.bbt.ticksPerBeat = jpos.ticks_per_beat;
  1486. timeInfo.bbt.beatsPerMinute = jpos.beats_per_minute;
  1487. }
  1488. else
  1489. {
  1490. timeInfo.bbt.valid = false;
  1491. }
  1492. }
  1493. pData->timeInfo = timeInfo;
  1494. }
  1495. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  1496. {
  1497. if (pData->aboutToClose)
  1498. {
  1499. if (float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut1], nframes))
  1500. carla_zeroFloats(audioOut1, nframes);
  1501. if (float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut2], nframes))
  1502. carla_zeroFloats(audioOut2, nframes);
  1503. }
  1504. else if (pData->curPluginCount == 0)
  1505. {
  1506. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn1], nframes);
  1507. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn2], nframes);
  1508. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut1], nframes);
  1509. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut2], nframes);
  1510. // assert buffers
  1511. CARLA_SAFE_ASSERT_RETURN(audioIn1 != nullptr,);
  1512. CARLA_SAFE_ASSERT_RETURN(audioIn2 != nullptr,);
  1513. CARLA_SAFE_ASSERT_RETURN(audioOut1 != nullptr,);
  1514. CARLA_SAFE_ASSERT_RETURN(audioOut2 != nullptr,);
  1515. // pass-through
  1516. carla_copyFloats(audioOut1, audioIn1, nframes);
  1517. carla_copyFloats(audioOut2, audioIn2, nframes);
  1518. // TODO pass-through MIDI as well
  1519. if (void* const eventOut = jackbridge_port_get_buffer(fRackPorts[kRackPortEventOut], nframes))
  1520. jackbridge_midi_clear_buffer(eventOut);
  1521. return;
  1522. }
  1523. }
  1524. if (pData->options.processMode == ENGINE_PROCESS_MODE_SINGLE_CLIENT)
  1525. {
  1526. for (uint i=0; i < pData->curPluginCount; ++i)
  1527. {
  1528. CarlaPlugin* const plugin(pData->plugins[i].plugin);
  1529. if (plugin != nullptr && plugin->isEnabled() && plugin->tryLock(fFreewheel))
  1530. {
  1531. plugin->initBuffers();
  1532. processPlugin(plugin, nframes);
  1533. plugin->unlock();
  1534. }
  1535. }
  1536. }
  1537. else if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK ||
  1538. pData->options.processMode == ENGINE_PROCESS_MODE_PATCHBAY)
  1539. {
  1540. CARLA_SAFE_ASSERT_RETURN(pData->events.in != nullptr,);
  1541. CARLA_SAFE_ASSERT_RETURN(pData->events.out != nullptr,);
  1542. // get buffers from jack
  1543. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn1], nframes);
  1544. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioIn2], nframes);
  1545. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut1], nframes);
  1546. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[kRackPortAudioOut2], nframes);
  1547. void* const eventIn = jackbridge_port_get_buffer(fRackPorts[kRackPortEventIn], nframes);
  1548. void* const eventOut = jackbridge_port_get_buffer(fRackPorts[kRackPortEventOut], nframes);
  1549. // assert buffers
  1550. CARLA_SAFE_ASSERT_RETURN(audioIn1 != nullptr,);
  1551. CARLA_SAFE_ASSERT_RETURN(audioIn2 != nullptr,);
  1552. CARLA_SAFE_ASSERT_RETURN(audioOut1 != nullptr,);
  1553. CARLA_SAFE_ASSERT_RETURN(audioOut2 != nullptr,);
  1554. // create audio buffers
  1555. const float* inBuf[2] = { audioIn1, audioIn2 };
  1556. /**/ float* outBuf[2] = { audioOut1, audioOut2 };
  1557. // initialize events
  1558. carla_zeroStructs(pData->events.in, kMaxEngineEventInternalCount);
  1559. carla_zeroStructs(pData->events.out, kMaxEngineEventInternalCount);
  1560. if (eventIn != nullptr)
  1561. {
  1562. ushort engineEventIndex = 0;
  1563. jack_midi_event_t jackEvent;
  1564. const uint32_t jackEventCount(jackbridge_midi_get_event_count(eventIn));
  1565. for (uint32_t jackEventIndex=0; jackEventIndex < jackEventCount; ++jackEventIndex)
  1566. {
  1567. if (! jackbridge_midi_event_get(&jackEvent, eventIn, jackEventIndex))
  1568. continue;
  1569. CARLA_SAFE_ASSERT_CONTINUE(jackEvent.size < 0xFF /* uint8_t max */);
  1570. EngineEvent& engineEvent(pData->events.in[engineEventIndex++]);
  1571. engineEvent.time = jackEvent.time;
  1572. engineEvent.fillFromMidiData(static_cast<uint8_t>(jackEvent.size), jackEvent.buffer, 0);
  1573. if (engineEventIndex >= kMaxEngineEventInternalCount)
  1574. break;
  1575. }
  1576. }
  1577. if (pData->options.processMode == ENGINE_PROCESS_MODE_CONTINUOUS_RACK)
  1578. {
  1579. pData->graph.processRack(pData, inBuf, outBuf, nframes);
  1580. }
  1581. else
  1582. {
  1583. const CarlaMutexLocker cml(fPatchbayProcThreadProtectionMutex);
  1584. pData->graph.process(pData, inBuf, outBuf, nframes);
  1585. }
  1586. // output control
  1587. if (eventOut != nullptr)
  1588. {
  1589. jackbridge_midi_clear_buffer(eventOut);
  1590. uint8_t size = 0;
  1591. uint8_t mdata[3] = { 0, 0, 0 };
  1592. uint8_t mdataTmp[EngineMidiEvent::kDataSize];
  1593. const uint8_t* mdataPtr;
  1594. for (ushort i=0; i < kMaxEngineEventInternalCount; ++i)
  1595. {
  1596. const EngineEvent& engineEvent(pData->events.out[i]);
  1597. /**/ if (engineEvent.type == kEngineEventTypeNull)
  1598. {
  1599. break;
  1600. }
  1601. else if (engineEvent.type == kEngineEventTypeControl)
  1602. {
  1603. const EngineControlEvent& ctrlEvent(engineEvent.ctrl);
  1604. size = ctrlEvent.convertToMidiData(engineEvent.channel, mdata);
  1605. mdataPtr = mdata;
  1606. }
  1607. else if (engineEvent.type == kEngineEventTypeMidi)
  1608. {
  1609. const EngineMidiEvent& midiEvent(engineEvent.midi);
  1610. size = midiEvent.size;
  1611. CARLA_SAFE_ASSERT_CONTINUE(size > 0);
  1612. if (size > EngineMidiEvent::kDataSize)
  1613. {
  1614. CARLA_SAFE_ASSERT_CONTINUE(midiEvent.dataExt != nullptr);
  1615. mdataPtr = midiEvent.dataExt;
  1616. }
  1617. else
  1618. {
  1619. // set first byte
  1620. mdataTmp[0] = static_cast<uint8_t>(midiEvent.data[0] | (engineEvent.channel & MIDI_CHANNEL_BIT));
  1621. // copy rest
  1622. carla_copy<uint8_t>(mdataTmp+1, midiEvent.data+1, size-1U);
  1623. // done
  1624. mdataPtr = mdataTmp;
  1625. }
  1626. }
  1627. else
  1628. {
  1629. continue;
  1630. }
  1631. if (size > 0)
  1632. jackbridge_midi_event_write(eventOut, engineEvent.time, mdataPtr, size);
  1633. }
  1634. }
  1635. }
  1636. if (pData->options.transportMode == ENGINE_TRANSPORT_MODE_JACK)
  1637. {
  1638. if (fTimebaseMaster)
  1639. {
  1640. const bool playing = jackbridge_transport_query(fClient, nullptr) == JackTransportRolling;
  1641. if (fTimebaseRolling != playing)
  1642. {
  1643. fTimebaseRolling = playing;
  1644. pData->timeInfo.playing = playing;
  1645. }
  1646. // Check if we are no longer timebase master
  1647. if (playing && fTimebaseUsecs != 0 && fTimebaseUsecs == pData->timeInfo.usecs)
  1648. {
  1649. carla_debug("No longer timerbase master");
  1650. fTimebaseMaster = false;
  1651. }
  1652. }
  1653. fTimebaseUsecs = pData->timeInfo.usecs;
  1654. }
  1655. #endif // ! BUILD_BRIDGE
  1656. }
  1657. void handleJackLatencyCallback(const jack_latency_callback_mode_t /*mode*/)
  1658. {
  1659. // TODO
  1660. }
  1661. #ifndef BUILD_BRIDGE
  1662. void handleJackTimebaseCallback(jack_nframes_t nframes, jack_position_t* const pos, const int new_pos)
  1663. {
  1664. if (new_pos)
  1665. pData->time.setNeedsReset();
  1666. pData->timeInfo.playing = fTimebaseRolling;
  1667. pData->timeInfo.frame = pos->frame;
  1668. pData->timeInfo.usecs = pos->usecs;
  1669. pData->time.fillJackTimeInfo(pos, nframes);
  1670. }
  1671. void handleJackClientRegistrationCallback(const char* const name, const bool reg)
  1672. {
  1673. CARLA_SAFE_ASSERT_RETURN(name != nullptr && name[0] != '\0',);
  1674. // ignore this if on internal patchbay mode
  1675. #if defined(HAVE_LIBLO) && !defined(BUILD_BRIDGE)
  1676. if (! (fExternalPatchbayHost || (fExternalPatchbayOsc && pData->osc.isControlRegisteredForTCP()))) return;
  1677. #else
  1678. if (! fExternalPatchbayHost) return;
  1679. #endif
  1680. // do nothing on client registration, wait for first port
  1681. if (reg) return;
  1682. const uint groupId(fUsedGroups.getGroupId(name));
  1683. // clients might have been registered without ports
  1684. if (groupId == 0) return;
  1685. GroupNameToId groupNameToId;
  1686. groupNameToId.setData(groupId, name);
  1687. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1688. ENGINE_CALLBACK_PATCHBAY_CLIENT_REMOVED, groupNameToId.group, 0, 0, 0, 0.0f, nullptr);
  1689. fUsedGroups.list.removeOne(groupNameToId);
  1690. }
  1691. void handleJackPortRegistrationCallback(const jack_port_id_t port, const bool reg)
  1692. {
  1693. // ignore this if on internal patchbay mode
  1694. #if defined(HAVE_LIBLO) && !defined(BUILD_BRIDGE)
  1695. if (! (fExternalPatchbayHost || (fExternalPatchbayOsc && pData->osc.isControlRegisteredForTCP()))) return;
  1696. #else
  1697. if (! fExternalPatchbayHost) return;
  1698. #endif
  1699. const jack_port_t* const jackPort(jackbridge_port_by_id(fClient, port));
  1700. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr,);
  1701. const char* const fullPortName(jackbridge_port_name(jackPort));
  1702. CARLA_SAFE_ASSERT_RETURN(fullPortName != nullptr && fullPortName[0] != '\0',);
  1703. if (reg)
  1704. {
  1705. const char* const shortPortName(jackbridge_port_short_name(jackPort));
  1706. CARLA_SAFE_ASSERT_RETURN(shortPortName != nullptr && shortPortName[0] != '\0',);
  1707. bool found;
  1708. CarlaString groupName(fullPortName);
  1709. groupName.truncate(groupName.rfind(shortPortName, &found)-1);
  1710. CARLA_SAFE_ASSERT_RETURN(found,);
  1711. const int jackPortFlags(jackbridge_port_flags(jackPort));
  1712. uint groupId(fUsedGroups.getGroupId(groupName));
  1713. if (groupId == 0)
  1714. {
  1715. groupId = ++fUsedGroups.lastId;
  1716. GroupNameToId groupNameToId;
  1717. groupNameToId.setData(groupId, groupName);
  1718. int pluginId = -1;
  1719. PatchbayIcon icon = (jackPortFlags & JackPortIsPhysical) ? PATCHBAY_ICON_HARDWARE : PATCHBAY_ICON_APPLICATION;
  1720. findPluginIdAndIcon(groupName, pluginId, icon);
  1721. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1722. ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED,
  1723. groupNameToId.group,
  1724. icon,
  1725. pluginId,
  1726. 0, 0.0f,
  1727. groupNameToId.name);
  1728. fUsedGroups.list.append(groupNameToId);
  1729. }
  1730. addPatchbayJackPort(fExternalPatchbayHost, fExternalPatchbayOsc,
  1731. groupId, jackPort, shortPortName, fullPortName, jackPortFlags);
  1732. }
  1733. else
  1734. {
  1735. const PortNameToId& portNameToId(fUsedPorts.getPortNameToId(fullPortName));
  1736. /* NOTE: Due to JACK2 async behaviour the port we get here might be the same of a previous rename-plugin request.
  1737. See the comment on CarlaEngineJack::renamePlugin() for more information. */
  1738. if (portNameToId.group <= 0 || portNameToId.port <= 0) return;
  1739. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1740. ENGINE_CALLBACK_PATCHBAY_PORT_REMOVED,
  1741. portNameToId.group,
  1742. static_cast<int>(portNameToId.port),
  1743. 0, 0, 0.0f, nullptr);
  1744. fUsedPorts.list.removeOne(portNameToId);
  1745. }
  1746. }
  1747. void handleJackPortConnectCallback(const jack_port_id_t a, const jack_port_id_t b, const bool connect)
  1748. {
  1749. // ignore this if on internal patchbay mode
  1750. #if defined(HAVE_LIBLO) && !defined(BUILD_BRIDGE)
  1751. if (! (fExternalPatchbayHost || (fExternalPatchbayOsc && pData->osc.isControlRegisteredForTCP()))) return;
  1752. #else
  1753. if (! fExternalPatchbayHost) return;
  1754. #endif
  1755. const jack_port_t* const jackPortA(jackbridge_port_by_id(fClient, a));
  1756. CARLA_SAFE_ASSERT_RETURN(jackPortA != nullptr,);
  1757. const jack_port_t* const jackPortB(jackbridge_port_by_id(fClient, b));
  1758. CARLA_SAFE_ASSERT_RETURN(jackPortB != nullptr,);
  1759. const char* const fullPortNameA(jackbridge_port_name(jackPortA));
  1760. CARLA_SAFE_ASSERT_RETURN(fullPortNameA != nullptr && fullPortNameA[0] != '\0',);
  1761. const char* const fullPortNameB(jackbridge_port_name(jackPortB));
  1762. CARLA_SAFE_ASSERT_RETURN(fullPortNameB != nullptr && fullPortNameB[0] != '\0',);
  1763. const PortNameToId& portNameToIdA(fUsedPorts.getPortNameToId(fullPortNameA));
  1764. const PortNameToId& portNameToIdB(fUsedPorts.getPortNameToId(fullPortNameB));
  1765. /* NOTE: Due to JACK2 async behaviour the port we get here might be the same of a previous rename-plugin request.
  1766. See the comment on CarlaEngineJack::renamePlugin() for more information. */
  1767. if (portNameToIdA.group <= 0 || portNameToIdA.port <= 0) return;
  1768. if (portNameToIdB.group <= 0 || portNameToIdB.port <= 0) return;
  1769. if (connect)
  1770. {
  1771. char strBuf[STR_MAX+1];
  1772. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i", portNameToIdA.group, portNameToIdA.port, portNameToIdB.group, portNameToIdB.port);
  1773. strBuf[STR_MAX] = '\0';
  1774. ConnectionToId connectionToId;
  1775. connectionToId.setData(++fUsedConnections.lastId, portNameToIdA.group, portNameToIdA.port, portNameToIdB.group, portNameToIdB.port);
  1776. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1777. ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED,
  1778. connectionToId.id,
  1779. 0, 0, 0, 0.0f,
  1780. strBuf);
  1781. fUsedConnections.list.append(connectionToId);
  1782. }
  1783. else
  1784. {
  1785. ConnectionToId connectionToId = { 0, 0, 0, 0, 0 };
  1786. bool found = false;
  1787. {
  1788. const CarlaMutexLocker cml(fUsedConnections.mutex);
  1789. for (LinkedList<ConnectionToId>::Itenerator it = fUsedConnections.list.begin2(); it.valid(); it.next())
  1790. {
  1791. connectionToId = it.getValue(connectionToId);
  1792. CARLA_SAFE_ASSERT_CONTINUE(connectionToId.id != 0);
  1793. if (connectionToId.groupA == portNameToIdA.group && connectionToId.portA == portNameToIdA.port &&
  1794. connectionToId.groupB == portNameToIdB.group && connectionToId.portB == portNameToIdB.port)
  1795. {
  1796. found = true;
  1797. fUsedConnections.list.remove(it);
  1798. break;
  1799. }
  1800. }
  1801. }
  1802. if (found) {
  1803. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1804. ENGINE_CALLBACK_PATCHBAY_CONNECTION_REMOVED,
  1805. connectionToId.id,
  1806. 0, 0, 0, 0.0f, nullptr);
  1807. }
  1808. }
  1809. }
  1810. void handleJackPortRenameCallback(const jack_port_id_t port, const char* const oldFullName, const char* const newFullName)
  1811. {
  1812. // ignore this if on internal patchbay mode
  1813. #if defined(HAVE_LIBLO) && !defined(BUILD_BRIDGE)
  1814. if (! (fExternalPatchbayHost || (fExternalPatchbayOsc && pData->osc.isControlRegisteredForTCP()))) return;
  1815. #else
  1816. if (! fExternalPatchbayHost) return;
  1817. #endif
  1818. CARLA_SAFE_ASSERT_RETURN(oldFullName != nullptr && oldFullName[0] != '\0',);
  1819. CARLA_SAFE_ASSERT_RETURN(newFullName != nullptr && newFullName[0] != '\0',);
  1820. const jack_port_t* const jackPort(jackbridge_port_by_id(fClient, port));
  1821. CARLA_SAFE_ASSERT_RETURN(jackPort != nullptr,);
  1822. const char* const shortPortName(jackbridge_port_short_name(jackPort));
  1823. CARLA_SAFE_ASSERT_RETURN(shortPortName != nullptr && shortPortName[0] != '\0',);
  1824. bool found;
  1825. CarlaString groupName(newFullName);
  1826. groupName.truncate(groupName.rfind(shortPortName, &found)-1);
  1827. CARLA_SAFE_ASSERT_RETURN(found,);
  1828. const uint groupId(fUsedGroups.getGroupId(groupName));
  1829. CARLA_SAFE_ASSERT_RETURN(groupId > 0,);
  1830. for (LinkedList<PortNameToId>::Itenerator it = fUsedPorts.list.begin2(); it.valid(); it.next())
  1831. {
  1832. static PortNameToId portNameFallback = { 0, 0, { '\0' }, { '\0' } };
  1833. PortNameToId& portNameToId(it.getValue(portNameFallback));
  1834. CARLA_SAFE_ASSERT_CONTINUE(portNameToId.group != 0);
  1835. if (std::strncmp(portNameToId.fullName, oldFullName, STR_MAX) == 0)
  1836. {
  1837. CARLA_SAFE_ASSERT_CONTINUE(portNameToId.group == groupId);
  1838. portNameToId.rename(shortPortName, newFullName);
  1839. callback(fExternalPatchbayHost, fExternalPatchbayOsc,
  1840. ENGINE_CALLBACK_PATCHBAY_PORT_RENAMED,
  1841. portNameToId.group,
  1842. static_cast<int>(portNameToId.port),
  1843. 0, 0, 0.0f,
  1844. portNameToId.name);
  1845. break;
  1846. }
  1847. }
  1848. }
  1849. #endif
  1850. void handleJackShutdownCallback()
  1851. {
  1852. #ifndef BUILD_BRIDGE
  1853. signalThreadShouldExit();
  1854. #endif
  1855. const PendingRtEventsRunner prt(this, pData->bufferSize);
  1856. for (uint i=0; i < pData->curPluginCount; ++i)
  1857. {
  1858. if (CarlaPlugin* const plugin = pData->plugins[i].plugin)
  1859. {
  1860. plugin->tryLock(true);
  1861. if (CarlaEngineJackClient* const client = (CarlaEngineJackClient*)plugin->getEngineClient())
  1862. client->invalidate();
  1863. plugin->unlock();
  1864. }
  1865. }
  1866. fClient = nullptr;
  1867. #ifndef BUILD_BRIDGE
  1868. carla_zeroPointers(fRackPorts, kRackPortCount);
  1869. #endif
  1870. callback(true, true, ENGINE_CALLBACK_QUIT, 0, 0, 0, 0, 0.0f, nullptr);
  1871. }
  1872. // -------------------------------------------------------------------
  1873. void handlePluginJackShutdownCallback(CarlaPlugin* const plugin)
  1874. {
  1875. CarlaEngineJackClient* const engineClient((CarlaEngineJackClient*)plugin->getEngineClient());
  1876. CARLA_SAFE_ASSERT_RETURN(engineClient != nullptr,);
  1877. plugin->tryLock(true);
  1878. engineClient->invalidate();
  1879. plugin->unlock();
  1880. //if (pData->nextAction.pluginId == plugin->getId())
  1881. // pData->nextAction.clearAndReset();
  1882. callback(true, true, ENGINE_CALLBACK_PLUGIN_UNAVAILABLE, plugin->getId(), 0, 0, 0, 0.0f, "Killed by JACK");
  1883. }
  1884. // -------------------------------------------------------------------
  1885. private:
  1886. jack_client_t* fClient;
  1887. bool fExternalPatchbayHost;
  1888. bool fExternalPatchbayOsc;
  1889. bool fFreewheel;
  1890. // -------------------------------------------------------------------
  1891. #ifdef BUILD_BRIDGE
  1892. bool fIsRunning;
  1893. #else
  1894. enum RackPorts {
  1895. kRackPortAudioIn1 = 0,
  1896. kRackPortAudioIn2 = 1,
  1897. kRackPortAudioOut1 = 2,
  1898. kRackPortAudioOut2 = 3,
  1899. kRackPortEventIn = 4,
  1900. kRackPortEventOut = 5,
  1901. kRackPortCount = 6
  1902. };
  1903. jack_port_t* fRackPorts[kRackPortCount];
  1904. bool fTimebaseMaster;
  1905. bool fTimebaseRolling;
  1906. uint64_t fTimebaseUsecs;
  1907. PatchbayGroupList fUsedGroups;
  1908. PatchbayPortList fUsedPorts;
  1909. PatchbayConnectionList fUsedConnections;
  1910. CarlaMutex fPatchbayProcThreadProtectionMutex;
  1911. mutable CharStringListPtr fRetConns;
  1912. void findPluginIdAndIcon(const char* const clientName, int& pluginId, PatchbayIcon& icon) const noexcept
  1913. {
  1914. carla_debug("CarlaEngineJack::findPluginIdAndIcon(\"%s\", ...)", clientName);
  1915. // TODO - this currently only works in multi-client mode
  1916. if (pData->options.processMode != ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS)
  1917. return;
  1918. const char* const uuidstr = jackbridge_get_uuid_for_client_name(fClient, clientName);
  1919. CARLA_SAFE_ASSERT_RETURN(uuidstr != nullptr && uuidstr[0] != '\0',);
  1920. jack_uuid_t uuid;
  1921. CARLA_SAFE_ASSERT_RETURN(buggy_jack2_uuid_parse(uuidstr, &uuid) == 0,);
  1922. {
  1923. char* value = nullptr;
  1924. char* type = nullptr;
  1925. if (! jackbridge_get_property(uuid, "https://kx.studio/ns/carla/plugin-id", &value, &type))
  1926. return;
  1927. CARLA_SAFE_ASSERT_RETURN(type != nullptr && type[0] != '\0',);
  1928. CARLA_SAFE_ASSERT_RETURN(value != nullptr && value[0] != '\0',);
  1929. CARLA_SAFE_ASSERT_RETURN(std::strcmp(type, "http://www.w3.org/2001/XMLSchema#integer") == 0,);
  1930. pluginId = std::atoi(value);
  1931. icon = PATCHBAY_ICON_PLUGIN;
  1932. }
  1933. {
  1934. char* value = nullptr;
  1935. char* type = nullptr;
  1936. if (! jackbridge_get_property(uuid, "https://kx.studio/ns/carla/plugin-icon", &value, &type))
  1937. return;
  1938. CARLA_SAFE_ASSERT_RETURN(type != nullptr && type[0] != '\0',);
  1939. CARLA_SAFE_ASSERT_RETURN(value != nullptr && value[0] != '\0',);
  1940. CARLA_SAFE_ASSERT_RETURN(std::strcmp(type, "text/plain") == 0,);
  1941. /**/ if (std::strcmp(value, "app") == 0)
  1942. icon = PATCHBAY_ICON_APPLICATION;
  1943. else if (std::strcmp(value, "application") == 0)
  1944. icon = PATCHBAY_ICON_APPLICATION;
  1945. else if (std::strcmp(value, "plugin") == 0)
  1946. icon = PATCHBAY_ICON_PLUGIN;
  1947. else if (std::strcmp(value, "hardware") == 0)
  1948. icon = PATCHBAY_ICON_HARDWARE;
  1949. else if (std::strcmp(value, "carla") == 0)
  1950. icon = PATCHBAY_ICON_CARLA;
  1951. else if (std::strcmp(value, "distrho") == 0)
  1952. icon = PATCHBAY_ICON_DISTRHO;
  1953. else if (std::strcmp(value, "file") == 0)
  1954. icon = PATCHBAY_ICON_FILE;
  1955. }
  1956. }
  1957. void initJackPatchbay(const bool sendHost, const bool sendOSC, const char* const ourName)
  1958. {
  1959. CARLA_SAFE_ASSERT_RETURN(pData->options.processMode != ENGINE_PROCESS_MODE_PATCHBAY ||
  1960. (fExternalPatchbayHost && sendHost) || (fExternalPatchbayOsc && sendOSC),);
  1961. CARLA_SAFE_ASSERT_RETURN(ourName != nullptr && ourName[0] != '\0',);
  1962. CarlaStringList parsedGroups;
  1963. // add our client first
  1964. {
  1965. parsedGroups.append(ourName);
  1966. GroupNameToId groupNameToId;
  1967. groupNameToId.setData(++fUsedGroups.lastId, ourName);
  1968. callback(sendHost, sendOSC,
  1969. ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED,
  1970. groupNameToId.group,
  1971. PATCHBAY_ICON_CARLA,
  1972. MAIN_CARLA_PLUGIN_ID,
  1973. 0, 0.0f,
  1974. groupNameToId.name);
  1975. fUsedGroups.list.append(groupNameToId);
  1976. }
  1977. // query all jack ports
  1978. {
  1979. const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, 0);
  1980. CARLA_SAFE_ASSERT_RETURN(ports != nullptr,);
  1981. for (int i=0; ports[i] != nullptr; ++i)
  1982. {
  1983. const char* const fullPortName(ports[i]);
  1984. CARLA_SAFE_ASSERT_CONTINUE(fullPortName != nullptr && fullPortName[0] != '\0');
  1985. const jack_port_t* const jackPort(jackbridge_port_by_name(fClient, fullPortName));
  1986. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  1987. const char* const shortPortName(jackbridge_port_short_name(jackPort));
  1988. CARLA_SAFE_ASSERT_CONTINUE(shortPortName != nullptr && shortPortName[0] != '\0');
  1989. const int jackPortFlags(jackbridge_port_flags(jackPort));
  1990. uint groupId = 0;
  1991. bool found;
  1992. CarlaString groupName(fullPortName);
  1993. groupName.truncate(groupName.rfind(shortPortName, &found)-1);
  1994. CARLA_SAFE_ASSERT_CONTINUE(found);
  1995. if (parsedGroups.contains(groupName))
  1996. {
  1997. groupId = fUsedGroups.getGroupId(groupName);
  1998. CARLA_SAFE_ASSERT_CONTINUE(groupId > 0);
  1999. }
  2000. else
  2001. {
  2002. groupId = ++fUsedGroups.lastId;
  2003. parsedGroups.append(groupName);
  2004. GroupNameToId groupNameToId;
  2005. groupNameToId.setData(groupId, groupName);
  2006. int pluginId = -1;
  2007. PatchbayIcon icon = (jackPortFlags & JackPortIsPhysical) ? PATCHBAY_ICON_HARDWARE : PATCHBAY_ICON_APPLICATION;
  2008. findPluginIdAndIcon(groupName, pluginId, icon);
  2009. callback(sendHost, sendOSC,
  2010. ENGINE_CALLBACK_PATCHBAY_CLIENT_ADDED,
  2011. groupNameToId.group,
  2012. icon,
  2013. pluginId,
  2014. 0, 0.0f,
  2015. groupNameToId.name);
  2016. fUsedGroups.list.append(groupNameToId);
  2017. }
  2018. addPatchbayJackPort(sendHost, sendOSC, groupId, jackPort, shortPortName, fullPortName, jackPortFlags);
  2019. }
  2020. jackbridge_free(ports);
  2021. }
  2022. // query connections, after all ports are in place
  2023. if (const char** const ports = jackbridge_get_ports(fClient, nullptr, nullptr, JackPortIsOutput))
  2024. {
  2025. char strBuf[STR_MAX+1];
  2026. for (int i=0; ports[i] != nullptr; ++i)
  2027. {
  2028. const char* const fullPortName(ports[i]);
  2029. CARLA_SAFE_ASSERT_CONTINUE(fullPortName != nullptr && fullPortName[0] != '\0');
  2030. const jack_port_t* const jackPort(jackbridge_port_by_name(fClient, fullPortName));
  2031. CARLA_SAFE_ASSERT_CONTINUE(jackPort != nullptr);
  2032. const PortNameToId& thisPort(fUsedPorts.getPortNameToId(fullPortName));
  2033. CARLA_SAFE_ASSERT_CONTINUE(thisPort.group > 0);
  2034. CARLA_SAFE_ASSERT_CONTINUE(thisPort.port > 0);
  2035. if (const char** const connections = jackbridge_port_get_all_connections(fClient, jackPort))
  2036. {
  2037. for (int j=0; connections[j] != nullptr; ++j)
  2038. {
  2039. const char* const connection(connections[j]);
  2040. CARLA_SAFE_ASSERT_CONTINUE(connection != nullptr && connection[0] != '\0');
  2041. const PortNameToId& targetPort(fUsedPorts.getPortNameToId(connection));
  2042. CARLA_SAFE_ASSERT_CONTINUE(targetPort.group > 0);
  2043. CARLA_SAFE_ASSERT_CONTINUE(targetPort.port > 0);
  2044. std::snprintf(strBuf, STR_MAX, "%i:%i:%i:%i", thisPort.group, thisPort.port, targetPort.group, targetPort.port);
  2045. strBuf[STR_MAX] = '\0';
  2046. ConnectionToId connectionToId;
  2047. connectionToId.setData(++fUsedConnections.lastId, thisPort.group, thisPort.port, targetPort.group, targetPort.port);
  2048. callback(sendHost, sendOSC,
  2049. ENGINE_CALLBACK_PATCHBAY_CONNECTION_ADDED,
  2050. connectionToId.id,
  2051. 0, 0, 0, 0.0f,
  2052. strBuf);
  2053. fUsedConnections.list.append(connectionToId);
  2054. }
  2055. jackbridge_free(connections);
  2056. }
  2057. }
  2058. jackbridge_free(ports);
  2059. }
  2060. }
  2061. void addPatchbayJackPort(const bool sendHost, const bool sendOSC,
  2062. const uint groupId, const jack_port_t* const jackPort,
  2063. const char* const shortPortName, const char* const fullPortName, const int jackPortFlags)
  2064. {
  2065. bool portIsInput = (jackPortFlags & JackPortIsInput);
  2066. bool portIsAudio = (std::strcmp(jackbridge_port_type(jackPort), JACK_DEFAULT_AUDIO_TYPE) == 0);
  2067. bool portIsMIDI = (std::strcmp(jackbridge_port_type(jackPort), JACK_DEFAULT_MIDI_TYPE) == 0);
  2068. bool portIsCV = false;
  2069. bool portIsOSC = false;
  2070. if (const jack_uuid_t uuid = jackbridge_port_uuid(jackPort))
  2071. {
  2072. char* value = nullptr;
  2073. char* type = nullptr;
  2074. if (jackbridge_get_property(uuid, JACKEY_SIGNAL_TYPE, &value, &type)
  2075. && value != nullptr
  2076. && type != nullptr
  2077. && std::strcmp(type, "text/plain") == 0)
  2078. {
  2079. portIsCV = (std::strcmp(value, "CV") == 0);
  2080. portIsOSC = (std::strcmp(value, "OSC") == 0);
  2081. }
  2082. }
  2083. uint canvasPortFlags = 0x0;
  2084. canvasPortFlags |= portIsInput ? PATCHBAY_PORT_IS_INPUT : 0x0;
  2085. /**/ if (portIsCV)
  2086. canvasPortFlags |= PATCHBAY_PORT_TYPE_CV;
  2087. else if (portIsOSC)
  2088. canvasPortFlags |= PATCHBAY_PORT_TYPE_OSC;
  2089. else if (portIsAudio)
  2090. canvasPortFlags |= PATCHBAY_PORT_TYPE_AUDIO;
  2091. else if (portIsMIDI)
  2092. canvasPortFlags |= PATCHBAY_PORT_TYPE_MIDI;
  2093. PortNameToId portNameToId;
  2094. portNameToId.setData(groupId, ++fUsedPorts.lastId, shortPortName, fullPortName);
  2095. callback(sendHost, sendOSC,
  2096. ENGINE_CALLBACK_PATCHBAY_PORT_ADDED,
  2097. portNameToId.group,
  2098. static_cast<int>(portNameToId.port),
  2099. static_cast<int>(canvasPortFlags),
  2100. 0, 0.0f,
  2101. portNameToId.name);
  2102. fUsedPorts.list.append(portNameToId);
  2103. }
  2104. #endif
  2105. // -------------------------------------------------------------------
  2106. void processPlugin(CarlaPlugin* const plugin, const uint32_t nframes)
  2107. {
  2108. const uint32_t audioInCount(plugin->getAudioInCount());
  2109. const uint32_t audioOutCount(plugin->getAudioOutCount());
  2110. const uint32_t cvInCount(plugin->getCVInCount());
  2111. const uint32_t cvOutCount(plugin->getCVOutCount());
  2112. const float* audioIn[audioInCount];
  2113. /* */ float* audioOut[audioOutCount];
  2114. const float* cvIn[cvInCount];
  2115. /* */ float* cvOut[cvOutCount];
  2116. for (uint32_t i=0; i < audioInCount; ++i)
  2117. {
  2118. CarlaEngineAudioPort* const port(plugin->getAudioInPort(i));
  2119. audioIn[i] = port->getBuffer();
  2120. }
  2121. for (uint32_t i=0; i < audioOutCount; ++i)
  2122. {
  2123. CarlaEngineAudioPort* const port(plugin->getAudioOutPort(i));
  2124. audioOut[i] = port->getBuffer();
  2125. }
  2126. for (uint32_t i=0; i < cvInCount; ++i)
  2127. {
  2128. CarlaEngineCVPort* const port(plugin->getCVInPort(i));
  2129. cvIn[i] = port->getBuffer();
  2130. }
  2131. for (uint32_t i=0; i < cvOutCount; ++i)
  2132. {
  2133. CarlaEngineCVPort* const port(plugin->getCVOutPort(i));
  2134. cvOut[i] = port->getBuffer();
  2135. }
  2136. float inPeaks[2] = { 0.0f };
  2137. float outPeaks[2] = { 0.0f };
  2138. for (uint32_t i=0; i < audioInCount && i < 2; ++i)
  2139. {
  2140. for (uint32_t j=0; j < nframes; ++j)
  2141. {
  2142. const float absV(std::abs(audioIn[i][j]));
  2143. if (absV > inPeaks[i])
  2144. inPeaks[i] = absV;
  2145. }
  2146. }
  2147. plugin->process(audioIn, audioOut, cvIn, cvOut, nframes);
  2148. for (uint32_t i=0; i < audioOutCount && i < 2; ++i)
  2149. {
  2150. for (uint32_t j=0; j < nframes; ++j)
  2151. {
  2152. const float absV(std::abs(audioOut[i][j]));
  2153. if (absV > outPeaks[i])
  2154. outPeaks[i] = absV;
  2155. }
  2156. }
  2157. setPluginPeaksRT(plugin->getId(), inPeaks, outPeaks);
  2158. }
  2159. #ifndef BUILD_BRIDGE
  2160. // -------------------------------------------------------------------
  2161. struct PostPonedJackEvent {
  2162. enum Type {
  2163. kTypeNull = 0,
  2164. kTypeClientRegister,
  2165. kTypePortRegister,
  2166. kTypePortConnect,
  2167. kTypePortRename
  2168. };
  2169. Type type;
  2170. bool action; // register or connect
  2171. jack_port_id_t port1;
  2172. jack_port_id_t port2;
  2173. char name1[STR_MAX+1];
  2174. char name2[STR_MAX+1];
  2175. };
  2176. LinkedList<PostPonedJackEvent> fPostPonedEvents;
  2177. CarlaMutex fPostPonedEventsMutex;
  2178. bool fIsInternalClient;
  2179. void postPoneJackCallback(const PostPonedJackEvent& ev)
  2180. {
  2181. const CarlaMutexLocker cml(fPostPonedEventsMutex);
  2182. fPostPonedEvents.append(ev);
  2183. }
  2184. void run() override
  2185. {
  2186. LinkedList<PostPonedJackEvent> events;
  2187. PostPonedJackEvent nullEvent;
  2188. carla_zeroStruct(nullEvent);
  2189. for (; ! shouldThreadExit();)
  2190. {
  2191. if (fIsInternalClient)
  2192. idle();
  2193. {
  2194. const CarlaMutexLocker cml(fPostPonedEventsMutex);
  2195. if (fPostPonedEvents.count() > 0)
  2196. fPostPonedEvents.moveTo(events);
  2197. }
  2198. if (fClient == nullptr)
  2199. break;
  2200. if (events.count() == 0)
  2201. {
  2202. carla_msleep(fIsInternalClient ? 50 : 200);
  2203. continue;
  2204. }
  2205. for (LinkedList<PostPonedJackEvent>::Itenerator it = events.begin2(); it.valid(); it.next())
  2206. {
  2207. const PostPonedJackEvent& ev(it.getValue(nullEvent));
  2208. CARLA_SAFE_ASSERT_CONTINUE(ev.type != PostPonedJackEvent::kTypeNull);
  2209. switch (ev.type)
  2210. {
  2211. case PostPonedJackEvent::kTypeNull:
  2212. break;
  2213. case PostPonedJackEvent::kTypeClientRegister:
  2214. handleJackClientRegistrationCallback(ev.name1, ev.action);
  2215. break;
  2216. case PostPonedJackEvent::kTypePortRegister:
  2217. handleJackPortRegistrationCallback(ev.port1, ev.action);
  2218. break;
  2219. case PostPonedJackEvent::kTypePortConnect:
  2220. handleJackPortConnectCallback(ev.port1, ev.port2, ev.action);
  2221. break;
  2222. case PostPonedJackEvent::kTypePortRename:
  2223. handleJackPortRenameCallback(ev.port1, ev.name1, ev.name2);
  2224. break;
  2225. }
  2226. }
  2227. events.clear();
  2228. }
  2229. events.clear();
  2230. }
  2231. #endif // BUILD_BRIDGE
  2232. // -------------------------------------------------------------------
  2233. // disable -Wattributes warnings
  2234. #if defined(__clang__)
  2235. # pragma clang diagnostic push
  2236. # pragma clang diagnostic ignored "-Wattributes"
  2237. #elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))
  2238. # pragma GCC diagnostic push
  2239. # pragma GCC diagnostic ignored "-Wattributes"
  2240. #endif
  2241. #define handlePtr ((CarlaEngineJack*)arg)
  2242. static void JACKBRIDGE_API carla_jack_thread_init_callback(void*)
  2243. {
  2244. #ifdef __SSE2_MATH__
  2245. // Set FTZ and DAZ flags
  2246. _mm_setcsr(_mm_getcsr() | 0x8040);
  2247. #endif
  2248. }
  2249. static int JACKBRIDGE_API carla_jack_bufsize_callback(jack_nframes_t newBufferSize, void* arg)
  2250. {
  2251. handlePtr->handleJackBufferSizeCallback(newBufferSize);
  2252. return 0;
  2253. }
  2254. static int JACKBRIDGE_API carla_jack_srate_callback(jack_nframes_t newSampleRate, void* arg)
  2255. {
  2256. handlePtr->handleJackSampleRateCallback(newSampleRate);
  2257. return 0;
  2258. }
  2259. static void JACKBRIDGE_API carla_jack_freewheel_callback(int starting, void* arg)
  2260. {
  2261. handlePtr->handleJackFreewheelCallback(bool(starting));
  2262. }
  2263. static void JACKBRIDGE_API carla_jack_latency_callback(jack_latency_callback_mode_t mode, void* arg)
  2264. {
  2265. handlePtr->handleJackLatencyCallback(mode);
  2266. }
  2267. static int JACKBRIDGE_API carla_jack_process_callback(jack_nframes_t nframes, void* arg) __attribute__((annotate("realtime")))
  2268. {
  2269. handlePtr->handleJackProcessCallback(nframes);
  2270. return 0;
  2271. }
  2272. #ifndef BUILD_BRIDGE
  2273. static void JACKBRIDGE_API carla_jack_timebase_callback(jack_transport_state_t, jack_nframes_t nframes, jack_position_t* pos, int new_pos, void* arg) __attribute__((annotate("realtime")))
  2274. {
  2275. handlePtr->handleJackTimebaseCallback(nframes, pos, new_pos);
  2276. }
  2277. static void JACKBRIDGE_API carla_jack_client_registration_callback(const char* name, int reg, void* arg)
  2278. {
  2279. PostPonedJackEvent ev;
  2280. carla_zeroStruct(ev);
  2281. ev.type = PostPonedJackEvent::kTypeClientRegister;
  2282. ev.action = (reg != 0);
  2283. std::strncpy(ev.name1, name, STR_MAX);
  2284. handlePtr->postPoneJackCallback(ev);
  2285. }
  2286. static void JACKBRIDGE_API carla_jack_port_registration_callback(jack_port_id_t port, int reg, void* arg)
  2287. {
  2288. PostPonedJackEvent ev;
  2289. carla_zeroStruct(ev);
  2290. ev.type = PostPonedJackEvent::kTypePortRegister;
  2291. ev.action = (reg != 0);
  2292. ev.port1 = port;
  2293. handlePtr->postPoneJackCallback(ev);
  2294. }
  2295. static void JACKBRIDGE_API carla_jack_port_connect_callback(jack_port_id_t a, jack_port_id_t b, int connect, void* arg)
  2296. {
  2297. PostPonedJackEvent ev;
  2298. carla_zeroStruct(ev);
  2299. ev.type = PostPonedJackEvent::kTypePortConnect;
  2300. ev.action = (connect != 0);
  2301. ev.port1 = a;
  2302. ev.port2 = b;
  2303. handlePtr->postPoneJackCallback(ev);
  2304. }
  2305. static void JACKBRIDGE_API carla_jack_port_rename_callback(jack_port_id_t port, const char* oldName, const char* newName, void* arg)
  2306. {
  2307. PostPonedJackEvent ev;
  2308. carla_zeroStruct(ev);
  2309. ev.type = PostPonedJackEvent::kTypePortRename;
  2310. ev.port1 = port;
  2311. std::strncpy(ev.name1, oldName, STR_MAX);
  2312. std::strncpy(ev.name2, newName, STR_MAX);
  2313. handlePtr->postPoneJackCallback(ev);
  2314. }
  2315. static int JACKBRIDGE_API carla_jack_xrun_callback(void* arg)
  2316. {
  2317. ++(handlePtr->pData->xruns);
  2318. return 0;
  2319. }
  2320. #endif
  2321. static void JACKBRIDGE_API carla_jack_shutdown_callback(void* arg)
  2322. {
  2323. handlePtr->handleJackShutdownCallback();
  2324. }
  2325. #undef handlePtr
  2326. // -------------------------------------------------------------------
  2327. #ifndef BUILD_BRIDGE
  2328. static int JACKBRIDGE_API carla_jack_process_callback_plugin(jack_nframes_t nframes, void* arg) __attribute__((annotate("realtime")))
  2329. {
  2330. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  2331. CARLA_SAFE_ASSERT_RETURN(plugin != nullptr && plugin->isEnabled(), 0);
  2332. CarlaEngineJack* const engine((CarlaEngineJack*)plugin->getEngine());
  2333. CARLA_SAFE_ASSERT_RETURN(engine != nullptr, 0);
  2334. if (plugin->tryLock(engine->fFreewheel))
  2335. {
  2336. plugin->initBuffers();
  2337. engine->processPlugin(plugin, nframes);
  2338. plugin->unlock();
  2339. }
  2340. return 0;
  2341. }
  2342. /*
  2343. static int JACKBRIDGE_API carla_jack_bufsize_callback_plugin(jack_nframes_t nframes, void* arg)
  2344. {
  2345. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  2346. CARLA_SAFE_ASSERT_RETURN(plugin != nullptr && plugin->isEnabled(), 0);
  2347. plugin->bufferSizeChanged(nframes);
  2348. return 1;
  2349. }
  2350. static int JACKBRIDGE_API carla_jack_srate_callback_plugin(jack_nframes_t nframes, void* arg)
  2351. {
  2352. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  2353. CARLA_SAFE_ASSERT_RETURN(plugin != nullptr && plugin->isEnabled(), 0);
  2354. plugin->sampleRateChanged(nframes);
  2355. return 1;
  2356. }
  2357. */
  2358. static void JACKBRIDGE_API carla_jack_latency_callback_plugin(jack_latency_callback_mode_t /*mode*/, void* /*arg*/)
  2359. {
  2360. // TODO
  2361. }
  2362. static void JACKBRIDGE_API carla_jack_shutdown_callback_plugin(void* arg)
  2363. {
  2364. CarlaPlugin* const plugin((CarlaPlugin*)arg);
  2365. CARLA_SAFE_ASSERT_RETURN(plugin != nullptr,);
  2366. CarlaEngineJack* const engine((CarlaEngineJack*)plugin->getEngine());
  2367. CARLA_SAFE_ASSERT_RETURN(engine != nullptr,);
  2368. engine->handlePluginJackShutdownCallback(plugin);
  2369. }
  2370. #endif
  2371. // enable -Wattributes again
  2372. #if defined(__clang__)
  2373. # pragma clang diagnostic pop
  2374. #elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))
  2375. # pragma GCC diagnostic pop
  2376. #endif
  2377. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJack)
  2378. };
  2379. // -----------------------------------------------------------------------
  2380. CarlaEngine* CarlaEngine::newJack()
  2381. {
  2382. carla_debug("CarlaEngine::newJack()");
  2383. return new CarlaEngineJack();
  2384. }
  2385. // -----------------------------------------------------------------------
  2386. CARLA_BACKEND_END_NAMESPACE
  2387. #ifndef BUILD_BRIDGE
  2388. // -----------------------------------------------------------------------
  2389. // internal jack client
  2390. CARLA_EXPORT
  2391. int jack_initialize (jack_client_t *client, const char *load_init);
  2392. CARLA_EXPORT
  2393. void jack_finish(void *arg);
  2394. #ifdef CARLA_OS_UNIX
  2395. # include "ThreadSafeFFTW.hpp"
  2396. static ThreadSafeFFTW sThreadSafeFFTW;
  2397. #endif
  2398. // -----------------------------------------------------------------------
  2399. CARLA_EXPORT
  2400. int jack_initialize(jack_client_t* const client, const char* const load_init)
  2401. {
  2402. CARLA_BACKEND_USE_NAMESPACE
  2403. EngineProcessMode mode;
  2404. if (load_init != nullptr && std::strcmp(load_init, "rack") == 0)
  2405. mode = ENGINE_PROCESS_MODE_CONTINUOUS_RACK;
  2406. else
  2407. mode = ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS;
  2408. #ifdef USING_JUCE
  2409. juce::initialiseJuce_GUI();
  2410. #endif
  2411. CarlaEngineJack* const engine = new CarlaEngineJack();
  2412. engine->setOption(ENGINE_OPTION_FORCE_STEREO, 1, nullptr);
  2413. engine->setOption(ENGINE_OPTION_AUDIO_DEVICE, 0, "Auto-Connect ON");
  2414. engine->setOption(ENGINE_OPTION_OSC_ENABLED, 1, nullptr);
  2415. engine->setOption(ENGINE_OPTION_OSC_PORT_TCP, 22752, nullptr);
  2416. engine->setOption(ENGINE_OPTION_OSC_PORT_UDP, 22752, nullptr);
  2417. engine->setOption(ENGINE_OPTION_PROCESS_MODE, mode, nullptr);
  2418. engine->setOption(ENGINE_OPTION_TRANSPORT_MODE, ENGINE_TRANSPORT_MODE_JACK, nullptr);
  2419. // FIXME
  2420. engine->setOption(ENGINE_OPTION_PATH_BINARIES, 0, "");
  2421. engine->setOption(ENGINE_OPTION_PATH_RESOURCES, 0, "");
  2422. if (engine->initInternal(client))
  2423. {
  2424. #ifdef CARLA_OS_UNIX
  2425. sThreadSafeFFTW.init();
  2426. #endif
  2427. return 0;
  2428. }
  2429. else
  2430. {
  2431. delete engine;
  2432. #ifdef USING_JUCE
  2433. juce::shutdownJuce_GUI();
  2434. #endif
  2435. return 1;
  2436. }
  2437. }
  2438. CARLA_EXPORT
  2439. void jack_finish(void *arg)
  2440. {
  2441. CARLA_BACKEND_USE_NAMESPACE
  2442. CarlaEngineJack* const engine = (CarlaEngineJack*)arg;;
  2443. CARLA_SAFE_ASSERT_RETURN(engine != nullptr,);
  2444. #ifdef CARLA_OS_UNIX
  2445. const ThreadSafeFFTW::Deinitializer tsfftwde(sThreadSafeFFTW);
  2446. #endif
  2447. engine->setAboutToClose();
  2448. engine->removeAllPlugins();
  2449. engine->close();
  2450. delete engine;
  2451. #ifdef USING_JUCE
  2452. juce::shutdownJuce_GUI();
  2453. #endif
  2454. }
  2455. // -----------------------------------------------------------------------
  2456. #endif