Linux_博客-调研tcp定时器

TCP的四种定时器

  1. 重传计时器:Retransmission Timer
  2. 坚持计时器:Persistent Timer
  3. 保活计时器:Keeplive Timer
  4. 时间等待计时器:Time_Wait Timer。

  • 重传计时器
    为了控制丢失的报文段或丢弃的报文段,也就是对报文段确认的等待时间。当TCP发送报文段时,就创建这个特定报文段的重传计时器,可能发生两种情况:若在计时器超时之前收到对报文段的确认,则撤销计时器;若在收到对特定报文段的确认之前计时器超时,则重传该报文,并把计时器复位;
    重传时间=2*RTT;
    RTT的值应该动态计算。常用的公式是:RTT=previous RTT*i + (1-i)*current RTT。i的值通常取90%,即新的RTT是以前的RTT值的90%加上当前RTT值的10%.

Karn算法:对重传报文,在计算新的RTT时,不考虑重传报文的RTT。因为无法推理出:发送端所收到的确认是对上一次报文段的确认还是对重传报文段的确认。干脆不计入。

  • 坚持计时器
    坚持定时器主要是解决零窗口大小通知可能导致的死锁问题。刚开始接收端向发送端发送了一个零窗口报文段。在不久之后,如果接收端的缓存区有一定的空间可以接收数据,此时接收端就会向发送端发送了一个非零窗口大小的报文段(即窗口更新),但是这个非零窗口大小的报文段在传输过程中丢失,导致发送端无法接收到该非零窗口大小的报文段。因此,发送端就会一直处于等待非零窗口大小的报文端通知,由于接收端已经发送了非零窗口大小的报文段,而且并不知道该报文段在传输过程中丢失,则接收端会一直处于等待接收数据状态,如果没有任何措施的话,这个死锁的局面会一直延续下去。
    为了解决上面这个问题,TCP 为每一个连接设有一个坚持定时器(也叫持续计数器)。当发送端收到零窗口的确认时,就启动坚持计时器,当坚持计时器截止期到时,发送端就发送一个特殊的报文段,叫探测报文段,这个报文段只有一个字节的数据。探测报文段有序号,但序号永远不需要确认,甚至在计算对其他部分数据的确认时这个序号也被忽略。探测报文段提醒接收端,确认已丢失,必须重传。
    坚持计时器的截止期设置为重传时间的值,但若没有收到来自接收端的响应,则发送另一个探测报文段,并将坚持计时器的值加倍和并复位,发送端继续发送探测报文段,将坚持计时器的值加倍和复位,直到这个值增大到阈值为止(通常为 60 秒)。在此之后,发送端每隔 60s 就发送一个报文段,直到窗口重新打开为止。
    坚持定时器的原理:当 TCP 服务器收到了客户端的 0 滑动窗口报文时,启动一个定时器来计时,并在定时器溢出的时向客户端查询窗口是否已经增大,如果得到非零的窗口就重新开始发送数据,如果得到零窗口就再开一个新的定时器准备下一次查询。

  • 保活计时器
    每当服务器收到客户的信息,就将keeplive timer复位,超时通常设置2小时,若服务器超过2小时还没有收到来自客户的信息,就发送探测报文段,若发送了10个探测报文段(没75秒发送一个)还没收到响应,则终止连接。

  • 2MSL计时器
    2MSL 定时器主要是解决以下两种情况:
    TIME_WAIT 确保有足够的时间让对端收到了ACK,如果被动关闭的那方没有收到 ACK,就会触发被动端重发 FIN。因为最后一次确认应答 ACK 报文段很有可能丢失,因而使被动关闭方处于在LIST_ACK 状态的,此时被动关闭方会重发这个 FIN+ACK 报文段,在这等待的 2MSL 时间内主动关闭方重新收到这个被动关闭方重发的 FIN+ACK 报文段,因此,主动关闭方会重新发送确认应答信息,从而重新启动 2MSL 计时器,直到通信双方都进入 CLOSED 状态。如果主动关闭方在 TIME_WAIT 状态不等待一段时间就直接释放连接并进入 CLOSED 状态,那么主动关闭方无法收到来自被动关闭方重发的 FIN+ACK 报文段,也就不会再发送一次确认 ACK 报文段,因此被动关闭方就无法正常进入CLOSED 状态。
    有足够的时间让这个连接不会跟后面的连接混在一起。防止已失效的请求连接出现在本连接中。在连接处于 2MSL 等待时,任何迟到的报文段将被丢弃,因为处于 2MSL等待的、由该插口(插口是IP和端口对的意思,socket)定义的连接在这段时间内将不能被再用,这样就可以使下一个新的连接中不会出现这种旧的连接之前延迟的报文段。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这段代码是在 Linux 内核中的一个 Wi-Fi 驱动程序中的一段初始化代码,主要作用是将一些硬件和平台相关的数据结构进行初始化,并将它们保存在一个名为 platform_data 的指针中,以便后续的函数可以访问这些数据。具体来说,这段代码的含义如下: - ecrnx_hw 是一个指向 Wi-Fi 硬件相关的数据结构的指针,这个数据结构包含了一些硬件相关的信息,比如 MAC 地址等。 - wiphy 是一个指向 Linux 内核中的一个 Wi-Fi 设备的数据结构的指针,这个数据结构包含了一些与 Wi-Fi 设备相关的信息,比如支持的 Wi-Fi 协议等。 - wiphy_priv 是一个 Linux 内核中的函数,它的作用是获取 wiphy 数据结构中的一个私有数据结构,也就是 ecrnx_hw 数据结构。 - ecrnx_hw->wiphy = wiphy; 这行代码的作用是将 ecrnx_hw 中的 wiphy 指针指向 wiphy 数据结构,这样就可以在后续的代码中访问 wiphy 中的信息了。 - ecrnx_hw->plat = ecrnx_plat; 这行代码的作用是将 ecrnx_hw 中的 plat 指针指向一个平台相关的数据结构,这个数据结构包含了一些与硬件平台相关的信息。 - ecrnx_hw->dev = ecrnx_platform_get_dev(ecrnx_plat); 这行代码的作用是根据平台相关的数据结构 ecrnx_plat,获取一个指向设备的数据结构的指针,并将这个指针保存在 ecrnx_hw 中的 dev 字段中。 - ecrnx_hw->mod_params = &ecrnx_mod_params; 这行代码的作用是将 ecrnx_hw 中的 mod_params 指针指向一个模块参数的结构体,这个结构体包含了一些驱动程序中需要使用的参数。 - ecrnx_hw->tcp_pacing_shift = 7; 这行代码的作用是将 ecrnx_hw 中的 tcp_pacing_shift 字段设置为 7。 - *platform_data = ecrnx_hw; 这行代码的作用是将指向 ecrnx_hw 的指针保存在 platform_data 指针所指向的位置,以便后续的函数可以访问 ecrnx_hw 中的数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值