blob: ad2979cb63c32b4893db68904c24e5ea053575e0 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellander65c7f672016-02-12 00:05:01 -08002 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellander65c7f672016-02-12 00:05:01 -08004 * 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.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
jbauch5869f502017-06-29 12:31:36 -070011#include <algorithm>
12#include <iterator>
kwiberg0eb15ed2015-12-17 03:04:15 -080013#include <utility>
14
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020015#include "pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000016
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "api/call/audio_sink.h"
18#include "media/base/mediaconstants.h"
19#include "media/base/rtputils.h"
20#include "rtc_base/bind.h"
21#include "rtc_base/byteorder.h"
22#include "rtc_base/checks.h"
23#include "rtc_base/copyonwritebuffer.h"
24#include "rtc_base/dscp.h"
25#include "rtc_base/logging.h"
26#include "rtc_base/networkroute.h"
27#include "rtc_base/ptr_util.h"
28#include "rtc_base/trace_event.h"
zhihuang38ede132017-06-15 12:52:32 -070029// Adding 'nogncheck' to disable the gn include headers check to support modular
30// WebRTC build targets.
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020031#include "media/engine/webrtcvoiceengine.h" // nogncheck
32#include "p2p/base/packettransportinternal.h"
33#include "pc/channelmanager.h"
Steve Anton4e70a722017-11-28 14:57:10 -080034#include "pc/rtpmediautils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000035
36namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000037using rtc::Bind;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000038
deadbeef2d110be2016-01-13 12:00:26 -080039namespace {
kwiberg31022942016-03-11 14:18:21 -080040// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080041bool SetRawAudioSink_w(VoiceMediaChannel* channel,
42 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080043 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
44 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080045 return true;
46}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020047
48struct SendPacketMessageData : public rtc::MessageData {
49 rtc::CopyOnWriteBuffer packet;
50 rtc::PacketOptions options;
51};
52
deadbeef2d110be2016-01-13 12:00:26 -080053} // namespace
54
henrike@webrtc.org28e20752013-07-10 00:45:36 +000055enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000056 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020057 MSG_SEND_RTP_PACKET,
58 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000059 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000062 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063};
64
henrike@webrtc.org28e20752013-07-10 00:45:36 +000065static const int kAgcMinus10db = -10;
66
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000067static void SafeSetError(const std::string& message, std::string* error_desc) {
68 if (error_desc) {
69 *error_desc = message;
70 }
71}
72
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000073struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020074 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000075 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020076 : ssrc(in_ssrc), error(in_error) {}
77 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000078 VoiceMediaChannel::Error error;
79};
80
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000081struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020082 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000083 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020084 : ssrc(in_ssrc), error(in_error) {}
85 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000086 VideoMediaChannel::Error error;
87};
88
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000089struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020090 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000091 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020092 : ssrc(in_ssrc), error(in_error) {}
93 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000094 DataMediaChannel::Error error;
95};
96
jbaucheec21bd2016-03-20 06:15:43 -070097static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000098 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -070099 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000100}
101
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700102template <class Codec>
103void RtpParametersFromMediaDescription(
104 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700105 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700106 RtpParameters<Codec>* params) {
107 // TODO(pthatcher): Remove this once we're sure no one will give us
Zhi Huang801b8682017-11-15 11:36:43 -0800108 // a description without codecs. Currently the ORTC implementation is relying
109 // on this.
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700110 if (desc->has_codecs()) {
111 params->codecs = desc->codecs();
112 }
113 // TODO(pthatcher): See if we really need
114 // rtp_header_extensions_set() and remove it if we don't.
115 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700116 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700117 }
deadbeef13871492015-12-09 12:37:51 -0800118 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700119}
120
nisse05103312016-03-16 02:22:50 -0700121template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700122void RtpSendParametersFromMediaDescription(
123 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700124 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700125 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700126 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700127 send_params->max_bandwidth_bps = desc->bandwidth();
128}
129
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200130BaseChannel::BaseChannel(rtc::Thread* worker_thread,
131 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800132 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800133 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700134 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800135 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800136 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200137 : worker_thread_(worker_thread),
138 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800139 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000140 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700141 rtcp_mux_required_(rtcp_mux_required),
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800142 unencrypted_rtp_transport_(
143 rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required)),
deadbeef7af91dd2016-12-13 11:29:11 -0800144 srtp_required_(srtp_required),
Zhi Huang1d88d742017-11-15 15:58:49 -0800145 media_channel_(std::move(media_channel)) {
Steve Anton8699a322017-11-06 15:53:33 -0800146 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800147 rtp_transport_ = unencrypted_rtp_transport_.get();
148 ConnectToRtpTransport();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100149 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000150}
151
152BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800153 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800154 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000155 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000156 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200157 // Eats any outstanding messages or packets.
158 worker_thread_->Clear(&invoker_);
159 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000160 // We must destroy the media channel before the transport channel, otherwise
161 // the media channel may try to send on the dead transport channel. NULLing
162 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800163 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100164 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200165}
166
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800167void BaseChannel::ConnectToRtpTransport() {
168 RTC_DCHECK(rtp_transport_);
169 rtp_transport_->SignalReadyToSend.connect(
170 this, &BaseChannel::OnTransportReadyToSend);
171 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
172 // with a callback interface later so that the demuxer can select which
173 // channel to signal.
174 rtp_transport_->SignalPacketReceived.connect(this,
175 &BaseChannel::OnPacketReceived);
176 rtp_transport_->SignalNetworkRouteChanged.connect(
177 this, &BaseChannel::OnNetworkRouteChanged);
178 rtp_transport_->SignalWritableState.connect(this,
179 &BaseChannel::OnWritableState);
180 rtp_transport_->SignalSentPacket.connect(this,
181 &BaseChannel::SignalSentPacket_n);
182}
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200183
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800184void BaseChannel::DisconnectFromRtpTransport() {
185 RTC_DCHECK(rtp_transport_);
186 rtp_transport_->SignalReadyToSend.disconnect(this);
187 rtp_transport_->SignalPacketReceived.disconnect(this);
188 rtp_transport_->SignalNetworkRouteChanged.disconnect(this);
189 rtp_transport_->SignalWritableState.disconnect(this);
190 rtp_transport_->SignalSentPacket.disconnect(this);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200191}
192
Steve Anton8699a322017-11-06 15:53:33 -0800193void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800194 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800195 rtc::PacketTransportInternal* rtp_packet_transport,
196 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800197 RTC_DCHECK_RUN_ON(worker_thread_);
198 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800199 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport,
200 rtp_packet_transport, rtcp_packet_transport);
201
202 if (rtcp_mux_required_) {
203 rtcp_mux_filter_.SetActive();
204 }
Steve Anton8699a322017-11-06 15:53:33 -0800205 });
206
deadbeeff5346592017-01-24 21:51:21 -0800207 // Both RTP and RTCP channels should be set, we can call SetInterface on
208 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000209 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000210}
211
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800212void BaseChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
213 RTC_DCHECK_RUN_ON(worker_thread_);
214 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
215 SetRtpTransport(rtp_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200216
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800217 if (rtcp_mux_required_) {
218 rtcp_mux_filter_.SetActive();
219 }
220 });
221
222 // Both RTP and RTCP channels should be set, we can call SetInterface on
223 // the media channel and it can set network options.
224 media_channel_->SetInterface(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200225}
226
wu@webrtc.org78187522013-10-07 23:32:02 +0000227void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200228 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000229 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200230 // Packets arrive on the network thread, processing packets calls virtual
231 // functions, so need to stop this process in Deinit that is called in
232 // derived classes destructor.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800233 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
234 FlushRtcpMessages_n();
235
236 if (dtls_srtp_transport_) {
237 dtls_srtp_transport_->SetDtlsTransports(nullptr, nullptr);
238 } else {
239 rtp_transport_->SetRtpPacketTransport(nullptr);
240 rtp_transport_->SetRtcpPacketTransport(nullptr);
241 }
242 // Clear pending read packets/messages.
243 network_thread_->Clear(&invoker_);
244 network_thread_->Clear(this);
245 });
wu@webrtc.org78187522013-10-07 23:32:02 +0000246}
247
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800248void BaseChannel::SetRtpTransport(webrtc::RtpTransportInternal* rtp_transport) {
249 if (!network_thread_->IsCurrent()) {
250 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
251 SetRtpTransport(rtp_transport);
252 return;
253 });
254 }
255
256 RTC_DCHECK(rtp_transport);
257
258 if (rtp_transport_) {
259 DisconnectFromRtpTransport();
260 }
261 rtp_transport_ = rtp_transport;
262 RTC_LOG(LS_INFO) << "Setting the RtpTransport for " << content_name();
263 ConnectToRtpTransport();
264
265 UpdateWritableState_n();
266}
267
zhihuangb2cdd932017-01-19 16:54:25 -0800268void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
269 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800270 network_thread_->Invoke<void>(
271 RTC_FROM_HERE,
272 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
273 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000274}
275
deadbeeff5346592017-01-24 21:51:21 -0800276void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800277 rtc::PacketTransportInternal* rtp_packet_transport,
278 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800279 network_thread_->Invoke<void>(
280 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
281 rtp_packet_transport, rtcp_packet_transport));
282}
zhihuangf5b251b2017-01-12 19:37:48 -0800283
deadbeeff5346592017-01-24 21:51:21 -0800284void BaseChannel::SetTransports_n(
285 DtlsTransportInternal* rtp_dtls_transport,
286 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800287 rtc::PacketTransportInternal* rtp_packet_transport,
288 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800289 RTC_DCHECK(network_thread_->IsCurrent());
290 // Validate some assertions about the input.
291 RTC_DCHECK(rtp_packet_transport);
292 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
293 if (rtp_dtls_transport || rtcp_dtls_transport) {
294 // DTLS/non-DTLS pointers should be to the same object.
295 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
296 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
297 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700298 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800299 } else {
300 // Can't go from DTLS to non-DTLS.
301 RTC_DCHECK(!rtp_dtls_transport_);
302 }
303 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800304 if (rtp_dtls_transport && rtcp_dtls_transport) {
305 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
306 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800307 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800308
309 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
310 // Nothing to do if transport isn't changing.
311 return;
312 }
313
deadbeeff5346592017-01-24 21:51:21 -0800314 std::string debug_name;
315 if (rtp_dtls_transport) {
316 transport_name_ = rtp_dtls_transport->transport_name();
317 debug_name = transport_name_;
318 } else {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800319 debug_name = rtp_packet_transport->transport_name();
deadbeeff5346592017-01-24 21:51:21 -0800320 }
deadbeefac22f702017-01-12 21:59:29 -0800321 // If this BaseChannel doesn't require RTCP mux and we haven't fully
322 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800323 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100324 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
325 << " on " << debug_name << " transport "
326 << rtcp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800327 SetTransport_n(/*rtcp=*/true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000328 }
329
Mirko Bonadei675513b2017-11-09 11:09:25 +0100330 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
331 << debug_name << " transport " << rtp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800332 SetTransport_n(/*rtcp=*/false, rtp_dtls_transport, rtp_packet_transport);
333
334 // Set DtlsTransport/PacketTransport for RTP-level transport.
335 if ((rtp_dtls_transport_ || rtcp_dtls_transport_) && dtls_srtp_transport_) {
336 // When setting the transport with non-null |dtls_srtp_transport_|, we are
337 // using DTLS-SRTP. This could happen for bundling. If the
338 // |dtls_srtp_transport| is null, we cannot tell if it doing DTLS-SRTP or
339 // SDES until the description is set. So don't call |EnableDtlsSrtp_n| here.
340 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport,
341 rtcp_dtls_transport);
342 } else {
343 rtp_transport_->SetRtpPacketTransport(rtp_packet_transport);
344 rtp_transport_->SetRtcpPacketTransport(rtcp_packet_transport);
345 }
guoweis46383312015-12-17 16:45:59 -0800346
deadbeefcbecd352015-09-23 11:50:27 -0700347 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700348 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200349 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000350}
351
deadbeeff5346592017-01-24 21:51:21 -0800352void BaseChannel::SetTransport_n(
353 bool rtcp,
354 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800355 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200356 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800357 if (new_dtls_transport) {
358 RTC_DCHECK(new_dtls_transport == new_packet_transport);
359 }
deadbeeff5346592017-01-24 21:51:21 -0800360 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800361 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700362 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700363 rtcp ? rtp_transport_->rtcp_packet_transport()
364 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800365
deadbeeff5346592017-01-24 21:51:21 -0800366 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700367 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000368 return;
369 }
zhihuangb2cdd932017-01-19 16:54:25 -0800370
deadbeeff5346592017-01-24 21:51:21 -0800371 RTC_DCHECK(old_packet_transport != new_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000372
deadbeeff5346592017-01-24 21:51:21 -0800373 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000374
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800375 // If there's no new transport, we're done.
deadbeeff5346592017-01-24 21:51:21 -0800376 if (!new_packet_transport) {
377 return;
378 }
379
380 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700381 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
382 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800383 << "should never happen.";
384 }
zstein56162b92017-04-24 16:54:35 -0700385
deadbeeff5346592017-01-24 21:51:21 -0800386 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
387 for (const auto& pair : socket_options) {
388 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800389 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000390}
391
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000392bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700393 worker_thread_->Invoke<void>(
394 RTC_FROM_HERE,
395 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
396 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000397 return true;
398}
399
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000400bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700401 return InvokeOnWorker<bool>(RTC_FROM_HERE,
402 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000403}
404
Peter Boström0c4e06b2015-10-07 12:23:21 +0200405bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700406 return InvokeOnWorker<bool>(
407 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000408}
409
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000410bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700411 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700412 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000413}
414
Peter Boström0c4e06b2015-10-07 12:23:21 +0200415bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700416 return InvokeOnWorker<bool>(
417 RTC_FROM_HERE,
418 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000419}
420
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000421bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000422 ContentAction action,
423 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100424 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700425 return InvokeOnWorker<bool>(
426 RTC_FROM_HERE,
427 Bind(&BaseChannel::SetLocalContent_w, this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000428}
429
430bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000431 ContentAction action,
432 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100433 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700434 return InvokeOnWorker<bool>(
435 RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w, this, content,
436 action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000437}
438
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000439void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800440 // We pass in the BaseChannel instead of the rtp_dtls_transport_
441 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000442 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200443 // We pass in the network thread because on that thread connection monitor
444 // will call BaseChannel::GetConnectionStats which must be called on the
445 // network thread.
446 connection_monitor_.reset(
447 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000448 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000449 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000450 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000451}
452
453void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000454 if (connection_monitor_) {
455 connection_monitor_->Stop();
456 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000457 }
458}
459
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000460bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200461 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800462 if (!rtp_dtls_transport_) {
463 return false;
464 }
zhihuangb2cdd932017-01-19 16:54:25 -0800465 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800466}
467
468bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800469 // If this BaseChannel doesn't require RTCP mux and we haven't fully
470 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700471 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000472}
473
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700474bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000475 // Receive data if we are enabled and have local content,
Steve Anton4e70a722017-11-28 14:57:10 -0800476 return enabled() &&
477 webrtc::RtpTransceiverDirectionHasRecv(local_content_direction_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000478}
479
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700480bool BaseChannel::IsReadyToSendMedia_w() const {
481 // Need to access some state updated on the network thread.
482 return network_thread_->Invoke<bool>(
483 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
484}
485
486bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000487 // Send outgoing data if we are enabled, have local and remote content,
488 // and we have had some form of connectivity.
Steve Anton4e70a722017-11-28 14:57:10 -0800489 return enabled() &&
490 webrtc::RtpTransceiverDirectionHasRecv(remote_content_direction_) &&
491 webrtc::RtpTransceiverDirectionHasSend(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700492 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000493}
494
jbaucheec21bd2016-03-20 06:15:43 -0700495bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700496 const rtc::PacketOptions& options) {
497 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000498}
499
jbaucheec21bd2016-03-20 06:15:43 -0700500bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700501 const rtc::PacketOptions& options) {
502 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000503}
504
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000505int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000506 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200507 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700508 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200509}
510
511int BaseChannel::SetOption_n(SocketType type,
512 rtc::Socket::Option opt,
513 int value) {
514 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800515 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000516 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000517 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700518 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700519 socket_options_.push_back(
520 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000521 break;
522 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700523 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700524 rtcp_socket_options_.push_back(
525 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000526 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000527 }
deadbeeff5346592017-01-24 21:51:21 -0800528 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000529}
530
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800531void BaseChannel::OnWritableState(bool writable) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200532 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800533 if (writable) {
534 // This is used to cover the scenario when the DTLS handshake is completed
535 // and DtlsTransport becomes writable before the remote description is set.
536 if (ShouldSetupDtlsSrtp_n()) {
537 EnableDtlsSrtp_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700538 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800539 ChannelWritable_n();
540 } else {
541 ChannelNotWritable_n();
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800542 }
543}
544
Zhi Huang942bc2e2017-11-13 13:26:07 -0800545void BaseChannel::OnNetworkRouteChanged(
546 rtc::Optional<rtc::NetworkRoute> network_route) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200547 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800548 rtc::NetworkRoute new_route;
549 if (network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800550 new_route = *(network_route);
Zhi Huang8c316c12017-11-13 21:13:45 +0000551 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800552 // Note: When the RTCP-muxing is not enabled, RTCP transport and RTP transport
553 // use the same transport name and MediaChannel::OnNetworkRouteChanged cannot
554 // work correctly. Intentionally leave it broken to simplify the code and
555 // encourage the users to stop using non-muxing RTCP.
Steve Anton8699a322017-11-06 15:53:33 -0800556 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800557 media_channel_->OnNetworkRouteChanged(transport_name_, new_route);
Steve Anton8699a322017-11-06 15:53:33 -0800558 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700559}
560
zstein56162b92017-04-24 16:54:35 -0700561void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800562 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
563 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000564}
565
stefanc1aeaf02015-10-15 07:26:07 -0700566bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700567 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700568 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200569 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
570 // If the thread is not our network thread, we will post to our network
571 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000572 // synchronize access to all the pieces of the send path, including
573 // SRTP and the inner workings of the transport channels.
574 // The only downside is that we can't return a proper failure code if
575 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200576 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000577 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200578 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
579 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800580 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700581 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700582 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000583 return true;
584 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200585 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000586
587 // Now that we are on the correct thread, ensure we have a place to send this
588 // packet before doing anything. (We might get RTCP packets that we don't
589 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
590 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700591 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000592 return false;
593 }
594
595 // Protect ourselves against crazy data.
596 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100597 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
598 << RtpRtcpStringLiteral(rtcp)
599 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000600 return false;
601 }
602
Zhi Huangcf990f52017-09-22 12:12:30 -0700603 if (!srtp_active()) {
604 if (srtp_required_) {
605 // The audio/video engines may attempt to send RTCP packets as soon as the
606 // streams are created, so don't treat this as an error for RTCP.
607 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
608 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000609 return false;
610 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700611 // However, there shouldn't be any RTP packets sent before SRTP is set up
612 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100613 RTC_LOG(LS_ERROR)
614 << "Can't send outgoing RTP packet when SRTP is inactive"
615 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700616 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800617 return false;
618 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800619
620 std::string packet_type = rtcp ? "RTCP" : "RTP";
621 RTC_LOG(LS_WARNING) << "Sending an " << packet_type
622 << " packet without encryption.";
623 } else {
624 // Make sure we didn't accidentally send any packets without encryption.
625 RTC_DCHECK(rtp_transport_ == sdes_transport_.get() ||
626 rtp_transport_ == dtls_srtp_transport_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000627 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000628 // Bon voyage.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800629 return rtcp ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
630 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000631}
632
zstein3dcf0e92017-06-01 13:22:42 -0700633bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700634 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000635}
636
zstein3dcf0e92017-06-01 13:22:42 -0700637void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700638 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700639 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000640 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000641 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700642 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000643 }
644
Zhi Huangcf990f52017-09-22 12:12:30 -0700645 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000646 // Our session description indicates that SRTP is required, but we got a
647 // packet before our SRTP filter is active. This means either that
648 // a) we got SRTP packets before we received the SDES keys, in which case
649 // we can't decrypt it anyway, or
650 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800651 // transports, so we haven't yet extracted keys, even if DTLS did
652 // complete on the transport that the packets are being sent on. It's
653 // really good practice to wait for both RTP and RTCP to be good to go
654 // before sending media, to prevent weird failure modes, so it's fine
655 // for us to just eat packets here. This is all sidestepped if RTCP mux
656 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100657 RTC_LOG(LS_WARNING)
658 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
659 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000660 return;
661 }
662
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200663 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700664 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700665 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200666}
667
zstein3dcf0e92017-06-01 13:22:42 -0700668void BaseChannel::ProcessPacket(bool rtcp,
669 const rtc::CopyOnWriteBuffer& packet,
670 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200671 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700672
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200673 // Need to copy variable because OnRtcpReceived/OnPacketReceived
674 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
675 rtc::CopyOnWriteBuffer data(packet);
676 if (rtcp) {
677 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000678 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200679 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000680 }
681}
682
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000683void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700684 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000685 if (enabled_)
686 return;
687
Mirko Bonadei675513b2017-11-09 11:09:25 +0100688 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700690 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000691}
692
693void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700694 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000695 if (!enabled_)
696 return;
697
Mirko Bonadei675513b2017-11-09 11:09:25 +0100698 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000699 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700700 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000701}
702
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200703void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700704 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700705 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700706 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700707 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700708 if (rtp_packet_transport && rtp_packet_transport->writable() &&
709 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200710 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700711 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200712 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700713 }
714}
715
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200716void BaseChannel::ChannelWritable_n() {
717 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800718 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000719 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800720 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000721
Mirko Bonadei675513b2017-11-09 11:09:25 +0100722 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
723 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000725 was_ever_writable_ = true;
726 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700727 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000728}
729
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200730bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800731 // Since DTLS is applied to all transports, checking RTP should be enough.
732 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000733}
734
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200735void BaseChannel::ChannelNotWritable_n() {
736 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000737 if (!writable_)
738 return;
739
Mirko Bonadei675513b2017-11-09 11:09:25 +0100740 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000741 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700742 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000743}
744
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200745bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700746 const MediaContentDescription* content,
747 ContentAction action,
748 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700749 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700750 std::string* error_desc) {
jbauch5869f502017-06-29 12:31:36 -0700751 std::vector<int> encrypted_extension_ids;
752 for (const webrtc::RtpExtension& extension : extensions) {
753 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100754 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
755 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700756 encrypted_extension_ids.push_back(extension.id);
757 }
758 }
759
deadbeef7af91dd2016-12-13 11:29:11 -0800760 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200761 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700762 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
jbauch5869f502017-06-29 12:31:36 -0700763 content, action, src, encrypted_extension_ids,
764 error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200765}
766
767bool BaseChannel::SetRtpTransportParameters_n(
768 const MediaContentDescription* content,
769 ContentAction action,
770 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700771 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200772 std::string* error_desc) {
773 RTC_DCHECK(network_thread_->IsCurrent());
774
jbauch5869f502017-06-29 12:31:36 -0700775 if (!SetSrtp_n(content->cryptos(), action, src, encrypted_extension_ids,
776 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700777 return false;
778 }
779
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200780 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700781 return false;
782 }
783
784 return true;
785}
786
zhihuangb2cdd932017-01-19 16:54:25 -0800787// |dtls| will be set to true if DTLS is active for transport and crypto is
788// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200789bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
790 bool* dtls,
791 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -0800792 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000793 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200794 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000795 return false;
796 }
797 return true;
798}
799
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800800void BaseChannel::EnableSdes_n() {
801 if (sdes_transport_) {
802 return;
Zhi Huangcf990f52017-09-22 12:12:30 -0700803 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800804 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
805 // time.
806 RTC_DCHECK(!dtls_srtp_transport_);
807 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800808 sdes_transport_ = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800809 std::move(unencrypted_rtp_transport_));
Zhi Huangd7455782017-11-30 14:50:52 -0800810#if defined(ENABLE_EXTERNAL_AUTH)
811 sdes_transport_->EnableExternalAuth();
812#endif
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800813 SetRtpTransport(sdes_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800814 RTC_LOG(LS_INFO) << "Wrapping RtpTransport in SrtpTransport.";
815}
816
817void BaseChannel::EnableDtlsSrtp_n() {
818 if (dtls_srtp_transport_) {
819 return;
820 }
821 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
822 // time.
823 RTC_DCHECK(!sdes_transport_);
824 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800825
826 auto srtp_transport = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800827 std::move(unencrypted_rtp_transport_));
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800828#if defined(ENABLE_EXTERNAL_AUTH)
829 srtp_transport->EnableExternalAuth();
830#endif
831 dtls_srtp_transport_ =
832 rtc::MakeUnique<webrtc::DtlsSrtpTransport>(std::move(srtp_transport));
833
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800834 SetRtpTransport(dtls_srtp_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800835 if (cached_send_extension_ids_) {
836 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
837 *cached_send_extension_ids_);
838 }
839 if (cached_recv_extension_ids_) {
840 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
841 *cached_recv_extension_ids_);
842 }
843 // Set the DtlsTransport and the |dtls_srtp_transport_| will handle the DTLS
844 // relate signal internally.
845 RTC_DCHECK(rtp_dtls_transport_);
846 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
847 rtcp_dtls_transport_);
848
849 RTC_LOG(LS_INFO) << "Wrapping SrtpTransport in DtlsSrtpTransport.";
Zhi Huangcf990f52017-09-22 12:12:30 -0700850}
851
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200852bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000853 ContentAction action,
854 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700855 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000856 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -0800857 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000858 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000859 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200860 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000861 if (!ret) {
862 return false;
863 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700864
865 // If SRTP was not required, but we're setting a description that uses SDES,
866 // we need to upgrade to an SrtpTransport.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800867 if (!sdes_transport_ && !dtls && !cryptos.empty()) {
868 EnableSdes_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700869 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800870
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800871 if ((action == CA_ANSWER || action == CA_PRANSWER) && dtls) {
872 EnableDtlsSrtp_n();
873 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800874
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800875 UpdateEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
876
877 if (!dtls) {
878 switch (action) {
879 case CA_OFFER:
Zhi Huangcf990f52017-09-22 12:12:30 -0700880 ret = sdes_negotiator_.SetOffer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800881 break;
882 case CA_PRANSWER:
Zhi Huangcf990f52017-09-22 12:12:30 -0700883 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800884 break;
885 case CA_ANSWER:
Zhi Huangcf990f52017-09-22 12:12:30 -0700886 ret = sdes_negotiator_.SetAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800887 break;
888 default:
889 break;
890 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700891
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800892 // If setting an SDES answer succeeded, apply the negotiated parameters
893 // to the SRTP transport.
894 if ((action == CA_PRANSWER || action == CA_ANSWER) && ret) {
895 if (sdes_negotiator_.send_cipher_suite() &&
896 sdes_negotiator_.recv_cipher_suite()) {
897 RTC_DCHECK(cached_send_extension_ids_);
898 RTC_DCHECK(cached_recv_extension_ids_);
899 ret = sdes_transport_->SetRtpParams(
900 *(sdes_negotiator_.send_cipher_suite()),
901 sdes_negotiator_.send_key().data(),
902 static_cast<int>(sdes_negotiator_.send_key().size()),
903 *(cached_send_extension_ids_),
904 *(sdes_negotiator_.recv_cipher_suite()),
905 sdes_negotiator_.recv_key().data(),
906 static_cast<int>(sdes_negotiator_.recv_key().size()),
907 *(cached_recv_extension_ids_));
908 } else {
909 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
910 if (action == CA_ANSWER && sdes_transport_) {
911 // Explicitly reset the |sdes_transport_| if no crypto param is
912 // provided in the answer. No need to call |ResetParams()| for
913 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
914 sdes_transport_->ResetParams();
915 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700916 }
917 }
918 }
919
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000920 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -0800921 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000922 return false;
923 }
924 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000925}
926
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200927bool BaseChannel::SetRtcpMux_n(bool enable,
928 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000929 ContentSource src,
930 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -0800931 // Provide a more specific error message for the RTCP mux "require" policy
932 // case.
zstein56162b92017-04-24 16:54:35 -0700933 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -0800934 SafeSetError(
935 "rtcpMuxPolicy is 'require', but media description does not "
936 "contain 'a=rtcp-mux'.",
937 error_desc);
938 return false;
939 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000940 bool ret = false;
941 switch (action) {
942 case CA_OFFER:
943 ret = rtcp_mux_filter_.SetOffer(enable, src);
944 break;
945 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -0800946 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700947 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000948 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
949 break;
950 case CA_ANSWER:
951 ret = rtcp_mux_filter_.SetAnswer(enable, src);
952 if (ret && rtcp_mux_filter_.IsActive()) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800953 ActivateRtcpMux();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000954 }
955 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000956 default:
957 break;
958 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000959 if (!ret) {
960 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
961 return false;
962 }
zsteine8ab5432017-07-12 11:48:11 -0700963 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000964 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -0800965 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
966 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000967 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000968 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -0700969 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200970 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000971 }
972 }
973
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000974 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000975}
976
977bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700978 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -0800979 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000980}
981
Peter Boström0c4e06b2015-10-07 12:23:21 +0200982bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700983 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000984 return media_channel()->RemoveRecvStream(ssrc);
985}
986
987bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000988 ContentAction action,
989 std::string* error_desc) {
Zhi Huang801b8682017-11-15 11:36:43 -0800990 if (!(action == CA_OFFER || action == CA_ANSWER || action == CA_PRANSWER))
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000991 return false;
992
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000993 // Check for streams that have been removed.
994 bool ret = true;
995 for (StreamParamsVec::const_iterator it = local_streams_.begin();
996 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000997 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000998 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000999 std::ostringstream desc;
1000 desc << "Failed to remove send stream with ssrc "
1001 << it->first_ssrc() << ".";
1002 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001003 ret = false;
1004 }
1005 }
1006 }
1007 // Check for new streams.
1008 for (StreamParamsVec::const_iterator it = streams.begin();
1009 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001010 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001011 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001012 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001013 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001014 std::ostringstream desc;
1015 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1016 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001017 ret = false;
1018 }
1019 }
1020 }
1021 local_streams_ = streams;
1022 return ret;
1023}
1024
1025bool BaseChannel::UpdateRemoteStreams_w(
1026 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001027 ContentAction action,
1028 std::string* error_desc) {
Zhi Huang801b8682017-11-15 11:36:43 -08001029 if (!(action == CA_OFFER || action == CA_ANSWER || action == CA_PRANSWER))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001030 return false;
1031
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001032 // Check for streams that have been removed.
1033 bool ret = true;
1034 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1035 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001036 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001037 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001038 std::ostringstream desc;
1039 desc << "Failed to remove remote stream with ssrc "
1040 << it->first_ssrc() << ".";
1041 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001042 ret = false;
1043 }
1044 }
1045 }
1046 // Check for new streams.
1047 for (StreamParamsVec::const_iterator it = streams.begin();
1048 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001049 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001050 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001051 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001052 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001053 std::ostringstream desc;
1054 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1055 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001056 ret = false;
1057 }
1058 }
1059 }
1060 remote_streams_ = streams;
1061 return ret;
1062}
1063
jbauch5869f502017-06-29 12:31:36 -07001064RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1065 const RtpHeaderExtensions& extensions) {
1066 if (!rtp_dtls_transport_ ||
1067 !rtp_dtls_transport_->crypto_options()
1068 .enable_encrypted_rtp_header_extensions) {
1069 RtpHeaderExtensions filtered;
1070 auto pred = [](const webrtc::RtpExtension& extension) {
1071 return !extension.encrypt;
1072 };
1073 std::copy_if(extensions.begin(), extensions.end(),
1074 std::back_inserter(filtered), pred);
1075 return filtered;
1076 }
1077
1078 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1079}
1080
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001081void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001082 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001083// Absolute Send Time extension id is used only with external auth,
1084// so do not bother searching for it and making asyncronious call to set
1085// something that is not used.
1086#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001087 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001088 webrtc::RtpExtension::FindHeaderExtensionByUri(
1089 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001090 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001091 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001092 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001093 RTC_FROM_HERE, network_thread_,
1094 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1095 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001096#endif
1097}
1098
1099void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1100 int rtp_abs_sendtime_extn_id) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001101 if (sdes_transport_) {
1102 sdes_transport_->CacheRtpAbsSendTimeHeaderExtension(
Zhi Huangcf990f52017-09-22 12:12:30 -07001103 rtp_abs_sendtime_extn_id);
Zhi Huang2a4d70c2017-11-29 15:41:59 -08001104 } else if (dtls_srtp_transport_) {
1105 dtls_srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1106 rtp_abs_sendtime_extn_id);
Zhi Huangcf990f52017-09-22 12:12:30 -07001107 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001108 RTC_LOG(LS_WARNING)
1109 << "Trying to cache the Absolute Send Time extension id "
1110 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001111 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001112}
1113
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001114void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001115 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001116 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001117 case MSG_SEND_RTP_PACKET:
1118 case MSG_SEND_RTCP_PACKET: {
1119 RTC_DCHECK(network_thread_->IsCurrent());
1120 SendPacketMessageData* data =
1121 static_cast<SendPacketMessageData*>(pmsg->pdata);
1122 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1123 SendPacket(rtcp, &data->packet, data->options);
1124 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001125 break;
1126 }
1127 case MSG_FIRSTPACKETRECEIVED: {
1128 SignalFirstPacketReceived(this);
1129 break;
1130 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001131 }
1132}
1133
zstein3dcf0e92017-06-01 13:22:42 -07001134void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001135 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001136}
1137
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001138void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001139 // Flush all remaining RTCP messages. This should only be called in
1140 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001141 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001142 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001143 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1144 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001145 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1146 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001147 }
1148}
1149
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001150void BaseChannel::SignalSentPacket_n(const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001151 RTC_DCHECK(network_thread_->IsCurrent());
1152 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001153 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001154 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1155}
1156
1157void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1158 RTC_DCHECK(worker_thread_->IsCurrent());
1159 SignalSentPacket(sent_packet);
1160}
1161
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001162void BaseChannel::UpdateEncryptedHeaderExtensionIds(
Zhi Huangc99b6c72017-11-10 16:44:46 -08001163 cricket::ContentSource source,
1164 const std::vector<int>& extension_ids) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001165 if (source == ContentSource::CS_LOCAL) {
1166 cached_recv_extension_ids_ = std::move(extension_ids);
1167 if (dtls_srtp_transport_) {
1168 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
1169 extension_ids);
1170 }
1171 } else {
1172 cached_send_extension_ids_ = std::move(extension_ids);
1173 if (dtls_srtp_transport_) {
1174 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
1175 extension_ids);
1176 }
1177 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001178}
1179
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001180void BaseChannel::ActivateRtcpMux() {
1181 // We permanently activated RTCP muxing; signal that we no longer need
1182 // the RTCP transport.
1183 std::string debug_name =
1184 transport_name_.empty()
1185 ? rtp_transport_->rtp_packet_transport()->transport_name()
1186 : transport_name_;
1187 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1188 << "; no longer need RTCP transport for " << debug_name;
1189 if (rtp_transport_->rtcp_packet_transport()) {
1190 SetTransport_n(/*rtcp=*/true, nullptr, nullptr);
1191 if (dtls_srtp_transport_) {
1192 RTC_DCHECK(rtp_dtls_transport_);
1193 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
1194 /*rtcp_dtls_transport_=*/nullptr);
1195 } else {
1196 rtp_transport_->SetRtcpPacketTransport(nullptr);
1197 }
1198 SignalRtcpMuxFullyActive(transport_name_);
Zhi Huangc99b6c72017-11-10 16:44:46 -08001199 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001200 UpdateWritableState_n();
Zhi Huangc99b6c72017-11-10 16:44:46 -08001201}
1202
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001203VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1204 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001205 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001206 MediaEngineInterface* media_engine,
Steve Anton8699a322017-11-06 15:53:33 -08001207 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001208 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001209 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001210 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001211 : BaseChannel(worker_thread,
1212 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001213 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001214 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001215 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001216 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001217 srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -08001218 media_engine_(media_engine) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001219
1220VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001221 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001222 StopAudioMonitor();
1223 StopMediaMonitor();
1224 // this can't be done in the base class, since it calls a virtual
1225 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001226 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001227}
1228
Peter Boström0c4e06b2015-10-07 12:23:21 +02001229bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001230 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001231 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001232 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001233 return InvokeOnWorker<bool>(
1234 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1235 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001236}
1237
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001238// TODO(juberti): Handle early media the right way. We should get an explicit
1239// ringing message telling us to start playing local ringback, which we cancel
1240// if any early media actually arrives. For now, we do the opposite, which is
1241// to wait 1 second for early media, and start playing local ringback if none
1242// arrives.
1243void VoiceChannel::SetEarlyMedia(bool enable) {
1244 if (enable) {
1245 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001246 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1247 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001248 } else {
1249 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001250 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001251 }
1252}
1253
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001254bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001255 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001256 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001257}
1258
Peter Boström0c4e06b2015-10-07 12:23:21 +02001259bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1260 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001261 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001262 return InvokeOnWorker<bool>(
1263 RTC_FROM_HERE,
1264 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001265}
1266
solenberg4bac9c52015-10-09 02:32:53 -07001267bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001268 return InvokeOnWorker<bool>(
1269 RTC_FROM_HERE,
1270 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001271}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001272
Tommif888bb52015-12-12 01:37:01 +01001273void VoiceChannel::SetRawAudioSink(
1274 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001275 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1276 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001277 // passing. So we invoke to our own little routine that gets a pointer to
1278 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001279 InvokeOnWorker<bool>(RTC_FROM_HERE,
1280 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001281}
1282
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001283webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001284 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001285 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001286}
1287
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001288webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1289 uint32_t ssrc) const {
1290 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001291}
1292
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001293bool VoiceChannel::SetRtpSendParameters(
1294 uint32_t ssrc,
1295 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001296 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001297 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001298 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001299}
1300
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001301bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1302 webrtc::RtpParameters parameters) {
1303 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1304}
1305
1306webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1307 uint32_t ssrc) const {
1308 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001309 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001310 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1311}
1312
1313webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1314 uint32_t ssrc) const {
1315 return media_channel()->GetRtpReceiveParameters(ssrc);
1316}
1317
1318bool VoiceChannel::SetRtpReceiveParameters(
1319 uint32_t ssrc,
1320 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001321 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001322 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001323 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1324}
1325
1326bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1327 webrtc::RtpParameters parameters) {
1328 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001329}
1330
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001331bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001332 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1333 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001334}
1335
hbos8d609f62017-04-10 07:39:05 -07001336std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1337 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001338 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1339}
1340
1341std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1342 RTC_DCHECK(worker_thread()->IsCurrent());
1343 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001344}
1345
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001346void VoiceChannel::StartMediaMonitor(int cms) {
1347 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001348 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001349 media_monitor_->SignalUpdate.connect(
1350 this, &VoiceChannel::OnMediaMonitorUpdate);
1351 media_monitor_->Start(cms);
1352}
1353
1354void VoiceChannel::StopMediaMonitor() {
1355 if (media_monitor_) {
1356 media_monitor_->Stop();
1357 media_monitor_->SignalUpdate.disconnect(this);
1358 media_monitor_.reset();
1359 }
1360}
1361
1362void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001363 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001364 audio_monitor_
1365 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1366 audio_monitor_->Start(cms);
1367}
1368
1369void VoiceChannel::StopAudioMonitor() {
1370 if (audio_monitor_) {
1371 audio_monitor_->Stop();
1372 audio_monitor_.reset();
1373 }
1374}
1375
1376bool VoiceChannel::IsAudioMonitorRunning() const {
1377 return (audio_monitor_.get() != NULL);
1378}
1379
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001380int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001381 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001382}
1383
1384int VoiceChannel::GetOutputLevel_w() {
1385 return media_channel()->GetOutputLevel();
1386}
1387
1388void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1389 media_channel()->GetActiveStreams(actives);
1390}
1391
zstein3dcf0e92017-06-01 13:22:42 -07001392void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001393 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001394 const rtc::PacketTime& packet_time) {
1395 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001396 // Set a flag when we've received an RTP packet. If we're waiting for early
1397 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001398 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001399 received_media_ = true;
1400 }
1401}
1402
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001403void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001404 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001405 invoker_.AsyncInvoke<void>(
1406 RTC_FROM_HERE, worker_thread_,
1407 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001408}
1409
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001410void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001411 // Render incoming data if we're the active call, and we have the local
1412 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001413 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001414 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001415
1416 // Send outgoing data if we're the active call, we have the remote content,
1417 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001418 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001419 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001420
Mirko Bonadei675513b2017-11-09 11:09:25 +01001421 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001422}
1423
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001424bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001425 ContentAction action,
1426 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001427 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001428 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001429 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001430
1431 const AudioContentDescription* audio =
1432 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001433 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001434 if (!audio) {
1435 SafeSetError("Can't find audio content in local description.", error_desc);
1436 return false;
1437 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001438
jbauch5869f502017-06-29 12:31:36 -07001439 RtpHeaderExtensions rtp_header_extensions =
1440 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1441
1442 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1443 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001444 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001445 }
1446
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001447 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001448 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001449 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001450 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001451 error_desc);
1452 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001453 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001454 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001455 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001456 }
1457 last_recv_params_ = recv_params;
1458
1459 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1460 // only give it to the media channel once we have a remote
1461 // description too (without a remote description, we won't be able
1462 // to send them anyway).
1463 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1464 SafeSetError("Failed to set local audio description streams.", error_desc);
1465 return false;
1466 }
1467
1468 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001469 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001470 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001471}
1472
1473bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001474 ContentAction action,
1475 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001476 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001477 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001478 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001479
1480 const AudioContentDescription* audio =
1481 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001482 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001483 if (!audio) {
1484 SafeSetError("Can't find audio content in remote description.", error_desc);
1485 return false;
1486 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001487
jbauch5869f502017-06-29 12:31:36 -07001488 RtpHeaderExtensions rtp_header_extensions =
1489 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1490
1491 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1492 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001493 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494 }
1495
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001496 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001497 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1498 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001499 if (audio->agc_minus_10db()) {
Oskar Sundbom36f8f3e2017-11-16 10:54:27 +01001500 send_params.options.adjust_agc_delta = kAgcMinus10db;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001501 }
skvladdc1c62c2016-03-16 19:07:43 -07001502
1503 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1504 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001505 SafeSetError("Failed to set remote audio description send parameters.",
1506 error_desc);
1507 return false;
1508 }
1509 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001510
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001511 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1512 // and only give it to the media channel once we have a local
1513 // description too (without a local description, we won't be able to
1514 // recv them anyway).
1515 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1516 SafeSetError("Failed to set remote audio description streams.", error_desc);
1517 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001518 }
1519
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001520 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001521 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001522 }
1523
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001524 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001525 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001526 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001527}
1528
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001529void VoiceChannel::HandleEarlyMediaTimeout() {
1530 // This occurs on the main thread, not the worker thread.
1531 if (!received_media_) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001532 RTC_LOG(LS_INFO) << "No early media received before timeout";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001533 SignalEarlyMediaTimeout(this);
1534 }
1535}
1536
Peter Boström0c4e06b2015-10-07 12:23:21 +02001537bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1538 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001539 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001540 if (!enabled()) {
1541 return false;
1542 }
solenberg1d63dd02015-12-02 12:35:09 -08001543 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001544}
1545
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001546void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001547 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001548 case MSG_EARLYMEDIATIMEOUT:
1549 HandleEarlyMediaTimeout();
1550 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001551 case MSG_CHANNEL_ERROR: {
1552 VoiceChannelErrorMessageData* data =
1553 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001554 delete data;
1555 break;
1556 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001557 default:
1558 BaseChannel::OnMessage(pmsg);
1559 break;
1560 }
1561}
1562
1563void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001564 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001565 SignalConnectionMonitor(this, infos);
1566}
1567
1568void VoiceChannel::OnMediaMonitorUpdate(
1569 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001570 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001571 SignalMediaMonitor(this, info);
1572}
1573
1574void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1575 const AudioInfo& info) {
1576 SignalAudioMonitor(this, info);
1577}
1578
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001579VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1580 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001581 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001582 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001583 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001584 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001585 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001586 : BaseChannel(worker_thread,
1587 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001588 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001589 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001590 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001591 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001592 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001593
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001594VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001595 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001596 StopMediaMonitor();
1597 // this can't be done in the base class, since it calls a virtual
1598 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001599
1600 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001601}
1602
nisse08582ff2016-02-04 01:24:52 -08001603bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001604 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001605 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001606 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001607 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001608 return true;
1609}
1610
deadbeef5a4a75a2016-06-02 16:23:38 -07001611bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001612 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001613 bool mute,
1614 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001615 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001616 return InvokeOnWorker<bool>(
1617 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1618 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001619}
1620
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001621webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001622 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001623 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001624}
1625
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001626webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1627 uint32_t ssrc) const {
1628 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001629}
1630
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001631bool VideoChannel::SetRtpSendParameters(
1632 uint32_t ssrc,
1633 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001634 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001635 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001636 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001637}
1638
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001639bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1640 webrtc::RtpParameters parameters) {
1641 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1642}
1643
1644webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1645 uint32_t ssrc) const {
1646 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001647 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001648 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1649}
1650
1651webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1652 uint32_t ssrc) const {
1653 return media_channel()->GetRtpReceiveParameters(ssrc);
1654}
1655
1656bool VideoChannel::SetRtpReceiveParameters(
1657 uint32_t ssrc,
1658 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001659 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001660 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001661 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1662}
1663
1664bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1665 webrtc::RtpParameters parameters) {
1666 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001667}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001668
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001669void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001670 // Send outgoing data if we're the active call, we have the remote content,
1671 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001672 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001673 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001674 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001675 // TODO(gangji): Report error back to server.
1676 }
1677
Mirko Bonadei675513b2017-11-09 11:09:25 +01001678 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001679}
1680
stefanf79ade12017-06-02 06:44:03 -07001681void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1682 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1683 media_channel(), bwe_info));
1684}
1685
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001686bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001687 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1688 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001689}
1690
1691void VideoChannel::StartMediaMonitor(int cms) {
1692 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001693 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001694 media_monitor_->SignalUpdate.connect(
1695 this, &VideoChannel::OnMediaMonitorUpdate);
1696 media_monitor_->Start(cms);
1697}
1698
1699void VideoChannel::StopMediaMonitor() {
1700 if (media_monitor_) {
1701 media_monitor_->Stop();
1702 media_monitor_.reset();
1703 }
1704}
1705
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001706bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001707 ContentAction action,
1708 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001709 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001710 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001711 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001712
1713 const VideoContentDescription* video =
1714 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001715 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001716 if (!video) {
1717 SafeSetError("Can't find video content in local description.", error_desc);
1718 return false;
1719 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001720
jbauch5869f502017-06-29 12:31:36 -07001721 RtpHeaderExtensions rtp_header_extensions =
1722 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1723
1724 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1725 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001726 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001727 }
1728
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001729 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001730 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001731 if (!media_channel()->SetRecvParameters(recv_params)) {
1732 SafeSetError("Failed to set local video description recv parameters.",
1733 error_desc);
1734 return false;
1735 }
1736 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001737 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001738 }
1739 last_recv_params_ = recv_params;
1740
1741 // TODO(pthatcher): Move local streams into VideoSendParameters, and
1742 // only give it to the media channel once we have a remote
1743 // description too (without a remote description, we won't be able
1744 // to send them anyway).
1745 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
1746 SafeSetError("Failed to set local video description streams.", error_desc);
1747 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001748 }
1749
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001750 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001751 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001752 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001753}
1754
1755bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001756 ContentAction action,
1757 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001758 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001759 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001760 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001761
1762 const VideoContentDescription* video =
1763 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001764 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001765 if (!video) {
1766 SafeSetError("Can't find video content in remote description.", error_desc);
1767 return false;
1768 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001769
jbauch5869f502017-06-29 12:31:36 -07001770 RtpHeaderExtensions rtp_header_extensions =
1771 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1772
1773 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1774 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001775 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001776 }
1777
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001778 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001779 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
1780 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001781 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08001782 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001783 }
skvladdc1c62c2016-03-16 19:07:43 -07001784
1785 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1786
1787 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001788 SafeSetError("Failed to set remote video description send parameters.",
1789 error_desc);
1790 return false;
1791 }
1792 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001793
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001794 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
1795 // and only give it to the media channel once we have a local
1796 // description too (without a local description, we won't be able to
1797 // recv them anyway).
1798 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
1799 SafeSetError("Failed to set remote video description streams.", error_desc);
1800 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001801 }
1802
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001803 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001804 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001805 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001806
1807 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001808 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001809 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001810}
1811
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001812void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001813 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001814 case MSG_CHANNEL_ERROR: {
1815 const VideoChannelErrorMessageData* data =
1816 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001817 delete data;
1818 break;
1819 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001820 default:
1821 BaseChannel::OnMessage(pmsg);
1822 break;
1823 }
1824}
1825
1826void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001827 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001828 SignalConnectionMonitor(this, infos);
1829}
1830
1831// TODO(pthatcher): Look into removing duplicate code between
1832// audio, video, and data, perhaps by using templates.
1833void VideoChannel::OnMediaMonitorUpdate(
1834 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001835 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001836 SignalMediaMonitor(this, info);
1837}
1838
deadbeef953c2ce2017-01-09 14:53:41 -08001839RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
1840 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001841 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001842 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08001843 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001844 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001845 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001846 : BaseChannel(worker_thread,
1847 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001848 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001849 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001850 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001851 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001852 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001853
deadbeef953c2ce2017-01-09 14:53:41 -08001854RtpDataChannel::~RtpDataChannel() {
1855 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001856 StopMediaMonitor();
1857 // this can't be done in the base class, since it calls a virtual
1858 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001859
1860 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001861}
1862
Steve Anton8699a322017-11-06 15:53:33 -08001863void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08001864 DtlsTransportInternal* rtp_dtls_transport,
1865 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08001866 rtc::PacketTransportInternal* rtp_packet_transport,
1867 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08001868 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
1869 rtp_packet_transport, rtcp_packet_transport);
1870
deadbeef953c2ce2017-01-09 14:53:41 -08001871 media_channel()->SignalDataReceived.connect(this,
1872 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001873 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001874 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001875}
1876
Zhi Huang2dfc42d2017-12-04 13:38:48 -08001877void RtpDataChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
1878 BaseChannel::Init_w(rtp_transport);
1879 media_channel()->SignalDataReceived.connect(this,
1880 &RtpDataChannel::OnDataReceived);
1881 media_channel()->SignalReadyToSend.connect(
1882 this, &RtpDataChannel::OnDataChannelReadyToSend);
1883}
1884
deadbeef953c2ce2017-01-09 14:53:41 -08001885bool RtpDataChannel::SendData(const SendDataParams& params,
1886 const rtc::CopyOnWriteBuffer& payload,
1887 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07001888 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001889 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
1890 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001891}
1892
deadbeef953c2ce2017-01-09 14:53:41 -08001893bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001894 const DataContentDescription* content,
1895 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001896 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
1897 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08001898 // It's been set before, but doesn't match. That's bad.
1899 if (is_sctp) {
1900 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
1901 error_desc);
1902 return false;
1903 }
1904 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001905}
1906
deadbeef953c2ce2017-01-09 14:53:41 -08001907bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
1908 ContentAction action,
1909 std::string* error_desc) {
1910 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001911 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001912 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001913
1914 const DataContentDescription* data =
1915 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001916 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001917 if (!data) {
1918 SafeSetError("Can't find data content in local description.", error_desc);
1919 return false;
1920 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001921
deadbeef953c2ce2017-01-09 14:53:41 -08001922 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001923 return false;
1924 }
1925
jbauch5869f502017-06-29 12:31:36 -07001926 RtpHeaderExtensions rtp_header_extensions =
1927 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1928
1929 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1930 rtp_header_extensions, error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001931 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001932 }
1933
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001934 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001935 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001936 if (!media_channel()->SetRecvParameters(recv_params)) {
1937 SafeSetError("Failed to set remote data description recv parameters.",
1938 error_desc);
1939 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001940 }
deadbeef953c2ce2017-01-09 14:53:41 -08001941 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001942 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001943 }
1944 last_recv_params_ = recv_params;
1945
1946 // TODO(pthatcher): Move local streams into DataSendParameters, and
1947 // only give it to the media channel once we have a remote
1948 // description too (without a remote description, we won't be able
1949 // to send them anyway).
1950 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
1951 SafeSetError("Failed to set local data description streams.", error_desc);
1952 return false;
1953 }
1954
1955 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001956 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001957 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001958}
1959
deadbeef953c2ce2017-01-09 14:53:41 -08001960bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
1961 ContentAction action,
1962 std::string* error_desc) {
1963 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001964 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001965
1966 const DataContentDescription* data =
1967 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001968 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001969 if (!data) {
1970 SafeSetError("Can't find data content in remote description.", error_desc);
1971 return false;
1972 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001973
Zhi Huang801b8682017-11-15 11:36:43 -08001974 // If the remote data doesn't have codecs, it must be empty, so ignore it.
1975 if (!data->has_codecs()) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001976 return true;
1977 }
1978
deadbeef953c2ce2017-01-09 14:53:41 -08001979 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001980 return false;
1981 }
1982
jbauch5869f502017-06-29 12:31:36 -07001983 RtpHeaderExtensions rtp_header_extensions =
1984 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1985
Mirko Bonadei675513b2017-11-09 11:09:25 +01001986 RTC_LOG(LS_INFO) << "Setting remote data description";
jbauch5869f502017-06-29 12:31:36 -07001987 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1988 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001989 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001990 }
1991
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001992 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001993 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
1994 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001995 if (!media_channel()->SetSendParameters(send_params)) {
1996 SafeSetError("Failed to set remote data description send parameters.",
1997 error_desc);
1998 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001999 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002000 last_send_params_ = send_params;
2001
2002 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2003 // and only give it to the media channel once we have a local
2004 // description too (without a local description, we won't be able to
2005 // recv them anyway).
2006 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2007 SafeSetError("Failed to set remote data description streams.",
2008 error_desc);
2009 return false;
2010 }
2011
2012 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002013 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002014 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002015}
2016
deadbeef953c2ce2017-01-09 14:53:41 -08002017void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002018 // Render incoming data if we're the active call, and we have the local
2019 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002020 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002021 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002022 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002023 }
2024
2025 // Send outgoing data if we're the active call, we have the remote content,
2026 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002027 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002028 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002029 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002030 }
2031
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002032 // Trigger SignalReadyToSendData asynchronously.
2033 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002034
Mirko Bonadei675513b2017-11-09 11:09:25 +01002035 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002036}
2037
deadbeef953c2ce2017-01-09 14:53:41 -08002038void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002039 switch (pmsg->message_id) {
2040 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002041 DataChannelReadyToSendMessageData* data =
2042 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002043 ready_to_send_data_ = data->data();
2044 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002045 delete data;
2046 break;
2047 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002048 case MSG_DATARECEIVED: {
2049 DataReceivedMessageData* data =
2050 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002051 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002052 delete data;
2053 break;
2054 }
2055 case MSG_CHANNEL_ERROR: {
2056 const DataChannelErrorMessageData* data =
2057 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002058 delete data;
2059 break;
2060 }
2061 default:
2062 BaseChannel::OnMessage(pmsg);
2063 break;
2064 }
2065}
2066
deadbeef953c2ce2017-01-09 14:53:41 -08002067void RtpDataChannel::OnConnectionMonitorUpdate(
2068 ConnectionMonitor* monitor,
2069 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002070 SignalConnectionMonitor(this, infos);
2071}
2072
deadbeef953c2ce2017-01-09 14:53:41 -08002073void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002074 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002075 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002076 media_monitor_->SignalUpdate.connect(this,
2077 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002078 media_monitor_->Start(cms);
2079}
2080
deadbeef953c2ce2017-01-09 14:53:41 -08002081void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002082 if (media_monitor_) {
2083 media_monitor_->Stop();
2084 media_monitor_->SignalUpdate.disconnect(this);
2085 media_monitor_.reset();
2086 }
2087}
2088
deadbeef953c2ce2017-01-09 14:53:41 -08002089void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2090 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002091 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002092 SignalMediaMonitor(this, info);
2093}
2094
deadbeef953c2ce2017-01-09 14:53:41 -08002095void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2096 const char* data,
2097 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002098 DataReceivedMessageData* msg = new DataReceivedMessageData(
2099 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002100 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002101}
2102
deadbeef953c2ce2017-01-09 14:53:41 -08002103void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2104 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002105 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2106 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002107 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002108}
2109
deadbeef953c2ce2017-01-09 14:53:41 -08002110void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002111 // This is usded for congestion control to indicate that the stream is ready
2112 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2113 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002114 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002115 new DataChannelReadyToSendMessageData(writable));
2116}
2117
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002118} // namespace cricket