blob: f9b0c33711c21f88773d2d97e8021b9532327460 [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;
Steve Anton3828c062017-12-06 10:34:51 -080038using webrtc::SdpType;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000039
deadbeef2d110be2016-01-13 12:00:26 -080040namespace {
kwiberg31022942016-03-11 14:18:21 -080041// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080042bool SetRawAudioSink_w(VoiceMediaChannel* channel,
43 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080044 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
45 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080046 return true;
47}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020048
49struct SendPacketMessageData : public rtc::MessageData {
50 rtc::CopyOnWriteBuffer packet;
51 rtc::PacketOptions options;
52};
53
deadbeef2d110be2016-01-13 12:00:26 -080054} // namespace
55
henrike@webrtc.org28e20752013-07-10 00:45:36 +000056enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000057 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020058 MSG_SEND_RTP_PACKET,
59 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000062 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000064};
65
henrike@webrtc.org28e20752013-07-10 00:45:36 +000066static const int kAgcMinus10db = -10;
67
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000068static void SafeSetError(const std::string& message, std::string* error_desc) {
69 if (error_desc) {
70 *error_desc = message;
71 }
72}
73
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000074struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020075 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000076 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020077 : ssrc(in_ssrc), error(in_error) {}
78 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000079 VoiceMediaChannel::Error error;
80};
81
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000082struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020083 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000084 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020085 : ssrc(in_ssrc), error(in_error) {}
86 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 VideoMediaChannel::Error error;
88};
89
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000090struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020091 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000092 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020093 : ssrc(in_ssrc), error(in_error) {}
94 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 DataMediaChannel::Error error;
96};
97
jbaucheec21bd2016-03-20 06:15:43 -070098static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000099 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -0700100 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000101}
102
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700103template <class Codec>
104void RtpParametersFromMediaDescription(
105 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700106 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700107 RtpParameters<Codec>* params) {
108 // TODO(pthatcher): Remove this once we're sure no one will give us
Zhi Huang801b8682017-11-15 11:36:43 -0800109 // a description without codecs. Currently the ORTC implementation is relying
110 // on this.
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700111 if (desc->has_codecs()) {
112 params->codecs = desc->codecs();
113 }
114 // TODO(pthatcher): See if we really need
115 // rtp_header_extensions_set() and remove it if we don't.
116 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700117 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700118 }
deadbeef13871492015-12-09 12:37:51 -0800119 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700120}
121
nisse05103312016-03-16 02:22:50 -0700122template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700123void RtpSendParametersFromMediaDescription(
124 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700125 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700126 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700127 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700128 send_params->max_bandwidth_bps = desc->bandwidth();
129}
130
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200131BaseChannel::BaseChannel(rtc::Thread* worker_thread,
132 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800133 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800134 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700135 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800136 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800137 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200138 : worker_thread_(worker_thread),
139 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800140 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000141 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700142 rtcp_mux_required_(rtcp_mux_required),
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800143 unencrypted_rtp_transport_(
144 rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required)),
deadbeef7af91dd2016-12-13 11:29:11 -0800145 srtp_required_(srtp_required),
Zhi Huang1d88d742017-11-15 15:58:49 -0800146 media_channel_(std::move(media_channel)) {
Steve Anton8699a322017-11-06 15:53:33 -0800147 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800148 rtp_transport_ = unencrypted_rtp_transport_.get();
149 ConnectToRtpTransport();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100150 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000151}
152
153BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800154 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800155 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000156 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000157 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200158 // Eats any outstanding messages or packets.
159 worker_thread_->Clear(&invoker_);
160 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000161 // We must destroy the media channel before the transport channel, otherwise
162 // the media channel may try to send on the dead transport channel. NULLing
163 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800164 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100165 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200166}
167
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800168void BaseChannel::ConnectToRtpTransport() {
169 RTC_DCHECK(rtp_transport_);
170 rtp_transport_->SignalReadyToSend.connect(
171 this, &BaseChannel::OnTransportReadyToSend);
172 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
173 // with a callback interface later so that the demuxer can select which
174 // channel to signal.
175 rtp_transport_->SignalPacketReceived.connect(this,
176 &BaseChannel::OnPacketReceived);
177 rtp_transport_->SignalNetworkRouteChanged.connect(
178 this, &BaseChannel::OnNetworkRouteChanged);
179 rtp_transport_->SignalWritableState.connect(this,
180 &BaseChannel::OnWritableState);
181 rtp_transport_->SignalSentPacket.connect(this,
182 &BaseChannel::SignalSentPacket_n);
183}
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200184
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800185void BaseChannel::DisconnectFromRtpTransport() {
186 RTC_DCHECK(rtp_transport_);
187 rtp_transport_->SignalReadyToSend.disconnect(this);
188 rtp_transport_->SignalPacketReceived.disconnect(this);
189 rtp_transport_->SignalNetworkRouteChanged.disconnect(this);
190 rtp_transport_->SignalWritableState.disconnect(this);
191 rtp_transport_->SignalSentPacket.disconnect(this);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200192}
193
Steve Anton8699a322017-11-06 15:53:33 -0800194void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800195 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800196 rtc::PacketTransportInternal* rtp_packet_transport,
197 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800198 RTC_DCHECK_RUN_ON(worker_thread_);
199 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800200 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport,
201 rtp_packet_transport, rtcp_packet_transport);
202
203 if (rtcp_mux_required_) {
204 rtcp_mux_filter_.SetActive();
205 }
Steve Anton8699a322017-11-06 15:53:33 -0800206 });
207
deadbeeff5346592017-01-24 21:51:21 -0800208 // Both RTP and RTCP channels should be set, we can call SetInterface on
209 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000210 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000211}
212
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800213void BaseChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
214 RTC_DCHECK_RUN_ON(worker_thread_);
215 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
216 SetRtpTransport(rtp_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200217
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800218 if (rtcp_mux_required_) {
219 rtcp_mux_filter_.SetActive();
220 }
221 });
222
223 // Both RTP and RTCP channels should be set, we can call SetInterface on
224 // the media channel and it can set network options.
225 media_channel_->SetInterface(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200226}
227
wu@webrtc.org78187522013-10-07 23:32:02 +0000228void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200229 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000230 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200231 // Packets arrive on the network thread, processing packets calls virtual
232 // functions, so need to stop this process in Deinit that is called in
233 // derived classes destructor.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800234 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
235 FlushRtcpMessages_n();
236
237 if (dtls_srtp_transport_) {
238 dtls_srtp_transport_->SetDtlsTransports(nullptr, nullptr);
239 } else {
240 rtp_transport_->SetRtpPacketTransport(nullptr);
241 rtp_transport_->SetRtcpPacketTransport(nullptr);
242 }
243 // Clear pending read packets/messages.
244 network_thread_->Clear(&invoker_);
245 network_thread_->Clear(this);
246 });
wu@webrtc.org78187522013-10-07 23:32:02 +0000247}
248
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800249void BaseChannel::SetRtpTransport(webrtc::RtpTransportInternal* rtp_transport) {
250 if (!network_thread_->IsCurrent()) {
251 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
252 SetRtpTransport(rtp_transport);
253 return;
254 });
255 }
256
257 RTC_DCHECK(rtp_transport);
258
259 if (rtp_transport_) {
260 DisconnectFromRtpTransport();
261 }
262 rtp_transport_ = rtp_transport;
263 RTC_LOG(LS_INFO) << "Setting the RtpTransport for " << content_name();
264 ConnectToRtpTransport();
265
266 UpdateWritableState_n();
267}
268
zhihuangb2cdd932017-01-19 16:54:25 -0800269void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
270 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800271 network_thread_->Invoke<void>(
272 RTC_FROM_HERE,
273 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
274 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000275}
276
deadbeeff5346592017-01-24 21:51:21 -0800277void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800278 rtc::PacketTransportInternal* rtp_packet_transport,
279 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800280 network_thread_->Invoke<void>(
281 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
282 rtp_packet_transport, rtcp_packet_transport));
283}
zhihuangf5b251b2017-01-12 19:37:48 -0800284
deadbeeff5346592017-01-24 21:51:21 -0800285void BaseChannel::SetTransports_n(
286 DtlsTransportInternal* rtp_dtls_transport,
287 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800288 rtc::PacketTransportInternal* rtp_packet_transport,
289 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800290 RTC_DCHECK(network_thread_->IsCurrent());
291 // Validate some assertions about the input.
292 RTC_DCHECK(rtp_packet_transport);
293 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
294 if (rtp_dtls_transport || rtcp_dtls_transport) {
295 // DTLS/non-DTLS pointers should be to the same object.
296 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
297 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
298 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700299 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800300 } else {
301 // Can't go from DTLS to non-DTLS.
302 RTC_DCHECK(!rtp_dtls_transport_);
303 }
304 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800305 if (rtp_dtls_transport && rtcp_dtls_transport) {
306 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
307 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800308 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800309
310 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
311 // Nothing to do if transport isn't changing.
312 return;
313 }
314
deadbeeff5346592017-01-24 21:51:21 -0800315 std::string debug_name;
316 if (rtp_dtls_transport) {
317 transport_name_ = rtp_dtls_transport->transport_name();
318 debug_name = transport_name_;
319 } else {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800320 debug_name = rtp_packet_transport->transport_name();
deadbeeff5346592017-01-24 21:51:21 -0800321 }
deadbeefac22f702017-01-12 21:59:29 -0800322 // If this BaseChannel doesn't require RTCP mux and we haven't fully
323 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800324 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100325 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
326 << " on " << debug_name << " transport "
327 << rtcp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800328 SetTransport_n(/*rtcp=*/true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000329 }
330
Mirko Bonadei675513b2017-11-09 11:09:25 +0100331 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
332 << debug_name << " transport " << rtp_packet_transport;
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800333 SetTransport_n(/*rtcp=*/false, rtp_dtls_transport, rtp_packet_transport);
334
335 // Set DtlsTransport/PacketTransport for RTP-level transport.
336 if ((rtp_dtls_transport_ || rtcp_dtls_transport_) && dtls_srtp_transport_) {
337 // When setting the transport with non-null |dtls_srtp_transport_|, we are
338 // using DTLS-SRTP. This could happen for bundling. If the
339 // |dtls_srtp_transport| is null, we cannot tell if it doing DTLS-SRTP or
340 // SDES until the description is set. So don't call |EnableDtlsSrtp_n| here.
341 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport,
342 rtcp_dtls_transport);
343 } else {
344 rtp_transport_->SetRtpPacketTransport(rtp_packet_transport);
345 rtp_transport_->SetRtcpPacketTransport(rtcp_packet_transport);
346 }
guoweis46383312015-12-17 16:45:59 -0800347
deadbeefcbecd352015-09-23 11:50:27 -0700348 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700349 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200350 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000351}
352
deadbeeff5346592017-01-24 21:51:21 -0800353void BaseChannel::SetTransport_n(
354 bool rtcp,
355 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800356 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200357 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800358 if (new_dtls_transport) {
359 RTC_DCHECK(new_dtls_transport == new_packet_transport);
360 }
deadbeeff5346592017-01-24 21:51:21 -0800361 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800362 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700363 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700364 rtcp ? rtp_transport_->rtcp_packet_transport()
365 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800366
deadbeeff5346592017-01-24 21:51:21 -0800367 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700368 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000369 return;
370 }
zhihuangb2cdd932017-01-19 16:54:25 -0800371
deadbeeff5346592017-01-24 21:51:21 -0800372 RTC_DCHECK(old_packet_transport != new_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000373
deadbeeff5346592017-01-24 21:51:21 -0800374 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000375
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800376 // If there's no new transport, we're done.
deadbeeff5346592017-01-24 21:51:21 -0800377 if (!new_packet_transport) {
378 return;
379 }
380
381 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700382 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
383 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800384 << "should never happen.";
385 }
zstein56162b92017-04-24 16:54:35 -0700386
deadbeeff5346592017-01-24 21:51:21 -0800387 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
388 for (const auto& pair : socket_options) {
389 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800390 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000391}
392
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000393bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700394 worker_thread_->Invoke<void>(
395 RTC_FROM_HERE,
396 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
397 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000398 return true;
399}
400
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000401bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700402 return InvokeOnWorker<bool>(RTC_FROM_HERE,
403 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000404}
405
Peter Boström0c4e06b2015-10-07 12:23:21 +0200406bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700407 return InvokeOnWorker<bool>(
408 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000409}
410
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000411bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700412 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700413 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000414}
415
Peter Boström0c4e06b2015-10-07 12:23:21 +0200416bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700417 return InvokeOnWorker<bool>(
418 RTC_FROM_HERE,
419 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000420}
421
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000422bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800423 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000424 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100425 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700426 return InvokeOnWorker<bool>(
427 RTC_FROM_HERE,
Steve Anton3828c062017-12-06 10:34:51 -0800428 Bind(&BaseChannel::SetLocalContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000429}
430
431bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800432 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000433 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100434 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700435 return InvokeOnWorker<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800436 RTC_FROM_HERE,
437 Bind(&BaseChannel::SetRemoteContent_w, this, content, type, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438}
439
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000440void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800441 // We pass in the BaseChannel instead of the rtp_dtls_transport_
442 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000443 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200444 // We pass in the network thread because on that thread connection monitor
445 // will call BaseChannel::GetConnectionStats which must be called on the
446 // network thread.
447 connection_monitor_.reset(
448 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000449 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000450 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000451 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000452}
453
454void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000455 if (connection_monitor_) {
456 connection_monitor_->Stop();
457 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000458 }
459}
460
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000461bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200462 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800463 if (!rtp_dtls_transport_) {
464 return false;
465 }
zhihuangb2cdd932017-01-19 16:54:25 -0800466 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800467}
468
469bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800470 // If this BaseChannel doesn't require RTCP mux and we haven't fully
471 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700472 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000473}
474
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700475bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000476 // Receive data if we are enabled and have local content,
Steve Anton4e70a722017-11-28 14:57:10 -0800477 return enabled() &&
478 webrtc::RtpTransceiverDirectionHasRecv(local_content_direction_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479}
480
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700481bool BaseChannel::IsReadyToSendMedia_w() const {
482 // Need to access some state updated on the network thread.
483 return network_thread_->Invoke<bool>(
484 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
485}
486
487bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000488 // Send outgoing data if we are enabled, have local and remote content,
489 // and we have had some form of connectivity.
Steve Anton4e70a722017-11-28 14:57:10 -0800490 return enabled() &&
491 webrtc::RtpTransceiverDirectionHasRecv(remote_content_direction_) &&
492 webrtc::RtpTransceiverDirectionHasSend(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700493 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000494}
495
jbaucheec21bd2016-03-20 06:15:43 -0700496bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700497 const rtc::PacketOptions& options) {
498 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000499}
500
jbaucheec21bd2016-03-20 06:15:43 -0700501bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700502 const rtc::PacketOptions& options) {
503 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000504}
505
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000506int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000507 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200508 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700509 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200510}
511
512int BaseChannel::SetOption_n(SocketType type,
513 rtc::Socket::Option opt,
514 int value) {
515 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800516 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000517 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000518 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700519 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700520 socket_options_.push_back(
521 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000522 break;
523 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700524 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700525 rtcp_socket_options_.push_back(
526 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000527 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000528 }
deadbeeff5346592017-01-24 21:51:21 -0800529 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000530}
531
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800532void BaseChannel::OnWritableState(bool writable) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200533 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800534 if (writable) {
535 // This is used to cover the scenario when the DTLS handshake is completed
536 // and DtlsTransport becomes writable before the remote description is set.
537 if (ShouldSetupDtlsSrtp_n()) {
538 EnableDtlsSrtp_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700539 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800540 ChannelWritable_n();
541 } else {
542 ChannelNotWritable_n();
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800543 }
544}
545
Zhi Huang942bc2e2017-11-13 13:26:07 -0800546void BaseChannel::OnNetworkRouteChanged(
547 rtc::Optional<rtc::NetworkRoute> network_route) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200548 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800549 rtc::NetworkRoute new_route;
550 if (network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800551 new_route = *(network_route);
Zhi Huang8c316c12017-11-13 21:13:45 +0000552 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800553 // Note: When the RTCP-muxing is not enabled, RTCP transport and RTP transport
554 // use the same transport name and MediaChannel::OnNetworkRouteChanged cannot
555 // work correctly. Intentionally leave it broken to simplify the code and
556 // encourage the users to stop using non-muxing RTCP.
Steve Anton8699a322017-11-06 15:53:33 -0800557 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800558 media_channel_->OnNetworkRouteChanged(transport_name_, new_route);
Steve Anton8699a322017-11-06 15:53:33 -0800559 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700560}
561
zstein56162b92017-04-24 16:54:35 -0700562void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800563 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
564 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000565}
566
stefanc1aeaf02015-10-15 07:26:07 -0700567bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700568 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700569 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200570 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
571 // If the thread is not our network thread, we will post to our network
572 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000573 // synchronize access to all the pieces of the send path, including
574 // SRTP and the inner workings of the transport channels.
575 // The only downside is that we can't return a proper failure code if
576 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200577 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000578 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200579 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
580 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800581 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700582 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700583 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000584 return true;
585 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200586 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000587
588 // Now that we are on the correct thread, ensure we have a place to send this
589 // packet before doing anything. (We might get RTCP packets that we don't
590 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
591 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700592 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000593 return false;
594 }
595
596 // Protect ourselves against crazy data.
597 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100598 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
599 << RtpRtcpStringLiteral(rtcp)
600 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000601 return false;
602 }
603
Zhi Huangcf990f52017-09-22 12:12:30 -0700604 if (!srtp_active()) {
605 if (srtp_required_) {
606 // The audio/video engines may attempt to send RTCP packets as soon as the
607 // streams are created, so don't treat this as an error for RTCP.
608 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
609 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000610 return false;
611 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700612 // However, there shouldn't be any RTP packets sent before SRTP is set up
613 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100614 RTC_LOG(LS_ERROR)
615 << "Can't send outgoing RTP packet when SRTP is inactive"
616 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700617 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800618 return false;
619 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800620
621 std::string packet_type = rtcp ? "RTCP" : "RTP";
622 RTC_LOG(LS_WARNING) << "Sending an " << packet_type
623 << " packet without encryption.";
624 } else {
625 // Make sure we didn't accidentally send any packets without encryption.
626 RTC_DCHECK(rtp_transport_ == sdes_transport_.get() ||
627 rtp_transport_ == dtls_srtp_transport_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000628 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000629 // Bon voyage.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800630 return rtcp ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
631 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000632}
633
zstein3dcf0e92017-06-01 13:22:42 -0700634bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700635 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000636}
637
zstein3dcf0e92017-06-01 13:22:42 -0700638void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700639 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700640 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000641 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000642 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700643 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000644 }
645
Zhi Huangcf990f52017-09-22 12:12:30 -0700646 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000647 // Our session description indicates that SRTP is required, but we got a
648 // packet before our SRTP filter is active. This means either that
649 // a) we got SRTP packets before we received the SDES keys, in which case
650 // we can't decrypt it anyway, or
651 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800652 // transports, so we haven't yet extracted keys, even if DTLS did
653 // complete on the transport that the packets are being sent on. It's
654 // really good practice to wait for both RTP and RTCP to be good to go
655 // before sending media, to prevent weird failure modes, so it's fine
656 // for us to just eat packets here. This is all sidestepped if RTCP mux
657 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100658 RTC_LOG(LS_WARNING)
659 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
660 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000661 return;
662 }
663
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200664 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700665 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700666 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200667}
668
zstein3dcf0e92017-06-01 13:22:42 -0700669void BaseChannel::ProcessPacket(bool rtcp,
670 const rtc::CopyOnWriteBuffer& packet,
671 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200672 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700673
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200674 // Need to copy variable because OnRtcpReceived/OnPacketReceived
675 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
676 rtc::CopyOnWriteBuffer data(packet);
677 if (rtcp) {
678 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000679 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200680 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000681 }
682}
683
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000684void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700685 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000686 if (enabled_)
687 return;
688
Mirko Bonadei675513b2017-11-09 11:09:25 +0100689 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000690 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700691 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000692}
693
694void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700695 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000696 if (!enabled_)
697 return;
698
Mirko Bonadei675513b2017-11-09 11:09:25 +0100699 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000700 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700701 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000702}
703
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200704void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700705 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700706 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700707 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700708 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700709 if (rtp_packet_transport && rtp_packet_transport->writable() &&
710 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200711 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700712 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200713 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700714 }
715}
716
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200717void BaseChannel::ChannelWritable_n() {
718 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800719 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000720 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800721 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000722
Mirko Bonadei675513b2017-11-09 11:09:25 +0100723 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
724 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000725
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000726 was_ever_writable_ = true;
727 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700728 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729}
730
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200731bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800732 // Since DTLS is applied to all transports, checking RTP should be enough.
733 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000734}
735
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200736void BaseChannel::ChannelNotWritable_n() {
737 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000738 if (!writable_)
739 return;
740
Mirko Bonadei675513b2017-11-09 11:09:25 +0100741 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000742 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700743 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000744}
745
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200746bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700747 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800748 SdpType type,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700749 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700750 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700751 std::string* error_desc) {
jbauch5869f502017-06-29 12:31:36 -0700752 std::vector<int> encrypted_extension_ids;
753 for (const webrtc::RtpExtension& extension : extensions) {
754 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100755 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
756 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700757 encrypted_extension_ids.push_back(extension.id);
758 }
759 }
760
deadbeef7af91dd2016-12-13 11:29:11 -0800761 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200762 return network_thread_->Invoke<bool>(
Steve Anton3828c062017-12-06 10:34:51 -0800763 RTC_FROM_HERE,
764 Bind(&BaseChannel::SetRtpTransportParameters_n, this, content, type, src,
765 encrypted_extension_ids, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200766}
767
768bool BaseChannel::SetRtpTransportParameters_n(
769 const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -0800770 SdpType type,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200771 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700772 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200773 std::string* error_desc) {
774 RTC_DCHECK(network_thread_->IsCurrent());
775
Steve Anton3828c062017-12-06 10:34:51 -0800776 if (!SetSrtp_n(content->cryptos(), type, src, encrypted_extension_ids,
777 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700778 return false;
779 }
780
Steve Anton3828c062017-12-06 10:34:51 -0800781 if (!SetRtcpMux_n(content->rtcp_mux(), type, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700782 return false;
783 }
784
785 return true;
786}
787
zhihuangb2cdd932017-01-19 16:54:25 -0800788// |dtls| will be set to true if DTLS is active for transport and crypto is
789// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200790bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
791 bool* dtls,
792 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -0800793 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000794 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200795 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000796 return false;
797 }
798 return true;
799}
800
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800801void BaseChannel::EnableSdes_n() {
802 if (sdes_transport_) {
803 return;
Zhi Huangcf990f52017-09-22 12:12:30 -0700804 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800805 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
806 // time.
807 RTC_DCHECK(!dtls_srtp_transport_);
808 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800809 sdes_transport_ = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800810 std::move(unencrypted_rtp_transport_));
Zhi Huangd7455782017-11-30 14:50:52 -0800811#if defined(ENABLE_EXTERNAL_AUTH)
812 sdes_transport_->EnableExternalAuth();
813#endif
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800814 SetRtpTransport(sdes_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800815 RTC_LOG(LS_INFO) << "Wrapping RtpTransport in SrtpTransport.";
816}
817
818void BaseChannel::EnableDtlsSrtp_n() {
819 if (dtls_srtp_transport_) {
820 return;
821 }
822 // DtlsSrtpTransport and SrtpTransport shouldn't be enabled at the same
823 // time.
824 RTC_DCHECK(!sdes_transport_);
825 RTC_DCHECK(unencrypted_rtp_transport_);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800826
827 auto srtp_transport = rtc::MakeUnique<webrtc::SrtpTransport>(
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800828 std::move(unencrypted_rtp_transport_));
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800829#if defined(ENABLE_EXTERNAL_AUTH)
830 srtp_transport->EnableExternalAuth();
831#endif
832 dtls_srtp_transport_ =
833 rtc::MakeUnique<webrtc::DtlsSrtpTransport>(std::move(srtp_transport));
834
Zhi Huang2dfc42d2017-12-04 13:38:48 -0800835 SetRtpTransport(dtls_srtp_transport_.get());
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800836 if (cached_send_extension_ids_) {
837 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
838 *cached_send_extension_ids_);
839 }
840 if (cached_recv_extension_ids_) {
841 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
842 *cached_recv_extension_ids_);
843 }
844 // Set the DtlsTransport and the |dtls_srtp_transport_| will handle the DTLS
845 // relate signal internally.
846 RTC_DCHECK(rtp_dtls_transport_);
847 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
848 rtcp_dtls_transport_);
849
850 RTC_LOG(LS_INFO) << "Wrapping SrtpTransport in DtlsSrtpTransport.";
Zhi Huangcf990f52017-09-22 12:12:30 -0700851}
852
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200853bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
Steve Anton3828c062017-12-06 10:34:51 -0800854 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000855 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700856 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000857 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -0800858 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000859 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000860 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200861 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000862 if (!ret) {
863 return false;
864 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700865
866 // If SRTP was not required, but we're setting a description that uses SDES,
867 // we need to upgrade to an SrtpTransport.
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800868 if (!sdes_transport_ && !dtls && !cryptos.empty()) {
869 EnableSdes_n();
Zhi Huangcf990f52017-09-22 12:12:30 -0700870 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800871
Steve Anton3828c062017-12-06 10:34:51 -0800872 if ((type == SdpType::kAnswer || type == SdpType::kPrAnswer) && dtls) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800873 EnableDtlsSrtp_n();
874 }
Zhi Huangc99b6c72017-11-10 16:44:46 -0800875
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800876 UpdateEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
877
878 if (!dtls) {
Steve Anton3828c062017-12-06 10:34:51 -0800879 switch (type) {
880 case SdpType::kOffer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700881 ret = sdes_negotiator_.SetOffer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800882 break;
Steve Anton3828c062017-12-06 10:34:51 -0800883 case SdpType::kPrAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700884 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800885 break;
Steve Anton3828c062017-12-06 10:34:51 -0800886 case SdpType::kAnswer:
Zhi Huangcf990f52017-09-22 12:12:30 -0700887 ret = sdes_negotiator_.SetAnswer(cryptos, src);
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800888 break;
889 default:
890 break;
891 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700892
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800893 // If setting an SDES answer succeeded, apply the negotiated parameters
894 // to the SRTP transport.
Steve Anton3828c062017-12-06 10:34:51 -0800895 if ((type == SdpType::kPrAnswer || type == SdpType::kAnswer) && ret) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800896 if (sdes_negotiator_.send_cipher_suite() &&
897 sdes_negotiator_.recv_cipher_suite()) {
898 RTC_DCHECK(cached_send_extension_ids_);
899 RTC_DCHECK(cached_recv_extension_ids_);
900 ret = sdes_transport_->SetRtpParams(
901 *(sdes_negotiator_.send_cipher_suite()),
902 sdes_negotiator_.send_key().data(),
903 static_cast<int>(sdes_negotiator_.send_key().size()),
904 *(cached_send_extension_ids_),
905 *(sdes_negotiator_.recv_cipher_suite()),
906 sdes_negotiator_.recv_key().data(),
907 static_cast<int>(sdes_negotiator_.recv_key().size()),
908 *(cached_recv_extension_ids_));
909 } else {
910 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
Steve Anton3828c062017-12-06 10:34:51 -0800911 if (type == SdpType::kAnswer && sdes_transport_) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800912 // Explicitly reset the |sdes_transport_| if no crypto param is
913 // provided in the answer. No need to call |ResetParams()| for
914 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
915 sdes_transport_->ResetParams();
916 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700917 }
918 }
919 }
920
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000921 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -0800922 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000923 return false;
924 }
925 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000926}
927
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200928bool BaseChannel::SetRtcpMux_n(bool enable,
Steve Anton3828c062017-12-06 10:34:51 -0800929 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000930 ContentSource src,
931 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -0800932 // Provide a more specific error message for the RTCP mux "require" policy
933 // case.
zstein56162b92017-04-24 16:54:35 -0700934 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -0800935 SafeSetError(
936 "rtcpMuxPolicy is 'require', but media description does not "
937 "contain 'a=rtcp-mux'.",
938 error_desc);
939 return false;
940 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941 bool ret = false;
Steve Anton3828c062017-12-06 10:34:51 -0800942 switch (type) {
943 case SdpType::kOffer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000944 ret = rtcp_mux_filter_.SetOffer(enable, src);
945 break;
Steve Anton3828c062017-12-06 10:34:51 -0800946 case SdpType::kPrAnswer:
zhihuangb2cdd932017-01-19 16:54:25 -0800947 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700948 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000949 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
950 break;
Steve Anton3828c062017-12-06 10:34:51 -0800951 case SdpType::kAnswer:
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000952 ret = rtcp_mux_filter_.SetAnswer(enable, src);
953 if (ret && rtcp_mux_filter_.IsActive()) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -0800954 ActivateRtcpMux();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000955 }
956 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000957 default:
958 break;
959 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000960 if (!ret) {
961 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
962 return false;
963 }
zsteine8ab5432017-07-12 11:48:11 -0700964 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
Steve Anton3828c062017-12-06 10:34:51 -0800965 // |rtcp_mux_filter_| can be active if |action| is SdpType::kPrAnswer or
966 // SdpType::kAnswer, but we only want to tear down the RTCP transport if we
967 // received a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000968 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000969 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -0700970 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200971 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000972 }
973 }
974
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000975 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000976}
977
978bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700979 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -0800980 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000981}
982
Peter Boström0c4e06b2015-10-07 12:23:21 +0200983bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700984 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000985 return media_channel()->RemoveRecvStream(ssrc);
986}
987
988bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -0800989 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000990 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000991 // Check for streams that have been removed.
992 bool ret = true;
993 for (StreamParamsVec::const_iterator it = local_streams_.begin();
994 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000995 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000996 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000997 std::ostringstream desc;
998 desc << "Failed to remove send stream with ssrc "
999 << it->first_ssrc() << ".";
1000 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001001 ret = false;
1002 }
1003 }
1004 }
1005 // Check for new streams.
1006 for (StreamParamsVec::const_iterator it = streams.begin();
1007 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001008 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001009 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001010 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001011 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001012 std::ostringstream desc;
1013 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1014 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001015 ret = false;
1016 }
1017 }
1018 }
1019 local_streams_ = streams;
1020 return ret;
1021}
1022
1023bool BaseChannel::UpdateRemoteStreams_w(
1024 const std::vector<StreamParams>& streams,
Steve Anton3828c062017-12-06 10:34:51 -08001025 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001026 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001027 // Check for streams that have been removed.
1028 bool ret = true;
1029 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1030 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001031 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001032 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001033 std::ostringstream desc;
1034 desc << "Failed to remove remote stream with ssrc "
1035 << it->first_ssrc() << ".";
1036 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001037 ret = false;
1038 }
1039 }
1040 }
1041 // Check for new streams.
1042 for (StreamParamsVec::const_iterator it = streams.begin();
1043 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001044 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001045 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001046 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001047 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001048 std::ostringstream desc;
1049 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1050 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001051 ret = false;
1052 }
1053 }
1054 }
1055 remote_streams_ = streams;
1056 return ret;
1057}
1058
jbauch5869f502017-06-29 12:31:36 -07001059RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1060 const RtpHeaderExtensions& extensions) {
1061 if (!rtp_dtls_transport_ ||
1062 !rtp_dtls_transport_->crypto_options()
1063 .enable_encrypted_rtp_header_extensions) {
1064 RtpHeaderExtensions filtered;
1065 auto pred = [](const webrtc::RtpExtension& extension) {
1066 return !extension.encrypt;
1067 };
1068 std::copy_if(extensions.begin(), extensions.end(),
1069 std::back_inserter(filtered), pred);
1070 return filtered;
1071 }
1072
1073 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1074}
1075
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001076void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001077 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001078// Absolute Send Time extension id is used only with external auth,
1079// so do not bother searching for it and making asyncronious call to set
1080// something that is not used.
1081#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001082 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001083 webrtc::RtpExtension::FindHeaderExtensionByUri(
1084 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001085 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001086 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001087 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001088 RTC_FROM_HERE, network_thread_,
1089 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1090 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001091#endif
1092}
1093
1094void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1095 int rtp_abs_sendtime_extn_id) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001096 if (sdes_transport_) {
1097 sdes_transport_->CacheRtpAbsSendTimeHeaderExtension(
Zhi Huangcf990f52017-09-22 12:12:30 -07001098 rtp_abs_sendtime_extn_id);
Zhi Huang2a4d70c2017-11-29 15:41:59 -08001099 } else if (dtls_srtp_transport_) {
1100 dtls_srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1101 rtp_abs_sendtime_extn_id);
Zhi Huangcf990f52017-09-22 12:12:30 -07001102 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001103 RTC_LOG(LS_WARNING)
1104 << "Trying to cache the Absolute Send Time extension id "
1105 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001106 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001107}
1108
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001109void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001110 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001111 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001112 case MSG_SEND_RTP_PACKET:
1113 case MSG_SEND_RTCP_PACKET: {
1114 RTC_DCHECK(network_thread_->IsCurrent());
1115 SendPacketMessageData* data =
1116 static_cast<SendPacketMessageData*>(pmsg->pdata);
1117 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1118 SendPacket(rtcp, &data->packet, data->options);
1119 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001120 break;
1121 }
1122 case MSG_FIRSTPACKETRECEIVED: {
1123 SignalFirstPacketReceived(this);
1124 break;
1125 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001126 }
1127}
1128
zstein3dcf0e92017-06-01 13:22:42 -07001129void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001130 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001131}
1132
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001133void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001134 // Flush all remaining RTCP messages. This should only be called in
1135 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001136 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001137 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001138 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1139 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001140 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1141 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001142 }
1143}
1144
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001145void BaseChannel::SignalSentPacket_n(const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001146 RTC_DCHECK(network_thread_->IsCurrent());
1147 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001148 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001149 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1150}
1151
1152void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1153 RTC_DCHECK(worker_thread_->IsCurrent());
1154 SignalSentPacket(sent_packet);
1155}
1156
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001157void BaseChannel::UpdateEncryptedHeaderExtensionIds(
Zhi Huangc99b6c72017-11-10 16:44:46 -08001158 cricket::ContentSource source,
1159 const std::vector<int>& extension_ids) {
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001160 if (source == ContentSource::CS_LOCAL) {
1161 cached_recv_extension_ids_ = std::move(extension_ids);
1162 if (dtls_srtp_transport_) {
1163 dtls_srtp_transport_->UpdateRecvEncryptedHeaderExtensionIds(
1164 extension_ids);
1165 }
1166 } else {
1167 cached_send_extension_ids_ = std::move(extension_ids);
1168 if (dtls_srtp_transport_) {
1169 dtls_srtp_transport_->UpdateSendEncryptedHeaderExtensionIds(
1170 extension_ids);
1171 }
1172 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001173}
1174
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001175void BaseChannel::ActivateRtcpMux() {
1176 // We permanently activated RTCP muxing; signal that we no longer need
1177 // the RTCP transport.
1178 std::string debug_name =
1179 transport_name_.empty()
1180 ? rtp_transport_->rtp_packet_transport()->transport_name()
1181 : transport_name_;
1182 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1183 << "; no longer need RTCP transport for " << debug_name;
1184 if (rtp_transport_->rtcp_packet_transport()) {
1185 SetTransport_n(/*rtcp=*/true, nullptr, nullptr);
1186 if (dtls_srtp_transport_) {
1187 RTC_DCHECK(rtp_dtls_transport_);
1188 dtls_srtp_transport_->SetDtlsTransports(rtp_dtls_transport_,
1189 /*rtcp_dtls_transport_=*/nullptr);
1190 } else {
1191 rtp_transport_->SetRtcpPacketTransport(nullptr);
1192 }
1193 SignalRtcpMuxFullyActive(transport_name_);
Zhi Huangc99b6c72017-11-10 16:44:46 -08001194 }
Zhi Huangcd3fc5d2017-11-29 10:41:57 -08001195 UpdateWritableState_n();
Zhi Huangc99b6c72017-11-10 16:44:46 -08001196}
1197
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001198VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1199 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001200 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001201 MediaEngineInterface* media_engine,
Steve Anton8699a322017-11-06 15:53:33 -08001202 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001203 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001204 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001205 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001206 : BaseChannel(worker_thread,
1207 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001208 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001209 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001210 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001211 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001212 srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -08001213 media_engine_(media_engine) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001214
1215VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001216 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001217 StopAudioMonitor();
1218 StopMediaMonitor();
1219 // this can't be done in the base class, since it calls a virtual
1220 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001221 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001222}
1223
Peter Boström0c4e06b2015-10-07 12:23:21 +02001224bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001225 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001226 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001227 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001228 return InvokeOnWorker<bool>(
1229 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1230 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001231}
1232
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001233// TODO(juberti): Handle early media the right way. We should get an explicit
1234// ringing message telling us to start playing local ringback, which we cancel
1235// if any early media actually arrives. For now, we do the opposite, which is
1236// to wait 1 second for early media, and start playing local ringback if none
1237// arrives.
1238void VoiceChannel::SetEarlyMedia(bool enable) {
1239 if (enable) {
1240 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001241 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1242 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001243 } else {
1244 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001245 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001246 }
1247}
1248
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001249bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001250 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001251 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001252}
1253
Peter Boström0c4e06b2015-10-07 12:23:21 +02001254bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1255 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001256 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001257 return InvokeOnWorker<bool>(
1258 RTC_FROM_HERE,
1259 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001260}
1261
solenberg4bac9c52015-10-09 02:32:53 -07001262bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001263 return InvokeOnWorker<bool>(
1264 RTC_FROM_HERE,
1265 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001266}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001267
Tommif888bb52015-12-12 01:37:01 +01001268void VoiceChannel::SetRawAudioSink(
1269 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001270 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1271 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001272 // passing. So we invoke to our own little routine that gets a pointer to
1273 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001274 InvokeOnWorker<bool>(RTC_FROM_HERE,
1275 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001276}
1277
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001278webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001279 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001280 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001281}
1282
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001283webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1284 uint32_t ssrc) const {
1285 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001286}
1287
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001288bool VoiceChannel::SetRtpSendParameters(
1289 uint32_t ssrc,
1290 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001291 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001292 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001293 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001294}
1295
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001296bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1297 webrtc::RtpParameters parameters) {
1298 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1299}
1300
1301webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1302 uint32_t ssrc) const {
1303 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001304 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001305 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1306}
1307
1308webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1309 uint32_t ssrc) const {
1310 return media_channel()->GetRtpReceiveParameters(ssrc);
1311}
1312
1313bool VoiceChannel::SetRtpReceiveParameters(
1314 uint32_t ssrc,
1315 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001316 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001317 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001318 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1319}
1320
1321bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1322 webrtc::RtpParameters parameters) {
1323 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001324}
1325
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001326bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001327 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1328 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001329}
1330
hbos8d609f62017-04-10 07:39:05 -07001331std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1332 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001333 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1334}
1335
1336std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1337 RTC_DCHECK(worker_thread()->IsCurrent());
1338 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001339}
1340
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001341void VoiceChannel::StartMediaMonitor(int cms) {
1342 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001343 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001344 media_monitor_->SignalUpdate.connect(
1345 this, &VoiceChannel::OnMediaMonitorUpdate);
1346 media_monitor_->Start(cms);
1347}
1348
1349void VoiceChannel::StopMediaMonitor() {
1350 if (media_monitor_) {
1351 media_monitor_->Stop();
1352 media_monitor_->SignalUpdate.disconnect(this);
1353 media_monitor_.reset();
1354 }
1355}
1356
1357void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001358 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001359 audio_monitor_
1360 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1361 audio_monitor_->Start(cms);
1362}
1363
1364void VoiceChannel::StopAudioMonitor() {
1365 if (audio_monitor_) {
1366 audio_monitor_->Stop();
1367 audio_monitor_.reset();
1368 }
1369}
1370
1371bool VoiceChannel::IsAudioMonitorRunning() const {
1372 return (audio_monitor_.get() != NULL);
1373}
1374
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001375int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001376 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001377}
1378
1379int VoiceChannel::GetOutputLevel_w() {
1380 return media_channel()->GetOutputLevel();
1381}
1382
1383void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1384 media_channel()->GetActiveStreams(actives);
1385}
1386
zstein3dcf0e92017-06-01 13:22:42 -07001387void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001388 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001389 const rtc::PacketTime& packet_time) {
1390 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001391 // Set a flag when we've received an RTP packet. If we're waiting for early
1392 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001393 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001394 received_media_ = true;
1395 }
1396}
1397
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001398void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001399 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001400 invoker_.AsyncInvoke<void>(
1401 RTC_FROM_HERE, worker_thread_,
1402 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001403}
1404
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001405void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001406 // Render incoming data if we're the active call, and we have the local
1407 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001408 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001409 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001410
1411 // Send outgoing data if we're the active call, we have the remote content,
1412 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001413 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001414 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001415
Mirko Bonadei675513b2017-11-09 11:09:25 +01001416 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001417}
1418
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001419bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001420 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001421 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001422 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001423 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001424 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001425
1426 const AudioContentDescription* audio =
1427 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001428 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001429 if (!audio) {
1430 SafeSetError("Can't find audio content in local description.", error_desc);
1431 return false;
1432 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001433
jbauch5869f502017-06-29 12:31:36 -07001434 RtpHeaderExtensions rtp_header_extensions =
1435 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1436
Steve Anton3828c062017-12-06 10:34:51 -08001437 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1438 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001439 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001440 }
1441
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001442 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001443 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001444 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001445 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001446 error_desc);
1447 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001448 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001449 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001450 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001451 }
1452 last_recv_params_ = recv_params;
1453
1454 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1455 // only give it to the media channel once we have a remote
1456 // description too (without a remote description, we won't be able
1457 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001458 if (!UpdateLocalStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001459 SafeSetError("Failed to set local audio description streams.", error_desc);
1460 return false;
1461 }
1462
1463 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001464 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001465 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001466}
1467
1468bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001469 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001470 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001471 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001472 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001473 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001474
1475 const AudioContentDescription* audio =
1476 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001477 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001478 if (!audio) {
1479 SafeSetError("Can't find audio content in remote description.", error_desc);
1480 return false;
1481 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001482
jbauch5869f502017-06-29 12:31:36 -07001483 RtpHeaderExtensions rtp_header_extensions =
1484 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1485
Steve Anton3828c062017-12-06 10:34:51 -08001486 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1487 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001488 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001489 }
1490
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001491 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001492 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1493 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001494 if (audio->agc_minus_10db()) {
Oskar Sundbom36f8f3e2017-11-16 10:54:27 +01001495 send_params.options.adjust_agc_delta = kAgcMinus10db;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001496 }
skvladdc1c62c2016-03-16 19:07:43 -07001497
1498 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1499 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001500 SafeSetError("Failed to set remote audio description send parameters.",
1501 error_desc);
1502 return false;
1503 }
1504 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001505
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001506 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1507 // and only give it to the media channel once we have a local
1508 // description too (without a local description, we won't be able to
1509 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001510 if (!UpdateRemoteStreams_w(audio->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001511 SafeSetError("Failed to set remote audio description streams.", error_desc);
1512 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001513 }
1514
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001515 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001516 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001517 }
1518
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001519 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001520 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001521 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001522}
1523
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001524void VoiceChannel::HandleEarlyMediaTimeout() {
1525 // This occurs on the main thread, not the worker thread.
1526 if (!received_media_) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001527 RTC_LOG(LS_INFO) << "No early media received before timeout";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001528 SignalEarlyMediaTimeout(this);
1529 }
1530}
1531
Peter Boström0c4e06b2015-10-07 12:23:21 +02001532bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1533 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001534 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001535 if (!enabled()) {
1536 return false;
1537 }
solenberg1d63dd02015-12-02 12:35:09 -08001538 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001539}
1540
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001541void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001542 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001543 case MSG_EARLYMEDIATIMEOUT:
1544 HandleEarlyMediaTimeout();
1545 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001546 case MSG_CHANNEL_ERROR: {
1547 VoiceChannelErrorMessageData* data =
1548 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001549 delete data;
1550 break;
1551 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001552 default:
1553 BaseChannel::OnMessage(pmsg);
1554 break;
1555 }
1556}
1557
1558void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001559 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001560 SignalConnectionMonitor(this, infos);
1561}
1562
1563void VoiceChannel::OnMediaMonitorUpdate(
1564 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001565 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001566 SignalMediaMonitor(this, info);
1567}
1568
1569void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1570 const AudioInfo& info) {
1571 SignalAudioMonitor(this, info);
1572}
1573
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001574VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1575 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001576 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001577 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001578 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001579 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001580 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001581 : BaseChannel(worker_thread,
1582 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001583 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001584 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001585 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001586 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001587 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001588
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001589VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001590 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001591 StopMediaMonitor();
1592 // this can't be done in the base class, since it calls a virtual
1593 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001594
1595 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001596}
1597
nisse08582ff2016-02-04 01:24:52 -08001598bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001599 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001600 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001601 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001602 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001603 return true;
1604}
1605
deadbeef5a4a75a2016-06-02 16:23:38 -07001606bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001607 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001608 bool mute,
1609 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001610 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001611 return InvokeOnWorker<bool>(
1612 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1613 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001614}
1615
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001616webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001617 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001618 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001619}
1620
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001621webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1622 uint32_t ssrc) const {
1623 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001624}
1625
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001626bool VideoChannel::SetRtpSendParameters(
1627 uint32_t ssrc,
1628 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001629 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001630 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001631 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001632}
1633
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001634bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1635 webrtc::RtpParameters parameters) {
1636 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1637}
1638
1639webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1640 uint32_t ssrc) const {
1641 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001642 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001643 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1644}
1645
1646webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1647 uint32_t ssrc) const {
1648 return media_channel()->GetRtpReceiveParameters(ssrc);
1649}
1650
1651bool VideoChannel::SetRtpReceiveParameters(
1652 uint32_t ssrc,
1653 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001654 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001655 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001656 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1657}
1658
1659bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1660 webrtc::RtpParameters parameters) {
1661 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001662}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001663
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001664void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001665 // Send outgoing data if we're the active call, we have the remote content,
1666 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001667 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001668 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001669 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001670 // TODO(gangji): Report error back to server.
1671 }
1672
Mirko Bonadei675513b2017-11-09 11:09:25 +01001673 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001674}
1675
stefanf79ade12017-06-02 06:44:03 -07001676void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1677 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1678 media_channel(), bwe_info));
1679}
1680
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001681bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001682 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1683 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001684}
1685
1686void VideoChannel::StartMediaMonitor(int cms) {
1687 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001688 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001689 media_monitor_->SignalUpdate.connect(
1690 this, &VideoChannel::OnMediaMonitorUpdate);
1691 media_monitor_->Start(cms);
1692}
1693
1694void VideoChannel::StopMediaMonitor() {
1695 if (media_monitor_) {
1696 media_monitor_->Stop();
1697 media_monitor_.reset();
1698 }
1699}
1700
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001701bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001702 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001703 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001704 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001705 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001706 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001707
1708 const VideoContentDescription* video =
1709 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001710 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001711 if (!video) {
1712 SafeSetError("Can't find video content in local description.", error_desc);
1713 return false;
1714 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001715
jbauch5869f502017-06-29 12:31:36 -07001716 RtpHeaderExtensions rtp_header_extensions =
1717 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1718
Steve Anton3828c062017-12-06 10:34:51 -08001719 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1720 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001721 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001722 }
1723
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001724 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001725 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001726 if (!media_channel()->SetRecvParameters(recv_params)) {
1727 SafeSetError("Failed to set local video description recv parameters.",
1728 error_desc);
1729 return false;
1730 }
1731 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001732 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001733 }
1734 last_recv_params_ = recv_params;
1735
1736 // TODO(pthatcher): Move local streams into VideoSendParameters, and
1737 // only give it to the media channel once we have a remote
1738 // description too (without a remote description, we won't be able
1739 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001740 if (!UpdateLocalStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001741 SafeSetError("Failed to set local video description streams.", error_desc);
1742 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001743 }
1744
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001745 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001746 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001747 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001748}
1749
1750bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001751 SdpType type,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001752 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001753 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001754 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001755 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001756
1757 const VideoContentDescription* video =
1758 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001759 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001760 if (!video) {
1761 SafeSetError("Can't find video content in remote description.", error_desc);
1762 return false;
1763 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001764
jbauch5869f502017-06-29 12:31:36 -07001765 RtpHeaderExtensions rtp_header_extensions =
1766 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1767
Steve Anton3828c062017-12-06 10:34:51 -08001768 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1769 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001770 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001771 }
1772
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001773 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001774 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
1775 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001776 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08001777 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001778 }
skvladdc1c62c2016-03-16 19:07:43 -07001779
1780 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1781
1782 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001783 SafeSetError("Failed to set remote video description send parameters.",
1784 error_desc);
1785 return false;
1786 }
1787 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001788
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001789 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
1790 // and only give it to the media channel once we have a local
1791 // description too (without a local description, we won't be able to
1792 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001793 if (!UpdateRemoteStreams_w(video->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001794 SafeSetError("Failed to set remote video description streams.", error_desc);
1795 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001796 }
1797
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001798 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001799 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001800 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001801
1802 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001803 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001804 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001805}
1806
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001807void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001808 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001809 case MSG_CHANNEL_ERROR: {
1810 const VideoChannelErrorMessageData* data =
1811 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001812 delete data;
1813 break;
1814 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001815 default:
1816 BaseChannel::OnMessage(pmsg);
1817 break;
1818 }
1819}
1820
1821void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001822 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001823 SignalConnectionMonitor(this, infos);
1824}
1825
1826// TODO(pthatcher): Look into removing duplicate code between
1827// audio, video, and data, perhaps by using templates.
1828void VideoChannel::OnMediaMonitorUpdate(
1829 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001830 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001831 SignalMediaMonitor(this, info);
1832}
1833
deadbeef953c2ce2017-01-09 14:53:41 -08001834RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
1835 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001836 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001837 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08001838 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001839 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001840 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001841 : BaseChannel(worker_thread,
1842 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001843 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001844 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001845 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001846 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08001847 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001848
deadbeef953c2ce2017-01-09 14:53:41 -08001849RtpDataChannel::~RtpDataChannel() {
1850 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001851 StopMediaMonitor();
1852 // this can't be done in the base class, since it calls a virtual
1853 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001854
1855 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001856}
1857
Steve Anton8699a322017-11-06 15:53:33 -08001858void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08001859 DtlsTransportInternal* rtp_dtls_transport,
1860 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08001861 rtc::PacketTransportInternal* rtp_packet_transport,
1862 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08001863 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
1864 rtp_packet_transport, rtcp_packet_transport);
1865
deadbeef953c2ce2017-01-09 14:53:41 -08001866 media_channel()->SignalDataReceived.connect(this,
1867 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00001868 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001869 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001870}
1871
Zhi Huang2dfc42d2017-12-04 13:38:48 -08001872void RtpDataChannel::Init_w(webrtc::RtpTransportInternal* rtp_transport) {
1873 BaseChannel::Init_w(rtp_transport);
1874 media_channel()->SignalDataReceived.connect(this,
1875 &RtpDataChannel::OnDataReceived);
1876 media_channel()->SignalReadyToSend.connect(
1877 this, &RtpDataChannel::OnDataChannelReadyToSend);
1878}
1879
deadbeef953c2ce2017-01-09 14:53:41 -08001880bool RtpDataChannel::SendData(const SendDataParams& params,
1881 const rtc::CopyOnWriteBuffer& payload,
1882 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07001883 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001884 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
1885 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001886}
1887
deadbeef953c2ce2017-01-09 14:53:41 -08001888bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001889 const DataContentDescription* content,
1890 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001891 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
1892 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08001893 // It's been set before, but doesn't match. That's bad.
1894 if (is_sctp) {
1895 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
1896 error_desc);
1897 return false;
1898 }
1899 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001900}
1901
deadbeef953c2ce2017-01-09 14:53:41 -08001902bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001903 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001904 std::string* error_desc) {
1905 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001906 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001907 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001908
1909 const DataContentDescription* data =
1910 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001911 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001912 if (!data) {
1913 SafeSetError("Can't find data content in local description.", error_desc);
1914 return false;
1915 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001916
deadbeef953c2ce2017-01-09 14:53:41 -08001917 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001918 return false;
1919 }
1920
jbauch5869f502017-06-29 12:31:36 -07001921 RtpHeaderExtensions rtp_header_extensions =
1922 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1923
Steve Anton3828c062017-12-06 10:34:51 -08001924 if (!SetRtpTransportParameters(content, type, CS_LOCAL, rtp_header_extensions,
1925 error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001926 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001927 }
1928
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001929 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001930 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001931 if (!media_channel()->SetRecvParameters(recv_params)) {
1932 SafeSetError("Failed to set remote data description recv parameters.",
1933 error_desc);
1934 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001935 }
deadbeef953c2ce2017-01-09 14:53:41 -08001936 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001937 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001938 }
1939 last_recv_params_ = recv_params;
1940
1941 // TODO(pthatcher): Move local streams into DataSendParameters, and
1942 // only give it to the media channel once we have a remote
1943 // description too (without a remote description, we won't be able
1944 // to send them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08001945 if (!UpdateLocalStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001946 SafeSetError("Failed to set local data description streams.", error_desc);
1947 return false;
1948 }
1949
1950 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001951 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001952 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001953}
1954
deadbeef953c2ce2017-01-09 14:53:41 -08001955bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
Steve Anton3828c062017-12-06 10:34:51 -08001956 SdpType type,
deadbeef953c2ce2017-01-09 14:53:41 -08001957 std::string* error_desc) {
1958 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001959 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001960
1961 const DataContentDescription* data =
1962 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001963 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001964 if (!data) {
1965 SafeSetError("Can't find data content in remote description.", error_desc);
1966 return false;
1967 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001968
Zhi Huang801b8682017-11-15 11:36:43 -08001969 // If the remote data doesn't have codecs, it must be empty, so ignore it.
1970 if (!data->has_codecs()) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001971 return true;
1972 }
1973
deadbeef953c2ce2017-01-09 14:53:41 -08001974 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001975 return false;
1976 }
1977
jbauch5869f502017-06-29 12:31:36 -07001978 RtpHeaderExtensions rtp_header_extensions =
1979 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
1980
Mirko Bonadei675513b2017-11-09 11:09:25 +01001981 RTC_LOG(LS_INFO) << "Setting remote data description";
Steve Anton3828c062017-12-06 10:34:51 -08001982 if (!SetRtpTransportParameters(content, type, CS_REMOTE,
1983 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001984 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001985 }
1986
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001987 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001988 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
1989 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001990 if (!media_channel()->SetSendParameters(send_params)) {
1991 SafeSetError("Failed to set remote data description send parameters.",
1992 error_desc);
1993 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001994 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001995 last_send_params_ = send_params;
1996
1997 // TODO(pthatcher): Move remote streams into DataRecvParameters,
1998 // and only give it to the media channel once we have a local
1999 // description too (without a local description, we won't be able to
2000 // recv them anyway).
Steve Anton3828c062017-12-06 10:34:51 -08002001 if (!UpdateRemoteStreams_w(data->streams(), type, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002002 SafeSetError("Failed to set remote data description streams.",
2003 error_desc);
2004 return false;
2005 }
2006
2007 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002008 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002009 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002010}
2011
deadbeef953c2ce2017-01-09 14:53:41 -08002012void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002013 // Render incoming data if we're the active call, and we have the local
2014 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002015 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002016 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002017 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002018 }
2019
2020 // Send outgoing data if we're the active call, we have the remote content,
2021 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002022 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002023 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002024 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002025 }
2026
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002027 // Trigger SignalReadyToSendData asynchronously.
2028 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002029
Mirko Bonadei675513b2017-11-09 11:09:25 +01002030 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002031}
2032
deadbeef953c2ce2017-01-09 14:53:41 -08002033void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002034 switch (pmsg->message_id) {
2035 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002036 DataChannelReadyToSendMessageData* data =
2037 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002038 ready_to_send_data_ = data->data();
2039 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002040 delete data;
2041 break;
2042 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002043 case MSG_DATARECEIVED: {
2044 DataReceivedMessageData* data =
2045 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002046 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002047 delete data;
2048 break;
2049 }
2050 case MSG_CHANNEL_ERROR: {
2051 const DataChannelErrorMessageData* data =
2052 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002053 delete data;
2054 break;
2055 }
2056 default:
2057 BaseChannel::OnMessage(pmsg);
2058 break;
2059 }
2060}
2061
deadbeef953c2ce2017-01-09 14:53:41 -08002062void RtpDataChannel::OnConnectionMonitorUpdate(
2063 ConnectionMonitor* monitor,
2064 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002065 SignalConnectionMonitor(this, infos);
2066}
2067
deadbeef953c2ce2017-01-09 14:53:41 -08002068void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002069 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002070 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002071 media_monitor_->SignalUpdate.connect(this,
2072 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002073 media_monitor_->Start(cms);
2074}
2075
deadbeef953c2ce2017-01-09 14:53:41 -08002076void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002077 if (media_monitor_) {
2078 media_monitor_->Stop();
2079 media_monitor_->SignalUpdate.disconnect(this);
2080 media_monitor_.reset();
2081 }
2082}
2083
deadbeef953c2ce2017-01-09 14:53:41 -08002084void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2085 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002086 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002087 SignalMediaMonitor(this, info);
2088}
2089
deadbeef953c2ce2017-01-09 14:53:41 -08002090void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2091 const char* data,
2092 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002093 DataReceivedMessageData* msg = new DataReceivedMessageData(
2094 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002095 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002096}
2097
deadbeef953c2ce2017-01-09 14:53:41 -08002098void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2099 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002100 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2101 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002102 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002103}
2104
deadbeef953c2ce2017-01-09 14:53:41 -08002105void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002106 // This is usded for congestion control to indicate that the stream is ready
2107 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2108 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002109 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002110 new DataChannelReadyToSendMessageData(writable));
2111}
2112
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002113} // namespace cricket