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.

318 lines
8.8KB

  1. /*
  2. * Carla Plugin 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. #ifdef BUILD_BRIDGE
  18. #include "CarlaEngineInternal.hpp"
  19. #include "CarlaBackendUtils.hpp"
  20. #include "CarlaMIDI.h"
  21. #include "../CarlaBridge.hpp"
  22. #include "CarlaShmUtils.hpp"
  23. #include "jackbridge/jackbridge.h"
  24. #include <ctime>
  25. //#ifdef CARLA_OS_WIN
  26. //# include <sys/time.h>
  27. //#endif
  28. CARLA_BACKEND_START_NAMESPACE
  29. #if 0
  30. } // Fix editor indentation
  31. #endif
  32. // -----------------------------------------
  33. class CarlaEngineBridge : public CarlaEngine,
  34. public CarlaThread
  35. {
  36. public:
  37. CarlaEngineBridge(const char* const audioBaseName, const char* const controlBaseName)
  38. : CarlaEngine(),
  39. fIsRunning(false),
  40. fQuitNow(false)
  41. {
  42. carla_debug("CarlaEngineBridge::CarlaEngineBridge()");
  43. fShmAudioPool.filename = "/carla-bridge_shm_";
  44. fShmAudioPool.filename += audioBaseName;
  45. fShmControl.filename = "/carla-bridge_shc_";
  46. fShmControl.filename += controlBaseName;
  47. }
  48. ~CarlaEngineBridge()
  49. {
  50. carla_debug("CarlaEngineBridge::~CarlaEngineBridge()");
  51. }
  52. // -------------------------------------
  53. // CarlaEngine virtual calls
  54. bool init(const char* const clientName)
  55. {
  56. carla_debug("CarlaEngineBridge::init(\"%s\")", clientName);
  57. // SHM Audio Pool
  58. {
  59. #ifdef CARLA_OS_WIN
  60. // TESTING!
  61. fShmAudioPool.shm = carla_shm_attach_linux((const char*)fShmAudioPool.filename);
  62. #else
  63. fShmAudioPool.shm = carla_shm_attach((const char*)fShmAudioPool.filename);
  64. #endif
  65. if (! carla_is_shm_valid(fShmAudioPool.shm))
  66. {
  67. _cleanup();
  68. carla_stdout("Failed to open or create shared memory file #1");
  69. return false;
  70. }
  71. }
  72. // SHM Control
  73. {
  74. #ifdef CARLA_OS_WIN
  75. // TESTING!
  76. fShmControl.shm = carla_shm_attach_linux((const char*)fShmControl.filename);
  77. #else
  78. fShmControl.shm = carla_shm_attach((const char*)fShmControl.filename);
  79. #endif
  80. if (! carla_is_shm_valid(fShmControl.shm))
  81. {
  82. _cleanup();
  83. carla_stdout("Failed to open or create shared memory file #2");
  84. return false;
  85. }
  86. if (! carla_shm_map<BridgeShmControl>(fShmControl.shm, fShmControl.data))
  87. {
  88. _cleanup();
  89. carla_stdout("Failed to mmap shared memory file");
  90. return false;
  91. }
  92. }
  93. // Read values from memory
  94. CARLA_ASSERT(rdwr_readOpcode(&fShmControl.data->ringBuffer) == kPluginBridgeOpcodeBufferSize);
  95. fBufferSize = rdwr_readInt(&fShmControl.data->ringBuffer);
  96. carla_stderr("BufferSize: %i", fBufferSize);
  97. CARLA_ASSERT(rdwr_readOpcode(&fShmControl.data->ringBuffer) == kPluginBridgeOpcodeSampleRate);
  98. fSampleRate = rdwr_readFloat(&fShmControl.data->ringBuffer);
  99. carla_stderr("SampleRate: %f", fSampleRate);
  100. fQuitNow = false;
  101. fIsRunning = true;
  102. CarlaThread::start();
  103. CarlaEngine::init(clientName);
  104. return true;
  105. }
  106. bool close()
  107. {
  108. carla_debug("CarlaEnginePlugin::close()");
  109. CarlaEngine::close();
  110. fQuitNow = true;
  111. CarlaThread::stop();
  112. _cleanup();
  113. return true;
  114. }
  115. bool isRunning() const
  116. {
  117. return fIsRunning;
  118. }
  119. bool isOffline() const
  120. {
  121. return false;
  122. }
  123. EngineType type() const
  124. {
  125. return kEngineTypeBridge;
  126. }
  127. // -------------------------------------
  128. // CarlaThread virtual calls
  129. void run()
  130. {
  131. // TODO - set RT permissions
  132. const int timeout = 50;
  133. while (! fQuitNow)
  134. {
  135. carla_debug("RUN 001");
  136. timespec ts_timeout;
  137. #if 0//def CARLA_OS_WIN
  138. timeval now;
  139. gettimeofday(&now, nullptr);
  140. ts_timeout.tv_sec = now.tv_sec;
  141. ts_timeout.tv_nsec = now.tv_usec * 1000;
  142. #else
  143. linux_clock_gettime_rt(&ts_timeout);
  144. #endif
  145. time_t seconds = timeout / 1000;
  146. ts_timeout.tv_sec += seconds;
  147. ts_timeout.tv_nsec += (timeout - seconds * 1000) * 1000000;
  148. if (ts_timeout.tv_nsec >= 1000000000) {
  149. ts_timeout.tv_nsec -= 1000000000;
  150. ts_timeout.tv_sec++;
  151. }
  152. if (linux_sem_timedwait(&fShmControl.data->runServer, &ts_timeout))
  153. {
  154. if (errno == ETIMEDOUT)
  155. {
  156. continue;
  157. }
  158. else
  159. {
  160. fQuitNow = true;
  161. break;
  162. }
  163. }
  164. carla_debug("RUN 004");
  165. while (rdwr_dataAvailable(&fShmControl.data->ringBuffer))
  166. {
  167. carla_debug("RUN 005");
  168. const PluginBridgeOpcode opcode = rdwr_readOpcode(&fShmControl.data->ringBuffer);
  169. switch (opcode)
  170. {
  171. case kPluginBridgeOpcodeNull:
  172. break;
  173. case kPluginBridgeOpcodeReadyWait:
  174. fShmAudioPool.data = (float*)carla_shm_map(fShmAudioPool.shm, rdwr_readInt(&fShmControl.data->ringBuffer));
  175. break;
  176. case kPluginBridgeOpcodeBufferSize:
  177. bufferSizeChanged(rdwr_readInt(&fShmControl.data->ringBuffer));
  178. break;
  179. case kPluginBridgeOpcodeSampleRate:
  180. sampleRateChanged(rdwr_readFloat(&fShmControl.data->ringBuffer));
  181. break;
  182. case kPluginBridgeOpcodeProcess:
  183. {
  184. CARLA_ASSERT(fShmAudioPool.data != nullptr);
  185. CarlaPlugin* const plugin(getPluginUnchecked(0));
  186. carla_debug("RUN 006");
  187. if (plugin != nullptr && plugin->enabled() && plugin->tryLock())
  188. {
  189. const uint32_t inCount = plugin->audioInCount();
  190. const uint32_t outCount = plugin->audioOutCount();
  191. float* inBuffer[inCount];
  192. float* outBuffer[outCount];
  193. for (uint32_t i=0; i < inCount; i++)
  194. inBuffer[i] = fShmAudioPool.data + i*fBufferSize;
  195. for (uint32_t i=0; i < outCount; i++)
  196. outBuffer[i] = fShmAudioPool.data + (i+inCount)*fBufferSize;
  197. plugin->initBuffers();
  198. plugin->setActive(true, false, false);
  199. plugin->process(inBuffer, outBuffer, fBufferSize);
  200. plugin->unlock();
  201. }
  202. break;
  203. }
  204. }
  205. }
  206. if (linux_sem_post(&fShmControl.data->runClient) != 0)
  207. carla_stderr2("Could not post to semaphore");
  208. }
  209. fIsRunning = false;
  210. }
  211. private:
  212. struct BridgeAudioPool {
  213. CarlaString filename;
  214. float* data;
  215. shm_t shm;
  216. BridgeAudioPool()
  217. : data(nullptr)
  218. {
  219. carla_shm_init(shm);
  220. }
  221. } fShmAudioPool;
  222. struct BridgeControl {
  223. CarlaString filename;
  224. BridgeShmControl* data;
  225. shm_t shm;
  226. BridgeControl()
  227. : data(nullptr)
  228. {
  229. carla_shm_init(shm);
  230. }
  231. } fShmControl;
  232. bool fIsRunning;
  233. bool fQuitNow;
  234. void _cleanup()
  235. {
  236. if (fShmAudioPool.filename.isNotEmpty())
  237. fShmAudioPool.filename.clear();
  238. if (fShmControl.filename.isNotEmpty())
  239. fShmControl.filename.clear();
  240. fShmAudioPool.data = nullptr;
  241. fShmControl.data = nullptr;
  242. if (carla_is_shm_valid(fShmAudioPool.shm))
  243. carla_shm_close(fShmAudioPool.shm);
  244. if (carla_is_shm_valid(fShmControl.shm))
  245. carla_shm_close(fShmControl.shm);
  246. }
  247. CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineBridge)
  248. };
  249. // -----------------------------------------
  250. CarlaEngine* CarlaEngine::newBridge(const char* const audioBaseName, const char* const controlBaseName)
  251. {
  252. return new CarlaEngineBridge(audioBaseName, controlBaseName);
  253. }
  254. CARLA_BACKEND_END_NAMESPACE
  255. #endif // BUILD_BRIDGE