This repo is the result of my research into the Juce Framework and is a good starting point to create a synthesiser app and/or plugin. It contains some good, reusable generic classes that can be used to build any kind of midi synthesiser with effects.
This code depends on the Juce Library to run, which has its own open source and commercial dual/triple licensing. Please see the LICENSE-NOTE.md file for more details. License information for the unique code in this repo can be found at LICENSE.md.
This synthesiser lets you put as many as 6 effects on the signal, in any order you like, in any number you like. This means you could place 6 distortions in-line or 6 reverbs if you really wanted to.... However - you probably DO NOT want to, as feedback, frequency spikes and general chaos will be the result. It's pretty fun to play with, but it gets out of hand quickly - especially if your effect gain dials aren't set low enough.
- So... before loading up on more than one effect of the same type, MAKE SURE TO TURN DOWN YOUR VOLUME!
The filter resonance can also be especially loud, even at low settings.
So in case you missed it, I'll say it again: TURN DOWN YOUR VOLUME!
You've been warned. I'm not liable for any damage to your speakers, ears, sanity, etc.
Cool stuff in here:
- A wavetable synthesiser implementation with waveform sweeping from sine to triangle to square, and any combination between each. (You can sweep continuously between sine and triangle and between triangle and square.)
- An effects section featuring six effect slots with variable gain that can be put in any order you like. (You might have heard about this above, in the Warning section...) The effects use juce::dsp modules, and right now have most of their parameters hard-coded except for the level. The available effects are:
- Distortion - mild wave distortion taken from an overdriven filter circuit
- Delay - ~400ms delay. Feedback (level of each repeat) is controlled by the effect level control.
- Chorus - Wet / dry mix is controlled by the effect level (1.0 = full wet)
- Reverb - Wet / dry mix is controlled by the effect level control.
- A low-pass filter with configurable cutoff frequency and resonance. (Pretty standard stuff, but please note that the filter goes into steep resonance pretty early on. The default resonance is 0.)
- (Have I mentioned that it's a good idea to turn down the volume before testing this synth??)
The project is a good starting point for any type of synthesiser project, and can serve as an example for how to work with Juce modules in an object-oriented way. It also serves as a proving ground for my method of creating a flexible effects chain that is dynamically malleable at runtime. Essential to the design is the choice to develop against a single fixed precision for floating point sample types. This allowed me to mostly reduce the usage of template code and create interfaces to encapsulate the behavior of effect processors and synths more flexibly -- plus allowing for the mix-and-match effect section. This came at the cost of not having double-precision support; to add double-precision support, it will take a little bit of doing but is very possible. (Please see Planned feature #1, below, or here for more details.)
Some of the main classes and interfaces are described here. A UML class diagram is located at doc/class-diagram.png. The generic, reusable code is stored in the "Source/juce_igutil
" subdirectory. At some point this will be pulled out into a proper Juce module.
- SynthAudioSource - an AudioSource - like interface for things that create audio and expect midi input. Implementations include:
- ConfigurableSynthAudioSource - the heart of the system. It's a reusable class which is an implementation of SynthAudioSource providing the ability to configure a synth with any Synthesiser, SynthesiserVoice, SynthesiserSound, and Processor (effect) types. The design is meant to allow strategies for all those things to be plugged in, and to make it easy for new implementations to be created. The SynthesiserVoice and SynthesiserSound classes are Juce abstract classes, but the idea is the same.
- WavetableSynth - an implementation of SynthAudioSource that decorates a ConfigurableSynthAudioSource. Uses WavetableSynthVoice/WavetableOscillator as well as UnlimitedSynthSound, and controls the special effects section.
- Oscillator - generic Oscillator interface for synthesisers. Implementations include:
- WavetableOscillator - implements Oscillator and provides a way to iterate through any provided wavetables to simulate a waveform generator. It's completely configurable to allow new wavetables. It's more flexible than
juce::Oscillator
because the wavetable can be a global shared pointer and is not unique to this Oscillator, which makes it more memory-efficient.
- WavetableOscillator - implements Oscillator and provides a way to iterate through any provided wavetables to simulate a waveform generator. It's completely configurable to allow new wavetables. It's more flexible than
- Processor - generic effect processor interface. Implementations:
- EffectProcessor - template processor class implements Processor and is meant to wrap template "
juce::dsp
" classes with the Processor interface. - ProcessorSequence - similar to "
juce::dsp::ProcessorChain
" but for my Processors. It's much more flexible than ProcessorChain because it uses runtime polymorphism rather than hardcoded compile-time typing. - DelayProcessor - effect that uses a juce DelayLine to produce a standard delay effect.
- EffectProcessor - template processor class implements Processor and is meant to wrap template "
- WavetableSynthVoice - juce::SynthesiserVoice implementation / extension that uses WavetableOscillator(s).
The malleable effects processing sequence was realized by the ProcessorSequence and some fancy footwork inside the WavetableSynth. At startup, all of the possible effects are created (to avoid processing delays when rendering). They are added to a pool of effects and then pulled out into a ProcessorSequence when selected in the UI. This effects section could probably be pulled out into a generic module.
To allow for control of gain parameters in each effect, a UI knob is created for each effect slot 1-6 with a corresponding Parameter. The parameters are read at the start of each render cycle and set appropriately through lambda functions attached to the effect. To get more detailed effect parameters, there needs to be fancier UI handling (see Planned Features below).
- Install Juce: https://juce.com/
- Note: Juce is a cross-platform framework that should work fine on Windows, Mac, and Linux; however I've only tested it on Windows. If you use a Mac (or you are a brave soul who does audio on Linux), please let me know if there are any incompatabilities.
- Install an appropriate C++ development environment to compile C++ projects. (For example, on Windows you'll need MS Visual Studio 2019)
- Open the "
midi-synthesiser.jucer
" file in the Projucer. - Change the global path to the Juce modules to match your system. By default it is set to "
../../../../../opt/juce/modules
" which has very low odds of matching your local setup. - Save the jucer project, then open the C++ project in your chosen IDE.
- Build the project. (In Visual Studio, select "
Build / Rebuild Solution
".)
- Standalone, after building in the IDE (easiest way):
- In the IDE (MS Visual Studio), click on the run button labelled "Local Windows Debugger" to run it in standalone mode, or double-click the EXE that is produced.
- As a plugin, inside an audio host program:
- Copy the *.vst3 file to the appropriate location and load it in your host. The VST is generated in
"./Builds/VisualStudio2019/x64/Debug/VST3/midi-synthesiser.vst3
", and on Windows systems you should copy it to "C:\Program Files\Common Files\VST3\
"
- Copy the *.vst3 file to the appropriate location and load it in your host. The VST is generated in
New features that I plan on working on in my spare time include:
- Double-precision support. Currently only single-precision audio processing is supported. Convert it to double-precision and add a buffer copy operation inside the single-precision processBlock() method to call the double-precision processor. Details on options for supporting multiple precisions easier can be found at my page https://github.com/invadergir/juce-double-precision-poc.
- Swappable "deep" controls for the effects, where a UI Component with lots of attached knobs is created/destroyed properly when a new effect is chosen. (For example, delay time, number of repeats, etc.) This requires figuring out a better way to handle the Params for each control without using too much memory.
- Pull out "
juce_igutil
" code into a separate repo containing a proper juce module or library. - Extract malleable effects processing section from WavetableSynth into reusable module.
- Profile memory usage and possibly / probably limit the number of duplicate effects to 2 or 3. Creating 6 effect instances of each type uses a bit of memory, especially the delays.
- Resolve various TODOs.
- Multiple instances of the same effect can be very loud. In case you haven't heard yet, you probably want to TURN DOWN YOUR VOLUME if you start using extreme effects settings or multiple effect instances of the same type.....
- The standalone audio device detection and usage is straight from Juce, and as such, it is not perfectly robust. On my dev system, I can only get driver buffer sizes of 144 samples but in Studio One it goes with what is on the host, and that goes as low as your hardware driver allows.