尽管FreeRTOS提供了软件计时器,但这些计时器有一些限制:

  • 最大分辨率等于RTOS滴答周期

  • 计时器回调从低优先级任务分派

硬件计时器不受这两个限制,但是通常它们使用起来不太方便。例如,应用组件可能需要定时器事件在将来的特定时间触发,但是硬件定时器仅包含一个用于中断产生的“比较”值。这意味着需要在硬件计时器之上构建一些功能来管理挂起事件列表,以便在发生相应的硬件中断时可以调度这些事件的回调。

esp_timer 一组API提供了一次性的计时器和定期的计时器,微秒级的时间分辨率以及64位范围。

基于ESP-IDF4.1

 1 #include <stdio.h>
2 #include <string.h>
3 #include <unistd.h>
4 #include "esp_timer.h"
5 #include "esp_log.h"
6 #include "esp_sleep.h"
7 #include "sdkconfig.h"
8
9 static void periodic_timer_callback(void* arg);
10 static void oneshot_timer_callback(void* arg);
11
12 static const char* TAG = "example";
13
14 void app_main(void)
15 {
16 /*
17 * 创建两个计时器:
18 * 1.每0.5秒运行的一个周期性计时器,并且打印消息
19 * 2.五秒后执行的一次性计时器,并且每1秒重启计时器
20 */
21
22 //创建周期性计时器
23 const esp_timer_create_args_t periodic_timer_args = {
24 .callback = &periodic_timer_callback,
25 //自定义名称
26 .name = "periodic"
27 };
28
29 esp_timer_handle_t periodic_timer;
30 ESP_ERROR_CHECK(esp_timer_create(&periodic_timer_args, &periodic_timer));
31
32 //创建一次性计时器
33 const esp_timer_create_args_t oneshot_timer_args = {
34 .callback = &oneshot_timer_callback,
35 //指定的参数将会传给计时器回调函数
36 .arg = (void*) periodic_timer,
37 .name = "one-shot"
38 };
39 esp_timer_handle_t oneshot_timer;
40 ESP_ERROR_CHECK(esp_timer_create(&oneshot_timer_args, &oneshot_timer));
41
42 //启动计时器
43 ESP_ERROR_CHECK(esp_timer_start_periodic(periodic_timer, 500000));
44 ESP_ERROR_CHECK(esp_timer_start_once(oneshot_timer, 5000000));
45 ESP_LOGI(TAG, "Started timers, time since boot: %lld us", esp_timer_get_time());
46
47 // 每2秒打印计时器调试信息到控制台
48 for (int i = 0; i < 5; ++i) {
49 ESP_ERROR_CHECK(esp_timer_dump(stdout));
50 usleep(2000000);
51 }
52
53
54 //轻度睡眠中,时间继续保持。轻度睡眠后计时器正确执行
55 ESP_LOGI(TAG, "Entering light sleep for 0.5s, time since boot: %lld us",esp_timer_get_time());
56 ESP_ERROR_CHECK(esp_sleep_enable_timer_wakeup(500000));
57 esp_light_sleep_start();
58 ESP_LOGI(TAG, "Woke up from light sleep, time since boot: %lld us", esp_timer_get_time());
59
60 //让计时器再运行一会
61 usleep(2000000);
62
63 //停止和清除计时器
64 ESP_ERROR_CHECK(esp_timer_stop(periodic_timer));
65 ESP_ERROR_CHECK(esp_timer_delete(periodic_timer));
66 ESP_ERROR_CHECK(esp_timer_delete(oneshot_timer));
67 ESP_LOGI(TAG, "Stopped and deleted timers");
68 }
69
70 //周期性计时器回调函数
71 static void periodic_timer_callback(void* arg)
72 {
73 int64_t time_since_boot = esp_timer_get_time();
74 ESP_LOGI(TAG, "Periodic timer called, time since boot: %lld us", time_since_boot);
75 }
76
77 //一次性计时器回调函数
78 static void oneshot_timer_callback(void* arg)
79 {
80 int64_t time_since_boot = esp_timer_get_time();
81 ESP_LOGI(TAG, "One-shot timer called, time since boot: %lld us", time_since_boot);
82 esp_timer_handle_t periodic_timer_handle = (esp_timer_handle_t) arg;
83 //启动运行的计时器,首先需要先停止它
84 ESP_ERROR_CHECK(esp_timer_stop(periodic_timer_handle));
85 ESP_ERROR_CHECK(esp_timer_start_periodic(periodic_timer_handle, 1000000));
86 time_since_boot = esp_timer_get_time();
87 ESP_LOGI(TAG, "Restarted periodic timer with 1s period, time since boot: %lld us",time_since_boot);
88 }

原文:https://gitee.com/EspressifSystems/esp-idf/tree/master/examples/system/esp_timer

ESP32高分辨率计时器笔记的更多相关文章

  1. 在Vue中使用计时器笔记

    在Vue中使用了计时器,一定要记得在生命周期destroyed()里清掉,不然第二次进入这个组件,会出现很大的问题 destroyed () { // (很重要)当跳转到其他页面的时候,要在生命周期的 ...

  2. ESP32存储blog笔记

    基于ESP-IDF4.1 1 #include <stdio.h> 2 #include "freertos/FreeRTOS.h" 3 #include " ...

  3. 引擎之旅 Chapter.1 高分辨率时钟

    目录 游戏中的时间线 真实时间线 游戏时间线 全局时钟的实现方式 我们如何理解时间.在现实生活中,时间就是一个有方向的直线.从一个无穷远到另一个无穷远.用数学去抽象地思考,它就是一个从无穷小到无穷大的 ...

  4. C#图解教程 第二十章 异步编程

    笔记 异步编程 什么是异步 示例 async/await特性的结构什么是异步方法 异步方法的控制流await表达式取消一个异步操作异常处理和await表达式在调用方法中同步地等待任务在异步方法中异步地 ...

  5. 游戏引擎架构 && windows 核心编程

    欲想正人,必先正己. 静坐当思己过,闲谈莫论人非. 垂直同步的作用: 为避免画面撕裂,许多渲染引擎会在交换缓冲区之前,等待显示器的垂直区间消隐,即电子枪重归屏幕上角的时间. 高分辨率计时器的时间漂移 ...

  6. MPI编程指南

    MPI编程指南 一.     MPI概述 1.1  MPI的发展史 MPI标准化涉及到大约60个国家的人们,他们主要来自于美国和欧洲的40个组织,这包括并行计算机的多数主要生产商,还有来自大学.政府实 ...

  7. 游戏引擎架构 (Jason Gregory 著)

    第一部分 基础 第1章 导论 (已看) 第2章 专业工具 (已看) 第3章 游戏软件工程基础 (已看) 第4章 游戏所需的三维数学 (已看) 第二部分 低阶引擎系统 第5章 游戏支持系统 (已看) 第 ...

  8. Netem参数说明

    Netem参数说明 本文主要内容来自Linux基金会Wiki网站Netem文档,点击这里访问原文 netem通过模拟广域网的特性为测试协议提供网络仿真功能.当前版本模拟可变延迟,丢失,重复和重新排序. ...

  9. 【大前端攻城狮之路·二】Javascript&QA⼯程师

    今天给大家分享的主题的是Javascript&QA⼯程师.看到这个主题,可能有人问:前端开发完就OK了,剩下的丢给测试就行,哪里还需要关心这些?但事实上呢,测试是前端开发非常重要的环节,也是迈 ...

随机推荐

  1. 如何在 Docker 环境下自动给 .NET 程序生成 Dump

    前言 之前"一线码农"大佬有写文章介绍了如何在 windows 下自动 dump,正好手里有个在 docker 环境下 dump 的需求,所以在参考大佬文章的基础上,有了本篇. ​ ...

  2. SpringBoot基础学习(三) 自定义配置、随机数设置及参数间引用

    自定义配置 SpringBoot免除了项目中大部分手动配置,可以说,几乎所有的配置都可以写在全局配置文件application.peroperties中,SpringBoot会自动加载全局配置文件从而 ...

  3. javascript数组排序之冒泡排序

    冒泡排序 作为一名程序员数组的排序算法是必须要掌握的,今天来说最简单的一种数组排序----冒泡排序 冒泡排序原理 冒泡排序算法是一种简单直观的排序算法.它重复地走访过要排序的数列,一次比较两个元素,如 ...

  4. SOLO: 按位置分割对象

    SOLO: 按位置分割对象 SOLO: Segmenting Objectsby Locations 论文链接: https://arxiv.org/pdf/1912.04488.pdf 代码链接: ...

  5. ieda引入jstl后报错解决办法

    报错如下: HTTP Status 500 - The absolute uri: http://java.sun.com/jsp/jstl/core cannot be resolved in ei ...

  6. Java接口以及匿名内部类,静态代码块

    接口 接口中只能定义public并且是final的公共静态常量,不允许定义变量. 抽象类可以定义抽象方法和非抽象方法,接口中只能定义公共的,抽象的实例方法. 接口只能由其他接口实现继承 子接口继承的目 ...

  7. 【NX二次开发】Block UI 双精度表

    属性说明 常规         类型 描述     BlockID     String 控件ID     Enable     Logical 是否可操作     Group     Logical ...

  8. Linux CentOS 配置Yaf框架

    简介 Yaf框架想必大家都有所了解,它是一个开源的高性能的PHP框架 官网地址:https://www.php.net/manual/zh/book.yaf.php Yaf开发文档:https://w ...

  9. 如何设置Python环境变量

    大家好,我是Yivies.相信很多python的初学者们在进行一顿下一步下一步的安装之后,在windows command命令行输入python的时候会出现这样的情况: 可我们希望它是这样子的: 其实 ...

  10. 迁移Report Server DataBase时遇到的坑

    1.项目背景 由于历史原因,公司部分系统的Report是基于SQL Server Report Service搭建的,且Reporting Services 和Report Server DataBa ...