Sample Rate Enhancements

This commit is contained in:
Lee Atkinson 2022-08-11 17:50:30 -04:00
parent e778112710
commit c3c1e307c4
5 changed files with 54 additions and 51 deletions

View File

@ -3863,6 +3863,11 @@ msgstr ""
msgid "rsplit(None,n)" msgid "rsplit(None,n)"
msgstr "" msgstr ""
#: shared-bindings/analogio/AnalogFastIn.c
msgid "sample rate must be 1.0-500000.0 per second"
msgstr ""
#: shared-bindings/analogio/AnalogFastIn.c
#: shared-bindings/audiocore/RawSample.c #: shared-bindings/audiocore/RawSample.c
msgid "" msgid ""
"sample_source buffer must be a bytearray or array of type 'h', 'H', 'b' or " "sample_source buffer must be a bytearray or array of type 'h', 'H', 'b' or "

View File

@ -29,16 +29,12 @@
#include "shared-bindings/microcontroller/Pin.h" #include "shared-bindings/microcontroller/Pin.h"
#include "py/runtime.h" #include "py/runtime.h"
#include "supervisor/shared/translate/translate.h" #include "supervisor/shared/translate/translate.h"
#include "src/rp2_common/hardware_adc/include/hardware/adc.h" #include "src/rp2_common/hardware_adc/include/hardware/adc.h"
#include "src/rp2_common/hardware_dma/include/hardware/dma.h" #include "src/rp2_common/hardware_dma/include/hardware/dma.h"
// /sdk/src/rp2_common/hardware_dma/include/hardware/dma.h // /sdk/src/rp2_common/hardware_dma/include/hardware/dma.h
// ports/raspberrypi/
// ports/raspberrypi/
#include "sdk/src/common/pico_stdlib/include/pico/stdlib.h" #include "sdk/src/common/pico_stdlib/include/pico/stdlib.h"
#define ADC_FIRST_PIN_NUMBER 26 #define ADC_FIRST_PIN_NUMBER 26
#define ADC_PIN_COUNT 4 #define ADC_PIN_COUNT 4
// Channel 0 is GPIO26 // Channel 0 is GPIO26
@ -46,30 +42,7 @@
#define CAPTURE_DEPTH 1000 #define CAPTURE_DEPTH 1000
uint8_t capture_buf[CAPTURE_DEPTH]; uint8_t capture_buf[CAPTURE_DEPTH];
// ADC unit8 or int8 ??? ---> unint16 void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *self, const mcu_pin_obj_t *pin, uint8_t *buffer, uint32_t len, uint8_t bytes_per_sample, bool samples_signed, mp_float_t sample_rate) {
//
// uint16_t common_hal_analogio_analogin_get_value(analogio_analogin_obj_t *self) {
// adc_select_input(self->pin->number - ADC_FIRST_PIN_NUMBER);
// uint16_t value = adc_read();
//
// // Stretch 12-bit ADC reading to 16-bit range
// return (value << 4) | (value >> 8);
// }
/*
typedef struct {
mp_obj_base_t base;
uint8_t number;
} mcu_pin_obj_t;
*/
// self->pin = pin;
// self->buffer = buffer;
// self->len = len;
// //self->bits_per_sample = bytes_per_sample * 8;
// self->samples_signed = samples_signed;
// self->sample_rate = sample_rate;
void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *self, const mcu_pin_obj_t *pin, uint8_t *buffer, uint32_t len, uint8_t bytes_per_sample, bool samples_signed, uint32_t sample_rate) {
// Set pin and channel // Set pin and channel
self->pin = pin; self->pin = pin;
@ -83,7 +56,6 @@ void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *sel
// checks on length here // checks on length here
// uint8_t bytes_per_sample // uint8_t bytes_per_sample
// Set sample rate // Set sample rate
// self->bits_per_sample = bytes_per_sample * 8; // self->bits_per_sample = bytes_per_sample * 8;
@ -102,7 +74,7 @@ void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *sel
true, // Enable DMA data request (DREQ) true, // Enable DMA data request (DREQ)
1, // DREQ (and IRQ) asserted when at least 1 sample present 1, // DREQ (and IRQ) asserted when at least 1 sample present
false, // We won't see the ERR bit because of 8 bit reads; disable. // ?? false, // We won't see the ERR bit because of 8 bit reads; disable. // ??
true // Shift each sample to 8 bits when pushing to FIFO // ?? false // Shift each sample to 8 bits when pushing to FIFO // ??
); );
// Divisor of 0 -> full speed. Free-running capture with the divider is // Divisor of 0 -> full speed. Free-running capture with the divider is
@ -112,28 +84,28 @@ void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *sel
// continuously) or > 95 (take samples less frequently than 96 cycle // continuously) or > 95 (take samples less frequently than 96 cycle
// intervals). This is all timed by the 48 MHz ADC clock. // intervals). This is all timed by the 48 MHz ADC clock.
// sample rate determines divisor, not zero. // sample rate determines divisor, not zero.
adc_set_clkdiv(0);
adc_set_clkdiv(48000000.0 / self->sample_rate);
// sleep_ms(1000); // sleep_ms(1000);
// Set up the DMA to start transferring data as soon as it appears in FIFO // Set up the DMA to start transferring data as soon as it appears in FIFO
uint dma_chan = dma_claim_unused_channel(true); uint dma_chan = dma_claim_unused_channel(true);
self->dma_chan = dma_chan; self->dma_chan = dma_chan;
dma_channel_config cfg = dma_channel_get_default_config(dma_chan);
// Set Config
self->cfg = dma_channel_get_default_config(dma_chan);
// Reading from constant address, writing to incrementing byte addresses // Reading from constant address, writing to incrementing byte addresses
channel_config_set_transfer_data_size(&cfg, DMA_SIZE_8); channel_config_set_transfer_data_size(&(self->cfg), DMA_SIZE_16);
channel_config_set_read_increment(&cfg, false); channel_config_set_read_increment(&(self->cfg), false);
channel_config_set_write_increment(&cfg, true); channel_config_set_write_increment(&(self->cfg), true);
// Pace transfers based on availability of ADC samples // Pace transfers based on availability of ADC samples
channel_config_set_dreq(&cfg, DREQ_ADC); channel_config_set_dreq(&(self->cfg), DREQ_ADC);
dma_channel_configure(dma_chan, &cfg, // clear any previous activity
capture_buf, // dst adc_fifo_drain();
&adc_hw->fifo, // src adc_run(false);
self->len, // CAPTURE_DEPTH, // transfer count
true // start immediately
);
} }
@ -146,8 +118,12 @@ void common_hal_analogio_analogfastin_deinit(analogio_analogfastin_obj_t *self)
return; return;
} }
// Release ADC Pin
reset_pin_number(self->pin->number); reset_pin_number(self->pin->number);
self->pin = NULL; self->pin = NULL;
// Release DMA Channel
dma_channel_unclaim(self->dma_chan);
} }
// ================================================================ // ================================================================
@ -155,15 +131,31 @@ void common_hal_analogio_analogfastin_deinit(analogio_analogfastin_obj_t *self)
// make this a bool so that later we can perform integrity checking // make this a bool so that later we can perform integrity checking
// ================================================================ // ================================================================
bool common_hal_analogio_analogfastin_capture(analogio_analogfastin_obj_t *self) { bool common_hal_analogio_analogfastin_capture(analogio_analogfastin_obj_t *self) {
// CONSIDER THESE ISSUES
// uint16_t value = adc_read(); // uint16_t value = adc_read();
// Stretch 12-bit ADC reading to 16-bit range // Stretch 12-bit ADC reading to 16-bit range
// return (value << 4) | (value >> 8); // return (value << 4) | (value >> 8);
uint32_t cdl = self->len / 2 - 1;
dma_channel_configure(self->dma_chan, &(self->cfg),
self->buffer, // dst
&adc_hw->fifo, // src
cdl, // CAPTURE_DEPTH, // transfer count
true // start immediately
);
// Start the ADC
adc_run(true); adc_run(true);
// Once DMA finishes, stop any new conversions from starting, and clean up // Once DMA finishes, stop any new conversions from starting, and clean up
// the FIFO in case the ADC was still mid-conversion. // the FIFO in case the ADC was still mid-conversion.
dma_channel_wait_for_finish_blocking(self->dma_chan); dma_channel_wait_for_finish_blocking(self->dma_chan);
// printf("Capture finished\n");
// Clean up
adc_run(false); adc_run(false);
adc_fifo_drain(); adc_fifo_drain();
return true; return true;
} }

View File

@ -28,6 +28,7 @@
#define MICROPY_INCLUDED_RASPBERRYPI_COMMON_HAL_ANALOGIO_ANALOGFASTIN_H #define MICROPY_INCLUDED_RASPBERRYPI_COMMON_HAL_ANALOGIO_ANALOGFASTIN_H
#include "common-hal/microcontroller/Pin.h" #include "common-hal/microcontroller/Pin.h"
#include "src/rp2_common/hardware_dma/include/hardware/dma.h"
#include "py/obj.h" #include "py/obj.h"
@ -39,9 +40,10 @@ typedef struct {
uint32_t len; uint32_t len;
uint8_t bytes_per_sample; uint8_t bytes_per_sample;
bool samples_signed; bool samples_signed;
uint32_t sample_rate; mp_float_t sample_rate;
uint8_t chan; uint8_t chan;
uint dma_chan; uint dma_chan;
dma_channel_config cfg;
// data_size = DMA_SIZE_8; // - default DMA_SIZE_8 // data_size = DMA_SIZE_8; // - default DMA_SIZE_8
// data_size = DMA_SIZE_16; // - default DMA_SIZE_16 // data_size = DMA_SIZE_16; // - default DMA_SIZE_16
// Either 12 bits in 16 or 12 over 2 bytes or truncate 12 to 8 in 8 // Either 12 bits in 16 or 12 over 2 bytes or truncate 12 to 8 in 8

View File

@ -37,15 +37,18 @@
#include "shared-bindings/util.h" #include "shared-bindings/util.h"
// pin, buffer, rate // pin, buffer, rate
STATIC void validate_rate(mp_float_t rate) {
if (rate < (mp_float_t)1.0f || rate > (mp_float_t)500000.0f) {
mp_raise_ValueError(translate("sample rate must be 1.0-500000.0 per second"));
}
}
STATIC mp_obj_t analogio_analogfastin_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *all_args) { STATIC mp_obj_t analogio_analogfastin_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *all_args) {
enum { ARG_pin, ARG_buffer, ARG_sample_rate }; enum { ARG_pin, ARG_buffer, ARG_sample_rate };
static const mp_arg_t allowed_args[] = { static const mp_arg_t allowed_args[] = {
{ MP_QSTR_pin, MP_ARG_REQUIRED | MP_ARG_OBJ }, { MP_QSTR_pin, MP_ARG_OBJ | MP_ARG_REQUIRED },
{ MP_QSTR_buffer, MP_ARG_OBJ | MP_ARG_REQUIRED }, { MP_QSTR_buffer, MP_ARG_OBJ | MP_ARG_REQUIRED },
{ MP_QSTR_sample_rate, MP_ARG_INT | MP_ARG_KW_ONLY, {.u_int = 8000} }, { MP_QSTR_sample_rate, MP_ARG_OBJ | MP_ARG_REQUIRED, {.u_obj = MP_OBJ_NEW_SMALL_INT(1)} },
}; };
mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)]; 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_arg_parse_all_kw_array(n_args, n_kw, all_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args);
@ -69,7 +72,8 @@ STATIC mp_obj_t analogio_analogfastin_make_new(const mp_obj_type_t *type, size_t
} }
// Validate sample rate here // Validate sample rate here
uint32_t sample_rate = args[ARG_sample_rate].u_int; mp_float_t sample_rate = mp_obj_get_float(args[ARG_sample_rate].u_obj);
validate_rate(sample_rate);
// Create local object // Create local object
analogio_analogfastin_obj_t *self = m_new_obj(analogio_analogfastin_obj_t); analogio_analogfastin_obj_t *self = m_new_obj(analogio_analogfastin_obj_t);

View File

@ -32,7 +32,7 @@
extern const mp_obj_type_t analogio_analogfastin_type; extern const mp_obj_type_t analogio_analogfastin_type;
void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *self, const mcu_pin_obj_t *pin, uint8_t *buffer, uint32_t len, uint8_t bytes_per_sample, bool samples_signed, uint32_t sample_rate); void common_hal_analogio_analogfastin_construct(analogio_analogfastin_obj_t *self, const mcu_pin_obj_t *pin, uint8_t *buffer, uint32_t len, uint8_t bytes_per_sample, bool samples_signed, mp_float_t sample_rate);
void common_hal_analogio_analogfastin_deinit(analogio_analogfastin_obj_t *self); void common_hal_analogio_analogfastin_deinit(analogio_analogfastin_obj_t *self);
bool common_hal_analogio_analogfastin_deinited(analogio_analogfastin_obj_t *self); bool common_hal_analogio_analogfastin_deinited(analogio_analogfastin_obj_t *self);
bool common_hal_analogio_analogfastin_capture(analogio_analogfastin_obj_t *self); bool common_hal_analogio_analogfastin_capture(analogio_analogfastin_obj_t *self);