mirror of
https://github.com/RGBCube/serenity
synced 2025-07-26 04:07:35 +00:00
LibDSP: Add Envelope abstraction
For the upcoming synthesizer, having an abstracted ADSR envelope concept is highly desirable. Additionally, Envelope is mostly constexpr and therefore super fast :^)
This commit is contained in:
parent
a1093abe26
commit
cc9aab7462
2 changed files with 103 additions and 0 deletions
70
Userland/Libraries/LibDSP/Envelope.h
Normal file
70
Userland/Libraries/LibDSP/Envelope.h
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021, kleines Filmröllchen <malu.bertsch@gmail.com>.
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/StdLibExtras.h>
|
||||||
|
|
||||||
|
namespace LibDSP {
|
||||||
|
|
||||||
|
enum class EnvelopeState : u8 {
|
||||||
|
Off,
|
||||||
|
Attack,
|
||||||
|
Decay,
|
||||||
|
Sustain,
|
||||||
|
Release,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Envelope {
|
||||||
|
constexpr Envelope() = default;
|
||||||
|
constexpr Envelope(double envelope)
|
||||||
|
: envelope(envelope)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool is_attack() const { return 0 <= envelope && envelope < 1; }
|
||||||
|
constexpr double attack() const { return clamp(envelope, 0, 1); }
|
||||||
|
constexpr void set_attack(double offset) { envelope = offset; }
|
||||||
|
static constexpr Envelope from_attack(double attack) { return Envelope(attack); }
|
||||||
|
|
||||||
|
constexpr bool is_decay() const { return 1 <= envelope && envelope < 2; }
|
||||||
|
constexpr double decay() const { return clamp(envelope, 1, 2) - 1; }
|
||||||
|
constexpr void set_decay(double offset) { envelope = 1 + offset; }
|
||||||
|
static constexpr Envelope from_decay(double decay) { return Envelope(decay + 1); }
|
||||||
|
|
||||||
|
constexpr bool is_sustain() const { return 2 <= envelope && envelope < 3; }
|
||||||
|
constexpr double sustain() const { return clamp(envelope, 2, 3) - 2; }
|
||||||
|
constexpr void set_sustain(double offset) { envelope = 2 + offset; }
|
||||||
|
static constexpr Envelope from_sustain(double decay) { return Envelope(decay + 2); }
|
||||||
|
|
||||||
|
constexpr bool is_release() const { return 3 <= envelope && envelope < 4; }
|
||||||
|
constexpr double release() const { return clamp(envelope, 3, 4) - 3; }
|
||||||
|
constexpr void set_release(double offset) { envelope = 3 + offset; }
|
||||||
|
static constexpr Envelope from_release(double decay) { return Envelope(decay + 3); }
|
||||||
|
|
||||||
|
constexpr bool is_active() const { return 0 <= envelope && envelope < 4; }
|
||||||
|
|
||||||
|
constexpr void reset() { envelope = -1; }
|
||||||
|
|
||||||
|
constexpr operator EnvelopeState()
|
||||||
|
{
|
||||||
|
if (!is_active())
|
||||||
|
return EnvelopeState::Off;
|
||||||
|
if (is_attack())
|
||||||
|
return EnvelopeState::Attack;
|
||||||
|
if (is_decay())
|
||||||
|
return EnvelopeState::Decay;
|
||||||
|
if (is_sustain())
|
||||||
|
return EnvelopeState::Sustain;
|
||||||
|
if (is_release())
|
||||||
|
return EnvelopeState::Release;
|
||||||
|
VERIFY_NOT_REACHED();
|
||||||
|
}
|
||||||
|
|
||||||
|
double envelope { -1 };
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -11,6 +11,7 @@
|
||||||
#include <AK/Variant.h>
|
#include <AK/Variant.h>
|
||||||
#include <AK/Vector.h>
|
#include <AK/Vector.h>
|
||||||
#include <LibAudio/Buffer.h>
|
#include <LibAudio/Buffer.h>
|
||||||
|
#include <LibDSP/Envelope.h>
|
||||||
|
|
||||||
namespace LibDSP {
|
namespace LibDSP {
|
||||||
|
|
||||||
|
@ -26,6 +27,38 @@ struct RollNote {
|
||||||
u32 off_sample;
|
u32 off_sample;
|
||||||
u8 pitch;
|
u8 pitch;
|
||||||
i8 velocity;
|
i8 velocity;
|
||||||
|
|
||||||
|
Envelope to_envelope(u32 time, u32 attack_samples, u32 decay_samples, u32 release_samples)
|
||||||
|
{
|
||||||
|
i64 time_since_end = static_cast<i64>(time) - static_cast<i64>(off_sample);
|
||||||
|
// We're before the end of this note.
|
||||||
|
if (time_since_end < 0) {
|
||||||
|
i64 time_since_start = static_cast<i64>(time) - static_cast<i64>(on_sample);
|
||||||
|
if (time_since_start < 0)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
if (time_since_start < attack_samples) {
|
||||||
|
if (attack_samples == 0)
|
||||||
|
return Envelope::from_attack(0);
|
||||||
|
return Envelope::from_attack(static_cast<double>(time_since_start) / static_cast<double>(attack_samples));
|
||||||
|
}
|
||||||
|
if (time_since_start < attack_samples + decay_samples) {
|
||||||
|
if (decay_samples == 0)
|
||||||
|
return Envelope::from_decay(0);
|
||||||
|
return Envelope::from_decay(static_cast<double>(time_since_start - attack_samples) / static_cast<double>(decay_samples));
|
||||||
|
}
|
||||||
|
// This is a note-dependent value!
|
||||||
|
u32 sustain_samples = length() - attack_samples - decay_samples;
|
||||||
|
return Envelope::from_sustain(static_cast<double>(time_since_start - attack_samples - decay_samples) / static_cast<double>(sustain_samples));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Overshot the release time
|
||||||
|
if (time_since_end > release_samples)
|
||||||
|
return {};
|
||||||
|
return Envelope::from_release(static_cast<double>(time_since_end) / static_cast<double>(release_samples));
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool is_playing(u32 time) { return on_sample <= time && time <= off_sample; }
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class SignalType : u8 {
|
enum class SignalType : u8 {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue