从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

  • 4
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 您好,ESP32是一款非常流行的微控制器,而ESP-IDF是它的开发框架。如果您想学习ESP-IDF的使用,可以参考以下步骤: 1. 准备开发环境:您需要安装ESP-IDF的开发环境,包括ESP-IDF工具链、Python、Git等。具体安装步骤可以参考ESP-IDF官方文档。 2. 学习ESP-IDF的基本概念:ESP-IDF是一个基于FreeRTOS的开发框架,它提供了许多API和组件,可以方便地开发ESP32的应用程序。您需要学习ESP-IDF的基本概念,包括任务、队列、定时器、中断等。 3. 编写应用程序:您可以使用ESP-IDF提供的示例程序作为参考,编写自己的应用程序。ESP-IDF提供了许多组件,包括WiFi、蓝牙、SPI、I2C等,您可以根据自己的需求选择相应的组件。 4. 调试和测试:在编写应用程序的过程中,您需要进行调试和测试。ESP-IDF提供了许多调试工具,包括GDB调试器、串口调试工具等。 总之,学习ESP-IDF需要一定的编程基础和硬件知识,但是它可以帮助您快速开发ESP32的应用程序。希望以上信息能对您有所帮助。 ### 回答2: ESP32Espressif Systems推出的一款支持Wi-Fi和蓝牙功能的微型芯片,由于其出色的性能和低成本,成为了物联网领域中广泛使用的芯片之一。ESP-IDFESP32的官方开发框架,为用户提供一套完善的开发工具和资源,以便于用户轻松开发出稳定和高效的应用程序。 ESP-IDF教学主要包括以下方面内容: 1.ESP-IDF开发环境搭建。 ESP-IDF适用于Linux、Windows和macOS系统,并且可以配合使用命令行、VS Code等工具,极大地方便了开发者的使用。搭建好开发环境对于后续的代码编写和调试至关重要。 2.ESP-IDF应用程序架构。 ESP-IDF的应用程序架构分为三层,包括API层、组件层、和应用层,这三个层次需要开发者熟练掌握。 3.ESP-IDF API函数使用。 ESP-IDF具有众多的API函数,可以实现GPIO控制、Wi-Fi和蓝牙通信、定时器、SPI和I2C接口等功能,学习这些函数的使用方法可以方便开发者快速搭建出各种应用程序。 4.ESP-IDF组件使用。 ESP-IDF还提供了多种组件,包括操作系统、TCP/IP协议栈、驱动程序、框架和示例,这些组件可以极大地提升开发效率,缩短开发周期。 5.ESP-IDF应用程序调试。 ESP-IDF提供了集成式调试器和自动化测试框架,可以帮助开发者快速排查问题并进行黑盒测试。 总结来说,学习ESP-IDF需要开发者具备一定的编程经验和开发基础,同时需要深入了解ESP32的基础知识和硬件架构,熟练使用ESP-IDF可以为开发者带来高效、稳定、可靠的应用程序开发体验。 ### 回答3: ESP32是一款非常流行的嵌入式设备,其配备的ESP-IDF系统是其开发的重要工具。ESP-IDF是一个开源的开发框架,可帮助开发者在ESP32中进行应用程序的高效开发。ESP-IDF使得开发人员可以使用C语言开发的单片机应用程序,从而可以控制硬件和访问数字传感器等设备。 在学习ESP-IDF教程之前,需要了解以下几个基本概念: 1. IIS:Integrated Development Environment (集成开发环境),用于编写、构建和调试ESP32应用程序。 2. SDK:Software Development Kit (软件开发工具包),包括必要的C库和头文件等,使开发人员可以编写ESP32应用程序。 3. IDF:IoT Development Framework (物联网开发框架),是ESP32的开发框架,包括了一系列API和头文件,可用于开发应用程序。 开始ESP-IDF教程的步骤如下: 1. 下载和安装基于ESP-IDF的IIS。可从ESP-IDF官网下载最新版本:https://docs.espressif.com/projects/esp-idf/en/latest/get-started/ 2. 创建ESP32项目。在IIS中创建一个ESP32项目,或使用ESP-IDF提供的模板项目。IIS可以自动生成项目文件。 3. 编译和烧写固件。编译ESP32应用程序并将其烧写到ESP32设备中。 4. 运行ESP32应用程序。使用IIS的调试器,在ESP32设备上启动ESP32应用程序。 ESP-IDF教程是ESP32开发的重要组成部分,可为开发人员提供快速上手和了解ESP32开发的指南。对于想要进入ESP32开发领域的初学者,建议先学会C语言基础,并熟悉ESP32开发板和其硬件接口。在进一步了解和掌握ESP-IDF教程之前,先掌握ESP32的基本知识,将有助于更好地理解ESP-IDF的学习。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值