【MultiTimer】简易小巧的软件定时器扩展模块

2019-12-13 18:31发布

本帖最后由 半导体 于 2016-9-23 22:44 编辑

简介:
        前几天在坛里分享的MultiButton按键模块看到大家对此感兴趣,这让我备受鼓舞!今天分享的是一个简易的软件定时器模块,跟MultiButton一样使用单向链表串连管理同样的套路。
        这种类似的软件定时器模块其实在很多SOC的SDK上都已经有自带的API接口了,但很多工程师在裸机上开发还是习惯使用判断定时变量计数或标志位的方式去传递执行一些定时任务,
        虽然这种方式简单高效,但随着工程增大会出现满天飞的定时全局变量和标志位,加大了各代码段之间的耦合程度,并且很不优雅。
        MultiTimer 是一个软件定时器扩展模块,可无限扩展你所需的定时器任务,取代传统的标志位判断方式, 更优雅更便捷地管理程序的时间触发时序。

使用方法:
        1.先申请一个定时器管理handle:
                struct Timer timer1;

        2.初始化定时器对象,注册定时器回调处理函数,设置定时时间(ms),循环定时触发时间:
                timer_init(struct Timer* handle, void(*timeout_cb)(), uint32_t timeout, uint32_t repeat);

        3.启动定时器
                timer_start(&timer1);

        4.设置1ms的硬件定时器循环调用 timer_ticks() 以提供时间基准
                void HAL_SYSTICK_Callback(void)
                {
                        timer_ticks();
                }

        5.在主循环调用定时器后台处理函数
                int main()
                {
                            while(1) {
                                ...
                                timer_loop();
                            }
                }

例程:
  1. #include "multi_timer.h"

  2. struct Timer timer1;
  3. struct Timer timer2;

  4. void timer1_callback()
  5. {
  6.     printf("timer1 timeout! ");
  7. }

  8. void timer2_callback()
  9. {
  10.     printf("timer2 timeout! ");
  11. }

  12. int main()
  13. {
  14.         timer_init(&timer1, timer1_callback, 1000, 0); //1s delay
  15.         timer_start(&timer1);
  16.        
  17.         timer_init(&timer2, timer2_callback, 50, 50); //50ms loop
  18.         timer_start(&timer2);
  19.        
  20.         while(1) {
  21.             
  22.             timer_loop();
  23.         }
  24. }

  25. void HAL_SYSTICK_Callback(void)
  26. {
  27.     timer_ticks(); //1ms ticks
  28. }
复制代码

模块下载地址:
https://github.com/0x1abin/MultiTimer
(注:使用github的原因是我后面的一些改动能够及时更新,附件没法及时更新)
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
77条回答
heianshaonian
1楼-- · 2019-12-18 03:02
本帖最后由 heianshaonian 于 2016-9-28 16:53 编辑

楼主的代码已经移植过来了 ,time_loop我暂时放到中断里面了 因为while(1)有个oled要跑 放while(1) 巡不过来  只有把中断服务程序弄短一点了
techbaby
2楼-- · 2019-12-18 07:46
看了LZ的代码,有两处问题说一下!

1.
时钟计数值_timer_ticks是32位无符号数,在51类单片机下读取时不是原子操作,读取时可能中断发生,导致读取到的数值出错,此时运行timer_loop()函数肯定会出错!
建议加入_timer_ticks读取保护措施!

2.
timer_loop()函数中判断超时部分可能出错:
由程序中判断语句 if(_timer_ticks >= target->timeout)可知,超时与否仅仅根据_timer_ticks 值的大小比较,当_timer_ticks 发生溢出时可能出错。
你能保证单个定时器模块的回调函数运行时间在1个ticks内,但如果多个回调函数刚好同时运行在单次调用timer_loop()时,总运行时间可能超出1个ticks,
甚至达到2~3个ticks。比如我们经常延时5ms,10ms,20ms,50ms等,当ticks=100ms时,可能要运行4个回调函数。
如果此时某个target->timeout=0xFFFFFFFF,而在_timer_ticks = 0xFFFFFFFF时恰好因为延误未调用到此模块,当_timer_ticks = 0 反转后,此回调函数再想运行就要等到猴年马月了!
        

techbaby
3楼-- · 2019-12-18 13:15
 精彩回答 2  元偷偷看……
半导体
4楼-- · 2019-12-18 14:04
本帖最后由 半导体 于 2016-9-30 16:14 编辑
techbaby 发表于 2016-9-30 13:55
看了LZ的代码,有两处问题说一下!

1.


1. MultiTimer属于中间层模块,timer_ticks()只提供时基累加服务,是不考虑平台环境的,所以由平台自行处理:
viod hardware_timer_ISR()
{
        CPU_INT_DIS();
        timer_ticks();
        CPU_INT_EN();
}

2. 我明白你的意思,这个问题我在27楼解释过了,32bit的tick可以计数49.7天,根据你的需求而定可以用64bit计数(350亿年后溢出)。
newphj
5楼-- · 2019-12-18 19:07
很不错,又可以替换了。
techbaby
6楼-- · 2019-12-18 21:15
半导体 发表于 2016-9-30 16:10
1. MultiTimer属于中间层模块,timer_ticks()只提供时基累加服务,是不考虑平台环境的,所以由平台自行处 ...

一周热门 更多>