diff --git a/src/codal_app/main.cpp b/src/codal_app/main.cpp index 00e91a9..c5f82e0 100644 --- a/src/codal_app/main.cpp +++ b/src/codal_app/main.cpp @@ -75,6 +75,9 @@ int main() { uBit.audio.setSpeakerEnabled(true); uBit.audio.setPinEnabled(false); + // Set the microphone gain to a reasonable value. + uBit.audio.processor->setGain(0.2); + mp_main(); return 0; } diff --git a/src/codal_app/microbithal.h b/src/codal_app/microbithal.h index 78d455f..78c75e9 100644 --- a/src/codal_app/microbithal.h +++ b/src/codal_app/microbithal.h @@ -165,9 +165,13 @@ int microbit_hal_compass_get_field_strength(void); int microbit_hal_compass_get_heading(void); void microbit_hal_microphone_init(void); +void microbit_hal_microphone_set_sensitivity(float value); void microbit_hal_microphone_set_threshold(int kind, int value); int microbit_hal_microphone_get_level(void); float microbit_hal_microphone_get_level_db(void); +void microbit_hal_microphone_start_recording(uint8_t *buf, size_t max_len, size_t *cur_len, int rate); +bool microbit_hal_microphone_is_recording(void); +void microbit_hal_microphone_stop_recording(void); const uint8_t *microbit_hal_get_font_data(char c); @@ -181,13 +185,15 @@ int microbit_hal_log_data(const char *key, const char *value); void microbit_hal_audio_select_pin(int pin); void microbit_hal_audio_select_speaker(bool enable); void microbit_hal_audio_set_volume(int value); +bool microbit_hal_audio_is_playing(void); bool microbit_hal_audio_is_expression_active(void); void microbit_hal_audio_play_expression(const char *expr); void microbit_hal_audio_stop_expression(void); -void microbit_hal_audio_init(uint32_t sample_rate); -void microbit_hal_audio_write_data(const uint8_t *buf, size_t num_samples); -void microbit_hal_audio_ready_callback(void); +void microbit_hal_audio_raw_init(uint32_t sample_rate); +void microbit_hal_audio_raw_set_rate(uint32_t sample_rate); +void microbit_hal_audio_raw_write_data(const uint8_t *buf, size_t num_samples); +void microbit_hal_audio_raw_ready_callback(void); void microbit_hal_audio_speech_init(uint32_t sample_rate); void microbit_hal_audio_speech_write_data(const uint8_t *buf, size_t num_samples); diff --git a/src/codal_app/microbithal_audio.cpp b/src/codal_app/microbithal_audio.cpp index d27b429..b60c172 100644 --- a/src/codal_app/microbithal_audio.cpp +++ b/src/codal_app/microbithal_audio.cpp @@ -53,7 +53,7 @@ class AudioSource : public DataSource { } }; -static AudioSource data_source; +static AudioSource raw_source; static AudioSource speech_source; extern "C" { @@ -86,6 +86,10 @@ void microbit_hal_sound_synth_callback(int event) { } } +bool microbit_hal_audio_is_playing(void) { + return uBit.audio.isPlaying(); +} + bool microbit_hal_audio_is_expression_active(void) { return sound_synth_active_count > 0; } @@ -105,23 +109,27 @@ void microbit_hal_audio_stop_expression(void) { uBit.audio.soundExpressions.stop(); } -void microbit_hal_audio_init(uint32_t sample_rate) { - if (!data_source.started) { +void microbit_hal_audio_raw_init(uint32_t sample_rate) { + if (!raw_source.started) { MicroBitAudio::requestActivation(); - data_source.started = true; - data_source.callback = microbit_hal_audio_ready_callback; - data_source.channel = uBit.audio.mixer.addChannel(data_source, sample_rate, 255); + raw_source.started = true; + raw_source.callback = microbit_hal_audio_raw_ready_callback; + raw_source.channel = uBit.audio.mixer.addChannel(raw_source, sample_rate, 255); } else { - data_source.channel->setSampleRate(sample_rate); + raw_source.channel->setSampleRate(sample_rate); } } -void microbit_hal_audio_write_data(const uint8_t *buf, size_t num_samples) { - if ((size_t)data_source.buf.length() != num_samples) { - data_source.buf = ManagedBuffer(num_samples); +void microbit_hal_audio_raw_set_rate(uint32_t sample_rate) { + raw_source.channel->setSampleRate(sample_rate); +} + +void microbit_hal_audio_raw_write_data(const uint8_t *buf, size_t num_samples) { + if ((size_t)raw_source.buf.length() != num_samples) { + raw_source.buf = ManagedBuffer(num_samples); } - memcpy(data_source.buf.getBytes(), buf, num_samples); - data_source.sink->pullRequest(); + memcpy(raw_source.buf.getBytes(), buf, num_samples); + raw_source.sink->pullRequest(); } void microbit_hal_audio_speech_init(uint32_t sample_rate) { diff --git a/src/codal_app/microbithal_microphone.cpp b/src/codal_app/microbithal_microphone.cpp index e530432..9455b0e 100644 --- a/src/codal_app/microbithal_microphone.cpp +++ b/src/codal_app/microbithal_microphone.cpp @@ -28,12 +28,65 @@ #include "microbithal.h" #include "MicroBitDevice.h" +#define MIN(a, b) ((a) < (b) ? (a) : (b)) + extern "C" void microbit_hal_level_detector_callback(int); static void level_detector_event_handler(Event evt) { microbit_hal_level_detector_callback(evt.value); } +class MyStreamRecording : public DataSink +{ + public: + SplitterChannel *upStream; + + public: + uint8_t *dest; + size_t *dest_pos_ptr; + size_t dest_max; + bool request_stop; + + MyStreamRecording(SplitterChannel *source); + virtual ~MyStreamRecording(); + + virtual int pullRequest(); +}; + +MyStreamRecording::MyStreamRecording(SplitterChannel *source) : upStream(source) +{ +} + +MyStreamRecording::~MyStreamRecording() +{ +} + +int MyStreamRecording::pullRequest() +{ + uint8_t *pull_buf = this->dest + *this->dest_pos_ptr; + size_t n = this->dest_max - *this->dest_pos_ptr; + + if (n > 0) { + n = this->upStream->pullInto(pull_buf, n) - pull_buf; + } + + if (n == 0 || this->request_stop) { + this->upStream->disconnect(); + this->request_stop = false; + } else { + // Convert signed 8-bit to unsigned 8-bit data. + for (size_t i = 0; i < n; ++i) { + pull_buf[i] += 128; + } + *this->dest_pos_ptr += n; + } + + return DEVICE_OK; +} + +static MyStreamRecording *recording = NULL; +static SplitterChannel *splitterChannel = NULL; + extern "C" { static bool microphone_init_done = false; @@ -46,6 +99,10 @@ void microbit_hal_microphone_init(void) { } } +void microbit_hal_microphone_set_sensitivity(float value) { + uBit.audio.processor->setGain(value); +} + void microbit_hal_microphone_set_threshold(int kind, int value) { if (kind == MICROBIT_HAL_MICROPHONE_SET_THRESHOLD_LOW) { uBit.audio.levelSPL->setLowThreshold(value); @@ -66,4 +123,41 @@ float microbit_hal_microphone_get_level_db(void) { return value; } +void microbit_hal_microphone_start_recording(uint8_t *buf, size_t max_len, size_t *cur_len, int rate) { + if (splitterChannel == NULL) { + splitterChannel = uBit.audio.splitter->createChannel(); + splitterChannel->setFormat(DATASTREAM_FORMAT_8BIT_UNSIGNED); + } + splitterChannel->requestSampleRate(rate); + + if (recording == NULL) { + recording = new MyStreamRecording(splitterChannel); + } else { + if (microbit_hal_microphone_is_recording()) { + microbit_hal_microphone_stop_recording(); + while (microbit_hal_microphone_is_recording()) { + microbit_hal_idle(); + } + } + } + + recording->dest = buf; + recording->dest_pos_ptr = cur_len; + *recording->dest_pos_ptr = 0; + recording->dest_max = max_len; + recording->request_stop = false; + + splitterChannel->connect(*recording); +} + +bool microbit_hal_microphone_is_recording(void) { + return recording != NULL && splitterChannel->isConnected(); +} + +void microbit_hal_microphone_stop_recording(void) { + if (recording != NULL) { + recording->request_stop = true; + } +} + } diff --git a/src/codal_port/Makefile b/src/codal_port/Makefile index c6b4643..64bd213 100644 --- a/src/codal_port/Makefile +++ b/src/codal_port/Makefile @@ -69,6 +69,8 @@ SRC_C += \ iters.c \ main.c \ microbit_accelerometer.c \ + microbit_audiorecording.c \ + microbit_audiotrack.c \ microbit_button.c \ microbit_compass.c \ microbit_display.c \ @@ -100,6 +102,7 @@ SRC_C += \ modspeech.c \ modthis.c \ mphalport.c \ + utils.c \ SRC_C += \ shared/readline/readline.c \ diff --git a/src/codal_port/microbit_audiorecording.c b/src/codal_port/microbit_audiorecording.c new file mode 100644 index 0000000..6de7974 --- /dev/null +++ b/src/codal_port/microbit_audiorecording.c @@ -0,0 +1,128 @@ +/* + * This file is part of the MicroPython project, http://micropython.org/ + * + * The MIT License (MIT) + * + * Copyright (c) 2024 Damien P. George + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +#include "py/mphal.h" +#include "drv_system.h" +#include "modmicrobit.h" +#include "modaudio.h" +#include "utils.h" + +mp_obj_t microbit_audio_recording_new(size_t num_bytes, uint32_t rate) { + // Make sure size is non-zero. + if (num_bytes == 0) { + num_bytes = 1; + } + + // Create and return the AudioRecording object. + uint8_t *data = m_new(uint8_t, num_bytes); + memset(data, 128, num_bytes); + return microbit_audio_track_new(MP_OBJ_NULL, num_bytes, data, rate); +} + +static mp_obj_t microbit_audio_recording_make_new(const mp_obj_type_t *type_in, size_t n_args, size_t n_kw, const mp_obj_t *all_args) { + (void)type_in; + + enum { ARG_duration, ARG_rate }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_duration, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_rom_obj = MP_OBJ_NULL} }, + { MP_QSTR_rate, MP_ARG_OBJ, {.u_rom_obj = MP_ROM_INT(AUDIO_TRACK_DEFAULT_SAMPLE_RATE)} }, + }; + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all_kw_array(n_args, n_kw, all_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + mp_int_t rate = mp_obj_get_int_allow_float(args[ARG_rate].u_obj); + if (rate <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + + mp_float_t duration_ms = mp_obj_get_float(args[ARG_duration].u_obj); + if (duration_ms <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("duration out of bounds")); + } + size_t num_bytes = duration_ms * rate / 1000; + + return microbit_audio_recording_new(num_bytes, rate); +} + +static mp_obj_t microbit_audio_recording_copy(mp_obj_t self_in) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + uint8_t *data = m_new(uint8_t, self->size); + memcpy(data, self->data, self->size); + return microbit_audio_track_new(MP_OBJ_NULL, self->size, data, self->rate); +} +static MP_DEFINE_CONST_FUN_OBJ_1(microbit_audio_recording_copy_obj, microbit_audio_recording_copy); + +static mp_obj_t microbit_audio_recording_track(size_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) { + enum { ARG_start_ms, ARG_end_ms }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_start_ms, MP_ARG_OBJ, {.u_rom_obj = MP_ROM_INT(0)} }, + { MP_QSTR_end_ms, MP_ARG_OBJ, {.u_rom_obj = MP_ROM_INT(-1)} }, + }; + // parse args + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all(n_args - 1, pos_args + 1, kw_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(pos_args[0]); + mp_int_t start_byte = mp_obj_get_float(args[ARG_start_ms].u_obj) * self->rate / 1000; + mp_int_t end_byte; + if (args[ARG_end_ms].u_obj == MP_OBJ_NEW_SMALL_INT(-1)) { + end_byte = self->size; + } else { + end_byte = mp_obj_get_float(args[ARG_end_ms].u_obj) * self->rate / 1000; + } + + // Truncate start_byte to fit in valid range. + start_byte = MAX(0, start_byte); + start_byte = MIN(start_byte, self->size); + + // Truncate end_byte to fit in valid range. + end_byte = MAX(0, end_byte); + end_byte = MIN(end_byte, self->size); + + // Calculate length of track, truncating negative lengths to 0. + size_t len = MAX(0, end_byte - start_byte); + + // Create and return new track. + return microbit_audio_track_new(pos_args[0], len, self->data + start_byte, self->rate); +} +static MP_DEFINE_CONST_FUN_OBJ_KW(microbit_audio_recording_track_obj, 1, microbit_audio_recording_track); + +static const mp_rom_map_elem_t microbit_audio_recording_locals_dict_table[] = { + { MP_ROM_QSTR(MP_QSTR_get_rate), MP_ROM_PTR(µbit_audio_track_get_rate_obj) }, + { MP_ROM_QSTR(MP_QSTR_set_rate), MP_ROM_PTR(µbit_audio_track_set_rate_obj) }, + { MP_ROM_QSTR(MP_QSTR_copy), MP_ROM_PTR(µbit_audio_recording_copy_obj) }, + { MP_ROM_QSTR(MP_QSTR_track), MP_ROM_PTR(µbit_audio_recording_track_obj) }, +}; +static MP_DEFINE_CONST_DICT(microbit_audio_recording_locals_dict, microbit_audio_recording_locals_dict_table); + +MP_DEFINE_CONST_OBJ_TYPE( + microbit_audio_recording_type, + MP_QSTR_AudioRecording, + MP_TYPE_FLAG_NONE, + make_new, microbit_audio_recording_make_new, + buffer, microbit_audio_track_get_buffer, + locals_dict, µbit_audio_recording_locals_dict + ); diff --git a/src/codal_port/microbit_audiotrack.c b/src/codal_port/microbit_audiotrack.c new file mode 100644 index 0000000..a645197 --- /dev/null +++ b/src/codal_port/microbit_audiotrack.c @@ -0,0 +1,199 @@ +/* + * This file is part of the MicroPython project, http://micropython.org/ + * + * The MIT License (MIT) + * + * Copyright (c) 2024 Damien P. George + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +#include "py/mphal.h" +#include "drv_system.h" +#include "modmicrobit.h" +#include "modaudio.h" +#include "utils.h" + +mp_obj_t microbit_audio_track_new(mp_obj_t buffer_obj, size_t len, uint8_t *data, uint32_t rate) { + microbit_audio_track_obj_t *self = m_new_obj(microbit_audio_track_obj_t); + if (buffer_obj == MP_OBJ_NULL) { + self->base.type = µbit_audio_recording_type; + } else { + self->base.type = µbit_audio_track_type; + if (mp_obj_is_type(buffer_obj, µbit_audio_track_type)) { + buffer_obj = ((microbit_audio_track_obj_t *)MP_OBJ_TO_PTR(buffer_obj))->buffer_obj; + } + } + self->buffer_obj = buffer_obj; + self->size = len; + self->rate = rate; + self->data = data; + return MP_OBJ_FROM_PTR(self); +} + +static mp_obj_t microbit_audio_track_make_new(const mp_obj_type_t *type_in, size_t n_args, size_t n_kw, const mp_obj_t *all_args) { + (void)type_in; + + enum { ARG_buffer, ARG_rate }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_buffer, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_rom_obj = MP_OBJ_NULL} }, + { MP_QSTR_rate, MP_ARG_OBJ, {.u_rom_obj = MP_ROM_INT(AUDIO_TRACK_DEFAULT_SAMPLE_RATE)} }, + }; + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all_kw_array(n_args, n_kw, all_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + mp_int_t rate = mp_obj_get_int_allow_float(args[ARG_rate].u_obj); + if (rate <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + + // Get buffer. + mp_buffer_info_t bufinfo; + mp_get_buffer_raise(args[ARG_buffer].u_obj, &bufinfo, MP_BUFFER_RW); + + // Create and return the AudioTrack object. + return microbit_audio_track_new(args[ARG_buffer].u_obj, bufinfo.len, bufinfo.buf, rate); +} + +static mp_obj_t microbit_audio_track_unary_op(mp_unary_op_t op, mp_obj_t self_in) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + switch (op) { + case MP_UNARY_OP_LEN: + return MP_OBJ_NEW_SMALL_INT(self->size); + default: + return MP_OBJ_NULL; // op not supported + } +} + +static mp_obj_t microbit_audio_track_binary_op(mp_binary_op_t op, mp_obj_t lhs_in, mp_obj_t rhs_in) { + microbit_audio_track_obj_t *lhs = MP_OBJ_TO_PTR(lhs_in); + + // Make a copy of LHS if the operation is not inplace. + if (op == MP_BINARY_OP_ADD || op == MP_BINARY_OP_SUBTRACT || op == MP_BINARY_OP_MULTIPLY) { + microbit_audio_track_obj_t *lhs_copy = microbit_audio_track_new(lhs->buffer_obj, lhs->size, lhs->data, lhs->rate); + memcpy(lhs_copy->data, lhs->data, lhs->size); + lhs = lhs_copy; + } + + switch(op) { + case MP_BINARY_OP_ADD: + case MP_BINARY_OP_SUBTRACT: + case MP_BINARY_OP_INPLACE_ADD: + case MP_BINARY_OP_INPLACE_SUBTRACT: + if (mp_obj_get_type(rhs_in) != µbit_audio_track_type) { + return MP_OBJ_NULL; // op not supported + } + microbit_audio_track_obj_t *rhs = MP_OBJ_TO_PTR(rhs_in); + size_t size = MIN(lhs->size, rhs->size); + microbit_audio_data_add_inplace(lhs->data, rhs->data, size, op == MP_BINARY_OP_ADD || op == MP_BINARY_OP_INPLACE_ADD); + return lhs; + case MP_BINARY_OP_MULTIPLY: + case MP_BINARY_OP_INPLACE_MULTIPLY: + microbit_audio_data_mult_inplace(lhs->data, lhs->size, mp_obj_get_float(rhs_in)); + return lhs; + default: + return MP_OBJ_NULL; // op not supported + } +} + +static mp_obj_t microbit_audio_track_subscr(mp_obj_t self_in, mp_obj_t index, mp_obj_t value_in) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + if (value_in == MP_OBJ_NULL) { + // delete + mp_raise_TypeError(MP_ERROR_TEXT("cannot delete elements of AudioTrack")); + } else if (value_in == MP_OBJ_SENTINEL) { + // load + if (mp_obj_is_type(index, &mp_type_slice)) { + mp_bound_slice_t slice; + if (!mp_seq_get_fast_slice_indexes(self->size, index, &slice)) { + mp_raise_NotImplementedError(MP_ERROR_TEXT("slices must have step=1")); + } + return microbit_audio_track_new(self->buffer_obj, slice.stop - slice.start, self->data + slice.start, self->rate); + } else { + size_t index_val = mp_get_index(self->base.type, self->size, index, false); + return MP_OBJ_NEW_SMALL_INT(self->data[index_val]); + } + } else { + // store + size_t index_val = mp_get_index(self->base.type, self->size, index, false); + mp_int_t value = mp_obj_get_int(value_in); + if (value < 0 || value > 255) { + mp_raise_ValueError(MP_ERROR_TEXT("value out of range")); + } + self->data[index_val] = value; + return mp_const_none; + } +} + +mp_int_t microbit_audio_track_get_buffer(mp_obj_t self_in, mp_buffer_info_t *bufinfo, mp_uint_t flags) { + (void)flags; + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + bufinfo->buf = self->data; + bufinfo->len = self->size; + bufinfo->typecode = 'B'; + return 0; +} + +static mp_obj_t microbit_audio_track_get_rate(mp_obj_t self_in) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + return MP_OBJ_NEW_SMALL_INT(self->rate); +} +MP_DEFINE_CONST_FUN_OBJ_1(microbit_audio_track_get_rate_obj, microbit_audio_track_get_rate); + +static mp_obj_t microbit_audio_track_set_rate(mp_obj_t self_in, mp_obj_t rate_in) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + mp_int_t rate = mp_obj_get_int_allow_float(rate_in); + if (rate <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + self->rate = rate; + // TODO: only set if this frame is currently being played + microbit_hal_audio_raw_set_rate(rate); + return mp_const_none; +} +MP_DEFINE_CONST_FUN_OBJ_2(microbit_audio_track_set_rate_obj, microbit_audio_track_set_rate); + +static mp_obj_t microbit_audio_track_copyfrom(mp_obj_t self_in, mp_obj_t other) { + microbit_audio_track_obj_t *self = MP_OBJ_TO_PTR(self_in); + mp_buffer_info_t bufinfo; + mp_get_buffer_raise(other, &bufinfo, MP_BUFFER_READ); + uint32_t len = MIN(bufinfo.len, self->size); + memcpy(self->data, bufinfo.buf, len); + return mp_const_none; +} +static MP_DEFINE_CONST_FUN_OBJ_2(microbit_audio_track_copyfrom_obj, microbit_audio_track_copyfrom); + +static const mp_rom_map_elem_t microbit_audio_track_locals_dict_table[] = { + { MP_ROM_QSTR(MP_QSTR_get_rate), MP_ROM_PTR(µbit_audio_track_get_rate_obj) }, + { MP_ROM_QSTR(MP_QSTR_set_rate), MP_ROM_PTR(µbit_audio_track_set_rate_obj) }, + { MP_ROM_QSTR(MP_QSTR_copyfrom), MP_ROM_PTR(µbit_audio_track_copyfrom_obj) }, +}; +static MP_DEFINE_CONST_DICT(microbit_audio_track_locals_dict, microbit_audio_track_locals_dict_table); + +MP_DEFINE_CONST_OBJ_TYPE( + microbit_audio_track_type, + MP_QSTR_AudioTrack, + MP_TYPE_FLAG_NONE, + make_new, microbit_audio_track_make_new, + unary_op, microbit_audio_track_unary_op, + binary_op, microbit_audio_track_binary_op, + subscr, microbit_audio_track_subscr, + buffer, microbit_audio_track_get_buffer, + locals_dict, µbit_audio_track_locals_dict + ); diff --git a/src/codal_port/microbit_microphone.c b/src/codal_port/microbit_microphone.c index cf37069..b3ccc79 100644 --- a/src/codal_port/microbit_microphone.c +++ b/src/codal_port/microbit_microphone.c @@ -26,6 +26,7 @@ #include "py/runtime.h" #include "py/mphal.h" +#include "modaudio.h" #include "modmicrobit.h" #define EVENT_HISTORY_SIZE (8) @@ -84,6 +85,13 @@ static uint8_t sound_event_from_obj(mp_obj_t sound) { mp_raise_ValueError(MP_ERROR_TEXT("invalid sound")); } +static mp_obj_t microbit_microphone_set_sensitivity(mp_obj_t self_in, mp_obj_t value_in) { + (void)self_in; + microbit_hal_microphone_set_sensitivity(mp_obj_get_float(value_in)); + return mp_const_none; +} +static MP_DEFINE_CONST_FUN_OBJ_2(microbit_microphone_set_sensitivity_obj, microbit_microphone_set_sensitivity); + static mp_obj_t microbit_microphone_set_threshold(mp_obj_t self_in, mp_obj_t sound_in, mp_obj_t value_in) { (void)self_in; uint8_t sound = sound_event_from_obj(sound_in); @@ -158,7 +166,114 @@ static mp_obj_t microbit_microphone_get_events(mp_obj_t self_in) { } static MP_DEFINE_CONST_FUN_OBJ_1(microbit_microphone_get_events_obj, microbit_microphone_get_events); +static void microbit_microphone_record_helper(microbit_audio_track_obj_t *audio_track, bool wait) { + // Start the recording. + microbit_hal_microphone_start_recording(audio_track->data, audio_track->size, &audio_track->size, audio_track->rate); + + if (wait) { + // Wait for the recording to finish. + while (microbit_hal_microphone_is_recording()) { + mp_handle_pending(true); + microbit_hal_idle(); + } + } +} + +static mp_obj_t microbit_microphone_record(mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) { + enum { ARG_duration, ARG_rate, }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_duration, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_rom_obj = MP_ROM_NONE} }, + { MP_QSTR_rate, MP_ARG_INT, {.u_int = AUDIO_TRACK_DEFAULT_SAMPLE_RATE} }, + }; + + // Parse the args. + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all(n_args - 1, pos_args + 1, kw_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + // Validate arguments. + mp_float_t duration = mp_obj_get_float(args[ARG_duration].u_obj); + if (duration <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("duration out of bounds")); + } + if (args[ARG_rate].u_int <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + + // Create the AudioRecording to record into. + size_t size = duration * args[ARG_rate].u_int / 1000; + microbit_audio_track_obj_t *audio_track = MP_OBJ_TO_PTR(microbit_audio_recording_new(size, args[ARG_rate].u_int)); + + // Start recording and wait. + microbit_microphone_record_helper(audio_track, true); + + // Return the new AudioFrame. + return MP_OBJ_FROM_PTR(audio_track); +} +static MP_DEFINE_CONST_FUN_OBJ_KW(microbit_microphone_record_obj, 1, microbit_microphone_record); + +static mp_obj_t microbit_microphone_record_into(mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) { + enum { ARG_buffer, ARG_wait }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_buffer, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_obj = MP_OBJ_NULL} }, + { MP_QSTR_wait, MP_ARG_KW_ONLY | MP_ARG_BOOL, {.u_bool = true} }, + }; + + // Parse the args. + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all(n_args - 1, pos_args + 1, kw_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + // Check that the buffer is an AudioFrame instance. + // TODO allow both AudioRecording and AudioTrack? yes + if (!mp_obj_is_type(args[ARG_buffer].u_obj, µbit_audio_track_type) + && !mp_obj_is_type(args[ARG_buffer].u_obj, µbit_audio_recording_type)) { + mp_raise_TypeError(MP_ERROR_TEXT("expecting an AudioTrack or AudioRecording")); + } + microbit_audio_track_obj_t *audio_track = MP_OBJ_TO_PTR(args[ARG_buffer].u_obj); + + // Create the AudioTrack to record into. + microbit_audio_track_obj_t *audio_track_new = MP_OBJ_TO_PTR(microbit_audio_track_new(args[ARG_buffer].u_obj, audio_track->size, audio_track->data, audio_track->rate)); + + // Start recording and wait if requested. + microbit_microphone_record_helper(audio_track_new, args[ARG_wait].u_bool); + + // Return the new AudioTrack. + return MP_OBJ_FROM_PTR(audio_track_new); +} +static MP_DEFINE_CONST_FUN_OBJ_KW(microbit_microphone_record_into_obj, 1, microbit_microphone_record_into); + +static mp_obj_t microbit_microphone_is_recording(mp_obj_t self_in) { + (void)self_in; + return mp_obj_new_bool(microbit_hal_microphone_is_recording()); +} +static MP_DEFINE_CONST_FUN_OBJ_1(microbit_microphone_is_recording_obj, microbit_microphone_is_recording); + +static mp_obj_t microbit_microphone_stop_recording(mp_obj_t self_in) { + (void)self_in; + microbit_hal_microphone_stop_recording(); + return mp_const_none; +} +static MP_DEFINE_CONST_FUN_OBJ_1(microbit_microphone_stop_recording_obj, microbit_microphone_stop_recording); + +#if MICROPY_OBJ_REPR == MICROPY_OBJ_REPR_A || MICROPY_OBJ_REPR == MICROPY_OBJ_REPR_B + +// The way float objects are defined depends on the object representation. Here we only +// support representation A and B which use the following struct for float objects. +// This is defined in py/objfloat.c and not exposed publicly, so must be defined here. + +typedef struct _mp_obj_float_t { + mp_obj_base_t base; + mp_float_t value; +} mp_obj_float_t; + +static const mp_obj_float_t microbit_microphone_sensitivity_low_obj = {{&mp_type_float}, (mp_float_t)0.079}; +static const mp_obj_float_t microbit_microphone_sensitivity_medium_obj = {{&mp_type_float}, (mp_float_t)0.2}; +static const mp_obj_float_t microbit_microphone_sensitivity_high_obj = {{&mp_type_float}, (mp_float_t)1.0}; + +#endif + static const mp_rom_map_elem_t microbit_microphone_locals_dict_table[] = { + // Methods. + { MP_ROM_QSTR(MP_QSTR_set_sensitivity), MP_ROM_PTR(µbit_microphone_set_sensitivity_obj) }, { MP_ROM_QSTR(MP_QSTR_set_threshold), MP_ROM_PTR(µbit_microphone_set_threshold_obj) }, { MP_ROM_QSTR(MP_QSTR_sound_level), MP_ROM_PTR(µbit_microphone_sound_level_obj) }, { MP_ROM_QSTR(MP_QSTR_sound_level_db), MP_ROM_PTR(µbit_microphone_sound_level_db_obj) }, @@ -166,6 +281,15 @@ static const mp_rom_map_elem_t microbit_microphone_locals_dict_table[] = { { MP_ROM_QSTR(MP_QSTR_is_event), MP_ROM_PTR(µbit_microphone_is_event_obj) }, { MP_ROM_QSTR(MP_QSTR_was_event), MP_ROM_PTR(µbit_microphone_was_event_obj) }, { MP_ROM_QSTR(MP_QSTR_get_events), MP_ROM_PTR(µbit_microphone_get_events_obj) }, + { MP_ROM_QSTR(MP_QSTR_record), MP_ROM_PTR(µbit_microphone_record_obj) }, + { MP_ROM_QSTR(MP_QSTR_record_into), MP_ROM_PTR(µbit_microphone_record_into_obj) }, + { MP_ROM_QSTR(MP_QSTR_is_recording), MP_ROM_PTR(µbit_microphone_is_recording_obj) }, + { MP_ROM_QSTR(MP_QSTR_stop_recording), MP_ROM_PTR(µbit_microphone_stop_recording_obj) }, + + // Constants. + { MP_ROM_QSTR(MP_QSTR_SENSITIVITY_LOW), MP_ROM_PTR(µbit_microphone_sensitivity_low_obj) }, + { MP_ROM_QSTR(MP_QSTR_SENSITIVITY_MEDIUM), MP_ROM_PTR(µbit_microphone_sensitivity_medium_obj) }, + { MP_ROM_QSTR(MP_QSTR_SENSITIVITY_HIGH), MP_ROM_PTR(µbit_microphone_sensitivity_high_obj) }, }; static MP_DEFINE_CONST_DICT(microbit_microphone_locals_dict, microbit_microphone_locals_dict_table); diff --git a/src/codal_port/modaudio.c b/src/codal_port/modaudio.c index 5962c22..f31ff88 100644 --- a/src/codal_port/modaudio.c +++ b/src/codal_port/modaudio.c @@ -25,16 +25,23 @@ * THE SOFTWARE. */ +#include #include "py/mphal.h" #include "drv_system.h" #include "modaudio.h" #include "modmicrobit.h" -#define audio_source_iter MP_STATE_PORT(audio_source) +// Convenience macros to access the root-pointer state. +#define audio_source_frame MP_STATE_PORT(audio_source_frame_state) +#define audio_source_track MP_STATE_PORT(audio_source_track_state) +#define audio_source_iter MP_STATE_PORT(audio_source_iter_state) +#ifndef AUDIO_OUTPUT_BUFFER_SIZE +#define AUDIO_OUTPUT_BUFFER_SIZE (64) +#endif + +#define DEFAULT_AUDIO_FRAME_SIZE (32) #define DEFAULT_SAMPLE_RATE (7812) -#define BUFFER_EXPANSION (4) // smooth out the samples via linear interpolation -#define OUT_CHUNK_SIZE (BUFFER_EXPANSION * AUDIO_CHUNK_SIZE) typedef enum { AUDIO_OUTPUT_STATE_IDLE, @@ -42,98 +49,180 @@ typedef enum { AUDIO_OUTPUT_STATE_DATA_WRITTEN, } audio_output_state_t; -static uint8_t audio_output_buffer[OUT_CHUNK_SIZE]; +static uint8_t audio_output_buffer[AUDIO_OUTPUT_BUFFER_SIZE]; +static size_t audio_output_buffer_offset; static volatile audio_output_state_t audio_output_state; -static volatile bool audio_fetcher_scheduled; +static size_t audio_source_frame_offset; +static uint32_t audio_current_sound_level; +static mp_sched_node_t audio_data_fetcher_sched_node; -microbit_audio_frame_obj_t *microbit_audio_frame_make_new(void); +static void audio_data_schedule_fetch(void); static inline bool audio_is_running(void) { - return audio_source_iter != NULL; + return audio_source_frame != NULL || audio_source_track != NULL || audio_source_iter != MP_OBJ_NULL; } void microbit_audio_stop(void) { - audio_source_iter = NULL; + audio_output_buffer_offset = 0; + audio_source_frame = NULL; + audio_source_track = NULL; + audio_source_iter = MP_OBJ_NULL; + audio_source_frame_offset = 0; + audio_current_sound_level = 0; microbit_hal_audio_stop_expression(); } -static void audio_buffer_ready(void) { - uint32_t atomic_state = MICROPY_BEGIN_ATOMIC_SECTION(); - audio_output_state_t old_state = audio_output_state; - audio_output_state = AUDIO_OUTPUT_STATE_DATA_READY; - MICROPY_END_ATOMIC_SECTION(atomic_state); - if (old_state == AUDIO_OUTPUT_STATE_IDLE) { - microbit_hal_audio_ready_callback(); +static void audio_data_pull_from_source(void) { + if (audio_source_frame != NULL) { + // An existing AudioFrame is being played, see if there's any data left. + if (audio_source_frame_offset >= audio_source_frame->alloc_size) { + // AudioFrame is exhausted. + audio_source_frame = NULL; + } + } else if (audio_source_track != NULL) { + // An existing AudioTrack is being played, see if there's any data left. + if (audio_source_frame_offset >= audio_source_track->size) { + // AudioTrack is exhausted. + audio_source_track = NULL; + } } -} -static void audio_data_fetcher(void) { - audio_fetcher_scheduled = false; - if (audio_source_iter == NULL) { - return; + if (audio_source_frame == NULL && audio_source_track == NULL) { + // There is no AudioFrame/AudioTrack, so try to get one from the audio iterator. + + if (audio_source_iter == MP_OBJ_NULL) { + // Audio iterator is already exhausted. + return; + } + + // Get the next item from the audio iterator. + nlr_buf_t nlr; + mp_obj_t frame_obj; + if (nlr_push(&nlr) == 0) { + frame_obj = mp_iternext_allow_raise(audio_source_iter); + nlr_pop(); + } else { + if (!mp_obj_is_subclass_fast(MP_OBJ_FROM_PTR(((mp_obj_base_t*)nlr.ret_val)->type), + MP_OBJ_FROM_PTR(&mp_type_StopIteration))) { + mp_sched_exception(MP_OBJ_FROM_PTR(nlr.ret_val)); + } + frame_obj = MP_OBJ_STOP_ITERATION; + } + if (frame_obj == MP_OBJ_STOP_ITERATION) { + // End of audio iterator. + audio_source_iter = MP_OBJ_NULL; + return; + } + + if (mp_obj_is_type(frame_obj, µbit_audio_frame_type)) { + // We have the next AudioFrame. + audio_source_frame = MP_OBJ_TO_PTR(frame_obj); + audio_source_frame_offset = 0; + microbit_hal_audio_raw_set_rate(audio_source_frame->rate); + } else if (mp_obj_is_type(frame_obj, µbit_audio_track_type) + || mp_obj_is_type(frame_obj, µbit_audio_recording_type)) { + // We have the next AudioTrack/AudioRecording. + audio_source_track = MP_OBJ_TO_PTR(frame_obj); + audio_source_frame_offset = 0; + microbit_hal_audio_raw_set_rate(audio_source_track->rate); + } else { + // Audio iterator did not return an AudioFrame/AudioTrack/AudioRecording. + audio_source_iter = MP_OBJ_NULL; + mp_sched_exception(mp_obj_new_exception_msg(&mp_type_TypeError, MP_ERROR_TEXT("not an AudioFrame"))); + return; + } } - mp_obj_t buffer_obj; - nlr_buf_t nlr; - if (nlr_push(&nlr) == 0) { - buffer_obj = mp_iternext_allow_raise(audio_source_iter); - nlr_pop(); +} + +static void audio_data_fetcher(mp_sched_node_t *node) { + audio_data_pull_from_source(); + uint8_t *dest = &audio_output_buffer[audio_output_buffer_offset]; + + if (audio_source_frame == NULL && audio_source_track == NULL) { + // Audio source is exhausted. + + if (audio_output_buffer_offset == 0) { + // No output data left, finish output streaming. + microbit_audio_stop(); + return; + } + + // Fill remaining audio_output_buffer bytes with silence, for the final output frame. + memset(dest, 128, AUDIO_OUTPUT_BUFFER_SIZE - audio_output_buffer_offset); + audio_output_buffer_offset = AUDIO_OUTPUT_BUFFER_SIZE; } else { - if (!mp_obj_is_subclass_fast(MP_OBJ_FROM_PTR(((mp_obj_base_t*)nlr.ret_val)->type), - MP_OBJ_FROM_PTR(&mp_type_StopIteration))) { - mp_sched_exception(MP_OBJ_FROM_PTR(nlr.ret_val)); + // Copy samples to the buffer. + const uint8_t *src; + size_t size; + if (audio_source_frame != NULL) { + src = &audio_source_frame->data[audio_source_frame_offset]; + size = audio_source_frame->alloc_size; + } else { + src = &audio_source_track->data[audio_source_frame_offset]; + size = audio_source_track->size; } - buffer_obj = MP_OBJ_STOP_ITERATION; + size_t src_len = MIN(size - audio_source_frame_offset, AUDIO_OUTPUT_BUFFER_SIZE - audio_output_buffer_offset); + memcpy(dest, src, src_len); + + // Update output and source offsets. + audio_output_buffer_offset += src_len; + audio_source_frame_offset += src_len; } - if (buffer_obj == MP_OBJ_STOP_ITERATION) { - // End of audio iterator - microbit_audio_stop(); - } else if (mp_obj_get_type(buffer_obj) != µbit_audio_frame_type) { - // Audio iterator did not return an AudioFrame - microbit_audio_stop(); - mp_sched_exception(mp_obj_new_exception_msg(&mp_type_TypeError, MP_ERROR_TEXT("not an AudioFrame"))); + + if (audio_output_buffer_offset < AUDIO_OUTPUT_BUFFER_SIZE) { + // Output buffer not full yet, so attempt to pull more data from the source. + audio_data_schedule_fetch(); } else { - microbit_audio_frame_obj_t *buffer = (microbit_audio_frame_obj_t *)buffer_obj; - uint8_t *dest = &audio_output_buffer[0]; - uint32_t last = dest[BUFFER_EXPANSION * AUDIO_CHUNK_SIZE - 1]; - for (int i = 0; i < AUDIO_CHUNK_SIZE; ++i) { - uint32_t cur = buffer->data[i]; - for (int j = 0; j < BUFFER_EXPANSION; ++j) { - // Get next sample with linear interpolation. - uint32_t sample = ((BUFFER_EXPANSION - 1 - j) * last + (j + 1) * cur) / BUFFER_EXPANSION; - // Write sample to the buffer. - *dest++ = sample; - } - last = cur; + // Output buffer is full, process it and prepare for next buffer fill. + audio_output_buffer_offset = 0; + + // Compute the sound level. + uint32_t sound_level = 0; + for (int i = 0; i < AUDIO_OUTPUT_BUFFER_SIZE; ++i) { + sound_level += (audio_output_buffer[i] - 128) * (audio_output_buffer[i] - 128); + } + audio_current_sound_level = sound_level / AUDIO_OUTPUT_BUFFER_SIZE; + + // Send the data to the lower levels of the audio pipeline. + uint32_t atomic_state = MICROPY_BEGIN_ATOMIC_SECTION(); + audio_output_state_t old_state = audio_output_state; + audio_output_state = AUDIO_OUTPUT_STATE_DATA_READY; + MICROPY_END_ATOMIC_SECTION(atomic_state); + if (old_state == AUDIO_OUTPUT_STATE_IDLE) { + microbit_hal_audio_raw_ready_callback(); } - audio_buffer_ready(); } } -static mp_obj_t audio_data_fetcher_wrapper(mp_obj_t arg) { - audio_data_fetcher(); - return mp_const_none; +static void audio_data_schedule_fetch(void) { + if (audio_source_track != NULL && audio_source_frame_offset < audio_source_track->size) { + // An existing AudioTrack is being played, and still has some data remaining, + // so fetch that immediately. This helps to keep audio playback smooth when the + // playback rate is large. + audio_data_fetcher(&audio_data_fetcher_sched_node); + } else { + // Schedule audio_data_fetcher to be executed ASAP to try and fetch more data. + mp_sched_schedule_node(&audio_data_fetcher_sched_node, audio_data_fetcher); + } } -static MP_DEFINE_CONST_FUN_OBJ_1(audio_data_fetcher_wrapper_obj, audio_data_fetcher_wrapper); -void microbit_hal_audio_ready_callback(void) { +void microbit_hal_audio_raw_ready_callback(void) { if (audio_output_state == AUDIO_OUTPUT_STATE_DATA_READY) { // there is data ready to send out to the audio pipeline, so send it - microbit_hal_audio_write_data(&audio_output_buffer[0], OUT_CHUNK_SIZE); + microbit_hal_audio_raw_write_data(&audio_output_buffer[0], AUDIO_OUTPUT_BUFFER_SIZE); audio_output_state = AUDIO_OUTPUT_STATE_DATA_WRITTEN; } else { // no data ready, need to call this function later when data is ready audio_output_state = AUDIO_OUTPUT_STATE_IDLE; } - if (!audio_fetcher_scheduled) { - // schedule audio_data_fetcher to be executed to prepare the next buffer - audio_fetcher_scheduled = mp_sched_schedule(MP_OBJ_FROM_PTR(&audio_data_fetcher_wrapper_obj), mp_const_none); - } + + // Schedule the next fetch of audio data. + audio_data_schedule_fetch(); } static void audio_init(uint32_t sample_rate) { - audio_fetcher_scheduled = false; audio_output_state = AUDIO_OUTPUT_STATE_IDLE; - microbit_hal_audio_init(BUFFER_EXPANSION * sample_rate); + microbit_hal_audio_raw_init(sample_rate); } void microbit_audio_play_source(mp_obj_t src, mp_obj_t pin_select, bool wait, uint32_t sample_rate) { @@ -149,6 +238,15 @@ void microbit_audio_play_source(mp_obj_t src, mp_obj_t pin_select, bool wait, ui sound_expr_data = sound->name; } else if (mp_obj_is_type(src, µbit_soundeffect_type)) { sound_expr_data = microbit_soundeffect_get_sound_expr_data(src); + } else if (mp_obj_is_type(src, µbit_audio_frame_type)) { + audio_source_frame = MP_OBJ_TO_PTR(src); + audio_source_frame_offset = 0; + microbit_hal_audio_raw_set_rate(audio_source_frame->rate); + } else if (mp_obj_is_type(src, µbit_audio_track_type) + || mp_obj_is_type(src, µbit_audio_recording_type)) { + audio_source_track = MP_OBJ_TO_PTR(src); + audio_source_frame_offset = 0; + microbit_hal_audio_raw_set_rate(audio_source_track->rate); } else if (mp_obj_is_type(src, &mp_type_tuple) || mp_obj_is_type(src, &mp_type_list)) { // A tuple/list passed in. Need to check if it contains SoundEffect instances. size_t len; @@ -166,7 +264,13 @@ void microbit_audio_play_source(mp_obj_t src, mp_obj_t pin_select, bool wait, ui } // Replace last "," with a string null terminator. data[-1] = '\0'; + } else { + // A tuple/list of AudioFrame instances. + audio_source_iter = mp_getiter(src, NULL); } + } else { + // An iterator of AudioFrame instances. + audio_source_iter = mp_getiter(src, NULL); } if (sound_expr_data != NULL) { @@ -189,16 +293,15 @@ void microbit_audio_play_source(mp_obj_t src, mp_obj_t pin_select, bool wait, ui return; } - // Get the iterator and start the audio running. + // Start the audio running. // The scheduler must be locked because audio_data_fetcher() can also be called from the scheduler. - audio_source_iter = mp_getiter(src, NULL); mp_sched_lock(); - audio_data_fetcher(); + audio_data_fetcher(&audio_data_fetcher_sched_node); mp_sched_unlock(); if (wait) { // Wait the audio to exhaust the iterator. - while (audio_is_running()) { + while (microbit_audio_is_playing()) { mp_handle_pending(true); microbit_hal_idle(); } @@ -230,7 +333,9 @@ static mp_obj_t play(mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_ar MP_DEFINE_CONST_FUN_OBJ_KW(microbit_audio_play_obj, 0, play); bool microbit_audio_is_playing(void) { - return audio_is_running() || microbit_hal_audio_is_expression_active(); + return audio_is_running() + || microbit_hal_audio_is_playing() + || microbit_hal_audio_is_expression_active(); } mp_obj_t is_playing(void) { @@ -238,12 +343,22 @@ mp_obj_t is_playing(void) { } MP_DEFINE_CONST_FUN_OBJ_0(microbit_audio_is_playing_obj, is_playing); +// Returns a number between 0 and 254, being the average intensity of the sound played +// from the most recent chunk of data. +static mp_obj_t microbit_audio_sound_level(void) { + return MP_OBJ_NEW_SMALL_INT(2 * sqrt(audio_current_sound_level)); +} +static MP_DEFINE_CONST_FUN_OBJ_0(microbit_audio_sound_level_obj, microbit_audio_sound_level); + static const mp_rom_map_elem_t audio_globals_table[] = { { MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_audio) }, { MP_ROM_QSTR(MP_QSTR_stop), MP_ROM_PTR(µbit_audio_stop_obj) }, { MP_ROM_QSTR(MP_QSTR_play), MP_ROM_PTR(µbit_audio_play_obj) }, { MP_ROM_QSTR(MP_QSTR_is_playing), MP_ROM_PTR(µbit_audio_is_playing_obj) }, + { MP_ROM_QSTR(MP_QSTR_sound_level), MP_ROM_PTR(µbit_audio_sound_level_obj) }, { MP_ROM_QSTR(MP_QSTR_AudioFrame), MP_ROM_PTR(µbit_audio_frame_type) }, + { MP_ROM_QSTR(MP_QSTR_AudioRecording), MP_ROM_PTR(µbit_audio_recording_type) }, + { MP_ROM_QSTR(MP_QSTR_AudioTrack), MP_ROM_PTR(µbit_audio_track_type) }, { MP_ROM_QSTR(MP_QSTR_SoundEffect), MP_ROM_PTR(µbit_soundeffect_type) }, }; static MP_DEFINE_CONST_DICT(audio_module_globals, audio_globals_table); @@ -258,17 +373,40 @@ MP_REGISTER_MODULE(MP_QSTR_audio, audio_module); /******************************************************************************/ // AudioFrame class -static mp_obj_t microbit_audio_frame_new(const mp_obj_type_t *type_in, mp_uint_t n_args, mp_uint_t n_kw, const mp_obj_t *args) { +static mp_obj_t microbit_audio_frame_new(const mp_obj_type_t *type_in, size_t n_args, size_t n_kw, const mp_obj_t *all_args) { (void)type_in; - (void)args; - mp_arg_check_num(n_args, n_kw, 0, 0, false); - return microbit_audio_frame_make_new(); + + enum { ARG_duration, ARG_rate }; + static const mp_arg_t allowed_args[] = { + { MP_QSTR_duration, MP_ARG_OBJ, {.u_rom_obj = MP_ROM_NONE} }, + { MP_QSTR_rate, MP_ARG_INT, {.u_int = DEFAULT_SAMPLE_RATE} }, + }; + mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; + mp_arg_parse_all_kw_array(n_args, n_kw, all_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args); + + mp_int_t rate = args[ARG_rate].u_int; + if (rate <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + + size_t size; + if (args[ARG_duration].u_obj == mp_const_none) { + size = DEFAULT_AUDIO_FRAME_SIZE; + } else { + mp_float_t duration = mp_obj_get_float(args[ARG_duration].u_obj); + if (duration <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("size out of bounds")); + } + size = duration * rate / 1000; + } + + return microbit_audio_frame_make_new(size, rate); } static mp_obj_t audio_frame_subscr(mp_obj_t self_in, mp_obj_t index_in, mp_obj_t value_in) { microbit_audio_frame_obj_t *self = (microbit_audio_frame_obj_t *)self_in; mp_int_t index = mp_obj_get_int(index_in); - if (index < 0 || index >= AUDIO_CHUNK_SIZE) { + if (index < 0 || index >= self->alloc_size) { mp_raise_ValueError(MP_ERROR_TEXT("index out of bounds")); } if (value_in == MP_OBJ_NULL) { @@ -288,10 +426,10 @@ static mp_obj_t audio_frame_subscr(mp_obj_t self_in, mp_obj_t index_in, mp_obj_t } static mp_obj_t audio_frame_unary_op(mp_unary_op_t op, mp_obj_t self_in) { - (void)self_in; + microbit_audio_frame_obj_t *self = (microbit_audio_frame_obj_t *)self_in; switch (op) { case MP_UNARY_OP_LEN: - return MP_OBJ_NEW_SMALL_INT(32); + return MP_OBJ_NEW_SMALL_INT(self->alloc_size); default: return MP_OBJ_NULL; // op not supported } @@ -301,26 +439,26 @@ static mp_int_t audio_frame_get_buffer(mp_obj_t self_in, mp_buffer_info_t *bufin (void)flags; microbit_audio_frame_obj_t *self = (microbit_audio_frame_obj_t *)self_in; bufinfo->buf = self->data; - bufinfo->len = AUDIO_CHUNK_SIZE; - bufinfo->typecode = 'b'; + bufinfo->len = self->alloc_size; + bufinfo->typecode = 'B'; return 0; } -static void add_into(microbit_audio_frame_obj_t *self, microbit_audio_frame_obj_t *other, bool add) { +void microbit_audio_data_add_inplace(uint8_t *lhs_data, const uint8_t *rhs_data, size_t size, bool add) { int mult = add ? 1 : -1; - for (int i = 0; i < AUDIO_CHUNK_SIZE; i++) { - unsigned val = (int)self->data[i] + mult*(other->data[i]-128); + for (int i = 0; i < size; i++) { + unsigned val = (int)lhs_data[i] + mult*(rhs_data[i]-128); // Clamp to 0-255 if (val > 255) { val = (1-(val>>31))*255; } - self->data[i] = val; + lhs_data[i] = val; } } static microbit_audio_frame_obj_t *copy(microbit_audio_frame_obj_t *self) { - microbit_audio_frame_obj_t *result = microbit_audio_frame_make_new(); - for (int i = 0; i < AUDIO_CHUNK_SIZE; i++) { + microbit_audio_frame_obj_t *result = microbit_audio_frame_make_new(self->alloc_size, self->rate); + for (int i = 0; i < self->alloc_size; i++) { result->data[i] = self->data[i]; } return result; @@ -330,7 +468,7 @@ mp_obj_t copyfrom(mp_obj_t self_in, mp_obj_t other) { microbit_audio_frame_obj_t *self = (microbit_audio_frame_obj_t *)self_in; mp_buffer_info_t bufinfo; mp_get_buffer_raise(other, &bufinfo, MP_BUFFER_READ); - uint32_t len = bufinfo.len > AUDIO_CHUNK_SIZE ? AUDIO_CHUNK_SIZE : bufinfo.len; + uint32_t len = MIN(bufinfo.len, self->alloc_size); for (uint32_t i = 0; i < len; i++) { self->data[i] = ((uint8_t *)bufinfo.buf)[i]; } @@ -365,14 +503,14 @@ int32_t float_to_fixed(float f, uint32_t scale) { return result; } -static void mult(microbit_audio_frame_obj_t *self, float f) { +void microbit_audio_data_mult_inplace(uint8_t *data, size_t size, float f) { int scaled = float_to_fixed(f, 15); - for (int i = 0; i < AUDIO_CHUNK_SIZE; i++) { - unsigned val = ((((int)self->data[i]-128) * scaled) >> 15)+128; + for (int i = 0; i < size; i++) { + unsigned val = ((((int)data[i]-128) * scaled) >> 15)+128; if (val > 255) { val = (1-(val>>31))*255; } - self->data[i] = val; + data[i] = val; } } @@ -390,19 +528,42 @@ static mp_obj_t audio_frame_binary_op(mp_binary_op_t op, mp_obj_t lhs_in, mp_obj if (mp_obj_get_type(rhs_in) != µbit_audio_frame_type) { return MP_OBJ_NULL; // op not supported } - add_into(lhs, (microbit_audio_frame_obj_t *)rhs_in, op==MP_BINARY_OP_ADD||op==MP_BINARY_OP_INPLACE_ADD); + microbit_audio_frame_obj_t *rhs = MP_OBJ_TO_PTR(rhs_in); + size_t size = MIN(lhs->alloc_size, rhs->alloc_size); + microbit_audio_data_add_inplace(lhs->data, rhs->data, size, op == MP_BINARY_OP_ADD || op == MP_BINARY_OP_INPLACE_ADD); return lhs; case MP_BINARY_OP_MULTIPLY: lhs = copy(lhs); case MP_BINARY_OP_INPLACE_MULTIPLY: - mult(lhs, mp_obj_get_float(rhs_in)); + microbit_audio_data_mult_inplace(lhs->data, lhs->alloc_size, mp_obj_get_float(rhs_in)); return lhs; default: return MP_OBJ_NULL; // op not supported } } +static mp_obj_t audio_frame_get_rate(mp_obj_t self_in) { + microbit_audio_frame_obj_t *self = MP_OBJ_TO_PTR(self_in); + return MP_OBJ_NEW_SMALL_INT(self->rate); +} +static MP_DEFINE_CONST_FUN_OBJ_1(audio_frame_get_rate_obj, audio_frame_get_rate); + +static mp_obj_t audio_frame_set_rate(mp_obj_t self_in, mp_obj_t rate_in) { + microbit_audio_frame_obj_t *self = MP_OBJ_TO_PTR(self_in); + mp_int_t rate = mp_obj_get_int(rate_in); + if (rate <= 0) { + mp_raise_ValueError(MP_ERROR_TEXT("rate out of bounds")); + } + self->rate = rate; + // TODO: only set if this frame is currently being played + microbit_hal_audio_raw_set_rate(rate); + return mp_const_none; +} +static MP_DEFINE_CONST_FUN_OBJ_2(audio_frame_set_rate_obj, audio_frame_set_rate); + static const mp_map_elem_t microbit_audio_frame_locals_dict_table[] = { + { MP_OBJ_NEW_QSTR(MP_QSTR_get_rate), (mp_obj_t)&audio_frame_get_rate_obj }, + { MP_OBJ_NEW_QSTR(MP_QSTR_set_rate), (mp_obj_t)&audio_frame_set_rate_obj }, { MP_OBJ_NEW_QSTR(MP_QSTR_copyfrom), (mp_obj_t)©from_obj }, }; static MP_DEFINE_CONST_DICT(microbit_audio_frame_locals_dict, microbit_audio_frame_locals_dict_table); @@ -419,11 +580,20 @@ MP_DEFINE_CONST_OBJ_TYPE( locals_dict, µbit_audio_frame_locals_dict ); -microbit_audio_frame_obj_t *microbit_audio_frame_make_new(void) { - microbit_audio_frame_obj_t *res = m_new_obj(microbit_audio_frame_obj_t); +microbit_audio_frame_obj_t *microbit_audio_frame_make_new(size_t size, uint32_t rate) { + // Make sure size is non-zero. + if (size == 0) { + size = 1; + } + + microbit_audio_frame_obj_t *res = m_new_obj_var(microbit_audio_frame_obj_t, data, uint8_t, size); res->base.type = µbit_audio_frame_type; - memset(res->data, 128, AUDIO_CHUNK_SIZE); + res->alloc_size = size; + res->rate = rate; + memset(res->data, 128, size); return res; } -MP_REGISTER_ROOT_POINTER(void *audio_source); +MP_REGISTER_ROOT_POINTER(struct _microbit_audio_frame_obj_t *audio_source_frame_state); +MP_REGISTER_ROOT_POINTER(struct _microbit_audio_track_obj_t *audio_source_track_state); +MP_REGISTER_ROOT_POINTER(mp_obj_t audio_source_iter_state); diff --git a/src/codal_port/modaudio.h b/src/codal_port/modaudio.h index 7111715..b3440e4 100644 --- a/src/codal_port/modaudio.h +++ b/src/codal_port/modaudio.h @@ -29,24 +29,44 @@ #include "py/runtime.h" -#define LOG_AUDIO_CHUNK_SIZE (5) -#define AUDIO_CHUNK_SIZE (1 << LOG_AUDIO_CHUNK_SIZE) - #define SOUND_EXPR_TOTAL_LENGTH (72) +#define AUDIO_TRACK_DEFAULT_SAMPLE_RATE (7812) typedef struct _microbit_audio_frame_obj_t { mp_obj_base_t base; - uint8_t data[AUDIO_CHUNK_SIZE]; + size_t alloc_size; + uint32_t rate; + uint8_t data[]; } microbit_audio_frame_obj_t; +typedef struct _microbit_audio_track_obj_t { + mp_obj_base_t base; + mp_obj_t buffer_obj; // so GC can't reclaim the underlying buffer + size_t size; + uint32_t rate; + uint8_t *data; +} microbit_audio_track_obj_t; + extern const mp_obj_type_t microbit_audio_frame_type; +extern const mp_obj_type_t microbit_audio_recording_type; +extern const mp_obj_type_t microbit_audio_track_type; extern const mp_obj_module_t audio_module; +void microbit_audio_data_add_inplace(uint8_t *lhs_data, const uint8_t *rhs_data, size_t size, bool add); +void microbit_audio_data_mult_inplace(uint8_t *data, size_t size, float f); + void microbit_audio_play_source(mp_obj_t src, mp_obj_t pin_select, bool wait, uint32_t sample_rate); void microbit_audio_stop(void); bool microbit_audio_is_playing(void); -microbit_audio_frame_obj_t *microbit_audio_frame_make_new(void); + +microbit_audio_frame_obj_t *microbit_audio_frame_make_new(size_t size, uint32_t rate); +mp_obj_t microbit_audio_track_new(mp_obj_t buffer_obj, size_t len, uint8_t *data, uint32_t rate); +mp_obj_t microbit_audio_recording_new(size_t num_bytes, uint32_t rate); const char *microbit_soundeffect_get_sound_expr_data(mp_obj_t self_in); +mp_int_t microbit_audio_track_get_buffer(mp_obj_t self_in, mp_buffer_info_t *bufinfo, mp_uint_t flags); +MP_DECLARE_CONST_FUN_OBJ_1(microbit_audio_track_get_rate_obj); +MP_DECLARE_CONST_FUN_OBJ_2(microbit_audio_track_set_rate_obj); + #endif // MICROPY_INCLUDED_MICROBIT_MODAUDIO_H diff --git a/src/codal_port/mpconfigport.h b/src/codal_port/mpconfigport.h index cca2378..8a4a3a7 100644 --- a/src/codal_port/mpconfigport.h +++ b/src/codal_port/mpconfigport.h @@ -61,6 +61,7 @@ #define MICROPY_MODULE_BUILTIN_INIT (1) #define MICROPY_USE_INTERNAL_ERRNO (1) #define MICROPY_ENABLE_SCHEDULER (1) +#define MICROPY_SCHEDULER_STATIC_NODES (1) // Fine control over Python builtins, classes, modules, etc #define MICROPY_PY_BUILTINS_STR_UNICODE (1) diff --git a/src/codal_port/utils.c b/src/codal_port/utils.c new file mode 100644 index 0000000..d070417 --- /dev/null +++ b/src/codal_port/utils.c @@ -0,0 +1,35 @@ +/* + * This file is part of the MicroPython project, http://micropython.org/ + * + * The MIT License (MIT) + * + * Copyright (c) 2024 Damien P. George + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +#include "utils.h" + +mp_int_t mp_obj_get_int_allow_float(mp_obj_t obj) { + if (mp_obj_is_integer(obj)) { + return mp_obj_get_int(obj); + } else { + return (mp_int_t)mp_obj_get_float(obj); + } +} diff --git a/src/codal_port/utils.h b/src/codal_port/utils.h new file mode 100644 index 0000000..a398e81 --- /dev/null +++ b/src/codal_port/utils.h @@ -0,0 +1,33 @@ +/* + * This file is part of the MicroPython project, http://micropython.org/ + * + * The MIT License (MIT) + * + * Copyright (c) 2024 Damien P. George + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +#ifndef MICROPY_INCLUDED_CODAL_PORT_UTILS_H +#define MICROPY_INCLUDED_CODAL_PORT_UTILS_H + +#include "py/runtime.h" + +mp_int_t mp_obj_get_int_allow_float(mp_obj_t obj); + +#endif // MICROPY_INCLUDED_CODAL_PORT_UTILS_H diff --git a/src/test_record.py b/src/test_record.py new file mode 100644 index 0000000..ef81f9e --- /dev/null +++ b/src/test_record.py @@ -0,0 +1,54 @@ +from microbit import * + +mouth_open = Image( + "09090:" + "00000:" + "09990:" + "90009:" + "09990" +) +mouth_closed = Image( + "09090:" + "00000:" + "00000:" + "99999:" + "00000" +) +play = Image( + "00000:" + "04740:" + "07970:" + "04740:" + "00000" +) + +my_recording = audio.AudioRecording(duration=5000) +my_track = None + +while True: + if button_a.was_pressed(): + my_track = microphone.record_into(my_recording, wait=False) + display.show([mouth_open, mouth_closed], loop=True, wait=False, delay=150) + while button_a.is_pressed() and microphone.is_recording(): + sleep(50) + microphone.stop_recording() + display.show(mouth_closed) + while button_a.is_pressed(): + sleep(50) + display.clear() + my_track *= 2 # amplify volume + if button_b.was_pressed() and my_track: + audio.play(my_track, wait=False) + level = 0 + while audio.is_playing(): + l = audio.sound_level() + if l > level: + level = l + else: + level *= 0.95 + display.show(play * min(1, level / 100)) + x = accelerometer.get_x() + my_track.set_rate(max(2250, scale(x, (-1000, 1000), (2250, 13374)))) + sleep(5) + display.clear() + sleep(100)