1use 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
50const BTLBW_FILTER_LEN: Duration = Duration::from_secs(10);
53
54const PROBE_RTT_INTERVAL: Duration = Duration::from_secs(10);
57
58const RTPROP_FILTER_LEN: Duration = PROBE_RTT_INTERVAL;
60
61const BBR_HIGH_GAIN: f64 = 2.89;
65
66const BBR_MIN_PIPE_CWND_PKTS: usize = 4;
68
69const BBR_GAIN_CYCLE_LEN: usize = 8;
71
72const PROBE_RTT_DURATION: Duration = Duration::from_millis(200);
75
76const 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
80const BTLBW_GROWTH_TARGET: f64 = 1.25;
82
83#[derive(Debug, PartialEq, Eq)]
85enum BBRStateMachine {
86 Startup,
87 Drain,
88 ProbeBW,
89 ProbeRTT,
90}
91
92pub struct State {
94 state: BBRStateMachine,
96
97 pacing_rate: u64,
100
101 btlbw: u64,
104
105 btlbwfilter: Minmax<u64>,
107
108 rtprop: Duration,
111
112 rtprop_stamp: Instant,
114
115 rtprop_expired: bool,
119
120 pacing_gain: f64,
123
124 cwnd_gain: f64,
127
128 filled_pipe: bool,
131
132 round_count: u64,
134
135 round_start: bool,
138
139 next_round_delivered: usize,
141
142 probe_rtt_done_stamp: Option<Instant>,
144
145 probe_rtt_round_done: bool,
147
148 packet_conservation: bool,
150
151 prior_cwnd: usize,
153
154 idle_restart: bool,
156
157 full_bw: u64,
159
160 full_bw_count: usize,
162
163 cycle_stamp: Instant,
165
166 cycle_index: usize,
168
169 target_cwnd: usize,
171
172 in_recovery: bool,
174
175 start_time: Instant,
177
178 newly_lost_bytes: usize,
180
181 newly_acked_bytes: usize,
183
184 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
252fn 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 r.bbr_state.next_round_delivered = r.delivery_rate.delivered();
266}
267
268fn 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
278fn 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 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 if !r.in_congestion_recovery(largest_lost_pkt.time_sent) {
325 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 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 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 let mut acked = ranges::RangeSet::default();
457 acked.insert(4..5);
458
459 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 assert_eq!(r.cwnd(), mss * 4);
474 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 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 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 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 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 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 assert!(r.congestion.bbr_state.filled_pipe);
660 assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::ProbeBW);
661
662 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 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 assert_eq!(r.congestion.bbr_state.state, BBRStateMachine::ProbeBW);
731
732 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 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;