当前位置:首页 > > 知晓编程
[导读]CPU内部自带的定时器模块,通过初始化、配置可以实现定时,定时时间到以后就会执行相应的定时器中断处理函数。硬件定时器一般都带有其它功能,比如PWM输出、输入捕获等等功能。但是缺点是硬件定时器数量少!!


简介

这里先介绍下软件定时器和硬件定时器的区别硬件定时器

CPU内部自带的定时器模块,通过初始化、配置可以实现定时,定时时间到以后就会执行相应的定时器中断处理函数。硬件定时器一般都带有其它功能,比如PWM输出、输入捕获等等功能。但是缺点是硬件定时器数量少!!

软件定时器

软件定时器允许设置一段时间,当设置的时间到达之后就执行指定的功能函数,被定时器调用的这个功能函数叫做定时器的回调函数。回调函数的两次执行间隔叫做定时器的定时周期,简而言之,当定时器的定时周期到了以后就会执行回调函数。在FreeRTOS中有专门的软件定时器功能,我们可以在MCU中简单的是实现“软件定时器”如下:
void timer_1000ms(void){ printf("timer_1000ms\r\n");}/*systick_ms在硬件定时器中每1ms加1*/int main(void){ static timer_tick = 0; timer_tick = systick_ms; while() { if((systick_ms-timer_tick)>1000) { timer_tick = systick_ms; timer_1000ms(); } }}
这就是简单的软件定时器,是的,这就是特别简洁版本的软件定时器。当然它是有缺点的,比如systick_ms每1ms加1,所以软件定时器的精度是ms为单位的,并且如果while(1)中有其他代码阻塞,软件定时器也会跟着阻塞的。这个简单的软件定时器毕竟很"简陋",大家可以自行封装丰富一下,或者参考已经有的开源方案:MultiTimer,一款可无限扩展的软件定时器。MultiTimer 是一个软件定时器扩展模块,可无限扩展你所需的定时器任务,取代传统的标志位判断方式, 更优雅更便捷地管理程序的时间触发时序。开源地址:https://github.com/0x1abin/MultiTimer
MultiTimer

MultiTimer的设计比较简洁,只有2个文件,并且只有4个函数,总共82行代码,稍微花一点功夫就可以理解明白。

移植步骤

  • 配置系统时间基准接口,安装定时器驱动

uint64_t PlatformTicksGetFunc(void){ /* Platform implementation */}MultiTimerInstall(PlatformTicksGetFunc);
  • 实例化一个定时器对象

MultiTimer timer1;
  • 设置定时时间,超时回调处理函数, 用户上下指针,启动定时器

int MultiTimerStart(&timer1, uint64_t timing, MultiTimerCallback_t callback, void* userData);
  • 在主循环调用定时器后台处理函数

int main(int argc, char *argv[]){ ... while (1) { ... MultiTimerYield(); }}

具体就不做手把手教程如何移植了,在STM32F207移植好的工程开源地址:

开源地址:https://github.com/strongercjd/STM32F207VCT6/tree/master/23-Timer-MultiTimer

下面分析一下MultiTimer在移植的第一步,配置系统时间基准接口,安装定时器驱动
uint64_t PlatformTicksGetFunc(void){ /* Platform implementation */}MultiTimerInstall(PlatformTicksGetFunc);
看一下MultiTimerInstall函数原型
typedef uint64_t (*PlatformTicksFunction_t)(void);static PlatformTicksFunction_t platformTicksFunction = NULL;int MultiTimerInstall(PlatformTicksFunction_t ticksFunc){ platformTicksFunction = ticksFunc; return 0;}
这其实就是函数指针实现的回调函数,具体详解看之前的文章《回调函数》,其实就是给MultiTimer提供一个计数器。
除去回调函数,该开源项目还是单链表的很好的示例,学习数据结构是比较乏味的,这个开源项目是单链表很好的应用落地,不太懂的同学可以学习下。下面摘取一下部分代码链表的删除
for (; *nextTimer; nextTimer = &(*nextTimer)->next) { if (timer == *nextTimer) { *nextTimer = timer->next; /* remove from list */ break; }}

插入链表

for (nextTimer = &timerList;; nextTimer = &(*nextTimer)->next) { if (!*nextTimer) { timer->next = NULL; *nextTimer = timer; break; } if (timer->deadline < (*nextTimer)->deadline) { timer->next = *nextTimer; *nextTimer = timer; break; }}

遍历链表

MultiTimer* entry = timerList;for (; entry; entry = entry->next) { /* Sorted list, just process with the front part. */ if (platformTicksFunction() < entry->deadline) { return (int)(entry->deadline - platformTicksFunction()); } /* remove expired timer from list */ timerList = entry->next;  /* call callback */ if (entry->callback) { entry->callback(entry, entry->userData); }}
这篇文章不会详细讲解链表的操作,不懂的同学可以看之前文章《链表在STM32中的应用》。

当然MultiTimer也是有缺点的,比如一个定时器是1000ms,另一个定时器是500ms,调度时就会冲突,也没有定时器调度抢占,会随着其他代码的阻塞而阻塞。这种类似的问题不再详述,大家使用的时候多测测就好。


任务调度

看了上面的操作,如果我们不叫软件定时器,叫它“任务”,是不是和FreeRTOS任务类似,感觉高端一些,甚至这篇文章标题可以修改为《一篇文章教你实现操作系统》,开个欢笑,不做标题党。

有些项目实时性要求高,需要任务抢占,所以需要使用FreeRTOS这样的操作系统,但它资源占用比例过大,不利于项目开发,在一般的小项目中也用不到RTOS的太多功能,使用上面的思路,你可以把每个任务设置不同的间隔时间周期性调用,如果有实时性要求很高的事件,就通过中断处理。

当然也可以使用开头的粗糙方法

if((systick_ms-timer_tick)>1000){ timer_tick = systick_ms; timer_1000ms();}

这样功能是可以实现的,但没有模块化,不利于代码的维护。我们可以借鉴MultiTimer思路封装一下软件接口。

并且,如果你的项目中,任务的个数是固定不变的,可以将MultiTimer中的链表拿掉,直接使用全局变量就可以,如果有额外的时间模仿FreeRTOS实现一些信号量,对列等,这就是自己的OS(无抢占)啊。(当然这属于重复造轮子,但对一些公司来讲,有适合自己业务的,最精简的代码框架是很有必要的)。

我简单粗糙的实现了一个,有兴趣的可以看一下


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭