Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

audiofilters: Add Distortion effect and implement LFO ticking #9776

Open
wants to merge 30 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
16575fc
Added `Distortion` and `DistortionMode` classes to `audiofilters` mod…
relic-se Oct 23, 2024
46ebae1
Remove separate DistortionMode code files.
relic-se Oct 31, 2024
3a16daf
Remove `audiofilters.DistortionMode.ATAN`
relic-se Oct 31, 2024
1008dd5
Simplify `audiofilters.DistortionMode.LOFI` sample processing with bi…
relic-se Oct 31, 2024
37b6b70
Fix error with null sample handling in `audiofilters.Distortion`.
relic-se Nov 4, 2024
064c3f3
Merge branch 'adafruit:main' into audiofilters_distortion
relic-se Nov 13, 2024
a7060f0
Merge branch 'adafruit:main' into audiofilters_distortion
relic-se Nov 27, 2024
6481b4e
Merge branch 'adafruit:main' into audiofilters_distortion
relic-se Dec 10, 2024
31c9095
Implement `synthio_block_slot_get_limited`.
relic-se Dec 10, 2024
155f197
Convert default float values from null checks to MP_ROM_INT.
relic-se Dec 10, 2024
89f2ae1
Remove unnecessary kwarg setters.
relic-se Dec 10, 2024
5c981f0
Use `MICROPY_FLOAT_CONST` and `MICROPY_FLOAT_C_FUN` within floating p…
relic-se Dec 10, 2024
222ce2c
Apply similar updates to audiofilters.Filter and audiodelays.Echo: MI…
relic-se Dec 10, 2024
0410d22
Added `soft_clip` property to toggle between hard clipping (default) …
relic-se Dec 11, 2024
57022f9
Implemented soft clipping and continued optimization of distortion al…
relic-se Dec 11, 2024
48ca21d
Add Distortion to unix port and make type conversions explicit.
relic-se Dec 11, 2024
4257c62
Variable number of samples within `shared_bindings_synthio_lfo_tick`.
relic-se Dec 12, 2024
0e64e1c
Implement block ticking within audio effects.
relic-se Dec 12, 2024
5fbbeed
Call `shared_bindings_synthio_lfo_tick` on audioeffects in `SYNTHIO_M…
relic-se Dec 12, 2024
e18e5b2
Merge branch 'adafruit:main' into audiofilters_distortion
relic-se Jan 17, 2025
cef94d7
Remove unnecessary deinit check.
relic-se Jan 17, 2025
b796f0d
Only calculate lofi bit mask when necessary.
relic-se Jan 17, 2025
84f8e31
Update `shared_bindings_synthio_lfo_tick` within `audiomixer` to use …
relic-se Jan 17, 2025
d84cdbc
Update Distortion class docstring.
relic-se Jan 17, 2025
8c40c56
Move `shared_bindings_synthio_lfo_tick` to avoid error if using unsig…
relic-se Jan 17, 2025
05db339
Remove truncation of `delay_ms` within buffer processing loop.
relic-se Jan 17, 2025
ec5b1e8
Add `mix_down_sample` to echo output.
relic-se Jan 17, 2025
48f272e
Fix build errors.
relic-se Jan 17, 2025
99b4fae
Remove unnecessary copies of `mix_down_sample`.
relic-se Jan 17, 2025
e049337
Allow variable mix down sample scale.
relic-se Jan 18, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
30 changes: 2 additions & 28 deletions shared-module/audiodelays/Echo.c
Original file line number Diff line number Diff line change
Expand Up @@ -258,32 +258,6 @@ void common_hal_audiodelays_echo_stop(audiodelays_echo_obj_t *self) {
return;
}

#define RANGE_LOW_16 (-28000)
#define RANGE_HIGH_16 (28000)
#define RANGE_SHIFT_16 (16)
#define RANGE_SCALE_16 (0xfffffff / (32768 * 2 - RANGE_HIGH_16)) // 2 for echo+sample
relic-se marked this conversation as resolved.
Show resolved Hide resolved

// dynamic range compression via a downward compressor with hard knee
//
// When the output value is within the range +-28000 (about 85% of full scale),
// it is unchanged. Otherwise, it undergoes a gain reduction so that the
// largest possible values, (+32768,-32767) * 2 (2 for echo and sample),
// still fit within the output range
//
// This produces a much louder overall volume with multiple voices, without
// much additional processing.
//
// https://en.wikipedia.org/wiki/Dynamic_range_compression
static
int16_t mix_down_sample(int32_t sample) {
if (sample < RANGE_LOW_16) {
sample = (((sample - RANGE_LOW_16) * RANGE_SCALE_16) >> RANGE_SHIFT_16) + RANGE_LOW_16;
} else if (sample > RANGE_HIGH_16) {
sample = (((sample - RANGE_HIGH_16) * RANGE_SCALE_16) >> RANGE_SHIFT_16) + RANGE_HIGH_16;
}
return sample;
}

audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t *self, bool single_channel_output, uint8_t channel,
uint8_t **buffer, uint32_t *buffer_length) {

Expand Down Expand Up @@ -454,7 +428,7 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t *
}

if (MP_LIKELY(self->bits_per_sample == 16)) {
word = mix_down_sample(word);
word = synthio_mix_down_sample(word);
if (self->freq_shift) {
for (uint32_t j = echo_buffer_pos >> 8; j < next_buffer_pos >> 8; j++) {
echo_buffer[j % echo_buf_len] = (int16_t)word;
Expand All @@ -479,7 +453,7 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t *
}

word = echo + sample_word;
relic-se marked this conversation as resolved.
Show resolved Hide resolved
word = mix_down_sample(word);
word = synthio_mix_down_sample(word);

if (MP_LIKELY(self->bits_per_sample == 16)) {
word_buffer[i] = (int16_t)((sample_word * (MICROPY_FLOAT_CONST(1.0) - mix)) + (word * mix));
Expand Down
1 change: 1 addition & 0 deletions shared-module/audiodelays/Echo.h
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
#include "py/obj.h"

#include "shared-module/audiocore/__init__.h"
#include "shared-module/synthio/__init__.h"
#include "shared-module/synthio/block.h"

extern const mp_obj_type_t audiodelays_echo_type;
Expand Down
28 changes: 1 addition & 27 deletions shared-module/audiofilters/Filter.c
Original file line number Diff line number Diff line change
Expand Up @@ -212,32 +212,6 @@ void common_hal_audiofilters_filter_stop(audiofilters_filter_obj_t *self) {
return;
}

#define RANGE_LOW_16 (-28000)
#define RANGE_HIGH_16 (28000)
#define RANGE_SHIFT_16 (16)
#define RANGE_SCALE_16 (0xfffffff / (32768 * 2 - RANGE_HIGH_16)) // 2 for echo+sample

// dynamic range compression via a downward compressor with hard knee
//
// When the output value is within the range +-28000 (about 85% of full scale),
// it is unchanged. Otherwise, it undergoes a gain reduction so that the
// largest possible values, (+32768,-32767) * 2 (2 for echo and sample),
// still fit within the output range
//
// This produces a much louder overall volume with multiple voices, without
// much additional processing.
//
// https://en.wikipedia.org/wiki/Dynamic_range_compression
static
int16_t mix_down_sample(int32_t sample) {
if (sample < RANGE_LOW_16) {
sample = (((sample - RANGE_LOW_16) * RANGE_SCALE_16) >> RANGE_SHIFT_16) + RANGE_LOW_16;
} else if (sample > RANGE_HIGH_16) {
sample = (((sample - RANGE_HIGH_16) * RANGE_SCALE_16) >> RANGE_SHIFT_16) + RANGE_HIGH_16;
}
return sample;
}

audioio_get_buffer_result_t audiofilters_filter_get_buffer(audiofilters_filter_obj_t *self, bool single_channel_output, uint8_t channel,
uint8_t **buffer, uint32_t *buffer_length) {
(void)channel;
Expand Down Expand Up @@ -341,7 +315,7 @@ audioio_get_buffer_result_t audiofilters_filter_get_buffer(audiofilters_filter_o
// Mix processed signal with original sample and transfer to output buffer
for (uint32_t j = 0; j < n_samples; j++) {
if (MP_LIKELY(self->bits_per_sample == 16)) {
word_buffer[i + j] = mix_down_sample((int32_t)((sample_src[i + j] * (MICROPY_FLOAT_CONST(1.0) - mix)) + (self->filter_buffer[j] * mix)));
word_buffer[i + j] = synthio_mix_down_sample((int32_t)((sample_src[i + j] * (MICROPY_FLOAT_CONST(1.0) - mix)) + (self->filter_buffer[j] * mix)));
if (!self->samples_signed) {
word_buffer[i + j] ^= 0x8000;
}
Expand Down
1 change: 1 addition & 0 deletions shared-module/audiofilters/Filter.h
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@

#include "shared-bindings/synthio/Biquad.h"
#include "shared-module/audiocore/__init__.h"
#include "shared-module/synthio/__init__.h"
#include "shared-module/synthio/block.h"
#include "shared-module/synthio/Biquad.h"

Expand Down
5 changes: 2 additions & 3 deletions shared-module/synthio/__init__.c
Original file line number Diff line number Diff line change
Expand Up @@ -145,8 +145,7 @@ static synthio_envelope_definition_t *synthio_synth_get_note_envelope(synthio_sy
// much additional processing.
//
// https://en.wikipedia.org/wiki/Dynamic_range_compression
static
int16_t mix_down_sample(int32_t sample) {
int16_t synthio_mix_down_sample(int32_t sample) {
if (sample < RANGE_LOW) {
sample = (((sample - RANGE_LOW) * RANGE_SCALE) >> RANGE_SHIFT) + RANGE_LOW;
} else if (sample > RANGE_HIGH) {
Expand Down Expand Up @@ -345,7 +344,7 @@ void synthio_synth_synthesize(synthio_synth_t *synth, uint8_t **bufptr, uint32_t
// mix down audio
for (size_t i = 0; i < dur * synth->channel_count; i++) {
int32_t sample = out_buffer32[i];
out_buffer16[i] = mix_down_sample(sample);
out_buffer16[i] = synthio_mix_down_sample(sample);
}

// advance envelope states
Expand Down
2 changes: 2 additions & 0 deletions shared-module/synthio/__init__.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,8 @@ bool synthio_span_change_note(synthio_synth_t *synth, mp_obj_t old_note, mp_obj_
void synthio_envelope_step(synthio_envelope_definition_t *definition, synthio_envelope_state_t *state, int n_samples);
void synthio_envelope_definition_set(synthio_envelope_definition_t *envelope, mp_obj_t obj, uint32_t sample_rate);

int16_t synthio_mix_down_sample(int32_t sample);

uint64_t synthio_frequency_convert_float_to_scaled(mp_float_t frequency_hz);
uint32_t synthio_frequency_convert_float_to_dds(mp_float_t frequency_hz, int32_t sample_rate);
uint32_t synthio_frequency_convert_scaled_to_dds(uint64_t frequency_scaled, int32_t sample_rate);
Expand Down