• TCP系列15—重传—5、Linux中RTO的计算


    之前我们介绍的都是协议中给出的RTO计算方法,下面我们看一下linux实现中RTO的计算方法。在linux中维护了srtt、mdev、mdev_max、rttvar、rtt_seq几个状态变量用来计算RTO,其中linux实现中的mdev变量相当于协议中的RTTVAR变量。rtt_seq状态变量用来控制一个RTT时间窗,linux在一个RTT时间窗内部更新状态变量的方式与RTT时间窗结束更新状态变量的方式不同,rtt_seq则用来判断当前是在RTT时间窗内部,还是一个RTT时间窗已经结束

    一、RTT时间窗的判断

    几个状态变量中的rtt_seq用来判断当前采样是否处于RTT时间窗内,我们简单的说一下如何判断

    在TCP窗口管理时候维护发送窗口有两个状态变量,一个是snd.una,另外一个是snd.nxt。其中snd.una表示还没有被ACK确认的数据包里面最早的系列号,snd.nxt表示下一个待发送的数据包。初始的时候设置rtt_seq =snd.nxt,随着数据包的发送和ACK报文的接收,snd.una和snd.nxt都会向前滑行,当更新RTT状态变量的时候,如果发现snd.una<=rtt_seq,说明之前发送的数据包还没有收到ACK,当前还处于RTT时间窗内部。如果发现snd.una>rtt_seq说明之前发送的数据包已经收到了对应的ACK确认,那么一个RTT时间窗结束,并把rtt_seq设置为snd.nxt继续下一个RTT时间窗的处理

    二、状态变量的更新

    1、在测量到第一个RTT采样之前,linux会先查看本地缓存中是否由目标ip地址的RTT缓存信息,如果由对应的缓存信息,则会根据缓存信息初始化RTO,如果没有对应的缓存信息,则会把RTO初始化为3s。

    我们可以使用ip tcp_metrics查看有所缓存,也可以使用ip tcp_metrics show ip命令查看某个ip地址相关的缓存信息

    1. ******@Inspiron:~$ ip tcp_metrics show 121.201.104.55
    2. 121.201.104.55 age 56.604sec cwnd 10 rtt 461481us rttvar 461481us source 192.168.1.103
    2、在linux测量到第一个RTT采样m的时候,按照如下初始化相关状态变量
    1. srtt = m
    2. mddev = m/2
    3. rttvar = max(mdev, min_rto)
    4. mdev_max = rttvar
    5. RTO = srtt + 4 * rttvar

    其中min_rto为该目标地址的最小RTT,如果路由中有配置那么使用配置值,如果没有配置则使用TCP_RTO_MIN,TCP_RTO_MIN常量为50ms(linux代码中这个常量为200ms实际为放大四倍后的值)。


    3、在随后再次收到RTT测量值m的时候,按照如下更新mdev

    1. if(m < (srtt - mdev))
    2.    mdev = (31/32) * mdev + (1/32) * |srtt - m|
    3. else
    4.    mdev = (3/4) * mdev + (1/4) * |srtt - m|

    我们之前说过linux实现中mdev变量相当于协议中的RTTVAR变量,这里mdev的更新与协议有了明显的不同,主要原因是如果链路时延突然大幅降低的时候,如果按照协议方法更新反而会大幅增加最后的RTO,因此linux在发现链路时延大幅下降的时候会降低RTO增长的幅度。其他状态变量更新如下

    1. mdev_max = max(mdev_max, mdev)
    2. srtt = (7/8) * srtt + (1/8) * m
    3. rttvar = mdev_max
    4. RTO = srtt + 4 * rttvar

    4、根据rtt_seq来判断如果当前是RTT时间窗结束,则执行如下流程

    1. if(mdev_max < rtt_var)
    2. {
    3.    rtt_var = (3/4) * rtt_var + (1/4) * mdev_max
    4. }
    5. mdev_max = min_rto
    注意我在if判断后面加了一个大括号来强调,不管if条件是否为真,只要RTT时间窗结束就要初始化mdev_max为min_rto。


    补充说明:

    1、linux中在测量到第一个RTT采样之前RTO的初始化参考tcp_init_metrics,RTT相关状态变量的更新参考tcp_rtt_estimator,其中mdev放大4倍,srtt放大8倍进行的保存和计算。





  • 相关阅读:
    C字符串处理函数
    C语言字符串函数大全
    那些闪亮的日子
    牛客网在线编程:幸运数
    牛客网在线编程:水仙花数
    [LeetCode]617.Merge Two Binary Trees
    [LeetCode]657.Judge Route Circle
    [LeetCode]141. Linked List Cycle
    五大算法:分治,贪心,动态规划,回溯,分支界定
    [LeetCode]387.First Unique Character in a String
  • 原文地址:https://www.cnblogs.com/lshs/p/6038541.html
Copyright © 2020-2023  润新知