2022-03-09 03:41:23 +11:00
|
|
|
// Crisp: a distortion plugin but not quite
|
2022-03-09 23:17:26 +11:00
|
|
|
// Copyright (C) 2022 Robbert van der Helm
|
2022-03-09 03:41:23 +11:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
2022-03-09 23:24:57 +11:00
|
|
|
#[macro_use]
|
|
|
|
extern crate nih_plug;
|
|
|
|
|
2022-03-09 03:41:23 +11:00
|
|
|
use nih_plug::prelude::*;
|
2022-03-09 07:10:36 +11:00
|
|
|
use pcg::Pcg32iState;
|
2022-03-09 03:41:23 +11:00
|
|
|
use std::pin::Pin;
|
2022-03-09 09:30:12 +11:00
|
|
|
use std::sync::Arc;
|
2022-03-09 03:41:23 +11:00
|
|
|
|
2022-03-09 23:24:57 +11:00
|
|
|
mod filter;
|
2022-03-09 06:42:59 +11:00
|
|
|
mod pcg;
|
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
/// The number of channels we support. Hardcoded to allow for easier SIMD-ifying in the future.
|
|
|
|
const NUM_CHANNELS: u32 = 2;
|
|
|
|
|
2022-03-09 05:02:21 +11:00
|
|
|
/// These seeds being fixed makes bouncing deterministic.
|
2022-03-09 07:10:36 +11:00
|
|
|
const INITIAL_PRNG_SEED: Pcg32iState = Pcg32iState::new(69, 420);
|
2022-03-09 05:02:21 +11:00
|
|
|
|
2022-03-09 09:20:30 +11:00
|
|
|
/// Allow 100% amount to scale the gain to a bit above 100%, to make the effect even less subtle.
|
|
|
|
const AMOUNT_GAIN_MULTIPLIER: f32 = 2.0;
|
|
|
|
|
2022-03-09 10:51:52 +11:00
|
|
|
/// This plugin essentially layers the sound with another copy of the signal ring modulated with
|
2022-03-09 23:24:57 +11:00
|
|
|
/// white (or filtered) noise. That other copy of the sound may have a low-pass filter applied to it
|
2022-03-09 10:51:52 +11:00
|
|
|
/// since this effect just turns into literal noise at high frequencies.
|
2022-03-09 03:41:23 +11:00
|
|
|
struct Crisp {
|
|
|
|
params: Pin<Box<CrispParams>>,
|
2022-03-09 05:02:21 +11:00
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
/// Needed for computing the filter coefficients.
|
|
|
|
sample_rate: f32,
|
|
|
|
|
2022-03-09 07:10:36 +11:00
|
|
|
/// A PRNG for generating noise, after that we'll implement PCG ourselves so we can easily
|
|
|
|
/// SIMD-ify this in the future.
|
|
|
|
prng: Pcg32iState,
|
2022-03-09 23:48:23 +11:00
|
|
|
|
|
|
|
/// Resonant filters for high passing the noise signal, to make it even brighter.
|
|
|
|
noise_hpf: [filter::Biquad<f32>; NUM_CHANNELS as usize],
|
2022-03-09 03:41:23 +11:00
|
|
|
}
|
|
|
|
|
2022-03-09 10:51:52 +11:00
|
|
|
// TODO: Add a filter for the RM input
|
2022-03-09 10:03:28 +11:00
|
|
|
// TODO: Add more kinds of noise
|
2022-03-09 03:41:23 +11:00
|
|
|
#[derive(Params)]
|
2022-03-09 05:02:21 +11:00
|
|
|
pub struct CrispParams {
|
|
|
|
/// On a range of `[0, 1]`, how much of the modulated sound to mix in.
|
|
|
|
#[id = "amount"]
|
|
|
|
amount: FloatParam,
|
2022-03-09 10:51:52 +11:00
|
|
|
/// What kind of RM to apply. The preset this was modelled after whether intentional or not only
|
|
|
|
/// RMs the positive part of the waveform.
|
2022-03-09 09:46:05 +11:00
|
|
|
#[id = "mode"]
|
|
|
|
mode: EnumParam<Mode>,
|
2022-03-09 10:03:28 +11:00
|
|
|
/// How to handle stereo signals. See [`StereoMode`].
|
|
|
|
#[id = "stereo"]
|
|
|
|
stereo_mode: EnumParam<StereoMode>,
|
2022-03-09 09:30:12 +11:00
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
/// The cutoff frequency for the high pass filter applied to the noise.
|
|
|
|
#[id = "nzhpff"]
|
|
|
|
noise_hpf_freq: FloatParam,
|
|
|
|
/// The Q parameter for the high pass filter applied to the noise.
|
|
|
|
#[id = "nzhpfq"]
|
|
|
|
noise_hpf_q: FloatParam,
|
|
|
|
|
2022-03-09 09:30:12 +11:00
|
|
|
/// Output gain, as voltage gain. Displayed in decibels.
|
|
|
|
#[id = "output"]
|
|
|
|
output_gain: FloatParam,
|
2022-03-09 05:02:21 +11:00
|
|
|
}
|
2022-03-09 03:41:23 +11:00
|
|
|
|
2022-03-09 09:46:05 +11:00
|
|
|
/// Controls the type of modulation to apply.
|
|
|
|
#[derive(Enum, Debug, PartialEq)]
|
|
|
|
enum Mode {
|
2022-03-09 10:51:52 +11:00
|
|
|
/// RM the entire waveform.
|
2022-03-09 09:46:05 +11:00
|
|
|
Crispy,
|
2022-03-09 10:51:52 +11:00
|
|
|
/// RM only the positive part of the waveform.
|
2022-03-09 09:46:05 +11:00
|
|
|
#[name = "Even Crispier"]
|
|
|
|
EvenCrispier,
|
2022-03-09 10:51:52 +11:00
|
|
|
/// RM only the negative part of the waveform.
|
2022-03-09 09:46:05 +11:00
|
|
|
#[name = "Even Crispier (alt)"]
|
|
|
|
EvenCrispierNegated,
|
|
|
|
}
|
|
|
|
|
2022-03-09 10:03:28 +11:00
|
|
|
/// Controls how to handle stereo input.
|
|
|
|
#[derive(Enum, Debug, PartialEq)]
|
|
|
|
enum StereoMode {
|
|
|
|
/// Use the same noise for both channels.
|
|
|
|
Mono,
|
|
|
|
/// Use a different noise source per channel.
|
|
|
|
Stereo,
|
|
|
|
}
|
|
|
|
|
2022-03-09 03:41:23 +11:00
|
|
|
impl Default for Crisp {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
params: Box::pin(CrispParams::default()),
|
2022-03-09 05:02:21 +11:00
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
sample_rate: 1.0,
|
|
|
|
|
2022-03-09 07:10:36 +11:00
|
|
|
prng: INITIAL_PRNG_SEED,
|
2022-03-09 23:48:23 +11:00
|
|
|
noise_hpf: [filter::Biquad::default(); NUM_CHANNELS as usize],
|
2022-03-09 03:41:23 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for CrispParams {
|
|
|
|
#[allow(clippy::derivable_impls)]
|
|
|
|
fn default() -> Self {
|
2022-03-09 05:02:21 +11:00
|
|
|
Self {
|
2022-03-09 10:05:37 +11:00
|
|
|
amount: FloatParam::new("Amount", 0.35, FloatRange::Linear { min: 0.0, max: 1.0 })
|
2022-03-09 05:02:21 +11:00
|
|
|
.with_smoother(SmoothingStyle::Linear(10.0))
|
|
|
|
.with_unit("%")
|
|
|
|
.with_value_to_string(formatters::f32_percentage(0))
|
|
|
|
.with_string_to_value(formatters::from_f32_percentage()),
|
2022-03-09 23:48:23 +11:00
|
|
|
|
2022-03-09 09:46:05 +11:00
|
|
|
mode: EnumParam::new("Mode", Mode::EvenCrispier),
|
2022-03-09 10:03:28 +11:00
|
|
|
stereo_mode: EnumParam::new("Stereo Mode", StereoMode::Stereo),
|
2022-03-09 23:48:23 +11:00
|
|
|
|
|
|
|
noise_hpf_freq: FloatParam::new(
|
|
|
|
"Noise HPF Frequency",
|
|
|
|
1.0,
|
|
|
|
FloatRange::Skewed {
|
|
|
|
min: 1.0,
|
|
|
|
max: 22_000.0,
|
|
|
|
factor: FloatRange::skew_factor(-1.0),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.with_smoother(SmoothingStyle::Logarithmic(100.0))
|
|
|
|
.with_unit(" Hz")
|
|
|
|
.with_value_to_string(Arc::new(|value| {
|
|
|
|
if value <= 1.0 {
|
|
|
|
String::from("Disabled")
|
|
|
|
} else {
|
|
|
|
format!("{:.0}", value)
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
.with_string_to_value(Arc::new(|string| {
|
|
|
|
if string == "Disabled" {
|
|
|
|
Some(1.0)
|
|
|
|
} else {
|
|
|
|
string.trim().trim_end_matches(" Hz").parse().ok()
|
|
|
|
}
|
|
|
|
})),
|
|
|
|
noise_hpf_q: FloatParam::new(
|
|
|
|
"Noise HPF Resonance",
|
|
|
|
2.0f32.sqrt() / 2.0,
|
|
|
|
FloatRange::Skewed {
|
|
|
|
min: 2.0f32.sqrt() / 2.0,
|
|
|
|
max: 10.0,
|
|
|
|
factor: FloatRange::skew_factor(-1.0),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.with_smoother(SmoothingStyle::Logarithmic(100.0))
|
|
|
|
.with_value_to_string(formatters::f32_rounded(2)),
|
|
|
|
|
2022-03-09 09:30:12 +11:00
|
|
|
output_gain: FloatParam::new(
|
|
|
|
"Output",
|
|
|
|
1.0,
|
|
|
|
// Because we're representing gain as decibels the range is already logarithmic
|
|
|
|
FloatRange::Linear {
|
|
|
|
min: util::db_to_gain(-24.0),
|
|
|
|
max: util::db_to_gain(0.0),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.with_smoother(SmoothingStyle::Logarithmic(10.0))
|
|
|
|
.with_unit(" dB")
|
|
|
|
.with_value_to_string(Arc::new(|value| format!("{:.2}", util::gain_to_db(value))))
|
|
|
|
.with_string_to_value(Arc::new(|string| {
|
|
|
|
string
|
|
|
|
.trim()
|
|
|
|
.trim_end_matches(" dB")
|
|
|
|
.parse()
|
|
|
|
.ok()
|
|
|
|
.map(util::db_to_gain)
|
|
|
|
})),
|
2022-03-09 05:02:21 +11:00
|
|
|
}
|
2022-03-09 03:41:23 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Plugin for Crisp {
|
|
|
|
const NAME: &'static str = "Crisp";
|
|
|
|
const VENDOR: &'static str = "Robbert van der Helm";
|
|
|
|
const URL: &'static str = "https://github.com/robbert-vdh/nih-plug";
|
|
|
|
const EMAIL: &'static str = "mail@robbertvanderhelm.nl";
|
|
|
|
|
|
|
|
const VERSION: &'static str = "0.1.0";
|
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
const DEFAULT_NUM_INPUTS: u32 = NUM_CHANNELS;
|
|
|
|
const DEFAULT_NUM_OUTPUTS: u32 = NUM_CHANNELS;
|
2022-03-09 03:41:23 +11:00
|
|
|
|
|
|
|
fn params(&self) -> Pin<&dyn Params> {
|
|
|
|
self.params.as_ref()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn accepts_bus_config(&self, config: &BusConfig) -> bool {
|
|
|
|
// We'll add a SIMD version in a bit which only supports stereo
|
2022-03-09 23:48:23 +11:00
|
|
|
config.num_input_channels == config.num_output_channels
|
|
|
|
&& config.num_input_channels == NUM_CHANNELS
|
|
|
|
}
|
|
|
|
|
|
|
|
fn initialize(
|
|
|
|
&mut self,
|
|
|
|
bus_config: &BusConfig,
|
|
|
|
buffer_config: &BufferConfig,
|
|
|
|
_context: &mut impl ProcessContext,
|
|
|
|
) -> bool {
|
|
|
|
nih_debug_assert_eq!(bus_config.num_input_channels, NUM_CHANNELS);
|
|
|
|
nih_debug_assert_eq!(bus_config.num_output_channels, NUM_CHANNELS);
|
|
|
|
self.sample_rate = buffer_config.sample_rate;
|
|
|
|
|
|
|
|
true
|
2022-03-09 03:41:23 +11:00
|
|
|
}
|
|
|
|
|
2022-03-09 05:02:21 +11:00
|
|
|
fn reset(&mut self) {
|
|
|
|
// By using the same seeds each time bouncing can be made deterministic
|
2022-03-09 07:10:36 +11:00
|
|
|
self.prng = INITIAL_PRNG_SEED;
|
2022-03-09 23:48:23 +11:00
|
|
|
|
|
|
|
for filter in &mut self.noise_hpf {
|
|
|
|
filter.reset();
|
|
|
|
}
|
2022-03-09 05:02:21 +11:00
|
|
|
}
|
|
|
|
|
2022-03-09 03:41:23 +11:00
|
|
|
fn process(
|
|
|
|
&mut self,
|
|
|
|
buffer: &mut Buffer,
|
|
|
|
_context: &mut impl ProcessContext,
|
|
|
|
) -> ProcessStatus {
|
2022-03-09 05:02:21 +11:00
|
|
|
for channel_samples in buffer.iter_mut() {
|
2022-03-09 09:20:30 +11:00
|
|
|
let amount = self.params.amount.smoothed.next() * AMOUNT_GAIN_MULTIPLIER;
|
2022-03-09 09:30:12 +11:00
|
|
|
let output_gain = self.params.output_gain.smoothed.next();
|
2022-03-09 05:02:21 +11:00
|
|
|
|
2022-03-09 23:48:23 +11:00
|
|
|
// Controls the HPF applied to the noise signal
|
|
|
|
self.maybe_update_filters();
|
|
|
|
|
2022-03-09 05:02:21 +11:00
|
|
|
// TODO: SIMD-ize this to process both channels at once
|
2022-03-09 10:03:28 +11:00
|
|
|
// TODO: Avoid branching twice here. Modern branch predictors are pretty good at this
|
|
|
|
// though.
|
|
|
|
match self.params.stereo_mode.value() {
|
|
|
|
StereoMode::Mono => {
|
2022-03-09 23:48:23 +11:00
|
|
|
let noise = self.gen_noise(0);
|
2022-03-09 10:03:28 +11:00
|
|
|
for sample in channel_samples {
|
2022-03-09 10:51:52 +11:00
|
|
|
*sample += self.do_ring_mod(*sample, noise) * amount;
|
2022-03-09 10:03:28 +11:00
|
|
|
*sample *= output_gain;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
StereoMode::Stereo => {
|
2022-03-09 23:48:23 +11:00
|
|
|
for (channel_idx, sample) in channel_samples.into_iter().enumerate() {
|
|
|
|
let noise = self.gen_noise(channel_idx);
|
2022-03-09 10:51:52 +11:00
|
|
|
*sample += self.do_ring_mod(*sample, noise) * amount;
|
2022-03-09 10:03:28 +11:00
|
|
|
*sample *= output_gain;
|
|
|
|
}
|
|
|
|
}
|
2022-03-09 05:02:21 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-09 03:41:23 +11:00
|
|
|
ProcessStatus::Normal
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-09 10:03:28 +11:00
|
|
|
impl Crisp {
|
2022-03-09 23:48:23 +11:00
|
|
|
/// Generate a new noise sample with the high pass filter applied.
|
|
|
|
fn gen_noise(&mut self, channel: usize) -> f32 {
|
|
|
|
let noise = self.prng.next_f32() * 2.0 - 1.0;
|
|
|
|
self.noise_hpf[channel].process(noise)
|
2022-03-09 10:03:28 +11:00
|
|
|
}
|
|
|
|
|
2022-03-09 10:51:52 +11:00
|
|
|
/// Perform the RM step depending on the mode.
|
|
|
|
fn do_ring_mod(&self, sample: f32, noise: f32) -> f32 {
|
2022-03-09 10:03:28 +11:00
|
|
|
// TODO: Avoid branching in the main loop, this just makes it a bit easier to prototype
|
|
|
|
match self.params.mode.value() {
|
|
|
|
Mode::Crispy => sample * noise,
|
|
|
|
Mode::EvenCrispier => sample.max(0.0) * noise,
|
|
|
|
Mode::EvenCrispierNegated => sample.max(0.0) * noise,
|
|
|
|
}
|
|
|
|
}
|
2022-03-09 23:48:23 +11:00
|
|
|
|
|
|
|
/// Update the filter coefficients if needed. Should be called once per sample.
|
|
|
|
fn maybe_update_filters(&mut self) {
|
|
|
|
if self.params.noise_hpf_freq.smoothed.is_smoothing()
|
|
|
|
|| self.params.noise_hpf_q.smoothed.is_smoothing()
|
|
|
|
{
|
|
|
|
let frequency = self.params.noise_hpf_freq.smoothed.next();
|
|
|
|
let q = self.params.noise_hpf_q.smoothed.next();
|
|
|
|
let coefficients = filter::BiquadCoefficients::highpass(self.sample_rate, frequency, q);
|
|
|
|
for filter in &mut self.noise_hpf {
|
|
|
|
filter.coefficients = coefficients;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-09 10:03:28 +11:00
|
|
|
}
|
|
|
|
|
2022-03-09 03:41:23 +11:00
|
|
|
impl ClapPlugin for Crisp {
|
|
|
|
const CLAP_ID: &'static str = "nl.robbertvanderhelm.crisp";
|
|
|
|
const CLAP_DESCRIPTION: &'static str = "Adds a bright crispy top end to low bass sounds";
|
|
|
|
const CLAP_FEATURES: &'static [&'static str] =
|
2022-03-09 10:29:46 +11:00
|
|
|
&["audio_effect", "stereo", "distortion", "filter"];
|
2022-03-09 03:41:23 +11:00
|
|
|
const CLAP_MANUAL_URL: &'static str = Self::URL;
|
|
|
|
const CLAP_SUPPORT_URL: &'static str = Self::URL;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Vst3Plugin for Crisp {
|
|
|
|
const VST3_CLASS_ID: [u8; 16] = *b"CrispPluginRvdH.";
|
|
|
|
const VST3_CATEGORIES: &'static str = "Fx|Filter|Distortion";
|
|
|
|
}
|
|
|
|
|
|
|
|
nih_export_clap!(Crisp);
|
|
|
|
nih_export_vst3!(Crisp);
|