samd/machine_adc: Add the machine.ADC class.
With the method read_u16(). Keyword arguments of the constructor are: - bits=n The resolution; default is 12. - average=n The average of samples, which are taken and cumulated. The default value is 16. Averaging by hw is faster than averaging in code. The ADC runs at a clock freq 1.5 MHz. A single 12 bit conversion takes 8 microseconds.
This commit is contained in:
parent
6765d4bbd6
commit
5c7e93ec48
@ -90,6 +90,7 @@ endif
|
||||
SRC_C = \
|
||||
clock_config.c \
|
||||
help.c \
|
||||
machine_adc.c \
|
||||
machine_led.c \
|
||||
machine_pin.c \
|
||||
main.c \
|
||||
@ -138,6 +139,7 @@ endif
|
||||
|
||||
# List of sources for qstr extraction
|
||||
SRC_QSTR += \
|
||||
machine_adc.c \
|
||||
machine_led.c \
|
||||
machine_pin.c \
|
||||
modutime.c \
|
||||
|
238
ports/samd/machine_adc.c
Normal file
238
ports/samd/machine_adc.c
Normal file
@ -0,0 +1,238 @@
|
||||
/*
|
||||
* This file is part of the MicroPython project, http://micropython.org/
|
||||
*
|
||||
* The MIT License (MIT)
|
||||
*
|
||||
* Copyright (c) 2021 Philipp Ebensberger
|
||||
* Copyright (c) 2022 Robert Hammelrath
|
||||
*
|
||||
* 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 <stdint.h>
|
||||
#include "py/obj.h"
|
||||
#include "py/runtime.h"
|
||||
#include "py/mphal.h"
|
||||
|
||||
#include "sam.h"
|
||||
#include "pin_af.h"
|
||||
#include "modmachine.h"
|
||||
|
||||
typedef struct _machine_adc_obj_t {
|
||||
mp_obj_base_t base;
|
||||
adc_config_t adc_config;
|
||||
uint8_t id;
|
||||
uint8_t avg;
|
||||
uint8_t bits;
|
||||
} machine_adc_obj_t;
|
||||
|
||||
#define DEFAULT_ADC_BITS 12
|
||||
#define DEFAULT_ADC_AVG 16
|
||||
|
||||
Adc *const adc_bases[] = ADC_INSTS;
|
||||
uint32_t busy_flags = 0;
|
||||
bool init_flags[2] = {false, false};
|
||||
static void adc_init(machine_adc_obj_t *self);
|
||||
static uint8_t resolution[] = {
|
||||
ADC_CTRLB_RESSEL_8BIT_Val, ADC_CTRLB_RESSEL_10BIT_Val, ADC_CTRLB_RESSEL_12BIT_Val
|
||||
};
|
||||
|
||||
// Calculate the floor value of log2(n)
|
||||
mp_int_t log2i(mp_int_t num) {
|
||||
mp_int_t res = 0;
|
||||
for (; num > 1; num >>= 1) {
|
||||
res += 1;
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
STATIC void adc_obj_print(const mp_print_t *print, mp_obj_t o, mp_print_kind_t kind) {
|
||||
(void)kind;
|
||||
machine_adc_obj_t *self = MP_OBJ_TO_PTR(o);
|
||||
|
||||
mp_printf(print, "ADC(P%c%02u, ADC%u, channel=%u, bits=%u, average=%u)",
|
||||
"ABCD"[self->id / 32], self->id % 32, self->adc_config.device,
|
||||
self->adc_config.channel, self->bits, 1 << self->avg);
|
||||
}
|
||||
|
||||
STATIC mp_obj_t adc_obj_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *all_args) {
|
||||
enum { ARG_id, ARG_bits, ARG_average };
|
||||
static const mp_arg_t allowed_args[] = {
|
||||
{ MP_QSTR_id, MP_ARG_REQUIRED | MP_ARG_OBJ },
|
||||
{ MP_QSTR_bits, MP_ARG_INT, {.u_int = DEFAULT_ADC_BITS} },
|
||||
{ MP_QSTR_average, MP_ARG_KW_ONLY | MP_ARG_INT, {.u_int = DEFAULT_ADC_AVG} },
|
||||
};
|
||||
|
||||
// Parse the arguments.
|
||||
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);
|
||||
|
||||
// Unpack and check, whther the pin has ADC capability
|
||||
int id = mp_hal_get_pin_obj(args[ARG_id].u_obj);
|
||||
adc_config_t adc_config = get_adc_config(id, busy_flags);
|
||||
|
||||
// Now that we have a valid device and channel, create and populate the ADC instance
|
||||
machine_adc_obj_t *self = mp_obj_malloc(machine_adc_obj_t, &machine_adc_type);
|
||||
self->id = id;
|
||||
self->adc_config = adc_config;
|
||||
self->bits = DEFAULT_ADC_BITS;
|
||||
uint16_t bits = args[ARG_bits].u_int;
|
||||
if (bits >= 8 && bits <= 12) {
|
||||
self->bits = bits;
|
||||
}
|
||||
uint32_t avg = log2i(args[ARG_average].u_int);
|
||||
self->avg = (avg <= 10 ? avg : 10);
|
||||
|
||||
// flag the device/channel as being in use.
|
||||
busy_flags |= (1 << (self->adc_config.device * 16 + self->adc_config.channel));
|
||||
|
||||
adc_init(self);
|
||||
|
||||
return MP_OBJ_FROM_PTR(self);
|
||||
}
|
||||
|
||||
// read_u16()
|
||||
STATIC mp_obj_t machine_adc_read_u16(mp_obj_t self_in) {
|
||||
machine_adc_obj_t *self = MP_OBJ_TO_PTR(self_in);
|
||||
Adc *adc = adc_bases[self->adc_config.device];
|
||||
// Set Input channel and resolution
|
||||
// Select the pin as positive input and gnd as negative input reference, non-diff mode by default
|
||||
adc->INPUTCTRL.reg = ADC_INPUTCTRL_MUXNEG_GND | self->adc_config.channel;
|
||||
// set resolution. Scale 8-16 to 0 - 4 for table access.
|
||||
adc->CTRLB.bit.RESSEL = resolution[(self->bits - 8) / 2];
|
||||
// Measure input voltage
|
||||
adc->SWTRIG.bit.START = 1;
|
||||
while (adc->INTFLAG.bit.RESRDY == 0) {
|
||||
}
|
||||
// Get and return the result
|
||||
return MP_OBJ_NEW_SMALL_INT(adc->RESULT.reg * (65536 / (1 << self->bits)));
|
||||
}
|
||||
STATIC MP_DEFINE_CONST_FUN_OBJ_1(machine_adc_read_u16_obj, machine_adc_read_u16);
|
||||
|
||||
// deinit() : release the ADC channel
|
||||
STATIC mp_obj_t machine_adc_deinit(mp_obj_t self_in) {
|
||||
machine_adc_obj_t *self = MP_OBJ_TO_PTR(self_in);
|
||||
|
||||
busy_flags &= ~((1 << (self->adc_config.device * 16 + self->adc_config.channel)));
|
||||
return mp_const_none;
|
||||
}
|
||||
STATIC MP_DEFINE_CONST_FUN_OBJ_1(machine_adc_deinit_obj, machine_adc_deinit);
|
||||
|
||||
void adc_deinit_all(void) {
|
||||
busy_flags = 0;
|
||||
init_flags[0] = 0;
|
||||
init_flags[1] = 0;
|
||||
}
|
||||
|
||||
STATIC const mp_rom_map_elem_t adc_locals_dict_table[] = {
|
||||
{ MP_ROM_QSTR(MP_QSTR_read_u16), MP_ROM_PTR(&machine_adc_read_u16_obj) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_deinit), MP_ROM_PTR(&machine_adc_deinit_obj) },
|
||||
};
|
||||
|
||||
STATIC MP_DEFINE_CONST_DICT(adc_locals_dict, adc_locals_dict_table);
|
||||
|
||||
MP_DEFINE_CONST_OBJ_TYPE(
|
||||
machine_adc_type,
|
||||
MP_QSTR_ADC,
|
||||
MP_TYPE_FLAG_NONE,
|
||||
make_new, adc_obj_make_new,
|
||||
print, adc_obj_print,
|
||||
locals_dict, &adc_locals_dict
|
||||
);
|
||||
|
||||
static void adc_init(machine_adc_obj_t *self) {
|
||||
// ADC & clock init is done only once per ADC
|
||||
if (init_flags[self->adc_config.device] == false) {
|
||||
Adc *adc = adc_bases[self->adc_config.device];
|
||||
|
||||
init_flags[self->adc_config.device] = true;
|
||||
|
||||
#if defined(MCU_SAMD21)
|
||||
// Configuration SAMD21
|
||||
// Enable APBD clocks and PCHCTRL clocks; GCLK2 at 48 MHz
|
||||
PM->APBCMASK.reg |= PM_APBCMASK_ADC;
|
||||
GCLK->CLKCTRL.reg = GCLK_CLKCTRL_CLKEN | GCLK_CLKCTRL_GEN_GCLK2 | GCLK_CLKCTRL_ID_ADC;
|
||||
while (GCLK->STATUS.bit.SYNCBUSY) {
|
||||
}
|
||||
// Reset ADC registers
|
||||
adc->CTRLA.bit.SWRST = 1;
|
||||
while (adc->CTRLA.bit.SWRST) {
|
||||
}
|
||||
// Get the calibration data
|
||||
uint32_t bias = (*((uint32_t *)ADC_FUSES_BIASCAL_ADDR) & ADC_FUSES_BIASCAL_Msk) >> ADC_FUSES_BIASCAL_Pos;
|
||||
uint32_t linearity = (*((uint32_t *)ADC_FUSES_LINEARITY_0_ADDR) & ADC_FUSES_LINEARITY_0_Msk) >> ADC_FUSES_LINEARITY_0_Pos;
|
||||
linearity |= ((*((uint32_t *)ADC_FUSES_LINEARITY_1_ADDR) & ADC_FUSES_LINEARITY_1_Msk) >> ADC_FUSES_LINEARITY_1_Pos) << 5;
|
||||
/* Write the calibration data. */
|
||||
ADC->CALIB.reg = ADC_CALIB_BIAS_CAL(bias) | ADC_CALIB_LINEARITY_CAL(linearity);
|
||||
// Divide 48MHz clock by 32 to obtain 1.5 MHz clock to adc
|
||||
adc->CTRLB.reg = ADC_CTRLB_PRESCALER_DIV32;
|
||||
// Select external AREFA as reference voltage.
|
||||
adc->REFCTRL.reg = ADC_REFCTRL_REFSEL_AREFA;
|
||||
// Average: Accumulate samples and scale them down accordingly
|
||||
adc->AVGCTRL.reg = self->avg | ADC_AVGCTRL_ADJRES(self->avg);
|
||||
// Enable ADC and wait to be ready
|
||||
adc->CTRLA.bit.ENABLE = 1;
|
||||
while (adc->STATUS.bit.SYNCBUSY) {
|
||||
}
|
||||
|
||||
#elif defined(MCU_SAMD51)
|
||||
// Configuration SAMD51
|
||||
// Enable APBD clocks and PCHCTRL clocks; GCLK2 at 48 MHz
|
||||
if (self->adc_config.device == 0) {
|
||||
GCLK->PCHCTRL[ADC0_GCLK_ID].reg = GCLK_PCHCTRL_GEN_GCLK2 | GCLK_PCHCTRL_CHEN;
|
||||
MCLK->APBDMASK.bit.ADC0_ = 1;
|
||||
} else {
|
||||
GCLK->PCHCTRL[ADC1_GCLK_ID].reg = GCLK_PCHCTRL_GEN_GCLK2 | GCLK_PCHCTRL_CHEN;
|
||||
MCLK->APBDMASK.bit.ADC1_ = 1;
|
||||
}
|
||||
// Reset ADC registers
|
||||
adc->CTRLA.bit.SWRST = 1;
|
||||
while (adc->CTRLA.bit.SWRST) {
|
||||
}
|
||||
// Get the calibration data
|
||||
uint32_t biascomp;
|
||||
uint32_t biasr2r;
|
||||
uint32_t biasrefbuf;
|
||||
if (self->adc_config.device == 0) {
|
||||
biascomp = (*((uint32_t *)ADC0_FUSES_BIASCOMP_ADDR) & ADC0_FUSES_BIASCOMP_Msk) >> ADC0_FUSES_BIASCOMP_Pos;
|
||||
biasr2r = (*((uint32_t *)ADC0_FUSES_BIASR2R_ADDR) & ADC0_FUSES_BIASR2R_Msk) >> ADC0_FUSES_BIASR2R_Pos;
|
||||
biasrefbuf = (*((uint32_t *)ADC0_FUSES_BIASREFBUF_ADDR) & ADC0_FUSES_BIASREFBUF_Msk) >> ADC0_FUSES_BIASREFBUF_Pos;
|
||||
} else {
|
||||
biascomp = (*((uint32_t *)ADC1_FUSES_BIASCOMP_ADDR) & ADC1_FUSES_BIASCOMP_Msk) >> ADC1_FUSES_BIASCOMP_Pos;
|
||||
biasr2r = (*((uint32_t *)ADC1_FUSES_BIASR2R_ADDR) & ADC1_FUSES_BIASR2R_Msk) >> ADC1_FUSES_BIASR2R_Pos;
|
||||
biasrefbuf = (*((uint32_t *)ADC1_FUSES_BIASREFBUF_ADDR) & ADC1_FUSES_BIASREFBUF_Msk) >> ADC1_FUSES_BIASREFBUF_Pos;
|
||||
}
|
||||
/* Write the calibration data. */
|
||||
adc->CALIB.reg = ADC_CALIB_BIASCOMP(biascomp) | ADC_CALIB_BIASR2R(biasr2r) | ADC_CALIB_BIASREFBUF(biasrefbuf);
|
||||
// Divide 48MHz clock by 32 to obtain 1.5 MHz clock to adc
|
||||
adc->CTRLA.reg = ADC_CTRLA_PRESCALER_DIV32;
|
||||
// Select external AREFA as reference voltage.
|
||||
adc->REFCTRL.reg = ADC_REFCTRL_REFSEL_AREFA;
|
||||
// Average: Accumulate samples and scale them down accordingly
|
||||
adc->AVGCTRL.reg = self->avg | ADC_AVGCTRL_ADJRES(self->avg);
|
||||
// Enable ADC and wait to be ready
|
||||
adc->CTRLA.bit.ENABLE = 1;
|
||||
while (adc->SYNCBUSY.bit.ENABLE) {
|
||||
}
|
||||
|
||||
#endif
|
||||
}
|
||||
// Set the port as given in self->id as ADC
|
||||
mp_hal_set_pin_mux(self->id, ALT_FCT_ADC);
|
||||
}
|
@ -33,6 +33,7 @@
|
||||
#include "shared/runtime/pyexec.h"
|
||||
|
||||
extern uint8_t _sstack, _estack, _sheap, _eheap;
|
||||
extern void adc_deinit_all(void);
|
||||
|
||||
void samd_main(void) {
|
||||
mp_stack_set_top(&_estack);
|
||||
@ -62,6 +63,7 @@ void samd_main(void) {
|
||||
}
|
||||
|
||||
mp_printf(MP_PYTHON_PRINTER, "MPY: soft reboot\n");
|
||||
adc_deinit_all();
|
||||
gc_sweep_all();
|
||||
mp_deinit();
|
||||
}
|
||||
|
@ -129,8 +129,10 @@ STATIC const mp_rom_map_elem_t machine_module_globals_table[] = {
|
||||
{ MP_ROM_QSTR(MP_QSTR_mem16), MP_ROM_PTR(&machine_mem16_obj) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_mem32), MP_ROM_PTR(&machine_mem32_obj) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_unique_id), MP_ROM_PTR(&machine_unique_id_obj) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_Pin), MP_ROM_PTR(&machine_pin_type) },
|
||||
|
||||
{ MP_ROM_QSTR(MP_QSTR_ADC), MP_ROM_PTR(&machine_adc_type) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_LED), MP_ROM_PTR(&machine_led_type) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_Pin), MP_ROM_PTR(&machine_pin_type) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_SoftI2C), MP_ROM_PTR(&mp_machine_soft_i2c_type) },
|
||||
{ MP_ROM_QSTR(MP_QSTR_SoftSPI), MP_ROM_PTR(&mp_machine_soft_spi_type) },
|
||||
};
|
||||
|
@ -28,7 +28,8 @@
|
||||
|
||||
#include "py/obj.h"
|
||||
|
||||
extern const mp_obj_type_t machine_pin_type;
|
||||
extern const mp_obj_type_t machine_adc_type;
|
||||
extern const mp_obj_type_t machine_led_type;
|
||||
extern const mp_obj_type_t machine_pin_type;
|
||||
|
||||
#endif // MICROPY_INCLUDED_SAMD_MODMACHINE_H
|
||||
|
Loading…
x
Reference in New Issue
Block a user