ESP32学习笔记(42)——硬件定时器接口使用

一、简介

ESP32 芯片包含两个硬件定时器组。每组有两个通用硬件定时器。它们都是基于 16 位预分频器和 64 位递增/递减计数器的 64 位通用定时器,能够自动重新加载。

ESP-IDF 编程指南——通用定时器

二、API说明

以下硬件定时器接口位于 driver/include/driver/timer.h

2.1 timer_init

2.2 timer_set_counter_value

2.3 timer_set_alarm_value

2.4 timer_enable_intr

2.5 timer_isr_register

2.6 timer_start

2.7 timer_get_counter_value

2.8 timer_spinlock_take

2.9 timer_spinlock_give

2.10 timer_group_get_intr_status_in_isr

2.11 timer_group_get_counter_value_in_isr

2.12 timer_group_clr_intr_status_in_isr

2.13 timer_group_set_alarm_value_in_isr

2.14 timer_group_enable_alarm_in_isr

三、编程流程

3.1 定时器初始化

首先,应该通过调用函数timer_init()并传递一个结构体timer_config_t来定义定时器应该如何操作来初始化定时器。特别是,可以设置以下定时器参数:

timer_config_t config = {
        .divider = TIMER_DIVIDER,       //分频倍数
        .counter_dir = TIMER_COUNT_UP,  //向上计数(1)/向下计数(0)
        .counter_en = TIMER_PAUSE,      //定时器运行开(1)关(0)
        .alarm_en = TIMER_ALARM_EN,     //定时器中断开(1)关(0)
        .auto_reload = TIMER_AUTORELOAD_EN,     //是(1)否(0)自动重装载
    }; // default clock source is APB

要获取计时器设置的当前值,请使用函数timer_get_config()

3.2 定时器控制

一旦定时器被启用,它的计数器就会开始运行。要启用计时器,请timer_init()使用counter_enset调用该函数true,或调用timer_start()

您可以通过调用指定计时器的初始计数器值timer_set_counter_value()
要检查计时器的当前值,请调用timer_get_counter_value()timer_get_counter_time_sec()

要随时暂停计时器,请调用timer_pause()
要恢复它,请调用timer_start()
要重新配置计时器,您可以调用timer_init()

3.3 定时器中断

3.3.1 中断设置

要设置中断报警,请调用该函数timer_set_alarm_value(),然后使用启用报警timer_set_alarm()。报警也可以在定时器初始化阶段启用,当timer_init()被调用时。

开启报警后,定时器达到报警值后,根据配置的不同,会发生以下两种动作:

  • 如果先前已配置,将触发中断。有关如何配置中断,请参阅中断部分。
  • auto_reload启用时,定时器计数器将自动重新加载开始从以前配置的值重新计数。这个值应该提前设置timer_set_counter_value()
  • 如果设置了报警值并且计时器已经达到该值,则会立即触发报警。
  • 一旦触发,报警将自动禁用,需要重新启用才能再次触发。

3.3.2 中断回调

ESP32 的硬件定时器中断有两种处理方式:

  • 回调配置法【推荐】:
    ESP32 硬件定时器在默认情况下有一个 ISR 程序,称作 “总 ISR 程序” 。它为我们执行我们配置的定时器的回调。换句话说,我们写的回调函数,配置后将属于这个ISR程序的一部分。因此这个回调函数又称为ISR 回调(ISR callback)
    这样做要求你的回调程序要尽可能的简短。

    创建ISR Callback函数

    static bool timer_callback(void *args){
        uint64_t val;
        BaseType_t pxHigherPriorityTaskWoken = pdFALSE;
      
        // xQueueSendFromISR(queue, (void*)&val, &pxHigherPriorityTaskWoken);
    
        return pxHigherPriorityTaskWoken;//请看第 3 行
    }
    

    添加、删除定时器Callback(注册中断)
    调用函数timer_isr_callback_add()添加回调
    调用函数timer_isr_callback_remove()删除回调

  • 自定义ISR法:
    如果使用此函数重新注册 ISR,则需要编写完整的 ISR。例如需要清除中断标志位,配置自旋锁之类。

    一般情况下必要的事:

    1. 获取自旋锁timer_spinlock_take()
    2. 显式清除中断状态timer_group_clr_intr_status_in_isr()
    3. 再次使能中断timer_group_enable_alarm_in_isr()
    4. 释放自旋锁timer_spinlock_take()

    创建ISR中断服务程序

    void IRAM_ATTR timerIsr(void *arg){
      timer_spinlock_take(0);    // 获取自旋锁
    
      /*
          代码区一
      */
      timer_group_clr_intr_status_in_isr(0, 0);
      timer_group_enable_alarm_in_isr(0, 0);
      /*
          代码区二
      */
      
      timer_spinlock_give(0);  // 释放自旋锁
    }
    

    注意:必须显式清除中断标志,以及重新使能中断。如上代码区一和代码区二之间的两个函数。(两个函数参数都是定时器组索引和定时器索引)

    注册中断
    调用函数timer_isr_register()配置
    例如,为定时器组0中的定时器0配置一个完整的ISR程序

    timer_isr_register(0, 0, timerIsr, &config, ESP_INTR_FLAG_IRAM, NULL); 
    

四、示例代码

定义一个硬件定时器,时长为3秒,每一次中断向一个任务传递定时器计数值。

根据 examples\peripherals\timer_group 中的例程修改

4.1 配置回调法

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

#define TIMER_DIVIDER         16  //  硬件定时器分频器
#define TIMER_FREQ           (TIMER_BASE_CLK / TIMER_DIVIDER)  // 定时器计数频率

static xQueueHandle queue;

static bool timer_callback(void *args){
    uint64_t val;
    BaseType_t pxHigherPriorityTaskWoken = pdFALSE;
    
    val = timer_group_get_counter_value_in_isr(0, 0);
    /*
     * 上行代码:
     * ————————————————————
     * 将定时器的值传给一个任务
     *(由于本示例使用的自动重装载模式,
     * 所以在本示例中这个val值无意义。
     * 只是为了展示在isr callback中获
     * 取定时器值函数的使用【必须
     * 调用带有_in_isr的函数】)
     * ————————————————————
     */
    
    //通过队列将 val 传给任务
    xQueueSendFromISR(queue, (void*)&val, &pxHigherPriorityTaskWoken);

    return pxHigherPriorityTaskWoken;
}


static void task(void *arg){
    uint64_t counts = 0ull;
    while(1){
        counts ++;
        uint64_t val;
        if(xQueueReceive(queue, &val, portMAX_DELAY)){
            printf("val %lld\n", val);
            printf("定时器第 %llu 次中断\n", counts);
        }
    }
}
void app_main(void)
{
    queue = xQueueCreate(10, sizeof(uint64_t));
    timer_config_t config = {
        .divider = TIMER_DIVIDER,
        .counter_dir = TIMER_COUNT_UP,
        .counter_en = TIMER_PAUSE,
        .alarm_en = TIMER_ALARM_EN,
        .auto_reload = TIMER_AUTORELOAD_EN,
    };
    timer_init(0, 0, &config);
    timer_set_counter_value(0, 0, 0x00ull);
    timer_set_alarm_value(0, 0, TIMER_FREQ * 3);
    timer_enable_intr(0, 0);
    
    timer_isr_callback_add(0, 0, timer_callback, NULL, ESP_INTR_FLAG_IRAM);

    xTaskCreate(task, "timer_test_task", 2048, NULL, 5, NULL);
    timer_start(0, 0);
    printf("定时器启动成功!");
}

4.2 自定义ISR法

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

#define TIMER_DIVIDER         16  //  Hardware timer clock divider
#define TIMER_FREQ           (TIMER_BASE_CLK / TIMER_DIVIDER)  // convert counter value to seconds

static xQueueHandle queue;
//回调函数
static void IRAM_ATTR timerIsr(void *arg){
    timer_spinlock_take(0);    

    uint64_t val = 0;
    val = timer_group_get_counter_value_in_isr(0, 0);
    /*
     * 上行代码:
     * ————————————————————
     * 将定时器的值传给一个任务
     *(由于本示例使用的自动重装载模式,
     * 所以在本示例中这个val值无意义。
     * 只是为了展示在isr callback中获
     * 取定时器值函数的使用【必须
     * 调用带有_in_isr的函数】)
     * ————————————————————
     */
    
    timer_group_clr_intr_status_in_isr(0, 0);
    timer_group_enable_alarm_in_isr(0, 0);
    
    xQueueSendFromISR(queue, &val, NULL);
    
    timer_spinlock_give(0);
}

static void task(void *arg){
    uint64_t counts = 0ull;
    while(1){
        counts ++;
        uint64_t val;
        if(xQueueReceive(queue, &val, portMAX_DELAY)){
            printf("val %lld\n", val);
            printf("定时器第 %llu 次中断\n", counts);
        }
    }
}
void app_main(void)
{
    queue = xQueueCreate(10, sizeof(uint64_t));
    timer_config_t config = {
        .divider = TIMER_DIVIDER,
        .counter_dir = TIMER_COUNT_UP,
        .counter_en = TIMER_PAUSE,
        .alarm_en = TIMER_ALARM_EN,
        .auto_reload = TIMER_AUTORELOAD_EN,
    };
    timer_init(0, 0, &config);
    timer_set_counter_value(0, 0, 0x00ull);
    timer_set_alarm_value(0, 0, TIMER_FREQ * 3);
    timer_enable_intr(0, 0);
    timer_isr_register(0, 0, timerIsr, &config, ESP_INTR_FLAG_IRAM, NULL);
    xTaskCreate(task, "timer_test_task", 2048, NULL, 5, NULL);
    timer_start(0, 0);
    printf("定时器启动成功!");
}

• 由 Leung 写于 2021 年 8 月 9 日

• 参考:ESP32 之 ESP-IDF 教学(三)——通用硬件定时器(Timer)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容