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.

380 lines
11KB

  1. /*
  2. * Carla Engine Bridge
  3. * Copyright (C) 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. #ifdef BUILD_BRIDGE
  18. #include "CarlaEngineInternal.hpp"
  19. #include "CarlaBackendUtils.hpp"
  20. #include "CarlaMIDI.h"
  21. #include "CarlaBridgeUtils.hpp"
  22. #include "CarlaShmUtils.hpp"
  23. #include "jackbridge/JackBridge.hpp"
  24. #include <ctime>
  25. CARLA_BACKEND_START_NAMESPACE
  26. #if 0
  27. } // Fix editor indentation
  28. #endif
  29. // -------------------------------------------------------------------
  30. class CarlaEngineBridge : public CarlaEngine,
  31. public QThread
  32. {
  33. public:
  34. CarlaEngineBridge(const char* const audioBaseName, const char* const controlBaseName)
  35. : CarlaEngine(),
  36. fIsRunning(false),
  37. fQuitNow(false),
  38. fEventsInPort(nullptr)
  39. {
  40. carla_debug("CarlaEngineBridge::CarlaEngineBridge()");
  41. fShmAudioPool.filename = "/carla-bridge_shm_";
  42. fShmAudioPool.filename += audioBaseName;
  43. fShmControl.filename = "/carla-bridge_shc_";
  44. fShmControl.filename += controlBaseName;
  45. }
  46. ~CarlaEngineBridge()
  47. {
  48. carla_debug("CarlaEngineBridge::~CarlaEngineBridge()");
  49. }
  50. // -------------------------------------
  51. // CarlaEngine virtual calls
  52. bool init(const char* const clientName)
  53. {
  54. carla_debug("CarlaEngineBridge::init(\"%s\")", clientName);
  55. // SHM Audio Pool
  56. {
  57. #ifdef CARLA_OS_WIN
  58. // TESTING!
  59. fShmAudioPool.shm = carla_shm_attach_linux((const char*)fShmAudioPool.filename);
  60. #else
  61. fShmAudioPool.shm = carla_shm_attach((const char*)fShmAudioPool.filename);
  62. #endif
  63. if (! carla_is_shm_valid(fShmAudioPool.shm))
  64. {
  65. _cleanup();
  66. carla_stdout("Failed to open or create shared memory file #1");
  67. return false;
  68. }
  69. }
  70. // SHM Control
  71. {
  72. #ifdef CARLA_OS_WIN
  73. // TESTING!
  74. fShmControl.shm = carla_shm_attach_linux((const char*)fShmControl.filename);
  75. #else
  76. fShmControl.shm = carla_shm_attach((const char*)fShmControl.filename);
  77. #endif
  78. if (! carla_is_shm_valid(fShmControl.shm))
  79. {
  80. _cleanup();
  81. carla_stdout("Failed to open or create shared memory file #2");
  82. return false;
  83. }
  84. if (! carla_shm_map<BridgeShmControl>(fShmControl.shm, fShmControl.data))
  85. {
  86. _cleanup();
  87. carla_stdout("Failed to mmap shared memory file");
  88. return false;
  89. }
  90. }
  91. // Read values from memory
  92. CARLA_ASSERT(rdwr_readOpcode(&fShmControl.data->ringBuffer) == kPluginBridgeOpcodeSetBufferSize);
  93. fBufferSize = rdwr_readInt(&fShmControl.data->ringBuffer);
  94. carla_stderr("BufferSize: %i", fBufferSize);
  95. CARLA_ASSERT(rdwr_readOpcode(&fShmControl.data->ringBuffer) == kPluginBridgeOpcodeSetSampleRate);
  96. fSampleRate = rdwr_readFloat(&fShmControl.data->ringBuffer);
  97. carla_stderr("SampleRate: %f", fSampleRate);
  98. fQuitNow = false;
  99. fIsRunning = true;
  100. QThread::start(QThread::TimeCriticalPriority);
  101. CarlaEngine::init(clientName);
  102. return true;
  103. }
  104. bool close()
  105. {
  106. carla_debug("CarlaEnginePlugin::close()");
  107. CarlaEngine::close();
  108. fQuitNow = true;
  109. QThread::wait();
  110. _cleanup();
  111. return true;
  112. }
  113. bool isRunning() const
  114. {
  115. return fIsRunning;
  116. }
  117. bool isOffline() const
  118. {
  119. return false;
  120. }
  121. EngineType type() const
  122. {
  123. return kEngineTypeBridge;
  124. }
  125. // -------------------------------------
  126. // CarlaThread virtual calls
  127. void run()
  128. {
  129. // TODO - set RT permissions
  130. while (! fQuitNow)
  131. {
  132. if (! jackbridge_sem_timedwait(&fShmControl.data->runServer, 5))
  133. {
  134. if (errno != ETIMEDOUT)
  135. continue;
  136. fQuitNow = true;
  137. return;
  138. }
  139. while (rdwr_dataAvailable(&fShmControl.data->ringBuffer))
  140. {
  141. const PluginBridgeOpcode opcode(rdwr_readOpcode(&fShmControl.data->ringBuffer));
  142. switch (opcode)
  143. {
  144. case kPluginBridgeOpcodeNull:
  145. break;
  146. case kPluginBridgeOpcodeSetAudioPool:
  147. {
  148. const int poolSize(rdwr_readInt(&fShmControl.data->ringBuffer));
  149. fShmAudioPool.data = (float*)carla_shm_map(fShmAudioPool.shm, poolSize);
  150. fEventsInPort = nullptr;
  151. CarlaPlugin* const plugin(getPluginUnchecked(0));
  152. if (plugin != nullptr && plugin->enabled())
  153. fEventsInPort = plugin->getDefaultEventInPort();
  154. break;
  155. }
  156. case kPluginBridgeOpcodeSetBufferSize:
  157. {
  158. const int bufferSize(rdwr_readInt(&fShmControl.data->ringBuffer));
  159. bufferSizeChanged(bufferSize);
  160. break;
  161. }
  162. case kPluginBridgeOpcodeSetSampleRate:
  163. {
  164. const float sampleRate(rdwr_readFloat(&fShmControl.data->ringBuffer));
  165. sampleRateChanged(sampleRate);
  166. break;
  167. }
  168. case kPluginBridgeOpcodeSetParameter:
  169. {
  170. const int index(rdwr_readInt(&fShmControl.data->ringBuffer));
  171. const float value(rdwr_readFloat(&fShmControl.data->ringBuffer));
  172. CarlaPlugin* const plugin(getPluginUnchecked(0));
  173. if (plugin != nullptr && plugin->enabled())
  174. {
  175. plugin->setParameterValueByRealIndex(index, value, false, false, false);
  176. plugin->postponeRtEvent(kPluginPostRtEventParameterChange, index, 0, value);
  177. }
  178. break;
  179. }
  180. case kPluginBridgeOpcodeSetProgram:
  181. {
  182. const int index(rdwr_readInt(&fShmControl.data->ringBuffer));
  183. CarlaPlugin* const plugin(getPluginUnchecked(0));
  184. if (plugin != nullptr && plugin->enabled())
  185. {
  186. plugin->setProgram(index, false, false, false);
  187. plugin->postponeRtEvent(kPluginPostRtEventProgramChange, index, 0, 0.0f);
  188. }
  189. break;
  190. }
  191. case kPluginBridgeOpcodeSetMidiProgram:
  192. {
  193. const int index(rdwr_readInt(&fShmControl.data->ringBuffer));
  194. CarlaPlugin* const plugin(getPluginUnchecked(0));
  195. if (plugin != nullptr && plugin->enabled())
  196. {
  197. plugin->setMidiProgram(index, false, false, false);
  198. plugin->postponeRtEvent(kPluginPostRtEventMidiProgramChange, index, 0, 0.0f);
  199. }
  200. break;
  201. }
  202. case kPluginBridgeOpcodeMidiEvent:
  203. {
  204. uint8_t data[4] = { 0 };
  205. const long time(rdwr_readLong(&fShmControl.data->ringBuffer));
  206. const int dataSize(rdwr_readInt(&fShmControl.data->ringBuffer));
  207. CARLA_ASSERT_INT(dataSize >= 1 && dataSize <= 4, dataSize);
  208. for (int i=0; i < dataSize && i < 4; ++i)
  209. data[i] = rdwr_readChar(&fShmControl.data->ringBuffer);
  210. CARLA_ASSERT(fEventsInPort != nullptr);
  211. if (fEventsInPort != nullptr)
  212. fEventsInPort->writeMidiEvent(time, data, dataSize);
  213. break;
  214. }
  215. case kPluginBridgeOpcodeProcess:
  216. {
  217. CARLA_ASSERT(fShmAudioPool.data != nullptr);
  218. CarlaPlugin* const plugin(getPluginUnchecked(0));
  219. if (plugin != nullptr && plugin->enabled() && plugin->tryLock())
  220. {
  221. const uint32_t inCount(plugin->audioInCount());
  222. const uint32_t outCount(plugin->audioOutCount());
  223. float* inBuffer[inCount];
  224. float* outBuffer[outCount];
  225. for (uint32_t i=0; i < inCount; ++i)
  226. inBuffer[i] = fShmAudioPool.data + i*fBufferSize;
  227. for (uint32_t i=0; i < outCount; ++i)
  228. outBuffer[i] = fShmAudioPool.data + (i+inCount)*fBufferSize;
  229. plugin->initBuffers();
  230. plugin->process(inBuffer, outBuffer, fBufferSize);
  231. plugin->unlock();
  232. if (fEventsInPort != nullptr)
  233. fEventsInPort->clearBuffer();
  234. }
  235. break;
  236. }
  237. case kPluginBridgeOpcodeQuit:
  238. fQuitNow = true;
  239. break;
  240. }
  241. }
  242. if (jackbridge_sem_post(&fShmControl.data->runClient) != 0)
  243. pass(); //carla_stderr2("Could not post to semaphore");
  244. }
  245. fIsRunning = false;
  246. }
  247. private:
  248. struct BridgeAudioPool {
  249. CarlaString filename;
  250. float* data;
  251. shm_t shm;
  252. BridgeAudioPool()
  253. : data(nullptr)
  254. {
  255. carla_shm_init(shm);
  256. }
  257. } fShmAudioPool;
  258. struct BridgeControl {
  259. CarlaString filename;
  260. BridgeShmControl* data;
  261. shm_t shm;
  262. BridgeControl()
  263. : data(nullptr)
  264. {
  265. carla_shm_init(shm);
  266. }
  267. } fShmControl;
  268. bool fIsRunning;
  269. bool fQuitNow;
  270. CarlaEngineEventPort* fEventsInPort;
  271. void _cleanup()
  272. {
  273. if (fShmAudioPool.filename.isNotEmpty())
  274. fShmAudioPool.filename.clear();
  275. if (fShmControl.filename.isNotEmpty())
  276. fShmControl.filename.clear();
  277. fShmAudioPool.data = nullptr;
  278. fShmControl.data = nullptr;
  279. if (carla_is_shm_valid(fShmAudioPool.shm))
  280. carla_shm_close(fShmAudioPool.shm);
  281. if (carla_is_shm_valid(fShmControl.shm))
  282. carla_shm_close(fShmControl.shm);
  283. }
  284. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineBridge)
  285. };
  286. // -----------------------------------------
  287. CarlaEngine* CarlaEngine::newBridge(const char* const audioBaseName, const char* const controlBaseName)
  288. {
  289. return new CarlaEngineBridge(audioBaseName, controlBaseName);
  290. }
  291. CARLA_BACKEND_END_NAMESPACE
  292. #endif // BUILD_BRIDGE