blob: d45e220d5c7843b9eab9099d7ccdd58d6b4e9dc6 [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
kwiberg0eb15ed2015-12-17 03:04:15 -080011#include <utility>
12
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010013#include "webrtc/pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000014
kjellandera69d9732016-08-31 07:33:05 -070015#include "webrtc/api/call/audio_sink.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000016#include "webrtc/base/bind.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000017#include "webrtc/base/byteorder.h"
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -070018#include "webrtc/base/checks.h"
jbaucheec21bd2016-03-20 06:15:43 -070019#include "webrtc/base/copyonwritebuffer.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000020#include "webrtc/base/dscp.h"
21#include "webrtc/base/logging.h"
Honghai Zhangcc411c02016-03-29 17:27:21 -070022#include "webrtc/base/networkroute.h"
Peter Boström6f28cf02015-12-07 23:17:15 +010023#include "webrtc/base/trace_event.h"
kjellanderf4752772016-03-02 05:42:30 -080024#include "webrtc/media/base/mediaconstants.h"
kjellandera96e2d72016-02-04 23:52:28 -080025#include "webrtc/media/base/rtputils.h"
hbos8d609f62017-04-10 07:39:05 -070026#include "webrtc/media/engine/webrtcvoiceengine.h"
deadbeef5bd5ca32017-02-10 11:31:50 -080027#include "webrtc/p2p/base/packettransportinternal.h"
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010028#include "webrtc/pc/channelmanager.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000029
30namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000031using rtc::Bind;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000032
deadbeef2d110be2016-01-13 12:00:26 -080033namespace {
kwiberg31022942016-03-11 14:18:21 -080034// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080035bool SetRawAudioSink_w(VoiceMediaChannel* channel,
36 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080037 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
38 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080039 return true;
40}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020041
42struct SendPacketMessageData : public rtc::MessageData {
43 rtc::CopyOnWriteBuffer packet;
44 rtc::PacketOptions options;
45};
46
isheriff6f8d6862016-05-26 11:24:55 -070047#if defined(ENABLE_EXTERNAL_AUTH)
48// Returns the named header extension if found among all extensions,
49// nullptr otherwise.
50const webrtc::RtpExtension* FindHeaderExtension(
51 const std::vector<webrtc::RtpExtension>& extensions,
52 const std::string& uri) {
53 for (const auto& extension : extensions) {
54 if (extension.uri == uri)
55 return &extension;
56 }
57 return nullptr;
58}
59#endif
60
deadbeef2d110be2016-01-13 12:00:26 -080061} // namespace
62
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000064 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020065 MSG_SEND_RTP_PACKET,
66 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000067 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000068 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000069 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000070 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000071};
72
73// Value specified in RFC 5764.
74static const char kDtlsSrtpExporterLabel[] = "EXTRACTOR-dtls_srtp";
75
76static const int kAgcMinus10db = -10;
77
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000078static void SafeSetError(const std::string& message, std::string* error_desc) {
79 if (error_desc) {
80 *error_desc = message;
81 }
82}
83
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000084struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020085 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000086 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020087 : ssrc(in_ssrc), error(in_error) {}
88 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000089 VoiceMediaChannel::Error error;
90};
91
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000092struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020093 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000094 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020095 : ssrc(in_ssrc), error(in_error) {}
96 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000097 VideoMediaChannel::Error error;
98};
99
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000100struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200101 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +0200103 : ssrc(in_ssrc), error(in_error) {}
104 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000105 DataMediaChannel::Error error;
106};
107
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000108static const char* PacketType(bool rtcp) {
109 return (!rtcp) ? "RTP" : "RTCP";
110}
111
jbaucheec21bd2016-03-20 06:15:43 -0700112static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000113 // Check the packet size. We could check the header too if needed.
114 return (packet &&
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000115 packet->size() >= (!rtcp ? kMinRtpPacketLen : kMinRtcpPacketLen) &&
116 packet->size() <= kMaxRtpPacketLen);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000117}
118
119static bool IsReceiveContentDirection(MediaContentDirection direction) {
120 return direction == MD_SENDRECV || direction == MD_RECVONLY;
121}
122
123static bool IsSendContentDirection(MediaContentDirection direction) {
124 return direction == MD_SENDRECV || direction == MD_SENDONLY;
125}
126
127static const MediaContentDescription* GetContentDescription(
128 const ContentInfo* cinfo) {
129 if (cinfo == NULL)
130 return NULL;
131 return static_cast<const MediaContentDescription*>(cinfo->description);
132}
133
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700134template <class Codec>
135void RtpParametersFromMediaDescription(
136 const MediaContentDescriptionImpl<Codec>* desc,
137 RtpParameters<Codec>* params) {
138 // TODO(pthatcher): Remove this once we're sure no one will give us
139 // a description without codecs (currently a CA_UPDATE with just
140 // streams can).
141 if (desc->has_codecs()) {
142 params->codecs = desc->codecs();
143 }
144 // TODO(pthatcher): See if we really need
145 // rtp_header_extensions_set() and remove it if we don't.
146 if (desc->rtp_header_extensions_set()) {
147 params->extensions = desc->rtp_header_extensions();
148 }
deadbeef13871492015-12-09 12:37:51 -0800149 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700150}
151
nisse05103312016-03-16 02:22:50 -0700152template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700153void RtpSendParametersFromMediaDescription(
154 const MediaContentDescriptionImpl<Codec>* desc,
nisse05103312016-03-16 02:22:50 -0700155 RtpSendParameters<Codec>* send_params) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700156 RtpParametersFromMediaDescription(desc, send_params);
157 send_params->max_bandwidth_bps = desc->bandwidth();
158}
159
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200160BaseChannel::BaseChannel(rtc::Thread* worker_thread,
161 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800162 rtc::Thread* signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -0700163 MediaChannel* media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700164 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800165 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800166 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200167 : worker_thread_(worker_thread),
168 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800169 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000170 content_name_(content_name),
zsteind48dbda2017-04-04 19:45:57 -0700171 rtp_transport_(rtcp_mux_required),
deadbeef7af91dd2016-12-13 11:29:11 -0800172 srtp_required_(srtp_required),
michaelt79e05882016-11-08 02:50:09 -0800173 media_channel_(media_channel),
174 selected_candidate_pair_(nullptr) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700175 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
jbauchdfcab722017-03-06 00:14:10 -0800176#if defined(ENABLE_EXTERNAL_AUTH)
177 srtp_filter_.EnableExternalAuth();
178#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000179 LOG(LS_INFO) << "Created channel for " << content_name;
180}
181
182BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800183 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700184 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
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.
193 delete media_channel_;
zhihuangf5b251b2017-01-12 19:37:48 -0800194 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_);
zsteind48dbda2017-04-04 19:45:57 -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_);
zsteind48dbda2017-04-04 19:45:57 -0700210 } else if (rtp_transport_.rtcp_packet_transport()) {
211 DisconnectFromPacketTransport(rtp_transport_.rtcp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200212 }
213
214 // Clear pending read packets/messages.
215 network_thread_->Clear(&invoker_);
216 network_thread_->Clear(this);
217}
218
zhihuangb2cdd932017-01-19 16:54:25 -0800219bool BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800220 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800221 rtc::PacketTransportInternal* rtp_packet_transport,
222 rtc::PacketTransportInternal* rtcp_packet_transport) {
skvlad6c87a672016-05-17 17:49:52 -0700223 if (!network_thread_->Invoke<bool>(
zhihuangb2cdd932017-01-19 16:54:25 -0800224 RTC_FROM_HERE, Bind(&BaseChannel::InitNetwork_n, this,
deadbeeff5346592017-01-24 21:51:21 -0800225 rtp_dtls_transport, rtcp_dtls_transport,
226 rtp_packet_transport, rtcp_packet_transport))) {
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000227 return false;
228 }
deadbeeff5346592017-01-24 21:51:21 -0800229 // Both RTP and RTCP channels should be set, we can call SetInterface on
230 // the media channel and it can set network options.
231 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.orgde305012013-10-31 15:40:38 +0000232 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000233 return true;
234}
235
deadbeeff5346592017-01-24 21:51:21 -0800236bool BaseChannel::InitNetwork_n(
237 DtlsTransportInternal* rtp_dtls_transport,
238 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800239 rtc::PacketTransportInternal* rtp_packet_transport,
240 rtc::PacketTransportInternal* rtcp_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200241 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800242 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport, rtp_packet_transport,
243 rtcp_packet_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200244
deadbeeff5346592017-01-24 21:51:21 -0800245 if (rtp_dtls_transport_ &&
246 !SetDtlsSrtpCryptoSuites_n(rtp_dtls_transport_, false)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200247 return false;
248 }
zhihuangb2cdd932017-01-19 16:54:25 -0800249 if (rtcp_dtls_transport_ &&
250 !SetDtlsSrtpCryptoSuites_n(rtcp_dtls_transport_, true)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200251 return false;
252 }
zsteind48dbda2017-04-04 19:45:57 -0700253 if (rtp_transport_.rtcp_mux_required()) {
deadbeefac22f702017-01-12 21:59:29 -0800254 rtcp_mux_filter_.SetActive();
255 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200256 return true;
257}
258
wu@webrtc.org78187522013-10-07 23:32:02 +0000259void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200260 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000261 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200262 // Packets arrive on the network thread, processing packets calls virtual
263 // functions, so need to stop this process in Deinit that is called in
264 // derived classes destructor.
265 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700266 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000267}
268
zhihuangb2cdd932017-01-19 16:54:25 -0800269void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
270 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800271 network_thread_->Invoke<void>(
272 RTC_FROM_HERE,
273 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
274 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000275}
276
deadbeeff5346592017-01-24 21:51:21 -0800277void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800278 rtc::PacketTransportInternal* rtp_packet_transport,
279 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800280 network_thread_->Invoke<void>(
281 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
282 rtp_packet_transport, rtcp_packet_transport));
283}
zhihuangf5b251b2017-01-12 19:37:48 -0800284
deadbeeff5346592017-01-24 21:51:21 -0800285void BaseChannel::SetTransports_n(
286 DtlsTransportInternal* rtp_dtls_transport,
287 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800288 rtc::PacketTransportInternal* rtp_packet_transport,
289 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800290 RTC_DCHECK(network_thread_->IsCurrent());
291 // Validate some assertions about the input.
292 RTC_DCHECK(rtp_packet_transport);
293 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
294 if (rtp_dtls_transport || rtcp_dtls_transport) {
295 // DTLS/non-DTLS pointers should be to the same object.
296 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
297 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
298 // Can't go from non-DTLS to DTLS.
zsteind48dbda2017-04-04 19:45:57 -0700299 RTC_DCHECK(!rtp_transport_.rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800300 } else {
301 // Can't go from DTLS to non-DTLS.
302 RTC_DCHECK(!rtp_dtls_transport_);
303 }
304 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800305 if (rtp_dtls_transport && rtcp_dtls_transport) {
306 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
307 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800308 }
deadbeeff5346592017-01-24 21:51:21 -0800309 std::string debug_name;
310 if (rtp_dtls_transport) {
311 transport_name_ = rtp_dtls_transport->transport_name();
312 debug_name = transport_name_;
313 } else {
314 debug_name = rtp_packet_transport->debug_name();
315 }
zsteind48dbda2017-04-04 19:45:57 -0700316 if (rtp_packet_transport == rtp_transport_.rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800317 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800318 return;
deadbeefcbecd352015-09-23 11:50:27 -0700319 }
320
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800321 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
322 // changes and wait until the DTLS handshake is complete to set the newly
323 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200324 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800325 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700326 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800327 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800328 srtp_filter_.ResetParams();
329 }
330
deadbeefac22f702017-01-12 21:59:29 -0800331 // If this BaseChannel doesn't require RTCP mux and we haven't fully
332 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800333 if (rtcp_packet_transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800334 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
deadbeeff5346592017-01-24 21:51:21 -0800335 << debug_name << " transport " << rtcp_packet_transport;
336 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000337 }
338
deadbeeff5346592017-01-24 21:51:21 -0800339 LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
340 << debug_name << " transport " << rtp_packet_transport;
341 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800342
deadbeefcbecd352015-09-23 11:50:27 -0700343 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700344 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200345 UpdateWritableState_n();
deadbeef062ce9f2016-08-26 21:42:15 -0700346 // We can only update ready-to-send after updating writability.
347 //
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700348 // On setting a new channel, assume it's ready to send if it's writable,
349 // because we have no way of knowing otherwise (the channel doesn't give us
350 // "was last send successful?").
351 //
352 // This won't always be accurate (the last SendPacket call from another
353 // BaseChannel could have resulted in an error), but even so, we'll just
354 // encounter the error again and update "ready to send" accordingly.
deadbeef062ce9f2016-08-26 21:42:15 -0700355 SetTransportChannelReadyToSend(
zsteind48dbda2017-04-04 19:45:57 -0700356 false, rtp_packet_transport && rtp_packet_transport->writable());
zhihuangb2cdd932017-01-19 16:54:25 -0800357 SetTransportChannelReadyToSend(
zsteind48dbda2017-04-04 19:45:57 -0700358 true, rtcp_packet_transport && rtcp_packet_transport->writable());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000359}
360
deadbeeff5346592017-01-24 21:51:21 -0800361void BaseChannel::SetTransport_n(
362 bool rtcp,
363 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800364 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200365 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800366 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800367 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700368 rtc::PacketTransportInternal* old_packet_transport =
369 rtcp ? rtp_transport_.rtcp_packet_transport()
370 : rtp_transport_.rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800371
deadbeeff5346592017-01-24 21:51:21 -0800372 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700373 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000374 return;
375 }
zhihuangb2cdd932017-01-19 16:54:25 -0800376
deadbeeff5346592017-01-24 21:51:21 -0800377 RTC_DCHECK(old_packet_transport != new_packet_transport);
378 if (old_dtls_transport) {
379 DisconnectFromDtlsTransport(old_dtls_transport);
380 } else if (old_packet_transport) {
381 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000382 }
383
zsteind48dbda2017-04-04 19:45:57 -0700384 if (rtcp) {
385 rtp_transport_.set_rtcp_packet_transport(new_packet_transport);
386 } else {
387 rtp_transport_.set_rtp_packet_transport(new_packet_transport);
388 }
deadbeeff5346592017-01-24 21:51:21 -0800389 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000390
deadbeeff5346592017-01-24 21:51:21 -0800391 // If there's no new transport, we're done after disconnecting from old one.
392 if (!new_packet_transport) {
393 return;
394 }
395
396 if (rtcp && new_dtls_transport) {
397 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
398 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
399 << "should never happen.";
400 }
401 if (new_dtls_transport) {
402 ConnectToDtlsTransport(new_dtls_transport);
403 } else {
404 ConnectToPacketTransport(new_packet_transport);
405 }
406 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
407 for (const auto& pair : socket_options) {
408 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800409 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000410}
411
deadbeeff5346592017-01-24 21:51:21 -0800412void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200413 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000414
zhihuangb2cdd932017-01-19 16:54:25 -0800415 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
416 transport->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
417 transport->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
418 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
419 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
420 transport->ice_transport()->SignalSelectedCandidatePairChanged.connect(
Honghai Zhangcc411c02016-03-29 17:27:21 -0700421 this, &BaseChannel::OnSelectedCandidatePairChanged);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000422}
423
deadbeeff5346592017-01-24 21:51:21 -0800424void BaseChannel::DisconnectFromDtlsTransport(
425 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200426 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800427 OnSelectedCandidatePairChanged(transport->ice_transport(), nullptr, -1,
428 false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000429
zhihuangb2cdd932017-01-19 16:54:25 -0800430 transport->SignalWritableState.disconnect(this);
431 transport->SignalReadPacket.disconnect(this);
432 transport->SignalReadyToSend.disconnect(this);
433 transport->SignalDtlsState.disconnect(this);
434 transport->SignalSentPacket.disconnect(this);
435 transport->ice_transport()->SignalSelectedCandidatePairChanged.disconnect(
436 this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000437}
438
deadbeeff5346592017-01-24 21:51:21 -0800439void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800440 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800441 RTC_DCHECK_RUN_ON(network_thread_);
442 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
443 transport->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
444 transport->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
445 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
446}
447
448void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800449 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800450 RTC_DCHECK_RUN_ON(network_thread_);
451 transport->SignalWritableState.disconnect(this);
452 transport->SignalReadPacket.disconnect(this);
453 transport->SignalReadyToSend.disconnect(this);
454 transport->SignalSentPacket.disconnect(this);
455}
456
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000457bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700458 worker_thread_->Invoke<void>(
459 RTC_FROM_HERE,
460 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
461 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000462 return true;
463}
464
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000465bool BaseChannel::AddRecvStream(const StreamParams& sp) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700466 return InvokeOnWorker(RTC_FROM_HERE,
467 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000468}
469
Peter Boström0c4e06b2015-10-07 12:23:21 +0200470bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700471 return InvokeOnWorker(RTC_FROM_HERE,
472 Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000473}
474
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000475bool BaseChannel::AddSendStream(const StreamParams& sp) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +0000476 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700477 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000478}
479
Peter Boström0c4e06b2015-10-07 12:23:21 +0200480bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700481 return InvokeOnWorker(RTC_FROM_HERE, Bind(&MediaChannel::RemoveSendStream,
482 media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000483}
484
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000485bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000486 ContentAction action,
487 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100488 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700489 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetLocalContent_w,
490 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491}
492
493bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000494 ContentAction action,
495 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100496 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700497 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w,
498 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000499}
500
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000501void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800502 // We pass in the BaseChannel instead of the rtp_dtls_transport_
503 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000504 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200505 // We pass in the network thread because on that thread connection monitor
506 // will call BaseChannel::GetConnectionStats which must be called on the
507 // network thread.
508 connection_monitor_.reset(
509 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000510 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000511 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000512 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000513}
514
515void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000516 if (connection_monitor_) {
517 connection_monitor_->Stop();
518 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000519 }
520}
521
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000522bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200523 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800524 if (!rtp_dtls_transport_) {
525 return false;
526 }
zhihuangb2cdd932017-01-19 16:54:25 -0800527 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800528}
529
530bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800531 // If this BaseChannel doesn't require RTCP mux and we haven't fully
532 // negotiated RTCP mux, we need an RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -0700533 return !rtp_transport_.rtcp_mux_required() &&
534 !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000535}
536
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700537bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000538 // Receive data if we are enabled and have local content,
539 return enabled() && IsReceiveContentDirection(local_content_direction_);
540}
541
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700542bool BaseChannel::IsReadyToSendMedia_w() const {
543 // Need to access some state updated on the network thread.
544 return network_thread_->Invoke<bool>(
545 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
546}
547
548bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000549 // Send outgoing data if we are enabled, have local and remote content,
550 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800551 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000552 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700553 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200554 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000555}
556
jbaucheec21bd2016-03-20 06:15:43 -0700557bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700558 const rtc::PacketOptions& options) {
559 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000560}
561
jbaucheec21bd2016-03-20 06:15:43 -0700562bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700563 const rtc::PacketOptions& options) {
564 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000565}
566
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000567int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000568 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200569 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700570 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200571}
572
573int BaseChannel::SetOption_n(SocketType type,
574 rtc::Socket::Option opt,
575 int value) {
576 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800577 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000578 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000579 case ST_RTP:
zsteind48dbda2017-04-04 19:45:57 -0700580 transport = rtp_transport_.rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700581 socket_options_.push_back(
582 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000583 break;
584 case ST_RTCP:
zsteind48dbda2017-04-04 19:45:57 -0700585 transport = rtp_transport_.rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700586 rtcp_socket_options_.push_back(
587 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000588 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000589 }
deadbeeff5346592017-01-24 21:51:21 -0800590 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000591}
592
jbauchcb560652016-08-04 05:20:32 -0700593bool BaseChannel::SetCryptoOptions(const rtc::CryptoOptions& crypto_options) {
594 crypto_options_ = crypto_options;
595 return true;
596}
597
deadbeef5bd5ca32017-02-10 11:31:50 -0800598void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteind48dbda2017-04-04 19:45:57 -0700599 RTC_DCHECK(transport == rtp_transport_.rtp_packet_transport() ||
600 transport == rtp_transport_.rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200601 RTC_DCHECK(network_thread_->IsCurrent());
602 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000603}
604
deadbeef5bd5ca32017-02-10 11:31:50 -0800605void BaseChannel::OnPacketRead(rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -0700606 const char* data,
607 size_t len,
608 const rtc::PacketTime& packet_time,
609 int flags) {
610 TRACE_EVENT0("webrtc", "BaseChannel::OnPacketRead");
611 // OnPacketRead gets called from P2PSocket; now pass data to MediaEngine
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200612 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000613
614 // When using RTCP multiplexing we might get RTCP packets on the RTP
615 // transport. We feed RTP traffic into the demuxer to determine if it is RTCP.
johand89ab142016-10-25 10:50:32 -0700616 bool rtcp = PacketIsRtcp(transport, data, len);
jbaucheec21bd2016-03-20 06:15:43 -0700617 rtc::CopyOnWriteBuffer packet(data, len);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000618 HandlePacket(rtcp, &packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000619}
620
deadbeef5bd5ca32017-02-10 11:31:50 -0800621void BaseChannel::OnReadyToSend(rtc::PacketTransportInternal* transport) {
zsteind48dbda2017-04-04 19:45:57 -0700622 RTC_DCHECK(transport == rtp_transport_.rtp_packet_transport() ||
623 transport == rtp_transport_.rtcp_packet_transport());
624 SetTransportChannelReadyToSend(
625 transport == rtp_transport_.rtcp_packet_transport(), true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000626}
627
zhihuangb2cdd932017-01-19 16:54:25 -0800628void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800629 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200630 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800631 return;
632 }
633
634 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
635 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800636 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800637 // TransportChannel) or when TransportChannel is attached after DTLS is
638 // negotiated.
639 if (state != DTLS_TRANSPORT_CONNECTED) {
640 srtp_filter_.ResetParams();
641 }
642}
643
Honghai Zhangcc411c02016-03-29 17:27:21 -0700644void BaseChannel::OnSelectedCandidatePairChanged(
zhihuangb2cdd932017-01-19 16:54:25 -0800645 IceTransportInternal* ice_transport,
Honghai Zhang52dce732016-03-31 12:37:31 -0700646 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700647 int last_sent_packet_id,
648 bool ready_to_send) {
deadbeeff5346592017-01-24 21:51:21 -0800649 RTC_DCHECK((rtp_dtls_transport_ &&
650 ice_transport == rtp_dtls_transport_->ice_transport()) ||
651 (rtcp_dtls_transport_ &&
652 ice_transport == rtcp_dtls_transport_->ice_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200653 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800654 selected_candidate_pair_ = selected_candidate_pair;
zhihuangb2cdd932017-01-19 16:54:25 -0800655 std::string transport_name = ice_transport->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700656 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700657 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700658 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700659 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700660 selected_candidate_pair->remote_candidate().network_id(),
661 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800662
663 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700664 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200665 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700666 RTC_FROM_HERE, worker_thread_,
667 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
668 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700669}
670
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700671void BaseChannel::SetTransportChannelReadyToSend(bool rtcp, bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200672 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700673 if (rtcp) {
674 rtcp_ready_to_send_ = ready;
675 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000676 rtp_ready_to_send_ = ready;
677 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000678
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200679 bool ready_to_send =
680 (rtp_ready_to_send_ &&
deadbeeff5346592017-01-24 21:51:21 -0800681 // In the case of rtcp mux |rtcp_packet_transport_| will be null.
zsteind48dbda2017-04-04 19:45:57 -0700682 (rtcp_ready_to_send_ || !rtp_transport_.rtcp_packet_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200683
684 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700685 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200686 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready_to_send));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000687}
688
deadbeef5bd5ca32017-02-10 11:31:50 -0800689bool BaseChannel::PacketIsRtcp(const rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -0700690 const char* data,
691 size_t len) {
zsteind48dbda2017-04-04 19:45:57 -0700692 return (transport == rtp_transport_.rtcp_packet_transport() ||
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000693 rtcp_mux_filter_.DemuxRtcp(data, static_cast<int>(len)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000694}
695
stefanc1aeaf02015-10-15 07:26:07 -0700696bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700697 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700698 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200699 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
700 // If the thread is not our network thread, we will post to our network
701 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000702 // synchronize access to all the pieces of the send path, including
703 // SRTP and the inner workings of the transport channels.
704 // The only downside is that we can't return a proper failure code if
705 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200706 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000707 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200708 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
709 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800710 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700711 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700712 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000713 return true;
714 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200715 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000716
717 // Now that we are on the correct thread, ensure we have a place to send this
718 // packet before doing anything. (We might get RTCP packets that we don't
719 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
720 // transport.
deadbeef5bd5ca32017-02-10 11:31:50 -0800721 rtc::PacketTransportInternal* transport =
zsteind48dbda2017-04-04 19:45:57 -0700722 (!rtcp || rtcp_mux_filter_.IsActive())
723 ? rtp_transport_.rtp_packet_transport()
724 : rtp_transport_.rtcp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800725 if (!transport || !transport->writable()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000726 return false;
727 }
728
729 // Protect ourselves against crazy data.
730 if (!ValidPacket(rtcp, packet)) {
731 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000732 << PacketType(rtcp)
733 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000734 return false;
735 }
736
stefanc1aeaf02015-10-15 07:26:07 -0700737 rtc::PacketOptions updated_options;
738 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000739 // Protect if needed.
740 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200741 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000742 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200743 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000744 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000745 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000746 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
747 // inside libsrtp for a RTP packet. A external HMAC module will be writing
748 // a fake HMAC value. This is ONLY done for a RTP packet.
749 // Socket layer will update rtp sendtime extension header if present in
750 // packet with current time before updating the HMAC.
751#if !defined(ENABLE_EXTERNAL_AUTH)
752 res = srtp_filter_.ProtectRtp(
753 data, len, static_cast<int>(packet->capacity()), &len);
754#else
jbauchd48f4882017-03-01 15:34:36 -0800755 if (!srtp_filter_.IsExternalAuthActive()) {
756 res = srtp_filter_.ProtectRtp(
757 data, len, static_cast<int>(packet->capacity()), &len);
758 } else {
759 updated_options.packet_time_params.rtp_sendtime_extension_id =
760 rtp_abs_sendtime_extn_id_;
761 res = srtp_filter_.ProtectRtp(
762 data, len, static_cast<int>(packet->capacity()), &len,
763 &updated_options.packet_time_params.srtp_packet_index);
764 // If protection succeeds, let's get auth params from srtp.
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000765 if (res) {
jbauchd48f4882017-03-01 15:34:36 -0800766 uint8_t* auth_key = NULL;
767 int key_len;
768 res = srtp_filter_.GetRtpAuthParams(
769 &auth_key, &key_len,
770 &updated_options.packet_time_params.srtp_auth_tag_len);
771 if (res) {
772 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
773 updated_options.packet_time_params.srtp_auth_key.assign(
774 auth_key, auth_key + key_len);
775 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000776 }
777 }
778#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000779 if (!res) {
780 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200781 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000782 GetRtpSeqNum(data, len, &seq_num);
783 GetRtpSsrc(data, len, &ssrc);
784 LOG(LS_ERROR) << "Failed to protect " << content_name_
785 << " RTP packet: size=" << len
786 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
787 return false;
788 }
789 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000790 res = srtp_filter_.ProtectRtcp(data, len,
791 static_cast<int>(packet->capacity()),
792 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000793 if (!res) {
794 int type = -1;
795 GetRtcpType(data, len, &type);
796 LOG(LS_ERROR) << "Failed to protect " << content_name_
797 << " RTCP packet: size=" << len << ", type=" << type;
798 return false;
799 }
800 }
801
802 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000803 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800804 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800805 // The audio/video engines may attempt to send RTCP packets as soon as the
806 // streams are created, so don't treat this as an error for RTCP.
807 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
808 if (rtcp) {
809 return false;
810 }
811 // However, there shouldn't be any RTP packets sent before SRTP is set up
812 // (and SetSend(true) is called).
813 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
814 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800815 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000816 return false;
817 }
818
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000819 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200820 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
zhihuangb2cdd932017-01-19 16:54:25 -0800821 int ret = transport->SendPacket(packet->data<char>(), packet->size(),
822 updated_options, flags);
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000823 if (ret != static_cast<int>(packet->size())) {
zhihuangb2cdd932017-01-19 16:54:25 -0800824 if (transport->GetError() == ENOTCONN) {
skvladc309e0e2016-07-28 17:15:20 -0700825 LOG(LS_WARNING) << "Got ENOTCONN from transport.";
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700826 SetTransportChannelReadyToSend(rtcp, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000827 }
828 return false;
829 }
830 return true;
831}
832
jbaucheec21bd2016-03-20 06:15:43 -0700833bool BaseChannel::WantsPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000834 // Protect ourselves against crazy data.
835 if (!ValidPacket(rtcp, packet)) {
836 LOG(LS_ERROR) << "Dropping incoming " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000837 << PacketType(rtcp)
838 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000839 return false;
840 }
pbos482b12e2015-11-16 10:19:58 -0800841 if (rtcp) {
842 // Permit all (seemingly valid) RTCP packets.
843 return true;
844 }
845 // Check whether we handle this payload.
jbaucheec21bd2016-03-20 06:15:43 -0700846 return bundle_filter_.DemuxPacket(packet->data(), packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000847}
848
jbaucheec21bd2016-03-20 06:15:43 -0700849void BaseChannel::HandlePacket(bool rtcp, rtc::CopyOnWriteBuffer* packet,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000850 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200851 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000852 if (!WantsPacket(rtcp, packet)) {
853 return;
854 }
855
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000856 // We are only interested in the first rtp packet because that
857 // indicates the media has started flowing.
858 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000859 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700860 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000861 }
862
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000863 // Unprotect the packet, if needed.
864 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200865 TRACE_EVENT0("webrtc", "SRTP Decode");
Karl Wiberg94784372015-04-20 14:03:07 +0200866 char* data = packet->data<char>();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000867 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000868 bool res;
869 if (!rtcp) {
870 res = srtp_filter_.UnprotectRtp(data, len, &len);
871 if (!res) {
872 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200873 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000874 GetRtpSeqNum(data, len, &seq_num);
875 GetRtpSsrc(data, len, &ssrc);
876 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
877 << " RTP packet: size=" << len
878 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
879 return;
880 }
881 } else {
882 res = srtp_filter_.UnprotectRtcp(data, len, &len);
883 if (!res) {
884 int type = -1;
885 GetRtcpType(data, len, &type);
886 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
887 << " RTCP packet: size=" << len << ", type=" << type;
888 return;
889 }
890 }
891
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000892 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800893 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000894 // Our session description indicates that SRTP is required, but we got a
895 // packet before our SRTP filter is active. This means either that
896 // a) we got SRTP packets before we received the SDES keys, in which case
897 // we can't decrypt it anyway, or
898 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800899 // transports, so we haven't yet extracted keys, even if DTLS did
900 // complete on the transport that the packets are being sent on. It's
901 // really good practice to wait for both RTP and RTCP to be good to go
902 // before sending media, to prevent weird failure modes, so it's fine
903 // for us to just eat packets here. This is all sidestepped if RTCP mux
904 // is used anyway.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905 LOG(LS_WARNING) << "Can't process incoming " << PacketType(rtcp)
906 << " packet when SRTP is inactive and crypto is required";
907 return;
908 }
909
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200910 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700911 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200912 Bind(&BaseChannel::OnPacketReceived, this, rtcp, *packet, packet_time));
913}
914
915void BaseChannel::OnPacketReceived(bool rtcp,
916 const rtc::CopyOnWriteBuffer& packet,
917 const rtc::PacketTime& packet_time) {
918 RTC_DCHECK(worker_thread_->IsCurrent());
919 // Need to copy variable because OnRtcpReceived/OnPacketReceived
920 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
921 rtc::CopyOnWriteBuffer data(packet);
922 if (rtcp) {
923 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000924 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200925 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000926 }
927}
928
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000929bool BaseChannel::PushdownLocalDescription(
930 const SessionDescription* local_desc, ContentAction action,
931 std::string* error_desc) {
932 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000933 const MediaContentDescription* content_desc =
934 GetContentDescription(content_info);
935 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000936 !SetLocalContent(content_desc, action, error_desc)) {
937 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
938 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000939 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000940 return true;
941}
942
943bool BaseChannel::PushdownRemoteDescription(
944 const SessionDescription* remote_desc, ContentAction action,
945 std::string* error_desc) {
946 const ContentInfo* content_info = GetFirstContent(remote_desc);
947 const MediaContentDescription* content_desc =
948 GetContentDescription(content_info);
949 if (content_desc && content_info && !content_info->rejected &&
950 !SetRemoteContent(content_desc, action, error_desc)) {
951 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
952 return false;
953 }
954 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000955}
956
957void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700958 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000959 if (enabled_)
960 return;
961
962 LOG(LS_INFO) << "Channel enabled";
963 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700964 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000965}
966
967void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700968 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000969 if (!enabled_)
970 return;
971
972 LOG(LS_INFO) << "Channel disabled";
973 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700974 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000975}
976
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200977void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700978 rtc::PacketTransportInternal* rtp_packet_transport =
979 rtp_transport_.rtp_packet_transport();
980 rtc::PacketTransportInternal* rtcp_packet_transport =
981 rtp_transport_.rtcp_packet_transport();
982 if (rtp_packet_transport && rtp_packet_transport->writable() &&
983 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200984 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700985 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200986 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700987 }
988}
989
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200990void BaseChannel::ChannelWritable_n() {
991 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800992 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000993 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800994 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000995
deadbeefcbecd352015-09-23 11:50:27 -0700996 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000997 << (was_ever_writable_ ? "" : " for the first time");
998
michaelt79e05882016-11-08 02:50:09 -0800999 if (selected_candidate_pair_)
1000 LOG(LS_INFO)
1001 << "Using "
1002 << selected_candidate_pair_->local_candidate().ToSensitiveString()
1003 << "->"
1004 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001005
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001006 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001007 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001008 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001009 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001010}
1011
deadbeef953c2ce2017-01-09 14:53:41 -08001012void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001013 RTC_DCHECK(network_thread_->IsCurrent());
1014 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001015 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -08001016 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001017}
1018
deadbeef953c2ce2017-01-09 14:53:41 -08001019void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001020 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -08001021 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001022}
1023
zhihuangb2cdd932017-01-19 16:54:25 -08001024bool BaseChannel::SetDtlsSrtpCryptoSuites_n(DtlsTransportInternal* transport,
1025 bool rtcp) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001026 std::vector<int> crypto_suites;
1027 // We always use the default SRTP crypto suites for RTCP, but we may use
1028 // different crypto suites for RTP depending on the media type.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001029 if (!rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001030 GetSrtpCryptoSuites_n(&crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001031 } else {
jbauchcb560652016-08-04 05:20:32 -07001032 GetDefaultSrtpCryptoSuites(crypto_options(), &crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001033 }
zhihuangb2cdd932017-01-19 16:54:25 -08001034 return transport->SetSrtpCryptoSuites(crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001035}
1036
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001037bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -08001038 // Since DTLS is applied to all transports, checking RTP should be enough.
1039 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001040}
1041
1042// This function returns true if either DTLS-SRTP is not in use
1043// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -08001044bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001045 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001046 bool ret = false;
1047
zhihuangb2cdd932017-01-19 16:54:25 -08001048 DtlsTransportInternal* transport =
1049 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -08001050 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -08001051 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001052
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001053 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001054
zhihuangb2cdd932017-01-19 16:54:25 -08001055 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001056 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001057 return false;
1058 }
1059
zhihuangb2cdd932017-01-19 16:54:25 -08001060 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name() << " "
1061 << PacketType(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001062
jbauchcb560652016-08-04 05:20:32 -07001063 int key_len;
1064 int salt_len;
1065 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
1066 &salt_len)) {
1067 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
1068 return false;
1069 }
1070
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001071 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -07001072 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001073
1074 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -08001075 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
1076 &dtls_buffer[0], dtls_buffer.size())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001077 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -08001078 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001079 return false;
1080 }
1081
1082 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -07001083 std::vector<unsigned char> client_write_key(key_len + salt_len);
1084 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001085 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -07001086 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
1087 offset += key_len;
1088 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
1089 offset += key_len;
1090 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
1091 offset += salt_len;
1092 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001093
1094 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001095 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -08001096 if (!transport->GetSslRole(&role)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00001097 LOG(LS_WARNING) << "GetSslRole failed";
1098 return false;
1099 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001100
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001101 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001102 send_key = &server_write_key;
1103 recv_key = &client_write_key;
1104 } else {
1105 send_key = &client_write_key;
1106 recv_key = &server_write_key;
1107 }
1108
zhihuangb2cdd932017-01-19 16:54:25 -08001109 if (rtcp) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001110 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
1111 static_cast<int>(send_key->size()),
1112 selected_crypto_suite, &(*recv_key)[0],
1113 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001114 } else {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001115 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
1116 static_cast<int>(send_key->size()),
1117 selected_crypto_suite, &(*recv_key)[0],
1118 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001119 }
1120
michaelt79e05882016-11-08 02:50:09 -08001121 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001122 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001123 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001124 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001125 UpdateTransportOverhead();
1126 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001127 return ret;
1128}
1129
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001130void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001131 if (srtp_filter_.IsActive()) {
1132 return;
1133 }
1134
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001135 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001136 return;
1137 }
1138
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001139 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001140 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001141 return;
1142 }
1143
zhihuangb2cdd932017-01-19 16:54:25 -08001144 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001145 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001146 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001147 return;
1148 }
1149 }
1150}
1151
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001152void BaseChannel::ChannelNotWritable_n() {
1153 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001154 if (!writable_)
1155 return;
1156
deadbeefcbecd352015-09-23 11:50:27 -07001157 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001158 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001159 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001160}
1161
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001162bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001163 const MediaContentDescription* content,
1164 ContentAction action,
1165 ContentSource src,
1166 std::string* error_desc) {
1167 if (action == CA_UPDATE) {
1168 // These parameters never get changed by a CA_UDPATE.
1169 return true;
1170 }
1171
deadbeef7af91dd2016-12-13 11:29:11 -08001172 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001173 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001174 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
1175 content, action, src, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001176}
1177
1178bool BaseChannel::SetRtpTransportParameters_n(
1179 const MediaContentDescription* content,
1180 ContentAction action,
1181 ContentSource src,
1182 std::string* error_desc) {
1183 RTC_DCHECK(network_thread_->IsCurrent());
1184
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001185 if (!SetSrtp_n(content->cryptos(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001186 return false;
1187 }
1188
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001189 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001190 return false;
1191 }
1192
1193 return true;
1194}
1195
zhihuangb2cdd932017-01-19 16:54:25 -08001196// |dtls| will be set to true if DTLS is active for transport and crypto is
1197// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001198bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1199 bool* dtls,
1200 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -08001201 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001202 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001203 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001204 return false;
1205 }
1206 return true;
1207}
1208
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001209bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001210 ContentAction action,
1211 ContentSource src,
1212 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001213 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001214 if (action == CA_UPDATE) {
1215 // no crypto params.
1216 return true;
1217 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001218 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001219 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001220 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001221 if (!ret) {
1222 return false;
1223 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001224 switch (action) {
1225 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001226 // If DTLS is already active on the channel, we could be renegotiating
1227 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001228 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001229 ret = srtp_filter_.SetOffer(cryptos, src);
1230 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001231 break;
1232 case CA_PRANSWER:
1233 // If we're doing DTLS-SRTP, we don't want to update the filter
1234 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001235 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001236 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1237 }
1238 break;
1239 case CA_ANSWER:
1240 // If we're doing DTLS-SRTP, we don't want to update the filter
1241 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001242 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001243 ret = srtp_filter_.SetAnswer(cryptos, src);
1244 }
1245 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001246 default:
1247 break;
1248 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001249 if (!ret) {
1250 SafeSetError("Failed to setup SRTP filter.", error_desc);
1251 return false;
1252 }
1253 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001254}
1255
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001256bool BaseChannel::SetRtcpMux_n(bool enable,
1257 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001258 ContentSource src,
1259 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001260 // Provide a more specific error message for the RTCP mux "require" policy
1261 // case.
zsteind48dbda2017-04-04 19:45:57 -07001262 if (rtp_transport_.rtcp_mux_required() && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001263 SafeSetError(
1264 "rtcpMuxPolicy is 'require', but media description does not "
1265 "contain 'a=rtcp-mux'.",
1266 error_desc);
1267 return false;
1268 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001269 bool ret = false;
1270 switch (action) {
1271 case CA_OFFER:
1272 ret = rtcp_mux_filter_.SetOffer(enable, src);
1273 break;
1274 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001275 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001276 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001277 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1278 break;
1279 case CA_ANSWER:
1280 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1281 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001282 // We permanently activated RTCP muxing; signal that we no longer need
1283 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001284 std::string debug_name =
1285 transport_name_.empty()
1286 ? rtp_transport_.rtp_packet_transport()->debug_name()
1287 : transport_name_;
deadbeefe814a0d2017-02-25 18:15:09 -08001288 ;
deadbeefcbecd352015-09-23 11:50:27 -07001289 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
deadbeefe814a0d2017-02-25 18:15:09 -08001290 << "; no longer need RTCP transport for " << debug_name;
zsteind48dbda2017-04-04 19:45:57 -07001291 if (rtp_transport_.rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001292 SetTransport_n(true, nullptr, nullptr);
1293 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001294 }
deadbeef062ce9f2016-08-26 21:42:15 -07001295 UpdateWritableState_n();
1296 SetTransportChannelReadyToSend(true, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001297 }
1298 break;
1299 case CA_UPDATE:
1300 // No RTCP mux info.
1301 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001302 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001303 default:
1304 break;
1305 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001306 if (!ret) {
1307 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1308 return false;
1309 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001310 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001311 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1312 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001313 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001314 // If the RTP transport is already writable, then so are we.
zsteind48dbda2017-04-04 19:45:57 -07001315 if (rtp_transport_.rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001316 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001317 }
1318 }
1319
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001320 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001321}
1322
1323bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001324 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001325 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001326}
1327
Peter Boström0c4e06b2015-10-07 12:23:21 +02001328bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001329 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001330 return media_channel()->RemoveRecvStream(ssrc);
1331}
1332
1333bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001334 ContentAction action,
1335 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001336 if (!(action == CA_OFFER || action == CA_ANSWER ||
1337 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001338 return false;
1339
1340 // If this is an update, streams only contain streams that have changed.
1341 if (action == CA_UPDATE) {
1342 for (StreamParamsVec::const_iterator it = streams.begin();
1343 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001344 const StreamParams* existing_stream =
1345 GetStreamByIds(local_streams_, it->groupid, it->id);
1346 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001347 if (media_channel()->AddSendStream(*it)) {
1348 local_streams_.push_back(*it);
1349 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1350 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001351 std::ostringstream desc;
1352 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1353 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001354 return false;
1355 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001356 } else if (existing_stream && !it->has_ssrcs()) {
1357 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001358 std::ostringstream desc;
1359 desc << "Failed to remove send stream with ssrc "
1360 << it->first_ssrc() << ".";
1361 SafeSetError(desc.str(), error_desc);
1362 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001363 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001364 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001365 } else {
1366 LOG(LS_WARNING) << "Ignore unsupported stream update";
1367 }
1368 }
1369 return true;
1370 }
1371 // Else streams are all the streams we want to send.
1372
1373 // Check for streams that have been removed.
1374 bool ret = true;
1375 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1376 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001377 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001378 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001379 std::ostringstream desc;
1380 desc << "Failed to remove send stream with ssrc "
1381 << it->first_ssrc() << ".";
1382 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001383 ret = false;
1384 }
1385 }
1386 }
1387 // Check for new streams.
1388 for (StreamParamsVec::const_iterator it = streams.begin();
1389 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001390 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001391 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001392 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001393 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001394 std::ostringstream desc;
1395 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1396 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001397 ret = false;
1398 }
1399 }
1400 }
1401 local_streams_ = streams;
1402 return ret;
1403}
1404
1405bool BaseChannel::UpdateRemoteStreams_w(
1406 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001407 ContentAction action,
1408 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001409 if (!(action == CA_OFFER || action == CA_ANSWER ||
1410 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001411 return false;
1412
1413 // If this is an update, streams only contain streams that have changed.
1414 if (action == CA_UPDATE) {
1415 for (StreamParamsVec::const_iterator it = streams.begin();
1416 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001417 const StreamParams* existing_stream =
1418 GetStreamByIds(remote_streams_, it->groupid, it->id);
1419 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001420 if (AddRecvStream_w(*it)) {
1421 remote_streams_.push_back(*it);
1422 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1423 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001424 std::ostringstream desc;
1425 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1426 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001427 return false;
1428 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001429 } else if (existing_stream && !it->has_ssrcs()) {
1430 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001431 std::ostringstream desc;
1432 desc << "Failed to remove remote stream with ssrc "
1433 << it->first_ssrc() << ".";
1434 SafeSetError(desc.str(), error_desc);
1435 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001436 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001437 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001438 } else {
1439 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001440 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001441 << " new stream = " << it->ToString();
1442 }
1443 }
1444 return true;
1445 }
1446 // Else streams are all the streams we want to receive.
1447
1448 // Check for streams that have been removed.
1449 bool ret = true;
1450 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1451 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001452 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001453 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001454 std::ostringstream desc;
1455 desc << "Failed to remove remote stream with ssrc "
1456 << it->first_ssrc() << ".";
1457 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001458 ret = false;
1459 }
1460 }
1461 }
1462 // Check for new streams.
1463 for (StreamParamsVec::const_iterator it = streams.begin();
1464 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001465 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001466 if (AddRecvStream_w(*it)) {
1467 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1468 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001469 std::ostringstream desc;
1470 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1471 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001472 ret = false;
1473 }
1474 }
1475 }
1476 remote_streams_ = streams;
1477 return ret;
1478}
1479
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001480void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001481 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001482// Absolute Send Time extension id is used only with external auth,
1483// so do not bother searching for it and making asyncronious call to set
1484// something that is not used.
1485#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001486 const webrtc::RtpExtension* send_time_extension =
1487 FindHeaderExtension(extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001488 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001489 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001490 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001491 RTC_FROM_HERE, network_thread_,
1492 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1493 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001494#endif
1495}
1496
1497void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1498 int rtp_abs_sendtime_extn_id) {
1499 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001500}
1501
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001502void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001503 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001504 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001505 case MSG_SEND_RTP_PACKET:
1506 case MSG_SEND_RTCP_PACKET: {
1507 RTC_DCHECK(network_thread_->IsCurrent());
1508 SendPacketMessageData* data =
1509 static_cast<SendPacketMessageData*>(pmsg->pdata);
1510 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1511 SendPacket(rtcp, &data->packet, data->options);
1512 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001513 break;
1514 }
1515 case MSG_FIRSTPACKETRECEIVED: {
1516 SignalFirstPacketReceived(this);
1517 break;
1518 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001519 }
1520}
1521
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001522void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001523 // Flush all remaining RTCP messages. This should only be called in
1524 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001525 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001526 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001527 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1528 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001529 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1530 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001531 }
1532}
1533
johand89ab142016-10-25 10:50:32 -07001534void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001535 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001536 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001537 RTC_DCHECK(network_thread_->IsCurrent());
1538 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001539 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001540 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1541}
1542
1543void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1544 RTC_DCHECK(worker_thread_->IsCurrent());
1545 SignalSentPacket(sent_packet);
1546}
1547
1548VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1549 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001550 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001551 MediaEngineInterface* media_engine,
1552 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001553 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001554 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001555 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001556 : BaseChannel(worker_thread,
1557 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001558 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001559 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001560 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001561 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001562 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001563 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001564 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001565
1566VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001567 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001568 StopAudioMonitor();
1569 StopMediaMonitor();
1570 // this can't be done in the base class, since it calls a virtual
1571 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001572 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001573}
1574
Peter Boström0c4e06b2015-10-07 12:23:21 +02001575bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001576 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001577 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001578 AudioSource* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001579 return InvokeOnWorker(RTC_FROM_HERE,
1580 Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001581 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001582}
1583
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001584// TODO(juberti): Handle early media the right way. We should get an explicit
1585// ringing message telling us to start playing local ringback, which we cancel
1586// if any early media actually arrives. For now, we do the opposite, which is
1587// to wait 1 second for early media, and start playing local ringback if none
1588// arrives.
1589void VoiceChannel::SetEarlyMedia(bool enable) {
1590 if (enable) {
1591 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001592 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1593 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001594 } else {
1595 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001596 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001597 }
1598}
1599
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001600bool VoiceChannel::CanInsertDtmf() {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001601 return InvokeOnWorker(
1602 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001603}
1604
Peter Boström0c4e06b2015-10-07 12:23:21 +02001605bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1606 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001607 int duration) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001608 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceChannel::InsertDtmf_w, this,
1609 ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001610}
1611
solenberg4bac9c52015-10-09 02:32:53 -07001612bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001613 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetOutputVolume,
1614 media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001615}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001616
Tommif888bb52015-12-12 01:37:01 +01001617void VoiceChannel::SetRawAudioSink(
1618 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001619 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1620 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001621 // passing. So we invoke to our own little routine that gets a pointer to
1622 // our local variable. This is OK since we're synchronously invoking.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001623 InvokeOnWorker(RTC_FROM_HERE,
1624 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001625}
1626
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001627webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001628 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001629 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001630}
1631
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001632webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1633 uint32_t ssrc) const {
1634 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001635}
1636
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001637bool VoiceChannel::SetRtpSendParameters(
1638 uint32_t ssrc,
1639 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001640 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001641 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001642 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001643}
1644
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001645bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1646 webrtc::RtpParameters parameters) {
1647 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1648}
1649
1650webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1651 uint32_t ssrc) const {
1652 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001653 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001654 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1655}
1656
1657webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1658 uint32_t ssrc) const {
1659 return media_channel()->GetRtpReceiveParameters(ssrc);
1660}
1661
1662bool VoiceChannel::SetRtpReceiveParameters(
1663 uint32_t ssrc,
1664 const webrtc::RtpParameters& parameters) {
1665 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001666 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001667 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1668}
1669
1670bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1671 webrtc::RtpParameters parameters) {
1672 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001673}
1674
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001675bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001676 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1677 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001678}
1679
hbos8d609f62017-04-10 07:39:05 -07001680std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1681 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
1682 RTC_FROM_HERE,
1683 Bind(&WebRtcVoiceMediaChannel::GetSources,
1684 static_cast<WebRtcVoiceMediaChannel*>(media_channel()), ssrc));
1685}
1686
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001687void VoiceChannel::StartMediaMonitor(int cms) {
1688 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001689 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001690 media_monitor_->SignalUpdate.connect(
1691 this, &VoiceChannel::OnMediaMonitorUpdate);
1692 media_monitor_->Start(cms);
1693}
1694
1695void VoiceChannel::StopMediaMonitor() {
1696 if (media_monitor_) {
1697 media_monitor_->Stop();
1698 media_monitor_->SignalUpdate.disconnect(this);
1699 media_monitor_.reset();
1700 }
1701}
1702
1703void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001704 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001705 audio_monitor_
1706 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1707 audio_monitor_->Start(cms);
1708}
1709
1710void VoiceChannel::StopAudioMonitor() {
1711 if (audio_monitor_) {
1712 audio_monitor_->Stop();
1713 audio_monitor_.reset();
1714 }
1715}
1716
1717bool VoiceChannel::IsAudioMonitorRunning() const {
1718 return (audio_monitor_.get() != NULL);
1719}
1720
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001721int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001722 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001723}
1724
1725int VoiceChannel::GetOutputLevel_w() {
1726 return media_channel()->GetOutputLevel();
1727}
1728
1729void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1730 media_channel()->GetActiveStreams(actives);
1731}
1732
deadbeef5bd5ca32017-02-10 11:31:50 -08001733void VoiceChannel::OnPacketRead(rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -07001734 const char* data,
1735 size_t len,
1736 const rtc::PacketTime& packet_time,
wu@webrtc.orga9890802013-12-13 00:21:03 +00001737 int flags) {
johand89ab142016-10-25 10:50:32 -07001738 BaseChannel::OnPacketRead(transport, data, len, packet_time, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001739 // Set a flag when we've received an RTP packet. If we're waiting for early
1740 // media, this will disable the timeout.
johand89ab142016-10-25 10:50:32 -07001741 if (!received_media_ && !PacketIsRtcp(transport, data, len)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001742 received_media_ = true;
1743 }
1744}
1745
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001746void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001747 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001748 invoker_.AsyncInvoke<void>(
1749 RTC_FROM_HERE, worker_thread_,
1750 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001751}
1752
michaelt79e05882016-11-08 02:50:09 -08001753int BaseChannel::GetTransportOverheadPerPacket() const {
1754 RTC_DCHECK(network_thread_->IsCurrent());
1755
1756 if (!selected_candidate_pair_)
1757 return 0;
1758
1759 int transport_overhead_per_packet = 0;
1760
1761 constexpr int kIpv4Overhaed = 20;
1762 constexpr int kIpv6Overhaed = 40;
1763 transport_overhead_per_packet +=
1764 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1765 ? kIpv4Overhaed
1766 : kIpv6Overhaed;
1767
1768 constexpr int kUdpOverhaed = 8;
1769 constexpr int kTcpOverhaed = 20;
1770 transport_overhead_per_packet +=
1771 selected_candidate_pair_->local_candidate().protocol() ==
1772 TCP_PROTOCOL_NAME
1773 ? kTcpOverhaed
1774 : kUdpOverhaed;
1775
1776 if (secure()) {
1777 int srtp_overhead = 0;
1778 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1779 transport_overhead_per_packet += srtp_overhead;
1780 }
1781
1782 return transport_overhead_per_packet;
1783}
1784
1785void BaseChannel::UpdateTransportOverhead() {
1786 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1787 if (transport_overhead_per_packet)
1788 invoker_.AsyncInvoke<void>(
1789 RTC_FROM_HERE, worker_thread_,
1790 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1791 transport_overhead_per_packet));
1792}
1793
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001794void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001795 // Render incoming data if we're the active call, and we have the local
1796 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001797 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001798 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001799
1800 // Send outgoing data if we're the active call, we have the remote content,
1801 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001802 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001803 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001804
1805 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1806}
1807
1808const ContentInfo* VoiceChannel::GetFirstContent(
1809 const SessionDescription* sdesc) {
1810 return GetFirstAudioContent(sdesc);
1811}
1812
1813bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001814 ContentAction action,
1815 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001816 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001817 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001818 LOG(LS_INFO) << "Setting local voice description";
1819
1820 const AudioContentDescription* audio =
1821 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001822 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001823 if (!audio) {
1824 SafeSetError("Can't find audio content in local description.", error_desc);
1825 return false;
1826 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001828 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001829 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001830 }
1831
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001832 AudioRecvParameters recv_params = last_recv_params_;
1833 RtpParametersFromMediaDescription(audio, &recv_params);
1834 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001835 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001836 error_desc);
1837 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001838 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001839 for (const AudioCodec& codec : audio->codecs()) {
1840 bundle_filter()->AddPayloadType(codec.id);
1841 }
1842 last_recv_params_ = recv_params;
1843
1844 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1845 // only give it to the media channel once we have a remote
1846 // description too (without a remote description, we won't be able
1847 // to send them anyway).
1848 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1849 SafeSetError("Failed to set local audio description streams.", error_desc);
1850 return false;
1851 }
1852
1853 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001854 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001855 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001856}
1857
1858bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001859 ContentAction action,
1860 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001861 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001862 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001863 LOG(LS_INFO) << "Setting remote voice description";
1864
1865 const AudioContentDescription* audio =
1866 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001867 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001868 if (!audio) {
1869 SafeSetError("Can't find audio content in remote description.", error_desc);
1870 return false;
1871 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001872
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001873 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001874 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001875 }
1876
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001877 AudioSendParameters send_params = last_send_params_;
1878 RtpSendParametersFromMediaDescription(audio, &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001879 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001880 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001881 }
skvladdc1c62c2016-03-16 19:07:43 -07001882
1883 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1884 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001885 SafeSetError("Failed to set remote audio description send parameters.",
1886 error_desc);
1887 return false;
1888 }
1889 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001890
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001891 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1892 // and only give it to the media channel once we have a local
1893 // description too (without a local description, we won't be able to
1894 // recv them anyway).
1895 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1896 SafeSetError("Failed to set remote audio description streams.", error_desc);
1897 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001898 }
1899
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001900 if (audio->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001901 MaybeCacheRtpAbsSendTimeHeaderExtension_w(audio->rtp_header_extensions());
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001902 }
1903
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001904 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001905 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001906 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001907}
1908
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001909void VoiceChannel::HandleEarlyMediaTimeout() {
1910 // This occurs on the main thread, not the worker thread.
1911 if (!received_media_) {
1912 LOG(LS_INFO) << "No early media received before timeout";
1913 SignalEarlyMediaTimeout(this);
1914 }
1915}
1916
Peter Boström0c4e06b2015-10-07 12:23:21 +02001917bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1918 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001919 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001920 if (!enabled()) {
1921 return false;
1922 }
solenberg1d63dd02015-12-02 12:35:09 -08001923 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001924}
1925
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001926void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001927 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001928 case MSG_EARLYMEDIATIMEOUT:
1929 HandleEarlyMediaTimeout();
1930 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001931 case MSG_CHANNEL_ERROR: {
1932 VoiceChannelErrorMessageData* data =
1933 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001934 delete data;
1935 break;
1936 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001937 default:
1938 BaseChannel::OnMessage(pmsg);
1939 break;
1940 }
1941}
1942
1943void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001944 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001945 SignalConnectionMonitor(this, infos);
1946}
1947
1948void VoiceChannel::OnMediaMonitorUpdate(
1949 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001950 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001951 SignalMediaMonitor(this, info);
1952}
1953
1954void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1955 const AudioInfo& info) {
1956 SignalAudioMonitor(this, info);
1957}
1958
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001959void VoiceChannel::GetSrtpCryptoSuites_n(
1960 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07001961 GetSupportedAudioCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001962}
1963
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001964VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1965 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001966 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001967 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001968 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001969 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001970 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001971 : BaseChannel(worker_thread,
1972 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001973 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001974 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001975 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001976 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001977 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001978
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001979VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001980 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001981 StopMediaMonitor();
1982 // this can't be done in the base class, since it calls a virtual
1983 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001984
1985 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001986}
1987
nisse08582ff2016-02-04 01:24:52 -08001988bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001989 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001990 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001991 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001992 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001993 return true;
1994}
1995
deadbeef5a4a75a2016-06-02 16:23:38 -07001996bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001997 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001998 bool mute,
1999 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08002000 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002001 return InvokeOnWorker(RTC_FROM_HERE,
2002 Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
deadbeef5a4a75a2016-06-02 16:23:38 -07002003 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07002004}
2005
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002006webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07002007 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002008 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07002009}
2010
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002011webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
2012 uint32_t ssrc) const {
2013 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07002014}
2015
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002016bool VideoChannel::SetRtpSendParameters(
2017 uint32_t ssrc,
2018 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07002019 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002020 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002021 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07002022}
2023
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002024bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
2025 webrtc::RtpParameters parameters) {
2026 return media_channel()->SetRtpSendParameters(ssrc, parameters);
2027}
2028
2029webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
2030 uint32_t ssrc) const {
2031 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002032 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002033 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
2034}
2035
2036webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
2037 uint32_t ssrc) const {
2038 return media_channel()->GetRtpReceiveParameters(ssrc);
2039}
2040
2041bool VideoChannel::SetRtpReceiveParameters(
2042 uint32_t ssrc,
2043 const webrtc::RtpParameters& parameters) {
2044 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002045 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002046 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
2047}
2048
2049bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
2050 webrtc::RtpParameters parameters) {
2051 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07002052}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002053
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002054void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002055 // Send outgoing data if we're the active call, we have the remote content,
2056 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002057 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002058 if (!media_channel()->SetSend(send)) {
2059 LOG(LS_ERROR) << "Failed to SetSend on video channel";
2060 // TODO(gangji): Report error back to server.
2061 }
2062
Peter Boström34fbfff2015-09-24 19:20:30 +02002063 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002064}
2065
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00002066bool VideoChannel::GetStats(VideoMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002067 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
2068 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002069}
2070
2071void VideoChannel::StartMediaMonitor(int cms) {
2072 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002073 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002074 media_monitor_->SignalUpdate.connect(
2075 this, &VideoChannel::OnMediaMonitorUpdate);
2076 media_monitor_->Start(cms);
2077}
2078
2079void VideoChannel::StopMediaMonitor() {
2080 if (media_monitor_) {
2081 media_monitor_->Stop();
2082 media_monitor_.reset();
2083 }
2084}
2085
2086const ContentInfo* VideoChannel::GetFirstContent(
2087 const SessionDescription* sdesc) {
2088 return GetFirstVideoContent(sdesc);
2089}
2090
2091bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002092 ContentAction action,
2093 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002094 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002095 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002096 LOG(LS_INFO) << "Setting local video description";
2097
2098 const VideoContentDescription* video =
2099 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002100 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002101 if (!video) {
2102 SafeSetError("Can't find video content in local description.", error_desc);
2103 return false;
2104 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002105
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002106 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002107 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002108 }
2109
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002110 VideoRecvParameters recv_params = last_recv_params_;
2111 RtpParametersFromMediaDescription(video, &recv_params);
2112 if (!media_channel()->SetRecvParameters(recv_params)) {
2113 SafeSetError("Failed to set local video description recv parameters.",
2114 error_desc);
2115 return false;
2116 }
2117 for (const VideoCodec& codec : video->codecs()) {
2118 bundle_filter()->AddPayloadType(codec.id);
2119 }
2120 last_recv_params_ = recv_params;
2121
2122 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2123 // only give it to the media channel once we have a remote
2124 // description too (without a remote description, we won't be able
2125 // to send them anyway).
2126 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2127 SafeSetError("Failed to set local video description streams.", error_desc);
2128 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002129 }
2130
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002131 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002132 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002133 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002134}
2135
2136bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002137 ContentAction action,
2138 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002139 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002140 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002141 LOG(LS_INFO) << "Setting remote video description";
2142
2143 const VideoContentDescription* video =
2144 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002145 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002146 if (!video) {
2147 SafeSetError("Can't find video content in remote description.", error_desc);
2148 return false;
2149 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002150
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002151 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002152 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002153 }
2154
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002155 VideoSendParameters send_params = last_send_params_;
2156 RtpSendParametersFromMediaDescription(video, &send_params);
2157 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002158 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002159 }
skvladdc1c62c2016-03-16 19:07:43 -07002160
2161 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2162
2163 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002164 SafeSetError("Failed to set remote video description send parameters.",
2165 error_desc);
2166 return false;
2167 }
2168 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002169
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002170 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2171 // and only give it to the media channel once we have a local
2172 // description too (without a local description, we won't be able to
2173 // recv them anyway).
2174 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2175 SafeSetError("Failed to set remote video description streams.", error_desc);
2176 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002177 }
2178
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002179 if (video->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002180 MaybeCacheRtpAbsSendTimeHeaderExtension_w(video->rtp_header_extensions());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002181 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002182
2183 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002184 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002185 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002186}
2187
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002188void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002189 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002190 case MSG_CHANNEL_ERROR: {
2191 const VideoChannelErrorMessageData* data =
2192 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002193 delete data;
2194 break;
2195 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002196 default:
2197 BaseChannel::OnMessage(pmsg);
2198 break;
2199 }
2200}
2201
2202void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002203 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002204 SignalConnectionMonitor(this, infos);
2205}
2206
2207// TODO(pthatcher): Look into removing duplicate code between
2208// audio, video, and data, perhaps by using templates.
2209void VideoChannel::OnMediaMonitorUpdate(
2210 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002211 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002212 SignalMediaMonitor(this, info);
2213}
2214
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002215void VideoChannel::GetSrtpCryptoSuites_n(
2216 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002217 GetSupportedVideoCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002218}
2219
deadbeef953c2ce2017-01-09 14:53:41 -08002220RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2221 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002222 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002223 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002224 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002225 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002226 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002227 : BaseChannel(worker_thread,
2228 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002229 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002230 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002231 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002232 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002233 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002234
deadbeef953c2ce2017-01-09 14:53:41 -08002235RtpDataChannel::~RtpDataChannel() {
2236 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002237 StopMediaMonitor();
2238 // this can't be done in the base class, since it calls a virtual
2239 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002240
2241 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002242}
2243
deadbeeff5346592017-01-24 21:51:21 -08002244bool RtpDataChannel::Init_w(
2245 DtlsTransportInternal* rtp_dtls_transport,
2246 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002247 rtc::PacketTransportInternal* rtp_packet_transport,
2248 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -08002249 if (!BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2250 rtp_packet_transport, rtcp_packet_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002251 return false;
2252 }
deadbeef953c2ce2017-01-09 14:53:41 -08002253 media_channel()->SignalDataReceived.connect(this,
2254 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002255 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002256 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002257 return true;
2258}
2259
deadbeef953c2ce2017-01-09 14:53:41 -08002260bool RtpDataChannel::SendData(const SendDataParams& params,
2261 const rtc::CopyOnWriteBuffer& payload,
2262 SendDataResult* result) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002263 return InvokeOnWorker(
2264 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2265 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002266}
2267
deadbeef953c2ce2017-01-09 14:53:41 -08002268const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002269 const SessionDescription* sdesc) {
2270 return GetFirstDataContent(sdesc);
2271}
2272
deadbeef953c2ce2017-01-09 14:53:41 -08002273bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002274 const DataContentDescription* content,
2275 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002276 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2277 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002278 // It's been set before, but doesn't match. That's bad.
2279 if (is_sctp) {
2280 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2281 error_desc);
2282 return false;
2283 }
2284 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002285}
2286
deadbeef953c2ce2017-01-09 14:53:41 -08002287bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2288 ContentAction action,
2289 std::string* error_desc) {
2290 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002291 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002292 LOG(LS_INFO) << "Setting local data description";
2293
2294 const DataContentDescription* data =
2295 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002296 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002297 if (!data) {
2298 SafeSetError("Can't find data content in local description.", error_desc);
2299 return false;
2300 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002301
deadbeef953c2ce2017-01-09 14:53:41 -08002302 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002303 return false;
2304 }
2305
deadbeef953c2ce2017-01-09 14:53:41 -08002306 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
2307 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002308 }
2309
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002310 DataRecvParameters recv_params = last_recv_params_;
2311 RtpParametersFromMediaDescription(data, &recv_params);
2312 if (!media_channel()->SetRecvParameters(recv_params)) {
2313 SafeSetError("Failed to set remote data description recv parameters.",
2314 error_desc);
2315 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002316 }
deadbeef953c2ce2017-01-09 14:53:41 -08002317 for (const DataCodec& codec : data->codecs()) {
2318 bundle_filter()->AddPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002319 }
2320 last_recv_params_ = recv_params;
2321
2322 // TODO(pthatcher): Move local streams into DataSendParameters, and
2323 // only give it to the media channel once we have a remote
2324 // description too (without a remote description, we won't be able
2325 // to send them anyway).
2326 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2327 SafeSetError("Failed to set local data description streams.", error_desc);
2328 return false;
2329 }
2330
2331 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002332 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002333 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002334}
2335
deadbeef953c2ce2017-01-09 14:53:41 -08002336bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2337 ContentAction action,
2338 std::string* error_desc) {
2339 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002340 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002341
2342 const DataContentDescription* data =
2343 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002344 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002345 if (!data) {
2346 SafeSetError("Can't find data content in remote description.", error_desc);
2347 return false;
2348 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002349
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002350 // If the remote data doesn't have codecs and isn't an update, it
2351 // must be empty, so ignore it.
2352 if (!data->has_codecs() && action != CA_UPDATE) {
2353 return true;
2354 }
2355
deadbeef953c2ce2017-01-09 14:53:41 -08002356 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002357 return false;
2358 }
2359
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002360 LOG(LS_INFO) << "Setting remote data description";
deadbeef953c2ce2017-01-09 14:53:41 -08002361 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002362 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002363 }
2364
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002365 DataSendParameters send_params = last_send_params_;
2366 RtpSendParametersFromMediaDescription<DataCodec>(data, &send_params);
2367 if (!media_channel()->SetSendParameters(send_params)) {
2368 SafeSetError("Failed to set remote data description send parameters.",
2369 error_desc);
2370 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002371 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002372 last_send_params_ = send_params;
2373
2374 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2375 // and only give it to the media channel once we have a local
2376 // description too (without a local description, we won't be able to
2377 // recv them anyway).
2378 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2379 SafeSetError("Failed to set remote data description streams.",
2380 error_desc);
2381 return false;
2382 }
2383
2384 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002385 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002386 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002387}
2388
deadbeef953c2ce2017-01-09 14:53:41 -08002389void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002390 // Render incoming data if we're the active call, and we have the local
2391 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002392 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002393 if (!media_channel()->SetReceive(recv)) {
2394 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2395 }
2396
2397 // Send outgoing data if we're the active call, we have the remote content,
2398 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002399 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002400 if (!media_channel()->SetSend(send)) {
2401 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2402 }
2403
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002404 // Trigger SignalReadyToSendData asynchronously.
2405 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002406
2407 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2408}
2409
deadbeef953c2ce2017-01-09 14:53:41 -08002410void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002411 switch (pmsg->message_id) {
2412 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002413 DataChannelReadyToSendMessageData* data =
2414 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002415 ready_to_send_data_ = data->data();
2416 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002417 delete data;
2418 break;
2419 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002420 case MSG_DATARECEIVED: {
2421 DataReceivedMessageData* data =
2422 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002423 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002424 delete data;
2425 break;
2426 }
2427 case MSG_CHANNEL_ERROR: {
2428 const DataChannelErrorMessageData* data =
2429 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002430 delete data;
2431 break;
2432 }
2433 default:
2434 BaseChannel::OnMessage(pmsg);
2435 break;
2436 }
2437}
2438
deadbeef953c2ce2017-01-09 14:53:41 -08002439void RtpDataChannel::OnConnectionMonitorUpdate(
2440 ConnectionMonitor* monitor,
2441 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002442 SignalConnectionMonitor(this, infos);
2443}
2444
deadbeef953c2ce2017-01-09 14:53:41 -08002445void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002446 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002447 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002448 media_monitor_->SignalUpdate.connect(this,
2449 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002450 media_monitor_->Start(cms);
2451}
2452
deadbeef953c2ce2017-01-09 14:53:41 -08002453void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002454 if (media_monitor_) {
2455 media_monitor_->Stop();
2456 media_monitor_->SignalUpdate.disconnect(this);
2457 media_monitor_.reset();
2458 }
2459}
2460
deadbeef953c2ce2017-01-09 14:53:41 -08002461void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2462 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002463 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002464 SignalMediaMonitor(this, info);
2465}
2466
deadbeef953c2ce2017-01-09 14:53:41 -08002467void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2468 const char* data,
2469 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002470 DataReceivedMessageData* msg = new DataReceivedMessageData(
2471 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002472 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002473}
2474
deadbeef953c2ce2017-01-09 14:53:41 -08002475void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2476 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002477 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2478 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002479 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002480}
2481
deadbeef953c2ce2017-01-09 14:53:41 -08002482void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002483 // This is usded for congestion control to indicate that the stream is ready
2484 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2485 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002486 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002487 new DataChannelReadyToSendMessageData(writable));
2488}
2489
deadbeef953c2ce2017-01-09 14:53:41 -08002490void RtpDataChannel::GetSrtpCryptoSuites_n(
2491 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002492 GetSupportedDataCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002493}
2494
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002495} // namespace cricket