0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

什么是软件定时器?软件定时器的实现原理

冬至子 来源:嵌入式大杂烩 作者:嵌入式大杂烩 2023-05-23 17:05 次阅读

1.什么是软件定时器

软件定时器是用程序模拟出来的定时器,可以由一个硬件定时器模拟出成千上万个软件定时器,这样程序在需要使用较多定时器的时候就不会受限于硬件资源的不足,这是软件定时器的一个优点,即数量不受限制。

但由于软件定时器是通过程序实现的,其运行和维护都需要耗费一定的CPU资源,同时精度也相对硬件定时器要差一些。

2.软件定时器的实现原理

Linux,uC/OS,FreeRTOS操作系统中,都带有软件定时器,原理大同小异。典型的实现方法是:通过一个硬件定时器产生固定的时钟节拍,每次硬件定时器中断到,就对一个全局的时间标记加一,每个软件定时器都保存着到期时间。

程序需要定期扫描所有运行中的软件定时器,将各个到期时间与全局时钟标记做比较,以判断对应软件定时器是否到期,到期则执行相应的回调函数,并关闭该定时器。

以上是单次定时器的实现,若要实现周期定时器,即到期后接着重新定时,只需要在执行完回调函数后,获取当前时间标记的值,加上延时时间作为下一次到期时间,继续运行软件定时器即可。

3.基于STM32的软件定时器

3.1 时钟节拍

软件定时器需要一个硬件时钟源作为基准,这个时钟源有一个固定的节拍(可以理解为秒针的每次滴答),用一个32位的全局变量tickCnt来记录这个节拍的变化:

static volatile uint32_t tickCnt = 0;    //软件定时器时钟节拍

每来一个节拍就对tickCnt加一(记录滴答了多少下):

/* 需在定时器中断内执行 */void tickCnt_Update(void){    tickCnt++;}

一旦开始运行,tickCnt将不停地加一,而每个软件定时器都记录着一个到期时间,只要tickCnt大于该到期时间,就代表定时器到期了。

3.2 数据结构

软件定时器的数据结构决定了其执行的性能和功能,一般可分为两种:数组结构和链表结构。什么意思呢?这是(多个)软件定时器在内存中的存储方式,可以用数组来存,也可以用链表来存。

两者的优劣之分就是两种数据结构的特性之分:数组方式的定时器查找较快,但数量固定,无法动态变化,数组大了容易浪费内存,数组小了又可能不够用,适用于定时事件明确且固定的系统;链表方式的定时器数量可动态增减,易造成内存碎片(如果没有内存管理),查找的时间开销相对数组大,适用于通用性强的系统,Linux,uC/OS,FreeRTOS等操作系统用的都是链表式的软件定时器。

本文使用数组结构:

static softTimer timer[TIMER_NUM];        //软件定时器数组

数组和链表是软件定时器整体的数据结构,当具体到单个定时器时,就涉及软件定时器结构体的定义,软件定时器所具有的功能与其结构体定义密切相关,以下是本文中软件定时器的结构体定义:

typedef struct softTimer {    uint8_t state;           //状态    uint8_t mode;            //模式    uint32_t match;          //到期时间    uint32_t period;         //定时周期    callback *cb;            //回调函数指针    void *argv;              //参数指针    uint16_t argc;           //参数个数}softTimer;

定时器的状态共有三种,默认是停止,启动后为运行,到期后为超时。

typedef enum tmrState {    SOFT_TIMER_STOPPED = 0,  //停止    SOFT_TIMER_RUNNING,      //运行    SOFT_TIMER_TIMEOUT       //超时}tmrState;

模式有两种:到期后就停止的是单次模式,到期后重新定时的是周期模式。

typedef enum tmrMode {    MODE_ONE_SHOT = 0,       //单次模式    MODE_PERIODIC,           //周期模式}tmrMode;

不管哪种模式,定时器到期后,都将执行回调函数,以下是该函数的定义,参数指针argv为void指针类型,便于传入不同类型的参数。

typedef void callback(void *argv, uint16_t argc);

上述结构体中的模式state和回调函数指针cb是可选的功能,如果系统不需要周期执行的定时器,或者不需要到期后自动执行某个函数,可删除此二者定义。

3.3 定时器操作

3.3.1 初始化

首先是软件定时器的初始化,对每个定时器结构体的成员赋初值,虽说static变量的初值为0,但个人觉得还是有必要保持初始化变量的习惯,避免出现一些奇奇怪怪的BUG。

void softTimer_Init(void)
{
    uint16_t i;
    for(i=0; i

3.3.2 启动

启动一个软件定时器不仅要改变其状态为运行状态,同时还要告诉定时器什么时候到期(当前tickCnt值加上延时时间即为到期时间),单次定时还是周期定时,到期后执行哪个函数,函数的参数是什么,交代好这些就可以开跑了。

void softTimer_Start(uint16_t id, tmrMode mode, uint32_t delay, callback *cb, void *argv, uint16_t argc)
{
    assert_param(id < TIMER_NUM);
    assert_param(mode == MODE_ONE_SHOT || mode == MODE_PERIODIC);
    
    timer[id].match = tickCnt_Get() + delay;
    timer[id].period = delay;
    timer[id].state = SOFT_TIMER_RUNNING;
    timer[id].mode = mode;
    timer[id].cb = cb;
    timer[id].argv = argv;
    timer[id].argc = argc;
}

上面函数中的assert_param()用于参数检查,类似于库函数assert()。

3.3.3 更新

本文中软件定时器有三种状态:停止,运行和超时,不同的状态做不同的事情。停止状态最简单,啥事都不做;运行状态需要不停地检查有没有到期,到期就执行回调函数并进入超时状态;超时状态判断定时器的模式,如果是周期模式就更新到期时间,继续运行,如果是单次模式就停止定时器。这些操作都由一个更新函数来实现:

void softTimer_Update(void)
{
    uint16_t i;
    
    for(i=0; i

3.3.4 停止

如果定时器跑到一半,想把它停掉,就需要一个停止函数,操作很简单,改变目标定时器的状态为停止即可:

void softTimer_Stop(uint16_t id)
{
    assert_param(id < TIMER_NUM);
    timer[id].state = SOFT_TIMER_STOPPED;
}

3.3.5 读状态

又如果想知道一个定时器是在跑着呢还是已经停下来?也很简单,返回它的状态:

uint8_t softTimer_GetState(uint16_t id)
{
    return timer[id].state;
}

或许这看起来很怪,为什么要返回,而不是直接读?别忘了在前面3.2节中定义的定时器数组是个静态全局变量,该变量只能被当前源文件访问,当外部文件需要访问它的时候只能通过函数返回,这是一种简单的封装,保持程序的模块化。

3.4 测试

最后,当然是来验证一下我们的软件定时器有没达到预想的功能。

定义三个定时器:

定时器TMR_STRING_PRINT只执行一次,1s后在串口1打印一串字符;

定时器TMR_TWINKLING为周期定时器,周期为0.5s,每次到期都将取反LED0的状态,实现LED0的闪烁;

定时器TMR_DELAY_ON执行一次,3s后点亮LED1,跟第一个定时器不同的是,此定时器的回调函数是个空函数nop(),点亮LED1的操作通过主循环中判断定时器的状态来实现,这种方式在某些场合可能会用到。

static uint8_t data[] = {1,2,3,4,5,6,7,8,9,0};

int main(void)
{
    USART1_Init(115200);
    TIM4_Init(TIME_BASE_MS);
    TIM4_NVIC_Config();
    LED_Init();
    
    printf("I just grabbed a spoon.\\r\\n");
    
    softTimer_Start(TMR_STRING_PRINT, MODE_ONE_SHOT, 1000, stringPrint, data, 5);
    softTimer_Start(TMR_TWINKLING, MODE_PERIODIC, 500, LED0_Twinkling, NULL, 0);
    softTimer_Start(TMR_DELAY_ON, MODE_ONE_SHOT, 3000, nop, NULL, 0);
    
    while(1) {
        softTimer_Update();
        if(softTimer_GetState(TMR_DELAY_ON) == SOFT_TIMER_TIMEOUT) {
            LED1_On();
        }
    }
}
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • led灯
    +关注

    关注

    22

    文章

    1592

    浏览量

    107979
  • 嵌入式系统
    +关注

    关注

    41

    文章

    3590

    浏览量

    129461
  • 软件定时器
    +关注

    关注

    0

    文章

    18

    浏览量

    6744
  • FreeRTOS
    +关注

    关注

    12

    文章

    484

    浏览量

    62159
收藏 人收藏

    评论

    相关推荐

    基于STM32的软件定时器设计

    软件定时器是用程序模拟出来的定时器,可以由一个硬件定时器模拟出成千上万个软件定时器,这样程序在需
    发表于 07-03 17:06 1060次阅读
    基于STM32的<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>设计

    什么是软件定时器软件定时器实现原理是什么?

    什么是软件定时器软件定时器实现原理是什么?
    发表于 11-24 06:43

    定时器初值计算软件工具

    定时器初值计算软件工具
    发表于 03-20 10:23 239次下载

    555定时器威廉希尔官方网站 设计软件

    555定时器威廉希尔官方网站 设计软件
    发表于 03-02 11:29 287次下载

    555定时器应用设计软件免费下载

    本文档的主要内容详细介绍的是555定时器应用设计软件免费下载,本软件是一款555定时器设计软件,使用非常方便。
    发表于 12-17 08:00 51次下载
    555<b class='flag-5'>定时器</b>应用设计<b class='flag-5'>软件</b>免费下载

    ESP8266的管脚的控制和软件定时器的使用

    先说定时器,ESP8266内部的定时器分为软件定时器和硬件定时器。手册中指出硬件定时器其实就跟单
    的头像 发表于 07-29 14:57 9510次阅读
    ESP8266的管脚的控制和<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>的使用

    设计软件定时器

    软件定时器搬来使用2、自己设计软件定时器这里我只介绍第二种方法,我们知道,硬件定时器是通过对系统时钟周期进行计数
    发表于 11-05 18:35 2次下载
    设计<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>

    基于硬件定时器软件定时器

    概括硬件定时器很精确,软件定时器无论如何都有延迟,主要用在不需要精确定时的地方,而且软件定时比较
    发表于 11-25 09:51 8次下载
    基于硬件<b class='flag-5'>定时器</b>的<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>

    STM32开发项目:软件虚拟定时器实现

    目录背景软件定时器库特性源码介绍softwaretimer.csoftwaretimer.h使用指南背景在不运行实时操作系统的单片机中,面对一些复杂的项目需求时,仅有的几个硬件定时器显得有些捉襟见肘
    发表于 12-24 19:15 1次下载
    STM32开发项目:<b class='flag-5'>软件</b>虚拟<b class='flag-5'>定时器</b>的<b class='flag-5'>实现</b>

    软件定时器简介及程序配置

      软件定时器就是允许函数设置一定的等待时间,然后执行。定时器执行的函数被称为定时器的回调函数。定时器从启动到执行回调函数之间的时间称为
    的头像 发表于 12-06 16:10 3909次阅读
    <b class='flag-5'>软件</b><b class='flag-5'>定时器</b>简介及程序配置

    单片机软件定时器实现方法

    定时器个数一般都比较少,在一些有多个周期性操作的应用场合就无法满足要求。这时,就可以基于硬件定时器派生出软件定时器,来满足这种多种周期性或多个单次延时操作的需求。
    的头像 发表于 01-17 15:14 4891次阅读
    单片机<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>的<b class='flag-5'>实现</b>方法

    freeRTOS软件定时器的使用

    freeRTOS中加入了软件定时器这个功能组件,是一个可选的、不属于freeRTOS内核的功能,由定时器服务(其实就是一个定时器任务)来提供。
    的头像 发表于 02-10 13:55 2160次阅读

    Free RTOS的软件定时器

    软件定时器是FreeRTOS中的一个重要模块,使用软件定时器可以方便的实现一些与超时或周期性相关的功能。
    的头像 发表于 02-10 15:53 2342次阅读
    Free RTOS的<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>

    关于软件定时器的一些讨论

    这就是简单的软件定时器,是的,这就是特别简洁版本的软件定时器。当然它是有缺点的,比如systick_ms每1ms加1,所以软件
    的头像 发表于 10-13 16:14 566次阅读
    关于<b class='flag-5'>软件</b><b class='flag-5'>定时器</b>的一些讨论

    如何实现一个软件定时器

    在Linux,uC/OS,FreeRTOS等操作系统中,都带有软件定时器,原理大同小异。典型的实现方法是:通过一个硬件定时器产生固定的时钟节拍,每次硬件
    的头像 发表于 04-29 11:00 647次阅读