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.

247 lines
8.1KB

  1. /*
  2. * DISTRHO Plugin Framework (DPF)
  3. * Copyright (C) 2012-2025 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * Permission to use, copy, modify, and/or distribute this software for any purpose with
  6. * or without fee is hereby granted, provided that the above copyright notice and this
  7. * permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
  10. * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
  11. * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  12. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
  13. * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  14. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #ifndef DISTRHO_PLUGIN_UTILS_HPP_INCLUDED
  17. #define DISTRHO_PLUGIN_UTILS_HPP_INCLUDED
  18. #include "DistrhoPlugin.hpp"
  19. START_NAMESPACE_DISTRHO
  20. /* ------------------------------------------------------------------------------------------------------------
  21. * Plugin related utilities */
  22. /**
  23. @defgroup PluginRelatedUtilities Plugin related utilities
  24. @{
  25. */
  26. /**
  27. Get the absolute filename of the plugin DSP/UI binary.@n
  28. Under certain systems or plugin formats the binary will be inside the plugin bundle.@n
  29. Also, in some formats or setups, the DSP and UI binaries are in different files.
  30. */
  31. const char* getBinaryFilename();
  32. /**
  33. Get a string representation of the current plugin format we are building against.@n
  34. This can be "AudioUnit", "JACK/Standalone", "LADSPA", "DSSI", "LV2", "VST2", "VST3" or "CLAP".@n
  35. This string is purely informational and must not be used to tweak plugin behaviour.
  36. @note DO NOT CHANGE PLUGIN BEHAVIOUR BASED ON PLUGIN FORMAT.
  37. */
  38. const char* getPluginFormatName() noexcept;
  39. /**
  40. List of supported OS-specific directories to be used for getSpecialDir.
  41. */
  42. enum SpecialDir {
  43. /** The user "home" directory */
  44. kSpecialDirHome,
  45. /** Directory intended to store persistent configuration data (in general) */
  46. kSpecialDirConfig,
  47. /** Directory intended to store persistent configuration data for the current plugin */
  48. kSpecialDirConfigForPlugin,
  49. /** Directory intended to store "documents" (in general) */
  50. kSpecialDirDocuments,
  51. /** Directory intended to store "documents" for the current plugin */
  52. kSpecialDirDocumentsForPlugin,
  53. };
  54. /**
  55. Get an OS-specific directory.@n
  56. Calling this function will ensure the dictory exists on the filesystem.@n
  57. The returned path always includes a final OS separator.
  58. */
  59. const char* getSpecialDir(SpecialDir dir);
  60. /**
  61. Get the path to where resources are stored within the plugin bundle.@n
  62. Requires a valid plugin bundle path.
  63. Returns a path inside the bundle where the plugin is meant to store its resources in.@n
  64. This path varies between systems and plugin formats, like so:
  65. - AU: <bundle>/Contents/Resources
  66. - CLAP+VST2 macOS: <bundle>/Contents/Resources
  67. - CLAP+VST2 non-macOS: <bundle>/resources (see note)
  68. - LV2: <bundle>/resources (can be stored anywhere inside the bundle really, DPF just uses this one)
  69. - VST3: <bundle>/Contents/Resources
  70. The other non-mentioned formats do not support bundles.@n
  71. @note For CLAP and VST2 on non-macOS systems, this assumes you have your plugin inside a dedicated directory
  72. rather than only shipping with the binary, like so:
  73. @code
  74. + myplugin.vst/
  75. - myplugin.dll
  76. + resources/
  77. - image1.png
  78. @endcode
  79. */
  80. const char* getResourcePath(const char* bundlePath) noexcept;
  81. /** @} */
  82. /* ------------------------------------------------------------------------------------------------------------
  83. * Plugin helper classes */
  84. /**
  85. @defgroup PluginHelperClasses Plugin helper classes
  86. @{
  87. */
  88. #if DISTRHO_PLUGIN_NUM_OUTPUTS > 0
  89. /**
  90. Handy class to help keep audio buffer in sync with incoming MIDI events.
  91. To use it, create a local variable (on the stack) and call nextEvent() until it returns false.
  92. @code
  93. for (AudioMidiSyncHelper amsh(outputs, frames, midiEvents, midiEventCount); amsh.nextEvent();)
  94. {
  95. float* const outL = amsh.outputs[0];
  96. float* const outR = amsh.outputs[1];
  97. for (uint32_t i=0; i<amsh.midiEventCount; ++i)
  98. {
  99. const MidiEvent& ev(amsh.midiEvents[i]);
  100. // ... do something with the midi event
  101. }
  102. renderSynth(outL, outR, amsh.frames);
  103. }
  104. @endcode
  105. Some important notes when using this class:
  106. 1. MidiEvent::frame retains its original value, but it is useless, do not use it.
  107. 2. The class variable names are the same as the default ones in the run function.
  108. Keep that in mind and try to avoid typos. :)
  109. */
  110. struct AudioMidiSyncHelper
  111. {
  112. /** Parameters from the run function, adjusted for event sync */
  113. float* outputs[DISTRHO_PLUGIN_NUM_OUTPUTS];
  114. uint32_t frames;
  115. const MidiEvent* midiEvents;
  116. uint32_t midiEventCount;
  117. /**
  118. Constructor, using values from the run function.
  119. */
  120. AudioMidiSyncHelper(float** const o, uint32_t f, const MidiEvent* m, uint32_t mc)
  121. : outputs(),
  122. frames(0),
  123. midiEvents(m),
  124. midiEventCount(0),
  125. remainingFrames(f),
  126. remainingMidiEventCount(mc),
  127. totalFramesUsed(0)
  128. {
  129. for (uint i=0; i<DISTRHO_PLUGIN_NUM_OUTPUTS; ++i)
  130. outputs[i] = o[i];
  131. }
  132. /**
  133. Process a batch of events untill no more are available.
  134. You must not read any more values from this class after this function returns false.
  135. */
  136. bool nextEvent()
  137. {
  138. // nothing else to do
  139. if (remainingFrames == 0)
  140. return false;
  141. // initial setup, need to find first MIDI event
  142. if (totalFramesUsed == 0)
  143. {
  144. // no MIDI events at all in this process cycle
  145. if (remainingMidiEventCount == 0)
  146. {
  147. frames = remainingFrames;
  148. remainingFrames = 0;
  149. totalFramesUsed += frames;
  150. return true;
  151. }
  152. // render audio until first midi event, if needed
  153. if (const uint32_t firstEventFrame = midiEvents[0].frame)
  154. {
  155. DISTRHO_SAFE_ASSERT_UINT2_RETURN(firstEventFrame < remainingFrames,
  156. firstEventFrame, remainingFrames, false);
  157. frames = firstEventFrame;
  158. remainingFrames -= firstEventFrame;
  159. totalFramesUsed += firstEventFrame;
  160. return true;
  161. }
  162. }
  163. else
  164. {
  165. for (uint32_t i=0; i<DISTRHO_PLUGIN_NUM_OUTPUTS; ++i)
  166. outputs[i] += frames;
  167. }
  168. // no more MIDI events available
  169. if (remainingMidiEventCount == 0)
  170. {
  171. frames = remainingFrames;
  172. midiEvents = nullptr;
  173. midiEventCount = 0;
  174. remainingFrames = 0;
  175. totalFramesUsed += frames;
  176. return true;
  177. }
  178. // if there were midi events before, increment pointer
  179. if (midiEventCount != 0)
  180. midiEvents += midiEventCount;
  181. const uint32_t firstEventFrame = midiEvents[0].frame;
  182. DISTRHO_SAFE_ASSERT_UINT2_RETURN(firstEventFrame >= totalFramesUsed,
  183. firstEventFrame, totalFramesUsed, false);
  184. midiEventCount = 1;
  185. while (midiEventCount < remainingMidiEventCount)
  186. {
  187. if (midiEvents[midiEventCount].frame == firstEventFrame)
  188. ++midiEventCount;
  189. else
  190. break;
  191. }
  192. frames = firstEventFrame - totalFramesUsed;
  193. remainingFrames -= frames;
  194. remainingMidiEventCount -= midiEventCount;
  195. totalFramesUsed += frames;
  196. return true;
  197. }
  198. private:
  199. /** @internal */
  200. uint32_t remainingFrames;
  201. uint32_t remainingMidiEventCount;
  202. uint32_t totalFramesUsed;
  203. };
  204. #endif
  205. /** @} */
  206. // -----------------------------------------------------------------------------------------------------------
  207. END_NAMESPACE_DISTRHO
  208. #endif // DISTRHO_PLUGIN_UTILS_HPP_INCLUDED