blob: 00aab7ff0b44c51b714a13efbadfad5e3f689be6 [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"
34#include "pc/rtptransport.h"
35#include "pc/srtptransport.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000036
37namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000038using rtc::Bind;
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
66// Value specified in RFC 5764.
67static const char kDtlsSrtpExporterLabel[] = "EXTRACTOR-dtls_srtp";
68
69static const int kAgcMinus10db = -10;
70
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000071static void SafeSetError(const std::string& message, std::string* error_desc) {
72 if (error_desc) {
73 *error_desc = message;
74 }
75}
76
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000077struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020078 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000079 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020080 : ssrc(in_ssrc), error(in_error) {}
81 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000082 VoiceMediaChannel::Error error;
83};
84
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000085struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020086 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020088 : ssrc(in_ssrc), error(in_error) {}
89 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000090 VideoMediaChannel::Error error;
91};
92
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000093struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020094 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020096 : ssrc(in_ssrc), error(in_error) {}
97 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000098 DataMediaChannel::Error error;
99};
100
jbaucheec21bd2016-03-20 06:15:43 -0700101static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -0700103 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000104}
105
106static bool IsReceiveContentDirection(MediaContentDirection direction) {
107 return direction == MD_SENDRECV || direction == MD_RECVONLY;
108}
109
110static bool IsSendContentDirection(MediaContentDirection direction) {
111 return direction == MD_SENDRECV || direction == MD_SENDONLY;
112}
113
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700114template <class Codec>
115void RtpParametersFromMediaDescription(
116 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700117 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700118 RtpParameters<Codec>* params) {
119 // TODO(pthatcher): Remove this once we're sure no one will give us
120 // a description without codecs (currently a CA_UPDATE with just
121 // streams can).
122 if (desc->has_codecs()) {
123 params->codecs = desc->codecs();
124 }
125 // TODO(pthatcher): See if we really need
126 // rtp_header_extensions_set() and remove it if we don't.
127 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700128 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700129 }
deadbeef13871492015-12-09 12:37:51 -0800130 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700131}
132
nisse05103312016-03-16 02:22:50 -0700133template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700134void RtpSendParametersFromMediaDescription(
135 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700136 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700137 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700138 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700139 send_params->max_bandwidth_bps = desc->bandwidth();
140}
141
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200142BaseChannel::BaseChannel(rtc::Thread* worker_thread,
143 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800144 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -0800145 std::unique_ptr<MediaChannel> media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700146 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800147 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800148 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200149 : worker_thread_(worker_thread),
150 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800151 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700153 rtcp_mux_required_(rtcp_mux_required),
deadbeef7af91dd2016-12-13 11:29:11 -0800154 srtp_required_(srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -0800155 media_channel_(std::move(media_channel)),
michaelt79e05882016-11-08 02:50:09 -0800156 selected_candidate_pair_(nullptr) {
Steve Anton8699a322017-11-06 15:53:33 -0800157 RTC_DCHECK_RUN_ON(worker_thread_);
Zhi Huangcf990f52017-09-22 12:12:30 -0700158 if (srtp_required) {
159 auto transport =
160 rtc::MakeUnique<webrtc::SrtpTransport>(rtcp_mux_required, content_name);
161 srtp_transport_ = transport.get();
162 rtp_transport_ = std::move(transport);
jbauchdfcab722017-03-06 00:14:10 -0800163#if defined(ENABLE_EXTERNAL_AUTH)
Zhi Huangcf990f52017-09-22 12:12:30 -0700164 srtp_transport_->EnableExternalAuth();
jbauchdfcab722017-03-06 00:14:10 -0800165#endif
Zhi Huangcf990f52017-09-22 12:12:30 -0700166 } else {
167 rtp_transport_ = rtc::MakeUnique<webrtc::RtpTransport>(rtcp_mux_required);
168 srtp_transport_ = nullptr;
169 }
zsteine8ab5432017-07-12 11:48:11 -0700170 rtp_transport_->SignalReadyToSend.connect(
zstein56162b92017-04-24 16:54:35 -0700171 this, &BaseChannel::OnTransportReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -0700172 // 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.
zsteine8ab5432017-07-12 11:48:11 -0700175 rtp_transport_->SignalPacketReceived.connect(this,
zstein398c3fd2017-07-19 13:38:02 -0700176 &BaseChannel::OnPacketReceived);
Zhi Huang942bc2e2017-11-13 13:26:07 -0800177 rtp_transport_->SignalNetworkRouteChanged.connect(
178 this, &BaseChannel::OnNetworkRouteChanged);
Mirko Bonadei675513b2017-11-09 11:09:25 +0100179 RTC_LOG(LS_INFO) << "Created channel for " << content_name;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000180}
181
182BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800183 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Steve Anton8699a322017-11-06 15:53:33 -0800184 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.org78187522013-10-07 23:32:02 +0000185 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000186 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200187 // Eats any outstanding messages or packets.
188 worker_thread_->Clear(&invoker_);
189 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000190 // We must destroy the media channel before the transport channel, otherwise
191 // the media channel may try to send on the dead transport channel. NULLing
192 // is not an effective strategy since the sends will come on another thread.
Steve Anton8699a322017-11-06 15:53:33 -0800193 media_channel_.reset();
Mirko Bonadei675513b2017-11-09 11:09:25 +0100194 RTC_LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200195}
196
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200197void BaseChannel::DisconnectTransportChannels_n() {
198 // Send any outstanding RTCP packets.
199 FlushRtcpMessages_n();
200
201 // Stop signals from transport channels, but keep them alive because
202 // media_channel may use them from a different thread.
zhihuangb2cdd932017-01-19 16:54:25 -0800203 if (rtp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800204 DisconnectFromDtlsTransport(rtp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700205 } else if (rtp_transport_->rtp_packet_transport()) {
206 DisconnectFromPacketTransport(rtp_transport_->rtp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200207 }
zhihuangb2cdd932017-01-19 16:54:25 -0800208 if (rtcp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800209 DisconnectFromDtlsTransport(rtcp_dtls_transport_);
zsteine8ab5432017-07-12 11:48:11 -0700210 } else if (rtp_transport_->rtcp_packet_transport()) {
211 DisconnectFromPacketTransport(rtp_transport_->rtcp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200212 }
213
zsteine8ab5432017-07-12 11:48:11 -0700214 rtp_transport_->SetRtpPacketTransport(nullptr);
215 rtp_transport_->SetRtcpPacketTransport(nullptr);
zstein3dcf0e92017-06-01 13:22:42 -0700216
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200217 // Clear pending read packets/messages.
218 network_thread_->Clear(&invoker_);
219 network_thread_->Clear(this);
220}
221
Steve Anton8699a322017-11-06 15:53:33 -0800222void BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800223 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800224 rtc::PacketTransportInternal* rtp_packet_transport,
225 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -0800226 RTC_DCHECK_RUN_ON(worker_thread_);
227 network_thread_->Invoke<void>(RTC_FROM_HERE, [&] {
228 return InitNetwork_n(rtp_dtls_transport, rtcp_dtls_transport,
229 rtp_packet_transport, rtcp_packet_transport);
230 });
231
deadbeeff5346592017-01-24 21:51:21 -0800232 // Both RTP and RTCP channels should be set, we can call SetInterface on
233 // the media channel and it can set network options.
wu@webrtc.orgde305012013-10-31 15:40:38 +0000234 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000235}
236
Steve Anton8699a322017-11-06 15:53:33 -0800237void BaseChannel::InitNetwork_n(
deadbeeff5346592017-01-24 21:51:21 -0800238 DtlsTransportInternal* rtp_dtls_transport,
239 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800240 rtc::PacketTransportInternal* rtp_packet_transport,
241 rtc::PacketTransportInternal* rtcp_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200242 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800243 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport, rtp_packet_transport,
244 rtcp_packet_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200245
zstein56162b92017-04-24 16:54:35 -0700246 if (rtcp_mux_required_) {
deadbeefac22f702017-01-12 21:59:29 -0800247 rtcp_mux_filter_.SetActive();
248 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200249}
250
wu@webrtc.org78187522013-10-07 23:32:02 +0000251void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200252 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000253 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200254 // Packets arrive on the network thread, processing packets calls virtual
255 // functions, so need to stop this process in Deinit that is called in
256 // derived classes destructor.
257 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700258 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000259}
260
zhihuangb2cdd932017-01-19 16:54:25 -0800261void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
262 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800263 network_thread_->Invoke<void>(
264 RTC_FROM_HERE,
265 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
266 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000267}
268
deadbeeff5346592017-01-24 21:51:21 -0800269void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800270 rtc::PacketTransportInternal* rtp_packet_transport,
271 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800272 network_thread_->Invoke<void>(
273 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
274 rtp_packet_transport, rtcp_packet_transport));
275}
zhihuangf5b251b2017-01-12 19:37:48 -0800276
deadbeeff5346592017-01-24 21:51:21 -0800277void BaseChannel::SetTransports_n(
278 DtlsTransportInternal* rtp_dtls_transport,
279 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800280 rtc::PacketTransportInternal* rtp_packet_transport,
281 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800282 RTC_DCHECK(network_thread_->IsCurrent());
283 // Validate some assertions about the input.
284 RTC_DCHECK(rtp_packet_transport);
285 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
286 if (rtp_dtls_transport || rtcp_dtls_transport) {
287 // DTLS/non-DTLS pointers should be to the same object.
288 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
289 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
290 // Can't go from non-DTLS to DTLS.
zsteine8ab5432017-07-12 11:48:11 -0700291 RTC_DCHECK(!rtp_transport_->rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800292 } else {
293 // Can't go from DTLS to non-DTLS.
294 RTC_DCHECK(!rtp_dtls_transport_);
295 }
296 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800297 if (rtp_dtls_transport && rtcp_dtls_transport) {
298 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
299 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800300 }
deadbeeff5346592017-01-24 21:51:21 -0800301 std::string debug_name;
302 if (rtp_dtls_transport) {
303 transport_name_ = rtp_dtls_transport->transport_name();
304 debug_name = transport_name_;
305 } else {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800306 debug_name = rtp_packet_transport->transport_name();
deadbeeff5346592017-01-24 21:51:21 -0800307 }
zsteine8ab5432017-07-12 11:48:11 -0700308 if (rtp_packet_transport == rtp_transport_->rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800309 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800310 return;
deadbeefcbecd352015-09-23 11:50:27 -0700311 }
312
Zhi Huangcf990f52017-09-22 12:12:30 -0700313 // When using DTLS-SRTP, we must reset the SrtpTransport every time the
314 // DtlsTransport changes and wait until the DTLS handshake is complete to set
315 // the newly negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200316 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800317 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700318 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800319 writable_ = false;
Zhi Huangcf990f52017-09-22 12:12:30 -0700320 dtls_active_ = false;
321 if (srtp_transport_) {
322 srtp_transport_->ResetParams();
323 }
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800324 }
325
deadbeefac22f702017-01-12 21:59:29 -0800326 // If this BaseChannel doesn't require RTCP mux and we haven't fully
327 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800328 if (rtcp_packet_transport) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100329 RTC_LOG(LS_INFO) << "Setting RTCP Transport for " << content_name()
330 << " on " << debug_name << " transport "
331 << rtcp_packet_transport;
deadbeeff5346592017-01-24 21:51:21 -0800332 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000333 }
334
Mirko Bonadei675513b2017-11-09 11:09:25 +0100335 RTC_LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
336 << debug_name << " transport " << rtp_packet_transport;
deadbeeff5346592017-01-24 21:51:21 -0800337 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800338
deadbeefcbecd352015-09-23 11:50:27 -0700339 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700340 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200341 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000342}
343
deadbeeff5346592017-01-24 21:51:21 -0800344void BaseChannel::SetTransport_n(
345 bool rtcp,
346 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800347 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200348 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800349 if (new_dtls_transport) {
350 RTC_DCHECK(new_dtls_transport == new_packet_transport);
351 }
deadbeeff5346592017-01-24 21:51:21 -0800352 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800353 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700354 rtc::PacketTransportInternal* old_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700355 rtcp ? rtp_transport_->rtcp_packet_transport()
356 : rtp_transport_->rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800357
deadbeeff5346592017-01-24 21:51:21 -0800358 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700359 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000360 return;
361 }
zhihuangb2cdd932017-01-19 16:54:25 -0800362
deadbeeff5346592017-01-24 21:51:21 -0800363 RTC_DCHECK(old_packet_transport != new_packet_transport);
364 if (old_dtls_transport) {
365 DisconnectFromDtlsTransport(old_dtls_transport);
366 } else if (old_packet_transport) {
367 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000368 }
369
zsteind48dbda2017-04-04 19:45:57 -0700370 if (rtcp) {
zsteine8ab5432017-07-12 11:48:11 -0700371 rtp_transport_->SetRtcpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700372 } else {
zsteine8ab5432017-07-12 11:48:11 -0700373 rtp_transport_->SetRtpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700374 }
deadbeeff5346592017-01-24 21:51:21 -0800375 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000376
deadbeeff5346592017-01-24 21:51:21 -0800377 // If there's no new transport, we're done after disconnecting from old one.
378 if (!new_packet_transport) {
379 return;
380 }
381
382 if (rtcp && new_dtls_transport) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700383 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_active()))
384 << "Setting RTCP for DTLS/SRTP after the DTLS is active "
deadbeeff5346592017-01-24 21:51:21 -0800385 << "should never happen.";
386 }
zstein56162b92017-04-24 16:54:35 -0700387
deadbeeff5346592017-01-24 21:51:21 -0800388 if (new_dtls_transport) {
389 ConnectToDtlsTransport(new_dtls_transport);
390 } else {
391 ConnectToPacketTransport(new_packet_transport);
392 }
393 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
394 for (const auto& pair : socket_options) {
395 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800396 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000397}
398
deadbeeff5346592017-01-24 21:51:21 -0800399void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200400 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000401
zstein56162b92017-04-24 16:54:35 -0700402 // TODO(zstein): de-dup with ConnectToPacketTransport
zhihuangb2cdd932017-01-19 16:54:25 -0800403 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
zhihuangb2cdd932017-01-19 16:54:25 -0800404 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
405 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000406}
407
deadbeeff5346592017-01-24 21:51:21 -0800408void BaseChannel::DisconnectFromDtlsTransport(
409 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200410 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800411 transport->SignalWritableState.disconnect(this);
zhihuangb2cdd932017-01-19 16:54:25 -0800412 transport->SignalDtlsState.disconnect(this);
413 transport->SignalSentPacket.disconnect(this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000414}
415
deadbeeff5346592017-01-24 21:51:21 -0800416void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800417 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800418 RTC_DCHECK_RUN_ON(network_thread_);
419 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
deadbeeff5346592017-01-24 21:51:21 -0800420 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
421}
422
423void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800424 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800425 RTC_DCHECK_RUN_ON(network_thread_);
426 transport->SignalWritableState.disconnect(this);
deadbeeff5346592017-01-24 21:51:21 -0800427 transport->SignalSentPacket.disconnect(this);
428}
429
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000430bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700431 worker_thread_->Invoke<void>(
432 RTC_FROM_HERE,
433 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
434 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000435 return true;
436}
437
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700439 return InvokeOnWorker<bool>(RTC_FROM_HERE,
440 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000441}
442
Peter Boström0c4e06b2015-10-07 12:23:21 +0200443bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700444 return InvokeOnWorker<bool>(
445 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000446}
447
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000448bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700449 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700450 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000451}
452
Peter Boström0c4e06b2015-10-07 12:23:21 +0200453bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700454 return InvokeOnWorker<bool>(
455 RTC_FROM_HERE,
456 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000457}
458
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000459bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000460 ContentAction action,
461 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100462 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700463 return InvokeOnWorker<bool>(
464 RTC_FROM_HERE,
465 Bind(&BaseChannel::SetLocalContent_w, this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000466}
467
468bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000469 ContentAction action,
470 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100471 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700472 return InvokeOnWorker<bool>(
473 RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w, this, content,
474 action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000475}
476
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000477void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800478 // We pass in the BaseChannel instead of the rtp_dtls_transport_
479 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000480 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200481 // We pass in the network thread because on that thread connection monitor
482 // will call BaseChannel::GetConnectionStats which must be called on the
483 // network thread.
484 connection_monitor_.reset(
485 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000486 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000487 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000488 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000489}
490
491void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000492 if (connection_monitor_) {
493 connection_monitor_->Stop();
494 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000495 }
496}
497
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000498bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200499 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800500 if (!rtp_dtls_transport_) {
501 return false;
502 }
zhihuangb2cdd932017-01-19 16:54:25 -0800503 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800504}
505
506bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800507 // If this BaseChannel doesn't require RTCP mux and we haven't fully
508 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700509 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000510}
511
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700512bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000513 // Receive data if we are enabled and have local content,
514 return enabled() && IsReceiveContentDirection(local_content_direction_);
515}
516
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700517bool BaseChannel::IsReadyToSendMedia_w() const {
518 // Need to access some state updated on the network thread.
519 return network_thread_->Invoke<bool>(
520 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
521}
522
523bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000524 // Send outgoing data if we are enabled, have local and remote content,
525 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800526 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000527 IsSendContentDirection(local_content_direction_) &&
Zhi Huangcf990f52017-09-22 12:12:30 -0700528 was_ever_writable() && (srtp_active() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000529}
530
jbaucheec21bd2016-03-20 06:15:43 -0700531bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700532 const rtc::PacketOptions& options) {
533 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000534}
535
jbaucheec21bd2016-03-20 06:15:43 -0700536bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700537 const rtc::PacketOptions& options) {
538 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000539}
540
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000541int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000542 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200543 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700544 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200545}
546
547int BaseChannel::SetOption_n(SocketType type,
548 rtc::Socket::Option opt,
549 int value) {
550 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800551 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000552 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000553 case ST_RTP:
zsteine8ab5432017-07-12 11:48:11 -0700554 transport = rtp_transport_->rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700555 socket_options_.push_back(
556 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000557 break;
558 case ST_RTCP:
zsteine8ab5432017-07-12 11:48:11 -0700559 transport = rtp_transport_->rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700560 rtcp_socket_options_.push_back(
561 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000562 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000563 }
deadbeeff5346592017-01-24 21:51:21 -0800564 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000565}
566
deadbeef5bd5ca32017-02-10 11:31:50 -0800567void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteine8ab5432017-07-12 11:48:11 -0700568 RTC_DCHECK(transport == rtp_transport_->rtp_packet_transport() ||
569 transport == rtp_transport_->rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200570 RTC_DCHECK(network_thread_->IsCurrent());
571 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000572}
573
zhihuangb2cdd932017-01-19 16:54:25 -0800574void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800575 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200576 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800577 return;
578 }
579
Zhi Huangcf990f52017-09-22 12:12:30 -0700580 // Reset the SrtpTransport if it's not the CONNECTED state. For the CONNECTED
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800581 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800582 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800583 // TransportChannel) or when TransportChannel is attached after DTLS is
584 // negotiated.
585 if (state != DTLS_TRANSPORT_CONNECTED) {
Zhi Huangcf990f52017-09-22 12:12:30 -0700586 dtls_active_ = false;
587 if (srtp_transport_) {
588 srtp_transport_->ResetParams();
589 }
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800590 }
591}
592
Zhi Huang942bc2e2017-11-13 13:26:07 -0800593void BaseChannel::OnNetworkRouteChanged(
594 rtc::Optional<rtc::NetworkRoute> network_route) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200595 RTC_DCHECK(network_thread_->IsCurrent());
Zhi Huang942bc2e2017-11-13 13:26:07 -0800596 rtc::NetworkRoute new_route;
597 if (network_route) {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800598 new_route = *(network_route);
Zhi Huang8c316c12017-11-13 21:13:45 +0000599 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800600 // Note: When the RTCP-muxing is not enabled, RTCP transport and RTP transport
601 // use the same transport name and MediaChannel::OnNetworkRouteChanged cannot
602 // work correctly. Intentionally leave it broken to simplify the code and
603 // encourage the users to stop using non-muxing RTCP.
Steve Anton8699a322017-11-06 15:53:33 -0800604 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_, [=] {
Zhi Huang942bc2e2017-11-13 13:26:07 -0800605 media_channel_->OnNetworkRouteChanged(transport_name_, new_route);
Steve Anton8699a322017-11-06 15:53:33 -0800606 });
Honghai Zhangcc411c02016-03-29 17:27:21 -0700607}
608
zstein56162b92017-04-24 16:54:35 -0700609void BaseChannel::OnTransportReadyToSend(bool ready) {
Steve Anton8699a322017-11-06 15:53:33 -0800610 invoker_.AsyncInvoke<void>(RTC_FROM_HERE, worker_thread_,
611 [=] { media_channel_->OnReadyToSend(ready); });
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000612}
613
stefanc1aeaf02015-10-15 07:26:07 -0700614bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700615 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700616 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200617 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
618 // If the thread is not our network thread, we will post to our network
619 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000620 // synchronize access to all the pieces of the send path, including
621 // SRTP and the inner workings of the transport channels.
622 // The only downside is that we can't return a proper failure code if
623 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200624 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000625 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200626 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
627 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800628 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700629 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700630 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000631 return true;
632 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200633 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000634
635 // Now that we are on the correct thread, ensure we have a place to send this
636 // packet before doing anything. (We might get RTCP packets that we don't
637 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
638 // transport.
zsteine8ab5432017-07-12 11:48:11 -0700639 if (!rtp_transport_->IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000640 return false;
641 }
642
643 // Protect ourselves against crazy data.
644 if (!ValidPacket(rtcp, packet)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100645 RTC_LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
646 << RtpRtcpStringLiteral(rtcp)
647 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000648 return false;
649 }
650
Zhi Huangcf990f52017-09-22 12:12:30 -0700651 if (!srtp_active()) {
652 if (srtp_required_) {
653 // The audio/video engines may attempt to send RTCP packets as soon as the
654 // streams are created, so don't treat this as an error for RTCP.
655 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
656 if (rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000657 return false;
658 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700659 // However, there shouldn't be any RTP packets sent before SRTP is set up
660 // (and SetSend(true) is called).
Mirko Bonadei675513b2017-11-09 11:09:25 +0100661 RTC_LOG(LS_ERROR)
662 << "Can't send outgoing RTP packet when SRTP is inactive"
663 << " and crypto is required";
Zhi Huangcf990f52017-09-22 12:12:30 -0700664 RTC_NOTREACHED();
deadbeef8f425f92016-12-01 12:26:27 -0800665 return false;
666 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700667 // Bon voyage.
Zhi Huang04eaa152017-10-04 14:08:30 -0700668 return rtcp
669 ? rtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
670 : rtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000671 }
Zhi Huangcf990f52017-09-22 12:12:30 -0700672 RTC_DCHECK(srtp_transport_);
673 RTC_DCHECK(srtp_transport_->IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000674 // Bon voyage.
Zhi Huangcf990f52017-09-22 12:12:30 -0700675 return rtcp ? srtp_transport_->SendRtcpPacket(packet, options, PF_SRTP_BYPASS)
676 : srtp_transport_->SendRtpPacket(packet, options, PF_SRTP_BYPASS);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000677}
678
zstein3dcf0e92017-06-01 13:22:42 -0700679bool BaseChannel::HandlesPayloadType(int packet_type) const {
zsteine8ab5432017-07-12 11:48:11 -0700680 return rtp_transport_->HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000681}
682
zstein3dcf0e92017-06-01 13:22:42 -0700683void BaseChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -0700684 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -0700685 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000686 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000687 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700688 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689 }
690
Zhi Huangcf990f52017-09-22 12:12:30 -0700691 if (!srtp_active() && srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000692 // Our session description indicates that SRTP is required, but we got a
693 // packet before our SRTP filter is active. This means either that
694 // a) we got SRTP packets before we received the SDES keys, in which case
695 // we can't decrypt it anyway, or
696 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800697 // transports, so we haven't yet extracted keys, even if DTLS did
698 // complete on the transport that the packets are being sent on. It's
699 // really good practice to wait for both RTP and RTCP to be good to go
700 // before sending media, to prevent weird failure modes, so it's fine
701 // for us to just eat packets here. This is all sidestepped if RTCP mux
702 // is used anyway.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100703 RTC_LOG(LS_WARNING)
704 << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
705 << " packet when SRTP is inactive and crypto is required";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000706 return;
707 }
708
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200709 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700710 RTC_FROM_HERE, worker_thread_,
zstein634977b2017-07-14 12:30:04 -0700711 Bind(&BaseChannel::ProcessPacket, this, rtcp, *packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200712}
713
zstein3dcf0e92017-06-01 13:22:42 -0700714void BaseChannel::ProcessPacket(bool rtcp,
715 const rtc::CopyOnWriteBuffer& packet,
716 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200717 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700718
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200719 // Need to copy variable because OnRtcpReceived/OnPacketReceived
720 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
721 rtc::CopyOnWriteBuffer data(packet);
722 if (rtcp) {
723 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200725 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000726 }
727}
728
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700730 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000731 if (enabled_)
732 return;
733
Mirko Bonadei675513b2017-11-09 11:09:25 +0100734 RTC_LOG(LS_INFO) << "Channel enabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000735 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700736 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000737}
738
739void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700740 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000741 if (!enabled_)
742 return;
743
Mirko Bonadei675513b2017-11-09 11:09:25 +0100744 RTC_LOG(LS_INFO) << "Channel disabled";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000745 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700746 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000747}
748
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200749void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700750 rtc::PacketTransportInternal* rtp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700751 rtp_transport_->rtp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700752 rtc::PacketTransportInternal* rtcp_packet_transport =
zsteine8ab5432017-07-12 11:48:11 -0700753 rtp_transport_->rtcp_packet_transport();
zsteind48dbda2017-04-04 19:45:57 -0700754 if (rtp_packet_transport && rtp_packet_transport->writable() &&
755 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200756 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700757 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200758 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700759 }
760}
761
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200762void BaseChannel::ChannelWritable_n() {
763 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800764 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000765 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800766 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000767
Mirko Bonadei675513b2017-11-09 11:09:25 +0100768 RTC_LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
769 << (was_ever_writable_ ? "" : " for the first time");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000770
michaelt79e05882016-11-08 02:50:09 -0800771 if (selected_candidate_pair_)
Mirko Bonadei675513b2017-11-09 11:09:25 +0100772 RTC_LOG(LS_INFO)
michaelt79e05882016-11-08 02:50:09 -0800773 << "Using "
774 << selected_candidate_pair_->local_candidate().ToSensitiveString()
775 << "->"
776 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000777
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000778 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200779 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000780 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700781 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000782}
783
deadbeef953c2ce2017-01-09 14:53:41 -0800784void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200785 RTC_DCHECK(network_thread_->IsCurrent());
786 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700787 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800788 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000789}
790
deadbeef953c2ce2017-01-09 14:53:41 -0800791void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700792 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800793 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000794}
795
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200796bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800797 // Since DTLS is applied to all transports, checking RTP should be enough.
798 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000799}
800
801// This function returns true if either DTLS-SRTP is not in use
802// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -0800803bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200804 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000805 bool ret = false;
806
zhihuangb2cdd932017-01-19 16:54:25 -0800807 DtlsTransportInternal* transport =
808 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -0800809 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -0800810 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000811
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800812 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000813
zhihuangb2cdd932017-01-19 16:54:25 -0800814 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100815 RTC_LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000816 return false;
817 }
818
Mirko Bonadei675513b2017-11-09 11:09:25 +0100819 RTC_LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name()
820 << " " << RtpRtcpStringLiteral(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000821
jbauchcb560652016-08-04 05:20:32 -0700822 int key_len;
823 int salt_len;
824 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
825 &salt_len)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100826 RTC_LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite"
827 << selected_crypto_suite;
jbauchcb560652016-08-04 05:20:32 -0700828 return false;
829 }
830
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000831 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700832 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000833
834 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -0800835 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
836 &dtls_buffer[0], dtls_buffer.size())) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100837 RTC_LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800838 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000839 return false;
840 }
841
842 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700843 std::vector<unsigned char> client_write_key(key_len + salt_len);
844 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000845 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700846 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
847 offset += key_len;
848 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
849 offset += key_len;
850 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
851 offset += salt_len;
852 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000853
854 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000855 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -0800856 if (!transport->GetSslRole(&role)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100857 RTC_LOG(LS_WARNING) << "GetSslRole failed";
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000858 return false;
859 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000860
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000861 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000862 send_key = &server_write_key;
863 recv_key = &client_write_key;
864 } else {
865 send_key = &client_write_key;
866 recv_key = &server_write_key;
867 }
868
Zhi Huangc99b6c72017-11-10 16:44:46 -0800869 // Use an empty encrypted header extension ID vector if not set. This could
870 // happen when the DTLS handshake is completed before processing the
871 // Offer/Answer which contains the encrypted header extension IDs.
872 std::vector<int> send_extension_ids;
873 std::vector<int> recv_extension_ids;
874 if (catched_send_extension_ids_) {
875 send_extension_ids = *catched_send_extension_ids_;
876 }
877 if (catched_recv_extension_ids_) {
878 recv_extension_ids = *catched_recv_extension_ids_;
879 }
880
Zhi Huangcf990f52017-09-22 12:12:30 -0700881 if (rtcp) {
882 if (!dtls_active()) {
883 RTC_DCHECK(srtp_transport_);
884 ret = srtp_transport_->SetRtcpParams(
885 selected_crypto_suite, &(*send_key)[0],
Zhi Huangc99b6c72017-11-10 16:44:46 -0800886 static_cast<int>(send_key->size()), send_extension_ids,
887 selected_crypto_suite, &(*recv_key)[0],
888 static_cast<int>(recv_key->size()), recv_extension_ids);
jbauch5869f502017-06-29 12:31:36 -0700889 } else {
Zhi Huangcf990f52017-09-22 12:12:30 -0700890 // RTCP doesn't need to call SetRtpParam because it is only used
891 // to make the updated encrypted RTP header extension IDs take effect.
892 ret = true;
jbauch5869f502017-06-29 12:31:36 -0700893 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000894 } else {
Zhi Huangcf990f52017-09-22 12:12:30 -0700895 RTC_DCHECK(srtp_transport_);
Zhi Huangc99b6c72017-11-10 16:44:46 -0800896 ret = srtp_transport_->SetRtpParams(
897 selected_crypto_suite, &(*send_key)[0],
898 static_cast<int>(send_key->size()), send_extension_ids,
899 selected_crypto_suite, &(*recv_key)[0],
900 static_cast<int>(recv_key->size()), recv_extension_ids);
Zhi Huangcf990f52017-09-22 12:12:30 -0700901 dtls_active_ = ret;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000902 }
903
michaelt79e05882016-11-08 02:50:09 -0800904 if (!ret) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100905 RTC_LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -0800906 }
Zhi Huang942bc2e2017-11-13 13:26:07 -0800907
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000908 return ret;
909}
910
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200911void BaseChannel::MaybeSetupDtlsSrtp_n() {
Zhi Huangcf990f52017-09-22 12:12:30 -0700912 if (dtls_active()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800913 return;
914 }
915
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200916 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800917 return;
918 }
919
Zhi Huangcf990f52017-09-22 12:12:30 -0700920 if (!srtp_transport_) {
921 EnableSrtpTransport_n();
922 }
923
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200924 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -0800925 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800926 return;
927 }
928
zhihuangb2cdd932017-01-19 16:54:25 -0800929 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200930 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -0800931 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800932 return;
933 }
934 }
935}
936
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200937void BaseChannel::ChannelNotWritable_n() {
938 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000939 if (!writable_)
940 return;
941
Mirko Bonadei675513b2017-11-09 11:09:25 +0100942 RTC_LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000943 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700944 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000945}
946
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200947bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700948 const MediaContentDescription* content,
949 ContentAction action,
950 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700951 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700952 std::string* error_desc) {
953 if (action == CA_UPDATE) {
954 // These parameters never get changed by a CA_UDPATE.
955 return true;
956 }
957
jbauch5869f502017-06-29 12:31:36 -0700958 std::vector<int> encrypted_extension_ids;
959 for (const webrtc::RtpExtension& extension : extensions) {
960 if (extension.encrypt) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100961 RTC_LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
962 << " encrypted extension: " << extension.ToString();
jbauch5869f502017-06-29 12:31:36 -0700963 encrypted_extension_ids.push_back(extension.id);
964 }
965 }
966
deadbeef7af91dd2016-12-13 11:29:11 -0800967 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200968 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700969 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
jbauch5869f502017-06-29 12:31:36 -0700970 content, action, src, encrypted_extension_ids,
971 error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200972}
973
974bool BaseChannel::SetRtpTransportParameters_n(
975 const MediaContentDescription* content,
976 ContentAction action,
977 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -0700978 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200979 std::string* error_desc) {
980 RTC_DCHECK(network_thread_->IsCurrent());
981
jbauch5869f502017-06-29 12:31:36 -0700982 if (!SetSrtp_n(content->cryptos(), action, src, encrypted_extension_ids,
983 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700984 return false;
985 }
986
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200987 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700988 return false;
989 }
990
991 return true;
992}
993
zhihuangb2cdd932017-01-19 16:54:25 -0800994// |dtls| will be set to true if DTLS is active for transport and crypto is
995// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200996bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
997 bool* dtls,
998 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -0800999 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001000 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001001 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001002 return false;
1003 }
1004 return true;
1005}
1006
Zhi Huangcf990f52017-09-22 12:12:30 -07001007void BaseChannel::EnableSrtpTransport_n() {
1008 if (srtp_transport_ == nullptr) {
1009 rtp_transport_->SignalReadyToSend.disconnect(this);
1010 rtp_transport_->SignalPacketReceived.disconnect(this);
Zhi Huang942bc2e2017-11-13 13:26:07 -08001011 rtp_transport_->SignalNetworkRouteChanged.disconnect(this);
Zhi Huangcf990f52017-09-22 12:12:30 -07001012
1013 auto transport = rtc::MakeUnique<webrtc::SrtpTransport>(
1014 std::move(rtp_transport_), content_name_);
1015 srtp_transport_ = transport.get();
1016 rtp_transport_ = std::move(transport);
1017
1018 rtp_transport_->SignalReadyToSend.connect(
1019 this, &BaseChannel::OnTransportReadyToSend);
1020 rtp_transport_->SignalPacketReceived.connect(
1021 this, &BaseChannel::OnPacketReceived);
Zhi Huang942bc2e2017-11-13 13:26:07 -08001022 rtp_transport_->SignalNetworkRouteChanged.connect(
1023 this, &BaseChannel::OnNetworkRouteChanged);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001024 RTC_LOG(LS_INFO) << "Wrapping RtpTransport in SrtpTransport.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001025 }
1026}
1027
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001028bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001029 ContentAction action,
1030 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001031 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001032 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001033 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001034 if (action == CA_UPDATE) {
1035 // no crypto params.
1036 return true;
1037 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001038 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001039 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001040 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001041 if (!ret) {
1042 return false;
1043 }
Zhi Huangcf990f52017-09-22 12:12:30 -07001044
1045 // If SRTP was not required, but we're setting a description that uses SDES,
1046 // we need to upgrade to an SrtpTransport.
1047 if (!srtp_transport_ && !dtls && !cryptos.empty()) {
1048 EnableSrtpTransport_n();
1049 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001050
1051 bool encrypted_header_extensions_id_changed =
1052 EncryptedHeaderExtensionIdsChanged(src, encrypted_extension_ids);
1053 CacheEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
1054
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001055 switch (action) {
1056 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001057 // If DTLS is already active on the channel, we could be renegotiating
1058 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001059 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001060 ret = sdes_negotiator_.SetOffer(cryptos, src);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001061 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001062 break;
1063 case CA_PRANSWER:
1064 // If we're doing DTLS-SRTP, we don't want to update the filter
1065 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001066 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001067 ret = sdes_negotiator_.SetProvisionalAnswer(cryptos, src);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001068 }
1069 break;
1070 case CA_ANSWER:
1071 // If we're doing DTLS-SRTP, we don't want to update the filter
1072 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001073 if (!dtls) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001074 ret = sdes_negotiator_.SetAnswer(cryptos, src);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001075 }
1076 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001077 default:
1078 break;
1079 }
Zhi Huangcf990f52017-09-22 12:12:30 -07001080
1081 // If setting an SDES answer succeeded, apply the negotiated parameters
1082 // to the SRTP transport.
1083 if ((action == CA_PRANSWER || action == CA_ANSWER) && !dtls && ret) {
1084 if (sdes_negotiator_.send_cipher_suite() &&
1085 sdes_negotiator_.recv_cipher_suite()) {
Zhi Huangc99b6c72017-11-10 16:44:46 -08001086 RTC_DCHECK(catched_send_extension_ids_);
1087 RTC_DCHECK(catched_recv_extension_ids_);
Zhi Huangcf990f52017-09-22 12:12:30 -07001088 ret = srtp_transport_->SetRtpParams(
1089 *(sdes_negotiator_.send_cipher_suite()),
1090 sdes_negotiator_.send_key().data(),
1091 static_cast<int>(sdes_negotiator_.send_key().size()),
Zhi Huangc99b6c72017-11-10 16:44:46 -08001092 *(catched_send_extension_ids_),
Zhi Huangcf990f52017-09-22 12:12:30 -07001093 *(sdes_negotiator_.recv_cipher_suite()),
1094 sdes_negotiator_.recv_key().data(),
Zhi Huangc99b6c72017-11-10 16:44:46 -08001095 static_cast<int>(sdes_negotiator_.recv_key().size()),
1096 *(catched_recv_extension_ids_));
Zhi Huangcf990f52017-09-22 12:12:30 -07001097 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001098 RTC_LOG(LS_INFO) << "No crypto keys are provided for SDES.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001099 if (action == CA_ANSWER && srtp_transport_) {
1100 // Explicitly reset the |srtp_transport_| if no crypto param is
1101 // provided in the answer. No need to call |ResetParams()| for
1102 // |sdes_negotiator_| because it resets the params inside |SetAnswer|.
1103 srtp_transport_->ResetParams();
1104 }
1105 }
1106 }
1107
Zhi Huangc99b6c72017-11-10 16:44:46 -08001108 // Only update SRTP transport if using DTLS. SDES is handled internally
jbauch5869f502017-06-29 12:31:36 -07001109 // by the SRTP filter.
Zhi Huangcf990f52017-09-22 12:12:30 -07001110 if (ret && dtls_active() && rtp_dtls_transport_ &&
Zhi Huangc99b6c72017-11-10 16:44:46 -08001111 rtp_dtls_transport_->dtls_state() == DTLS_TRANSPORT_CONNECTED &&
1112 encrypted_header_extensions_id_changed) {
1113 ret = SetupDtlsSrtp_n(/*rtcp=*/false);
jbauch5869f502017-06-29 12:31:36 -07001114 }
Zhi Huangc99b6c72017-11-10 16:44:46 -08001115
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001116 if (!ret) {
Zhi Huangc99b6c72017-11-10 16:44:46 -08001117 SafeSetError("Failed to setup SRTP.", error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001118 return false;
1119 }
1120 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001121}
1122
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001123bool BaseChannel::SetRtcpMux_n(bool enable,
1124 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001125 ContentSource src,
1126 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001127 // Provide a more specific error message for the RTCP mux "require" policy
1128 // case.
zstein56162b92017-04-24 16:54:35 -07001129 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001130 SafeSetError(
1131 "rtcpMuxPolicy is 'require', but media description does not "
1132 "contain 'a=rtcp-mux'.",
1133 error_desc);
1134 return false;
1135 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001136 bool ret = false;
1137 switch (action) {
1138 case CA_OFFER:
1139 ret = rtcp_mux_filter_.SetOffer(enable, src);
1140 break;
1141 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001142 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001143 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001144 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1145 break;
1146 case CA_ANSWER:
1147 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1148 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001149 // We permanently activated RTCP muxing; signal that we no longer need
1150 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001151 std::string debug_name =
1152 transport_name_.empty()
Zhi Huang942bc2e2017-11-13 13:26:07 -08001153 ? rtp_transport_->rtp_packet_transport()->transport_name()
zsteind48dbda2017-04-04 19:45:57 -07001154 : transport_name_;
Mirko Bonadei675513b2017-11-09 11:09:25 +01001155 RTC_LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1156 << "; no longer need RTCP transport for "
1157 << debug_name;
zsteine8ab5432017-07-12 11:48:11 -07001158 if (rtp_transport_->rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001159 SetTransport_n(true, nullptr, nullptr);
1160 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001161 }
deadbeef062ce9f2016-08-26 21:42:15 -07001162 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001163 }
1164 break;
1165 case CA_UPDATE:
1166 // No RTCP mux info.
1167 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001168 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001169 default:
1170 break;
1171 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001172 if (!ret) {
1173 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1174 return false;
1175 }
zsteine8ab5432017-07-12 11:48:11 -07001176 rtp_transport_->SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001177 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001178 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1179 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001180 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001181 // If the RTP transport is already writable, then so are we.
zsteine8ab5432017-07-12 11:48:11 -07001182 if (rtp_transport_->rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001183 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001184 }
1185 }
1186
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001187 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001188}
1189
1190bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001191 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001192 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001193}
1194
Peter Boström0c4e06b2015-10-07 12:23:21 +02001195bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001196 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001197 return media_channel()->RemoveRecvStream(ssrc);
1198}
1199
1200bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001201 ContentAction action,
1202 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001203 if (!(action == CA_OFFER || action == CA_ANSWER ||
1204 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001205 return false;
1206
1207 // If this is an update, streams only contain streams that have changed.
1208 if (action == CA_UPDATE) {
1209 for (StreamParamsVec::const_iterator it = streams.begin();
1210 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001211 const StreamParams* existing_stream =
1212 GetStreamByIds(local_streams_, it->groupid, it->id);
1213 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001214 if (media_channel()->AddSendStream(*it)) {
1215 local_streams_.push_back(*it);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001216 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001217 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001218 std::ostringstream desc;
1219 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1220 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001221 return false;
1222 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001223 } else if (existing_stream && !it->has_ssrcs()) {
1224 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001225 std::ostringstream desc;
1226 desc << "Failed to remove send stream with ssrc "
1227 << it->first_ssrc() << ".";
1228 SafeSetError(desc.str(), error_desc);
1229 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001230 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001231 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001232 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001233 RTC_LOG(LS_WARNING) << "Ignore unsupported stream update";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001234 }
1235 }
1236 return true;
1237 }
1238 // Else streams are all the streams we want to send.
1239
1240 // Check for streams that have been removed.
1241 bool ret = true;
1242 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1243 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001244 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001245 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001246 std::ostringstream desc;
1247 desc << "Failed to remove send stream with ssrc "
1248 << it->first_ssrc() << ".";
1249 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001250 ret = false;
1251 }
1252 }
1253 }
1254 // Check for new streams.
1255 for (StreamParamsVec::const_iterator it = streams.begin();
1256 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001257 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001258 if (media_channel()->AddSendStream(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001259 RTC_LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001260 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001261 std::ostringstream desc;
1262 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1263 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001264 ret = false;
1265 }
1266 }
1267 }
1268 local_streams_ = streams;
1269 return ret;
1270}
1271
1272bool BaseChannel::UpdateRemoteStreams_w(
1273 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001274 ContentAction action,
1275 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001276 if (!(action == CA_OFFER || action == CA_ANSWER ||
1277 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001278 return false;
1279
1280 // If this is an update, streams only contain streams that have changed.
1281 if (action == CA_UPDATE) {
1282 for (StreamParamsVec::const_iterator it = streams.begin();
1283 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001284 const StreamParams* existing_stream =
1285 GetStreamByIds(remote_streams_, it->groupid, it->id);
1286 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001287 if (AddRecvStream_w(*it)) {
1288 remote_streams_.push_back(*it);
Mirko Bonadei675513b2017-11-09 11:09:25 +01001289 RTC_LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001290 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001291 std::ostringstream desc;
1292 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1293 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001294 return false;
1295 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001296 } else if (existing_stream && !it->has_ssrcs()) {
1297 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001298 std::ostringstream desc;
1299 desc << "Failed to remove remote stream with ssrc "
1300 << it->first_ssrc() << ".";
1301 SafeSetError(desc.str(), error_desc);
1302 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001303 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001304 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001305 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001306 RTC_LOG(LS_WARNING)
1307 << "Ignore unsupported stream update."
1308 << " Stream exists? " << (existing_stream != nullptr)
1309 << " new stream = " << it->ToString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001310 }
1311 }
1312 return true;
1313 }
1314 // Else streams are all the streams we want to receive.
1315
1316 // Check for streams that have been removed.
1317 bool ret = true;
1318 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1319 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001320 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001321 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001322 std::ostringstream desc;
1323 desc << "Failed to remove remote stream with ssrc "
1324 << it->first_ssrc() << ".";
1325 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001326 ret = false;
1327 }
1328 }
1329 }
1330 // Check for new streams.
1331 for (StreamParamsVec::const_iterator it = streams.begin();
1332 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001333 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001334 if (AddRecvStream_w(*it)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001335 RTC_LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001336 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001337 std::ostringstream desc;
1338 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1339 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001340 ret = false;
1341 }
1342 }
1343 }
1344 remote_streams_ = streams;
1345 return ret;
1346}
1347
jbauch5869f502017-06-29 12:31:36 -07001348RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1349 const RtpHeaderExtensions& extensions) {
1350 if (!rtp_dtls_transport_ ||
1351 !rtp_dtls_transport_->crypto_options()
1352 .enable_encrypted_rtp_header_extensions) {
1353 RtpHeaderExtensions filtered;
1354 auto pred = [](const webrtc::RtpExtension& extension) {
1355 return !extension.encrypt;
1356 };
1357 std::copy_if(extensions.begin(), extensions.end(),
1358 std::back_inserter(filtered), pred);
1359 return filtered;
1360 }
1361
1362 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1363}
1364
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001365void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001366 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001367// Absolute Send Time extension id is used only with external auth,
1368// so do not bother searching for it and making asyncronious call to set
1369// something that is not used.
1370#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001371 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001372 webrtc::RtpExtension::FindHeaderExtensionByUri(
1373 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001374 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001375 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001376 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001377 RTC_FROM_HERE, network_thread_,
1378 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1379 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001380#endif
1381}
1382
1383void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1384 int rtp_abs_sendtime_extn_id) {
Zhi Huangcf990f52017-09-22 12:12:30 -07001385 if (srtp_transport_) {
1386 srtp_transport_->CacheRtpAbsSendTimeHeaderExtension(
1387 rtp_abs_sendtime_extn_id);
1388 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001389 RTC_LOG(LS_WARNING)
1390 << "Trying to cache the Absolute Send Time extension id "
1391 "but the SRTP is not active.";
Zhi Huangcf990f52017-09-22 12:12:30 -07001392 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001393}
1394
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001395void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001396 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001397 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001398 case MSG_SEND_RTP_PACKET:
1399 case MSG_SEND_RTCP_PACKET: {
1400 RTC_DCHECK(network_thread_->IsCurrent());
1401 SendPacketMessageData* data =
1402 static_cast<SendPacketMessageData*>(pmsg->pdata);
1403 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1404 SendPacket(rtcp, &data->packet, data->options);
1405 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001406 break;
1407 }
1408 case MSG_FIRSTPACKETRECEIVED: {
1409 SignalFirstPacketReceived(this);
1410 break;
1411 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001412 }
1413}
1414
zstein3dcf0e92017-06-01 13:22:42 -07001415void BaseChannel::AddHandledPayloadType(int payload_type) {
zsteine8ab5432017-07-12 11:48:11 -07001416 rtp_transport_->AddHandledPayloadType(payload_type);
zstein3dcf0e92017-06-01 13:22:42 -07001417}
1418
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001419void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001420 // Flush all remaining RTCP messages. This should only be called in
1421 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001422 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001423 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001424 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1425 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001426 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1427 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001428 }
1429}
1430
johand89ab142016-10-25 10:50:32 -07001431void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001432 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001433 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001434 RTC_DCHECK(network_thread_->IsCurrent());
1435 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001436 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001437 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1438}
1439
1440void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1441 RTC_DCHECK(worker_thread_->IsCurrent());
1442 SignalSentPacket(sent_packet);
1443}
1444
Zhi Huangc99b6c72017-11-10 16:44:46 -08001445void BaseChannel::CacheEncryptedHeaderExtensionIds(
1446 cricket::ContentSource source,
1447 const std::vector<int>& extension_ids) {
1448 source == ContentSource::CS_LOCAL
1449 ? catched_recv_extension_ids_.emplace(extension_ids)
1450 : catched_send_extension_ids_.emplace(extension_ids);
1451}
1452
1453bool BaseChannel::EncryptedHeaderExtensionIdsChanged(
1454 cricket::ContentSource source,
1455 const std::vector<int>& new_extension_ids) {
1456 if (source == ContentSource::CS_LOCAL) {
1457 return !catched_recv_extension_ids_ ||
1458 (*catched_recv_extension_ids_) != new_extension_ids;
1459 } else {
1460 return !catched_send_extension_ids_ ||
1461 (*catched_send_extension_ids_) != new_extension_ids;
1462 }
1463}
1464
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001465VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1466 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001467 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001468 MediaEngineInterface* media_engine,
Steve Anton8699a322017-11-06 15:53:33 -08001469 std::unique_ptr<VoiceMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001470 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001471 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001472 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001473 : BaseChannel(worker_thread,
1474 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001475 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001476 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001477 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001478 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001479 srtp_required),
Steve Anton8699a322017-11-06 15:53:33 -08001480 media_engine_(media_engine) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001481
1482VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001483 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001484 StopAudioMonitor();
1485 StopMediaMonitor();
1486 // this can't be done in the base class, since it calls a virtual
1487 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001488 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001489}
1490
Peter Boström0c4e06b2015-10-07 12:23:21 +02001491bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001492 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001493 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001494 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001495 return InvokeOnWorker<bool>(
1496 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1497 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001498}
1499
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001500// TODO(juberti): Handle early media the right way. We should get an explicit
1501// ringing message telling us to start playing local ringback, which we cancel
1502// if any early media actually arrives. For now, we do the opposite, which is
1503// to wait 1 second for early media, and start playing local ringback if none
1504// arrives.
1505void VoiceChannel::SetEarlyMedia(bool enable) {
1506 if (enable) {
1507 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001508 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1509 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001510 } else {
1511 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001512 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001513 }
1514}
1515
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001516bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001517 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001518 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001519}
1520
Peter Boström0c4e06b2015-10-07 12:23:21 +02001521bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1522 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001523 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001524 return InvokeOnWorker<bool>(
1525 RTC_FROM_HERE,
1526 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001527}
1528
solenberg4bac9c52015-10-09 02:32:53 -07001529bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001530 return InvokeOnWorker<bool>(
1531 RTC_FROM_HERE,
1532 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001533}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001534
Tommif888bb52015-12-12 01:37:01 +01001535void VoiceChannel::SetRawAudioSink(
1536 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001537 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1538 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001539 // passing. So we invoke to our own little routine that gets a pointer to
1540 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001541 InvokeOnWorker<bool>(RTC_FROM_HERE,
1542 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001543}
1544
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001545webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001546 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001547 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001548}
1549
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001550webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1551 uint32_t ssrc) const {
1552 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001553}
1554
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001555bool VoiceChannel::SetRtpSendParameters(
1556 uint32_t ssrc,
1557 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001558 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001559 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001560 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001561}
1562
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001563bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1564 webrtc::RtpParameters parameters) {
1565 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1566}
1567
1568webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1569 uint32_t ssrc) const {
1570 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001571 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001572 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1573}
1574
1575webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1576 uint32_t ssrc) const {
1577 return media_channel()->GetRtpReceiveParameters(ssrc);
1578}
1579
1580bool VoiceChannel::SetRtpReceiveParameters(
1581 uint32_t ssrc,
1582 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001583 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001584 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001585 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1586}
1587
1588bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1589 webrtc::RtpParameters parameters) {
1590 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001591}
1592
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001593bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001594 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1595 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001596}
1597
hbos8d609f62017-04-10 07:39:05 -07001598std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1599 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001600 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1601}
1602
1603std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1604 RTC_DCHECK(worker_thread()->IsCurrent());
1605 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001606}
1607
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001608void VoiceChannel::StartMediaMonitor(int cms) {
1609 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001610 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001611 media_monitor_->SignalUpdate.connect(
1612 this, &VoiceChannel::OnMediaMonitorUpdate);
1613 media_monitor_->Start(cms);
1614}
1615
1616void VoiceChannel::StopMediaMonitor() {
1617 if (media_monitor_) {
1618 media_monitor_->Stop();
1619 media_monitor_->SignalUpdate.disconnect(this);
1620 media_monitor_.reset();
1621 }
1622}
1623
1624void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001625 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001626 audio_monitor_
1627 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1628 audio_monitor_->Start(cms);
1629}
1630
1631void VoiceChannel::StopAudioMonitor() {
1632 if (audio_monitor_) {
1633 audio_monitor_->Stop();
1634 audio_monitor_.reset();
1635 }
1636}
1637
1638bool VoiceChannel::IsAudioMonitorRunning() const {
1639 return (audio_monitor_.get() != NULL);
1640}
1641
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001642int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001643 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001644}
1645
1646int VoiceChannel::GetOutputLevel_w() {
1647 return media_channel()->GetOutputLevel();
1648}
1649
1650void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1651 media_channel()->GetActiveStreams(actives);
1652}
1653
zstein3dcf0e92017-06-01 13:22:42 -07001654void VoiceChannel::OnPacketReceived(bool rtcp,
zstein634977b2017-07-14 12:30:04 -07001655 rtc::CopyOnWriteBuffer* packet,
zstein3dcf0e92017-06-01 13:22:42 -07001656 const rtc::PacketTime& packet_time) {
1657 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001658 // Set a flag when we've received an RTP packet. If we're waiting for early
1659 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001660 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001661 received_media_ = true;
1662 }
1663}
1664
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001665void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001666 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001667 invoker_.AsyncInvoke<void>(
1668 RTC_FROM_HERE, worker_thread_,
1669 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001670}
1671
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001672void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001673 // Render incoming data if we're the active call, and we have the local
1674 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001675 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001676 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001677
1678 // Send outgoing data if we're the active call, we have the remote content,
1679 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001680 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001681 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001682
Mirko Bonadei675513b2017-11-09 11:09:25 +01001683 RTC_LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001684}
1685
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001686bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001687 ContentAction action,
1688 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001689 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001690 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001691 RTC_LOG(LS_INFO) << "Setting local voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001692
1693 const AudioContentDescription* audio =
1694 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001695 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001696 if (!audio) {
1697 SafeSetError("Can't find audio content in local description.", error_desc);
1698 return false;
1699 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001700
jbauch5869f502017-06-29 12:31:36 -07001701 RtpHeaderExtensions rtp_header_extensions =
1702 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1703
1704 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1705 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001706 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001707 }
1708
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001709 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001710 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001711 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001712 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001713 error_desc);
1714 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001715 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001716 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001717 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001718 }
1719 last_recv_params_ = recv_params;
1720
1721 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1722 // only give it to the media channel once we have a remote
1723 // description too (without a remote description, we won't be able
1724 // to send them anyway).
1725 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1726 SafeSetError("Failed to set local audio description streams.", error_desc);
1727 return false;
1728 }
1729
1730 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001731 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001732 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001733}
1734
1735bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001736 ContentAction action,
1737 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001738 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001739 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001740 RTC_LOG(LS_INFO) << "Setting remote voice description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001741
1742 const AudioContentDescription* audio =
1743 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001744 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001745 if (!audio) {
1746 SafeSetError("Can't find audio content in remote description.", error_desc);
1747 return false;
1748 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001749
jbauch5869f502017-06-29 12:31:36 -07001750 RtpHeaderExtensions rtp_header_extensions =
1751 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1752
1753 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1754 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001755 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001756 }
1757
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001758 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001759 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1760 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001761 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001762 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001763 }
skvladdc1c62c2016-03-16 19:07:43 -07001764
1765 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1766 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001767 SafeSetError("Failed to set remote audio description send parameters.",
1768 error_desc);
1769 return false;
1770 }
1771 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001772
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001773 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1774 // and only give it to the media channel once we have a local
1775 // description too (without a local description, we won't be able to
1776 // recv them anyway).
1777 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1778 SafeSetError("Failed to set remote audio description streams.", error_desc);
1779 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001780 }
1781
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001782 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001783 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001784 }
1785
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001786 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001787 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001788 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001789}
1790
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001791void VoiceChannel::HandleEarlyMediaTimeout() {
1792 // This occurs on the main thread, not the worker thread.
1793 if (!received_media_) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001794 RTC_LOG(LS_INFO) << "No early media received before timeout";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001795 SignalEarlyMediaTimeout(this);
1796 }
1797}
1798
Peter Boström0c4e06b2015-10-07 12:23:21 +02001799bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1800 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001801 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001802 if (!enabled()) {
1803 return false;
1804 }
solenberg1d63dd02015-12-02 12:35:09 -08001805 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001806}
1807
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001808void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001809 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001810 case MSG_EARLYMEDIATIMEOUT:
1811 HandleEarlyMediaTimeout();
1812 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001813 case MSG_CHANNEL_ERROR: {
1814 VoiceChannelErrorMessageData* data =
1815 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001816 delete data;
1817 break;
1818 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001819 default:
1820 BaseChannel::OnMessage(pmsg);
1821 break;
1822 }
1823}
1824
1825void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001826 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827 SignalConnectionMonitor(this, infos);
1828}
1829
1830void VoiceChannel::OnMediaMonitorUpdate(
1831 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001832 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001833 SignalMediaMonitor(this, info);
1834}
1835
1836void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1837 const AudioInfo& info) {
1838 SignalAudioMonitor(this, info);
1839}
1840
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001841VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1842 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001843 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001844 std::unique_ptr<VideoMediaChannel> media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001845 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001846 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001847 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001848 : BaseChannel(worker_thread,
1849 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001850 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08001851 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07001852 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001853 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001854 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001855
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001856VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001857 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001858 StopMediaMonitor();
1859 // this can't be done in the base class, since it calls a virtual
1860 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001861
1862 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001863}
1864
nisse08582ff2016-02-04 01:24:52 -08001865bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001866 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001867 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001868 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001869 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001870 return true;
1871}
1872
deadbeef5a4a75a2016-06-02 16:23:38 -07001873bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001874 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001875 bool mute,
1876 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001877 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001878 return InvokeOnWorker<bool>(
1879 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1880 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001881}
1882
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001883webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001884 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001885 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001886}
1887
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001888webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1889 uint32_t ssrc) const {
1890 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001891}
1892
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001893bool VideoChannel::SetRtpSendParameters(
1894 uint32_t ssrc,
1895 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001896 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001897 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001898 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001899}
1900
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001901bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1902 webrtc::RtpParameters parameters) {
1903 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1904}
1905
1906webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1907 uint32_t ssrc) const {
1908 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001909 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001910 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1911}
1912
1913webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1914 uint32_t ssrc) const {
1915 return media_channel()->GetRtpReceiveParameters(ssrc);
1916}
1917
1918bool VideoChannel::SetRtpReceiveParameters(
1919 uint32_t ssrc,
1920 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001921 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001922 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001923 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1924}
1925
1926bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1927 webrtc::RtpParameters parameters) {
1928 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001929}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001930
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001931void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001932 // Send outgoing data if we're the active call, we have the remote content,
1933 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001934 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001935 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01001936 RTC_LOG(LS_ERROR) << "Failed to SetSend on video channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001937 // TODO(gangji): Report error back to server.
1938 }
1939
Mirko Bonadei675513b2017-11-09 11:09:25 +01001940 RTC_LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001941}
1942
stefanf79ade12017-06-02 06:44:03 -07001943void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
1944 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
1945 media_channel(), bwe_info));
1946}
1947
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001948bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001949 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1950 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001951}
1952
1953void VideoChannel::StartMediaMonitor(int cms) {
1954 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001955 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001956 media_monitor_->SignalUpdate.connect(
1957 this, &VideoChannel::OnMediaMonitorUpdate);
1958 media_monitor_->Start(cms);
1959}
1960
1961void VideoChannel::StopMediaMonitor() {
1962 if (media_monitor_) {
1963 media_monitor_->Stop();
1964 media_monitor_.reset();
1965 }
1966}
1967
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001968bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001969 ContentAction action,
1970 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001971 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001972 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01001973 RTC_LOG(LS_INFO) << "Setting local video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001974
1975 const VideoContentDescription* video =
1976 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001977 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001978 if (!video) {
1979 SafeSetError("Can't find video content in local description.", error_desc);
1980 return false;
1981 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001982
jbauch5869f502017-06-29 12:31:36 -07001983 RtpHeaderExtensions rtp_header_extensions =
1984 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
1985
1986 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1987 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001988 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001989 }
1990
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001991 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001992 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001993 if (!media_channel()->SetRecvParameters(recv_params)) {
1994 SafeSetError("Failed to set local video description recv parameters.",
1995 error_desc);
1996 return false;
1997 }
1998 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001999 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002000 }
2001 last_recv_params_ = recv_params;
2002
2003 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2004 // only give it to the media channel once we have a remote
2005 // description too (without a remote description, we won't be able
2006 // to send them anyway).
2007 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2008 SafeSetError("Failed to set local video description streams.", error_desc);
2009 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002010 }
2011
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002012 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002013 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002014 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002015}
2016
2017bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002018 ContentAction action,
2019 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002020 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002021 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01002022 RTC_LOG(LS_INFO) << "Setting remote video description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002023
2024 const VideoContentDescription* video =
2025 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002026 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002027 if (!video) {
2028 SafeSetError("Can't find video content in remote description.", error_desc);
2029 return false;
2030 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002031
jbauch5869f502017-06-29 12:31:36 -07002032 RtpHeaderExtensions rtp_header_extensions =
2033 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2034
2035 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2036 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002037 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002038 }
2039
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002040 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002041 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
2042 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002043 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002044 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002045 }
skvladdc1c62c2016-03-16 19:07:43 -07002046
2047 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2048
2049 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002050 SafeSetError("Failed to set remote video description send parameters.",
2051 error_desc);
2052 return false;
2053 }
2054 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002055
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002056 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2057 // and only give it to the media channel once we have a local
2058 // description too (without a local description, we won't be able to
2059 // recv them anyway).
2060 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2061 SafeSetError("Failed to set remote video description streams.", error_desc);
2062 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002063 }
2064
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002065 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07002066 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002067 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002068
2069 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002070 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002071 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002072}
2073
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002074void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002075 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002076 case MSG_CHANNEL_ERROR: {
2077 const VideoChannelErrorMessageData* data =
2078 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002079 delete data;
2080 break;
2081 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002082 default:
2083 BaseChannel::OnMessage(pmsg);
2084 break;
2085 }
2086}
2087
2088void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002089 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002090 SignalConnectionMonitor(this, infos);
2091}
2092
2093// TODO(pthatcher): Look into removing duplicate code between
2094// audio, video, and data, perhaps by using templates.
2095void VideoChannel::OnMediaMonitorUpdate(
2096 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002097 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002098 SignalMediaMonitor(this, info);
2099}
2100
deadbeef953c2ce2017-01-09 14:53:41 -08002101RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2102 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002103 rtc::Thread* signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08002104 std::unique_ptr<DataMediaChannel> media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002105 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002106 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002107 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002108 : BaseChannel(worker_thread,
2109 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002110 signaling_thread,
Steve Anton8699a322017-11-06 15:53:33 -08002111 std::move(media_channel),
deadbeefcbecd352015-09-23 11:50:27 -07002112 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002113 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002114 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002115
deadbeef953c2ce2017-01-09 14:53:41 -08002116RtpDataChannel::~RtpDataChannel() {
2117 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002118 StopMediaMonitor();
2119 // this can't be done in the base class, since it calls a virtual
2120 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002121
2122 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002123}
2124
Steve Anton8699a322017-11-06 15:53:33 -08002125void RtpDataChannel::Init_w(
deadbeeff5346592017-01-24 21:51:21 -08002126 DtlsTransportInternal* rtp_dtls_transport,
2127 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002128 rtc::PacketTransportInternal* rtp_packet_transport,
2129 rtc::PacketTransportInternal* rtcp_packet_transport) {
Steve Anton8699a322017-11-06 15:53:33 -08002130 BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2131 rtp_packet_transport, rtcp_packet_transport);
2132
deadbeef953c2ce2017-01-09 14:53:41 -08002133 media_channel()->SignalDataReceived.connect(this,
2134 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002135 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002136 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002137}
2138
deadbeef953c2ce2017-01-09 14:53:41 -08002139bool RtpDataChannel::SendData(const SendDataParams& params,
2140 const rtc::CopyOnWriteBuffer& payload,
2141 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07002142 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002143 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2144 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002145}
2146
deadbeef953c2ce2017-01-09 14:53:41 -08002147bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002148 const DataContentDescription* content,
2149 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002150 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2151 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002152 // It's been set before, but doesn't match. That's bad.
2153 if (is_sctp) {
2154 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2155 error_desc);
2156 return false;
2157 }
2158 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002159}
2160
deadbeef953c2ce2017-01-09 14:53:41 -08002161bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2162 ContentAction action,
2163 std::string* error_desc) {
2164 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002165 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
Mirko Bonadei675513b2017-11-09 11:09:25 +01002166 RTC_LOG(LS_INFO) << "Setting local data description";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002167
2168 const DataContentDescription* data =
2169 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002170 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002171 if (!data) {
2172 SafeSetError("Can't find data content in local description.", error_desc);
2173 return false;
2174 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002175
deadbeef953c2ce2017-01-09 14:53:41 -08002176 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002177 return false;
2178 }
2179
jbauch5869f502017-06-29 12:31:36 -07002180 RtpHeaderExtensions rtp_header_extensions =
2181 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2182
2183 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2184 rtp_header_extensions, error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08002185 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002186 }
2187
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002188 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002189 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002190 if (!media_channel()->SetRecvParameters(recv_params)) {
2191 SafeSetError("Failed to set remote data description recv parameters.",
2192 error_desc);
2193 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002194 }
deadbeef953c2ce2017-01-09 14:53:41 -08002195 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002196 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002197 }
2198 last_recv_params_ = recv_params;
2199
2200 // TODO(pthatcher): Move local streams into DataSendParameters, and
2201 // only give it to the media channel once we have a remote
2202 // description too (without a remote description, we won't be able
2203 // to send them anyway).
2204 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2205 SafeSetError("Failed to set local data description streams.", error_desc);
2206 return false;
2207 }
2208
2209 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002210 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002211 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002212}
2213
deadbeef953c2ce2017-01-09 14:53:41 -08002214bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2215 ContentAction action,
2216 std::string* error_desc) {
2217 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002218 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002219
2220 const DataContentDescription* data =
2221 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002222 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002223 if (!data) {
2224 SafeSetError("Can't find data content in remote description.", error_desc);
2225 return false;
2226 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002227
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002228 // If the remote data doesn't have codecs and isn't an update, it
2229 // must be empty, so ignore it.
2230 if (!data->has_codecs() && action != CA_UPDATE) {
2231 return true;
2232 }
2233
deadbeef953c2ce2017-01-09 14:53:41 -08002234 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002235 return false;
2236 }
2237
jbauch5869f502017-06-29 12:31:36 -07002238 RtpHeaderExtensions rtp_header_extensions =
2239 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2240
Mirko Bonadei675513b2017-11-09 11:09:25 +01002241 RTC_LOG(LS_INFO) << "Setting remote data description";
jbauch5869f502017-06-29 12:31:36 -07002242 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2243 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002244 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002245 }
2246
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002247 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002248 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
2249 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002250 if (!media_channel()->SetSendParameters(send_params)) {
2251 SafeSetError("Failed to set remote data description send parameters.",
2252 error_desc);
2253 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002254 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002255 last_send_params_ = send_params;
2256
2257 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2258 // and only give it to the media channel once we have a local
2259 // description too (without a local description, we won't be able to
2260 // recv them anyway).
2261 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2262 SafeSetError("Failed to set remote data description streams.",
2263 error_desc);
2264 return false;
2265 }
2266
2267 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002268 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002269 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002270}
2271
deadbeef953c2ce2017-01-09 14:53:41 -08002272void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002273 // Render incoming data if we're the active call, and we have the local
2274 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002275 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002276 if (!media_channel()->SetReceive(recv)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002277 RTC_LOG(LS_ERROR) << "Failed to SetReceive on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002278 }
2279
2280 // Send outgoing data if we're the active call, we have the remote content,
2281 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002282 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002283 if (!media_channel()->SetSend(send)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +01002284 RTC_LOG(LS_ERROR) << "Failed to SetSend on data channel";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002285 }
2286
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002287 // Trigger SignalReadyToSendData asynchronously.
2288 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002289
Mirko Bonadei675513b2017-11-09 11:09:25 +01002290 RTC_LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002291}
2292
deadbeef953c2ce2017-01-09 14:53:41 -08002293void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002294 switch (pmsg->message_id) {
2295 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002296 DataChannelReadyToSendMessageData* data =
2297 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002298 ready_to_send_data_ = data->data();
2299 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002300 delete data;
2301 break;
2302 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002303 case MSG_DATARECEIVED: {
2304 DataReceivedMessageData* data =
2305 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002306 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002307 delete data;
2308 break;
2309 }
2310 case MSG_CHANNEL_ERROR: {
2311 const DataChannelErrorMessageData* data =
2312 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002313 delete data;
2314 break;
2315 }
2316 default:
2317 BaseChannel::OnMessage(pmsg);
2318 break;
2319 }
2320}
2321
deadbeef953c2ce2017-01-09 14:53:41 -08002322void RtpDataChannel::OnConnectionMonitorUpdate(
2323 ConnectionMonitor* monitor,
2324 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002325 SignalConnectionMonitor(this, infos);
2326}
2327
deadbeef953c2ce2017-01-09 14:53:41 -08002328void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002329 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002330 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002331 media_monitor_->SignalUpdate.connect(this,
2332 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002333 media_monitor_->Start(cms);
2334}
2335
deadbeef953c2ce2017-01-09 14:53:41 -08002336void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002337 if (media_monitor_) {
2338 media_monitor_->Stop();
2339 media_monitor_->SignalUpdate.disconnect(this);
2340 media_monitor_.reset();
2341 }
2342}
2343
deadbeef953c2ce2017-01-09 14:53:41 -08002344void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2345 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002346 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002347 SignalMediaMonitor(this, info);
2348}
2349
deadbeef953c2ce2017-01-09 14:53:41 -08002350void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2351 const char* data,
2352 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002353 DataReceivedMessageData* msg = new DataReceivedMessageData(
2354 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002355 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002356}
2357
deadbeef953c2ce2017-01-09 14:53:41 -08002358void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2359 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002360 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2361 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002362 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002363}
2364
deadbeef953c2ce2017-01-09 14:53:41 -08002365void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002366 // This is usded for congestion control to indicate that the stream is ready
2367 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2368 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002369 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002370 new DataChannelReadyToSendMessageData(writable));
2371}
2372
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002373} // namespace cricket