2022-02-05 17:13:07 +01:00
|
|
|
# NIH-plug
|
|
|
|
|
2022-04-11 16:10:16 +02:00
|
|
|
[![Automated builds](https://github.com/robbert-vdh/nih-plug/actions/workflows/build.yml/badge.svg?branch=master)](https://github.com/robbert-vdh/nih-plug/actions/workflows/build.yml?query=branch%3Amaster)
|
2022-03-07 19:57:47 +01:00
|
|
|
[![Tests](https://github.com/robbert-vdh/nih-plug/actions/workflows/test.yml/badge.svg?branch=master)](https://github.com/robbert-vdh/nih-plug/actions/workflows/test.yml?query=branch%3Amaster)
|
2022-04-29 18:28:40 +02:00
|
|
|
[![Docs](https://github.com/robbert-vdh/nih-plug/actions/workflows/docs.yml/badge.svg?branch=master)](https://nih-plug.robbertvanderhelm.nl/)
|
2022-01-24 17:32:27 +01:00
|
|
|
|
2022-03-16 15:29:39 +01:00
|
|
|
This is a work in progress API-agnostic audio plugin framework written in Rust
|
2022-05-13 15:08:05 +02:00
|
|
|
as well as a small collection of plugins. The idea is to have a statefull but
|
|
|
|
simple plugin API that gets rid of as much unnecessary ceremony wherever
|
|
|
|
possible, while also keeping the amount of magic to minimum. Since this is not
|
|
|
|
quite meant for general use just yet, the plugin API surface is currently
|
|
|
|
limited to the functionality that I either needed myself or that was requested
|
|
|
|
by others. See the [current features](#current-features) section for more
|
|
|
|
information on the project's current status.
|
2022-01-28 15:02:55 +01:00
|
|
|
|
2022-04-11 16:57:59 +02:00
|
|
|
Come join us on the [Rust Audio Discord](https://discord.gg/ykxU3rt4Cb), or
|
2022-05-13 15:08:05 +02:00
|
|
|
check out the work in progress
|
|
|
|
[documentation](https://nih-plug.robbertvanderhelm.nl/) for some pointers on how
|
|
|
|
to get started.
|
2022-03-01 18:48:00 +01:00
|
|
|
|
2022-02-12 16:27:57 +01:00
|
|
|
### Table of contents
|
|
|
|
|
2022-02-13 18:42:05 +01:00
|
|
|
- [Plugins](#plugins)
|
|
|
|
- [Framework](#framework)
|
2022-03-16 15:29:39 +01:00
|
|
|
- [Current features](#current-features)
|
2022-02-13 18:42:05 +01:00
|
|
|
- [Building](#building)
|
2022-02-28 13:47:45 +01:00
|
|
|
- [Plugin formats](#plugin-formats)
|
2022-02-13 18:42:05 +01:00
|
|
|
- [Example plugins](#example-plugins)
|
2022-02-12 16:27:57 +01:00
|
|
|
- [Licensing](#licensing)
|
|
|
|
|
|
|
|
## Plugins
|
|
|
|
|
2022-05-24 23:56:21 +02:00
|
|
|
Check each plugin's readme for more details on what the plugin actually does.
|
|
|
|
You can download the development binaries for Linux, Windows and macOS from the
|
|
|
|
[automated
|
2022-06-10 15:59:18 +02:00
|
|
|
builds](https://github.com/robbert-vdh/nih-plug/actions/workflows/build.yml?query=branch%3Amaster)
|
2022-05-24 23:56:21 +02:00
|
|
|
page. Or if you're not signed in on GitHub, then you can also find the latest
|
|
|
|
nightly build [here](https://nightly.link/robbert-vdh/nih-plug/workflows/build/master).
|
|
|
|
|
|
|
|
Scroll down for more information on the plugin framework.
|
2022-02-12 16:27:57 +01:00
|
|
|
|
2022-03-08 17:41:23 +01:00
|
|
|
- [**Crisp**](plugins/crisp) adds a bright crispy top end to any low bass sound.
|
|
|
|
Inspired by Polarity's [Fake Distortion](https://youtu.be/MKfFn4L1zeg) video.
|
2022-05-29 13:42:45 +02:00
|
|
|
- [**Crossover**](plugins/crossover) is as boring as it sounds. It splits
|
|
|
|
cleanly splits the signal into two to five bands using a variety of algorithms
|
|
|
|
and outputs those through auxiliary outputs so they can be accessed and
|
|
|
|
processed individually. Meant as an alternative to Bitwig's Multiband FX
|
2022-06-07 21:39:32 +02:00
|
|
|
devices but with cleaner crossovers and a linear-phase option.
|
2022-02-12 17:27:23 +01:00
|
|
|
- [**Diopser**](plugins/diopser) is a totally original phase rotation plugin.
|
2022-02-12 16:27:57 +01:00
|
|
|
Useful for oomphing up kickdrums and basses, transforming synths into their
|
|
|
|
evil phase-y cousin, and making everything sound like a cheap Sci-Fi laser
|
2022-02-15 00:00:41 +01:00
|
|
|
beam.
|
2022-04-27 15:37:17 +02:00
|
|
|
- [**Loudness War Winner**](plugins/loudness_war_winner) does what it says on
|
|
|
|
the tin. Have you ever wanted to show off your dominance by winning the
|
|
|
|
loudness war? Neither have I. Dissatisfaction guaranteed.
|
2022-03-07 20:08:47 +01:00
|
|
|
- [**Puberty Simulator**](plugins/puberty_simulator) is that patent pending One
|
|
|
|
Weird Plugin that simulates the male voice change during puberty! If it was
|
|
|
|
not already obvious from that sentence, this plugin is a joke, but it might
|
2022-03-07 19:51:38 +01:00
|
|
|
actually be useful (or at least interesting) in some situations. This plugin
|
|
|
|
pitches the signal down an octave, but it also has the side effect of causing
|
2022-03-07 20:08:47 +01:00
|
|
|
things to sound like a cracking voice or to make them sound slightly out of
|
|
|
|
tune.
|
2022-05-24 23:57:11 +02:00
|
|
|
- [**Safety Limiter**](plugins/safety_limiter) is a simple tool to prevent ear
|
|
|
|
damage. As soon as there is a peak above 0 dBFS or the specified threshold,
|
|
|
|
the plugin will cut over to playing SOS in Morse code, gradually fading out
|
|
|
|
again when the input returns back to safe levels. Made for personal use during
|
|
|
|
plugin development and intense sound design sessions, but maybe you'll find it
|
|
|
|
useful too!
|
2022-02-12 16:27:57 +01:00
|
|
|
|
|
|
|
## Framework
|
|
|
|
|
2022-03-16 15:29:39 +01:00
|
|
|
### Current features
|
2022-02-06 18:31:49 +01:00
|
|
|
|
2022-03-16 15:29:39 +01:00
|
|
|
- Supports both VST3 and [CLAP](https://github.com/free-audio/clap) by simply
|
|
|
|
adding the corresponding `nih_export_<api>!(Foo)` macro to your plugin's
|
|
|
|
library.
|
2022-04-22 17:00:59 +02:00
|
|
|
- Standalone binaries can be made by calling `nih_export_standalone(Foo)` from
|
2022-06-15 01:01:57 +02:00
|
|
|
your `main()` function. Standalones come with a CLI for configuration and full
|
|
|
|
JACK audio, MIDI, and transport support.
|
2022-03-16 15:29:39 +01:00
|
|
|
- Declarative parameter handling without any boilerplate.
|
|
|
|
- Define parameters for your plugin by adding `FloatParam`, `IntParam`,
|
|
|
|
`BoolParam`, and `EnumParam<T>` fields to your parameter struct, assign
|
|
|
|
stable IDs to them with the `#[id = "foobar"]`, and a `#[derive(Params)]`
|
|
|
|
does all of the boring work for you.
|
|
|
|
- Parameters can have complex value distributions and the parameter objects
|
|
|
|
come with built-in smoothers and callbacks.
|
|
|
|
- Use simple enums deriving the `Enum` trait with the `EnumParam<T>` parameter
|
|
|
|
type for parameters that allow the user to choose between multiple discrete
|
|
|
|
options. That way you can use regular Rust pattern matching when working
|
|
|
|
with these values without having to do any conversions yourself.
|
|
|
|
- Store additional non-parameter state for your plugin by adding any field
|
2022-03-16 17:04:38 +01:00
|
|
|
that can be serialized with [Serde](https://serde.rs/) to your plugin's
|
2022-03-16 15:29:39 +01:00
|
|
|
`Params` object and annotating them with `#[persist = "key"]`.
|
|
|
|
- Group your parameters into logical groups by nesting `Params` objects using
|
|
|
|
the `#[nested = "Group Name"]`attribute.
|
2022-03-23 17:36:58 +01:00
|
|
|
- When needed, you can also provide your own implementation for the `Params`
|
|
|
|
trait to enable dynamically generated parameters and arrays of if mostly
|
|
|
|
identical parameter objects.
|
2022-03-16 15:29:39 +01:00
|
|
|
- Stateful. Behaves mostly like JUCE, just without all of the boilerplate.
|
|
|
|
- Does not make any assumptions on how you want to process audio, but does come
|
|
|
|
with utilities and adapters to help with common access patterns.
|
|
|
|
- Efficiently iterate over an audio buffer either per-sample per-channel,
|
|
|
|
per-block per-channel, or even per-block per-sample-per-channel with the
|
|
|
|
option to manually index the buffer or get access to a channel slice at any
|
|
|
|
time.
|
|
|
|
- Easily leverage per-channel SIMD using the SIMD adapters on the buffer and
|
|
|
|
block iterators.
|
|
|
|
- Comes with bring-your-own-FFT adapters for common (inverse) short-time
|
|
|
|
Fourier Transform operations. More to come.
|
|
|
|
- Optional sample accurate automation support for VST3 and CLAP that can be
|
|
|
|
enabled by setting the `Plugin::SAMPLE_ACCURATE_AUTOMATION` constant to
|
|
|
|
`true`.
|
2022-07-06 19:44:03 +02:00
|
|
|
- Support for CLAP's polyphonic modulation on a per-parameter basis.
|
2022-03-16 15:29:39 +01:00
|
|
|
- Comes with adapters for popular Rust GUI frameworks as well as some basic
|
|
|
|
widgets for them that integrate with NIH-plug's parameter system. Currently
|
2022-03-17 23:14:45 +01:00
|
|
|
there's support for [egui](nih_plug_egui), [iced](nih_plug_iced) and
|
|
|
|
[VIZIA](nih_plug_vizia).
|
2022-04-07 17:42:52 +02:00
|
|
|
- A simple and safe API for state saving and restoring from the editor is
|
|
|
|
provided by the framework if you want to do your own internal preset
|
|
|
|
management.
|
2022-04-11 19:52:51 +02:00
|
|
|
- Full support for receiving and outputting both modern polyphonic note
|
|
|
|
expression events as well as MIDI CCs, channel pressure, and pitch bend for
|
|
|
|
CLAP and VST3.
|
2022-05-23 17:07:48 +02:00
|
|
|
- Support for flexible dynamic buffer configurations, including multiple input
|
|
|
|
and output busses.
|
2022-03-16 15:29:39 +01:00
|
|
|
- A plugin bundler accessible through the
|
2022-04-05 16:42:41 +02:00
|
|
|
`cargo xtask bundle <package> <build_arguments>` command that automatically
|
|
|
|
detects which plugin targets your plugin exposes and creates the correct
|
|
|
|
plugin bundles for your target operating system and architecture, with
|
|
|
|
cross-compilation support. The cargo subcommand can easily be added to [your
|
|
|
|
own project](https://github.com/robbert-vdh/nih-plug/tree/master/nih_plug_xtask)
|
|
|
|
as an alias or [globally](https://github.com/robbert-vdh/nih-plug/tree/master/cargo_nih_plug)
|
2022-03-16 15:29:39 +01:00
|
|
|
as a regular cargo subcommand.
|
2022-04-05 16:42:41 +02:00
|
|
|
- Tested on Linux and Windows, with limited testing on macOS. Windows support
|
|
|
|
has mostly been tested through Wine with
|
2022-03-16 15:29:39 +01:00
|
|
|
[yabridge](https://github.com/robbert-vdh/yabridge).
|
|
|
|
- See the [`Plugin`](src/plugin.rs) trait's documentation for an incomplete list
|
|
|
|
of the functionlaity that has currently not yet been implemented.
|
2022-02-06 18:31:49 +01:00
|
|
|
|
2022-02-12 16:27:57 +01:00
|
|
|
### Building
|
2022-01-29 18:38:09 +01:00
|
|
|
|
2022-02-12 16:27:57 +01:00
|
|
|
NIH-plug works with the latest stable Rust compiler.
|
2022-02-05 17:14:28 +01:00
|
|
|
|
2022-02-15 18:30:45 +01:00
|
|
|
After installing [Rust](https://rustup.rs/), you can compile any of the plugins
|
2022-01-29 18:38:09 +01:00
|
|
|
in the `plugins` directory in the following way, replacing `gain` with the name
|
|
|
|
of the plugin:
|
|
|
|
|
|
|
|
```shell
|
2022-02-26 20:12:08 +01:00
|
|
|
cargo xtask bundle gain --release
|
2022-01-29 18:38:09 +01:00
|
|
|
```
|
|
|
|
|
2022-02-28 13:47:45 +01:00
|
|
|
### Plugin formats
|
|
|
|
|
2022-03-03 18:31:27 +01:00
|
|
|
NIH-plug can currently export VST3 and
|
|
|
|
[CLAP](https://github.com/free-audio/clap) plugins. Exporting a specific plugin
|
|
|
|
format for a plugin is as simple as calling the `nih_export_<format>!(Foo);`
|
2022-06-15 13:35:20 +00:00
|
|
|
macro. The `cargo xtask bundle` command will detect which plugin formats your
|
2022-03-03 18:31:27 +01:00
|
|
|
plugin supports and create the appropriate bundles accordingly, even when cross
|
|
|
|
compiling.
|
2022-02-28 13:47:45 +01:00
|
|
|
|
2022-02-12 16:27:57 +01:00
|
|
|
### Example plugins
|
2022-02-04 02:57:29 +01:00
|
|
|
|
|
|
|
The best way to get an idea for what the API looks like is to look at the
|
|
|
|
examples.
|
|
|
|
|
2022-02-13 21:04:07 +01:00
|
|
|
- [**gain**](plugins/examples/gain) is a simple smoothed gain plugin that shows
|
|
|
|
off a couple other parts of the API, like support for storing arbitrary
|
|
|
|
serializable state.
|
2022-03-17 23:20:16 +01:00
|
|
|
- **gain-gui** is the same plugin as gain, but with a GUI to control the
|
|
|
|
parameter and a digital peak meter. Comes in three exciting flavors:
|
2022-04-14 23:53:14 +02:00
|
|
|
[egui](plugins/examples/gain_gui_egui),
|
|
|
|
[iced](plugins/examples/gain_gui_iced), and
|
|
|
|
[VIZIA](plugins/examples/gain_gui_vizia).
|
2022-07-06 13:55:53 +02:00
|
|
|
- [**midi_inverter**](plugins/examples/midi_inverter) takes note/MIDI events and
|
2022-04-11 20:47:00 +02:00
|
|
|
flips around the note, channel, expression, pressure, and CC values. This
|
|
|
|
example demonstrates how to receive and output those events.
|
2022-07-06 13:55:53 +02:00
|
|
|
- [**poly_mod_synth**](plugins/examples/poly_mod_synth) is a simple polyphonic
|
|
|
|
synthesizer with support for polyphonic modulation in supported CLAP hosts.
|
|
|
|
This demonstrates how polyphonic modulation can be used in NIH-plug.
|
2022-02-13 21:04:07 +01:00
|
|
|
- [**sine**](plugins/examples/sine) is a simple test tone generator plugin with
|
|
|
|
frequency smoothing that can also make use of MIDI input instead of generating
|
|
|
|
a static signal based on the plugin's parameters.
|
2022-03-06 02:07:53 +01:00
|
|
|
- [**stft**](plugins/examples/stft) shows off some of NIH-plug's other optional
|
2022-03-06 18:54:18 +01:00
|
|
|
higher level helper features, such as an adapter to process audio with a
|
|
|
|
short-term Fourier transform using the overlap-add method, all using the
|
|
|
|
compositional `Buffer` interfaces.
|
2022-02-04 02:57:29 +01:00
|
|
|
|
2022-01-28 15:02:55 +01:00
|
|
|
## Licensing
|
|
|
|
|
2022-02-13 20:10:43 +01:00
|
|
|
The framework, its libraries, and the example plugins in `plugins/examples/` are
|
|
|
|
all licensed under the [ISC license](https://www.isc.org/licenses/). However,
|
|
|
|
the [VST3 bindings](https://github.com/RustAudio/vst3-sys) used by
|
|
|
|
`nih_export_vst3!()` are licensed under the GPLv3 license. This means that
|
|
|
|
unless you replace these bindings with your own bindings made from scratch, any
|
|
|
|
VST3 plugins built with NIH-plug need to be able to comply with the terms of the
|
|
|
|
GPLv3 license.
|
|
|
|
|
|
|
|
The other plugins in the `plugins/` directory may be licensed under the GPLv3
|
|
|
|
license. Check the plugin's `Cargo.toml` file for more information.
|