-
Notifications
You must be signed in to change notification settings - Fork 1.2k
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
base: main
Are you sure you want to change the base?
Changes from all commits
16575fc
46ebae1
3a16daf
1008dd5
37b6b70
064c3f3
a7060f0
6481b4e
31c9095
155f197
89f2ae1
5c981f0
222ce2c
0410d22
57022f9
48ca21d
4257c62
0e64e1c
5fbbeed
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
// This file is part of the CircuitPython project: https://circuitpython.org | ||
// | ||
// SPDX-FileCopyrightText: Copyright (c) 2024 Cooper Dalrymple | ||
// | ||
// SPDX-License-Identifier: MIT | ||
|
||
#pragma once | ||
|
||
#include "shared-module/audiofilters/Distortion.h" | ||
|
||
extern const mp_obj_type_t audiofilters_distortion_type; | ||
extern const mp_obj_type_t audiofilters_distortion_mode_type; | ||
|
||
void common_hal_audiofilters_distortion_construct(audiofilters_distortion_obj_t *self, | ||
mp_obj_t drive, mp_obj_t pre_gain, mp_obj_t post_gain, | ||
audiofilters_distortion_mode mode, bool soft_clip, mp_obj_t mix, | ||
uint32_t buffer_size, uint8_t bits_per_sample, bool samples_signed, | ||
uint8_t channel_count, uint32_t sample_rate); | ||
|
||
void common_hal_audiofilters_distortion_deinit(audiofilters_distortion_obj_t *self); | ||
bool common_hal_audiofilters_distortion_deinited(audiofilters_distortion_obj_t *self); | ||
|
||
uint32_t common_hal_audiofilters_distortion_get_sample_rate(audiofilters_distortion_obj_t *self); | ||
uint8_t common_hal_audiofilters_distortion_get_channel_count(audiofilters_distortion_obj_t *self); | ||
uint8_t common_hal_audiofilters_distortion_get_bits_per_sample(audiofilters_distortion_obj_t *self); | ||
|
||
mp_obj_t common_hal_audiofilters_distortion_get_drive(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_drive(audiofilters_distortion_obj_t *self, mp_obj_t arg); | ||
|
||
mp_obj_t common_hal_audiofilters_distortion_get_pre_gain(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_pre_gain(audiofilters_distortion_obj_t *self, mp_obj_t arg); | ||
|
||
mp_obj_t common_hal_audiofilters_distortion_get_post_gain(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_post_gain(audiofilters_distortion_obj_t *self, mp_obj_t arg); | ||
|
||
audiofilters_distortion_mode common_hal_audiofilters_distortion_get_mode(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_mode(audiofilters_distortion_obj_t *self, audiofilters_distortion_mode mode); | ||
|
||
bool common_hal_audiofilters_distortion_get_soft_clip(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_soft_clip(audiofilters_distortion_obj_t *self, bool soft_clip); | ||
|
||
mp_obj_t common_hal_audiofilters_distortion_get_mix(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_set_mix(audiofilters_distortion_obj_t *self, mp_obj_t arg); | ||
|
||
bool common_hal_audiofilters_distortion_get_playing(audiofilters_distortion_obj_t *self); | ||
void common_hal_audiofilters_distortion_play(audiofilters_distortion_obj_t *self, mp_obj_t sample, bool loop); | ||
void common_hal_audiofilters_distortion_stop(audiofilters_distortion_obj_t *self); |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -57,17 +57,17 @@ void common_hal_audiodelays_echo_construct(audiodelays_echo_obj_t *self, uint32_ | |
|
||
// If we did not receive a BlockInput we need to create a default float value | ||
if (decay == MP_OBJ_NULL) { | ||
decay = mp_obj_new_float(0.7); | ||
decay = mp_obj_new_float(MICROPY_FLOAT_CONST(0.7)); | ||
} | ||
synthio_block_assign_slot(decay, &self->decay, MP_QSTR_decay); | ||
|
||
if (delay_ms == MP_OBJ_NULL) { | ||
delay_ms = mp_obj_new_float(250.0); | ||
delay_ms = mp_obj_new_float(MICROPY_FLOAT_CONST(250.0)); | ||
} | ||
synthio_block_assign_slot(delay_ms, &self->delay_ms, MP_QSTR_delay_ms); | ||
|
||
if (mix == MP_OBJ_NULL) { | ||
mix = mp_obj_new_float(0.5); | ||
mix = mp_obj_new_float(MICROPY_FLOAT_CONST(0.5)); | ||
} | ||
synthio_block_assign_slot(mix, &self->mix, MP_QSTR_mix); | ||
|
||
|
@@ -77,7 +77,7 @@ void common_hal_audiodelays_echo_construct(audiodelays_echo_obj_t *self, uint32_ | |
|
||
// Allocate the echo buffer for the max possible delay, echo is always 16-bit | ||
self->max_delay_ms = max_delay_ms; | ||
self->max_echo_buffer_len = (uint32_t)(self->sample_rate / 1000.0f * max_delay_ms) * (self->channel_count * sizeof(uint16_t)); // bytes | ||
self->max_echo_buffer_len = (uint32_t)(self->sample_rate / MICROPY_FLOAT_CONST(1000.0) * max_delay_ms) * (self->channel_count * sizeof(uint16_t)); // bytes | ||
self->echo_buffer = m_malloc(self->max_echo_buffer_len); | ||
if (self->echo_buffer == NULL) { | ||
common_hal_audiodelays_echo_deinit(self); | ||
|
@@ -284,15 +284,6 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
channel = 0; | ||
} | ||
|
||
// get the effect values we need from the BlockInput. These may change at run time so you need to do bounds checking if required | ||
mp_float_t mix = MIN(1.0, MAX(synthio_block_slot_get(&self->mix), 0.0)); | ||
mp_float_t decay = MIN(1.0, MAX(synthio_block_slot_get(&self->decay), 0.0)); | ||
|
||
uint32_t delay_ms = (uint32_t)synthio_block_slot_get(&self->delay_ms); | ||
if (self->current_delay_ms != delay_ms) { | ||
recalculate_delay(self, delay_ms); | ||
} | ||
|
||
// Switch our buffers to the other buffer | ||
self->last_buf_idx = !self->last_buf_idx; | ||
|
||
|
@@ -303,16 +294,6 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
|
||
// The echo buffer is always stored as a 16-bit value internally | ||
int16_t *echo_buffer = (int16_t *)self->echo_buffer; | ||
uint32_t echo_buf_len = self->echo_buffer_len / sizeof(uint16_t); | ||
|
||
// Set our echo buffer position accounting for stereo | ||
uint32_t echo_buffer_pos = 0; | ||
if (self->freq_shift) { | ||
echo_buffer_pos = self->echo_buffer_left_pos; | ||
if (channel == 1) { | ||
echo_buffer_pos = self->echo_buffer_right_pos; | ||
} | ||
} | ||
|
||
// Loop over the entire length of our buffer to fill it, this may require several calls to get data from the sample | ||
while (length != 0) { | ||
|
@@ -334,9 +315,38 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
} | ||
} | ||
|
||
// Determine how many bytes we can process to our buffer, the less of the sample we have left and our buffer remaining | ||
uint32_t n; | ||
if (self->sample == NULL) { | ||
n = MIN(length, SYNTHIO_MAX_DUR * self->channel_count); | ||
} else { | ||
n = MIN(MIN(self->sample_buffer_length, length), SYNTHIO_MAX_DUR * self->channel_count); | ||
} | ||
|
||
// get the effect values we need from the BlockInput. These may change at run time so you need to do bounds checking if required | ||
shared_bindings_synthio_lfo_tick(self->sample_rate, n / self->channel_count); | ||
mp_float_t mix = synthio_block_slot_get_limited(&self->mix, MICROPY_FLOAT_CONST(0.0), MICROPY_FLOAT_CONST(1.0)); | ||
mp_float_t decay = synthio_block_slot_get_limited(&self->decay, MICROPY_FLOAT_CONST(0.0), MICROPY_FLOAT_CONST(1.0)); | ||
|
||
uint32_t delay_ms = (uint32_t)synthio_block_slot_get(&self->delay_ms); | ||
if (self->current_delay_ms != delay_ms) { | ||
recalculate_delay(self, delay_ms); | ||
} | ||
|
||
uint32_t echo_buf_len = self->echo_buffer_len / sizeof(uint16_t); | ||
|
||
// Set our echo buffer position accounting for stereo | ||
uint32_t echo_buffer_pos = 0; | ||
if (self->freq_shift) { | ||
echo_buffer_pos = self->echo_buffer_left_pos; | ||
if (channel == 1) { | ||
echo_buffer_pos = self->echo_buffer_right_pos; | ||
} | ||
} | ||
|
||
// If we have no sample keep the echo echoing | ||
if (self->sample == NULL) { | ||
if (mix <= 0.01) { // Mix of 0 is pure sample sound. We have no sample so no sound | ||
if (mix <= MICROPY_FLOAT_CONST(0.01)) { // Mix of 0 is pure sample sound. We have no sample so no sound | ||
if (self->samples_signed) { | ||
memset(word_buffer, 0, length * (self->bits_per_sample / 8)); | ||
} else { | ||
|
@@ -400,13 +410,10 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
length = 0; | ||
} else { | ||
// we have a sample to play and echo | ||
// Determine how many bytes we can process to our buffer, the less of the sample we have left and our buffer remaining | ||
uint32_t n = MIN(self->sample_buffer_length, length); | ||
|
||
int16_t *sample_src = (int16_t *)self->sample_remaining_buffer; // for 16-bit samples | ||
int8_t *sample_hsrc = (int8_t *)self->sample_remaining_buffer; // for 8-bit samples | ||
|
||
if (mix <= 0.01) { // if mix is zero pure sample only | ||
if (mix <= MICROPY_FLOAT_CONST(0.01)) { // if mix is zero pure sample only | ||
for (uint32_t i = 0; i < n; i++) { | ||
if (MP_LIKELY(self->bits_per_sample == 16)) { | ||
word_buffer[i] = sample_src[i]; | ||
|
@@ -467,12 +474,12 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
word = echo + sample_word; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If you happen to do a commit before the final review and have a moment after this line could you add: So the final chunk is:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
we've now got 3 copies of mix_down_sample. If not as part of this PR, please make this have extern linkage somewhere and get rid of the duplicates. |
||
|
||
if (MP_LIKELY(self->bits_per_sample == 16)) { | ||
word_buffer[i] = (int16_t)((sample_word * (1.0 - mix)) + (word * mix)); | ||
word_buffer[i] = (int16_t)((sample_word * (MICROPY_FLOAT_CONST(1.0) - mix)) + (word * mix)); | ||
if (!self->samples_signed) { | ||
word_buffer[i] ^= 0x8000; | ||
} | ||
} else { | ||
int8_t mixed = (int16_t)((sample_word * (1.0 - mix)) + (word * mix)); | ||
int8_t mixed = (int16_t)((sample_word * (MICROPY_FLOAT_CONST(1.0) - mix)) + (word * mix)); | ||
if (self->samples_signed) { | ||
hword_buffer[i] = mixed; | ||
} else { | ||
|
@@ -500,13 +507,13 @@ audioio_get_buffer_result_t audiodelays_echo_get_buffer(audiodelays_echo_obj_t * | |
self->sample_remaining_buffer += (n * (self->bits_per_sample / 8)); | ||
self->sample_buffer_length -= n; | ||
} | ||
} | ||
|
||
if (self->freq_shift) { | ||
if (channel == 0) { | ||
self->echo_buffer_left_pos = echo_buffer_pos; | ||
} else if (channel == 1) { | ||
self->echo_buffer_right_pos = echo_buffer_pos; | ||
if (self->freq_shift) { | ||
if (channel == 0) { | ||
self->echo_buffer_left_pos = echo_buffer_pos; | ||
} else if (channel == 1) { | ||
self->echo_buffer_right_pos = echo_buffer_pos; | ||
} | ||
} | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We may want to check that delay >= 0. Never did in the original code (my bad) but a LFO could go negative and weird things happen.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Casting it to an unsigned integer does force it to be >= 0, but a negative result might cause it to wrap around. If it's 0, I believe it will do a minimum of 1 sample of delay. I'll look into it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
maybe it should be
synthio_block_slot_get_limited(1, some_calculated_maximum)
then?Technically under the C99 standard, conversion from a negative floating point value to an unsigned integer value is undefined behavior.