xref: /aosp_15_r20/external/webrtc/call/rtp_bitrate_configurator.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2018 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "call/rtp_bitrate_configurator.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <algorithm>
14*d9f75844SAndroid Build Coastguard Worker 
15*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
16*d9f75844SAndroid Build Coastguard Worker 
17*d9f75844SAndroid Build Coastguard Worker namespace {
18*d9f75844SAndroid Build Coastguard Worker 
19*d9f75844SAndroid Build Coastguard Worker // Returns its smallest positive argument. If neither argument is positive,
20*d9f75844SAndroid Build Coastguard Worker // returns an arbitrary nonpositive value.
MinPositive(int a,int b)21*d9f75844SAndroid Build Coastguard Worker int MinPositive(int a, int b) {
22*d9f75844SAndroid Build Coastguard Worker   if (a <= 0) {
23*d9f75844SAndroid Build Coastguard Worker     return b;
24*d9f75844SAndroid Build Coastguard Worker   }
25*d9f75844SAndroid Build Coastguard Worker   if (b <= 0) {
26*d9f75844SAndroid Build Coastguard Worker     return a;
27*d9f75844SAndroid Build Coastguard Worker   }
28*d9f75844SAndroid Build Coastguard Worker   return std::min(a, b);
29*d9f75844SAndroid Build Coastguard Worker }
30*d9f75844SAndroid Build Coastguard Worker 
31*d9f75844SAndroid Build Coastguard Worker }  // namespace
32*d9f75844SAndroid Build Coastguard Worker 
33*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
RtpBitrateConfigurator(const BitrateConstraints & bitrate_config)34*d9f75844SAndroid Build Coastguard Worker RtpBitrateConfigurator::RtpBitrateConfigurator(
35*d9f75844SAndroid Build Coastguard Worker     const BitrateConstraints& bitrate_config)
36*d9f75844SAndroid Build Coastguard Worker     : bitrate_config_(bitrate_config), base_bitrate_config_(bitrate_config) {
37*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GE(bitrate_config.min_bitrate_bps, 0);
38*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GE(bitrate_config.start_bitrate_bps,
39*d9f75844SAndroid Build Coastguard Worker                 bitrate_config.min_bitrate_bps);
40*d9f75844SAndroid Build Coastguard Worker   if (bitrate_config.max_bitrate_bps != -1) {
41*d9f75844SAndroid Build Coastguard Worker     RTC_DCHECK_GE(bitrate_config.max_bitrate_bps,
42*d9f75844SAndroid Build Coastguard Worker                   bitrate_config.start_bitrate_bps);
43*d9f75844SAndroid Build Coastguard Worker   }
44*d9f75844SAndroid Build Coastguard Worker }
45*d9f75844SAndroid Build Coastguard Worker 
46*d9f75844SAndroid Build Coastguard Worker RtpBitrateConfigurator::~RtpBitrateConfigurator() = default;
47*d9f75844SAndroid Build Coastguard Worker 
GetConfig() const48*d9f75844SAndroid Build Coastguard Worker BitrateConstraints RtpBitrateConfigurator::GetConfig() const {
49*d9f75844SAndroid Build Coastguard Worker   return bitrate_config_;
50*d9f75844SAndroid Build Coastguard Worker }
51*d9f75844SAndroid Build Coastguard Worker 
52*d9f75844SAndroid Build Coastguard Worker absl::optional<BitrateConstraints>
UpdateWithSdpParameters(const BitrateConstraints & bitrate_config)53*d9f75844SAndroid Build Coastguard Worker RtpBitrateConfigurator::UpdateWithSdpParameters(
54*d9f75844SAndroid Build Coastguard Worker     const BitrateConstraints& bitrate_config) {
55*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GE(bitrate_config.min_bitrate_bps, 0);
56*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_NE(bitrate_config.start_bitrate_bps, 0);
57*d9f75844SAndroid Build Coastguard Worker   if (bitrate_config.max_bitrate_bps != -1) {
58*d9f75844SAndroid Build Coastguard Worker     RTC_DCHECK_GT(bitrate_config.max_bitrate_bps, 0);
59*d9f75844SAndroid Build Coastguard Worker   }
60*d9f75844SAndroid Build Coastguard Worker 
61*d9f75844SAndroid Build Coastguard Worker   absl::optional<int> new_start;
62*d9f75844SAndroid Build Coastguard Worker   // Only update the "start" bitrate if it's set, and different from the old
63*d9f75844SAndroid Build Coastguard Worker   // value. In practice, this value comes from the x-google-start-bitrate codec
64*d9f75844SAndroid Build Coastguard Worker   // parameter in SDP, and setting the same remote description twice shouldn't
65*d9f75844SAndroid Build Coastguard Worker   // restart bandwidth estimation.
66*d9f75844SAndroid Build Coastguard Worker   if (bitrate_config.start_bitrate_bps != -1 &&
67*d9f75844SAndroid Build Coastguard Worker       bitrate_config.start_bitrate_bps !=
68*d9f75844SAndroid Build Coastguard Worker           base_bitrate_config_.start_bitrate_bps) {
69*d9f75844SAndroid Build Coastguard Worker     new_start.emplace(bitrate_config.start_bitrate_bps);
70*d9f75844SAndroid Build Coastguard Worker   }
71*d9f75844SAndroid Build Coastguard Worker   base_bitrate_config_ = bitrate_config;
72*d9f75844SAndroid Build Coastguard Worker   return UpdateConstraints(new_start);
73*d9f75844SAndroid Build Coastguard Worker }
74*d9f75844SAndroid Build Coastguard Worker 
75*d9f75844SAndroid Build Coastguard Worker absl::optional<BitrateConstraints>
UpdateWithClientPreferences(const BitrateSettings & bitrate_mask)76*d9f75844SAndroid Build Coastguard Worker RtpBitrateConfigurator::UpdateWithClientPreferences(
77*d9f75844SAndroid Build Coastguard Worker     const BitrateSettings& bitrate_mask) {
78*d9f75844SAndroid Build Coastguard Worker   bitrate_config_mask_ = bitrate_mask;
79*d9f75844SAndroid Build Coastguard Worker   return UpdateConstraints(bitrate_mask.start_bitrate_bps);
80*d9f75844SAndroid Build Coastguard Worker }
81*d9f75844SAndroid Build Coastguard Worker 
82*d9f75844SAndroid Build Coastguard Worker // Relay cap can change only max bitrate.
UpdateWithRelayCap(DataRate cap)83*d9f75844SAndroid Build Coastguard Worker absl::optional<BitrateConstraints> RtpBitrateConfigurator::UpdateWithRelayCap(
84*d9f75844SAndroid Build Coastguard Worker     DataRate cap) {
85*d9f75844SAndroid Build Coastguard Worker   if (cap.IsFinite()) {
86*d9f75844SAndroid Build Coastguard Worker     RTC_DCHECK(!cap.IsZero());
87*d9f75844SAndroid Build Coastguard Worker   }
88*d9f75844SAndroid Build Coastguard Worker   max_bitrate_over_relay_ = cap;
89*d9f75844SAndroid Build Coastguard Worker   return UpdateConstraints(absl::nullopt);
90*d9f75844SAndroid Build Coastguard Worker }
91*d9f75844SAndroid Build Coastguard Worker 
UpdateConstraints(const absl::optional<int> & new_start)92*d9f75844SAndroid Build Coastguard Worker absl::optional<BitrateConstraints> RtpBitrateConfigurator::UpdateConstraints(
93*d9f75844SAndroid Build Coastguard Worker     const absl::optional<int>& new_start) {
94*d9f75844SAndroid Build Coastguard Worker   BitrateConstraints updated;
95*d9f75844SAndroid Build Coastguard Worker   updated.min_bitrate_bps =
96*d9f75844SAndroid Build Coastguard Worker       std::max(bitrate_config_mask_.min_bitrate_bps.value_or(0),
97*d9f75844SAndroid Build Coastguard Worker                base_bitrate_config_.min_bitrate_bps);
98*d9f75844SAndroid Build Coastguard Worker 
99*d9f75844SAndroid Build Coastguard Worker   updated.max_bitrate_bps =
100*d9f75844SAndroid Build Coastguard Worker       MinPositive(bitrate_config_mask_.max_bitrate_bps.value_or(-1),
101*d9f75844SAndroid Build Coastguard Worker                   base_bitrate_config_.max_bitrate_bps);
102*d9f75844SAndroid Build Coastguard Worker   updated.max_bitrate_bps =
103*d9f75844SAndroid Build Coastguard Worker       MinPositive(updated.max_bitrate_bps, max_bitrate_over_relay_.bps_or(-1));
104*d9f75844SAndroid Build Coastguard Worker 
105*d9f75844SAndroid Build Coastguard Worker   // If the combined min ends up greater than the combined max, the max takes
106*d9f75844SAndroid Build Coastguard Worker   // priority.
107*d9f75844SAndroid Build Coastguard Worker   if (updated.max_bitrate_bps != -1 &&
108*d9f75844SAndroid Build Coastguard Worker       updated.min_bitrate_bps > updated.max_bitrate_bps) {
109*d9f75844SAndroid Build Coastguard Worker     updated.min_bitrate_bps = updated.max_bitrate_bps;
110*d9f75844SAndroid Build Coastguard Worker   }
111*d9f75844SAndroid Build Coastguard Worker 
112*d9f75844SAndroid Build Coastguard Worker   // If there is nothing to update (min/max unchanged, no new bandwidth
113*d9f75844SAndroid Build Coastguard Worker   // estimation start value), return early.
114*d9f75844SAndroid Build Coastguard Worker   if (updated.min_bitrate_bps == bitrate_config_.min_bitrate_bps &&
115*d9f75844SAndroid Build Coastguard Worker       updated.max_bitrate_bps == bitrate_config_.max_bitrate_bps &&
116*d9f75844SAndroid Build Coastguard Worker       !new_start) {
117*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
118*d9f75844SAndroid Build Coastguard Worker   }
119*d9f75844SAndroid Build Coastguard Worker 
120*d9f75844SAndroid Build Coastguard Worker   if (new_start) {
121*d9f75844SAndroid Build Coastguard Worker     // Clamp start by min and max.
122*d9f75844SAndroid Build Coastguard Worker     updated.start_bitrate_bps = MinPositive(
123*d9f75844SAndroid Build Coastguard Worker         std::max(*new_start, updated.min_bitrate_bps), updated.max_bitrate_bps);
124*d9f75844SAndroid Build Coastguard Worker   } else {
125*d9f75844SAndroid Build Coastguard Worker     updated.start_bitrate_bps = -1;
126*d9f75844SAndroid Build Coastguard Worker   }
127*d9f75844SAndroid Build Coastguard Worker   BitrateConstraints config_to_return = updated;
128*d9f75844SAndroid Build Coastguard Worker   if (!new_start) {
129*d9f75844SAndroid Build Coastguard Worker     updated.start_bitrate_bps = bitrate_config_.start_bitrate_bps;
130*d9f75844SAndroid Build Coastguard Worker   }
131*d9f75844SAndroid Build Coastguard Worker   bitrate_config_ = updated;
132*d9f75844SAndroid Build Coastguard Worker   return config_to_return;
133*d9f75844SAndroid Build Coastguard Worker }
134*d9f75844SAndroid Build Coastguard Worker 
135*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
136