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.

3644 lines
127KB

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