quiche/recovery/gcongestion/bbr2/
startup.rs1use std::time::Instant;
32
33use super::mode::Mode;
34use super::mode::ModeImpl;
35use super::network_model::BBRv2NetworkModel;
36use super::Acked;
37use super::BBRv2CongestionEvent;
38use super::Limits;
39use super::Lost;
40use super::PARAMS;
41
42#[derive(Debug)]
43pub(super) struct Startup {
44 pub(super) model: BBRv2NetworkModel,
45}
46
47impl ModeImpl for Startup {
48 fn is_probing_for_bandwidth(&self) -> bool {
49 true
50 }
51
52 fn on_congestion_event(
53 mut self, _prior_in_flight: usize, event_time: std::time::Instant,
54 _acked_packets: &[Acked], _lost_packets: &[Lost],
55 congestion_event: &mut BBRv2CongestionEvent,
56 _target_bytes_inflight: usize,
57 ) -> Mode {
58 if self.model.full_bandwidth_reached() {
59 return self.into_drain(event_time, Some(congestion_event));
60 }
61
62 if !congestion_event.end_of_round_trip {
63 return Mode::Startup(self);
64 }
65
66 let has_bandwidth_growth =
67 self.model.has_bandwidth_growth(congestion_event);
68
69 #[allow(clippy::absurd_extreme_comparisons)]
70 if PARAMS.max_startup_queue_rounds > 0 && !has_bandwidth_growth {
71 self.model.check_persistent_queue(1.75);
75 }
76 if !congestion_event.last_packet_send_state.is_app_limited &&
80 !has_bandwidth_growth
81 {
82 self.check_excessive_losses(congestion_event);
83 }
84
85 if self.model.full_bandwidth_reached() {
86 self.into_drain(event_time, Some(congestion_event))
87 } else {
88 Mode::Startup(self)
89 }
90 }
91
92 fn get_cwnd_limits(&self) -> Limits<usize> {
93 Limits {
94 lo: 0,
95 hi: self.model.inflight_lo(),
96 }
97 }
98
99 fn on_exit_quiescence(
100 self, _now: Instant, _quiescence_start_time: Instant,
101 ) -> Mode {
102 Mode::Startup(self)
103 }
104
105 fn enter(
106 &mut self, _now: Instant,
107 _congestion_event: Option<&BBRv2CongestionEvent>,
108 ) {
109 unreachable!("Enter should never be called for startup")
110 }
111
112 fn leave(
113 &mut self, _now: Instant,
114 _congestion_event: Option<&BBRv2CongestionEvent>,
115 ) {
116 self.model.clear_bandwidth_lo();
118 }
119}
120
121impl Startup {
122 fn into_drain(
123 mut self, now: Instant, congestion_event: Option<&BBRv2CongestionEvent>,
124 ) -> Mode {
125 self.leave(now, congestion_event);
126 let mut next_mode = Mode::drain(self.model);
127 next_mode.enter(now, congestion_event);
128 next_mode
129 }
130
131 fn check_excessive_losses(
132 &mut self, congestion_event: &mut BBRv2CongestionEvent,
133 ) {
134 if self.model.full_bandwidth_reached() {
135 return;
136 }
137
138 if self.model.is_inflight_too_high(
140 congestion_event,
141 PARAMS.startup_full_loss_count,
142 ) {
143 let mut new_inflight_hi = self.model.bdp0();
144
145 if PARAMS.startup_loss_exit_use_max_delivered_for_inflight_hi {
146 new_inflight_hi = new_inflight_hi
147 .max(self.model.max_bytes_delivered_in_round());
148 }
149
150 self.model.set_inflight_hi(new_inflight_hi);
151 self.model.set_full_bandwidth_reached();
152 }
153 }
154}