【快速上手ESP32(基于ESP-IDFVSCode)】03-定时器

ESP32中的通用定时器

通用定时器是 ESP32 定时器组外设的驱动程序。ESP32 硬件定时器分辨率高,具有灵活的报警功能。定时器内部计数器达到特定目标数值的行为被称为定时器报警。定时器报警时将调用用户注册的不同定时器回调函数。

在ESP32-S3中,一共有两个定时器组,每个定时器组中各有两个通用定时器以及一个看门狗定时器。

每个通用定时器都有16位预分频器和54位可自动重新加载向上/向下计数器。

通用定时器通常在以下场景中使用:

  • 如同挂钟一般自由运行,随时随地获取高分辨率时间戳;

  • 生成周期性警报,定期触发事件;

  • 生成一次性警报,在目标时间内响应。

使用通用定时器

#include "driver/gptimer.h"

我们跟着编程指南来,第一步需要实例化一个定时器句柄。

可以配置的有五个点。

第一个是时钟源,我顺着编程指南的链接往上找,结果找到的结果就两个,甚至我觉得这两个是一个意思。

但是立创开发板提供的文档中有三个选项。

但是其实也不用纠结使用什么,我们就选默认的就行。因为ESP32的定时器与STM32的定时器在设置上不一样,STM32中我们需要通过时钟源频率去计算该如何配置自动重装载计数器和预分频器,但是在ESP32中我们可以不用计算,直接指定溢出频率,也就是上面的第三个。

第三个结构体成员设置内部计数器的分辨率,其实也就是溢出频率。

第二个结构体成员指定计数方向,一般我们就选向上计数。

第四个指定优先级,第五个设置是否将中断源共享,这俩我们在中断比较少的情况下可以不设置。

优先级不必说,数值越小越大(0是分配一个默认的优先级,并不是优先级最高)。

第五个是否中断源共享其实我们在STM32中就使用过了,不过STM32是默认就是共享的,因为不同的中断可能会进入同一个中断函数,这就是中断源共享。

配置完之后需要使用 gptimer_new_timer 实例化定时器并获取句柄,传入的参数就是刚刚结构体变量和定时器句柄的地址,因此除了上面的配置,我们还需要定义一个定时器句柄,可以参考下面的代码。

    // 定义一个通用定时器
    gptimer_handle_t gptimer = NULL;

    // 配置定时器参数
    gptimer_config_t timer_config = {
        .clk_src = GPTIMER_CLK_SRC_DEFAULT, // 定时器时钟来源 选择APB作为默认选项
        .direction = GPTIMER_COUNT_UP,      // 向上计数
        .resolution_hz = 1e6,
    };
    // 将配置设置到定时器
    gptimer_new_timer(&timer_config, &gptimer);

也可以参考编程指南的示例代码。

当我们不再需要定时器的时候,需要先禁用定时器,然后删除,具体流程在下面。

我们也可以通过下面两个函数去获取和修改计数器的值。

做完上面的准备工作之后,按照STM32中的流程,我们应该要写中断函数了吧。

在ESP32中,我们换了个叫法,我们叫警报。所以流程就是配置定时器,然后配置警报,接着绑定警报的回调函数,这个回调函数可以理解成就是STM32中的中断函数。

一共需要配置的结构体成员有三个。

第一个设置触发警报事件的目标计数值,也就是当计数器的值到达这个目标值的时候触发警报。

第二个设置重装载的值,一般咱就选择重装成0,然后计数模式是向上计数。

第三个选择是否自动重载,也就是说我们是否要这个定时器警报周期性的给我们警报。

给结构体的成员配置完之后,还需要使用gptimer_set_alarm_action这个函数去激活,第一个参数是定时器的句柄,第二个参数是上面配置好的结构体变量的地址。

可以参考下面的代码。

    // 通用定时器的报警值设置
    gptimer_alarm_config_t alarm_config = {
        .reload_count = 0,                  // 重载计数值为0
        .alarm_count = 5e5,         // 报警目标计数值 500000 = 500ms
        .flags.auto_reload_on_alarm = true, // 开启重加载
    };
    // 设置触发报警动作
    gptimer_set_alarm_action(gptimer, &alarm_config);

接下来就该绑定警报事件的回调函数了。

这个回调函数虽然不像STM32那样定死了,但是也是给我们限制住了,必须确保这个函数不会试图阻塞,甚至是使用FreeRTOS的API。

没错,ESP-IDF给我们自带了FreeRTOS的API,我们导入头文件之后就可以直接使用FreeRTOS了,这一点还是非常方便的。

不过我在编程指南中没找到提供的可用的回调函数,估计是我英文看不懂,翻译成中文有些偏差,搞得我迟迟没有找到。

不过没关系,我们直接抄别人现成的写好的来分析。

下面是立创开发板提供的文档中的代码。最后面我会贴出出完整版的修改过的代码,下面把代码切开来是为了方便分析。

首先先是包含头文件。

#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"

 可以看得出包含的头文件里有freertos的字样,也就是说我们通过ESP-IDF创建的工程确实可以直接使用FreeRTOS。

下面这一段是绑定回调函数,第三行中的TimerCallBback是回调函数的名字。

后面创建了一个队列,是用在最后设置定时器的回调函数时的第三个参数,这个参数是用户数据,会直接传递给回调函数。

    // 绑定一个回调函数
    gptimer_event_callbacks_t cbs = {
        .on_alarm = TimerCallback,
    };
    // 创建一个队列
    QueueHandle_t queue = xQueueCreate(10, sizeof(10));
    // 设置定时器gptimer的 回调函数为cbs  传入的参数为NULL
    gptimer_register_event_callbacks(gptimer, &cbs, queue);

下面是回调函数,函数名,返回值,形参的类型是固定的,我们可以在下面我写注释的地方写上我们要执行的逻辑,因为这个回调函数里不能有阻塞的风险,因此我们最好就是只写简单的逻辑甚至是不写,那么我们应该写在哪里呢?

static bool IRAM_ATTR TimerCallback(gptimer_handle_t timer, const gptimer_alarm_event_data_t *edata, void *user_data)
{
    BaseType_t high_task_awoken = pdFALSE;
    // 将传进来的队列保存
    QueueHandle_t queue = (QueueHandle_t)user_data;

    /*
        这里可以写要执行的逻辑,但是不建议写.
    */
    static int time = 0;
    // 从中断服务程序(ISR)中发送数据到队列
    xQueueSendFromISR(queue, &time, &high_task_awoken);

    return (high_task_awoken == pdTRUE);
}

写在主循环里面。

    int number = 0;
    while(1){
        //从队列中接收一个数据,不能在中断服务函数使用
        if (xQueueReceive(queue, &number, pdMS_TO_TICKS(2000))) {
            //触发了警报之后这里会执行
        } else {
            //出现错误
        }
    }

剩下就是使能定时器以及启动了。

    // 使能定时器
    gptimer_enable(gptimer);
    // 开始定时器开始工作
    gptimer_start(gptimer);

那么接下来就完整地展示一下通过定时器来让LED闪烁的代码。

#include <stdio.h>
#include "driver/gptimer.h"
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"

#include "driver/gpio.h"

void initGPIO(){
    gpio_config_t init;
    init.intr_type=GPIO_INTR_DISABLE;             //失能中断;
    init.mode=GPIO_MODE_OUTPUT|GPIO_MODE_INPUT;   
    init.pin_bit_mask=(1ULL<<18);                 //GPIO18
    init.pull_down_en=GPIO_PULLDOWN_DISABLE;      //失能下拉模式
    init.pull_up_en=GPIO_PULLUP_ENABLE;           //使能上拉模式

    gpio_config(&init);
}

/**
 * @函数说明        定时器回调函数
 * @传入参数
 * @函数返回
 */
static bool IRAM_ATTR TimerCallback(gptimer_handle_t timer, const gptimer_alarm_event_data_t *edata, void *user_data)
{
    BaseType_t high_task_awoken = pdFALSE;
    // 将传进来的队列保存
    QueueHandle_t queue = (QueueHandle_t)user_data;

    static int time = 0;
    // 从中断服务程序(ISR)中发送数据到队列
    xQueueSendFromISR(queue, &time, &high_task_awoken);

    return (high_task_awoken == pdTRUE);
}

void app_main(void){
    initGPIO();
    // 定义一个通用定时器
    gptimer_handle_t gptimer = NULL;
    // 配置定时器参数
    gptimer_config_t timer_config = {
        .clk_src = GPTIMER_CLK_SRC_DEFAULT, // 定时器时钟来源 选择APB作为默认选项
        .direction = GPTIMER_COUNT_UP,      // 向上计数
        .resolution_hz = 1e6,
    };
    // 将配置设置到定时器
    gptimer_new_timer(&timer_config, &gptimer);


    // 通用定时器的报警值设置
    gptimer_alarm_config_t alarm_config = {
        .reload_count = 0,                  // 重载计数值为0
        .alarm_count = 5e5,         // 报警目标计数值 500000 = 500ms
        .flags.auto_reload_on_alarm = true, // 开启重加载
    };
    // 设置触发报警动作
    gptimer_set_alarm_action(gptimer, &alarm_config);
    

    // 绑定一个回调函数
    gptimer_event_callbacks_t cbs = {
        .on_alarm = TimerCallback,
    };
    // 创建一个队列
    QueueHandle_t queue = xQueueCreate(10, sizeof(10));
    // 设置定时器gptimer的 回调函数为cbs  传入的参数为NULL
    gptimer_register_event_callbacks(gptimer, &cbs, queue);


    // 使能定时器
    gptimer_enable(gptimer);
    // 开始定时器开始工作
    gptimer_start(gptimer);
    
    int number = 0;
    
    while(1){
        //从队列中接收一个数据,不能在中断服务函数使用
        if (xQueueReceive(queue, &number, pdMS_TO_TICKS(2000))) {
           gpio_set_level(18,!gpio_get_level(18));
        } else {
            printf("error\r\n");
        }
    }
}

ESP32中的软件定时器

上面介绍的通用定时器实际上是硬件定时器,所以相对的,我们能用的还有软件定时器。

软件定时器就是系统模拟出来的定时器,而通用定时器是实实在在有的硬件定时器,因此硬件定时器的精度会更高,而软件定时器使用起来可以有很多个,并且代码编写方面也比较简单,但是相对的,精度会略微下降。

在编程指南里面,我们可以在系统API里找到软件定时器的用法,而我们之前的通用定时器的用法在外设API里,这也侧面说明了两种用法的区别。

使用软件定时器

使用软件定时器首先需要包含头文件。

#include "esp_timer.h"

接着我们需要构造一个esp_timer示例。

我们分别看看需要哪两个参数。

第一个esp_timer_create_args_t

由它来设置我们的回调函数,我们只需要设置前两个结构体成员即可,剩下的可以不配置。

第二个是软件定时器的句柄,我们直接创建一个,然后取地址放进去即可。

可以参考下面的例子。

    esp_timer_handle_t  timer1=0;
    
    esp_timer_create_args_t timer1_arg = {
            .callback = &timer1Callback,
            .arg = NULL
    };

	esp_timer_create(&timer1_arg , &timer1);

下一步有了配置好了的定时器的句柄之后我们就可以直接启动了。

 一共有两种启动方式,一种是周期性定时,另一种是一次性定时。

传入的第一个参数是定时器句柄,第二个是启动的时间,单位是微秒。

剩下就是上面三个函数:停止定时器,重新启动定时器,删除定时器,看看上面的解释就可以知道三者之间的联系与区别了。

然后就结束了。软件定时器就是这么简单。

接下来大家结合着我下面的示例代码就可以理解了。

#include "driver/gpio.h"
#include <unistd.h>
#include "esp_timer.h"

void initGPIO(){
    gpio_config_t init = {
        .intr_type = GPIO_INTR_DISABLE,             // 失能中断;
        .mode = GPIO_MODE_OUTPUT | GPIO_MODE_INPUT, // 输出模式&输入模式(为了读取电平来翻转)
        .pin_bit_mask = (1ULL << 18),               // GPIO18
        .pull_down_en = GPIO_PULLDOWN_DISABLE,      // 失能下拉模式
        .pull_up_en = GPIO_PULLUP_ENABLE,           // 使能上拉模式
    };
    gpio_config(&init);
}

esp_timer_handle_t timer1 = 0;
esp_timer_handle_t timer2 = 0;

void timer1Callback(void *arg){
    esp_timer_stop(timer2);   // 删除前需要停止
    esp_timer_delete(timer2); // 删除定时器
}

void timer2Callback(void *arg){
    gpio_set_level(18, !gpio_get_level(18)); // 翻转GPIO口电平
}

void initTimer(void){
    esp_timer_create_args_t timer1_arg = {
        .callback = &timer1Callback,
        .arg = NULL
    };

    esp_timer_create_args_t timer2_arg = {
        .callback = &timer2Callback,
        .arg = NULL
    };

    esp_timer_create(&timer1_arg, &timer1);
    esp_timer_start_once(timer1, 5 * 1000 * 1000);  //5s后执行一次

    esp_timer_create(&timer2_arg, &timer2);
    esp_timer_start_periodic(timer2, 1000 * 1000);  //1s执行一次,周期执行
}
void app_main(void){
    initGPIO();
    initTimer();
    while (1){
        usleep(1000*10);        //加个10ms延时,以免运行时产生报错
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/517995.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

HTML:框架

案例&#xff1a; <frameset cols"5%,*" ><frame src"left_frame.html"><frame src"right_frame.html"> </frameset> 一、<frameset>标签 <frameset>标签&#xff1a;称为框架标记&#xff0c;将一个HTML…

全网最强JavaWeb笔记 | 万字长文爆肝JavaWeb开发——day06_数据库-MySQL-02

万字长文爆肝黑马程序员2023最新版JavaWeb教程。这套教程打破常规&#xff0c;不再局限于过时的老套JavaWeb技术&#xff0c;而是与时俱进&#xff0c;运用的都是企业中流行的前沿技术。笔者认真跟着这个教程&#xff0c;再一次认真学习一遍JavaWeb教程&#xff0c;温故而知新&…

vue-cli打包 nodejs内存溢出 vue2.x Last few GCs

遇到这种情况百度各种博客&#xff0c;什么改package.json里的配置&#xff0c;什么安装increase-memory-limit &#xff0c;都尝试了并没什么用处&#xff0c;最后解决方案为执行下方名单&#xff0c;再次打包就成功了&#xff1a; export NODE_OPTIONS--max_old_space_size4…

spring事务那些事

实际工作中还会面临千奇百怪的问题&#xff0c;看下面返个例子&#xff08;注意MySql数据库测试&#xff09;&#xff1a; //1.hello1Service 调用 hello2Service Transactional(propagation Propagation.REQUIRED,rollbackFor Exception.class) public void doUpdate() {//…

重建大师地物实体shp该怎样获取?(如下图)

问题如图 一般是基于自己的模型去提取的&#xff0c;可以使用地物外轮廓功能生成&#xff0c;或者这边有DLG也可以实现。同时&#xff0c;用重建大师可以提取地物外轮廓。 重建大师是一款专为超大规模实景三维数据生产而设计的集群并行处理软件&#xff0c;输入倾斜照片&#x…

微信小程序 ---- 慕尚花坊 代码优化

代码优化 1. 分享功能 思路分析&#xff1a; 目前小程序页面都没有配置分享功能&#xff0c;需要给小程序页面设置分享功能。 但是并不是所有页面都需要设置分享功能&#xff0c; 具体哪些页面需要设置分享功能&#xff0c;可以和产品经理进行协商。 首页商品列表商品详情…

[StartingPoint][Tier1]Crocodile

Task 1 What Nmap scanning switch employs the use of default scripts during a scan? (哪些 Nmap 扫描开关在扫描期间使用默认脚本&#xff1f;) -sC Task 2 What service version is found to be running on port 21? 发现端口 21 上运行的服务版本是什么&#xff1f…

前端 基于响应式数据 实现拖拽排序和移动

在外层父元素添加拖拽相关监听事件 <divdragstart"handleDragstart"dragover"handleDragover"dragenter"handleDragenter"drag"handleDrag"drop"handleDrop">其中&#xff0c;start是drag起始元素&#xff0c;over会…

【计算机网络】epoll

IO多路转接 - epoll 一、I/O多路转接之 epoll1. epoll 接口&#xff08;1&#xff09;epoll_create()&#xff08;2&#xff09;epoll_wait()&#xff08;3&#xff09;epoll_ctl() 2. epoll 原理3. epoll 的优点4. epoll 的使用5. epoll 的工作模式&#xff08;1&#xff09;水…

Python网络爬虫(四):b站评论

首先来看一下采集的数据格式: 本文不对数据采集的过程做探讨,直接上代码。首先要在程序入口处bvids列表内替换成自己想要采集的视频bvid号,然后将self.cookies替换成自己的(需要字典格式),代码可以同时爬取多个视频的评论,且爬取的评论较为完整,亲测有效: im…

Calico IPIP和BGP TOR的数据包走向

IPIP Mesh全网互联 文字描述 APOD eth0 10.7.75.132 -----> APOD 网关 -----> A宿主机 cali76174826315网卡 -----> Atunl0 10.7.75.128 封装 ----> Aeth0 10.120.181.20 -----> 通过网关 10.120.181.254 -----> 下一跳 BNODE eth0 10.120.179.8 解封装 --…

带头双向循环链表,顺序表和链表的比较

双向链表 单链表结点中只有一个指向其后继的指针&#xff0c;使得单链表只能从前往后依次遍历&#xff0c;要访问某个结点的前驱&#xff08;插入、删除操作时&#xff09;&#xff0c;只能从头开始遍历&#xff0c;访问前驱的时间复杂度为O(N)。为了克服这个缺点&#xff0c;…

SSM框架学习——Eclipse创建Spring MVC maven项目

Spring MVC项目创建 什么是Spring MVC Spring MVC是Spring内置的&#xff0c;实现了Web MVC设计模式的框架。 它解决了Web开发过程中很多的问题&#xff0c;例如参数接收、表单验证等。另外它采用松散耦合可插拔组件等结构&#xff0c;具有相对较高的灵活性和扩展性。 Spri…

Coursera上托福专项课程02:TOEFL Speaking and Writing Sections Skills Mastery 学习笔记

TOEFL Speaking and Writing Sections Skills Mastery Course Certificate 本文是学习 https://www.coursera.org/learn/toefl-speaking-writing-sections-skills-mastery 这门课的学习笔记&#xff0c;如有侵权&#xff0c;请联系删除。 文章目录 TOEFL Speaking and Writing…

《崩溃》社会如何选择成败兴亡 - 三余书屋 3ysw.net

崩溃&#xff1a;社会如何选择成败兴亡 《崩溃&#xff1a;社会如何选择成败兴亡》深入对人类大历史的思考&#xff0c;解答人类社会成败兴亡的秘密。这本书主要聚焦在人类社会兴盛与环境之间的纠葛。我们将一同探讨历史上哪些伟大文明因为环境破坏而崩溃&#xff0c;还有哪些…

v-for之对象和对象信息

如下图所示&#xff1a; 看打印&#xff1a;尤其是这个对象信息的打印 也可以在打印对象信息的时候取出索引信息&#xff1a;

kafka 高吞吐设计分析

说明 本文基于 kafka 2.7 编写。author blog.jellyfishmix.com / JellyfishMIX - githubLICENSE GPL-2.0 概括 支撑 kafka 高吞吐的设计主要有以下几个方面: 网络 nio 主从 reactor 设计模式 顺序写。 零拷贝。 producer producer 开启压缩后是批量压缩&#xff0c;bro…

一分钟快速用上号称“音乐版ChatGPT”的suno AI,适合普通人的超简单教程!

随着AI的应用变广&#xff0c;各类AI程序已逐渐普及。AI已逐渐深入到人们的工作生活方方面面。而AI涉及的行业也越来越多&#xff0c;从最初的写作&#xff0c;到医疗教育&#xff0c;再到现在的音乐。 Suno是一个专业高质量的AI歌曲和音乐创作平台&#xff0c;用户只需输入简…

JAVA IO流学习

File类&#xff1a; File类是java.io包中很重要的一个类 File类的对象可以代表一个文件或者目录&#xff0c;可以修改文件大小、文件最后修改日期、文件名等 File对象不能操作文件的具体数据&#xff0c;即不能对文件进行读和写的操作 File的构造方法&#xff1a; File&…

Redis中的Sentinel(一)

Sentinel 概述 Sentinel(哨岗、哨兵)是Redis的高可用性(high availability)解决方案:由一个或多个Sentinel实例(instance)组成的Sentinel系统(system)可以监视任意多个主服务器&#xff0c;以及这些主服务器属下的所有从服务器,并在被监视的主服务器进入下线状态时&#xff0…