Sample Rate Enhancements
This commit is contained in:
parent
e778112710
commit
c3c1e307c4
@ -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 "
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user