The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
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.

340 lines
21KB

  1. The Juce Polymorphic Plugin Project!
  2. ====================================
  3. (c) 2015 by ROLI, visit www.juce.com for more info.
  4. -----------------------------------------------------------------------------------------------------
  5. The purpose of this framework is to make is simple to write an audio plugin in a generic
  6. way, which can then be compiled as a VST, AudioUnit, RTAS, or any combination of these.
  7. It's "polymorphic" because the output is a single binary module that acts as all of the
  8. different plugin formats at the same time. This means that you only need to maintain one
  9. project, and only need to perform one build to create a multi-format plugin.
  10. Also included are some helper classes that make it easy to create a stand-alone app to
  11. run your plugin without a host. This might be useful in its own right, but can also be very
  12. handy when developing your plugin, because you can build, test and debug it without needing
  13. to keep restarting a 3rd-party host.
  14. How does it work?
  15. =================
  16. To create your plugin, you just create a subclass of the AudioPluginInstance class to
  17. perform the processing. And your plugin UI is written like any normal Juce UI component.
  18. All the platform-specific code is hidden away in wrapper classes that you just add to
  19. your project - you should (hopefully) never need to even see the inner workings of these.
  20. Licensing issues
  21. ================
  22. Juce is released under the GPL (Gnu Public License) - this means that you're free to use
  23. and redistribute it as long as your products are also released under the GPL. Basically
  24. this means that if you use it, you also have to give away your source code.
  25. If you want to release a closed-source application, you can buy a commercial license
  26. that lets you avoid this restriction - see http://www.juce.com for more info,
  27. or see the comments at the top of all the Juce source files.
  28. If you're building the VST projects or releasing a VST, you'll need have a look at Steinberg's
  29. developer site to see what licensing rules apply these days. Their website's at
  30. http://www.steinberg.net
  31. If you're building an RTAS then you'll need to sign Digidesign's developer license to get
  32. their SDK. Visit http://www.digidesign.com for more info.
  33. Getting Started
  34. ===============
  35. The best advice for creating a plugin is to USE THE INTROJUCER! This will auto-generate a
  36. cross-platform project for you with all the correct settings, and some framework classes to
  37. get you started.
  38. If you choose to go it alone and not use the introjucer, then you'll have a lot of painful
  39. project set-up to do, and will need to scour this document and the juce forum for the details.
  40. There's a demo plugin in the juce/extras folder - this contains an example plugin which can
  41. be built in all the different formats.
  42. Have a look at the demo classes to see how it works, and then to create a real plugin,
  43. you'll need to replace the demo files with your own code.
  44. I've tried to add helpful comments where you might run across common compile errors, to
  45. help describe what you might be doing wrong, as getting a build set-up for some of these
  46. formats can be a bit of a pain. Please let me know if you find there's anything missing
  47. from these instructions or anything I could change to help smooth the build process along
  48. a bit.
  49. -----------------------------------------------------------------------------------------------------
  50. Prerequisites for building a VST
  51. ================================
  52. - Visit http://www.steinberg.net and jump through whatever hoops are necessary to download
  53. and install the VST SDK.
  54. -----------------------------------------------------------------------------------------------------
  55. Prerequisites for building an RTAS
  56. ==================================
  57. - Contact Digidesign, ask to become a Digidesign Development Partner, sign the relevent
  58. agreements and NDAs.
  59. - From the Digidesign website, download their latest Plug-In SDK
  60. - Install the SDK and build some of the demo plugins to make sure it all works.
  61. - In Visual Studio: Using the Digidesign demo projects in the SDK, make sure you've compiled
  62. debug and release versions of the following static libraries:
  63. DAE.lib, DigiExt.lib, DSI.lib, PlugInLib.lib.
  64. - In XCode: After installing the Digidesign SDK, make sure you've run the config_SDK_for_Mac
  65. command in the SDK's root directory. This sets up some of the tools that it needs.
  66. -----------------------------------------------------------------------------------------------------
  67. Choosing the formats to build
  68. =============================
  69. Each plugin project needs to contain an AppConfig.h file, which holds all the plugin-specific build details.
  70. (The introjucer will auto-generate this file for you when you create a project). In here, there are three macros
  71. that you can set to enable each of the available formats:
  72. #define JucePlugin_Build_VST 1
  73. #define JucePlugin_Build_RTAS 1
  74. #define JucePlugin_Build_AU 1
  75. The introjucer contains options in its config panel that let you set these values automatically without
  76. needing to manually change any files.
  77. -----------------------------------------------------------------------------------------------------
  78. Manually Creating a PC VST/RTAS plugin in Visual Studio
  79. =======================================================
  80. NOTE! That the following tips are provided only for the case where you're not using the introjucer..
  81. If you're sensible, you'll let the introjucer do all the heavy lifting, and won't need to read this.
  82. - First try loading the VST demo project in JuceAudioPlugin/demo/build. Hopefully this
  83. should build correctly.
  84. - Create a new, empty, win32 project using Visual Studio. Choose "DLL" as the type of
  85. product to build
  86. - If building an RTAS, add to your project all the juce_RTAS_*.cpp files from the wrapper/RTAS folder.
  87. - If building a VST, add to your project all the juce_VST_*.cpp files from the wrapper/VST folder.
  88. - Create yourself a JucePluginCharacteristics.h file, starting with a copy of the one in the
  89. demo project. Go through each item inside it carefully, and set them to the appropriate value
  90. for your plugin.
  91. - Under "Additional Include Directories", add the folder in which you're going to put
  92. your JucePluginCharacteristics.h file.
  93. - If you're doing an RTAS, change these project settings (these can all be ignored if you're only doing a VST):
  94. - Make sure all of these are on your include path:
  95. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\EffectClasses
  96. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\ProcessClasses
  97. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\ProcessClasses\Interfaces
  98. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\Utilities
  99. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\RTASP_Adapt
  100. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\CoreClasses
  101. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\Controls
  102. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\Meters
  103. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\ViewClasses
  104. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\DSPClasses
  105. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\PluginLibrary\Interfaces
  106. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\common
  107. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\common\Platform
  108. c:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugins\SignalProcessing\Public
  109. C:\yourdirectory\PT_80_SDK\AlturaPorts\TDMPlugIns\DSPManager\Interfaces
  110. c:\yourdirectory\PT_80_SDK\AlturaPorts\SADriver\Interfaces
  111. c:\yourdirectory\PT_80_SDK\AlturaPorts\DigiPublic\Interfaces
  112. c:\yourdirectory\PT_80_SDK\AlturaPorts\Fic\Interfaces\DAEClient
  113. c:\yourdirectory\PT_80_SDK\AlturaPorts\NewFileLibs\Cmn
  114. c:\yourdirectory\PT_80_SDK\AlturaPorts\NewFileLibs\DOA
  115. c:\yourdirectory\PT_80_SDK\AlturaPorts\AlturaSource\PPC_H
  116. c:\yourdirectory\PT_80_SDK\AlturaPorts\AlturaSource\AppSupport
  117. c:\yourdirectory\PT_80_SDK\AvidCode\AVX2sdk\AVX\avx2\avx2sdk\inc
  118. C:\yourdirectory\PT_80_SDK\xplat\AVX\avx2\avx2sdk\inc
  119. - Set "C++/Code Generation/Runtime Library" to be "Multi-threaded DLL" or "Multi-threaded Debug DLL"
  120. - Set the "Linker/Input/Module Definition file" to point to "wrapper/RTAS/juce_RTAS_WinExports.def"
  121. - Under "Linker/Input/Delay loaded DLLs", add the following:
  122. "DAE.dll; DigiExt.dll; DSI.dll; PluginLib.dll; DSPManager.dll"
  123. - You may (or may not) need to add "libcmtd.lib; libcmt.lib" to the "Linker/Input/Ignore Specific Library" setting.
  124. - For ONLY the following files:
  125. juce_RTAS_Wrapper.cpp, juce_RTAS_DigiCode1.cpp, juce_RTAS_DigiCode2.cpp, juce_RTAS_DigiCode3.cpp,
  126. change their "C++/Advanced/Calling Convention" property to "__stdcall". All other files should
  127. be left with the default calling convention of "__cdecl"
  128. - Set the "Linker/General/Output File" property to "$(OutDir)\$(ProjectName).dpm" (If you're building
  129. a polymorphic VST/RTAS, then you can simply copy or rename the finished .dpm file to a .dll, and
  130. it'll function as a VST)
  131. - Under "Custom build step", add the following command:
  132. copy /Y "\yourdirectory\juce\extras\audio plugins\wrapper\RTAS\juce_RTAS_WinResources.rsr" "$(TargetPath)".rsr
  133. The corresponding "Outputs" setting for this must be set to "$(TargetPath)".rsr
  134. (This will copy and rename the juce_RTAS_WinResources.rsr file to sit next to the finished .dpm file. It's
  135. a dummy resource file, but PT will refuse to load the plugin unless it has a corresponding .rsr file)
  136. - Because the RTAS code duplicates some win32 constants, you might need to force it to link correctly
  137. by adding "/FORCE:multiple" to the linker's additional command line options.
  138. - You might want to change the output directory to "\Program Files\Common Files\Digidesign\DAE\Plug-Ins\"
  139. if you want the built plugin to go directly into the PT plugins folder
  140. - When setting properties, remember to change them for both your debug and release builds!
  141. - Create your actual plugin classes and add them to the project. Obviously this is the hard bit!
  142. - Add the amalgamated juce source file to the project - have a look at the demo app for neat ways of doing this.
  143. - NOTE: on Windows, because RTAS uses the altura mac-style code, there are annoying clashes caused if
  144. you also include the Apple QuickTime headers, so you might need to turn off quicktime by setting the
  145. juce config macro: #define JUCE_QUICKTIME 0
  146. - NOTE: If you're using MSVC2005 to build your plugin, the users will need to
  147. have the Microsoft VC8 Runtime installed on their machines, otherwise the DLL will
  148. silently fail to load. You should probably add the runtime to your plugin's installer,
  149. and you can get a copy of it here:
  150. http://www.microsoft.com/downloads/details.aspx?FamilyID=32bc1bee-a3f9-4c13-9c99-220b62a191ee&DisplayLang=en
  151. -----------------------------------------------------------------------------------------------------
  152. Creating a Mac AU/VST/RTAS plugin in XCode
  153. ==========================================
  154. NOTE! That the following tips are provided only for the case where you're not using the introjucer..
  155. If you're sensible, you'll let the introjucer do all the heavy lifting, and won't need to read this.
  156. - Some of the RTAS SDK files in the demo project might be highlighted red to indicate that they're
  157. missing - if you're not bulding an RTAS, then you can just delete them from the project. If you are
  158. building an RTAS and have the SDK, you might need to update their paths to match your SDK location.
  159. - For an AU, there are a bunch of Apple cpp files that need to be included directly in your project - these
  160. used to be found in /Developer/Examples/CoreAudio/ but in the 10.6 SDK, Apple have moved them to a slightly
  161. more sensible location at /Developer/Extras/CoreAudio. The Juce demo is hardwired to refer to their new
  162. 10.6 location, so if you haven't yet installed the 10.6 SDK, the easiest way to make it all happy is just
  163. to duplicate your /Developer/Examples/CoreAudio folder as /Developer/Extras/CoreAudio. That way old and new
  164. juce plugin projects will all compile without changing any project paths.
  165. - For an AU, make sure that the JucePlugin_Build_AU is enabled in your JucePluginCharacteristics.h
  166. - In XCode, create a new project based on the "Audio Unit Effect" template
  167. - XCode will create a bunch of template source files for you - you can remove all of these from the project
  168. and delete them
  169. - In the target settings, clear the "Exported Symbols File" setting. The exports are specified by directives
  170. within the wrapper code, so don't need to be listed explicitly.
  171. - All all the Apple frameworks that Juce normally requires to the "External Frameworks" list
  172. - Add all the juce_AU_* files from the /wrapper/AU directory to your project
  173. - The template project creates an AUPublic group that contains lots of AudioUnit source files. But
  174. it leaves out files that it thinks you might not need, e.g. if you chose an "Audio Unit Effect" project,
  175. then it won't add the classes for handling MIDI. So you'll probably need to go into this folder
  176. and check that it contains AUMIDIBase.cpp, AUMidiEffectBase.cpp, MusicDeviceBase.cpp, etc
  177. - As for the PC, you'll need to make sure your project contains a correctly set-up JucePluginCharacteristics.h
  178. file - start with a copy of the one in the demo plugin project, and go through it making sure that
  179. all the values make sense for your plugin.
  180. - The JucePluginCharacteristics.h file is included not only by the code, but also by the resources files - so it
  181. needs to be locatable on both your normal header search path, and also on your resource include path, which is
  182. the project setting called 'Rez Search Paths'
  183. - Because of the flat naming structure used by Objective-C, if a host loads several different plugins which
  184. all contain slightly different versions of the juce library, you can get nasty situations where all their obj-C
  185. classes are cross-linked to the similarly-named class in other modules, and everything turns into a big mess...
  186. To avoid this, you're advised to set a unique JUCE_ObjCExtraSuffix value (you'll find this in juce_mac_NativeCode.mm,
  187. or if you're using the amalgamated version, you can just set it before including juce_amalgamated.cpp). Choose a
  188. suffix that's unique to both the name and version of your plugin.
  189. - Create your actual plugin classes and add them to the project. Obviously this is the hard bit!
  190. You should now be able to build a functional AU! If you want VST support as well, then read on...
  191. - Make sure that the JucePlugin_Build_VST is enabled in your JucePluginCharacteristics.h
  192. - For VST support, add all the juce_VST_* files from /wrapper/VST
  193. - In your target info settings, add the vstsdk2_4 folder to your "Header Search Paths" list
  194. - Make sure that in your Info.plist, the "Bundle Name" value is correctly set to the name of your plugin.
  195. Now, if you compile, the resulting bundle should work as both a VST and AU - you can simply copy or rename it,
  196. changing the suffix to ".vst", and put it in your VST folder.
  197. NOTE! In order to copy and rename the plugin to the various different plugin folders, the demo plugin project
  198. contains a VERY useful script, which you'll almost certainly want to copy into your own projects - have
  199. a look in the demo target for the build phase called "Copy into the different plugin folders", and use the
  200. inspector to have a look at the script. It uses advanced cunningness to copy only to the correct target
  201. folders for the types of plugin you're targeting, and should be able to be used in your own project
  202. without needing to edit it.
  203. NOTE! If you use the Finder to rename your xyz.component file to xyz.vst, it might look like it's done
  204. exactly this... but in fact, the Finder may have secretly renamed it as "xyz.vst.component", even though
  205. it shows "xyz.vst" as the name on the screen. I have wasted quite a lot of time angrily wondering why my VSTs
  206. don't seem to work, before realising that this is what has happened. You should use the command-line to rename
  207. it correctly.
  208. If you also want to build an RTAS, then carry on reading...
  209. - Make sure that the JucePlugin_Build_RTAS is enabled in your JucePluginCharacteristics.h
  210. - After installing the Digidesign SDK, make sure you've run the config_SDK_for_Mac command in
  211. its root directory. This sets up some of the tools that it needs.
  212. - Add the files from /wrapper/RTAS to your project. Obviously a couple of these are for Windows, so
  213. you shouldn't add those
  214. - In the Digi SDK, in /AlturaPorts/TDMPlugins/common/mac, there are two config files:
  215. CommonDebugSettings.xconfig and CommonReleaseSettings.xconfig
  216. These contain lots of Digi hackery to get their stuff to compile, so you should add them to your project
  217. and change your project's settings to use these files as their base config. Even so, it's all a bit of a mess,
  218. and you may need to tweak them a bit to get it to work on your system.
  219. - In your target settings, add a custom build setting called "MacBag", and set this to the path where the
  220. "MacBag" folder of the Digi SDK lives.
  221. - Add the following to your "Header Search Paths" setting (it's easiest to copy-and-paste this setting from
  222. the demo project):
  223. "$(MacBag)/../AlturaPorts/TDMPlugIns/PlugInLibrary/**"
  224. "$(MacBag)/../AlturaPorts/TDMPlugIns/DSPManager/**"
  225. "$(MacBag)/../AlturaPorts/TDMPlugIns/SupplementalPlugInLib/Encryption"
  226. "$(MacBag)/../AlturaPorts/TDMPlugIns/SupplementalPlugInLib/GraphicsExtensions"
  227. "$(MacBag)/../AlturaPorts/TDMPlugIns/common"
  228. "$(MacBag)/../AlturaPorts/TDMPlugIns/common/PI_LibInterface"
  229. "$(MacBag)/../AlturaPorts/TDMPlugIns/PACEProtection/**"
  230. "$(MacBag)/../AlturaPorts/OMS/Headers"
  231. "$(MacBag)/../AlturaPorts/Fic/Interfaces/**"
  232. "$(MacBag)/../AlturaPorts/Fic/Source/SignalNets"
  233. "$(MacBag)/../AlturaPorts/DSIPublicInterface/PublicHeaders"
  234. "$(MacBag)/../DAEWin/Include"
  235. "$(MacBag)/../AlturaPorts/DigiPublic/Interfaces"
  236. "$(MacBag)/../AlturaPorts/DigiPublic"
  237. "$(MacBag)/../AlturaPorts/NewFileLibs/DOA"
  238. "$(MacBag)/../AlturaPorts/NewFileLibs/Cmn"
  239. "$(MacBag)/../AlturaPorts/TDMPlugIns/SignalProcessing/**"
  240. "$(MacBag)/../AvidCode/AVX2sdk/AVX/avx2/avx2sdk/inc"
  241. "$(MacBag)/../AvidCode/AVX2sdk/AVX/avx2/avx2sdk/utils"
  242. - If you're using the Digi files CommonDebugSettings.xcconfig and CommonReleaseSettings.xcconfig,
  243. then you'll probably have to remove the "-x c++" option from their OTHER_CFLAGS setting, because
  244. that prevents it compiling obj-C. Also, you might need to comment-out the GCC_PREFIX_HEADER setting,
  245. unless you can persuade precompiled headers to work (I've never managed to get them working myself..)
  246. You'll also probably want to add a "MacBag" setting to these files, rather than putting it into
  247. your project - e.g. "MacBag = /Users/jules/SDKs/PT_80_SDK/MacBag"
  248. - If you get include errors compiling some of the DigiDesign code, you may need to
  249. add "/Developer/Headers/FlatCarbon" to your header search path.
  250. - In the SDK, find the PluginLibrary.xcodeproj file, and add this to your "External frameworks and Libraries".
  251. If you've already compiled this library, you can open its item in your XCode project treeview, to find
  252. the "libPluginLibrary.a" item inside it. Drag this subitem down to your Target/"Link Binary With Libraries"
  253. build stage and drop it there to add it to the link process.
  254. - In your Info.plist, change the "Bundle OS Type Code" to "TDMw", and the "Bundle Creator OS Type Code" to
  255. "PTul".
  256. - You may need to remove the "OTHER_CFLAGS = -x c++" from the RTAS settings file to stop it complaining about
  257. obj-C code
  258. You should now be able to build an RTAS! Again, just renaming the finished bundle to ".dpm" and
  259. putting it in your RTAS folder should be do the trick.
  260. If you get any weird build problems, a good tip is to try comparing the demo plugin's build settings with your
  261. own - this should usually show up what's missing.
  262. Note about exported symbols:
  263. When XCode builds the plugin, I've had unpredictable results when trying to stop it from exporting
  264. all of the internal functions as public symbols. There are some flags that are supposed to turn this
  265. off, but sometimes they don't seem to have any effect, and using an explicit exports file also
  266. seems a bit hit-and-miss. (If anyone knows better and can get this working, please let me know!)
  267. Anyway, as well as being wasteful and showing everyone what's inside your plugin, leaving all
  268. the symbols in there will cause fatal crashes when used with Tracktion, or alongside any other
  269. Juce-based plugins. A way of making sure your plugin is stripped is to use the command
  270. "strip -x -S YourPlugin.vst/Contents/MacOS/YourPlugin" after bulding it, which removes the
  271. unnecessary symbols (although in my experience this also doesn't seem to work all the time,
  272. so it's a good idea to check it using the unix "nm" command).