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.

CarlaEngineJack.cpp 76KB

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