当前位置: 首页 > web >正文

LWIP的超时事件笔记

那个马蜂佬,刚发就给我两个赞

lwIP超时事件处理简介

为每个与外界网络连接的任务都设定了timeout属性,即等待超时时间,例如TCP建立连接超时、ARP缓存表项的时间管理等,都需要超时操作来处理

lwIP超时事件机制 一共有四种

2.1,超时事件如何管理

2.2,超时事件如何注册     

2.3,超时事件如何删除   

2.4,超时事件如何查询   

LWIP超时事件结构体

typedef void (* lwip_cyclic_timer_handler)(void); 
struct lwip_cyclic_timer 
{ u32_t interval_ms; 					/* 超时时间 */ lwip_cyclic_timer_handler handler; 	/* 超时处理函数 */ 
};

LWIP超时管理结构体

typedef void (* sys_timeout_handler)(void *arg); 
struct sys_timeo 
{ 
struct sys_timeo *next; 	/* 下一个超时事件的指针 */ 
u32_t time; 			/* 当前超时事件的等待时间 */ 
sys_timeout_handler h; 	/* 指向超时的回调函数 */ 
void *arg; 			/* 超时的回调函数形数 */ 
};

超时事件注册

void sys_timeouts_init(void) 
{ 
size_t i; /* 遍历轮询超时事件数组 */ 
for (i = (LWIP_TCP ? 1 : 0); i < LWIP_ARRAYSIZE(lwip_cyclic_timers); i++) 
{ 
/* 注册超时事件 */ 
sys_timeout(lwip_cyclic_timers[i].interval_ms, lwip_cyclic_timer,LWIP_CONST_CAST(void *, &lwip_cyclic_timers[i])); 
} 
}
static void
#if LWIP_DEBUG_TIMERNAMES
sys_timeout_abs(u32_t abs_time, sys_timeout_handler handler, void *arg, const char *handler_name)
#else /* LWIP_DEBUG_TIMERNAMES */
sys_timeout_abs(u32_t abs_time, sys_timeout_handler handler, void *arg)
#endif
{struct sys_timeo *timeout, *t;timeout = (struct sys_timeo *)memp_malloc(MEMP_SYS_TIMEOUT);//申请内存,申请一个超时管理大小的内存if (timeout == NULL) {LWIP_ASSERT("sys_timeout: timeout != NULL, pool MEMP_SYS_TIMEOUT is empty", timeout != NULL);return;//申请失败}timeout->next = NULL;  //第一个超时事件先指向为空,在前插入timeout->h = handler;  //超时回调函数timeout->arg = arg;    //超时回调参数timeout->time = abs_time;//超时等待时间#if LWIP_DEBUG_TIMERNAMEStimeout->handler_name = handler_name;LWIP_DEBUGF(TIMERS_DEBUG, ("sys_timeout: %p abs_time=%"U32_F" handler=%s arg=%p\n",(void *)timeout, abs_time, handler_name, (void *)arg));
#endif /* LWIP_DEBUG_TIMERNAMES */if (next_timeout == NULL) {//指向链表的表头next_timeout = timeout;//赋值第一个超时事件的地址return;  //第一次从这里退出  第二次可以往下流}//if (TIME_LESS_THAN(timeout->time, next_timeout->time)) {//看当前的超时事件时间  如果当前超时时间时间大于刚才注册的时间 就往前插入 否则往后插入timeout->next = next_timeout;//往后插入  当前指向刚才注册事件next_timeout = timeout;} else {//往前插入for (t = next_timeout; t != NULL; t = t->next) {//遍历if ((t->next == NULL) || TIME_LESS_THAN(timeout->time, t->next->time)) {//重复对比当前事件和前面事件的大小,找到合适的位置timeout->next = t->next;t->next = timeout;break;}}}
}

next:下一个超时事件的指针

time:当前超时事件的等待时间

h:指向超时的回调函数

arg:超时的回调函数形数 

PS:注意arg0 是第一个超时事件  next_timeout是程序的一个临时变量

超时事件如何删除

void sys_untimeout(sys_timeout_handler handler, void *arg) 
{ 
struct sys_timeo *prev_t, *t; 
if (next_timeout == NULL) 
{ return; } 
/* 从链表头开始遍历这个链表 */ 
for (t = next_timeout, prev_t = NULL; t != NULL; prev_t = t, t = t->next){ 
/* 查找删除的超时事件,判断超时事件的回调函数与函数参数释放一致 */ 
if ((t->h == handler) && (t->arg == arg)) 
{ 
if (prev_t == NULL) { next_timeout = t->next; } 
else { prev_t->next = t->next; } memp_free(MEMP_SYS_TIMEOUT, t); return; 
} 
} 
return; }

把删除的事前的后一个事件指向   删除事件的前一个事件

超时事件如何查询

timeouts.c

Middlewares\lwip\src\core\timeouts.c

超时事件数组定义

每一个宏都是该协议的超时事件的使能

参考  :  LWIP超时事件结构体

/** This array contains all stack-internal cyclic timers. To get the number of* timers, use LWIP_ARRAYSIZE() */
const struct lwip_cyclic_timer lwip_cyclic_timers[] = {
#if LWIP_TCP/* The TCP timer is a special case: it does not have to run always andis triggered to start from TCP using tcp_timer_needed() */{TCP_TMR_INTERVAL, HANDLER(tcp_tmr)},
#endif /* LWIP_TCP */
#if LWIP_IPV4
#if IP_REASSEMBLY{IP_TMR_INTERVAL, HANDLER(ip_reass_tmr)},
#endif /* IP_REASSEMBLY */
#if LWIP_ARP{ARP_TMR_INTERVAL, HANDLER(etharp_tmr)},
#endif /* LWIP_ARP */
#if LWIP_DHCP{DHCP_COARSE_TIMER_MSECS, HANDLER(dhcp_coarse_tmr)},{DHCP_FINE_TIMER_MSECS, HANDLER(dhcp_fine_tmr)},
#endif /* LWIP_DHCP */
#if LWIP_AUTOIP{AUTOIP_TMR_INTERVAL, HANDLER(autoip_tmr)},
#endif /* LWIP_AUTOIP */
#if LWIP_IGMP{IGMP_TMR_INTERVAL, HANDLER(igmp_tmr)},
#endif /* LWIP_IGMP */
#endif /* LWIP_IPV4 */
#if LWIP_DNS{DNS_TMR_INTERVAL, HANDLER(dns_tmr)},
#endif /* LWIP_DNS */
#if LWIP_IPV6{ND6_TMR_INTERVAL, HANDLER(nd6_tmr)},
#if LWIP_IPV6_REASS{IP6_REASS_TMR_INTERVAL, HANDLER(ip6_reass_tmr)},
#endif /* LWIP_IPV6_REASS */
#if LWIP_IPV6_MLD{MLD6_TMR_INTERVAL, HANDLER(mld6_tmr)},
#endif /* LWIP_IPV6_MLD */
#if LWIP_IPV6_DHCP6{DHCP6_TIMER_MSECS, HANDLER(dhcp6_tmr)},
#endif /* LWIP_IPV6_DHCP6 */
#endif /* LWIP_IPV6 */
};

http://www.xdnf.cn/news/4527.html

相关文章:

  • 【python】使用Python和BERT进行文本摘要:从数据预处理到模型训练与生成
  • vllm命令行启动方式并发性能实测
  • 联想Horizon 2系列电脑 参数
  • SpringBoot学生宿舍管理系统开发实现
  • 浏览器跨标签通信的实现原理
  • feign负载均衡
  • linux(centos)联网情况下部署
  • 第一章——typec电路
  • SpirngAI框架 Advisor API详解
  • 【无标题】如何在sheel中运行Spark
  • 基于Django框架开发的企业级IT资产管理系统
  • Topic和Partition的关系是什么?为什么需要分区? (Topic是逻辑分类,Partition是物理分片;提升并行度和扩展性)
  • 【信息系统项目管理师-论文真题】2005下半年论文详解(包括解题思路和写作要点)
  • mint系统详解详细解释
  • 开源数学推理模型DeepSeek-Prover-V2:88.9%通过率+超长推理链
  • 数造科技携 DataBuilder 亮相安徽科交会,展现“DataOps +AI”双引擎魅力
  • 机器学习之嵌入(Embeddings):从理论到实践
  • LangChain第二讲:不设置环境变量也能调用LLM大模型吗?(更简单地调用LLM)
  • LabVIEW表面粗糙度测量及算法解析
  • Python cv2视频处理基础:从入门到实战
  • 我如何在ubuntu截图和屏幕录制,有什么好用的免费的软件吗?
  • C++ 基础复习
  • 蓝牙L2CAP协议概述
  • 微机控制电液伺服拉扭疲劳试验系统
  • 004 Linux基本指令
  • C语言| 递归求两个数的最大公约数
  • 17.Three.js 光照系统之《LightProbe》详解指南(含 Vue 3示例)
  • 准确--Notepad++ 实用的插件介绍
  • 【论文阅读】HunyuanVideo: A Systematic Framework For Large Video Generative Models
  • Linux系统安装指南