blob: e31aafb77d7cc2bb0487d1e93f09ad7e01736512 [file] [log] [blame]
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +00001/*
2 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 *
10 */
11
pbos@webrtc.org2e10b8e2013-07-16 12:54:53 +000012#include "webrtc/modules/bitrate_controller/bitrate_controller_impl.h"
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000013
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +000014#include <algorithm>
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000015#include <utility>
16
pbos@webrtc.org2e10b8e2013-07-16 12:54:53 +000017#include "webrtc/modules/rtp_rtcp/interface/rtp_rtcp_defines.h"
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000018
19namespace webrtc {
20
andresp@webrtc.org16b75c22014-03-21 14:00:51 +000021class BitrateControllerImpl::RtcpBandwidthObserverImpl
22 : public RtcpBandwidthObserver {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000023 public:
24 explicit RtcpBandwidthObserverImpl(BitrateControllerImpl* owner)
25 : owner_(owner) {
26 }
27 virtual ~RtcpBandwidthObserverImpl() {
28 }
29 // Received RTCP REMB or TMMBR.
pbos@webrtc.org4fac8a42013-07-31 15:16:20 +000030 virtual void OnReceivedEstimatedBitrate(const uint32_t bitrate) OVERRIDE {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000031 owner_->OnReceivedEstimatedBitrate(bitrate);
32 }
33 // Received RTCP receiver block.
34 virtual void OnReceivedRtcpReceiverReport(
stefan@webrtc.org28a331e2013-09-17 07:49:56 +000035 const ReportBlockList& report_blocks,
36 uint16_t rtt,
37 int64_t now_ms) OVERRIDE {
38 if (report_blocks.empty())
39 return;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000040
stefan@webrtc.org28a331e2013-09-17 07:49:56 +000041 int fraction_lost_aggregate = 0;
42 int total_number_of_packets = 0;
43
44 // Compute the a weighted average of the fraction loss from all report
45 // blocks.
46 for (ReportBlockList::const_iterator it = report_blocks.begin();
47 it != report_blocks.end(); ++it) {
48 std::map<uint32_t, uint32_t>::iterator seq_num_it =
49 ssrc_to_last_received_extended_high_seq_num_.find(it->sourceSSRC);
50
51 int number_of_packets = 0;
52 if (seq_num_it != ssrc_to_last_received_extended_high_seq_num_.end())
53 number_of_packets = it->extendedHighSeqNum -
54 seq_num_it->second;
55
56 fraction_lost_aggregate += number_of_packets * it->fractionLost;
57 total_number_of_packets += number_of_packets;
58
59 // Update last received for this SSRC.
60 ssrc_to_last_received_extended_high_seq_num_[it->sourceSSRC] =
61 it->extendedHighSeqNum;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000062 }
stefan@webrtc.org28a331e2013-09-17 07:49:56 +000063 if (total_number_of_packets == 0)
64 fraction_lost_aggregate = 0;
65 else
66 fraction_lost_aggregate = (fraction_lost_aggregate +
67 total_number_of_packets / 2) / total_number_of_packets;
68 if (fraction_lost_aggregate > 255)
69 return;
70
71 owner_->OnReceivedRtcpReceiverReport(fraction_lost_aggregate, rtt,
72 total_number_of_packets, now_ms);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000073 }
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +000074
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000075 private:
76 std::map<uint32_t, uint32_t> ssrc_to_last_received_extended_high_seq_num_;
77 BitrateControllerImpl* owner_;
78};
79
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +000080BitrateController* BitrateController::CreateBitrateController(
andresp@webrtc.orgda077372014-03-25 12:48:42 +000081 Clock* clock,
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +000082 bool enforce_min_bitrate) {
andresp@webrtc.orgda077372014-03-25 12:48:42 +000083 return new BitrateControllerImpl(clock, enforce_min_bitrate);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000084}
85
andresp@webrtc.orgda077372014-03-25 12:48:42 +000086BitrateControllerImpl::BitrateControllerImpl(Clock* clock,
87 bool enforce_min_bitrate)
88 : clock_(clock),
89 last_bitrate_update_ms_(clock_->TimeInMilliseconds()),
90 critsect_(CriticalSectionWrapper::CreateCriticalSection()),
91 enforce_min_bitrate_(enforce_min_bitrate),
92 last_bitrate_bps_(0),
93 last_fraction_loss_(0),
94 last_rtt_ms_(0),
95 last_enforce_min_bitrate_(enforce_min_bitrate_),
96 bitrate_observers_modified_(false) {}
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000097
98BitrateControllerImpl::~BitrateControllerImpl() {
stefan@webrtc.org1281dc02012-08-13 16:13:09 +000099 BitrateObserverConfList::iterator it =
pwestin@webrtc.orge9727cd2012-05-03 11:32:25 +0000100 bitrate_observers_.begin();
101 while (it != bitrate_observers_.end()) {
102 delete it->second;
103 bitrate_observers_.erase(it);
104 it = bitrate_observers_.begin();
105 }
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000106 delete critsect_;
107}
108
109RtcpBandwidthObserver* BitrateControllerImpl::CreateRtcpBandwidthObserver() {
110 return new RtcpBandwidthObserverImpl(this);
111}
112
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000113BitrateControllerImpl::BitrateObserverConfList::iterator
114BitrateControllerImpl::FindObserverConfigurationPair(const BitrateObserver*
115 observer) {
116 BitrateObserverConfList::iterator it = bitrate_observers_.begin();
117 for (; it != bitrate_observers_.end(); ++it) {
118 if (it->first == observer) {
119 return it;
120 }
121 }
122 return bitrate_observers_.end();
123}
124
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000125void BitrateControllerImpl::SetBitrateObserver(
126 BitrateObserver* observer,
127 const uint32_t start_bitrate,
128 const uint32_t min_bitrate,
129 const uint32_t max_bitrate) {
130 CriticalSectionScoped cs(critsect_);
131
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000132 BitrateObserverConfList::iterator it = FindObserverConfigurationPair(
133 observer);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000134
135 if (it != bitrate_observers_.end()) {
136 // Update current configuration.
137 it->second->start_bitrate_ = start_bitrate;
138 it->second->min_bitrate_ = min_bitrate;
139 it->second->max_bitrate_ = max_bitrate;
140 } else {
141 // Add new settings.
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000142 bitrate_observers_.push_back(BitrateObserverConfiguration(observer,
143 new BitrateConfiguration(start_bitrate, min_bitrate, max_bitrate)));
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000144 bitrate_observers_modified_ = true;
145
146 // TODO(andresp): This is a ugly way to set start bitrate.
147 //
148 // Only change start bitrate if we have exactly one observer. By definition
149 // you can only have one start bitrate, once we have our first estimate we
150 // will adapt from there.
151 if (bitrate_observers_.size() == 1) {
152 bandwidth_estimation_.SetSendBitrate(start_bitrate);
153 }
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000154 }
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000155
156 UpdateMinMaxBitrate();
157}
158
159void BitrateControllerImpl::UpdateMinMaxBitrate() {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000160 uint32_t sum_start_bitrate = 0;
161 uint32_t sum_min_bitrate = 0;
162 uint32_t sum_max_bitrate = 0;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000163 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000164 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
165 sum_start_bitrate += it->second->start_bitrate_;
166 sum_min_bitrate += it->second->min_bitrate_;
167 sum_max_bitrate += it->second->max_bitrate_;
168 }
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000169 if (sum_max_bitrate == 0) {
170 // No max configured use 1Gbit/s.
171 sum_max_bitrate = 1000000000;
172 }
173 if (enforce_min_bitrate_ == false) {
174 // If not enforcing min bitrate, allow the bandwidth estimation to
175 // go as low as 10 kbps.
176 sum_min_bitrate = std::min(sum_min_bitrate, 10000u);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000177 }
178 bandwidth_estimation_.SetMinMaxBitrate(sum_min_bitrate,
179 sum_max_bitrate);
180}
181
182void BitrateControllerImpl::RemoveBitrateObserver(BitrateObserver* observer) {
183 CriticalSectionScoped cs(critsect_);
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000184 BitrateObserverConfList::iterator it = FindObserverConfigurationPair(
185 observer);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000186 if (it != bitrate_observers_.end()) {
187 delete it->second;
188 bitrate_observers_.erase(it);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000189 bitrate_observers_modified_ = true;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000190 }
191}
192
henrik.lundin@webrtc.orgb56d0e32013-10-24 09:24:06 +0000193void BitrateControllerImpl::EnforceMinBitrate(bool enforce_min_bitrate) {
194 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000195 enforce_min_bitrate_ = enforce_min_bitrate;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000196 UpdateMinMaxBitrate();
henrik.lundin@webrtc.org845862f2014-03-06 07:19:28 +0000197}
198
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000199void BitrateControllerImpl::OnReceivedEstimatedBitrate(const uint32_t bitrate) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000200 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000201 bandwidth_estimation_.UpdateReceiverEstimate(bitrate);
202 MaybeTriggerOnNetworkChanged();
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000203}
204
andresp@webrtc.orgda077372014-03-25 12:48:42 +0000205int32_t BitrateControllerImpl::TimeUntilNextProcess() {
206 enum { kBitrateControllerUpdateIntervalMs = 25 };
207 CriticalSectionScoped cs(critsect_);
208 int time_since_update_ms =
209 clock_->TimeInMilliseconds() - last_bitrate_update_ms_;
210 return std::max(0, kBitrateControllerUpdateIntervalMs - time_since_update_ms);
211}
212
213int32_t BitrateControllerImpl::Process() {
214 if (TimeUntilNextProcess() > 0)
215 return 0;
216 {
217 CriticalSectionScoped cs(critsect_);
218 bandwidth_estimation_.UpdateEstimate(clock_->TimeInMilliseconds());
219 MaybeTriggerOnNetworkChanged();
220 }
221 last_bitrate_update_ms_ = clock_->TimeInMilliseconds();
222 return 0;
223}
224
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000225void BitrateControllerImpl::OnReceivedRtcpReceiverReport(
226 const uint8_t fraction_loss,
227 const uint32_t rtt,
228 const int number_of_packets,
229 const uint32_t now_ms) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000230 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000231 bandwidth_estimation_.UpdateReceiverBlock(
232 fraction_loss, rtt, number_of_packets, now_ms);
233 MaybeTriggerOnNetworkChanged();
234}
235
236void BitrateControllerImpl::MaybeTriggerOnNetworkChanged() {
237 uint32_t bitrate;
238 uint8_t fraction_loss;
239 uint32_t rtt;
240 bandwidth_estimation_.CurrentEstimate(&bitrate, &fraction_loss, &rtt);
241
andresp@webrtc.orgda077372014-03-25 12:48:42 +0000242 if (bitrate_observers_modified_ || bitrate != last_bitrate_bps_ ||
243 fraction_loss != last_fraction_loss_ || rtt != last_rtt_ms_ ||
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000244 last_enforce_min_bitrate_ != enforce_min_bitrate_) {
andresp@webrtc.orgda077372014-03-25 12:48:42 +0000245 last_bitrate_bps_ = bitrate;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000246 last_fraction_loss_ = fraction_loss;
andresp@webrtc.orgda077372014-03-25 12:48:42 +0000247 last_rtt_ms_ = rtt;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000248 last_enforce_min_bitrate_ = enforce_min_bitrate_;
249 bitrate_observers_modified_ = false;
250 OnNetworkChanged(bitrate, fraction_loss, rtt);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000251 }
252}
253
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000254void BitrateControllerImpl::OnNetworkChanged(const uint32_t bitrate,
255 const uint8_t fraction_loss,
256 const uint32_t rtt) {
257 // Sanity check.
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000258 if (bitrate_observers_.empty())
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000259 return;
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000260
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000261 uint32_t sum_min_bitrates = 0;
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000262 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000263 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
264 sum_min_bitrates += it->second->min_bitrate_;
265 }
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000266 if (bitrate <= sum_min_bitrates)
267 return LowRateAllocation(bitrate, fraction_loss, rtt, sum_min_bitrates);
268 else
269 return NormalRateAllocation(bitrate, fraction_loss, rtt, sum_min_bitrates);
270}
271
272void BitrateControllerImpl::NormalRateAllocation(uint32_t bitrate,
273 uint8_t fraction_loss,
274 uint32_t rtt,
275 uint32_t sum_min_bitrates) {
276 uint32_t number_of_observers = bitrate_observers_.size();
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000277 uint32_t bitrate_per_observer = (bitrate - sum_min_bitrates) /
278 number_of_observers;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000279 // Use map to sort list based on max bitrate.
280 ObserverSortingMap list_max_bitrates;
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000281 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000282 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
283 list_max_bitrates.insert(std::pair<uint32_t, ObserverConfiguration*>(
284 it->second->max_bitrate_,
285 new ObserverConfiguration(it->first, it->second->min_bitrate_)));
286 }
287 ObserverSortingMap::iterator max_it = list_max_bitrates.begin();
288 while (max_it != list_max_bitrates.end()) {
289 number_of_observers--;
290 uint32_t observer_allowance = max_it->second->min_bitrate_ +
291 bitrate_per_observer;
292 if (max_it->first < observer_allowance) {
293 // We have more than enough for this observer.
294 // Carry the remainder forward.
295 uint32_t remainder = observer_allowance - max_it->first;
296 if (number_of_observers != 0) {
297 bitrate_per_observer += remainder / number_of_observers;
298 }
299 max_it->second->observer_->OnNetworkChanged(max_it->first, fraction_loss,
300 rtt);
301 } else {
302 max_it->second->observer_->OnNetworkChanged(observer_allowance,
303 fraction_loss, rtt);
304 }
305 delete max_it->second;
306 list_max_bitrates.erase(max_it);
307 // Prepare next iteration.
308 max_it = list_max_bitrates.begin();
309 }
310}
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000311
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000312void BitrateControllerImpl::LowRateAllocation(uint32_t bitrate,
313 uint8_t fraction_loss,
314 uint32_t rtt,
315 uint32_t sum_min_bitrates) {
316 if (enforce_min_bitrate_) {
317 // Min bitrate to all observers.
318 BitrateControllerImpl::BitrateObserverConfList::iterator it;
319 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end();
320 ++it) {
321 it->first->OnNetworkChanged(it->second->min_bitrate_, fraction_loss, rtt);
322 }
323 // Set sum of min to current send bitrate.
324 bandwidth_estimation_.SetSendBitrate(sum_min_bitrates);
325 } else {
326 // Allocate up to |min_bitrate_| to one observer at a time, until
327 // |bitrate| is depleted.
328 uint32_t remainder = bitrate;
329 BitrateControllerImpl::BitrateObserverConfList::iterator it;
330 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end();
331 ++it) {
332 uint32_t allocation = std::min(remainder, it->second->min_bitrate_);
333 it->first->OnNetworkChanged(allocation, fraction_loss, rtt);
334 remainder -= allocation;
335 }
336 // Set |bitrate| to current send bitrate.
337 bandwidth_estimation_.SetSendBitrate(bitrate);
338 }
339}
340
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000341bool BitrateControllerImpl::AvailableBandwidth(uint32_t* bandwidth) const {
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000342 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000343 uint32_t bitrate;
344 uint8_t fraction_loss;
345 uint32_t rtt;
346 bandwidth_estimation_.CurrentEstimate(&bitrate, &fraction_loss, &rtt);
347 if (bitrate) {
348 *bandwidth = bitrate;
349 return true;
350 }
351 return false;
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000352}
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +0000353
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000354} // namespace webrtc