quiche/recovery/congestion/bbr/
mod.rs

1// Copyright (C) 2022, 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
27//! BBR Congestion Control
28//!
29//! This implementation is based on the following draft:
30//! <https://tools.ietf.org/html/draft-cardwell-iccrg-bbr-congestion-control-00>
31
32use super::*;
33use crate::minmax::Minmax;
34
35use std::time::Duration;
36
37use super::CongestionControlOps;
38
39pub(crate) static BBR: CongestionControlOps = CongestionControlOps {
40    on_init,
41    on_packet_sent,
42    on_packets_acked,
43    congestion_event,
44    checkpoint,
45    rollback,
46    has_custom_pacing,
47    debug_fmt,
48};
49
50/// A constant specifying the length of the BBR.BtlBw max filter window for
51/// BBR.BtlBwFilter, BtlBwFilterLen is 10 packet-timed round trips.
52const BTLBW_FILTER_LEN: Duration = Duration::from_secs(10);
53
54/// A constant specifying the minimum time interval between ProbeRTT states: 10
55/// secs.
56const PROBE_RTT_INTERVAL: Duration = Duration::from_secs(10);
57
58/// A constant specifying the length of the RTProp min filter window.
59const RTPROP_FILTER_LEN: Duration = PROBE_RTT_INTERVAL;
60
61/// A constant specifying the minimum gain value that will allow the sending
62/// rate to double each round (2/ln(2) ~= 2.89), used in Startup mode for both
63/// BBR.pacing_gain and BBR.cwnd_gain.
64const BBR_HIGH_GAIN: f64 = 2.89;
65
66/// The minimal cwnd value BBR tries to target using: 4 packets, or 4 * SMSS
67const BBR_MIN_PIPE_CWND_PKTS: usize = 4;
68
69/// The number of phases in the BBR ProbeBW gain cycle: 8.
70const BBR_GAIN_CYCLE_LEN: usize = 8;
71
72/// A constant specifying the minimum duration for which ProbeRTT state holds
73/// inflight to BBRMinPipeCwnd or fewer packets: 200 ms.
74const PROBE_RTT_DURATION: Duration = Duration::from_millis(200);
75
76/// Pacing Gain Cycle.
77const PACING_GAIN_CYCLE: [f64; BBR_GAIN_CYCLE_LEN] =
78    [5.0 / 4.0, 3.0 / 4.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0];
79
80/// A constant to check BBR.BtlBW is still growing.
81const BTLBW_GROWTH_TARGET: f64 = 1.25;
82
83/// BBR Internal State Machine.
84#[derive(Debug, PartialEq, Eq)]
85enum BBRStateMachine {
86    Startup,
87    Drain,
88    ProbeBW,
89    ProbeRTT,
90}
91
92/// BBR Specific State Variables.
93pub struct State {
94    // The current state of a BBR flow in the BBR state machine.
95    state: BBRStateMachine,
96
97    // The current pacing rate for a BBR flow, which controls inter-packet
98    // spacing.
99    pacing_rate: u64,
100
101    // BBR's estimated bottleneck bandwidth available to the transport flow,
102    // estimated from the maximum delivery rate sample in a sliding window.
103    btlbw: u64,
104
105    // The max filter used to estimate BBR.BtlBw.
106    btlbwfilter: Minmax<u64>,
107
108    // BBR's estimated two-way round-trip propagation delay of the path,
109    // estimated from the windowed minimum recent round-trip delay sample.
110    rtprop: Duration,
111
112    // The wall clock time at which the current BBR.RTProp sample was obtained.
113    rtprop_stamp: Instant,
114
115    // A boolean recording whether the BBR.RTprop has expired and is due for a
116    // refresh with an application idle period or a transition into ProbeRTT
117    // state.
118    rtprop_expired: bool,
119
120    // The dynamic gain factor used to scale BBR.BtlBw to produce
121    // BBR.pacing_rate.
122    pacing_gain: f64,
123
124    // The dynamic gain factor used to scale the estimated BDP to produce a
125    // congestion window (cwnd).
126    cwnd_gain: f64,
127
128    // A boolean that records whether BBR estimates that it has ever fully
129    // utilized its available bandwidth ("filled the pipe").
130    filled_pipe: bool,
131
132    // Count of packet-timed round trips elapsed so far.
133    round_count: u64,
134
135    // A boolean that BBR sets to true once per packet-timed round trip,
136    // on ACKs that advance BBR.round_count.
137    round_start: bool,
138
139    // packet.delivered value denoting the end of a packet-timed round trip.
140    next_round_delivered: usize,
141
142    // Timestamp when ProbeRTT state ends.
143    probe_rtt_done_stamp: Option<Instant>,
144
145    // Checking if a roundtrip in ProbeRTT state ends.
146    probe_rtt_round_done: bool,
147
148    // Checking if in the packet conservation mode during recovery.
149    packet_conservation: bool,
150
151    // Saved cwnd before loss recovery.
152    prior_cwnd: usize,
153
154    // Checking if restarting from idle.
155    idle_restart: bool,
156
157    // Baseline level delivery rate for full pipe estimator.
158    full_bw: u64,
159
160    // The number of round for full pipe estimator without much growth.
161    full_bw_count: usize,
162
163    // Last time cycle_index is updated.
164    cycle_stamp: Instant,
165
166    // Current index of pacing_gain_cycle[].
167    cycle_index: usize,
168
169    // The upper bound on the volume of data BBR allows in flight.
170    target_cwnd: usize,
171
172    // Whether in the recovery episode.
173    in_recovery: bool,
174
175    // Start time of the connection.
176    start_time: Instant,
177
178    // Newly marked lost data size in bytes.
179    newly_lost_bytes: usize,
180
181    // Newly acked data size in bytes.
182    newly_acked_bytes: usize,
183
184    // bytes_in_flight before processing this ACK.
185    prior_bytes_in_flight: usize,
186}
187
188impl State {
189    pub fn new() -> Self {
190        let now = Instant::now();
191
192        State {
193            state: BBRStateMachine::Startup,
194
195            pacing_rate: 0,
196
197            btlbw: 0,
198
199            btlbwfilter: Minmax::new(0),
200
201            rtprop: Duration::ZERO,
202
203            rtprop_stamp: now,
204
205            rtprop_expired: false,
206
207            pacing_gain: 0.0,
208
209            cwnd_gain: 0.0,
210
211            filled_pipe: false,
212
213            round_count: 0,
214
215            round_start: false,
216
217            next_round_delivered: 0,
218
219            probe_rtt_done_stamp: None,
220
221            probe_rtt_round_done: false,
222
223            packet_conservation: false,
224
225            prior_cwnd: 0,
226
227            idle_restart: false,
228
229            full_bw: 0,
230
231            full_bw_count: 0,
232
233            cycle_stamp: now,
234
235            cycle_index: 0,
236
237            target_cwnd: 0,
238
239            in_recovery: false,
240
241            start_time: now,
242
243            newly_lost_bytes: 0,
244
245            newly_acked_bytes: 0,
246
247            prior_bytes_in_flight: 0,
248        }
249    }
250}
251
252// When entering the recovery episode.
253fn bbr_enter_recovery(r: &mut Congestion, in_flight: usize, now: Instant) {
254    r.bbr_state.prior_cwnd = per_ack::bbr_save_cwnd(r);
255
256    r.congestion_window = in_flight.max(r.max_datagram_size);
257    r.congestion_recovery_start_time = Some(now);
258
259    r.bbr_state.packet_conservation = true;
260    r.bbr_state.in_recovery = true;
261
262    r.bbr_state.newly_lost_bytes = 0;
263
264    // Start round now.
265    r.bbr_state.next_round_delivered = r.delivery_rate.delivered();
266}
267
268// When exiting the recovery episode.
269fn bbr_exit_recovery(r: &mut Congestion) {
270    r.congestion_recovery_start_time = None;
271
272    r.bbr_state.packet_conservation = false;
273    r.bbr_state.in_recovery = false;
274
275    per_ack::bbr_restore_cwnd(r);
276}
277
278// Congestion Control Hooks.
279//
280fn on_init(r: &mut Congestion) {
281    init::bbr_init(r);
282}
283
284fn on_packet_sent(
285    r: &mut Congestion, _sent_bytes: usize, bytes_in_flight: usize, _now: Instant,
286) {
287    per_transmit::bbr_on_transmit(r, bytes_in_flight);
288}
289
290fn on_packets_acked(
291    r: &mut Congestion, bytes_in_flight: usize, packets: &mut Vec<Acked>,
292    now: Instant, _rtt_stats: &RttStats,
293) {
294    r.bbr_state.prior_bytes_in_flight = bytes_in_flight;
295
296    r.bbr_state.newly_acked_bytes =
297        packets.drain(..).fold(0, |acked_bytes, p| {
298            r.bbr_state.prior_bytes_in_flight -= p.size;
299
300            per_ack::bbr_update_model_and_state(r, &p, bytes_in_flight, now);
301
302            acked_bytes + p.size
303        });
304
305    if let Some(pkt) = packets.last() {
306        if !r.in_congestion_recovery(pkt.time_sent) && r.bbr_state.in_recovery {
307            // Upon exiting loss recovery.
308            bbr_exit_recovery(r);
309        }
310    }
311
312    per_ack::bbr_update_control_parameters(r, bytes_in_flight, now);
313
314    r.bbr_state.newly_lost_bytes = 0;
315}
316
317fn congestion_event(
318    r: &mut Congestion, bytes_in_flight: usize, lost_bytes: usize,
319    largest_lost_pkt: &Sent, now: Instant,
320) {
321    r.bbr_state.newly_lost_bytes = lost_bytes;
322
323    // Upon entering Fast Recovery.
324    if !r.in_congestion_recovery(largest_lost_pkt.time_sent) {
325        // Upon entering Fast Recovery.
326        bbr_enter_recovery(r, bytes_in_flight - lost_bytes, now);
327    }
328}
329
330fn checkpoint(_r: &mut Congestion) {}
331
332fn rollback(_r: &mut Congestion) -> bool {
333    false
334}
335
336fn has_custom_pacing() -> bool {
337    true
338}
339
340fn debug_fmt(r: &Congestion, f: &mut std::fmt::Formatter) -> std::fmt::Result {
341    let bbr = &r.bbr_state;
342
343    write!(
344         f,
345         "bbr={{ state={:?} btlbw={} rtprop={:?} pacing_rate={} pacing_gain={} cwnd_gain={} target_cwnd={} send_quantum={} filled_pipe={} round_count={} }}",
346         bbr.state, bbr.btlbw, bbr.rtprop, bbr.pacing_rate, bbr.pacing_gain, bbr.cwnd_gain, bbr.target_cwnd, r.send_quantum(), bbr.filled_pipe, bbr.round_count
347    )
348}
349
350#[cfg(test)]
351mod tests {
352    use super::*;
353
354    use crate::packet;
355    use crate::ranges;
356    use crate::recovery::congestion::recovery::LegacyRecovery;
357    use crate::recovery::congestion::test_sender::TestSender;
358    use crate::recovery::HandshakeStatus;
359    use crate::recovery::RecoveryOps;
360
361    use smallvec::smallvec;
362
363    fn test_sender() -> TestSender {
364        TestSender::new(CongestionControlAlgorithm::BBR, false)
365    }
366
367    #[test]
368    fn bbr_init() {
369        let mut cfg = crate::Config::new(crate::PROTOCOL_VERSION).unwrap();
370        cfg.set_cc_algorithm(CongestionControlAlgorithm::BBR);
371
372        let r = LegacyRecovery::new(&cfg);
373
374        assert_eq!(
375            r.cwnd(),
376            r.max_datagram_size * cfg.initial_congestion_window_packets
377        );
378        assert_eq!(r.bytes_in_flight, 0);
379
380        assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::Startup);
381    }
382
383    #[test]
384    fn bbr_startup() {
385        let mut sender = test_sender();
386        let mss = sender.max_datagram_size;
387
388        let rtt = Duration::from_millis(50);
389        sender.update_rtt(rtt);
390        sender.advance_time(rtt);
391
392        // Send 5 packets.
393        for _ in 0..5 {
394            sender.send_packet(mss);
395        }
396
397        sender.advance_time(rtt);
398
399        let cwnd_prev = sender.congestion_window;
400
401        sender.ack_n_packets(5, mss);
402
403        assert_eq!(sender.bbr_state.state, BBRStateMachine::Startup);
404        assert_eq!(sender.congestion_window, cwnd_prev + mss * 5);
405        assert_eq!(sender.bytes_in_flight, 0);
406        assert_eq!(
407            sender.delivery_rate(),
408            ((mss * 5) as f64 / rtt.as_secs_f64()) as u64
409        );
410        assert_eq!(sender.bbr_state.btlbw, sender.delivery_rate());
411    }
412
413    #[test]
414    fn bbr_congestion_event() {
415        let mut cfg = crate::Config::new(crate::PROTOCOL_VERSION).unwrap();
416        cfg.set_cc_algorithm(CongestionControlAlgorithm::BBR);
417
418        let mut r = LegacyRecovery::new(&cfg);
419        let now = Instant::now();
420        let mss = r.max_datagram_size;
421
422        // Send 5 packets.
423        for pn in 0..5 {
424            let pkt = Sent {
425                pkt_num: pn,
426                frames: smallvec![],
427                time_sent: now,
428                time_acked: None,
429                time_lost: None,
430                size: mss,
431                ack_eliciting: true,
432                in_flight: true,
433                delivered: 0,
434                delivered_time: now,
435                first_sent_time: now,
436                is_app_limited: false,
437                tx_in_flight: 0,
438                lost: 0,
439                has_data: false,
440                pmtud: false,
441            };
442
443            r.on_packet_sent(
444                pkt,
445                packet::Epoch::Application,
446                HandshakeStatus::default(),
447                now,
448                "",
449            );
450        }
451
452        let rtt = Duration::from_millis(50);
453        let now = now + rtt;
454
455        // Make a packet loss to trigger a congestion event.
456        let mut acked = ranges::RangeSet::default();
457        acked.insert(4..5);
458
459        // 1 acked, 2 x MSS lost.
460        assert_eq!(
461            r.on_ack_received(
462                &acked,
463                25,
464                packet::Epoch::Application,
465                HandshakeStatus::default(),
466                now,
467                "",
468            ),
469            (2, 2 * mss, mss),
470        );
471
472        // Sent: 0, 1, 2, 3, 4, Acked 4.
473        assert_eq!(r.cwnd(), mss * 4);
474        // Stil in flight: 2, 3.
475        assert_eq!(r.bytes_in_flight, mss * 2);
476    }
477
478    #[test]
479    fn bbr_drain() {
480        let mut cfg = crate::Config::new(crate::PROTOCOL_VERSION).unwrap();
481        cfg.set_cc_algorithm(CongestionControlAlgorithm::BBR);
482
483        let mut r = LegacyRecovery::new(&cfg);
484        let now = Instant::now();
485        let mss = r.max_datagram_size;
486
487        let mut pn = 0;
488
489        // Stop right before filled_pipe=true.
490        for _ in 0..3 {
491            let pkt = Sent {
492                pkt_num: pn,
493                frames: smallvec![],
494                time_sent: now,
495                time_acked: None,
496                time_lost: None,
497                size: mss,
498                ack_eliciting: true,
499                in_flight: true,
500                delivered: r.congestion.delivery_rate.delivered(),
501                delivered_time: now,
502                first_sent_time: now,
503                is_app_limited: false,
504                tx_in_flight: 0,
505                lost: 0,
506                has_data: false,
507                pmtud: false,
508            };
509
510            r.on_packet_sent(
511                pkt,
512                packet::Epoch::Application,
513                HandshakeStatus::default(),
514                now,
515                "",
516            );
517
518            pn += 1;
519
520            let rtt = Duration::from_millis(50);
521
522            let now = now + rtt;
523
524            let mut acked = ranges::RangeSet::default();
525            acked.insert(0..pn);
526
527            assert_eq!(
528                r.on_ack_received(
529                    &acked,
530                    25,
531                    packet::Epoch::Application,
532                    HandshakeStatus::default(),
533                    now,
534                    "",
535                ),
536                (0, 0, mss),
537            );
538        }
539
540        // Stop at right before filled_pipe=true.
541        for _ in 0..5 {
542            let pkt = Sent {
543                pkt_num: pn,
544                frames: smallvec![],
545                time_sent: now,
546                time_acked: None,
547                time_lost: None,
548                size: mss,
549                ack_eliciting: true,
550                in_flight: true,
551                delivered: r.congestion.delivery_rate.delivered(),
552                delivered_time: now,
553                first_sent_time: now,
554                is_app_limited: false,
555                tx_in_flight: 0,
556                lost: 0,
557                has_data: false,
558                pmtud: false,
559            };
560
561            r.on_packet_sent(
562                pkt,
563                packet::Epoch::Application,
564                HandshakeStatus::default(),
565                now,
566                "",
567            );
568
569            pn += 1;
570        }
571
572        let rtt = Duration::from_millis(50);
573        let now = now + rtt;
574
575        let mut acked = ranges::RangeSet::default();
576
577        // We sent 5 packets, but ack only one, to stay
578        // in Drain state.
579        acked.insert(0..pn - 4);
580
581        assert_eq!(
582            r.on_ack_received(
583                &acked,
584                25,
585                packet::Epoch::Application,
586                HandshakeStatus::default(),
587                now,
588                "",
589            ),
590            (0, 0, mss),
591        );
592
593        // Now we are in Drain state.
594        assert!(r.congestion.bbr_state.filled_pipe);
595        assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::Drain);
596        assert!(r.congestion.bbr_state.pacing_gain < 1.0);
597    }
598
599    #[test]
600    fn bbr_probe_bw() {
601        let mut cfg = crate::Config::new(crate::PROTOCOL_VERSION).unwrap();
602        cfg.set_cc_algorithm(CongestionControlAlgorithm::BBR);
603
604        let mut r = LegacyRecovery::new(&cfg);
605        let now = Instant::now();
606        let mss = r.max_datagram_size;
607
608        // At 4th roundtrip, filled_pipe=true and switch to Drain,
609        // but move to ProbeBW immediately because bytes_in_flight is
610        // smaller than BBRInFlight(1).
611        for (pn, _) in (0..4).enumerate() {
612            let pkt = Sent {
613                pkt_num: pn as u64,
614                frames: smallvec![],
615                time_sent: now,
616                time_acked: None,
617                time_lost: None,
618                size: mss,
619                ack_eliciting: true,
620                in_flight: true,
621                delivered: r.congestion.delivery_rate.delivered(),
622                delivered_time: now,
623                first_sent_time: now,
624                is_app_limited: false,
625                tx_in_flight: 0,
626                lost: 0,
627                has_data: false,
628                pmtud: false,
629            };
630
631            r.on_packet_sent(
632                pkt,
633                packet::Epoch::Application,
634                HandshakeStatus::default(),
635                now,
636                "",
637            );
638
639            let rtt = Duration::from_millis(50);
640            let now = now + rtt;
641
642            let mut acked = ranges::RangeSet::default();
643            acked.insert(0..pn as u64 + 1);
644
645            assert_eq!(
646                r.on_ack_received(
647                    &acked,
648                    25,
649                    packet::Epoch::Application,
650                    HandshakeStatus::default(),
651                    now,
652                    "",
653                ),
654                (0, 0, mss),
655            );
656        }
657
658        // Now we are in ProbeBW state.
659        assert!(r.congestion.bbr_state.filled_pipe);
660        assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::ProbeBW);
661
662        // In the first ProbeBW cycle, pacing_gain should be >= 1.0.
663        assert!(r.congestion.bbr_state.pacing_gain >= 1.0);
664    }
665
666    #[test]
667    fn bbr_probe_rtt() {
668        let mut cfg = crate::Config::new(crate::PROTOCOL_VERSION).unwrap();
669        cfg.set_cc_algorithm(CongestionControlAlgorithm::BBR);
670
671        let mut r = LegacyRecovery::new(&cfg);
672        let now = Instant::now();
673        let mss = r.max_datagram_size;
674
675        let mut pn = 0;
676
677        // At 4th roundtrip, filled_pipe=true and switch to Drain,
678        // but move to ProbeBW immediately because bytes_in_flight is
679        // smaller than BBRInFlight(1).
680        for _ in 0..4 {
681            let pkt = Sent {
682                pkt_num: pn,
683                frames: smallvec![],
684                time_sent: now,
685                time_acked: None,
686                time_lost: None,
687                size: mss,
688                ack_eliciting: true,
689                in_flight: true,
690                delivered: r.congestion.delivery_rate.delivered(),
691                delivered_time: now,
692                first_sent_time: now,
693                is_app_limited: false,
694                tx_in_flight: 0,
695                lost: 0,
696                has_data: false,
697                pmtud: false,
698            };
699
700            r.on_packet_sent(
701                pkt,
702                packet::Epoch::Application,
703                HandshakeStatus::default(),
704                now,
705                "",
706            );
707
708            pn += 1;
709
710            let rtt = Duration::from_millis(50);
711            let now = now + rtt;
712
713            let mut acked = ranges::RangeSet::default();
714            acked.insert(0..pn);
715
716            assert_eq!(
717                r.on_ack_received(
718                    &acked,
719                    25,
720                    packet::Epoch::Application,
721                    HandshakeStatus::default(),
722                    now,
723                    "",
724                ),
725                (0, 0, mss),
726            );
727        }
728
729        // Now we are in ProbeBW state.
730        assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::ProbeBW);
731
732        // After RTPROP_FILTER_LEN (10s), switch to ProbeRTT.
733        let now = now + RTPROP_FILTER_LEN;
734
735        let pkt = Sent {
736            pkt_num: pn,
737            frames: smallvec![],
738            time_sent: now,
739            time_acked: None,
740            time_lost: None,
741            size: mss,
742            ack_eliciting: true,
743            in_flight: true,
744            delivered: r.congestion.delivery_rate.delivered(),
745            delivered_time: now,
746            first_sent_time: now,
747            is_app_limited: false,
748            tx_in_flight: 0,
749            lost: 0,
750            has_data: false,
751            pmtud: false,
752        };
753
754        r.on_packet_sent(
755            pkt,
756            packet::Epoch::Application,
757            HandshakeStatus::default(),
758            now,
759            "",
760        );
761
762        pn += 1;
763
764        // Don't update rtprop by giving larger rtt than before.
765        // If rtprop is updated, rtprop expiry check is reset.
766        let rtt = Duration::from_millis(100);
767        let now = now + rtt;
768
769        let mut acked = ranges::RangeSet::default();
770        acked.insert(0..pn);
771
772        assert_eq!(
773            r.on_ack_received(
774                &acked,
775                25,
776                packet::Epoch::Application,
777                HandshakeStatus::default(),
778                now,
779                "",
780            ),
781            (0, 0, mss),
782        );
783
784        assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::ProbeRTT);
785        assert_eq!(r.congestion.bbr_state.pacing_gain, 1.0);
786    }
787}
788
789mod init;
790mod pacing;
791mod per_ack;
792mod per_transmit;