blob: 605190d879315d895a9d19cfdfe2a1b0bef1289e [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.
stefan@webrtc.orgedeea912014-12-08 19:46:23 +000030 virtual void OnReceivedEstimatedBitrate(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.org44caf012014-03-26 21:00:21 +000081 Clock* clock,
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +000082 bool enforce_min_bitrate) {
andresp@webrtc.org44caf012014-03-26 21:00:21 +000083 return new BitrateControllerImpl(clock, enforce_min_bitrate);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +000084}
85
sprang@webrtc.org9b791972014-12-18 11:53:59 +000086BitrateControllerImpl::BitrateControllerImpl(Clock* clock,
87 bool enforce_min_bitrate)
andresp@webrtc.org44caf012014-03-26 21:00:21 +000088 : clock_(clock),
89 last_bitrate_update_ms_(clock_->TimeInMilliseconds()),
90 critsect_(CriticalSectionWrapper::CreateCriticalSection()),
solenberg@webrtc.org4e656022014-03-26 14:32:47 +000091 bandwidth_estimation_(),
92 bitrate_observers_(),
93 enforce_min_bitrate_(enforce_min_bitrate),
andresp@webrtc.org44caf012014-03-26 21:00:21 +000094 reserved_bitrate_bps_(0),
95 last_bitrate_bps_(0),
solenberg@webrtc.org4e656022014-03-26 14:32:47 +000096 last_fraction_loss_(0),
andresp@webrtc.org44caf012014-03-26 21:00:21 +000097 last_rtt_ms_(0),
solenberg@webrtc.org4e656022014-03-26 14:32:47 +000098 last_enforce_min_bitrate_(!enforce_min_bitrate_),
99 bitrate_observers_modified_(false),
sprang@webrtc.org9b791972014-12-18 11:53:59 +0000100 last_reserved_bitrate_bps_(0),
101 remb_suppressor_(new RembSuppressor(clock)) {
102}
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000103
104BitrateControllerImpl::~BitrateControllerImpl() {
solenberg@webrtc.org4e656022014-03-26 14:32:47 +0000105 BitrateObserverConfList::iterator it = bitrate_observers_.begin();
pwestin@webrtc.orge9727cd2012-05-03 11:32:25 +0000106 while (it != bitrate_observers_.end()) {
107 delete it->second;
108 bitrate_observers_.erase(it);
109 it = bitrate_observers_.begin();
110 }
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000111 delete critsect_;
112}
113
114RtcpBandwidthObserver* BitrateControllerImpl::CreateRtcpBandwidthObserver() {
115 return new RtcpBandwidthObserverImpl(this);
116}
117
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000118BitrateControllerImpl::BitrateObserverConfList::iterator
119BitrateControllerImpl::FindObserverConfigurationPair(const BitrateObserver*
120 observer) {
121 BitrateObserverConfList::iterator it = bitrate_observers_.begin();
122 for (; it != bitrate_observers_.end(); ++it) {
123 if (it->first == observer) {
124 return it;
125 }
126 }
127 return bitrate_observers_.end();
128}
129
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000130void BitrateControllerImpl::SetBitrateObserver(
131 BitrateObserver* observer,
stefan@webrtc.orgedeea912014-12-08 19:46:23 +0000132 uint32_t start_bitrate,
133 uint32_t min_bitrate,
134 uint32_t max_bitrate) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000135 CriticalSectionScoped cs(critsect_);
136
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000137 BitrateObserverConfList::iterator it = FindObserverConfigurationPair(
138 observer);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000139
140 if (it != bitrate_observers_.end()) {
141 // Update current configuration.
142 it->second->start_bitrate_ = start_bitrate;
143 it->second->min_bitrate_ = min_bitrate;
144 it->second->max_bitrate_ = max_bitrate;
stefan@webrtc.org077593b2014-06-19 12:13:00 +0000145 // Set the send-side bandwidth to the max of the sum of start bitrates and
146 // the current estimate, so that if the user wants to immediately use more
147 // bandwidth, that can be enforced.
148 uint32_t sum_start_bitrate = 0;
149 BitrateObserverConfList::iterator it;
150 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end();
151 ++it) {
152 sum_start_bitrate += it->second->start_bitrate_;
153 }
154 uint32_t current_estimate;
155 uint8_t loss;
156 uint32_t rtt;
157 bandwidth_estimation_.CurrentEstimate(&current_estimate, &loss, &rtt);
158 bandwidth_estimation_.SetSendBitrate(std::max(sum_start_bitrate,
159 current_estimate));
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000160 } else {
161 // Add new settings.
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000162 bitrate_observers_.push_back(BitrateObserverConfiguration(observer,
163 new BitrateConfiguration(start_bitrate, min_bitrate, max_bitrate)));
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000164 bitrate_observers_modified_ = true;
165
166 // TODO(andresp): This is a ugly way to set start bitrate.
167 //
168 // Only change start bitrate if we have exactly one observer. By definition
169 // you can only have one start bitrate, once we have our first estimate we
170 // will adapt from there.
171 if (bitrate_observers_.size() == 1) {
172 bandwidth_estimation_.SetSendBitrate(start_bitrate);
173 }
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000174 }
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000175
176 UpdateMinMaxBitrate();
177}
178
179void BitrateControllerImpl::UpdateMinMaxBitrate() {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000180 uint32_t sum_min_bitrate = 0;
181 uint32_t sum_max_bitrate = 0;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000182 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000183 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000184 sum_min_bitrate += it->second->min_bitrate_;
185 sum_max_bitrate += it->second->max_bitrate_;
186 }
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000187 if (sum_max_bitrate == 0) {
188 // No max configured use 1Gbit/s.
189 sum_max_bitrate = 1000000000;
190 }
191 if (enforce_min_bitrate_ == false) {
192 // If not enforcing min bitrate, allow the bandwidth estimation to
193 // go as low as 10 kbps.
194 sum_min_bitrate = std::min(sum_min_bitrate, 10000u);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000195 }
196 bandwidth_estimation_.SetMinMaxBitrate(sum_min_bitrate,
197 sum_max_bitrate);
198}
199
200void BitrateControllerImpl::RemoveBitrateObserver(BitrateObserver* observer) {
201 CriticalSectionScoped cs(critsect_);
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000202 BitrateObserverConfList::iterator it = FindObserverConfigurationPair(
203 observer);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000204 if (it != bitrate_observers_.end()) {
205 delete it->second;
206 bitrate_observers_.erase(it);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000207 bitrate_observers_modified_ = true;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000208 }
209}
210
henrik.lundin@webrtc.orgb56d0e32013-10-24 09:24:06 +0000211void BitrateControllerImpl::EnforceMinBitrate(bool enforce_min_bitrate) {
212 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000213 enforce_min_bitrate_ = enforce_min_bitrate;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000214 UpdateMinMaxBitrate();
henrik.lundin@webrtc.org845862f2014-03-06 07:19:28 +0000215}
216
solenberg@webrtc.org4e656022014-03-26 14:32:47 +0000217void BitrateControllerImpl::SetReservedBitrate(uint32_t reserved_bitrate_bps) {
218 CriticalSectionScoped cs(critsect_);
219 reserved_bitrate_bps_ = reserved_bitrate_bps;
220 MaybeTriggerOnNetworkChanged();
221}
222
stefan@webrtc.orgedeea912014-12-08 19:46:23 +0000223void BitrateControllerImpl::OnReceivedEstimatedBitrate(uint32_t bitrate) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000224 CriticalSectionScoped cs(critsect_);
sprang@webrtc.org9b791972014-12-18 11:53:59 +0000225 if (remb_suppressor_->SuppresNewRemb(bitrate)) {
226 return;
227 }
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000228 bandwidth_estimation_.UpdateReceiverEstimate(bitrate);
229 MaybeTriggerOnNetworkChanged();
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000230}
231
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +0000232int64_t BitrateControllerImpl::TimeUntilNextProcess() {
233 const int64_t kBitrateControllerUpdateIntervalMs = 25;
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000234 CriticalSectionScoped cs(critsect_);
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +0000235 int64_t time_since_update_ms =
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000236 clock_->TimeInMilliseconds() - last_bitrate_update_ms_;
pkasting@chromium.org0b1534c2014-12-15 22:09:40 +0000237 return std::max<int64_t>(
238 kBitrateControllerUpdateIntervalMs - time_since_update_ms, 0);
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000239}
240
241int32_t BitrateControllerImpl::Process() {
242 if (TimeUntilNextProcess() > 0)
243 return 0;
244 {
245 CriticalSectionScoped cs(critsect_);
246 bandwidth_estimation_.UpdateEstimate(clock_->TimeInMilliseconds());
247 MaybeTriggerOnNetworkChanged();
248 }
249 last_bitrate_update_ms_ = clock_->TimeInMilliseconds();
250 return 0;
251}
252
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000253void BitrateControllerImpl::OnReceivedRtcpReceiverReport(
stefan@webrtc.orgedeea912014-12-08 19:46:23 +0000254 uint8_t fraction_loss,
255 uint32_t rtt,
256 int number_of_packets,
257 int64_t now_ms) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000258 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000259 bandwidth_estimation_.UpdateReceiverBlock(
260 fraction_loss, rtt, number_of_packets, now_ms);
261 MaybeTriggerOnNetworkChanged();
262}
263
264void BitrateControllerImpl::MaybeTriggerOnNetworkChanged() {
265 uint32_t bitrate;
266 uint8_t fraction_loss;
267 uint32_t rtt;
268 bandwidth_estimation_.CurrentEstimate(&bitrate, &fraction_loss, &rtt);
solenberg@webrtc.org4e656022014-03-26 14:32:47 +0000269 bitrate -= std::min(bitrate, reserved_bitrate_bps_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000270
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000271 if (bitrate_observers_modified_ ||
272 bitrate != last_bitrate_bps_ ||
273 fraction_loss != last_fraction_loss_ ||
274 rtt != last_rtt_ms_ ||
solenberg@webrtc.org4e656022014-03-26 14:32:47 +0000275 last_enforce_min_bitrate_ != enforce_min_bitrate_ ||
276 last_reserved_bitrate_bps_ != reserved_bitrate_bps_) {
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000277 last_bitrate_bps_ = bitrate;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000278 last_fraction_loss_ = fraction_loss;
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000279 last_rtt_ms_ = rtt;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000280 last_enforce_min_bitrate_ = enforce_min_bitrate_;
andresp@webrtc.org44caf012014-03-26 21:00:21 +0000281 last_reserved_bitrate_bps_ = reserved_bitrate_bps_;
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000282 bitrate_observers_modified_ = false;
283 OnNetworkChanged(bitrate, fraction_loss, rtt);
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000284 }
285}
286
stefan@webrtc.orgedeea912014-12-08 19:46:23 +0000287void BitrateControllerImpl::OnNetworkChanged(uint32_t bitrate,
288 uint8_t fraction_loss,
289 uint32_t rtt) {
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000290 // Sanity check.
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000291 if (bitrate_observers_.empty())
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000292 return;
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000293
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000294 uint32_t sum_min_bitrates = 0;
stefan@webrtc.org1281dc02012-08-13 16:13:09 +0000295 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000296 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
297 sum_min_bitrates += it->second->min_bitrate_;
298 }
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000299 if (bitrate <= sum_min_bitrates)
300 return LowRateAllocation(bitrate, fraction_loss, rtt, sum_min_bitrates);
301 else
302 return NormalRateAllocation(bitrate, fraction_loss, rtt, sum_min_bitrates);
303}
304
305void BitrateControllerImpl::NormalRateAllocation(uint32_t bitrate,
306 uint8_t fraction_loss,
307 uint32_t rtt,
308 uint32_t sum_min_bitrates) {
309 uint32_t number_of_observers = bitrate_observers_.size();
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000310 uint32_t bitrate_per_observer = (bitrate - sum_min_bitrates) /
311 number_of_observers;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000312 // Use map to sort list based on max bitrate.
313 ObserverSortingMap list_max_bitrates;
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000314 BitrateObserverConfList::iterator it;
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000315 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end(); ++it) {
316 list_max_bitrates.insert(std::pair<uint32_t, ObserverConfiguration*>(
317 it->second->max_bitrate_,
318 new ObserverConfiguration(it->first, it->second->min_bitrate_)));
319 }
320 ObserverSortingMap::iterator max_it = list_max_bitrates.begin();
321 while (max_it != list_max_bitrates.end()) {
322 number_of_observers--;
323 uint32_t observer_allowance = max_it->second->min_bitrate_ +
324 bitrate_per_observer;
325 if (max_it->first < observer_allowance) {
326 // We have more than enough for this observer.
327 // Carry the remainder forward.
328 uint32_t remainder = observer_allowance - max_it->first;
329 if (number_of_observers != 0) {
330 bitrate_per_observer += remainder / number_of_observers;
331 }
332 max_it->second->observer_->OnNetworkChanged(max_it->first, fraction_loss,
333 rtt);
334 } else {
335 max_it->second->observer_->OnNetworkChanged(observer_allowance,
336 fraction_loss, rtt);
337 }
338 delete max_it->second;
339 list_max_bitrates.erase(max_it);
340 // Prepare next iteration.
341 max_it = list_max_bitrates.begin();
342 }
343}
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000344
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000345void BitrateControllerImpl::LowRateAllocation(uint32_t bitrate,
346 uint8_t fraction_loss,
347 uint32_t rtt,
348 uint32_t sum_min_bitrates) {
349 if (enforce_min_bitrate_) {
350 // Min bitrate to all observers.
351 BitrateControllerImpl::BitrateObserverConfList::iterator it;
352 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end();
353 ++it) {
354 it->first->OnNetworkChanged(it->second->min_bitrate_, fraction_loss, rtt);
355 }
356 // Set sum of min to current send bitrate.
357 bandwidth_estimation_.SetSendBitrate(sum_min_bitrates);
358 } else {
359 // Allocate up to |min_bitrate_| to one observer at a time, until
360 // |bitrate| is depleted.
361 uint32_t remainder = bitrate;
362 BitrateControllerImpl::BitrateObserverConfList::iterator it;
363 for (it = bitrate_observers_.begin(); it != bitrate_observers_.end();
364 ++it) {
365 uint32_t allocation = std::min(remainder, it->second->min_bitrate_);
366 it->first->OnNetworkChanged(allocation, fraction_loss, rtt);
367 remainder -= allocation;
368 }
369 // Set |bitrate| to current send bitrate.
370 bandwidth_estimation_.SetSendBitrate(bitrate);
371 }
372}
373
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000374bool BitrateControllerImpl::AvailableBandwidth(uint32_t* bandwidth) const {
andresp@webrtc.org16b75c22014-03-21 14:00:51 +0000375 CriticalSectionScoped cs(critsect_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000376 uint32_t bitrate;
377 uint8_t fraction_loss;
378 uint32_t rtt;
379 bandwidth_estimation_.CurrentEstimate(&bitrate, &fraction_loss, &rtt);
380 if (bitrate) {
solenberg@webrtc.org4e656022014-03-26 14:32:47 +0000381 *bandwidth = bitrate - std::min(bitrate, reserved_bitrate_bps_);
andresp@webrtc.org07bc7342014-03-21 16:51:01 +0000382 return true;
383 }
384 return false;
pwestin@webrtc.orga2cd7322012-04-23 08:32:47 +0000385}
henrik.lundin@webrtc.org29dd0de2013-10-21 14:00:01 +0000386
sprang@webrtc.org9b791972014-12-18 11:53:59 +0000387void BitrateControllerImpl::SetBitrateSent(uint32_t bitrate_sent_bps) {
388 CriticalSectionScoped cs(critsect_);
389 remb_suppressor_->SetBitrateSent(bitrate_sent_bps);
390}
391
392void BitrateControllerImpl::SetCodecMode(webrtc::VideoCodecMode mode) {
393 CriticalSectionScoped cs(critsect_);
394 remb_suppressor_->SetEnabled(mode == kScreensharing);
395}
396
pwestin@webrtc.org1cd11622012-04-19 12:13:52 +0000397} // namespace webrtc