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.

1485 lines
49KB

  1. /*
  2. * Carla JACK Engine
  3. * Copyright (C) 2012-2013 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 GPL.txt file
  16. */
  17. #if 1//def WANT_JACK
  18. #include "CarlaEngineInternal.hpp"
  19. #include "CarlaBackendUtils.hpp"
  20. #include "CarlaMIDI.h"
  21. #include "jackbridge/jackbridge.h"
  22. #include <cmath>
  23. CARLA_BACKEND_START_NAMESPACE
  24. #if 0
  25. } // Fix editor indentation
  26. #endif
  27. // -------------------------------------------------------------------
  28. // Helpers, defined in CarlaPlugin.cpp
  29. extern CarlaEngine* CarlaPluginGetEngine(CarlaPlugin* const plugin);
  30. extern CarlaEngineAudioPort* CarlaPluginGetAudioInPort(CarlaPlugin* const plugin, uint32_t index);
  31. extern CarlaEngineAudioPort* CarlaPluginGetAudioOutPort(CarlaPlugin* const plugin, uint32_t index);
  32. // -------------------------------------------------------------------------------------------------------------------
  33. // Carla Engine JACK-Audio port
  34. class CarlaEngineJackAudioPort : public CarlaEngineAudioPort
  35. {
  36. public:
  37. CarlaEngineJackAudioPort(const bool isInput, const ProcessMode processMode, jack_client_t* const client, jack_port_t* const port)
  38. : CarlaEngineAudioPort(isInput, processMode),
  39. kClient(client),
  40. kPort(port)
  41. {
  42. carla_debug("CarlaEngineJackAudioPort::CarlaEngineJackAudioPort(%s, %s, %p, %p)", bool2str(isInput), ProcessMode2Str(processMode), client, port);
  43. if (processMode == PROCESS_MODE_SINGLE_CLIENT || processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  44. {
  45. CARLA_ASSERT(client != nullptr && port != nullptr);
  46. }
  47. else
  48. {
  49. CARLA_ASSERT(client == nullptr && port == nullptr);
  50. }
  51. }
  52. ~CarlaEngineJackAudioPort()
  53. {
  54. carla_debug("CarlaEngineJackAudioPort::~CarlaEngineJackAudioPort()");
  55. if (kClient != nullptr && kPort != nullptr)
  56. jackbridge_port_unregister(kClient, kPort);
  57. }
  58. void initBuffer(CarlaEngine* const engine)
  59. {
  60. CARLA_ASSERT(engine != nullptr);
  61. if (engine == nullptr)
  62. {
  63. fBuffer = nullptr;
  64. return;
  65. }
  66. if (kPort == nullptr)
  67. return CarlaEngineAudioPort::initBuffer(engine);
  68. fBuffer = (float*)jackbridge_port_get_buffer(kPort, engine->getBufferSize());
  69. if (! kIsInput)
  70. carla_zeroFloat(fBuffer, engine->getBufferSize());
  71. }
  72. private:
  73. jack_client_t* const kClient;
  74. jack_port_t* const kPort;
  75. friend class CarlaEngineJack;
  76. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackAudioPort)
  77. };
  78. // -------------------------------------------------------------------------------------------------------------------
  79. // Carla Engine JACK-Event port
  80. static const EngineEvent kFallbackJackEngineEvent;
  81. class CarlaEngineJackEventPort : public CarlaEngineEventPort
  82. {
  83. public:
  84. CarlaEngineJackEventPort(const bool isInput, const ProcessMode processMode, jack_client_t* const client, jack_port_t* const port)
  85. : CarlaEngineEventPort(isInput, processMode),
  86. kClient(client),
  87. kPort(port),
  88. fJackBuffer(nullptr)
  89. {
  90. carla_debug("CarlaEngineJackEventPort::CarlaEngineJackEventPort(%s, %s, %p, %p)", bool2str(isInput), ProcessMode2Str(processMode), client, port);
  91. if (processMode == PROCESS_MODE_SINGLE_CLIENT || processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  92. {
  93. CARLA_ASSERT(client != nullptr && port != nullptr);
  94. }
  95. else
  96. {
  97. CARLA_ASSERT(client == nullptr && port == nullptr);
  98. }
  99. }
  100. ~CarlaEngineJackEventPort()
  101. {
  102. carla_debug("CarlaEngineJackEventPort::~CarlaEngineJackEventPort()");
  103. if (kClient != nullptr && kPort != nullptr)
  104. jackbridge_port_unregister(kClient, kPort);
  105. }
  106. void initBuffer(CarlaEngine* const engine)
  107. {
  108. CARLA_ASSERT(engine != nullptr);
  109. if (engine == nullptr)
  110. {
  111. fJackBuffer = nullptr;
  112. return;
  113. }
  114. if (kPort == nullptr)
  115. return CarlaEngineEventPort::initBuffer(engine);
  116. fJackBuffer = jackbridge_port_get_buffer(kPort, engine->getBufferSize());
  117. if (! kIsInput)
  118. jackbridge_midi_clear_buffer(fJackBuffer);
  119. }
  120. uint32_t getEventCount()
  121. {
  122. if (kPort == nullptr)
  123. return CarlaEngineEventPort::getEventCount();
  124. CARLA_ASSERT(kIsInput);
  125. CARLA_ASSERT(fJackBuffer != nullptr);
  126. if (! kIsInput)
  127. return 0;
  128. if (fJackBuffer == nullptr)
  129. return 0;
  130. return jackbridge_midi_get_event_count(fJackBuffer);
  131. }
  132. const EngineEvent& getEvent(const uint32_t index)
  133. {
  134. if (kPort == nullptr)
  135. return CarlaEngineEventPort::getEvent(index);
  136. CARLA_ASSERT(kIsInput);
  137. CARLA_ASSERT(fJackBuffer != nullptr);
  138. if (! kIsInput)
  139. return kFallbackJackEngineEvent;
  140. if (fJackBuffer == nullptr)
  141. return kFallbackJackEngineEvent;
  142. jack_midi_event_t jackEvent;
  143. if (jackbridge_midi_event_get(&jackEvent, fJackBuffer, index) != 0 || jackEvent.size > 3)
  144. return kFallbackJackEngineEvent;
  145. fRetEvent.clear();
  146. const uint8_t midiStatus = MIDI_GET_STATUS_FROM_DATA(jackEvent.buffer);
  147. const uint8_t midiChannel = MIDI_GET_CHANNEL_FROM_DATA(jackEvent.buffer);
  148. fRetEvent.time = jackEvent.time;
  149. fRetEvent.channel = midiChannel;
  150. if (MIDI_IS_STATUS_CONTROL_CHANGE(midiStatus))
  151. {
  152. const uint8_t midiControl = jackEvent.buffer[1];
  153. fRetEvent.type = kEngineEventTypeControl;
  154. if (MIDI_IS_CONTROL_BANK_SELECT(midiControl))
  155. {
  156. const uint8_t midiBank = jackEvent.buffer[2];
  157. fRetEvent.ctrl.type = kEngineControlEventTypeMidiBank;
  158. fRetEvent.ctrl.param = midiBank;
  159. fRetEvent.ctrl.value = 0.0;
  160. }
  161. else if (midiControl == MIDI_CONTROL_ALL_SOUND_OFF)
  162. {
  163. fRetEvent.ctrl.type = kEngineControlEventTypeAllSoundOff;
  164. fRetEvent.ctrl.param = 0;
  165. fRetEvent.ctrl.value = 0.0;
  166. }
  167. else if (midiControl == MIDI_CONTROL_ALL_NOTES_OFF)
  168. {
  169. fRetEvent.ctrl.type = kEngineControlEventTypeAllNotesOff;
  170. fRetEvent.ctrl.param = 0;
  171. fRetEvent.ctrl.value = 0.0;
  172. }
  173. else
  174. {
  175. const uint8_t midiValue = jackEvent.buffer[2];
  176. fRetEvent.ctrl.type = kEngineControlEventTypeParameter;
  177. fRetEvent.ctrl.param = midiControl;
  178. fRetEvent.ctrl.value = double(midiValue)/127.0;
  179. }
  180. }
  181. else if (MIDI_IS_STATUS_PROGRAM_CHANGE(midiStatus))
  182. {
  183. const uint8_t midiProgram = jackEvent.buffer[1];
  184. fRetEvent.type = kEngineEventTypeControl;
  185. fRetEvent.ctrl.type = kEngineControlEventTypeMidiProgram;
  186. fRetEvent.ctrl.param = midiProgram;
  187. fRetEvent.ctrl.value = 0.0;
  188. }
  189. else
  190. {
  191. fRetEvent.type = kEngineEventTypeMidi;
  192. fRetEvent.midi.data[0] = midiStatus;
  193. fRetEvent.midi.data[1] = jackEvent.buffer[1];
  194. fRetEvent.midi.data[2] = jackEvent.buffer[2];
  195. fRetEvent.midi.size = static_cast<uint8_t>(jackEvent.size);
  196. }
  197. return fRetEvent;
  198. }
  199. void writeControlEvent(const uint32_t time, const uint8_t channel, const EngineControlEventType type, const uint16_t param, const double value)
  200. {
  201. if (kPort == nullptr)
  202. return CarlaEngineEventPort::writeControlEvent(time, channel, type, param, value);
  203. CARLA_ASSERT(! kIsInput);
  204. CARLA_ASSERT(fJackBuffer != nullptr);
  205. CARLA_ASSERT(type != kEngineControlEventTypeNull);
  206. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  207. CARLA_ASSERT(param < MAX_MIDI_VALUE);
  208. CARLA_SAFE_ASSERT(value >= 0.0 && value <= 1.0);
  209. if (kIsInput)
  210. return;
  211. if (fJackBuffer == nullptr)
  212. return;
  213. if (type == kEngineControlEventTypeNull)
  214. return;
  215. if (channel >= MAX_MIDI_CHANNELS)
  216. return;
  217. if (param >= MAX_MIDI_VALUE)
  218. return;
  219. if (type == kEngineControlEventTypeParameter)
  220. {
  221. CARLA_ASSERT(! MIDI_IS_CONTROL_BANK_SELECT(param));
  222. }
  223. const double fixedValue = carla_fixValue<double>(0.0, 1.0, value);
  224. uint8_t data[3] = { 0 };
  225. uint8_t size = 0;
  226. switch (type)
  227. {
  228. case kEngineControlEventTypeNull:
  229. break;
  230. case kEngineControlEventTypeParameter:
  231. data[0] = MIDI_STATUS_CONTROL_CHANGE + channel;
  232. data[1] = static_cast<uint8_t>(param);
  233. data[2] = uint8_t(fixedValue * 127.0);
  234. size = 3;
  235. break;
  236. case kEngineControlEventTypeMidiBank:
  237. data[0] = MIDI_STATUS_CONTROL_CHANGE + channel;
  238. data[1] = MIDI_CONTROL_BANK_SELECT;
  239. data[2] = static_cast<uint8_t>(param);
  240. size = 3;
  241. break;
  242. case kEngineControlEventTypeMidiProgram:
  243. data[0] = MIDI_STATUS_PROGRAM_CHANGE + channel;
  244. data[1] = static_cast<uint8_t>(param);
  245. size = 2;
  246. break;
  247. case kEngineControlEventTypeAllSoundOff:
  248. data[0] = MIDI_STATUS_CONTROL_CHANGE + channel;
  249. data[1] = MIDI_CONTROL_ALL_SOUND_OFF;
  250. size = 2;
  251. break;
  252. case kEngineControlEventTypeAllNotesOff:
  253. data[0] = MIDI_STATUS_CONTROL_CHANGE + channel;
  254. data[1] = MIDI_CONTROL_ALL_NOTES_OFF;
  255. size = 2;
  256. break;
  257. }
  258. if (size > 0)
  259. jackbridge_midi_event_write(fJackBuffer, time, data, size);
  260. }
  261. void writeMidiEvent(const uint32_t time, const uint8_t channel, const uint8_t port, const uint8_t* const data, const uint8_t size)
  262. {
  263. if (kPort == nullptr)
  264. return CarlaEngineEventPort::writeMidiEvent(time, channel, port, data, size);
  265. CARLA_ASSERT(! kIsInput);
  266. CARLA_ASSERT(fJackBuffer != nullptr);
  267. CARLA_ASSERT(channel < MAX_MIDI_CHANNELS);
  268. CARLA_ASSERT(data != nullptr);
  269. CARLA_ASSERT(size > 0);
  270. if (kIsInput)
  271. return;
  272. if (fJackBuffer == nullptr)
  273. return;
  274. if (channel >= MAX_MIDI_CHANNELS)
  275. return;
  276. if (data == nullptr)
  277. return;
  278. if (size == 0)
  279. return;
  280. uint8_t jdata[size];
  281. carla_copy<uint8_t>(jdata, data, size);
  282. jdata[0] = data[0] + channel;
  283. jackbridge_midi_event_write(fJackBuffer, time, jdata, size);
  284. }
  285. private:
  286. jack_client_t* const kClient;
  287. jack_port_t* const kPort;
  288. void* fJackBuffer;
  289. EngineEvent fRetEvent;
  290. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackEventPort)
  291. };
  292. // -------------------------------------------------------------------------------------------------------------------
  293. // Jack Engine client
  294. class CarlaEngineJackClient : public CarlaEngineClient
  295. {
  296. public:
  297. CarlaEngineJackClient(const EngineType engineType, const ProcessMode processMode, jack_client_t* const client)
  298. : CarlaEngineClient(engineType, processMode),
  299. kClient(client),
  300. kUseClient(processMode == PROCESS_MODE_SINGLE_CLIENT || processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  301. {
  302. carla_debug("CarlaEngineJackClient::CarlaEngineJackClient(%s, %s, %p)", EngineType2Str(engineType), ProcessMode2Str(processMode), client);
  303. if (kUseClient)
  304. {
  305. CARLA_ASSERT(kClient != nullptr);
  306. }
  307. else
  308. {
  309. CARLA_ASSERT(kClient == nullptr);
  310. }
  311. }
  312. ~CarlaEngineJackClient()
  313. {
  314. carla_debug("CarlaEngineClient::~CarlaEngineClient()");
  315. if (kProcessMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  316. {
  317. if (kClient)
  318. jackbridge_client_close(kClient);
  319. }
  320. }
  321. void activate()
  322. {
  323. carla_debug("CarlaEngineJackClient::activate()");
  324. if (kProcessMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  325. {
  326. CARLA_ASSERT(kClient && ! fActive);
  327. if (kClient && ! fActive)
  328. jackbridge_activate(kClient);
  329. }
  330. CarlaEngineClient::activate();
  331. }
  332. void deactivate()
  333. {
  334. carla_debug("CarlaEngineJackClient::deactivate()");
  335. if (kProcessMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  336. {
  337. CARLA_ASSERT(kClient && fActive);
  338. if (kClient && fActive)
  339. jackbridge_deactivate(kClient);
  340. }
  341. CarlaEngineClient::deactivate();
  342. }
  343. bool isOk() const
  344. {
  345. carla_debug("CarlaEngineJackClient::isOk()");
  346. if (kUseClient)
  347. return bool(kClient);
  348. return CarlaEngineClient::isOk();
  349. }
  350. void setLatency(const uint32_t samples)
  351. {
  352. CarlaEngineClient::setLatency(samples);
  353. if (kUseClient)
  354. jackbridge_recompute_total_latencies(kClient);
  355. }
  356. CarlaEnginePort* addPort(const EnginePortType portType, const char* const name, const bool isInput)
  357. {
  358. carla_debug("CarlaEngineJackClient::addPort(%s, \"%s\", %s)", EnginePortType2Str(portType), name, bool2str(isInput));
  359. jack_port_t* port = nullptr;
  360. // Create JACK port first, if needed
  361. if (kUseClient)
  362. {
  363. switch (portType)
  364. {
  365. case kEnginePortTypeNull:
  366. break;
  367. case kEnginePortTypeAudio:
  368. port = jackbridge_port_register(kClient, name, JACK_DEFAULT_AUDIO_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  369. break;
  370. case kEnginePortTypeEvent:
  371. port = jackbridge_port_register(kClient, name, JACK_DEFAULT_MIDI_TYPE, isInput ? JackPortIsInput : JackPortIsOutput, 0);
  372. break;
  373. }
  374. }
  375. // Create Engine port
  376. switch (portType)
  377. {
  378. case kEnginePortTypeNull:
  379. break;
  380. case kEnginePortTypeAudio:
  381. return new CarlaEngineJackAudioPort(isInput, kProcessMode, kClient, port);
  382. case kEnginePortTypeEvent:
  383. return new CarlaEngineJackEventPort(isInput, kProcessMode, kClient, port);
  384. }
  385. carla_stderr("CarlaEngineJackClient::addPort(%s, \"%s\", %s) - invalid type", EnginePortType2Str(portType), name, bool2str(isInput));
  386. return nullptr;
  387. }
  388. private:
  389. jack_client_t* const kClient;
  390. const bool kUseClient;
  391. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJackClient)
  392. };
  393. // -------------------------------------------------------------------------------------------------------------------
  394. // Jack Engine
  395. class CarlaEngineJack : public CarlaEngine
  396. {
  397. public:
  398. CarlaEngineJack()
  399. : CarlaEngine(),
  400. fClient(nullptr),
  401. fTransportState(JackTransportStopped),
  402. fFreewheel(false),
  403. #ifdef BUILD_BRIDGE
  404. fHasQuit(false)
  405. #else
  406. fRackPorts{nullptr},
  407. fLastGroupId(0),
  408. fLastPortId(0),
  409. fLastConnectionId(0)
  410. #endif
  411. {
  412. carla_debug("CarlaEngineJack::CarlaEngineJack()");
  413. #ifdef BUILD_BRIDGE
  414. fOptions.processMode = PROCESS_MODE_MULTIPLE_CLIENTS;
  415. #endif
  416. carla_zeroStruct<jack_position_t>(fTransportPos);
  417. }
  418. ~CarlaEngineJack()
  419. {
  420. carla_debug("CarlaEngineJack::~CarlaEngineJack()");
  421. CARLA_ASSERT(fClient == nullptr);
  422. #ifndef BUILD_BRIDGE
  423. fUsedGroupNames.clear();
  424. fUsedPortNames.clear();
  425. fUsedConnections.clear();
  426. #endif
  427. }
  428. // -------------------------------------------------------------------
  429. // Maximum values
  430. unsigned int maxClientNameSize()
  431. {
  432. if (fOptions.processMode == PROCESS_MODE_SINGLE_CLIENT || fOptions.processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  433. return static_cast<unsigned int>(jackbridge_client_name_size());
  434. return CarlaEngine::maxClientNameSize();
  435. }
  436. unsigned int maxPortNameSize()
  437. {
  438. if (fOptions.processMode == PROCESS_MODE_SINGLE_CLIENT || fOptions.processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  439. return static_cast<unsigned int>(jackbridge_port_name_size());
  440. return CarlaEngine::maxPortNameSize();
  441. }
  442. // -------------------------------------------------------------------
  443. // Virtual, per-engine type calls
  444. bool init(const char* const clientName)
  445. {
  446. carla_debug("CarlaEngineJack::init(\"%s\")", clientName);
  447. fFreewheel = false;
  448. fTransportState = JackTransportStopped;
  449. carla_zeroStruct<jack_position_t>(fTransportPos);
  450. #ifndef BUILD_BRIDGE
  451. fLastGroupId = 0;
  452. fLastPortId = 0;
  453. fLastConnectionId = 0;
  454. fUsedGroupNames.clear();
  455. fUsedPortNames.clear();
  456. fUsedConnections.clear();
  457. fClient = jackbridge_client_open(clientName, JackNullOption, nullptr);
  458. if (fClient != nullptr)
  459. {
  460. fBufferSize = jackbridge_get_buffer_size(fClient);
  461. fSampleRate = jackbridge_get_sample_rate(fClient);
  462. jackbridge_set_buffer_size_callback(fClient, carla_jack_bufsize_callback, this);
  463. jackbridge_set_sample_rate_callback(fClient, carla_jack_srate_callback, this);
  464. jackbridge_set_freewheel_callback(fClient, carla_jack_freewheel_callback, this);
  465. jackbridge_set_process_callback(fClient, carla_jack_process_callback, this);
  466. jackbridge_set_latency_callback(fClient, carla_jack_latency_callback, this);
  467. jackbridge_on_shutdown(fClient, carla_jack_shutdown_callback, this);
  468. if (fOptions.processMode == PROCESS_MODE_CONTINUOUS_RACK)
  469. {
  470. fRackPorts[rackPortAudioIn1] = jackbridge_port_register(fClient, "audio-in1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  471. fRackPorts[rackPortAudioIn2] = jackbridge_port_register(fClient, "audio-in2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0);
  472. fRackPorts[rackPortAudioOut1] = jackbridge_port_register(fClient, "audio-out1", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  473. fRackPorts[rackPortAudioOut2] = jackbridge_port_register(fClient, "audio-out2", JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0);
  474. fRackPorts[rackPortEventIn] = jackbridge_port_register(fClient, "events-in", JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0);
  475. fRackPorts[rackPortEventOut] = jackbridge_port_register(fClient, "events-out", JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput, 0);
  476. }
  477. // TODO - update jackbridge
  478. jack_set_client_registration_callback(fClient, carla_jack_client_registration_callback, this);
  479. jack_set_port_registration_callback(fClient, carla_jack_port_registration_callback, this);
  480. jack_set_port_connect_callback(fClient, carla_jack_port_connect_callback, this);
  481. if (jack_set_port_rename_callback)
  482. jack_set_port_rename_callback(fClient, carla_jack_port_rename_callback, this);
  483. if (jackbridge_activate(fClient) == 0)
  484. {
  485. const char* const jackClientName = jackbridge_get_client_name(fClient);
  486. return CarlaEngine::init(jackClientName);
  487. }
  488. else
  489. {
  490. setLastError("Failed to activate the JACK client");
  491. jackbridge_client_close(fClient);
  492. fClient = nullptr;
  493. }
  494. }
  495. else
  496. setLastError("Failed to create new JACK client");
  497. return false;
  498. #else
  499. if (fBufferSize == 0 || fSampleRate == 0.0)
  500. {
  501. // open temp client to get initial buffer-size and sample-rate values
  502. if (jack_client_t* tmpClient = jackbridge_client_open(clientName, JackNullOption, nullptr))
  503. {
  504. fBufferSize = jackbridge_get_buffer_size(tmpClient);
  505. fSampleRate = jackbridge_get_sample_rate(tmpClient);
  506. jackbridge_client_close(tmpClient);
  507. }
  508. }
  509. return CarlaEngine::init(clientName);
  510. #endif
  511. }
  512. bool close()
  513. {
  514. carla_debug("CarlaEngineJack::close()");
  515. CarlaEngine::close();
  516. #ifdef BUILD_BRIDGE
  517. fClient = nullptr;
  518. fHasQuit = true;
  519. return true;
  520. #else
  521. if (jackbridge_deactivate(fClient) == 0)
  522. {
  523. if (fOptions.processMode == PROCESS_MODE_CONTINUOUS_RACK)
  524. {
  525. jackbridge_port_unregister(fClient, fRackPorts[rackPortAudioIn1]);
  526. jackbridge_port_unregister(fClient, fRackPorts[rackPortAudioIn2]);
  527. jackbridge_port_unregister(fClient, fRackPorts[rackPortAudioOut1]);
  528. jackbridge_port_unregister(fClient, fRackPorts[rackPortAudioOut2]);
  529. jackbridge_port_unregister(fClient, fRackPorts[rackPortEventIn]);
  530. jackbridge_port_unregister(fClient, fRackPorts[rackPortEventOut]);
  531. }
  532. if (jackbridge_client_close(fClient) == 0)
  533. {
  534. fClient = nullptr;
  535. return true;
  536. }
  537. else
  538. setLastError("Failed to close the JACK client");
  539. }
  540. else
  541. setLastError("Failed to deactivate the JACK client");
  542. fClient = nullptr;
  543. fUsedGroupNames.clear();
  544. fUsedPortNames.clear();
  545. fUsedConnections.clear();
  546. #endif
  547. return false;
  548. }
  549. bool isRunning() const
  550. {
  551. #ifdef BUILD_BRIDGE
  552. return (fClient != nullptr || ! fHasQuit);
  553. #else
  554. return (fClient != nullptr);
  555. #endif
  556. }
  557. bool isOffline() const
  558. {
  559. return fFreewheel;
  560. }
  561. EngineType type() const
  562. {
  563. return kEngineTypeJack;
  564. }
  565. CarlaEngineClient* addClient(CarlaPlugin* const plugin)
  566. {
  567. jack_client_t* client = nullptr;
  568. #ifdef BUILD_BRIDGE
  569. client = fClient = jackbridge_client_open(plugin->name(), JackNullOption, nullptr);
  570. fBufferSize = jackbridge_get_buffer_size(client);
  571. fSampleRate = jackbridge_get_sample_rate(client);
  572. jackbridge_set_buffer_size_callback(client, carla_jack_bufsize_callback, this);
  573. jackbridge_set_sample_rate_callback(client, carla_jack_srate_callback, this);
  574. jackbridge_set_freewheel_callback(client, carla_jack_freewheel_callback, this);
  575. jackbridge_set_process_callback(client, carla_jack_process_callback, this);
  576. jackbridge_set_latency_callback(client, carla_jack_latency_callback, this);
  577. jackbridge_on_shutdown(client, carla_jack_shutdown_callback, this);
  578. #else
  579. if (fOptions.processMode == PROCESS_MODE_SINGLE_CLIENT)
  580. {
  581. client = fClient;
  582. }
  583. else if (fOptions.processMode == PROCESS_MODE_MULTIPLE_CLIENTS)
  584. {
  585. client = jackbridge_client_open(plugin->name(), JackNullOption, nullptr);
  586. jackbridge_set_process_callback(client, carla_jack_process_callback_plugin, plugin);
  587. jackbridge_set_latency_callback(client, carla_jack_latency_callback_plugin, plugin);
  588. }
  589. #endif
  590. return new CarlaEngineJackClient(kEngineTypeJack, fOptions.processMode, client);
  591. }
  592. // -------------------------------------------------------------------
  593. // Transport
  594. void transportPlay()
  595. {
  596. if (fOptions.transportMode == TRANSPORT_MODE_INTERNAL)
  597. CarlaEngine::transportPlay();
  598. else if (fClient != nullptr)
  599. jackbridge_transport_start(fClient);
  600. }
  601. void transportPause()
  602. {
  603. if (fOptions.transportMode == TRANSPORT_MODE_INTERNAL)
  604. CarlaEngine::transportPause();
  605. else if (fClient != nullptr)
  606. jackbridge_transport_stop(fClient);
  607. }
  608. void transportRelocate(const uint32_t frame)
  609. {
  610. if (fOptions.transportMode == TRANSPORT_MODE_INTERNAL)
  611. CarlaEngine::transportRelocate(frame);
  612. else if (fClient != nullptr)
  613. jackbridge_transport_locate(fClient, frame);
  614. }
  615. // -------------------------------------
  616. protected:
  617. void handleJackBufferSizeCallback(const uint32_t newBufferSize)
  618. {
  619. if (fBufferSize != newBufferSize)
  620. {
  621. fBufferSize = newBufferSize;
  622. bufferSizeChanged(newBufferSize);
  623. }
  624. }
  625. void handleJackSampleRateCallback(const double newSampleRate)
  626. {
  627. if (fSampleRate != newSampleRate)
  628. {
  629. fSampleRate = newSampleRate;
  630. sampleRateChanged(newSampleRate);
  631. }
  632. }
  633. void handleJackFreewheelCallback(const bool isFreewheel)
  634. {
  635. fFreewheel = isFreewheel;
  636. }
  637. void saveTransportInfo()
  638. {
  639. if (fOptions.transportMode != TRANSPORT_MODE_JACK)
  640. return;
  641. fTransportPos.unique_1 = fTransportPos.unique_2 + 1; // invalidate
  642. fTransportState = jackbridge_transport_query(fClient, &fTransportPos);
  643. fTimeInfo.playing = (fTransportState == JackTransportRolling);
  644. if (fTransportPos.unique_1 == fTransportPos.unique_2)
  645. {
  646. fTimeInfo.frame = fTransportPos.frame;
  647. fTimeInfo.time = fTransportPos.usecs;
  648. if (fTransportPos.valid & JackPositionBBT)
  649. {
  650. fTimeInfo.valid = EngineTimeInfo::ValidBBT;
  651. fTimeInfo.bbt.bar = fTransportPos.bar;
  652. fTimeInfo.bbt.beat = fTransportPos.beat;
  653. fTimeInfo.bbt.tick = fTransportPos.tick;
  654. fTimeInfo.bbt.barStartTick = fTransportPos.bar_start_tick;
  655. fTimeInfo.bbt.beatsPerBar = fTransportPos.beats_per_bar;
  656. fTimeInfo.bbt.beatType = fTransportPos.beat_type;
  657. fTimeInfo.bbt.ticksPerBeat = fTransportPos.ticks_per_beat;
  658. fTimeInfo.bbt.beatsPerMinute = fTransportPos.beats_per_minute;
  659. }
  660. else
  661. fTimeInfo.valid = 0x0;
  662. }
  663. else
  664. {
  665. fTimeInfo.frame = 0;
  666. fTimeInfo.valid = 0x0;
  667. }
  668. }
  669. void handleJackProcessCallback(const uint32_t nframes)
  670. {
  671. #ifndef BUILD_BRIDGE
  672. if (kData->curPluginCount == 0)
  673. {
  674. // pass-through
  675. if (fOptions.processMode == PROCESS_MODE_CONTINUOUS_RACK)
  676. {
  677. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioIn1], nframes);
  678. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioIn2], nframes);
  679. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioOut1], nframes);
  680. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioOut2], nframes);
  681. void* const eventOut = jackbridge_port_get_buffer(fRackPorts[rackPortEventOut], nframes);
  682. CARLA_ASSERT(audioIn1 != nullptr);
  683. CARLA_ASSERT(audioIn2 != nullptr);
  684. CARLA_ASSERT(audioOut1 != nullptr);
  685. CARLA_ASSERT(audioOut2 != nullptr);
  686. CARLA_ASSERT(eventOut != nullptr);
  687. carla_copyFloat(audioOut1, audioIn1, nframes);
  688. carla_copyFloat(audioOut2, audioIn2, nframes);
  689. jackbridge_midi_clear_buffer(eventOut);
  690. }
  691. return proccessPendingEvents();
  692. }
  693. #endif
  694. saveTransportInfo();
  695. #ifdef BUILD_BRIDGE
  696. CarlaPlugin* const plugin = getPluginUnchecked(0);
  697. if (plugin && plugin->enabled())
  698. {
  699. plugin->initBuffers();
  700. processPlugin(plugin, nframes);
  701. }
  702. #else
  703. if (fOptions.processMode == PROCESS_MODE_SINGLE_CLIENT)
  704. {
  705. for (unsigned int i=0; i < kData->curPluginCount; i++)
  706. {
  707. CarlaPlugin* const plugin = getPluginUnchecked(i);
  708. if (plugin && plugin->enabled())
  709. {
  710. plugin->initBuffers();
  711. processPlugin(plugin, nframes);
  712. }
  713. }
  714. }
  715. else if (fOptions.processMode == PROCESS_MODE_CONTINUOUS_RACK)
  716. {
  717. // get buffers from jack
  718. float* const audioIn1 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioIn1], nframes);
  719. float* const audioIn2 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioIn2], nframes);
  720. float* const audioOut1 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioOut1], nframes);
  721. float* const audioOut2 = (float*)jackbridge_port_get_buffer(fRackPorts[rackPortAudioOut2], nframes);
  722. void* const eventIn = jackbridge_port_get_buffer(fRackPorts[rackPortEventIn], nframes);
  723. void* const eventOut = jackbridge_port_get_buffer(fRackPorts[rackPortEventOut], nframes);
  724. // assert buffers
  725. CARLA_ASSERT(audioIn1 != nullptr);
  726. CARLA_ASSERT(audioIn2 != nullptr);
  727. CARLA_ASSERT(audioOut1 != nullptr);
  728. CARLA_ASSERT(audioOut2 != nullptr);
  729. CARLA_ASSERT(eventIn != nullptr);
  730. CARLA_ASSERT(eventOut != nullptr);
  731. // create audio buffers
  732. float* inBuf[2] = { audioIn1, audioIn2 };
  733. float* outBuf[2] = { audioOut1, audioOut2 };
  734. // initialize input events
  735. carla_zeroMem(kData->rack.in, sizeof(EngineEvent)*RACK_EVENT_COUNT);
  736. {
  737. uint32_t engineEventIndex = 0;
  738. jack_midi_event_t jackEvent;
  739. const uint32_t jackEventCount = jackbridge_midi_get_event_count(eventIn);
  740. for (uint32_t jackEventIndex=0; jackEventIndex < jackEventCount; jackEventIndex++)
  741. {
  742. if (jackbridge_midi_event_get(&jackEvent, eventIn, jackEventIndex) != 0)
  743. continue;
  744. EngineEvent* const engineEvent = &kData->rack.in[engineEventIndex++];
  745. engineEvent->clear();
  746. const uint8_t midiStatus = MIDI_GET_STATUS_FROM_DATA(jackEvent.buffer);
  747. const uint8_t midiChannel = MIDI_GET_CHANNEL_FROM_DATA(jackEvent.buffer);
  748. engineEvent->time = jackEvent.time;
  749. engineEvent->channel = midiChannel;
  750. if (MIDI_IS_STATUS_CONTROL_CHANGE(midiStatus))
  751. {
  752. const uint8_t midiControl = jackEvent.buffer[1];
  753. engineEvent->type = kEngineEventTypeControl;
  754. if (MIDI_IS_CONTROL_BANK_SELECT(midiControl))
  755. {
  756. const uint8_t midiBank = jackEvent.buffer[2];
  757. engineEvent->ctrl.type = kEngineControlEventTypeMidiBank;
  758. engineEvent->ctrl.param = midiBank;
  759. engineEvent->ctrl.value = 0.0;
  760. }
  761. else if (midiControl == MIDI_CONTROL_ALL_SOUND_OFF)
  762. {
  763. engineEvent->ctrl.type = kEngineControlEventTypeAllSoundOff;
  764. engineEvent->ctrl.param = 0;
  765. engineEvent->ctrl.value = 0.0;
  766. }
  767. else if (midiControl == MIDI_CONTROL_ALL_NOTES_OFF)
  768. {
  769. engineEvent->ctrl.type = kEngineControlEventTypeAllNotesOff;
  770. engineEvent->ctrl.param = 0;
  771. engineEvent->ctrl.value = 0.0;
  772. }
  773. else
  774. {
  775. const uint8_t midiValue = jackEvent.buffer[2];
  776. engineEvent->ctrl.type = kEngineControlEventTypeParameter;
  777. engineEvent->ctrl.param = midiControl;
  778. engineEvent->ctrl.value = double(midiValue)/127.0;
  779. }
  780. }
  781. else if (MIDI_IS_STATUS_PROGRAM_CHANGE(midiStatus))
  782. {
  783. const uint8_t midiProgram = jackEvent.buffer[1];
  784. engineEvent->type = kEngineEventTypeControl;
  785. engineEvent->ctrl.type = kEngineControlEventTypeMidiProgram;
  786. engineEvent->ctrl.param = midiProgram;
  787. engineEvent->ctrl.value = 0.0;
  788. }
  789. else
  790. {
  791. engineEvent->type = kEngineEventTypeMidi;
  792. engineEvent->midi.data[0] = midiStatus;
  793. engineEvent->midi.data[1] = jackEvent.buffer[1];
  794. engineEvent->midi.data[2] = jackEvent.buffer[2];
  795. engineEvent->midi.size = static_cast<uint8_t>(jackEvent.size);
  796. }
  797. if (engineEventIndex >= RACK_EVENT_COUNT)
  798. break;
  799. }
  800. }
  801. // process rack
  802. processRack(inBuf, outBuf, nframes);
  803. // output control
  804. {
  805. jackbridge_midi_clear_buffer(eventOut);
  806. for (unsigned short i=0; i < RACK_EVENT_COUNT; i++)
  807. {
  808. EngineEvent* const engineEvent = &kData->rack.out[i];
  809. uint8_t data[3] = { 0 };
  810. uint8_t size = 0;
  811. switch (engineEvent->type)
  812. {
  813. case kEngineEventTypeNull:
  814. break;
  815. case kEngineEventTypeControl:
  816. {
  817. EngineControlEvent* const ctrlEvent = &engineEvent->ctrl;
  818. if (ctrlEvent->type == kEngineControlEventTypeParameter && MIDI_IS_CONTROL_BANK_SELECT(ctrlEvent->param))
  819. {
  820. // FIXME?
  821. ctrlEvent->type = kEngineControlEventTypeMidiBank;
  822. ctrlEvent->param = ctrlEvent->value;
  823. ctrlEvent->value = 0.0;
  824. }
  825. switch (ctrlEvent->type)
  826. {
  827. case kEngineControlEventTypeNull:
  828. break;
  829. case kEngineControlEventTypeParameter:
  830. data[0] = MIDI_STATUS_CONTROL_CHANGE + engineEvent->channel;
  831. data[1] = static_cast<uint8_t>(ctrlEvent->param);
  832. data[2] = uint8_t(ctrlEvent->value * 127.0);
  833. size = 3;
  834. break;
  835. case kEngineControlEventTypeMidiBank:
  836. data[0] = MIDI_STATUS_CONTROL_CHANGE + engineEvent->channel;
  837. data[1] = MIDI_CONTROL_BANK_SELECT;
  838. data[2] = static_cast<uint8_t>(ctrlEvent->param);
  839. size = 3;
  840. break;
  841. case kEngineControlEventTypeMidiProgram:
  842. data[0] = MIDI_STATUS_PROGRAM_CHANGE + engineEvent->channel;
  843. data[1] = static_cast<uint8_t>(ctrlEvent->param);
  844. size = 2;
  845. break;
  846. case kEngineControlEventTypeAllSoundOff:
  847. data[0] = MIDI_STATUS_CONTROL_CHANGE + engineEvent->channel;
  848. data[1] = MIDI_CONTROL_ALL_SOUND_OFF;
  849. size = 2;
  850. break;
  851. case kEngineControlEventTypeAllNotesOff:
  852. data[0] = MIDI_STATUS_CONTROL_CHANGE + engineEvent->channel;
  853. data[1] = MIDI_CONTROL_ALL_NOTES_OFF;
  854. size = 2;
  855. break;
  856. }
  857. break;
  858. }
  859. case kEngineEventTypeMidi:
  860. {
  861. EngineMidiEvent* const midiEvent = &engineEvent->midi;
  862. data[0] = midiEvent->data[0];
  863. data[1] = midiEvent->data[1];
  864. data[2] = midiEvent->data[2];
  865. size = midiEvent->size;
  866. break;
  867. }
  868. }
  869. if (size > 0)
  870. jackbridge_midi_event_write(eventOut, engineEvent->time, data, size);
  871. }
  872. }
  873. }
  874. #endif // ! BUILD_BRIDGE
  875. proccessPendingEvents();
  876. }
  877. void handleJackLatencyCallback(const jack_latency_callback_mode_t mode)
  878. {
  879. if (fOptions.processMode != PROCESS_MODE_SINGLE_CLIENT)
  880. return;
  881. for (unsigned int i=0; i < kData->curPluginCount; i++)
  882. {
  883. CarlaPlugin* const plugin = getPluginUnchecked(i);
  884. if (plugin && plugin->enabled())
  885. latencyPlugin(plugin, mode);
  886. }
  887. }
  888. #ifndef BUILD_BRIDGE
  889. void handleJackClientRegistrationCallback(const char* name, bool reg)
  890. {
  891. if (reg)
  892. {
  893. GroupNameToId groupNameToId;
  894. groupNameToId.id = fLastGroupId;
  895. groupNameToId.name = name;
  896. callback(CALLBACK_PATCHBAY_CLIENT_ADDED, 0, fLastGroupId, 0, 0.0f, name);
  897. fUsedGroupNames.append(groupNameToId);
  898. fLastGroupId++;
  899. }
  900. else
  901. {
  902. for (int i=0, count=fUsedGroupNames.count(); i < count; i++)
  903. {
  904. if (fUsedGroupNames[i].name == name)
  905. {
  906. callback(CALLBACK_PATCHBAY_CLIENT_REMOVED, 0, fUsedGroupNames[i].id, 0, 0.0f, nullptr);
  907. fUsedGroupNames.takeAt(i);
  908. break;
  909. }
  910. }
  911. }
  912. }
  913. void handleJackPortRegistrationCallback(jack_port_id_t port, bool reg)
  914. {
  915. jack_port_t* jackPort = jack_port_by_id(fClient, port);
  916. QString fullName(jack_port_name(jackPort));
  917. QString groupName = fullName.split(":").at(0);
  918. int groupId = getGroupId(groupName);
  919. const char* portName = jack_port_short_name(jackPort);
  920. if (reg)
  921. {
  922. bool portIsInput = (jack_port_flags(jackPort) & JackPortIsInput);
  923. bool portIsAudio = (std::strcmp(jack_port_type(jackPort), JACK_DEFAULT_AUDIO_TYPE) == 0);
  924. unsigned int portFlags = 0x0;
  925. portFlags |= portIsInput ? PATCHBAY_PORT_IS_INPUT : PATCHBAY_PORT_IS_OUTPUT;
  926. portFlags |= portIsAudio ? PATCHBAY_PORT_IS_AUDIO : PATCHBAY_PORT_IS_MIDI;
  927. PortNameToId portNameToId;
  928. portNameToId.groupId = groupId;
  929. portNameToId.portId = fLastPortId;
  930. portNameToId.name = portName;
  931. fUsedPortNames.append(portNameToId);
  932. callback(CALLBACK_PATCHBAY_PORT_ADDED, 0, groupId, fLastPortId, portFlags, portName);
  933. fLastPortId++;
  934. }
  935. else
  936. {
  937. for (int i=0, count=fUsedPortNames.count(); i < count; i++)
  938. {
  939. if (fUsedPortNames[i].groupId == groupId && fUsedPortNames[i].name == portName)
  940. {
  941. callback(CALLBACK_PATCHBAY_PORT_REMOVED, 0, fUsedPortNames[i].portId, 0, 0.0f, nullptr);
  942. fUsedPortNames.takeAt(i);
  943. break;
  944. }
  945. }
  946. }
  947. }
  948. void handleJackPortConnectCallback(jack_port_id_t a, jack_port_id_t b, bool connect)
  949. {
  950. jack_port_t* jackPortA = jack_port_by_id(fClient, a);
  951. jack_port_t* jackPortB = jack_port_by_id(fClient, b);
  952. int portIdA = getPortId(QString(jack_port_name(jackPortA)));
  953. int portIdB = getPortId(QString(jack_port_name(jackPortB)));
  954. if (connect)
  955. {
  956. ConnectionToId connectionToId;
  957. connectionToId.id = fLastConnectionId;
  958. connectionToId.portOut = portIdA;
  959. connectionToId.portIn = portIdB;
  960. fUsedConnections.append(connectionToId);
  961. callback(CALLBACK_PATCHBAY_CONNECTION_ADDED, 0, fLastConnectionId, portIdA, portIdB, nullptr);
  962. fLastConnectionId++;
  963. }
  964. else
  965. {
  966. for (int i=0, count=fUsedConnections.count(); i < count; i++)
  967. {
  968. if (fUsedConnections[i].portOut == portIdA && fUsedConnections[i].portIn == portIdB)
  969. {
  970. callback(CALLBACK_PATCHBAY_CONNECTION_REMOVED, 0, fUsedConnections[i].id, 0, 0.0f, nullptr);
  971. fUsedConnections.takeAt(i);
  972. break;
  973. }
  974. }
  975. }
  976. }
  977. int handleJackPortRenameCallback(jack_port_id_t port, const char* oldName, const char* newName)
  978. {
  979. }
  980. #endif
  981. void handleJackShutdownCallback()
  982. {
  983. for (unsigned int i=0; i < kData->curPluginCount; i++)
  984. {
  985. //CarlaPlugin* const plugin = getPluginUnchecked(i);
  986. //if (plugin)
  987. // plugin->x_client = nullptr;
  988. }
  989. fClient = nullptr;
  990. callback(CALLBACK_QUIT, 0, 0, 0, 0.0f, nullptr);
  991. }
  992. // -------------------------------------
  993. private:
  994. jack_client_t* fClient;
  995. jack_position_t fTransportPos;
  996. jack_transport_state_t fTransportState;
  997. bool fFreewheel;
  998. // -------------------------------------
  999. #ifdef BUILD_BRIDGE
  1000. bool fHasQuit;
  1001. #else
  1002. enum RackPorts {
  1003. rackPortAudioIn1 = 0,
  1004. rackPortAudioIn2 = 1,
  1005. rackPortAudioOut1 = 2,
  1006. rackPortAudioOut2 = 3,
  1007. rackPortEventIn = 4,
  1008. rackPortEventOut = 5,
  1009. rackPortCount = 8
  1010. };
  1011. jack_port_t* fRackPorts[rackPortCount];
  1012. struct GroupNameToId {
  1013. int id;
  1014. QString name;
  1015. };
  1016. struct PortNameToId {
  1017. int groupId;
  1018. int portId;
  1019. QString name;
  1020. };
  1021. struct ConnectionToId {
  1022. int id;
  1023. int portOut;
  1024. int portIn;
  1025. };
  1026. int fLastGroupId;
  1027. int fLastPortId;
  1028. int fLastConnectionId ;
  1029. QList<GroupNameToId> fUsedGroupNames;
  1030. QList<PortNameToId> fUsedPortNames;
  1031. QList<ConnectionToId> fUsedConnections;
  1032. int getGroupId(QString groupName)
  1033. {
  1034. for (int i=0, count=fUsedGroupNames.count(); i < count; i++)
  1035. {
  1036. if (fUsedGroupNames[i].name == groupName)
  1037. {
  1038. return fUsedGroupNames[i].id;
  1039. }
  1040. }
  1041. return -1;
  1042. }
  1043. int getPortId(QString fullPortName)
  1044. {
  1045. QString groupName = fullPortName.split(":").at(0);
  1046. QString portName = fullPortName.replace(groupName+":", "");
  1047. int groupId = getGroupId(groupName);
  1048. for (int i=0, count=fUsedPortNames.count(); i < count; i++)
  1049. {
  1050. if (fUsedPortNames[i].groupId == groupId && fUsedPortNames[i].name == portName)
  1051. {
  1052. return fUsedPortNames[i].portId;
  1053. }
  1054. }
  1055. return -1;
  1056. }
  1057. #endif
  1058. // -------------------------------------
  1059. void processPlugin(CarlaPlugin* const plugin, const uint32_t nframes)
  1060. {
  1061. const uint32_t inCount = plugin->audioInCount();
  1062. const uint32_t outCount = plugin->audioOutCount();
  1063. float* inBuffer[inCount];
  1064. float* outBuffer[outCount];
  1065. float inPeaks[inCount];
  1066. float outPeaks[outCount];
  1067. if (inCount > 0)
  1068. carla_zeroFloat(inPeaks, inCount);
  1069. if (outCount > 0)
  1070. carla_zeroFloat(outPeaks, outCount);
  1071. for (uint32_t i=0; i < inCount; i++)
  1072. {
  1073. CarlaEngineAudioPort* const port = CarlaPluginGetAudioInPort(plugin, i);
  1074. inBuffer[i] = port->getBuffer();
  1075. }
  1076. for (uint32_t i=0; i < outCount; i++)
  1077. {
  1078. CarlaEngineAudioPort* const port = CarlaPluginGetAudioOutPort(plugin, i);
  1079. outBuffer[i] = port->getBuffer();
  1080. }
  1081. for (uint32_t i=0; i < inCount; i++)
  1082. {
  1083. for (uint32_t j=0; j < nframes; j++)
  1084. {
  1085. const float absV = std::fabs(inBuffer[i][j]);
  1086. if (absV > inPeaks[i])
  1087. inPeaks[i] = absV;
  1088. }
  1089. }
  1090. plugin->process(inBuffer, outBuffer, nframes);
  1091. for (uint32_t i=0; i < outCount; i++)
  1092. {
  1093. for (uint32_t j=0; j < nframes; j++)
  1094. {
  1095. const float absV = std::fabs(outBuffer[i][j]);
  1096. if (absV > outPeaks[i])
  1097. outPeaks[i] = absV;
  1098. }
  1099. }
  1100. setPeaks(plugin->id(), inPeaks, outPeaks);
  1101. }
  1102. void latencyPlugin(CarlaPlugin* const plugin, jack_latency_callback_mode_t mode)
  1103. {
  1104. const uint32_t inCount = plugin->audioInCount();
  1105. const uint32_t outCount = plugin->audioOutCount();
  1106. jack_latency_range_t range;
  1107. uint32_t pluginLatency = plugin->latency();
  1108. if (pluginLatency == 0)
  1109. return;
  1110. if (mode == JackCaptureLatency)
  1111. {
  1112. for (uint32_t i=0; i < inCount; i++)
  1113. {
  1114. uint aOutI = (i >= outCount) ? outCount : i;
  1115. jack_port_t* const portIn = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioInPort(plugin, i))->kPort;
  1116. jack_port_t* const portOut = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioOutPort(plugin, aOutI))->kPort;
  1117. jackbridge_port_get_latency_range(portIn, mode, &range);
  1118. range.min += pluginLatency;
  1119. range.max += pluginLatency;
  1120. jackbridge_port_set_latency_range(portOut, mode, &range);
  1121. }
  1122. }
  1123. else
  1124. {
  1125. for (uint32_t i=0; i < outCount; i++)
  1126. {
  1127. uint aInI = (i >= inCount) ? inCount : i;
  1128. jack_port_t* const portIn = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioInPort(plugin, aInI))->kPort;
  1129. jack_port_t* const portOut = ((CarlaEngineJackAudioPort*)CarlaPluginGetAudioOutPort(plugin, i))->kPort;
  1130. jackbridge_port_get_latency_range(portOut, mode, &range);
  1131. range.min += pluginLatency;
  1132. range.max += pluginLatency;
  1133. jackbridge_port_set_latency_range(portIn, mode, &range);
  1134. }
  1135. }
  1136. }
  1137. // -------------------------------------
  1138. #define handlePtr ((CarlaEngineJack*)arg)
  1139. static int carla_jack_srate_callback(jack_nframes_t newSampleRate, void* arg)
  1140. {
  1141. handlePtr->handleJackSampleRateCallback(newSampleRate);
  1142. return 0;
  1143. }
  1144. static int carla_jack_bufsize_callback(jack_nframes_t newBufferSize, void* arg)
  1145. {
  1146. handlePtr->handleJackBufferSizeCallback(newBufferSize);
  1147. return 0;
  1148. }
  1149. static void carla_jack_freewheel_callback(int starting, void* arg)
  1150. {
  1151. handlePtr->handleJackFreewheelCallback(bool(starting));
  1152. }
  1153. static int carla_jack_process_callback(jack_nframes_t nframes, void* arg)
  1154. {
  1155. handlePtr->handleJackProcessCallback(nframes);
  1156. return 0;
  1157. }
  1158. static void carla_jack_latency_callback(jack_latency_callback_mode_t mode, void* arg)
  1159. {
  1160. handlePtr->handleJackLatencyCallback(mode);
  1161. }
  1162. #ifndef BUILD_BRIDGE
  1163. static void carla_jack_client_registration_callback(const char* name, int reg, void* arg)
  1164. {
  1165. handlePtr->handleJackClientRegistrationCallback(name, (reg != 0));
  1166. }
  1167. static void carla_jack_port_registration_callback(jack_port_id_t port, int reg, void* arg)
  1168. {
  1169. handlePtr->handleJackPortRegistrationCallback(port, (reg != 0));
  1170. }
  1171. static void carla_jack_port_connect_callback(jack_port_id_t a, jack_port_id_t b, int connect, void* arg)
  1172. {
  1173. handlePtr->handleJackPortConnectCallback(a, b, (connect != 0));
  1174. }
  1175. static int carla_jack_port_rename_callback(jack_port_id_t port, const char* oldName, const char* newName, void* arg)
  1176. {
  1177. handlePtr->handleJackPortRenameCallback(port, oldName, newName);
  1178. return 0;
  1179. }
  1180. #endif
  1181. static void carla_jack_shutdown_callback(void* arg)
  1182. {
  1183. handlePtr->handleJackShutdownCallback();
  1184. }
  1185. #undef handlePtr
  1186. // -------------------------------------
  1187. #ifndef BUILD_BRIDGE
  1188. static int carla_jack_process_callback_plugin(jack_nframes_t nframes, void* arg)
  1189. {
  1190. CarlaPlugin* const plugin = (CarlaPlugin*)arg;
  1191. if (plugin != nullptr && plugin->enabled())
  1192. {
  1193. CarlaEngineJack* const engine = (CarlaEngineJack*)CarlaPluginGetEngine(plugin);
  1194. plugin->initBuffers();
  1195. engine->saveTransportInfo();
  1196. engine->processPlugin(plugin, nframes);
  1197. }
  1198. return 0;
  1199. }
  1200. static void carla_jack_latency_callback_plugin(jack_latency_callback_mode_t mode, void* arg)
  1201. {
  1202. CarlaPlugin* const plugin = (CarlaPlugin*)arg;
  1203. if (plugin != nullptr && plugin->enabled())
  1204. {
  1205. CarlaEngineJack* const engine = (CarlaEngineJack*)CarlaPluginGetEngine(plugin);
  1206. engine->latencyPlugin(plugin, mode);
  1207. }
  1208. }
  1209. #endif
  1210. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineJack)
  1211. };
  1212. // -----------------------------------------
  1213. CarlaEngine* CarlaEngine::newJack()
  1214. {
  1215. return new CarlaEngineJack();
  1216. }
  1217. // -----------------------------------------
  1218. CARLA_BACKEND_END_NAMESPACE
  1219. #endif // CARLA_ENGINE_JACK