esp32s2: Add support to LEDC PWM

This commit is contained in:
Alan Carvalho de Assis 2023-01-31 16:17:18 -03:00 committed by Xiang Xiao
parent d7b66adbeb
commit 39729bb635
5 changed files with 3736 additions and 0 deletions

View File

@ -442,6 +442,14 @@ config ESP32S2_I2C1
select ESP32S2_I2C
select I2C
config ESP32S2_LEDC
bool "LEDC (PWM)"
default n
select PWM
select ARCH_HAVE_PWM_MULTICHAN
---help---
Enable support to PWM on ESP32S2 using LEDC peripheral.
config ESP32S2_RT_TIMER
bool "Real-time Timer"
select ESP32S2_TIMER
@ -869,6 +877,91 @@ config ESP32S2_FREERUN
endmenu # Timer/Counter Configuration
menu "LEDC Configuration"
depends on ESP32S2_LEDC
menuconfig ESP32S2_LEDC_TIM0
bool "Timer 0"
default n
if ESP32S2_LEDC_TIM0
config ESP32S2_LEDC_TIM0_CHANNELS
int "Number of Timer 0 channels"
default 2
endif # ESP32S2_LEDC_TIM0
menuconfig ESP32S2_LEDC_TIM1
bool "Timer 1"
default n
if ESP32S2_LEDC_TIM1
config ESP32S2_LEDC_TIM1_CHANNELS
int "Number of Timer 1 channels"
default 2
endif # ESP32S2_LEDC_TIM1
menuconfig ESP32S2_LEDC_TIM2
bool "Timer 2"
default n
if ESP32S2_LEDC_TIM2
config ESP32S2_LEDC_TIM2_CHANNELS
int "Number of Timer 2 channels"
default 2
endif # ESP32S2_LEDC_TIM2
menuconfig ESP32S2_LEDC_TIM3
bool "Timer 3"
default n
if ESP32S2_LEDC_TIM3
config ESP32S2_LEDC_TIM3_CHANNELS
int "Number of Timer 3 channels"
default 2
endif # ESP32S2_LEDC_TIM2
config ESP32S2_LEDC_CHANNEL0_PIN
int "Channel 0 pin"
default 2
config ESP32S2_LEDC_CHANNEL1_PIN
int "Channel 1 pin"
default 3
config ESP32S2_LEDC_CHANNEL2_PIN
int "Channel 2 pin"
default 4
config ESP32S2_LEDC_CHANNEL3_PIN
int "Channel 3 pin"
default 5
config ESP32S2_LEDC_CHANNEL4_PIN
int "Channel 4 pin"
default 6
config ESP32S2_LEDC_CHANNEL5_PIN
int "Channel 5 pin"
default 7
config ESP32S2_LEDC_CHANNEL6_PIN
int "Channel 6 pin"
default 8
config ESP32S2_LEDC_CHANNEL7_PIN
int "Channel 7 pin"
default 9
endmenu # LEDC configuration
config ESP32S2_HAVE_OTA_PARTITION
bool
default n

View File

@ -60,6 +60,10 @@ ifeq ($(CONFIG_ESP32S2_I2S),y)
CHIP_CSRCS += esp32s2_i2s.c
endif
ifeq ($(CONFIG_ESP32S2_LEDC),y)
CHIP_CSRCS += esp32s2_ledc.c
endif
ifeq ($(CONFIG_ESP32S2_SPI),y)
CHIP_CSRCS += esp32s2_spi.c
endif

View File

@ -0,0 +1,813 @@
/****************************************************************************
* arch/xtensa/src/esp32s2/esp32s2_ledc.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <inttypes.h>
#include <stdint.h>
#include <stdbool.h>
#include <stdio.h>
#include <string.h>
#include <debug.h>
#include <errno.h>
#include "esp32s2_clockconfig.h"
#include "esp32s2_gpio.h"
#include "esp32s2_ledc.h"
#include "xtensa.h"
#include "hardware/esp32s2_ledc.h"
#include "hardware/esp32s2_system.h"
#include "hardware/esp32s2_gpio_sigmap.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/* LEDC total timers */
#define LEDC_TIMERS (4)
/* LEDC total channels */
#if defined(CONFIG_PWM_NCHANNELS) && CONFIG_PWM_NCHANNELS > 1
# define LEDC_CHANNELS (8)
#else
# define LEDC_CHANNELS (4)
#endif
/* LEDC timer0 channels and offset */
#ifdef CONFIG_ESP32S2_LEDC_TIM0
# if defined(CONFIG_PWM_NCHANNELS) && CONFIG_PWM_NCHANNELS > 1
# define LEDC_TIM0_CHANS CONFIG_ESP32S2_LEDC_TIM0_CHANNELS
# else
# define LEDC_TIM0_CHANS (1)
# endif
# define LEDC_TIM0_CHANS_OFF (0)
#endif
/* LEDC timer1 channels and offset */
#ifdef CONFIG_ESP32S2_LEDC_TIM1
# if defined(CONFIG_PWM_NCHANNELS) && CONFIG_PWM_NCHANNELS > 1
# define LEDC_TIM1_CHANS CONFIG_ESP32S2_LEDC_TIM1_CHANNELS
# else
# define LEDC_TIM1_CHANS (1)
# endif
# define LEDC_TIM1_CHANS_OFF (LEDC_TIM0_CHANS_OFF + LEDC_TIM0_CHANS)
#endif
/* LEDC timer2 channels and offset */
#ifdef CONFIG_ESP32S2_LEDC_TIM2
# if defined(CONFIG_PWM_NCHANNELS) && CONFIG_PWM_NCHANNELS > 1
# define LEDC_TIM2_CHANS CONFIG_ESP32S2_LEDC_TIM2_CHANNELS
# else
# define LEDC_TIM2_CHANS (1)
# endif
# define LEDC_TIM2_CHANS_OFF (LEDC_TIM1_CHANS_OFF + LEDC_TIM1_CHANS)
#endif
/* LEDC timer3 channels and offset */
#ifdef CONFIG_ESP32S2_LEDC_TIM3
# if defined(CONFIG_PWM_NCHANNELS) && CONFIG_PWM_NCHANNELS > 1
# define LEDC_TIM3_CHANS CONFIG_ESP32S2_LEDC_TIM3_CHANNELS
# else
# define LEDC_TIM3_CHANS (1)
# endif
# define LEDC_TIM3_CHANS_OFF (LEDC_TIM2_CHANS_OFF + LEDC_TIM2_CHANS)
#endif
/* LEDC clock resource */
#define LEDC_CLK_RES (1) /* APB clock */
/* LEDC timer max reload */
#define LEDC_RELOAD_MAX (16384) /* 2^14 */
/* LEDC timer max clock divider parameter */
#define LEDC_CLKDIV_MAX (262144) /* 2^18 */
/* LEDC timer registers mapping */
#define LEDC_TIMER_REG(r, n) ((r) + (n) * (LEDC_TIMER1_CONF_REG - \
LEDC_TIMER0_CONF_REG))
/* LEDC timer channel registers mapping */
#define setbits(bs, a) modifyreg32(a, 0, bs)
#define resetbits(bs, a) modifyreg32(a, bs, 0)
#define LEDC_CHAN_REG(r, n) ((r) + (n) * (LEDC_CH1_CONF0_REG - \
LEDC_CH0_CONF0_REG))
#define SET_TIMER_BITS(t, r, b) setbits(b, LEDC_TIMER_REG(r, (t)->num));
#define SET_TIMER_REG(t, r, v) putreg32(v, LEDC_TIMER_REG(r, (t)->num));
#define SET_CHAN_BITS(c, r, b) setbits(b, LEDC_CHAN_REG(r, (c)->num));
#define SET_CHAN_REG(c, r, v) putreg32(v, LEDC_CHAN_REG(r, (c)->num));
#ifndef MIN
# define MIN(a, b) (((a) < (b)) ? (a) : (b))
#endif
/****************************************************************************
* Private Types
****************************************************************************/
/* LEDC timer channel configuration */
struct esp32s2_ledc_chan_s
{
const uint8_t num; /* Timer channel ID */
const uint8_t pin; /* Timer channel GPIO pin number */
uint16_t duty; /* Timer channel current duty */
};
/* This structure represents the state of one LEDC timer */
struct esp32s2_ledc_s
{
const struct pwm_ops_s *ops; /* PWM operations */
const uint8_t num; /* Timer ID */
const uint8_t channels; /* Timer channels number */
struct esp32s2_ledc_chan_s *chans; /* Timer channels pointer */
uint32_t frequency; /* Timer current frequency */
uint32_t reload; /* Timer current reload */
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static int pwm_setup(struct pwm_lowerhalf_s *dev);
static int pwm_shutdown(struct pwm_lowerhalf_s *dev);
static int pwm_start(struct pwm_lowerhalf_s *dev,
const struct pwm_info_s *info);
static int pwm_stop(struct pwm_lowerhalf_s *dev);
static int pwm_ioctl(struct pwm_lowerhalf_s *dev, int cmd,
unsigned long arg);
/****************************************************************************
* Private Data
****************************************************************************/
/* LEDC PWM operations */
static const struct pwm_ops_s g_pwmops =
{
.setup = pwm_setup,
.shutdown = pwm_shutdown,
.start = pwm_start,
.stop = pwm_stop,
.ioctl = pwm_ioctl
};
/* LEDC channels table */
static struct esp32s2_ledc_chan_s g_ledc_chans[LEDC_CHANNELS] =
{
{
.num = 0,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL0_PIN
},
{
.num = 1,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL1_PIN
},
{
.num = 2,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL2_PIN
},
{
.num = 3,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL3_PIN
},
#if LEDC_CHANNELS > 4
{
.num = 4,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL4_PIN
},
{
.num = 5,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL5_PIN
},
{
.num = 6,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL6_PIN
},
{
.num = 7,
.pin = CONFIG_ESP32S2_LEDC_CHANNEL7_PIN
}
#endif
};
/* LEDC timer0 private data */
#ifdef CONFIG_ESP32S2_LEDC_TIM0
static struct esp32s2_ledc_s g_pwm0dev =
{
.ops = &g_pwmops,
.num = 0,
.channels = LEDC_TIM0_CHANS,
.chans = &g_ledc_chans[LEDC_TIM0_CHANS_OFF]
};
#endif /* CONFIG_ESP32S2_LEDC_TIM0 */
/* LEDC timer1 private data */
#ifdef CONFIG_ESP32S2_LEDC_TIM1
static struct esp32s2_ledc_s g_pwm1dev =
{
.ops = &g_pwmops,
.num = 1,
.channels = LEDC_TIM1_CHANS,
.chans = &g_ledc_chans[LEDC_TIM1_CHANS_OFF]
};
#endif /* CONFIG_ESP32S2_LEDC_TIM1 */
/* LEDC timer2 private data */
#ifdef CONFIG_ESP32S2_LEDC_TIM2
static struct esp32s2_ledc_s g_pwm2dev =
{
.ops = &g_pwmops,
.num = 2,
.channels = LEDC_TIM2_CHANS,
.chans = &g_ledc_chans[LEDC_TIM2_CHANS_OFF]
};
#endif /* CONFIG_ESP32S2_LEDC_TIM2 */
/* LEDC timer3 private data */
#ifdef CONFIG_ESP32S2_LEDC_TIM3
static struct esp32s2_ledc_s g_pwm3dev =
{
.ops = &g_pwmops,
.num = 3,
.channels = LEDC_TIM3_CHANS,
.chans = &g_ledc_chans[LEDC_TIM3_CHANS_OFF]
};
#endif /* CONFIG_ESP32S2_LEDC_TIM3 */
/* Clock reference count */
static uint32_t g_clk_ref;
/****************************************************************************
* Private functions
****************************************************************************/
/****************************************************************************
* Name: ledc_enable_clk
*
* Description:
* Enable LEDC clock.
*
* Input Parameters:
* None
*
* Returned Value:
* None.
*
****************************************************************************/
static void ledc_enable_clk(void)
{
irqstate_t flags;
flags = enter_critical_section();
if (g_clk_ref == 0)
{
setbits(SYSTEM_LEDC_CLK_EN, SYSTEM_PERIP_CLK_EN0_REG);
resetbits(SYSTEM_LEDC_RST, SYSTEM_PERIP_RST_EN0_REG);
putreg32(LEDC_CLK_RES, LEDC_CONF_REG);
putreg32(LEDC_CLK_EN, LEDC_CONF_REG);
pwminfo("Enable ledc clock\n");
}
g_clk_ref++;
leave_critical_section(flags);
}
/****************************************************************************
* Name: ledc_disable_clk
*
* Description:
* Disable LEDC clock.
*
* Input Parameters:
* None
*
* Returned Value:
* None.
*
****************************************************************************/
static void ledc_disable_clk(void)
{
irqstate_t flags;
flags = enter_critical_section();
g_clk_ref--;
if (g_clk_ref == 0)
{
pwminfo("Disable ledc clock\n");
setbits(SYSTEM_LEDC_RST, SYSTEM_PERIP_RST_EN0_REG);
resetbits(SYSTEM_LEDC_CLK_EN, SYSTEM_PERIP_CLK_EN0_REG);
}
leave_critical_section(flags);
}
/****************************************************************************
* Name: setup_timer
*
* Description:
* Setup LEDC timer frequency and reload.
*
* Input Parameters:
* priv - A reference to the LEDC timer state structure
*
* Returned Value:
* None
*
****************************************************************************/
static void setup_timer(struct esp32s2_ledc_s *priv)
{
irqstate_t flags;
uint32_t regval;
uint32_t reload;
uint32_t prescaler;
uint32_t shift = 1;
uint64_t pwmclk = esp_clk_apb_freq();
/* Reset timer */
SET_TIMER_BITS(priv, LEDC_TIMER0_CONF_REG, LEDC_TIMER0_RST);
/* Calculate optimal values for the timer prescaler and for the timer
* modulo register. If 'frequency' is the desired frequency, then
*
* tpmclk = pwmclk / presc
* frequency = tpmclk / reload
*
* ==>
*
* reload = pwmclk / presc / frequency
*
* In ESP32S2, there are 3 clock resources for PWM:
*
* 1. APB clock (80 MHz)
* 2. RTC clock (8 MHz)
* 3. XTAL_CLK
*
* We mostly use APB clock generally.
*
* There are many solutions to this, but the best solution will be the one
* that has the largest reload value and the smallest prescaler value.
* That is the solution that should give us the most accuracy in the timer
* control. Subject to:
*
* 2 <= presc <= 2^18(262144)
* 1 <= clkdiv <= 2^10
*
* clkdiv has 8-bit decimal precision, so
* clkdiv = pwmclk * 256 / 16384 / frequency would be optimal.
*
* Example:
*
* pwmclk = 80 MHz
* frequency = 100 Hz
*
* presc = 80,000,000 * 256 / 16,384 / 100
* = 12,500
* timclk = 80,000,000 / (12,500 / 256)
* = 1,638,400
* counter = 1,638,400 / 100
* = 16,384
* = 2^14
* shift = 14
*/
reload = (pwmclk * 256 / priv->frequency + LEDC_CLKDIV_MAX) /
LEDC_CLKDIV_MAX;
if (reload == 0)
{
reload = 1;
}
else if (reload > LEDC_RELOAD_MAX)
{
reload = LEDC_RELOAD_MAX;
}
for (int c = 2; c <= LEDC_RELOAD_MAX; c *= 2)
{
if (c * 2 > reload)
{
reload = c;
break;
}
shift++;
}
prescaler = pwmclk * 256 / reload / priv->frequency;
pwminfo("PWM timer%" PRIu8 " frequency=%0.4f reload=%" PRIu32 " shift=%"
PRIu32 " prescaler=%0.4f\n",
priv->num, (float)pwmclk / reload / ((float)prescaler / 256),
reload, shift, (float)prescaler / 256);
/* Store reload for channel duty */
priv->reload = reload;
flags = enter_critical_section();
/* Set timer clock divide and reload */
regval = (shift << LEDC_TIMER0_DUTY_RES_S) |
(prescaler << LEDC_CLK_DIV_TIMER0_S);
SET_TIMER_REG(priv, LEDC_TIMER0_CONF_REG, regval);
/* Setup to timer to use APB clock (80MHz) */
SET_TIMER_BITS(priv, LEDC_TIMER0_CONF_REG, LEDC_TICK_SEL_TIMER0);
/* Update clock divide and reload to hardware */
SET_TIMER_BITS(priv, LEDC_TIMER0_CONF_REG, LEDC_TIMER0_PARA_UP);
leave_critical_section(flags);
}
/****************************************************************************
* Name: setup_channel
*
* Description:
* Setup LEDC timer channel duty.
*
* Input Parameters:
* priv - A reference to the LEDC timer state structure
* cn - Timer channel number
*
* Returned Value:
* None
*
****************************************************************************/
static void setup_channel(struct esp32s2_ledc_s *priv, int cn)
{
irqstate_t flags;
uint32_t regval;
struct esp32s2_ledc_chan_s *chan = &priv->chans[cn];
/* Duty cycle:
*
* duty cycle = duty / 65536 * reload (fractional value)
*/
regval = b16toi(chan->duty * priv->reload + b16HALF);
pwminfo("channel=%" PRIu8 " duty=%" PRIu16 "(%0.4f) regval=%" PRIu32
" reload=%" PRIu32 "\n",
chan->num, chan->duty, (float)chan->duty / UINT16_MAX,
regval, priv->reload);
flags = enter_critical_section();
/* Reset config 0 & 1 registers */
SET_CHAN_REG(chan, LEDC_CH0_CONF0_REG, 0);
SET_CHAN_REG(chan, LEDC_CH0_CONF1_REG, 0);
/* Set pulse phase 0 */
SET_CHAN_REG(chan, LEDC_CH0_HPOINT_REG, 0);
/* Duty register uses bits [18:4] */
SET_CHAN_REG(chan, LEDC_CH0_DUTY_REG, regval << 4);
/* Start GPIO output */
SET_CHAN_BITS(chan, LEDC_CH0_CONF0_REG, LEDC_SIG_OUT_EN_CH0);
/* Start Duty counter */
SET_CHAN_BITS(chan, LEDC_CH0_CONF1_REG, LEDC_DUTY_START_CH0);
/* Update duty and phase to hardware */
SET_CHAN_BITS(chan, LEDC_CH0_CONF0_REG, LEDC_PARA_UP_CH0);
leave_critical_section(flags);
}
/****************************************************************************
* Name: pwm_setup
*
* Description:
* This method is called when the driver is opened. The lower half driver
* should configure and initialize the device so that it is ready for use.
* It should not, however, output pulses until the start method is called.
*
* Input Parameters:
* dev - A reference to the lower half PWM driver state structure
*
* Returned Value:
* Zero on success; a negated errno value on failure
*
****************************************************************************/
static int pwm_setup(struct pwm_lowerhalf_s *dev)
{
struct esp32s2_ledc_s *priv = (struct esp32s2_ledc_s *)dev;
pwminfo("PWM timer%u\n", priv->num);
ledc_enable_clk();
/* Setup channel GPIO pins */
for (int i = 0; i < priv->channels; i++)
{
pwminfo("channel%d --> pin%d\n", priv->chans[i].num,
priv->chans[i].pin);
esp32s2_configgpio(priv->chans[i].pin, OUTPUT | PULLUP);
esp32s2_gpio_matrix_out(priv->chans[i].pin,
LEDC_LS_SIG_OUT0_IDX + priv->chans[i].num,
0, 0);
}
return 0;
}
/****************************************************************************
* Name: pwm_shutdown
*
* Description:
* This method is called when the driver is closed. The lower half driver
* stop pulsed output, free any resources, disable the timer hardware, and
* put the system into the lowest possible power usage state
*
* Input Parameters:
* dev - A reference to the lower half PWM driver state structure
*
* Returned Value:
* Zero on success; a negated errno value on failure
*
****************************************************************************/
static int pwm_shutdown(struct pwm_lowerhalf_s *dev)
{
struct esp32s2_ledc_s *priv = (struct esp32s2_ledc_s *)dev;
#ifdef CONFIG_PWM_NCHANNELS
int channels = MIN(priv->channels, CONFIG_PWM_NCHANNELS);
#else
int channels = 1;
#endif
/* Stop timer */
pwm_stop(dev);
/* Clear timer and channel configuration */
priv->frequency = 0;
priv->reload = 0;
for (int i = 0; i < channels; i++)
{
priv->chans[i].duty = 0;
}
ledc_disable_clk();
return 0;
}
/****************************************************************************
* Name: pwm_start
*
* Description:
* (Re-)initialize the timer resources and start the pulsed output
*
* Input Parameters:
* dev - A reference to the lower half PWM driver state structure
* info - A reference to the characteristics of the pulsed output
*
* Returned Value:
* Zero on success; a negated errno value on failure
*
****************************************************************************/
static int pwm_start(struct pwm_lowerhalf_s *dev,
const struct pwm_info_s *info)
{
struct esp32s2_ledc_s *priv = (struct esp32s2_ledc_s *)dev;
#ifdef CONFIG_PWM_NCHANNELS
int channels = MIN(priv->channels, CONFIG_PWM_NCHANNELS);
#else
int channels = 1;
#endif
pwminfo("PWM timer%d\n", priv->num);
/* Update timer with given PWM timer frequency */
if (priv->frequency != info->frequency)
{
priv->frequency = info->frequency;
setup_timer(priv);
}
/* Update timer with given PWM channel duty */
for (int i = 0; i < channels; i++)
{
#ifdef CONFIG_PWM_NCHANNELS
if (priv->chans[i].duty != info->channels[i].duty)
#else
if (priv->chans[i].duty != info[i].duty)
#endif
{
#ifdef CONFIG_PWM_NCHANNELS
priv->chans[i].duty = info->channels[i].duty;
#else
priv->chans[i].duty = info[i].duty;
#endif
setup_channel(priv, i);
}
}
return 0;
}
/****************************************************************************
* Name: pwm_stop
*
* Description:
* Stop the pulsed output and reset the timer resources.
*
* Input Parameters:
* dev - A reference to the lower half PWM driver state structure
*
* Returned Value:
* Zero on success; a negated errno value on failure
*
****************************************************************************/
static int pwm_stop(struct pwm_lowerhalf_s *dev)
{
irqstate_t flags;
struct esp32s2_ledc_s *priv = (struct esp32s2_ledc_s *)dev;
pwminfo("PWM timer%d\n", priv->num);
flags = enter_critical_section();
/* Stop timer */
SET_TIMER_BITS(priv, LEDC_TIMER0_CONF_REG, LEDC_TIMER0_PAUSE);
/* Reset timer */
SET_TIMER_BITS(priv, LEDC_TIMER0_CONF_REG, LEDC_TIMER0_RST);
leave_critical_section(flags);
return 0;
}
/****************************************************************************
* Name: pwm_ioctl
*
* Description:
* Lower-half logic may support platform-specific ioctl commands
*
* Input Parameters:
* dev - A reference to the lower half PWM driver state structure
* cmd - The ioctl command
* arg - The argument accompanying the ioctl command
*
* Returned Value:
* Zero on success; a negated errno value on failure
*
****************************************************************************/
static int pwm_ioctl(struct pwm_lowerhalf_s *dev, int cmd,
unsigned long arg)
{
#ifdef CONFIG_DEBUG_PWM_INFO
struct esp32s2_ledc_s *priv = (struct esp32s2_ledc_s *)dev;
pwminfo("PWM timer%d\n", priv->num);
#endif
return -ENOTTY;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: esp32s2_ledc_init
*
* Description:
* Initialize one LEDC timer for use with the upper_level PWM driver.
*
* Input Parameters:
* timer - A number identifying the timer use.
*
* Returned Value:
* On success, a pointer to the ESP32S2 LEDC lower half PWM driver is
* returned. NULL is returned on any failure.
*
****************************************************************************/
struct pwm_lowerhalf_s *esp32s2_ledc_init(int timer)
{
struct esp32s2_ledc_s *lower = NULL;
pwminfo("TIM%u\n", timer);
switch (timer)
{
#ifdef CONFIG_ESP32S2_LEDC_TIM0
case 0:
lower = &g_pwm0dev;
break;
#endif
#ifdef CONFIG_ESP32S2_LEDC_TIM1
case 1:
lower = &g_pwm1dev;
break;
#endif
#ifdef CONFIG_ESP32S2_LEDC_TIM2
case 2:
lower = &g_pwm2dev;
break;
#endif
#ifdef CONFIG_ESP32S2_LEDC_TIM3
case 3:
lower = &g_pwm3dev;
break;
#endif
default:
pwmerr("ERROR: No such timer configured %d\n", timer);
lower = NULL;
break;
}
return (struct pwm_lowerhalf_s *)lower;
}

View File

@ -0,0 +1,52 @@
/****************************************************************************
* arch/xtensa/src/esp32s2/esp32s2_ledc.h
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
#ifndef __ARCH_XTENSA_SRC_ESP32S2_ESP32S2_LEDC_H
#define __ARCH_XTENSA_SRC_ESP32S2_ESP32S2_LEDC_H
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <nuttx/timers/pwm.h>
/****************************************************************************
* Public functions
****************************************************************************/
/****************************************************************************
* Name: esp32s2_ledc_init
*
* Description:
* Initialize one LEDC timer for use with the upper_level PWM driver.
*
* Input Parameters:
* timer - A number identifying the timer use.
*
* Returned Value:
* On success, a pointer to the ESP32S2-C3 LEDC lower half PWM driver is
* returned. NULL is returned on any failure.
*
****************************************************************************/
struct pwm_lowerhalf_s *esp32s2_ledc_init(int timer);
#endif /* __ARCH_RISCV_SRC_ESP32S2_ESP32S2_LEDC_H */

File diff suppressed because it is too large Load Diff