基于DWC_ether_qos的以太网驱动开发-LWIP的定时器模块详解

原创 嵌入式Lee 2023-09-15 17:50

一. 前言

LWIP的定时器模块,实现了通用的软件定时器,用于内部的周期事件处理,比如arptcp的超时等,用户也可以使用。这一篇来分析该模块的实现。

.代码分析

2.1源码

源码位于

timeouts.c

timeouts.h

会按照如下条件编译

#if LWIP_TIMERS && !LWIP_TIMERS_CUSTOM

LWIP_TIMERS1 LWIP_TIMERS_CUSTOM0才会编译,也是默认配置。

2.2数据结构

定时器的核心数据结构是一个单向链表,链表的节点如下

struct sys_timeo {
struct sys_timeo *next;
u32_t time;
sys_timeout_handler h;
void *arg;
#if LWIP_DEBUG_TIMERNAMES
const char* handler_name;
#endif /* LWIP_DEBUG_TIMERNAMES */
};

Next构成单向链表

time为绝对时间,即当前绝对时间滞后该值则表示定时器超时需要执行回调函数h

arg可以传入参数,

handler_namedebug打印信息用。

2.3超时比较算法

定时器使用的是绝对时间,即定时器的time和当前now时间比较,time<=now则表示定时器已经超时了需要处理,否则定时器还未到时间无需处理。

但是这里会有个问题,溢出的问题,time<=now就一定表示time的时刻提前于now吗,不一定,也可能是到了定时器值的最大值绕回了,

比如如果定时器的值是32位的,

now0xFFFFFFFFtime0x00000001

time 那么可能是time提前于now 时间0xFFFFFFFE (0xFFFFFFFF-0x00000001)

也可能time滞后于now2的时间 即(0x100000000-0xFFFFFFFF)+0x00000001.

我们更倾向于是后者,因为后者的时间差更小,更符合实际情况,因为我们定时时间一般都很小。

这里的实现是

#define LWIP_MAX_TIMEOUT 0x7fffffff

#define TIME_LESS_THAN(t, compare_to) ( (((u32_t)((t)-(compare_to))) > LWIP_MAX_TIMEOUT) ? 1 : 0 )

实际上用到的就是我们上面提到的思想,我们更倾向于时间差更小的为实际情况,

实际该算法还有专门的文章进行讨论,网上可以搜到。

即定义定时器最大范围的一半,比如32位最大范围是0~0xFFFFFFFF,共0x100000000的范围,其一半的范围是0x80000000,即0~0x7fffffff,作为基准,最大就只能定时器该时间,大于该时间认为不合理,实际是绕回反向的。

这里((u32_t)((t)-(compare_to)))按照无符号32位进行计算

如果t

如果t1compare_to2((u32_t)((t)-(compare_to)))结果为0xFFFFFFFF

实际上就是0x100000000-0x02 + 0x01.

 

如果t>compare_to

t2compare_to1

((u32_t)((t)-(compare_to)))结果为0x1

 

该表达式即可以理解为t滞后compare_to的时间(未来时间)

更形象的理解是a-bb需要追赶多少到a,即compare_to追赶到t需要多久,有可能绕回。

如果该滞后时间比较大,大于总时间的一半即0x7fffffff我们则认为,实际不是滞后而是超前。

因为倾向于时间间隔短的符合实际情况。

所以如果tcompare_to大的非常多,大于TIME_LESS_THAN,我们也认为t不是滞后compare_to而是提前于compare_to,只是是绕回了。

1.总结

我们可以用跑圈追赶的角度来理解,即tcompare_to在环形世界赛跑,某一刻我们并不知道谁跑在前,谁跑在后,因为有可能套圈, 于是我们有一条假设, tcompare_to跑的速度差异不是特别大(类似我们定时器的定时时间不是特别长),所以如果t追赶到compare_to的距离大于跑道的一半我们认为不合理,所以认为是compare_to追赶t

所以该算法能工作的前提条件是定时时间不能大于LWIP_MAX_TIMEOUT

当然去处理查询定时器超时的间隔也不能大于LWIP_MAX_TIMEOUT,否则由于套多圈无法区分。

2.4内建定时器

定义了一个数组

const struct lwip_cyclic_timer lwip_cyclic_timers[]

提供构建定时器的必要信息(注意不是定时器本身,而是提供定时器信息,sys_timeouts_init根据此创建定时器)

数组成员结构体如下

struct lwip_cyclic_timer {
u32_t interval_ms;
lwip_cyclic_timer_handler handler;
#if LWIP_DEBUG_TIMERNAMES
const char* handler_name;
#endif /* LWIP_DEBUG_TIMERNAMES */
};

interval_ms为定时器执行周期,上述定时器要求是绝对时间,为什么这里是间隔时间呢,因为now+间隔时间就是绝对时间了,初始化时会自动设置。

handler是回调函数

handler_name是用于打印定时器的名字。

默认根据宏定义了一些内建的定时器

比如,使能了LWIP_ARP则使能该定时器,回调函数是etharp_tmr,间隔时间是1S

用户可以配置这些宏来进行定时器的使能配置和周期配置。

#if LWIP_ARP

{ARP_TMR_INTERVAL, HANDLER(etharp_tmr)},

#endif /* LWIP_ARP */

#define ARP_TMR_INTERVAL 1000

初始化sys_timeouts_init时遍历lwip_cyclic_timers

通过sys_timeout->sys_timeout_abs动态创建定时器,定时器的绝对时间自动会在now基础上增加间隔(u32_t)(sys_now() + msecs);

这里i = (LWIP_TCP ? 1 : 0),如果有LWIP_TCP则从1开始, 0TCP定时器单独处理,因为它不需要总是运行,没有tcp连接就不需要该定时器了,所以手动调用tcp_timer_needed()处理。

2.5接口代码

sys_timeouts_sleeptime

后面定时器轮询有分析,计算定时器链表中,头定时器,离当前时间的时间,

返回0表示头定时器已经超时需要处理,返回SYS_TIMEOUTS_SLEEPTIME_INFINITE表示没有定时器,其他值为头定时器离现在的时间间隔。因为定时器是按照时间从小到大排列,所以只需要判断头定时器即可。

sys_restart_timeouts

以定时器链表第一个定时器为基准设置为now绝对时间,后续的按照和第一个定时器的偏差设置。

在长时间没有调用sys_check_timeouts时,重新设置时基,来触发一次时间调度。

这样保证在长时间没有调用sys_check_timeouts的期间导致的定时器没有执行,这时能弥补下执行一次。

sys_check_timeouts

查询定时器,从链表头开始查询,如果超时时间到则执行对应的回调函数,并释放定时器。

因为已经排序不需要查询到末尾,查询到第一个为超时的定时器即可结束,因为后面的值更大肯定不会超时。

OS时用户手动调用该函数

OS时,tcpip线程自动调用。

注意定时器都是单次的,一次执行完后会删除,周期执行需要重新创建。

这里个人觉得每次都删除和释放不是很好,尤其是嵌入式平台,多了mem等操作一方面内存碎片的问题(如果使用内存池实现还好,如果共用堆管理则会有些影响,尤其堆本来就很小的资源受限平台),一方面效率降低。

sys_untimeout

从定时器链表删除一个定时器

sys_timeout->sys_timeout_abs

创建定时器,按照定时器值从小大到插入到链表

sys_timeouts_init

初始化内建定时器,前面已经分析过

lwip_cyclic_timer

内建定时器回调处理

由于定时器都是单次的,所以周期定时器需要重新创建定时器。

内建定时器时都是设置的该回调函数

sys_timeout(lwip_cyclic_timers[i].interval_ms, lwip_cyclic_timer, LWIP_CONST_CAST(void *, &lwip_cyclic_timers[i]));

通过参数再回调具体的不同的回调函数

const struct lwip_cyclic_timer *cyclic = (const struct lwip_cyclic_timer *)arg;
cyclic->handler();

tcp_timer_needed/tcpip_tcp_timer

tcp定时器单独处理,创建一个tcp定时器

tcpip_tcp_timer会根据是否有tcp连接来确认是否需要重复定时器。

2.6定时器轮询

OS时手动周期调用

sys_check_timeouts

OS时在tcpip_thread线程中

TCPIP_MBOX_FETCHtcpip_timeouts_mbox_fetch会自动调用

sys_check_timeouts

我们来分析下tcpip_timeouts_mbox_fetch

首先sleeptime = sys_timeouts_sleeptime(); 获取最近一个将要超时的定时器到现在的时间间隔,这样mbox_fetch时就以该间隔时间作为超时时间sleeptime,这样如果在这个超时时间之前获取到了mbox则处理消息,下一个循环继续重复上述处理。否则等到超时再调用sys_check_timeouts();处理定时器。

sys_timeouts_sleeptime先要判断是否有定时器,如果next_timeout为空则说明没有定时器需要处理则超时时间sleeptime可以设置为无限大。

如果有定时器则只需要判断next_timeout头定时器得timenow比较即可,因为定时器是按照time从小到大排列的,所以最先超时得肯定是头定时器。如果next_timeouttime小于now,说明该定时器已经超时了设置为0,后面会马上调用sys_check_timeouts()处理。

否则计算next_timeouttime减去now为间隔时间。

也就是对应

if (sleeptime == SYS_TIMEOUTS_SLEEPTIME_INFINITE) {
UNLOCK_TCPIP_CORE();
sys_arch_mbox_fetch(mbox, msg, 0);
LOCK_TCPIP_CORE();
return;
} else if (sleeptime == 0) {
sys_check_timeouts();
/* We try again to fetch a message from the mbox. */
goto again;
}

如果没有定时器sleeptime == SYS_TIMEOUTS_SLEEPTIME_INFINITEsys_arch_mbox_fetch(mbox, msg, 0); 参数0表示无限超时时间。

直到获取到消息才会return,否则就一直在此等待。

这里个人觉得有个BUG,如果刚开始没有定时器,且此时没有消息,则在此之后新创建的定时器将得不到处理,因为一直在这里等待消息了,虽然一开始基本都会有定时器所以不会进到这里,但是逻辑上来说还是不严谨。虽然这里无限等待可以有利于效率,因为没有消息该线程就不执行了,但是个人觉得设置一个固定的超时间隔可能更安全,这样保证该线程不会卡死在这里,超过时间没有消息也跳过重新执行,这样保证新创建的定时器能执行,最大误差就是该设置的固定间隔。这个间隔可以根据允许误差和效率均衡考虑设置,这样也不至于影响效率,也能保证定时器始终能执行。

sleeptime已经有定时器超时了sleeptime == 0则马上调用sys_check_timeouts()处理。因为没有消息所以goto again;重复,无需return

如果sleeptime不是0也不是无限大,则按需设置超时时间

res = sys_arch_mbox_fetch(mbox, msg, sleeptime);

如果res返回超时则调用sys_check_timeouts处理定时器,goto again;重复上述过程,因为没有消息所以无需return。有消息则return到上一层去处理消息。

2.7DEBUG

lwipopts.h中定义LWIP_DEBUG_TIMERNAMES宏使能相关debug代码,

否则根据LWIP_DEBUG决定

如果定义了LWIP_DEBUGLWIP_DEBUG_TIMERNAMESSYS_DEBUG,否则为0

SYS_DEBUG默认为LWIP_DBG_OFF,可以该为LWIP_DBG_ON

#ifndef LWIP_DEBUG_TIMERNAMES
#ifdef LWIP_DEBUG
#define LWIP_DEBUG_TIMERNAMES SYS_DEBUG
#else /* LWIP_DEBUG */
#define LWIP_DEBUG_TIMERNAMES 0
#endif /* LWIP_DEBUG*/
#endif

以上使能相关调试代码之后,还需要lwipopts.h中使能

TIMERS_DEBUG

按如下配置使能

#define TIMERS_DEBUG LWIP_DBG_ON
#define LWIP_DEBUG_TIMERNAMES 1

当然也要使能DEBUG

#define LWIP_DEBUG 1

LWIP_PLATFORM_DIAG打印的接口宏。

此时可以看到打印信息如下,可以通过打印确定定时是否正确,定时器是否工作

sct calling h=ip_reass_tmr t=0 arg=0x2001548c
tcpip: ip_reass_tmr()
sys_timeout: 0x28213e48 abs_time=6223 handler=ip_reass_tmr arg=0x2001548c
sct calling h=etharp_tmr t=0 arg=0x20015498
tcpip: etharp_tmr()
sys_timeout: 0x28213e68 abs_time=6224 handler=etharp_tmr arg=0x20015498
sct calling h=ip_reass_tmr t=0 arg=0x2001548c
tcpip: ip_reass_tmr()
sys_timeout: 0x28213e48 abs_time=7223 handler=ip_reass_tmr arg=0x2001548c
sct calling h=etharp_tmr t=0 arg=0x20015498
tcpip: etharp_tmr()
sys_timeout: 0x28213e68 abs_time=7224 handler=etharp_tmr arg=0x20015498
sct calling h=ip_reass_tmr t=0 arg=0x2001548c
tcpip: ip_reass_tmr()
sys_timeout: 0x28213e48 abs_time=8223 handler=ip_reass_tmr arg=0x2001548c
sct calling h=ip_reass_tmr t=0 arg=0x2001548c
tcpip: ip_reass_tmr()
sys_timeout: 0x28213e48 abs_time=16223 handler=ip_reass_tmr arg=0x2001548c
sct calling h=etharp_tmr t=0 arg=0x20015498
tcpip: etharp_tmr()
sys_timeout: 0x28213e68 abs_time=16224 handler=etharp_tmr arg=0x20015498

.总结

重点理解定时器的超时判断算法,

注意定时器是单次的每次超时处理完都会删除,需要重新创建,这个需要注意,并且注意频繁的创建和删除对堆管理的影响。

了解内建定时器的定时周期的配置,以及定时器的调试方法。


评论
  • 戴上XR眼镜去“追龙”是种什么体验?2024年11月30日,由上海自然博物馆(上海科技馆分馆)与三湘印象联合出品、三湘印象旗下观印象艺术发展有限公司(下简称“观印象”)承制的《又见恐龙》XR嘉年华在上海自然博物馆重磅开幕。该体验项目将于12月1日正式对公众开放,持续至2025年3月30日。双向奔赴,恐龙IP撞上元宇宙不久前,上海市经济和信息化委员会等部门联合印发了《上海市超高清视听产业发展行动方案》,特别提到“支持博物馆、主题乐园等场所推动超高清视听技术应用,丰富线下文旅消费体验”。作为上海自然
    电子与消费 2024-11-30 22:03 68浏览
  • 国产光耦合器因其在电子系统中的重要作用而受到认可,可提供可靠的电气隔离并保护敏感电路免受高压干扰。然而,随着行业向5G和高频数据传输等高速应用迈进,对其性能和寿命的担忧已成为焦点。本文深入探讨了国产光耦合器在高频环境中面临的挑战,并探索了克服这些限制的创新方法。高频性能:一个持续关注的问题信号传输中的挑战国产光耦合器传统上利用LED和光电晶体管进行信号隔离。虽然这些组件对于标准应用有效,但在高频下面临挑战。随着工作频率的增加,信号延迟和数据保真度降低很常见,限制了它们在电信和高速计算等领域的有效
    腾恩科技-彭工 2024-11-29 16:11 106浏览
  • 最近几年,新能源汽车愈发受到消费者的青睐,其销量也是一路走高。据中汽协公布的数据显示,2024年10月,新能源汽车产销分别完成146.3万辆和143万辆,同比分别增长48%和49.6%。而结合各家新能源车企所公布的销量数据来看,比亚迪再度夺得了销冠宝座,其10月新能源汽车销量达到了502657辆,同比增长66.53%。众所周知,比亚迪是新能源汽车领域的重要参与者,其一举一动向来为外界所关注。日前,比亚迪汽车旗下品牌方程豹汽车推出了新车方程豹豹8,该款车型一上市就迅速吸引了消费者的目光,成为SUV
    刘旷 2024-12-02 09:32 58浏览
  • 《高速PCB设计经验规则应用实践》+PCB绘制学习与验证读书首先看目录,我感兴趣的是这一节;作者在书中列举了一条经典规则,然后进行详细分析,通过公式推导图表列举说明了传统的这一规则是受到电容加工特点影响的,在使用了MLCC陶瓷电容后这一条规则已经不再实用了。图书还列举了高速PCB设计需要的专业工具和仿真软件,当然由于篇幅所限,只是介绍了一点点设计步骤;我最感兴趣的部分还是元件布局的经验规则,在这里列举如下:在这里,演示一下,我根据书本知识进行电机驱动的布局:这也算知行合一吧。对于布局书中有一句:
    wuyu2009 2024-11-30 20:30 84浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2024-12-02 10:40 53浏览
  • 学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&
    youyeye 2024-11-30 14:30 62浏览
  • 国产光耦合器正以其创新性和多样性引领行业发展。凭借强大的研发能力,国内制造商推出了适应汽车、电信等领域独特需求的专业化光耦合器,为各行业的技术进步提供了重要支持。本文将重点探讨国产光耦合器的技术创新与产品多样性,以及它们在推动产业升级中的重要作用。国产光耦合器创新的作用满足现代需求的创新模式新设计正在满足不断变化的市场需求。例如,高速光耦合器满足了电信和数据处理系统中快速信号传输的需求。同时,栅极驱动光耦合器支持电动汽车(EV)和工业电机驱动器等大功率应用中的精确高效控制。先进材料和设计将碳化硅
    克里雅半导体科技 2024-11-29 16:18 157浏览
  • 艾迈斯欧司朗全新“样片申请”小程序,逾160种LED、传感器、多芯片组合等产品样片一触即达。轻松3步完成申请,境内免费包邮到家!本期热荐性能显著提升的OSLON® Optimal,GF CSSRML.24ams OSRAM 基于最新芯片技术推出全新LED产品OSLON® Optimal系列,实现了显著的性能升级。该系列提供五种不同颜色的光源选项,包括Hyper Red(660 nm,PDN)、Red(640 nm)、Deep Blue(450 nm,PDN)、Far Red(730 nm)及Ho
    艾迈斯欧司朗 2024-11-29 16:55 152浏览
  • By Toradex胡珊逢简介嵌入式领域的部分应用对安全、可靠、实时性有切实的需求,在诸多实现该需求的方案中,QNX 是经行业验证的选择。在 QNX SDP 8.0 上 BlackBerry 推出了 QNX Everywhere 项目,个人用户可以出于非商业目的免费使用 QNX 操作系统。得益于 Toradex 和 QNX 的良好合作伙伴关系,用户能够在 Apalis iMX8QM 和 Verdin iMX8MP 模块上轻松测试和评估 QNX 8 系统。下面将基于 Apalis iMX8QM 介
    hai.qin_651820742 2024-11-29 15:29 150浏览
  • 光耦合器作为关键技术组件,在确保安全性、可靠性和效率方面发挥着不可或缺的作用。无论是混合动力和电动汽车(HEV),还是军事和航空航天系统,它们都以卓越的性能支持高要求的应用环境,成为现代复杂系统中的隐形功臣。在迈向更环保技术和先进系统的过程中,光耦合器的重要性愈加凸显。1.混合动力和电动汽车中的光耦合器电池管理:保护动力源在电动汽车中,电池管理系统(BMS)是最佳充电、放电和性能监控背后的大脑。光耦合器在这里充当守门人,将高压电池组与敏感的低压电路隔离开来。这不仅可以防止潜在的损坏,还可以提高乘
    腾恩科技-彭工 2024-11-29 16:12 117浏览
  • 在电子技术快速发展的今天,KLV15002光耦固态继电器以高性能和强可靠性完美解决行业需求。该光继电器旨在提供无与伦比的电气隔离和无缝切换,是现代系统的终极选择。无论是在电信、工业自动化还是测试环境中,KLV15002光耦合器固态继电器都完美融合了效率和耐用性,可满足当今苛刻的应用需求。为什么选择KLV15002光耦合器固态继电器?不妥协的电压隔离从本质上讲,KLV15002优先考虑安全性。输入到输出隔离达到3750Vrms(后缀为V的型号为5000Vrms),确保即使在高压情况下,敏感的低功耗
    克里雅半导体科技 2024-11-29 16:15 119浏览
  • RDDI-DAP错误通常与调试接口相关,特别是在使用CMSIS-DAP协议进行嵌入式系统开发时。以下是一些可能的原因和解决方法: 1. 硬件连接问题:     检查调试器(如ST-Link)与目标板之间的连接是否牢固。     确保所有必要的引脚都已正确连接,没有松动或短路。 2. 电源问题:     确保目标板和调试器都有足够的电源供应。     检查电源电压是否符合目标板的规格要求。 3. 固件问题: &n
    丙丁先生 2024-12-01 17:37 57浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦