mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
387 lines
12 KiB
C
387 lines
12 KiB
C
/*
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_esp32
|
|
* @ingroup drivers_periph_pwm
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level PWM driver implementation
|
|
*
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
* @}
|
|
*/
|
|
|
|
#include "bitarithm.h"
|
|
#include "board.h"
|
|
#include "cpu.h"
|
|
#include "gpio_arch.h"
|
|
#include "kernel_defines.h"
|
|
#include "log.h"
|
|
#include "irq_arch.h"
|
|
#include "periph/pwm.h"
|
|
#include "periph/gpio.h"
|
|
|
|
#include "esp_common.h"
|
|
#include "esp_rom_gpio.h"
|
|
#include "hal/ledc_hal.h"
|
|
#include "soc/ledc_struct.h"
|
|
#include "soc/rtc.h"
|
|
|
|
#include "esp_idf_api/periph_ctrl.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#if defined(PWM0_GPIOS) || defined(PWM1_GPIOS) || defined(PWM2_GPIOS) || defined(PWM3_GPIOS)
|
|
|
|
#define SOC_LEDC_CLK_DIV_BIT_NUM 18
|
|
#define SOC_LEDC_CLK_DIV_INT_BIT_NUM 10 /* integral part of CLK divider */
|
|
#define SOC_LEDC_CLK_DIV_FRAC_BIT_NUM 8 /* fractional part of CLK divider */
|
|
|
|
#define PWM_HW_RES_MAX ((uint32_t)1 << SOC_LEDC_TIMER_BIT_WIDE_NUM)
|
|
#define PWM_HW_RES_MIN ((uint32_t)1 << 1)
|
|
|
|
#define _DEV _pwm_dev[pwm] /* shortcut for PWM device descriptor */
|
|
#define _CFG pwm_config[pwm] /* shortcut for PWM device configuration */
|
|
|
|
/* data structure for dynamic channel parameters */
|
|
typedef struct {
|
|
uint32_t duty; /* actual duty value */
|
|
uint32_t hpoint; /* actual hpoing value */
|
|
bool used; /* true if the channel is set by pwm_set */
|
|
uint8_t ch; /* actual channel index within used channel group */
|
|
} _pwm_ch_t;
|
|
|
|
/* data structure for device handling at runtime */
|
|
typedef struct {
|
|
uint32_t freq; /* specified frequency parameter */
|
|
uint32_t res; /* specified resolution parameter */
|
|
uint32_t hw_freq; /* used hardware frequency */
|
|
uint32_t hw_res; /* used hardware resolution */
|
|
uint32_t hw_clk_div; /* used hardware clock divider */
|
|
_pwm_ch_t ch[PWM_CH_NUMOF_MAX]; /* dynamic channel parameters at runtime */
|
|
ledc_hal_context_t hw; /* used hardware device context */
|
|
pwm_mode_t mode; /* specified mode */
|
|
ledc_timer_bit_t hw_res_bit; /* used hardware resolution in bit */
|
|
bool enabled; /* true if the device is used (powered on) */
|
|
} _pwm_dev_t;
|
|
|
|
static _pwm_dev_t _pwm_dev[PWM_NUMOF] = { };
|
|
|
|
/* if pwm_init is called first time, it checks the pwm configuration */
|
|
static bool _pwm_initialized = false;
|
|
|
|
/* static configuration checks and initialization on first pwm_init */
|
|
static bool _pwm_initialize(void);
|
|
|
|
/* Initialize PWM device */
|
|
uint32_t pwm_init(pwm_t pwm, pwm_mode_t mode, uint32_t freq, uint16_t res)
|
|
{
|
|
_Static_assert(PWM_NUMOF <= PWM_NUMOF_MAX, "Too many PWM devices defined");
|
|
|
|
if (!_pwm_initialized) {
|
|
if (!_pwm_initialize()) {
|
|
return 0;
|
|
}
|
|
_pwm_initialized = true;
|
|
}
|
|
|
|
assert(pwm < PWM_NUMOF);
|
|
assert(freq > 0);
|
|
|
|
_DEV.enabled = true;
|
|
_DEV.res = res;
|
|
_DEV.freq = freq;
|
|
_DEV.mode = mode;
|
|
|
|
if ((res < PWM_HW_RES_MIN) || (_DEV.res > PWM_HW_RES_MAX)) {
|
|
LOG_TAG_ERROR("pwm", "Resolution of PWM device %u to be in "
|
|
"range [%"PRIu32", %"PRIu32"]\n",
|
|
pwm, PWM_HW_RES_MIN, PWM_HW_RES_MAX);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* The hardware resolution must be a power of two, so we determine the
|
|
* next power of two, which covers the desired resolution
|
|
*/
|
|
ledc_timer_bit_t hw_res_bit = bitarithm_msb(res - 1);
|
|
if (hw_res_bit < SOC_LEDC_TIMER_BIT_WIDE_NUM) {
|
|
hw_res_bit++;
|
|
}
|
|
|
|
uint32_t hw_res = 1 << hw_res_bit;
|
|
|
|
uint32_t hw_ticks_max = rtc_clk_apb_freq_get();
|
|
uint32_t hw_ticks_min = hw_ticks_max / (1 << SOC_LEDC_CLK_DIV_INT_BIT_NUM);
|
|
uint32_t hw_freq_min = hw_ticks_min / (1 << SOC_LEDC_TIMER_BIT_WIDE_NUM) + 1;
|
|
|
|
if (freq < hw_freq_min) {
|
|
LOG_TAG_ERROR("pwm", "Frequency of %"PRIu32" Hz is too less, minimum "
|
|
"frequency is %"PRIu32" Hz\n", freq, hw_freq_min);
|
|
return 0;
|
|
}
|
|
|
|
/* number of hardware ticks required, at maximum it can be APB clock */
|
|
uint32_t hw_ticks = MIN(freq * hw_res, rtc_clk_apb_freq_get());
|
|
|
|
/*
|
|
* if the number of required ticks is less than minimum ticks supported by
|
|
* the hardware supports, we have to increase the resolution.
|
|
*/
|
|
while (hw_ticks < hw_ticks_min) {
|
|
hw_res_bit++;
|
|
hw_res = 1 << hw_res_bit;
|
|
hw_ticks = freq * hw_res;
|
|
}
|
|
|
|
/* LEDC_CLK_DIV is given in Q10.8 format */
|
|
uint32_t hw_clk_div =
|
|
((uint64_t)rtc_clk_apb_freq_get() << SOC_LEDC_CLK_DIV_FRAC_BIT_NUM) / hw_ticks;
|
|
|
|
_DEV.freq = freq;
|
|
_DEV.res = res;
|
|
_DEV.hw_freq = hw_ticks / hw_res;
|
|
_DEV.hw_res = hw_res;
|
|
_DEV.hw_res_bit = hw_res_bit;
|
|
_DEV.hw_clk_div = hw_clk_div;
|
|
|
|
DEBUG("%s hw_freq=%"PRIu32" hw_res=%"PRIu32" hw_ticks=%"PRIu32
|
|
" hw_clk_div=%"PRIu32"\n", __func__,
|
|
_DEV.hw_freq, _DEV.hw_res, hw_ticks, _DEV.hw_clk_div);
|
|
|
|
for (int i = 0; i < _CFG.ch_numof; i++) {
|
|
/* initialize channel data */
|
|
_DEV.ch[i].used = false;
|
|
_DEV.ch[i].duty = 0;
|
|
|
|
/* reset GPIO usage type if the pins were used already for PWM before
|
|
to make it possible to reinitialize PWM with new parameters */
|
|
if (gpio_get_pin_usage(_CFG.gpios[i]) == _PWM) {
|
|
gpio_set_pin_usage(_CFG.gpios[i], _GPIO);
|
|
}
|
|
|
|
if (gpio_get_pin_usage(_CFG.gpios[i]) != _GPIO) {
|
|
LOG_TAG_ERROR("pwm", "GPIO%d is used for %s and cannot be used as "
|
|
"PWM output\n",
|
|
i, gpio_get_pin_usage_str(_CFG.gpios[i]));
|
|
return 0;
|
|
}
|
|
|
|
/* initialize the GPIOs and route the PWM signal output to the GPIO */
|
|
if (gpio_init(_CFG.gpios[i], GPIO_OUT) < 0) {
|
|
return 0;
|
|
}
|
|
|
|
gpio_set_pin_usage(_CFG.gpios[i], _PWM);
|
|
gpio_clear(_CFG.gpios[i]);
|
|
|
|
esp_rom_gpio_connect_out_signal(
|
|
_CFG.gpios[i],
|
|
ledc_periph_signal[_CFG.group].sig_out0_idx + _DEV.ch[i].ch, 0, 0);
|
|
|
|
}
|
|
|
|
pwm_poweron(pwm);
|
|
|
|
return _DEV.hw_freq;
|
|
}
|
|
|
|
uint8_t pwm_channels(pwm_t pwm)
|
|
{
|
|
assert(pwm < PWM_NUMOF);
|
|
return _CFG.ch_numof;
|
|
}
|
|
|
|
void pwm_set(pwm_t pwm, uint8_t channel, uint16_t value)
|
|
{
|
|
DEBUG("%s pwm=%u channel=%u value=%u\n", __func__, pwm, channel, value);
|
|
|
|
assert(pwm < PWM_NUMOF);
|
|
assert(channel < _CFG.ch_numof);
|
|
|
|
value = MIN(value, _DEV.res);
|
|
|
|
_DEV.ch[channel].used = true;
|
|
_DEV.ch[channel].duty = value * _DEV.hw_res / _DEV.res;
|
|
_DEV.ch[channel].hpoint = 0;
|
|
|
|
switch (_DEV.mode) {
|
|
case PWM_LEFT:
|
|
_DEV.ch[channel].hpoint = 0;
|
|
break;
|
|
case PWM_RIGHT:
|
|
_DEV.ch[channel].hpoint = _DEV.hw_res - 1 - _DEV.ch[channel].duty;
|
|
break;
|
|
case PWM_CENTER:
|
|
_DEV.ch[channel].hpoint = (_DEV.hw_res - _DEV.ch[channel].duty) >> 1;
|
|
break;
|
|
}
|
|
|
|
DEBUG("%s pwm=%u duty=%"PRIu32" hpoint=%"PRIu32"\n",
|
|
__func__, pwm, _DEV.ch[channel].duty, _DEV.ch[channel].hpoint);
|
|
|
|
unsigned ch = _DEV.ch[channel].ch; /* internal channel mapping */
|
|
|
|
critical_enter();
|
|
ledc_hal_set_duty_int_part(&_DEV.hw, ch, _DEV.ch[channel].duty);
|
|
ledc_hal_set_hpoint(&_DEV.hw, ch, _DEV.ch[channel].hpoint);
|
|
ledc_hal_set_sig_out_en(&_DEV.hw, ch, true);
|
|
ledc_hal_ls_channel_update(&_DEV.hw, ch);
|
|
ledc_hal_set_duty_start(&_DEV.hw, ch, true);
|
|
critical_exit();
|
|
}
|
|
|
|
void pwm_poweron(pwm_t pwm)
|
|
{
|
|
DEBUG("%s pwm=%u\n", __func__, pwm);
|
|
|
|
/* enable and init the module and select the right clock source */
|
|
esp_idf_periph_module_enable(_CFG.module);
|
|
ledc_hal_init(&_DEV.hw, _CFG.group);
|
|
ledc_hal_set_slow_clk_sel(&_DEV.hw, LEDC_SLOW_CLK_APB);
|
|
ledc_hal_set_clock_source(&_DEV.hw, _CFG.timer, LEDC_APB_CLK);
|
|
|
|
/* update the timer according to determined parameters */
|
|
ledc_hal_set_clock_divider(&_DEV.hw, _CFG.timer, _DEV.hw_clk_div);
|
|
ledc_hal_set_duty_resolution(&_DEV.hw, _CFG.timer, _DEV.hw_res_bit);
|
|
ledc_hal_ls_timer_update(&_DEV.hw, _CFG.timer);
|
|
ledc_hal_timer_rst(&_DEV.hw, _CFG.timer);
|
|
|
|
critical_enter();
|
|
for (unsigned i = 0; i < _CFG.ch_numof; i++) {
|
|
/* static configuration of the channel, no fading */
|
|
ledc_hal_set_duty_direction(&_DEV.hw, _DEV.ch[i].ch, 1);
|
|
ledc_hal_set_duty_num(&_DEV.hw, _DEV.ch[i].ch, 1);
|
|
ledc_hal_set_duty_cycle(&_DEV.hw, _DEV.ch[i].ch, 1);
|
|
ledc_hal_set_duty_scale(&_DEV.hw, _DEV.ch[i].ch, 0);
|
|
ledc_hal_set_fade_end_intr(&_DEV.hw, _DEV.ch[i].ch, 0);
|
|
|
|
/* bind the channel to the timer and disable the output for now */
|
|
ledc_hal_bind_channel_timer(&_DEV.hw, _DEV.ch[i].ch, _CFG.timer);
|
|
|
|
/* restore used parameters */
|
|
ledc_hal_set_duty_int_part(&_DEV.hw, _DEV.ch[i].ch, _DEV.ch[i].duty);
|
|
ledc_hal_set_hpoint(&_DEV.hw, _DEV.ch[i].ch, _DEV.ch[i].hpoint);
|
|
ledc_hal_set_sig_out_en(&_DEV.hw, _DEV.ch[i].ch, _DEV.ch[i].used);
|
|
ledc_hal_ls_channel_update(&_DEV.hw, _DEV.ch[i].ch);
|
|
ledc_hal_set_duty_start(&_DEV.hw, _DEV.ch[i].ch, _DEV.ch[i].used);
|
|
}
|
|
_DEV.enabled = true;
|
|
critical_exit();
|
|
}
|
|
|
|
void pwm_poweroff(pwm_t pwm)
|
|
{
|
|
DEBUG("%s pwm=%u\n", __func__, pwm);
|
|
|
|
if (!_pwm_dev[pwm].enabled) {
|
|
return;
|
|
}
|
|
|
|
unsigned i;
|
|
|
|
/* disable the signal of all channels */
|
|
critical_enter();
|
|
for (i = 0; i < _CFG.ch_numof; i++) {
|
|
ledc_hal_set_idle_level(&_DEV.hw, _DEV.ch[i].ch, 0);
|
|
ledc_hal_set_sig_out_en(&_DEV.hw, _DEV.ch[i].ch, false);
|
|
ledc_hal_set_duty_start(&_DEV.hw, _DEV.ch[i].ch, false);
|
|
ledc_hal_ls_channel_update(&_DEV.hw, _DEV.ch[i].ch);
|
|
}
|
|
_DEV.enabled = false;
|
|
critical_exit();
|
|
|
|
/* check whether all devices of the same hardware module are disabled */
|
|
for (i = 0; i < PWM_NUMOF; i++) {
|
|
if ((_CFG.module == pwm_config[i].module) && _pwm_dev[i].enabled) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* if all devices of the same hardware module are disable, it is powered off */
|
|
if (i == PWM_NUMOF) {
|
|
esp_idf_periph_module_disable(_CFG.module);
|
|
}
|
|
}
|
|
|
|
void pwm_print_config(void)
|
|
{
|
|
for (unsigned pwm = 0; pwm < PWM_NUMOF; pwm++) {
|
|
printf("\tPWM_DEV(%d)\tchannels=[ ", pwm);
|
|
for (int i = 0; i < _CFG.ch_numof; i++) {
|
|
printf("%d ", _CFG.gpios[i]);
|
|
}
|
|
printf("]\n");
|
|
}
|
|
}
|
|
|
|
/* do static configuration checks */
|
|
static bool _pwm_initialize(void)
|
|
{
|
|
unsigned ch_numof[2] = {};
|
|
|
|
for (unsigned pwm = 0; pwm < PWM_NUMOF; pwm++) {
|
|
/* compute the channel indices */
|
|
for (unsigned i = 0; i < _CFG.ch_numof; i++) {
|
|
_pwm_dev[pwm].ch[i].ch = ch_numof[_CFG.group] + i;
|
|
}
|
|
ch_numof[_CFG.group] += _CFG.ch_numof;
|
|
if (_CFG.ch_numof > PWM_CH_NUMOF_MAX) {
|
|
LOG_TAG_ERROR("pwm", "Number of PWM channels of device %d is %d, "
|
|
"only %d channels per PWM device are supported\n",
|
|
pwm, _CFG.ch_numof, PWM_CH_NUMOF_MAX);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
unsigned total_ch_numof = ch_numof[0] + ch_numof[1];
|
|
|
|
if (total_ch_numof > (SOC_LEDC_CHANNEL_NUM * ARRAY_SIZE(ledc_periph_signal))) {
|
|
LOG_TAG_ERROR("pwm", "Total number of PWM channels is %d, only "
|
|
"%d channels are supported at maximum\n", total_ch_numof,
|
|
PWM_CH_NUMOF_MAX * ARRAY_SIZE(ledc_periph_signal));
|
|
return false;
|
|
}
|
|
|
|
bool multiple_used = false;
|
|
for (unsigned pi = 0; pi < PWM_NUMOF; pi++) {
|
|
for (unsigned ci = 0; ci < pwm_config[pi].ch_numof; ci++) {
|
|
for (unsigned pj = 0; pj < PWM_NUMOF; pj++) {
|
|
if (pi == pj) {
|
|
continue;
|
|
}
|
|
for (unsigned cj = 0; cj < pwm_config[pj].ch_numof; cj++) {
|
|
if (pwm_config[pi].gpios[ci] == pwm_config[pj].gpios[cj]) {
|
|
LOG_TAG_ERROR("pwm", "GPIO%d is used multiple times in "
|
|
"PWM devices %d and %d\n",
|
|
pwm_config[pi].gpios[ci], pi, pj);
|
|
multiple_used = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (multiple_used) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
#else
|
|
|
|
void pwm_print_config(void)
|
|
{
|
|
LOG_TAG_INFO("pwm", "no PWM devices\n");
|
|
}
|
|
|
|
#endif
|