IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> RT-Thread定时器 -> 正文阅读

[嵌入式]RT-Thread定时器

目录

定时器管理

?定时器超时函数

定时器管理接口

创建定时器

删除定时器

初始化定时器

脱离定时器

启动定时器

停止定时器

控制定时器

定时器执行上下文

定时器管理

定时器,是指从指定的时刻开始,经过一个指定的时间,然后触发一个事件,类似定个时间提醒第二天能够按时起床,定时器有软件定时器和硬件定时器之分;

  • 硬件定时器是芯片本身提供的定时功能。一般是由外部晶振提供给芯片输入时钟,芯片向软件模块提供一组配置寄存器,接受控制输入,到达设定时间值后芯片中断控制器产生时钟中断。硬件定时器的精度一般很高,可以达到纳秒级别,并且是中断触发方式。
  • 软件定时器是由操作系统提供的一类系统接口(函数),它在构建硬件定时器基础之上,使系统能够提供不受数目限制的定时器服务。

在操作系统中,通常软件定时器以系统节拍(tick)为单位。节拍长度指的是周期性硬件定时器两次中断间的间隔时间长度。这个周期性硬件定时器也称之为操作系统时钟。软件定时器以这个节拍时间长度为单位,数值必须是这个节拍的整数倍,例如节拍是10ms,那么上层软件定时器只能是10ms,20ms,100ms等,而不能取值为15ms。由于节拍定义了系统中定时器能够分辨的精度,系统可以根据实际系统CPU的处理能力和实时性需求设置合适的数值,tick值设置越小,精度越高,但是系统开销也将越大(在1S中系统用于处理时钟中断的次数也就越多)。RT-Thread的定时器也基于类似的系统节拍,提供了基于节拍整数倍的定时能力。

RT-Thread的定时器提供两类定时器机制,第一类是单次触发定时器,这类定时器在启动之后只会触发一次定时器,然后定时器自动停止,第二类是周期性触发定时器,这类定时器会周期性得分触发定时事件,直到用户手动的停止定时器,否则将永远的持续执行下去。

下面以一个实际代码来说明RT-Thread软件定时器的基本工作原理,在RT-Thread定时器模块中维护着这两个重要的全局变量:

  • 当前系统经过的tick时间rt_tick(当硬件定时器中断来临时,它将加1)
  • 定时器链表rt_time_list。系统新创建并激活的定时器都会按照以超时时间排序的方式插入到rt_time_list链表中

如下图示,系统当前的tick值为20,在当前系统中已经创建并启动了三个定时器,分别是定时时间为50个tick的Timer1、100个tick的Timer2和500个tick的Timer3,这三个定时器分别加上系统当前时间rt_tick=20,从小到大排序链接在rt_timer_list链表中,形成如下图示的定时器链表结构

而rt_tick随着硬件定时器的触发一直在增长(每次硬件定时器中断来临,rt_tick变量会加1) ,50个tick以后,rt_tick从20增长到70,与Timer1的timeout值相等,这时会触发与Timer1定时器相关的超时函数,同时将Timer1从rt_timer_list链表上删除,同理100个tick和500个tick过去后,与Timer2和Timer3相关联的超时函数会被触发,接着将Timer2和Timer3定时器相关联的超时函数会被触发,接着将Time2和Timer3定时器从rt_timer_list链表中删除

如果系统当前定时器状态在10个tick以后(rt_tick=30)有一个任务创建了一个tick值为300的Timer4定时器,由于Timer4定时器的timeout=rt_tick+300=330,因此它将被插入到Timer2和Timer3定时器的中间,形成如图所示的链表结构

?定时器超时函数

软定时器最主要的目的是在经过设定的时间后,系统能够自动执行用户设定的动作。当定时器设定的时间到了,即超时时,执行的动作函数称之为定时器的超市函数。与线程不同的是,超市函数的执行上下文环境并未显式给出。

在RT-Thread实时操作系统中,定时器超时函数存在着两种情况:

  • 超时函数在(系统时钟)中断上下文环境中执行
  • 超时函数在线程的上下文环境中执行

如果超时函数是在中断上下文环境中执行,显然对于超时函数的要求与中断服务例程的要求相同,执行时间尽量短,执行时不应导致当前上下文挂起,等待。例如在中断上下文中执行的超时函数不应该试图去申请动态内存、释放动态内存等(其中一个就包括rt_timer_delete函数调用)。

而在超时函数在线程上下文中执行,则不会有这个限制,但是通常也要求超时函数执行时间应该尽量短,而不影响到其他定时器或本次定时器的下一次周期性超时,这两种情况在RT-Thread定时器控制块中分别使用参数:RT_TIMER_FLAG_HARD_TIMER和RT_TIMER_FLAG_SOFT_TIMER指定,HARD_TIMER代表的是定时器超时函数执行上下文是在中断上下文中执行,SOFT_TIMER代表的是定时器函数执行的上下文是timer线程(在rtconfig.h头文件中定义宏RT_USING_TIMER_SOFT使得timer线程能被使用)

定时器管理控制块’

struct rt_timer
{
    struct rt_object parent;                            /**< inherit from rt_object */

    rt_list_t        row[RT_TIMER_SKIP_LIST_LEVEL];

    void (*timeout_func)(void *parameter);              /**< timeout function */
    void            *parameter;                         /**< timeout function's parameter */

    rt_tick_t        init_tick;                         /**< timer timeout tick */
    rt_tick_t        timeout_tick;                      /**< timeout tick */
};

定时器控制块由struct rt_timer结构体定义,并形成定时器内核对象再链接到内核对象容器中进行管理。list成员则用户把一个激活的(已经启动的)定时器链接到rt_timer_list链表中。

定时器管理接口

定时器管理系统初始化

初始化定时器管理系统,可以通过下面的接口函数实现:

void rt_system_timer_init(void)
{
    int i;

    for (i = 0; i < sizeof(rt_timer_list) / sizeof(rt_timer_list[0]); i++)
    {
        rt_list_init(rt_timer_list + i);
    }
}

上面的使用的是硬件定时器初始化,如果要使用SOFT_TIMER,则系统初始化时,应该调用下面的接口:

void rt_system_timer_thread_init(void)
{
#ifdef RT_USING_TIMER_SOFT
    int i;

    for (i = 0;
         i < sizeof(rt_soft_timer_list) / sizeof(rt_soft_timer_list[0]);
         i++)
    {
        rt_list_init(rt_soft_timer_list + i);
    }

    /* start software timer thread */
    rt_thread_init(&timer_thread,
                   "timer",
                   rt_thread_timer_entry,
                   RT_NULL,
                   &timer_thread_stack[0],
                   sizeof(timer_thread_stack),
                   RT_TIMER_THREAD_PRIO,
                   10);

    /* startup */
    rt_thread_startup(&timer_thread);
#endif
}

创建定时器

动态创建一个定时器:

rt_timer_t rt_timer_create(const char *name,
                           void (*timeout)(void *parameter),
                           void       *parameter,
                           rt_tick_t   time,
                           rt_uint8_t  flag)

参数

const char*name : 定时器的名称

void (timeout)(void parameter):定时器超时函数指针(定时器超时时,系统会调用这个函数)

void *parameter:定时器超时函数的入口参数(当定时器超时时,调用超时回调函数会把这个参数做为入口参数传递给超时函数

rt_tick_t time:定时器的超时函数,单位是系统节拍

rt_uint8_t flag:定时器创建时的参数,支持的值包括(可以用或关系取多个值)

rt_def.h中定义了一些定时器相关的宏:

#define RT_TIMER_FLAG_DEACTIVATED 0x0 /* 定时器为非激活态 */
#define RT_TIMER_FLAG_ACTIVATED 0x1 /* 定时器为激活状态 */
#define RT_TIMER_FLAG_ONE_SHOT 0x0 /* 单次定时 */
#define RT_TIMER_FLAG_PERIODIC 0x2 /* 周期定时 */
#define RT_TIMER_FLAG_HARD_TIMER 0x0 /* 硬件定时器 */
#define RT_TIMER_FLAG_SOFT_TIMER 0x4 /* 软件定时器 */

当指定的flag为RT_TIMER_FLAG_HARD_TIMER时,如果定时器超时,定时器的回调函数将在时钟中断的服务例程上下文调用,当指定的flag为RT_TIMER_FLAG_SOFT_TIMER,如果定时器超时,定时器的回调函数将在系统时钟timer线程的上下文中被调用。

函数返回:如果定时器创建成功,返回定时器句柄,否则返回失败RT_NULL。

删除定时器

系统不在使用特定定时器,可以使用下面的函数接口

rt_err_t rt_timer_delete(rt_timer_t timer)

?调用后,系统会把这个定时器从rt_timer_list链表中删除,然后释放相应的定时器控制块占有的内存。

初始化定时器

void rt_timer_init(rt_timer_t  timer,
                   const char *name,
                   void (*timeout)(void *parameter),
                   void       *parameter,
                   rt_tick_t   time,
                   rt_uint8_t  flag)

当选择静态创建定时器时,可利用rt_timer_init()来初始化该定时器,函数会初始化相应的定时器控制块和定时器名称,超时函数等。?

当指定的flag为RT_IMER_FLAG_HARD_TIMER时,如果定时器超时,定时器的回调函数
将在时钟中断的服务例程上下文中被调用;当指定的flag为RT_TIMER_FLAG_SOFT_TIMER
时,如果定时器超时,定时器的回调函数将在系统时钟timer线程的上下文中被调用。

脱离定时器

rt_err_t rt_timer_detach(rt_timer_t timer)

?脱离定时器,系统会把定时器对象从系统容器链表中删除,但是定时器对象所占有的内存不被释放。

启动定时器

rt_err_t rt_timer_start(rt_timer_t timer)

当定时器创建之后或者初始化之后,并不会被立即启动,必须在调用启动定时器函数接口后,才开始工作。调用定时器启动函数之后,定时器的状态将更改为激活状态(RT_TIMER_FLAG_ACTIVATED),并按照超时顺序插入到rt_timer_list链表中。

停止定时器

rt_err_t rt_timer_stop(rt_timer_t timer)
调用定时器停止函数接口后,定时器状态将更改为停止状态,并从rt_timer_list链表中 脱离出来不参与定时器超时检查。当一个(周期性)定时器超时时,也可以调用这个函数接 口停止这个(周期性)定时器本身。

控制定时器

/**
 * This function will get or set some options of the timer
 *
 * @param timer the timer to be get or set
 * @param cmd the control command
 * @param arg the argument
 *
 * @return RT_EOK
 */
rt_err_t rt_timer_control(rt_timer_t timer, int cmd, void *arg)
{

定时器执行上下文

RT-Thread里定时器默认的方式是HARD_TIMER定时器,即定时器超时后,超时函数是在系统时钟中断上下文中运行的,在中断上下文中执行的方式决定了定时器的超时函数不应调用任何会让当前上下文挂起的系统函数,也不能执行非常长的时间,否则会导致其他中断的响应时间加长或抢占其他线程执行的时间。

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2022-09-30 01:07:48  更:2022-09-30 01:09:54 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 20:54:57-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码