358 lines
13 KiB
C
358 lines
13 KiB
C
/*
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
*
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2016 Damien P. George
|
|
* Copyright (c) 2023 Vekatech Ltd.
|
|
*
|
|
* 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/runtime.h"
|
|
#include "py/mphal.h"
|
|
#include "py/mperrno.h"
|
|
#include "extmod/machine_pwm.h"
|
|
#include "pin.h"
|
|
#include "ra/ra_gpt.h"
|
|
#include "modmachine.h"
|
|
|
|
#if MICROPY_HW_ENABLE_HW_PWM
|
|
|
|
typedef struct _machine_pwm_obj_t {
|
|
mp_obj_base_t base;
|
|
R_GPT0_Type *pwm_inst;
|
|
uint8_t active;
|
|
uint8_t ch;
|
|
uint8_t id;
|
|
uint8_t duty;
|
|
uint32_t freq;
|
|
mp_hal_pin_obj_t pwm;
|
|
} machine_pwm_obj_t;
|
|
|
|
STATIC machine_pwm_obj_t machine_pwm_obj[] = {
|
|
#if defined(MICROPY_HW_PWM_0A)
|
|
{{&machine_pwm_type}, R_GPT0, 0, 0, 'A', 0, 0ul, MICROPY_HW_PWM_0A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_0B)
|
|
{{&machine_pwm_type}, R_GPT0, 0, 0, 'B', 0, 0ul, MICROPY_HW_PWM_0B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_1A)
|
|
{{&machine_pwm_type}, R_GPT1, 0, 1, 'A', 0, 0ul, MICROPY_HW_PWM_1A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_1B)
|
|
{{&machine_pwm_type}, R_GPT1, 0, 1, 'B', 0, 0ul, MICROPY_HW_PWM_1B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_2A)
|
|
{{&machine_pwm_type}, R_GPT2, 0, 2, 'A', 0, 0ul, MICROPY_HW_PWM_2A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_2B)
|
|
{{&machine_pwm_type}, R_GPT2, 0, 2, 'B', 0, 0ul, MICROPY_HW_PWM_2B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_3A)
|
|
{{&machine_pwm_type}, R_GPT3, 0, 3, 'A', 0, 0ul, MICROPY_HW_PWM_3A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_3B)
|
|
{{&machine_pwm_type}, R_GPT3, 0, 3, 'B', 0, 0ul, MICROPY_HW_PWM_3B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_4A)
|
|
{{&machine_pwm_type}, R_GPT4, 0, 4, 'A', 0, 0ul, MICROPY_HW_PWM_4A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_4B)
|
|
{{&machine_pwm_type}, R_GPT4, 0, 4, 'B', 0, 0ul, MICROPY_HW_PWM_4B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_5A)
|
|
{{&machine_pwm_type}, R_GPT5, 0, 5, 'A', 0, 0ul, MICROPY_HW_PWM_5A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_5B)
|
|
{{&machine_pwm_type}, R_GPT5, 0, 5, 'B', 0, 0ul, MICROPY_HW_PWM_5B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_6A)
|
|
{{&machine_pwm_type}, R_GPT6, 0, 6, 'A', 0, 0ul, MICROPY_HW_PWM_6A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_6B)
|
|
{{&machine_pwm_type}, R_GPT6, 0, 6, 'B', 0, 0ul, MICROPY_HW_PWM_6B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_7A)
|
|
{{&machine_pwm_type}, R_GPT7, 0, 7, 'A', 0, 0ul, MICROPY_HW_PWM_7A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_7B)
|
|
{{&machine_pwm_type}, R_GPT7, 0, 7, 'B', 0, 0ul, MICROPY_HW_PWM_7B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_8A)
|
|
{{&machine_pwm_type}, R_GPT8, 0, 8, 'A', 0, 0ul, MICROPY_HW_PWM_8A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_8B)
|
|
{{&machine_pwm_type}, R_GPT8, 0, 8, 'B', 0, 0ul, MICROPY_HW_PWM_8B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_9A)
|
|
{{&machine_pwm_type}, R_GPT9, 0, 9, 'A', 0, 0ul, MICROPY_HW_PWM_9A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_9B)
|
|
{{&machine_pwm_type}, R_GPT9, 0, 9, 'B', 0, 0ul, MICROPY_HW_PWM_9B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_10A)
|
|
{{&machine_pwm_type}, R_GPT10, 0, 10, 'A', 0, 0ul, MICROPY_HW_PWM_10A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_10B)
|
|
{{&machine_pwm_type}, R_GPT10, 0, 10, 'B', 0, 0ul, MICROPY_HW_PWM_10B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_11A)
|
|
{{&machine_pwm_type}, R_GPT11, 0, 11, 'A', 0, 0ul, MICROPY_HW_PWM_11A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_11B)
|
|
{{&machine_pwm_type}, R_GPT11, 0, 11, 'B', 0, 0ul, MICROPY_HW_PWM_11B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_12A)
|
|
{{&machine_pwm_type}, R_GPT12, 0, 12, 'A', 0, 0ul, MICROPY_HW_PWM_12A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_12B)
|
|
{{&machine_pwm_type}, R_GPT12, 0, 12, 'B', 0, 0ul, MICROPY_HW_PWM_12B},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_13A)
|
|
{{&machine_pwm_type}, R_GPT13, 0, 13, 'A', 0, 0ul, MICROPY_HW_PWM_13A},
|
|
#endif
|
|
#if defined(MICROPY_HW_PWM_13B)
|
|
{{&machine_pwm_type}, R_GPT13, 0, 13, 'B', 0, 0ul, MICROPY_HW_PWM_13B}
|
|
#endif
|
|
};
|
|
|
|
/******************************************************************************/
|
|
// MicroPython bindings for PWM
|
|
|
|
STATIC void mp_machine_pwm_print(const mp_print_t *print, mp_obj_t self_in, mp_print_kind_t kind) {
|
|
machine_pwm_obj_t *self = MP_OBJ_TO_PTR(self_in);
|
|
mp_printf(print, "PWM(GTIOC %d%c[#%d], runing=%u, freq=%u, duty=%u)", self->ch, self->id, self->pwm->pin, self->active, self->freq, self->duty);
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_init_helper(machine_pwm_obj_t *self, size_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) {
|
|
uint32_t D = 0ul;
|
|
|
|
enum { ARG_freq, ARG_duty };
|
|
static const mp_arg_t allowed_args[] = {
|
|
{ MP_QSTR_freq, MP_ARG_INT, {.u_int = -1} },
|
|
{ MP_QSTR_duty, MP_ARG_KW_ONLY | MP_ARG_INT, {.u_int = -1} }
|
|
};
|
|
|
|
mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)];
|
|
mp_arg_parse_all(n_args, pos_args, kw_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args);
|
|
|
|
if (args[ARG_freq].u_int != -1) {
|
|
if ((args[ARG_freq].u_int < 0) || (args[ARG_freq].u_int > 24000000)) {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("freq should be 0-24000000"));
|
|
} else {
|
|
self->freq = args[ARG_freq].u_int;
|
|
}
|
|
}
|
|
|
|
ra_gpt_timer_init(self->pwm->pin, self->ch, self->id, 0, (float)self->freq);
|
|
|
|
if (args[ARG_duty].u_int != -1) {
|
|
if ((args[ARG_duty].u_int < 0) || (args[ARG_duty].u_int > 100)) {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("duty should be 0-100"));
|
|
} else {
|
|
self->duty = args[ARG_duty].u_int;
|
|
}
|
|
}
|
|
|
|
D = self->duty * ra_gpt_timer_get_period(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, (uint32_t)(D / 100));
|
|
|
|
if (self->duty && self->freq) {
|
|
ra_gpt_timer_start(self->ch);
|
|
self->active = 1;
|
|
}
|
|
}
|
|
|
|
STATIC mp_obj_t mp_machine_pwm_make_new(const mp_obj_type_t *type, size_t n_args, size_t n_kw, const mp_obj_t *args) {
|
|
mp_hal_pin_obj_t pin_id = MP_OBJ_NULL;
|
|
machine_pwm_obj_t *self = MP_OBJ_NULL;
|
|
|
|
enum { ARG_pin, ARG_freq, ARG_duty };
|
|
static const mp_arg_t allowed_args[] = {
|
|
{ MP_QSTR_pin, MP_ARG_REQUIRED | MP_ARG_OBJ, {.u_obj = MP_OBJ_NULL} },
|
|
{ MP_QSTR_freq, MP_ARG_INT, {.u_int = -1} },
|
|
{ MP_QSTR_duty, MP_ARG_KW_ONLY | MP_ARG_INT, {.u_int = -1} }
|
|
};
|
|
|
|
mp_arg_check_num(n_args, n_kw, 1, 3, true);
|
|
mp_arg_val_t init_args[MP_ARRAY_SIZE(allowed_args)];
|
|
mp_arg_parse_all_kw_array(n_args, n_kw, args, MP_ARRAY_SIZE(allowed_args), allowed_args, init_args);
|
|
|
|
// Get GPIO and optional device to connect to PWM.
|
|
pin_id = mp_hal_get_pin_obj(init_args[ARG_pin].u_obj);
|
|
|
|
if (pin_id) {
|
|
for (int i = 0; i < MP_ARRAY_SIZE(machine_pwm_obj); i++) {
|
|
if (pin_id->pin == machine_pwm_obj[i].pwm->pin) {
|
|
self = &machine_pwm_obj[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (self) {
|
|
if (ra_gpt_timer_is_pwm_pin(self->pwm->pin)) {
|
|
// start the PWM running for this channel
|
|
mp_map_t kw_args;
|
|
mp_map_init_fixed_table(&kw_args, n_kw, args + n_args);
|
|
mp_machine_pwm_init_helper(self, n_args - 1, args + 1, &kw_args);
|
|
} else {
|
|
mp_raise_msg_varg(&mp_type_ValueError, MP_ERROR_TEXT("Pin(%d) has no timer output"), self->pwm->pin);
|
|
}
|
|
} else {
|
|
mp_raise_msg_varg(&mp_type_ValueError, MP_ERROR_TEXT("Pin(%d) is used with other peripheral"), pin_id->pin);
|
|
}
|
|
} else {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("Pin doesn't exist"));
|
|
}
|
|
|
|
return MP_OBJ_FROM_PTR(self);
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_deinit(machine_pwm_obj_t *self) {
|
|
ra_gpt_timer_deinit(self->pwm->pin, self->ch, self->id);
|
|
self->active = 0;
|
|
self->ch = 0;
|
|
self->id = ' ';
|
|
self->duty = 0;
|
|
self->freq = 0;
|
|
}
|
|
|
|
STATIC mp_obj_t mp_machine_pwm_freq_get(machine_pwm_obj_t *self) {
|
|
return MP_OBJ_NEW_SMALL_INT((uint32_t)ra_gpt_timer_get_freq(self->ch));
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_freq_set(machine_pwm_obj_t *self, mp_int_t freq) {
|
|
if (freq) {
|
|
ra_gpt_timer_set_freq(self->ch, (float)freq);
|
|
self->freq = (uint32_t)ra_gpt_timer_get_freq(self->ch);
|
|
if (!self->freq) {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("freq should be 0-24000000"));
|
|
} else
|
|
if (!self->active) {
|
|
ra_gpt_timer_start(self->ch);
|
|
self->active = 1;
|
|
}
|
|
} else {
|
|
ra_gpt_timer_stop(self->ch);
|
|
ra_gpt_timer_set_freq(self->ch, (float)freq);
|
|
self->freq = 0;
|
|
self->active = 0;
|
|
}
|
|
}
|
|
|
|
STATIC mp_obj_t mp_machine_pwm_duty_get(machine_pwm_obj_t *self) {
|
|
// give the result in %
|
|
uint64_t Dc = ra_gpt_timer_get_duty(self->ch, self->id) * 100;
|
|
return MP_OBJ_NEW_SMALL_INT(Dc / ra_gpt_timer_get_period(self->ch));
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_duty_set(machine_pwm_obj_t *self, mp_int_t duty) {
|
|
// assume duty is in %
|
|
if (duty < 0 || duty > 100) {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("duty should be 0-100"));
|
|
} else {
|
|
if (duty) {
|
|
uint64_t D = (uint8_t)(duty) * ra_gpt_timer_get_period(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, (uint32_t)(D / 100));
|
|
self->duty = (uint8_t)duty;
|
|
|
|
if (!self->active && self->freq) {
|
|
ra_gpt_timer_start(self->ch);
|
|
self->active = 1;
|
|
}
|
|
} else {
|
|
if (self->active) {
|
|
ra_gpt_timer_stop(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, 0);
|
|
self->duty = 0;
|
|
self->active = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
STATIC mp_obj_t mp_machine_pwm_duty_get_u16(machine_pwm_obj_t *self) {
|
|
// give the result in ratio (u16 / 65535)
|
|
uint64_t Dc = ra_gpt_timer_get_duty(self->ch, self->id) * 65535;
|
|
return MP_OBJ_NEW_SMALL_INT(Dc / ra_gpt_timer_get_period(self->ch));
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_duty_set_u16(machine_pwm_obj_t *self, mp_int_t duty_u16) {
|
|
// assume duty is a ratio (u16 / 65535)
|
|
if (duty_u16 < 0 || duty_u16 > 65535) {
|
|
mp_raise_ValueError(MP_ERROR_TEXT("duty should be 0-65535"));
|
|
} else {
|
|
if (duty_u16) {
|
|
uint64_t D = duty_u16 * ra_gpt_timer_get_period(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, (uint32_t)(D / 65535));
|
|
self->duty = (uint8_t)((duty_u16 * 100) / 65535);
|
|
|
|
if (!self->active && self->freq) {
|
|
ra_gpt_timer_start(self->ch);
|
|
self->active = 1;
|
|
}
|
|
} else {
|
|
if (self->active) {
|
|
ra_gpt_timer_stop(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, 0);
|
|
self->duty = 0;
|
|
self->active = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
STATIC mp_obj_t mp_machine_pwm_duty_get_ns(machine_pwm_obj_t *self) {
|
|
// give the result in ns
|
|
float ns = ra_gpt_timer_tick_time(self->ch);
|
|
ns *= (float)ra_gpt_timer_get_duty(self->ch, self->id);
|
|
return MP_OBJ_NEW_SMALL_INT(ns);
|
|
}
|
|
|
|
STATIC void mp_machine_pwm_duty_set_ns(machine_pwm_obj_t *self, mp_int_t duty_ns) {
|
|
// assume duty is ns
|
|
uint32_t ns_min = (uint32_t)ra_gpt_timer_tick_time(self->ch);
|
|
uint32_t ns_max = ns_min * ra_gpt_timer_get_period(self->ch);
|
|
if (duty_ns) {
|
|
if (duty_ns < ns_min || duty_ns > ns_max) {
|
|
mp_raise_msg_varg(&mp_type_ValueError, MP_ERROR_TEXT("duty should be in period range %d-%d"), ns_min, ns_max);
|
|
} else {
|
|
uint32_t D = (uint32_t)duty_ns / ns_min;
|
|
ra_gpt_timer_set_duty(self->ch, self->id, D);
|
|
D *= ns_min * 100;
|
|
self->duty = D / ns_max;
|
|
|
|
if (!self->active && self->freq) {
|
|
ra_gpt_timer_start(self->ch);
|
|
self->active = 1;
|
|
}
|
|
}
|
|
} else {
|
|
if (self->active) {
|
|
ra_gpt_timer_stop(self->ch);
|
|
ra_gpt_timer_set_duty(self->ch, self->id, 0);
|
|
self->duty = 0;
|
|
self->active = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif // MICROPY_HW_ENABLE_HW_PWM
|