mirror of
https://github.com/micropython/micropython.git
synced 2025-12-16 09:50:15 +01:00
esp32/machine_uart: Implement UART.RX_IDLE based on machine.Timer.
The UART.IRQ_IDLE callback is called about two character times after the last byte, or 1 ms, whichever is larger. For the irq, timer 0 is used. machine_timer.c had to be reworked to make it's mechanisms available for machine_uart.c. The irq.flags() value is change only at a requested event. Otherwise keep the state. Signed-off-by: robert-hh <robert@hammelrath.com>
This commit is contained in:
@@ -38,6 +38,7 @@
|
||||
#include "hal/timer_hal.h"
|
||||
#include "hal/timer_ll.h"
|
||||
#include "soc/timer_periph.h"
|
||||
#include "machine_timer.h"
|
||||
|
||||
#define TIMER_DIVIDER 8
|
||||
|
||||
@@ -46,27 +47,8 @@
|
||||
|
||||
#define TIMER_FLAGS 0
|
||||
|
||||
typedef struct _machine_timer_obj_t {
|
||||
mp_obj_base_t base;
|
||||
|
||||
timer_hal_context_t hal_context;
|
||||
mp_uint_t group;
|
||||
mp_uint_t index;
|
||||
|
||||
mp_uint_t repeat;
|
||||
// ESP32 timers are 64 or 54-bit
|
||||
uint64_t period;
|
||||
|
||||
mp_obj_t callback;
|
||||
|
||||
intr_handle_t handle;
|
||||
|
||||
struct _machine_timer_obj_t *next;
|
||||
} machine_timer_obj_t;
|
||||
|
||||
const mp_obj_type_t machine_timer_type;
|
||||
|
||||
static void machine_timer_disable(machine_timer_obj_t *self);
|
||||
static mp_obj_t machine_timer_init_helper(machine_timer_obj_t *self, mp_uint_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args);
|
||||
|
||||
void machine_timer_deinit_all(void) {
|
||||
@@ -91,17 +73,16 @@ static void machine_timer_print(const mp_print_t *print, mp_obj_t self_in, mp_pr
|
||||
#endif
|
||||
}
|
||||
|
||||
static mp_obj_t machine_timer_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *args) {
|
||||
mp_arg_check_num(n_args, n_kw, 1, MP_OBJ_FUN_ARGS_MAX, true);
|
||||
#if CONFIG_IDF_TARGET_ESP32C3
|
||||
mp_uint_t group = mp_obj_get_int(args[0]) & 1;
|
||||
mp_uint_t index = 0;
|
||||
#else
|
||||
mp_uint_t group = (mp_obj_get_int(args[0]) >> 1) & 1;
|
||||
mp_uint_t index = mp_obj_get_int(args[0]) & 1;
|
||||
#endif
|
||||
machine_timer_obj_t *machine_timer_create(mp_uint_t timer) {
|
||||
|
||||
machine_timer_obj_t *self = NULL;
|
||||
#if CONFIG_IDF_TARGET_ESP32C3
|
||||
mp_uint_t group = timer & 1;
|
||||
mp_uint_t index = 0;
|
||||
#else
|
||||
mp_uint_t group = (timer >> 1) & 1;
|
||||
mp_uint_t index = timer & 1;
|
||||
#endif
|
||||
|
||||
// Check whether the timer is already initialized, if so use it
|
||||
for (machine_timer_obj_t *t = MP_STATE_PORT(machine_timer_obj_head); t; t = t->next) {
|
||||
@@ -120,6 +101,14 @@ static mp_obj_t machine_timer_make_new(const mp_obj_type_t *type, size_t n_args,
|
||||
self->next = MP_STATE_PORT(machine_timer_obj_head);
|
||||
MP_STATE_PORT(machine_timer_obj_head) = self;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
static mp_obj_t machine_timer_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *args) {
|
||||
mp_arg_check_num(n_args, n_kw, 1, MP_OBJ_FUN_ARGS_MAX, true);
|
||||
|
||||
// Create the new timer.
|
||||
machine_timer_obj_t *self = machine_timer_create(mp_obj_get_int(args[0]));
|
||||
|
||||
if (n_args > 1 || n_kw > 0) {
|
||||
mp_map_t kw_args;
|
||||
@@ -130,7 +119,7 @@ static mp_obj_t machine_timer_make_new(const mp_obj_type_t *type, size_t n_args,
|
||||
return self;
|
||||
}
|
||||
|
||||
static void machine_timer_disable(machine_timer_obj_t *self) {
|
||||
void machine_timer_disable(machine_timer_obj_t *self) {
|
||||
if (self->hal_context.dev != NULL) {
|
||||
// Disable the counter and alarm.
|
||||
timer_ll_enable_counter(self->hal_context.dev, self->index, false);
|
||||
@@ -162,7 +151,7 @@ static void machine_timer_isr(void *self_in) {
|
||||
}
|
||||
}
|
||||
|
||||
static void machine_timer_enable(machine_timer_obj_t *self) {
|
||||
void machine_timer_enable(machine_timer_obj_t *self, void (*timer_isr)) {
|
||||
// Initialise the timer.
|
||||
timer_hal_init(&self->hal_context, self->group, self->index);
|
||||
timer_ll_enable_counter(self->hal_context.dev, self->index, false);
|
||||
@@ -176,7 +165,7 @@ static void machine_timer_enable(machine_timer_obj_t *self) {
|
||||
timer_ll_clear_intr_status(self->hal_context.dev, TIMER_LL_EVENT_ALARM(self->index));
|
||||
ESP_ERROR_CHECK(
|
||||
esp_intr_alloc(timer_group_periph_signals.groups[self->group].timer_irq_id[self->index],
|
||||
TIMER_FLAGS, machine_timer_isr, self, &self->handle)
|
||||
TIMER_FLAGS, timer_isr, self, &self->handle)
|
||||
);
|
||||
timer_ll_enable_intr(self->hal_context.dev, TIMER_LL_EVENT_ALARM(self->index), true);
|
||||
|
||||
@@ -234,7 +223,7 @@ static mp_obj_t machine_timer_init_helper(machine_timer_obj_t *self, mp_uint_t n
|
||||
self->callback = args[ARG_callback].u_obj;
|
||||
self->handle = NULL;
|
||||
|
||||
machine_timer_enable(self);
|
||||
machine_timer_enable(self, machine_timer_isr);
|
||||
|
||||
return mp_const_none;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user