从0学起的esp-idf之旅——外设篇ledc(led pwm)

官方例程详细注释

#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "driver/ledc.h"
#include "esp_err.h"

/*
 * 关于这个例子
 *
 * 1. 从初始化 LEDC 模块开始:
 *    a. 首先设置LEDC的定时器,这决定了PWM的频率和分辨率。
 *    b. 然后设置您要使用的 LEDC 通道,并与其中一个定时器进行绑定。
 *
 * 2. 您需要先安装默认的渐变功能,然后才能使用渐变 API。
 *
 * 3. 您还可以直接设置目标占空比而不用渐变功能。
 *
 * 4. 本例使用 GPIO18/19/4/5 作为 LEDC 输出,它会反复改变占空比。
 * 	通过给这几个引脚接上led可以看到呼吸灯的效果
 *
 * 5. GPIO18/19 来自高速通道组。GPIO4/5 来自低速通道组。
 *    
 */
#define LEDC_HS_TIMER          LEDC_TIMER_0
#define LEDC_HS_MODE           LEDC_HIGH_SPEED_MODE
#define LEDC_HS_CH0_GPIO       (18)
#define LEDC_HS_CH0_CHANNEL    LEDC_CHANNEL_0
#define LEDC_HS_CH1_GPIO       (19)
#define LEDC_HS_CH1_CHANNEL    LEDC_CHANNEL_1
#define LEDC_LS_TIMER          LEDC_TIMER_1
#define LEDC_LS_MODE           LEDC_LOW_SPEED_MODE

#define LEDC_LS_CH2_GPIO       (4)
#define LEDC_LS_CH2_CHANNEL    LEDC_CHANNEL_2
#define LEDC_LS_CH3_GPIO       (5)
#define LEDC_LS_CH3_CHANNEL    LEDC_CHANNEL_3

#define LEDC_TEST_CH_NUM       (4)
#define LEDC_TEST_DUTY         (4000)
#define LEDC_TEST_FADE_TIME    (3000)

void app_main(void)
{
    int ch;

    /*
     * 设置LEDC的定时器的配置
     */
    ledc_timer_config_t ledc_timer_ls = {
        .duty_resolution = LEDC_TIMER_13_BIT, // 设置分辨率,最大为2^13-1
        .freq_hz = 5000,                      // PWM信号频率
        .speed_mode = LEDC_LS_MODE,           // 定时器模式(“高速”或“低速”)
        .timer_num = LEDC_LS_TIMER,            // 设置定时器源(0-3)
        .clk_cfg = LEDC_AUTO_CLK,              // 配置LEDC时钟源(这里是自动选择)
    };
    // 初始化ledc的定时器配置
    ledc_timer_config(&ledc_timer_hs);
    ledc_timer_config_t ledc_timer_hs= {
        .duty_resolution = LEDC_TIMER_13_BIT, // 设置分辨率,最大为2^13-1
        .freq_hz = 5000,                      // PWM信号频率
        .speed_mode = LEDC_HS_MODE,           // 定时器模式(“高速”或“低速”)
        .timer_num = LEDC_HS_TIMER,            // 设置定时器源(0-3)
        .clk_cfg = LEDC_AUTO_CLK,              // 配置LEDC时钟源(这里是自动选择)
    };
    ledc_timer_config(&ledc_timer_hs);
    /*
     * 通过选择为 LEDC 控制器的每个通道准备单独的配置:
     * - 控制器的通道号
     * - 输出占空比,初始设置为 0
     * - LEDC 连接到的 GPIO 编号
     * - 速度模式,高或低
     * - 为LEDC通道指定定时器
     *   注意: 如果不同通道使用一个定时器,那么这些通道的频率和占空比分辨率将相同
     */
    ledc_channel_config_t ledc_channel[LEDC_TEST_CH_NUM] = {
        {
            .channel    = LEDC_HS_CH0_CHANNEL, 	// LEDC通道(0-7)
            .duty       = 0,					// 初始化占空比
            .gpio_num   = LEDC_HS_CH0_GPIO,		// pwm输出的gpio
            .speed_mode = LEDC_HS_MODE,			// 高速还是低速模式
            .hpoint     = 0,					// LEDC通道hpoint值
            .timer_sel  = LEDC_HS_TIMER			// LEDC依赖的定时器
        },
        {
            .channel    = LEDC_HS_CH1_CHANNEL,
            .duty       = 0,
            .gpio_num   = LEDC_HS_CH1_GPIO,
            .speed_mode = LEDC_HS_MODE,
            .hpoint     = 0,
            .timer_sel  = LEDC_HS_TIMER
        },
        {
            .channel    = LEDC_LS_CH2_CHANNEL,
            .duty       = 0,
            .gpio_num   = LEDC_LS_CH2_GPIO,
            .speed_mode = LEDC_LS_MODE,
            .hpoint     = 0,
            .timer_sel  = LEDC_LS_TIMER
        },
        {
            .channel    = LEDC_LS_CH3_CHANNEL,
            .duty       = 0,
            .gpio_num   = LEDC_LS_CH3_GPIO,
            .speed_mode = LEDC_LS_MODE,
            .hpoint     = 0,
            .timer_sel  = LEDC_LS_TIMER
        },
    };

    // 初始化ledc的通道
    for (ch = 0; ch < LEDC_TEST_CH_NUM; ch++) {
        ledc_channel_config(&ledc_channel[ch]);
    }

    // 安装渐变功能
    ledc_fade_func_install(0);

    while (1) {
        printf("1. LEDC fade up to duty = %d\n", LEDC_TEST_DUTY);
        for (ch = 0; ch < LEDC_TEST_CH_NUM; ch++) 
        {
        	// 设置LEDC渐变功能
            ledc_set_fade_with_time(ledc_channel[ch].speed_mode,ledc_channel[ch].channel, LEDC_TEST_DUTY, LEDC_TEST_FADE_TIME);
            // 开始执行渐变功能
            ledc_fade_start(ledc_channel[ch].speed_mode,ledc_channel[ch].channel, LEDC_FADE_NO_WAIT);
        }
        vTaskDelay(LEDC_TEST_FADE_TIME / portTICK_PERIOD_MS);

        printf("2. LEDC fade down to duty = 0\n");
        for (ch = 0; ch < LEDC_TEST_CH_NUM; ch++) 
        {
        	// 设置LEDC占空比渐变到0									
            ledc_set_fade_with_time(ledc_channel[ch].speed_mode,ledc_channel[ch].channel, 0, LEDC_TEST_FADE_TIME);
            ledc_fade_start(ledc_channel[ch].speed_mode,ledc_channel[ch].channel, LEDC_FADE_NO_WAIT);
        }
        vTaskDelay(LEDC_TEST_FADE_TIME / portTICK_PERIOD_MS);

        printf("3. LEDC set duty = %d without fade\n", LEDC_TEST_DUTY);
        for (ch = 0; ch < LEDC_TEST_CH_NUM; ch++) 
        {
        	// 直接改变占空比
            ledc_set_duty(ledc_channel[ch].speed_mode, ledc_channel[ch].channel, LEDC_TEST_DUTY);
            ledc_update_duty(ledc_channel[ch].speed_mode, ledc_channel[ch].channel);
        }
        vTaskDelay(1000 / portTICK_PERIOD_MS);

        printf("4. LEDC set duty = 0 without fade\n");
        for (ch = 0; ch < LEDC_TEST_CH_NUM; ch++) 
        {
            ledc_set_duty(ledc_channel[ch].speed_mode, ledc_channel[ch].channel, 0);
            ledc_update_duty(ledc_channel[ch].speed_mode, ledc_channel[ch].channel);
        }
        vTaskDelay(1000 / portTICK_PERIOD_MS);
    }
}

注意

关于这个pwm,乐鑫搞出了两个。led pwm区别于mcpwm,一个是用于ledc的简单的pwm生成器,另一个mcpwm则是用于电机的pwm生成器,分别对两种不同的情况有着不同的优化。比如led pwm中就加入了渐变的功能,可以方便的制作呼吸灯。频率越高,分别率也就越低。这个看个人需要去配置,乐鑫最高支持 40 MHz的pwm,但是只能实现百分之50占空比的方波,也就是duty_resolution = 2^1-1。官方的例子中支持了esp32c3和esp32s2导致代码比较多,这里做了删减只支持eps32
支持S2和C3的部分

解析官方例子

整个过程基本就是先配置底层的定时器,然后配置通道,最后如果需要渐变的功能就安装渐变功能的驱动。然后就进行整体的调用,整个代码用到了宏,关键的信息我们可以通过宏去直接修改。这里对于他的通道还有个疑问就是hpoint是什么?
hpoint
这里是官方手册里的图上面的一直上升的信号指的是定时器,定时器一直递增的基数,计数到溢出。这里hpoint指的是一个周期内上升沿的时间点。通过占空比和hpoint实际上lpoint是能算出来的,故没有接口去设置,实际上在大多数情况下一个周期内什么时候开始上升沿并不重要。所以一般情况都是0。

总结

ledc总结脑图

相关官方资料

ledc例程:https://github.com/espressif/esp-idf/tree/release/v4.3/examples/peripherals/ledc
ledc相关:https://docs.espressif.com/projects/esp-idf/zh_CN/latest/esp32/api-reference/peripherals/ledc.html
技术参考手册:https://www.espressif.com/sites/default/files/documentation/esp32_technical_reference_manual_cn.pdf

ESP-IDFEspressif IoT Development Framework)是Espressif Systems为开发基于其ESP32ESP32-S系列芯片的物联网设备提供的开源框架。它包含了一个丰富的库,其中包括PWM(Pulse Width Modulation)模块,用于控制数字信号的占空比,常用于电机驱动、LED照明、传感器采样等场景。 在ESP-IDF中,PWM可以通过`esp_timer`或`driver/pwm`模块来管理。例如,你可以创建一个PWM通道并设置其频率、周期以及高电平持续时间(即脉宽)。下面是一个简单的例子: ```c #include "esp_pwm.h" esp_err_t create_pwm_channel(esp_pwm_handle_t *handle, int channel, uint32_t frequency) { esp_pwm_config_t config = { .duty_cycle = 0, // 初始占空比为0 .freq = frequency, .pin_num = channel, .mode = PWM_MODE_MS, .type = PWM_WIDTH Schneider, }; return esp_pwm_create(handle, &config); } void setup_pwm(void) { esp_pwm_handle_t pwm_handle; if (create_pwm_channel(&pwm_handle, GPIO_ID_0, 5000) == ESP_OK) { // 设置GPIO作为PWM输出 gpio_set_direction(GPIO_ID_0, GPIO_MODE_OUTPUT); // 开始PWM定时器 esp_pwm_start(pwm_handle); // 更新PWM duty cycle esp_pwm_set_duty(pwm_handle, 50); // 设置50%占空比 // 关闭PWM定时器时记得释放资源 esp_pwm_stop(pwm_handle); esp_pwm_delete(pwm_handle); } else { ESP_LOGE(TAG, "Failed to create PWM handle"); } } ``` 在这个例子中,我们首先创建了一个PWM通道,并设置了5kHz的频率,然后将GPIO 0配置为PWM输出。通过`esp_pwm_set_duty`函数可以改变PWM的占空比。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值