quiche/recovery/
rtt.rs

1// Copyright (C) 2024, Cloudflare, Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8//     * Redistributions of source code must retain the above copyright notice,
9//       this list of conditions and the following disclaimer.
10//
11//     * Redistributions in binary form must reproduce the above copyright
12//       notice, this list of conditions and the following disclaimer in the
13//       documentation and/or other materials provided with the distribution.
14//
15// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
16// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
19// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27use std::time::Duration;
28use std::time::Instant;
29
30use crate::minmax::Minmax;
31use crate::recovery::GRANULARITY;
32
33pub(crate) const INITIAL_RTT: Duration = Duration::from_millis(333);
34
35pub(crate) const RTT_WINDOW: Duration = Duration::from_secs(300);
36
37pub struct RttStats {
38    pub(super) latest_rtt: Duration,
39
40    pub(super) smoothed_rtt: Duration,
41
42    pub(super) rttvar: Duration,
43
44    pub(super) min_rtt: Minmax<Duration>,
45
46    pub(super) max_ack_delay: Duration,
47
48    pub(super) first_rtt_sample: Option<Instant>,
49}
50
51impl std::fmt::Debug for RttStats {
52    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
53        f.debug_struct("RttStats")
54            .field("lastest_rtt", &self.latest_rtt)
55            .field("srtt", &self.smoothed_rtt)
56            .field("minrtt", &*self.min_rtt)
57            .field("rttvar", &self.rttvar)
58            .finish()
59    }
60}
61
62impl RttStats {
63    pub(crate) fn new(max_ack_delay: Duration) -> Self {
64        RttStats {
65            latest_rtt: Duration::ZERO,
66            min_rtt: Minmax::new(INITIAL_RTT),
67            smoothed_rtt: INITIAL_RTT,
68            rttvar: INITIAL_RTT / 2,
69            first_rtt_sample: None,
70            max_ack_delay,
71        }
72    }
73
74    pub(crate) fn update_rtt(
75        &mut self, latest_rtt: Duration, mut ack_delay: Duration, now: Instant,
76        handshake_confirmed: bool,
77    ) {
78        self.latest_rtt = latest_rtt;
79
80        if self.first_rtt_sample.is_none() {
81            self.min_rtt.reset(now, latest_rtt);
82            self.smoothed_rtt = latest_rtt;
83            self.rttvar = latest_rtt / 2;
84            self.first_rtt_sample = Some(now);
85            return;
86        }
87
88        // min_rtt ignores acknowledgment delay.
89        self.min_rtt.running_min(RTT_WINDOW, now, latest_rtt);
90
91        // Limit ack_delay by max_ack_delay after handshake confirmation.
92        if handshake_confirmed {
93            ack_delay = ack_delay.min(self.max_ack_delay);
94        }
95
96        // Adjust for acknowledgment delay if plausible.
97        let mut adjusted_rtt = latest_rtt;
98        if latest_rtt >= *self.min_rtt + ack_delay {
99            adjusted_rtt = latest_rtt - ack_delay;
100        }
101
102        self.rttvar = self.rttvar * 3 / 4 +
103            Duration::from_nanos(
104                self.smoothed_rtt
105                    .as_nanos()
106                    .abs_diff(adjusted_rtt.as_nanos()) as u64 /
107                    4,
108            );
109
110        self.smoothed_rtt = self.smoothed_rtt * 7 / 8 + adjusted_rtt / 8;
111    }
112
113    pub(crate) fn rtt(&self) -> Duration {
114        self.smoothed_rtt
115    }
116
117    #[allow(dead_code)]
118    pub(crate) fn latest_rtt(&self) -> Duration {
119        self.latest_rtt
120    }
121
122    pub(crate) fn rttvar(&self) -> Duration {
123        self.rttvar
124    }
125
126    pub(crate) fn min_rtt(&self) -> Option<Duration> {
127        self.min_rtt.ne(&Duration::ZERO).then_some(*self.min_rtt)
128    }
129
130    pub(crate) fn loss_delay(&self, time_thresh: f64) -> Duration {
131        self.latest_rtt
132            .max(self.smoothed_rtt)
133            .mul_f64(time_thresh)
134            .max(GRANULARITY)
135    }
136}