quiche/recovery/gcongestion/bbr2/drain.rs
1// Copyright (c) 2015 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// Copyright (C) 2023, Cloudflare, Inc.
6// All rights reserved.
7//
8// Redistribution and use in source and binary forms, with or without
9// modification, are permitted provided that the following conditions are
10// met:
11//
12// * Redistributions of source code must retain the above copyright notice,
13// this list of conditions and the following disclaimer.
14//
15// * Redistributions in binary form must reproduce the above copyright
16// notice, this list of conditions and the following disclaimer in the
17// documentation and/or other materials provided with the distribution.
18//
19// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
20// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
23// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
24// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
25// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
26// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
27// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
28// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
29// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31use std::time::Instant;
32
33use crate::recovery::gcongestion::Acked;
34use crate::recovery::gcongestion::Lost;
35
36use super::mode::Cycle;
37use super::mode::Mode;
38use super::mode::ModeImpl;
39use super::network_model::BBRv2NetworkModel;
40use super::BBRv2CongestionEvent;
41use super::Limits;
42use super::PARAMS;
43
44#[derive(Debug)]
45pub(super) struct Drain {
46 pub(super) model: BBRv2NetworkModel,
47 pub(super) cycle: Cycle,
48}
49
50impl ModeImpl for Drain {
51 fn is_probing_for_bandwidth(&self) -> bool {
52 false
53 }
54
55 fn on_congestion_event(
56 mut self, _prior_in_flight: usize, event_time: Instant,
57 _acked_packets: &[Acked], _lost_packets: &[Lost],
58 congestion_event: &mut BBRv2CongestionEvent,
59 _target_bytes_inflight: usize,
60 ) -> Mode {
61 self.model.set_pacing_gain(PARAMS.drain_pacing_gain);
62 // Only STARTUP can transition to DRAIN, both of them use the same cwnd
63 // gain.
64 self.model.set_cwnd_gain(PARAMS.drain_cwnd_gain);
65
66 let drain_target = self.drain_target();
67 if congestion_event.bytes_in_flight <= drain_target {
68 return self.into_probe_bw(event_time, Some(congestion_event));
69 }
70
71 Mode::Drain(self)
72 }
73
74 fn get_cwnd_limits(&self) -> Limits<usize> {
75 Limits {
76 lo: 0,
77 hi: self.model.inflight_lo(),
78 }
79 }
80
81 fn on_exit_quiescence(
82 self, _now: Instant, _quiescence_start_time: Instant,
83 ) -> Mode {
84 Mode::Drain(self)
85 }
86
87 fn enter(&mut self, _: Instant, _: Option<&BBRv2CongestionEvent>) {}
88
89 fn leave(&mut self, _: Instant, _: Option<&BBRv2CongestionEvent>) {}
90}
91
92impl Drain {
93 fn into_probe_bw(
94 mut self, now: Instant, congestion_event: Option<&BBRv2CongestionEvent>,
95 ) -> Mode {
96 self.leave(now, congestion_event);
97 let mut next_mode = Mode::probe_bw(self.model, self.cycle);
98 next_mode.enter(now, congestion_event);
99 next_mode
100 }
101
102 fn drain_target(&self) -> usize {
103 self.model.bdp0()
104 }
105}