blob: 59fca4a9d20c287626b4f2d7abbaa33bbf79d3af [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
zsteind48dbda2017-04-04 19:45:57 -0700245 if (rtp_transport_.rtcp_mux_required()) {
deadbeefac22f702017-01-12 21:59:29 -0800246 rtcp_mux_filter_.SetActive();
247 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200248 return true;
249}
250
wu@webrtc.org78187522013-10-07 23:32:02 +0000251void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200252 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000253 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200254 // Packets arrive on the network thread, processing packets calls virtual
255 // functions, so need to stop this process in Deinit that is called in
256 // derived classes destructor.
257 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700258 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000259}
260
zhihuangb2cdd932017-01-19 16:54:25 -0800261void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
262 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800263 network_thread_->Invoke<void>(
264 RTC_FROM_HERE,
265 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
266 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000267}
268
deadbeeff5346592017-01-24 21:51:21 -0800269void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800270 rtc::PacketTransportInternal* rtp_packet_transport,
271 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800272 network_thread_->Invoke<void>(
273 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
274 rtp_packet_transport, rtcp_packet_transport));
275}
zhihuangf5b251b2017-01-12 19:37:48 -0800276
deadbeeff5346592017-01-24 21:51:21 -0800277void BaseChannel::SetTransports_n(
278 DtlsTransportInternal* rtp_dtls_transport,
279 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800280 rtc::PacketTransportInternal* rtp_packet_transport,
281 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800282 RTC_DCHECK(network_thread_->IsCurrent());
283 // Validate some assertions about the input.
284 RTC_DCHECK(rtp_packet_transport);
285 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
286 if (rtp_dtls_transport || rtcp_dtls_transport) {
287 // DTLS/non-DTLS pointers should be to the same object.
288 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
289 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
290 // Can't go from non-DTLS to DTLS.
zsteind48dbda2017-04-04 19:45:57 -0700291 RTC_DCHECK(!rtp_transport_.rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800292 } else {
293 // Can't go from DTLS to non-DTLS.
294 RTC_DCHECK(!rtp_dtls_transport_);
295 }
296 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800297 if (rtp_dtls_transport && rtcp_dtls_transport) {
298 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
299 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800300 }
deadbeeff5346592017-01-24 21:51:21 -0800301 std::string debug_name;
302 if (rtp_dtls_transport) {
303 transport_name_ = rtp_dtls_transport->transport_name();
304 debug_name = transport_name_;
305 } else {
306 debug_name = rtp_packet_transport->debug_name();
307 }
zsteind48dbda2017-04-04 19:45:57 -0700308 if (rtp_packet_transport == rtp_transport_.rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800309 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800310 return;
deadbeefcbecd352015-09-23 11:50:27 -0700311 }
312
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800313 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
314 // changes and wait until the DTLS handshake is complete to set the newly
315 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200316 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800317 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700318 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800319 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800320 srtp_filter_.ResetParams();
321 }
322
deadbeefac22f702017-01-12 21:59:29 -0800323 // If this BaseChannel doesn't require RTCP mux and we haven't fully
324 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800325 if (rtcp_packet_transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800326 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
deadbeeff5346592017-01-24 21:51:21 -0800327 << debug_name << " transport " << rtcp_packet_transport;
328 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000329 }
330
deadbeeff5346592017-01-24 21:51:21 -0800331 LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
332 << debug_name << " transport " << rtp_packet_transport;
333 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800334
deadbeefcbecd352015-09-23 11:50:27 -0700335 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700336 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200337 UpdateWritableState_n();
deadbeef062ce9f2016-08-26 21:42:15 -0700338 // We can only update ready-to-send after updating writability.
339 //
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700340 // On setting a new channel, assume it's ready to send if it's writable,
341 // because we have no way of knowing otherwise (the channel doesn't give us
342 // "was last send successful?").
343 //
344 // This won't always be accurate (the last SendPacket call from another
345 // BaseChannel could have resulted in an error), but even so, we'll just
346 // encounter the error again and update "ready to send" accordingly.
deadbeef062ce9f2016-08-26 21:42:15 -0700347 SetTransportChannelReadyToSend(
zsteind48dbda2017-04-04 19:45:57 -0700348 false, rtp_packet_transport && rtp_packet_transport->writable());
zhihuangb2cdd932017-01-19 16:54:25 -0800349 SetTransportChannelReadyToSend(
zsteind48dbda2017-04-04 19:45:57 -0700350 true, rtcp_packet_transport && rtcp_packet_transport->writable());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000351}
352
deadbeeff5346592017-01-24 21:51:21 -0800353void BaseChannel::SetTransport_n(
354 bool rtcp,
355 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800356 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200357 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800358 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800359 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700360 rtc::PacketTransportInternal* old_packet_transport =
361 rtcp ? rtp_transport_.rtcp_packet_transport()
362 : rtp_transport_.rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800363
deadbeeff5346592017-01-24 21:51:21 -0800364 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700365 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000366 return;
367 }
zhihuangb2cdd932017-01-19 16:54:25 -0800368
deadbeeff5346592017-01-24 21:51:21 -0800369 RTC_DCHECK(old_packet_transport != new_packet_transport);
370 if (old_dtls_transport) {
371 DisconnectFromDtlsTransport(old_dtls_transport);
372 } else if (old_packet_transport) {
373 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000374 }
375
zsteind48dbda2017-04-04 19:45:57 -0700376 if (rtcp) {
377 rtp_transport_.set_rtcp_packet_transport(new_packet_transport);
378 } else {
379 rtp_transport_.set_rtp_packet_transport(new_packet_transport);
380 }
deadbeeff5346592017-01-24 21:51:21 -0800381 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000382
deadbeeff5346592017-01-24 21:51:21 -0800383 // If there's no new transport, we're done after disconnecting from old one.
384 if (!new_packet_transport) {
385 return;
386 }
387
388 if (rtcp && new_dtls_transport) {
389 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
390 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
391 << "should never happen.";
392 }
393 if (new_dtls_transport) {
394 ConnectToDtlsTransport(new_dtls_transport);
395 } else {
396 ConnectToPacketTransport(new_packet_transport);
397 }
398 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
399 for (const auto& pair : socket_options) {
400 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800401 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000402}
403
deadbeeff5346592017-01-24 21:51:21 -0800404void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200405 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000406
zhihuangb2cdd932017-01-19 16:54:25 -0800407 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
408 transport->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
409 transport->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
410 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
411 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
412 transport->ice_transport()->SignalSelectedCandidatePairChanged.connect(
Honghai Zhangcc411c02016-03-29 17:27:21 -0700413 this, &BaseChannel::OnSelectedCandidatePairChanged);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000414}
415
deadbeeff5346592017-01-24 21:51:21 -0800416void BaseChannel::DisconnectFromDtlsTransport(
417 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200418 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800419 OnSelectedCandidatePairChanged(transport->ice_transport(), nullptr, -1,
420 false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000421
zhihuangb2cdd932017-01-19 16:54:25 -0800422 transport->SignalWritableState.disconnect(this);
423 transport->SignalReadPacket.disconnect(this);
424 transport->SignalReadyToSend.disconnect(this);
425 transport->SignalDtlsState.disconnect(this);
426 transport->SignalSentPacket.disconnect(this);
427 transport->ice_transport()->SignalSelectedCandidatePairChanged.disconnect(
428 this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000429}
430
deadbeeff5346592017-01-24 21:51:21 -0800431void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800432 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800433 RTC_DCHECK_RUN_ON(network_thread_);
434 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
435 transport->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
436 transport->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
437 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
438}
439
440void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800441 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800442 RTC_DCHECK_RUN_ON(network_thread_);
443 transport->SignalWritableState.disconnect(this);
444 transport->SignalReadPacket.disconnect(this);
445 transport->SignalReadyToSend.disconnect(this);
446 transport->SignalSentPacket.disconnect(this);
447}
448
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000449bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700450 worker_thread_->Invoke<void>(
451 RTC_FROM_HERE,
452 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
453 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000454 return true;
455}
456
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000457bool BaseChannel::AddRecvStream(const StreamParams& sp) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700458 return InvokeOnWorker(RTC_FROM_HERE,
459 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000460}
461
Peter Boström0c4e06b2015-10-07 12:23:21 +0200462bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700463 return InvokeOnWorker(RTC_FROM_HERE,
464 Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000465}
466
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000467bool BaseChannel::AddSendStream(const StreamParams& sp) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +0000468 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700469 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000470}
471
Peter Boström0c4e06b2015-10-07 12:23:21 +0200472bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700473 return InvokeOnWorker(RTC_FROM_HERE, Bind(&MediaChannel::RemoveSendStream,
474 media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000475}
476
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000477bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000478 ContentAction action,
479 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100480 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700481 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetLocalContent_w,
482 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000483}
484
485bool BaseChannel::SetRemoteContent(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::SetRemoteContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700489 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w,
490 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491}
492
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000493void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800494 // We pass in the BaseChannel instead of the rtp_dtls_transport_
495 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000496 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200497 // We pass in the network thread because on that thread connection monitor
498 // will call BaseChannel::GetConnectionStats which must be called on the
499 // network thread.
500 connection_monitor_.reset(
501 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000502 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000503 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000504 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000505}
506
507void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000508 if (connection_monitor_) {
509 connection_monitor_->Stop();
510 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000511 }
512}
513
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000514bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200515 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800516 if (!rtp_dtls_transport_) {
517 return false;
518 }
zhihuangb2cdd932017-01-19 16:54:25 -0800519 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800520}
521
522bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800523 // If this BaseChannel doesn't require RTCP mux and we haven't fully
524 // negotiated RTCP mux, we need an RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -0700525 return !rtp_transport_.rtcp_mux_required() &&
526 !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000527}
528
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700529bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000530 // Receive data if we are enabled and have local content,
531 return enabled() && IsReceiveContentDirection(local_content_direction_);
532}
533
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700534bool BaseChannel::IsReadyToSendMedia_w() const {
535 // Need to access some state updated on the network thread.
536 return network_thread_->Invoke<bool>(
537 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
538}
539
540bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000541 // Send outgoing data if we are enabled, have local and remote content,
542 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800543 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000544 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700545 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200546 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000547}
548
jbaucheec21bd2016-03-20 06:15:43 -0700549bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700550 const rtc::PacketOptions& options) {
551 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000552}
553
jbaucheec21bd2016-03-20 06:15:43 -0700554bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700555 const rtc::PacketOptions& options) {
556 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000557}
558
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000559int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000560 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200561 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700562 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200563}
564
565int BaseChannel::SetOption_n(SocketType type,
566 rtc::Socket::Option opt,
567 int value) {
568 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800569 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000570 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000571 case ST_RTP:
zsteind48dbda2017-04-04 19:45:57 -0700572 transport = rtp_transport_.rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700573 socket_options_.push_back(
574 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000575 break;
576 case ST_RTCP:
zsteind48dbda2017-04-04 19:45:57 -0700577 transport = rtp_transport_.rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700578 rtcp_socket_options_.push_back(
579 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000580 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000581 }
deadbeeff5346592017-01-24 21:51:21 -0800582 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000583}
584
deadbeef5bd5ca32017-02-10 11:31:50 -0800585void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteind48dbda2017-04-04 19:45:57 -0700586 RTC_DCHECK(transport == rtp_transport_.rtp_packet_transport() ||
587 transport == rtp_transport_.rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200588 RTC_DCHECK(network_thread_->IsCurrent());
589 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000590}
591
deadbeef5bd5ca32017-02-10 11:31:50 -0800592void BaseChannel::OnPacketRead(rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -0700593 const char* data,
594 size_t len,
595 const rtc::PacketTime& packet_time,
596 int flags) {
597 TRACE_EVENT0("webrtc", "BaseChannel::OnPacketRead");
598 // OnPacketRead gets called from P2PSocket; now pass data to MediaEngine
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200599 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000600
601 // When using RTCP multiplexing we might get RTCP packets on the RTP
602 // transport. We feed RTP traffic into the demuxer to determine if it is RTCP.
johand89ab142016-10-25 10:50:32 -0700603 bool rtcp = PacketIsRtcp(transport, data, len);
jbaucheec21bd2016-03-20 06:15:43 -0700604 rtc::CopyOnWriteBuffer packet(data, len);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000605 HandlePacket(rtcp, &packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000606}
607
deadbeef5bd5ca32017-02-10 11:31:50 -0800608void BaseChannel::OnReadyToSend(rtc::PacketTransportInternal* transport) {
zsteind48dbda2017-04-04 19:45:57 -0700609 RTC_DCHECK(transport == rtp_transport_.rtp_packet_transport() ||
610 transport == rtp_transport_.rtcp_packet_transport());
611 SetTransportChannelReadyToSend(
612 transport == rtp_transport_.rtcp_packet_transport(), true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000613}
614
zhihuangb2cdd932017-01-19 16:54:25 -0800615void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800616 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200617 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800618 return;
619 }
620
621 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
622 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800623 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800624 // TransportChannel) or when TransportChannel is attached after DTLS is
625 // negotiated.
626 if (state != DTLS_TRANSPORT_CONNECTED) {
627 srtp_filter_.ResetParams();
628 }
629}
630
Honghai Zhangcc411c02016-03-29 17:27:21 -0700631void BaseChannel::OnSelectedCandidatePairChanged(
zhihuangb2cdd932017-01-19 16:54:25 -0800632 IceTransportInternal* ice_transport,
Honghai Zhang52dce732016-03-31 12:37:31 -0700633 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700634 int last_sent_packet_id,
635 bool ready_to_send) {
deadbeeff5346592017-01-24 21:51:21 -0800636 RTC_DCHECK((rtp_dtls_transport_ &&
637 ice_transport == rtp_dtls_transport_->ice_transport()) ||
638 (rtcp_dtls_transport_ &&
639 ice_transport == rtcp_dtls_transport_->ice_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200640 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800641 selected_candidate_pair_ = selected_candidate_pair;
zhihuangb2cdd932017-01-19 16:54:25 -0800642 std::string transport_name = ice_transport->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700643 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700644 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700645 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700646 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700647 selected_candidate_pair->remote_candidate().network_id(),
648 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800649
650 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700651 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200652 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700653 RTC_FROM_HERE, worker_thread_,
654 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
655 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700656}
657
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700658void BaseChannel::SetTransportChannelReadyToSend(bool rtcp, bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200659 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700660 if (rtcp) {
661 rtcp_ready_to_send_ = ready;
662 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000663 rtp_ready_to_send_ = ready;
664 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000665
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200666 bool ready_to_send =
667 (rtp_ready_to_send_ &&
deadbeeff5346592017-01-24 21:51:21 -0800668 // In the case of rtcp mux |rtcp_packet_transport_| will be null.
zsteind48dbda2017-04-04 19:45:57 -0700669 (rtcp_ready_to_send_ || !rtp_transport_.rtcp_packet_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200670
671 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700672 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200673 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready_to_send));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000674}
675
deadbeef5bd5ca32017-02-10 11:31:50 -0800676bool BaseChannel::PacketIsRtcp(const rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -0700677 const char* data,
678 size_t len) {
zsteind48dbda2017-04-04 19:45:57 -0700679 return (transport == rtp_transport_.rtcp_packet_transport() ||
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000680 rtcp_mux_filter_.DemuxRtcp(data, static_cast<int>(len)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000681}
682
stefanc1aeaf02015-10-15 07:26:07 -0700683bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700684 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700685 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200686 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
687 // If the thread is not our network thread, we will post to our network
688 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000689 // synchronize access to all the pieces of the send path, including
690 // SRTP and the inner workings of the transport channels.
691 // The only downside is that we can't return a proper failure code if
692 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200693 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000694 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200695 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
696 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800697 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700698 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700699 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000700 return true;
701 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200702 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000703
704 // Now that we are on the correct thread, ensure we have a place to send this
705 // packet before doing anything. (We might get RTCP packets that we don't
706 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
707 // transport.
deadbeef5bd5ca32017-02-10 11:31:50 -0800708 rtc::PacketTransportInternal* transport =
zsteind48dbda2017-04-04 19:45:57 -0700709 (!rtcp || rtcp_mux_filter_.IsActive())
710 ? rtp_transport_.rtp_packet_transport()
711 : rtp_transport_.rtcp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800712 if (!transport || !transport->writable()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000713 return false;
714 }
715
716 // Protect ourselves against crazy data.
717 if (!ValidPacket(rtcp, packet)) {
718 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000719 << PacketType(rtcp)
720 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000721 return false;
722 }
723
stefanc1aeaf02015-10-15 07:26:07 -0700724 rtc::PacketOptions updated_options;
725 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000726 // Protect if needed.
727 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200728 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200730 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000731 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000732 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000733 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
734 // inside libsrtp for a RTP packet. A external HMAC module will be writing
735 // a fake HMAC value. This is ONLY done for a RTP packet.
736 // Socket layer will update rtp sendtime extension header if present in
737 // packet with current time before updating the HMAC.
738#if !defined(ENABLE_EXTERNAL_AUTH)
739 res = srtp_filter_.ProtectRtp(
740 data, len, static_cast<int>(packet->capacity()), &len);
741#else
jbauchd48f4882017-03-01 15:34:36 -0800742 if (!srtp_filter_.IsExternalAuthActive()) {
743 res = srtp_filter_.ProtectRtp(
744 data, len, static_cast<int>(packet->capacity()), &len);
745 } else {
746 updated_options.packet_time_params.rtp_sendtime_extension_id =
747 rtp_abs_sendtime_extn_id_;
748 res = srtp_filter_.ProtectRtp(
749 data, len, static_cast<int>(packet->capacity()), &len,
750 &updated_options.packet_time_params.srtp_packet_index);
751 // If protection succeeds, let's get auth params from srtp.
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000752 if (res) {
jbauchd48f4882017-03-01 15:34:36 -0800753 uint8_t* auth_key = NULL;
754 int key_len;
755 res = srtp_filter_.GetRtpAuthParams(
756 &auth_key, &key_len,
757 &updated_options.packet_time_params.srtp_auth_tag_len);
758 if (res) {
759 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
760 updated_options.packet_time_params.srtp_auth_key.assign(
761 auth_key, auth_key + key_len);
762 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000763 }
764 }
765#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000766 if (!res) {
767 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200768 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000769 GetRtpSeqNum(data, len, &seq_num);
770 GetRtpSsrc(data, len, &ssrc);
771 LOG(LS_ERROR) << "Failed to protect " << content_name_
772 << " RTP packet: size=" << len
773 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
774 return false;
775 }
776 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000777 res = srtp_filter_.ProtectRtcp(data, len,
778 static_cast<int>(packet->capacity()),
779 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000780 if (!res) {
781 int type = -1;
782 GetRtcpType(data, len, &type);
783 LOG(LS_ERROR) << "Failed to protect " << content_name_
784 << " RTCP packet: size=" << len << ", type=" << type;
785 return false;
786 }
787 }
788
789 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000790 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800791 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800792 // The audio/video engines may attempt to send RTCP packets as soon as the
793 // streams are created, so don't treat this as an error for RTCP.
794 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
795 if (rtcp) {
796 return false;
797 }
798 // However, there shouldn't be any RTP packets sent before SRTP is set up
799 // (and SetSend(true) is called).
800 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
801 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800802 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000803 return false;
804 }
805
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000806 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200807 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
zhihuangb2cdd932017-01-19 16:54:25 -0800808 int ret = transport->SendPacket(packet->data<char>(), packet->size(),
809 updated_options, flags);
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000810 if (ret != static_cast<int>(packet->size())) {
zhihuangb2cdd932017-01-19 16:54:25 -0800811 if (transport->GetError() == ENOTCONN) {
skvladc309e0e2016-07-28 17:15:20 -0700812 LOG(LS_WARNING) << "Got ENOTCONN from transport.";
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700813 SetTransportChannelReadyToSend(rtcp, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000814 }
815 return false;
816 }
817 return true;
818}
819
jbaucheec21bd2016-03-20 06:15:43 -0700820bool BaseChannel::WantsPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000821 // Protect ourselves against crazy data.
822 if (!ValidPacket(rtcp, packet)) {
823 LOG(LS_ERROR) << "Dropping incoming " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000824 << PacketType(rtcp)
825 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000826 return false;
827 }
pbos482b12e2015-11-16 10:19:58 -0800828 if (rtcp) {
829 // Permit all (seemingly valid) RTCP packets.
830 return true;
831 }
832 // Check whether we handle this payload.
jbaucheec21bd2016-03-20 06:15:43 -0700833 return bundle_filter_.DemuxPacket(packet->data(), packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000834}
835
jbaucheec21bd2016-03-20 06:15:43 -0700836void BaseChannel::HandlePacket(bool rtcp, rtc::CopyOnWriteBuffer* packet,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000837 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200838 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000839 if (!WantsPacket(rtcp, packet)) {
840 return;
841 }
842
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000843 // We are only interested in the first rtp packet because that
844 // indicates the media has started flowing.
845 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000846 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700847 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000848 }
849
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000850 // Unprotect the packet, if needed.
851 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200852 TRACE_EVENT0("webrtc", "SRTP Decode");
Karl Wiberg94784372015-04-20 14:03:07 +0200853 char* data = packet->data<char>();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000854 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000855 bool res;
856 if (!rtcp) {
857 res = srtp_filter_.UnprotectRtp(data, len, &len);
858 if (!res) {
859 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200860 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000861 GetRtpSeqNum(data, len, &seq_num);
862 GetRtpSsrc(data, len, &ssrc);
863 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
864 << " RTP packet: size=" << len
865 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
866 return;
867 }
868 } else {
869 res = srtp_filter_.UnprotectRtcp(data, len, &len);
870 if (!res) {
871 int type = -1;
872 GetRtcpType(data, len, &type);
873 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
874 << " RTCP packet: size=" << len << ", type=" << type;
875 return;
876 }
877 }
878
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000879 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800880 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000881 // Our session description indicates that SRTP is required, but we got a
882 // packet before our SRTP filter is active. This means either that
883 // a) we got SRTP packets before we received the SDES keys, in which case
884 // we can't decrypt it anyway, or
885 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800886 // transports, so we haven't yet extracted keys, even if DTLS did
887 // complete on the transport that the packets are being sent on. It's
888 // really good practice to wait for both RTP and RTCP to be good to go
889 // before sending media, to prevent weird failure modes, so it's fine
890 // for us to just eat packets here. This is all sidestepped if RTCP mux
891 // is used anyway.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000892 LOG(LS_WARNING) << "Can't process incoming " << PacketType(rtcp)
893 << " packet when SRTP is inactive and crypto is required";
894 return;
895 }
896
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200897 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700898 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200899 Bind(&BaseChannel::OnPacketReceived, this, rtcp, *packet, packet_time));
900}
901
902void BaseChannel::OnPacketReceived(bool rtcp,
903 const rtc::CopyOnWriteBuffer& packet,
904 const rtc::PacketTime& packet_time) {
905 RTC_DCHECK(worker_thread_->IsCurrent());
906 // Need to copy variable because OnRtcpReceived/OnPacketReceived
907 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
908 rtc::CopyOnWriteBuffer data(packet);
909 if (rtcp) {
910 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000911 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200912 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000913 }
914}
915
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000916bool BaseChannel::PushdownLocalDescription(
917 const SessionDescription* local_desc, ContentAction action,
918 std::string* error_desc) {
919 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000920 const MediaContentDescription* content_desc =
921 GetContentDescription(content_info);
922 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000923 !SetLocalContent(content_desc, action, error_desc)) {
924 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
925 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000926 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000927 return true;
928}
929
930bool BaseChannel::PushdownRemoteDescription(
931 const SessionDescription* remote_desc, ContentAction action,
932 std::string* error_desc) {
933 const ContentInfo* content_info = GetFirstContent(remote_desc);
934 const MediaContentDescription* content_desc =
935 GetContentDescription(content_info);
936 if (content_desc && content_info && !content_info->rejected &&
937 !SetRemoteContent(content_desc, action, error_desc)) {
938 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
939 return false;
940 }
941 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000942}
943
944void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700945 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000946 if (enabled_)
947 return;
948
949 LOG(LS_INFO) << "Channel enabled";
950 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700951 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000952}
953
954void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700955 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000956 if (!enabled_)
957 return;
958
959 LOG(LS_INFO) << "Channel disabled";
960 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700961 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000962}
963
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200964void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700965 rtc::PacketTransportInternal* rtp_packet_transport =
966 rtp_transport_.rtp_packet_transport();
967 rtc::PacketTransportInternal* rtcp_packet_transport =
968 rtp_transport_.rtcp_packet_transport();
969 if (rtp_packet_transport && rtp_packet_transport->writable() &&
970 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200971 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700972 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200973 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700974 }
975}
976
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200977void BaseChannel::ChannelWritable_n() {
978 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800979 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000980 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800981 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000982
deadbeefcbecd352015-09-23 11:50:27 -0700983 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000984 << (was_ever_writable_ ? "" : " for the first time");
985
michaelt79e05882016-11-08 02:50:09 -0800986 if (selected_candidate_pair_)
987 LOG(LS_INFO)
988 << "Using "
989 << selected_candidate_pair_->local_candidate().ToSensitiveString()
990 << "->"
991 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000992
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000993 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200994 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000995 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700996 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000997}
998
deadbeef953c2ce2017-01-09 14:53:41 -0800999void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001000 RTC_DCHECK(network_thread_->IsCurrent());
1001 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001002 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -08001003 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001004}
1005
deadbeef953c2ce2017-01-09 14:53:41 -08001006void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001007 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -08001008 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001009}
1010
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001011bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -08001012 // Since DTLS is applied to all transports, checking RTP should be enough.
1013 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001014}
1015
1016// This function returns true if either DTLS-SRTP is not in use
1017// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -08001018bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001019 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001020 bool ret = false;
1021
zhihuangb2cdd932017-01-19 16:54:25 -08001022 DtlsTransportInternal* transport =
1023 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -08001024 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -08001025 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001026
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001027 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001028
zhihuangb2cdd932017-01-19 16:54:25 -08001029 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001030 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001031 return false;
1032 }
1033
zhihuangb2cdd932017-01-19 16:54:25 -08001034 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name() << " "
1035 << PacketType(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001036
jbauchcb560652016-08-04 05:20:32 -07001037 int key_len;
1038 int salt_len;
1039 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
1040 &salt_len)) {
1041 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
1042 return false;
1043 }
1044
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001045 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -07001046 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001047
1048 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -08001049 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
1050 &dtls_buffer[0], dtls_buffer.size())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001051 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -08001052 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001053 return false;
1054 }
1055
1056 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -07001057 std::vector<unsigned char> client_write_key(key_len + salt_len);
1058 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001059 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -07001060 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
1061 offset += key_len;
1062 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
1063 offset += key_len;
1064 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
1065 offset += salt_len;
1066 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001067
1068 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001069 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -08001070 if (!transport->GetSslRole(&role)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00001071 LOG(LS_WARNING) << "GetSslRole failed";
1072 return false;
1073 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001074
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001075 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001076 send_key = &server_write_key;
1077 recv_key = &client_write_key;
1078 } else {
1079 send_key = &client_write_key;
1080 recv_key = &server_write_key;
1081 }
1082
zhihuangb2cdd932017-01-19 16:54:25 -08001083 if (rtcp) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001084 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
1085 static_cast<int>(send_key->size()),
1086 selected_crypto_suite, &(*recv_key)[0],
1087 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001088 } else {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001089 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
1090 static_cast<int>(send_key->size()),
1091 selected_crypto_suite, &(*recv_key)[0],
1092 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001093 }
1094
michaelt79e05882016-11-08 02:50:09 -08001095 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001096 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001097 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001098 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001099 UpdateTransportOverhead();
1100 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001101 return ret;
1102}
1103
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001104void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001105 if (srtp_filter_.IsActive()) {
1106 return;
1107 }
1108
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001109 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001110 return;
1111 }
1112
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001113 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001114 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001115 return;
1116 }
1117
zhihuangb2cdd932017-01-19 16:54:25 -08001118 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001119 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001120 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001121 return;
1122 }
1123 }
1124}
1125
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001126void BaseChannel::ChannelNotWritable_n() {
1127 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001128 if (!writable_)
1129 return;
1130
deadbeefcbecd352015-09-23 11:50:27 -07001131 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001132 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001133 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001134}
1135
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001136bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001137 const MediaContentDescription* content,
1138 ContentAction action,
1139 ContentSource src,
1140 std::string* error_desc) {
1141 if (action == CA_UPDATE) {
1142 // These parameters never get changed by a CA_UDPATE.
1143 return true;
1144 }
1145
deadbeef7af91dd2016-12-13 11:29:11 -08001146 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001147 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001148 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
1149 content, action, src, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001150}
1151
1152bool BaseChannel::SetRtpTransportParameters_n(
1153 const MediaContentDescription* content,
1154 ContentAction action,
1155 ContentSource src,
1156 std::string* error_desc) {
1157 RTC_DCHECK(network_thread_->IsCurrent());
1158
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001159 if (!SetSrtp_n(content->cryptos(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001160 return false;
1161 }
1162
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001163 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001164 return false;
1165 }
1166
1167 return true;
1168}
1169
zhihuangb2cdd932017-01-19 16:54:25 -08001170// |dtls| will be set to true if DTLS is active for transport and crypto is
1171// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001172bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1173 bool* dtls,
1174 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -08001175 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001176 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001177 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001178 return false;
1179 }
1180 return true;
1181}
1182
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001183bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001184 ContentAction action,
1185 ContentSource src,
1186 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001187 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001188 if (action == CA_UPDATE) {
1189 // no crypto params.
1190 return true;
1191 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001192 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001193 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001194 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001195 if (!ret) {
1196 return false;
1197 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001198 switch (action) {
1199 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001200 // If DTLS is already active on the channel, we could be renegotiating
1201 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001202 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001203 ret = srtp_filter_.SetOffer(cryptos, src);
1204 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001205 break;
1206 case CA_PRANSWER:
1207 // If we're doing DTLS-SRTP, we don't want to update the filter
1208 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001209 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001210 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1211 }
1212 break;
1213 case CA_ANSWER:
1214 // If we're doing DTLS-SRTP, we don't want to update the filter
1215 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001216 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001217 ret = srtp_filter_.SetAnswer(cryptos, src);
1218 }
1219 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001220 default:
1221 break;
1222 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001223 if (!ret) {
1224 SafeSetError("Failed to setup SRTP filter.", error_desc);
1225 return false;
1226 }
1227 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001228}
1229
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001230bool BaseChannel::SetRtcpMux_n(bool enable,
1231 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001232 ContentSource src,
1233 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001234 // Provide a more specific error message for the RTCP mux "require" policy
1235 // case.
zsteind48dbda2017-04-04 19:45:57 -07001236 if (rtp_transport_.rtcp_mux_required() && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001237 SafeSetError(
1238 "rtcpMuxPolicy is 'require', but media description does not "
1239 "contain 'a=rtcp-mux'.",
1240 error_desc);
1241 return false;
1242 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001243 bool ret = false;
1244 switch (action) {
1245 case CA_OFFER:
1246 ret = rtcp_mux_filter_.SetOffer(enable, src);
1247 break;
1248 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001249 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001250 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001251 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1252 break;
1253 case CA_ANSWER:
1254 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1255 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001256 // We permanently activated RTCP muxing; signal that we no longer need
1257 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001258 std::string debug_name =
1259 transport_name_.empty()
1260 ? rtp_transport_.rtp_packet_transport()->debug_name()
1261 : transport_name_;
deadbeefe814a0d2017-02-25 18:15:09 -08001262 ;
deadbeefcbecd352015-09-23 11:50:27 -07001263 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
deadbeefe814a0d2017-02-25 18:15:09 -08001264 << "; no longer need RTCP transport for " << debug_name;
zsteind48dbda2017-04-04 19:45:57 -07001265 if (rtp_transport_.rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001266 SetTransport_n(true, nullptr, nullptr);
1267 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001268 }
deadbeef062ce9f2016-08-26 21:42:15 -07001269 UpdateWritableState_n();
1270 SetTransportChannelReadyToSend(true, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001271 }
1272 break;
1273 case CA_UPDATE:
1274 // No RTCP mux info.
1275 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001276 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001277 default:
1278 break;
1279 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001280 if (!ret) {
1281 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1282 return false;
1283 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001284 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001285 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1286 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001287 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001288 // If the RTP transport is already writable, then so are we.
zsteind48dbda2017-04-04 19:45:57 -07001289 if (rtp_transport_.rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001290 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001291 }
1292 }
1293
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001294 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001295}
1296
1297bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001298 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001299 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001300}
1301
Peter Boström0c4e06b2015-10-07 12:23:21 +02001302bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001303 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001304 return media_channel()->RemoveRecvStream(ssrc);
1305}
1306
1307bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001308 ContentAction action,
1309 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001310 if (!(action == CA_OFFER || action == CA_ANSWER ||
1311 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001312 return false;
1313
1314 // If this is an update, streams only contain streams that have changed.
1315 if (action == CA_UPDATE) {
1316 for (StreamParamsVec::const_iterator it = streams.begin();
1317 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001318 const StreamParams* existing_stream =
1319 GetStreamByIds(local_streams_, it->groupid, it->id);
1320 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001321 if (media_channel()->AddSendStream(*it)) {
1322 local_streams_.push_back(*it);
1323 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1324 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001325 std::ostringstream desc;
1326 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1327 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001328 return false;
1329 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001330 } else if (existing_stream && !it->has_ssrcs()) {
1331 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001332 std::ostringstream desc;
1333 desc << "Failed to remove send stream with ssrc "
1334 << it->first_ssrc() << ".";
1335 SafeSetError(desc.str(), error_desc);
1336 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001337 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001338 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001339 } else {
1340 LOG(LS_WARNING) << "Ignore unsupported stream update";
1341 }
1342 }
1343 return true;
1344 }
1345 // Else streams are all the streams we want to send.
1346
1347 // Check for streams that have been removed.
1348 bool ret = true;
1349 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1350 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001351 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001352 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001353 std::ostringstream desc;
1354 desc << "Failed to remove send stream with ssrc "
1355 << it->first_ssrc() << ".";
1356 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001357 ret = false;
1358 }
1359 }
1360 }
1361 // Check for new streams.
1362 for (StreamParamsVec::const_iterator it = streams.begin();
1363 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001364 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001365 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001366 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001367 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001368 std::ostringstream desc;
1369 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1370 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001371 ret = false;
1372 }
1373 }
1374 }
1375 local_streams_ = streams;
1376 return ret;
1377}
1378
1379bool BaseChannel::UpdateRemoteStreams_w(
1380 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001381 ContentAction action,
1382 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001383 if (!(action == CA_OFFER || action == CA_ANSWER ||
1384 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001385 return false;
1386
1387 // If this is an update, streams only contain streams that have changed.
1388 if (action == CA_UPDATE) {
1389 for (StreamParamsVec::const_iterator it = streams.begin();
1390 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001391 const StreamParams* existing_stream =
1392 GetStreamByIds(remote_streams_, it->groupid, it->id);
1393 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001394 if (AddRecvStream_w(*it)) {
1395 remote_streams_.push_back(*it);
1396 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1397 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001398 std::ostringstream desc;
1399 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1400 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001401 return false;
1402 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001403 } else if (existing_stream && !it->has_ssrcs()) {
1404 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001405 std::ostringstream desc;
1406 desc << "Failed to remove remote stream with ssrc "
1407 << it->first_ssrc() << ".";
1408 SafeSetError(desc.str(), error_desc);
1409 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001410 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001411 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001412 } else {
1413 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001414 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001415 << " new stream = " << it->ToString();
1416 }
1417 }
1418 return true;
1419 }
1420 // Else streams are all the streams we want to receive.
1421
1422 // Check for streams that have been removed.
1423 bool ret = true;
1424 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1425 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001426 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001427 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001428 std::ostringstream desc;
1429 desc << "Failed to remove remote stream with ssrc "
1430 << it->first_ssrc() << ".";
1431 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001432 ret = false;
1433 }
1434 }
1435 }
1436 // Check for new streams.
1437 for (StreamParamsVec::const_iterator it = streams.begin();
1438 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001439 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001440 if (AddRecvStream_w(*it)) {
1441 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1442 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001443 std::ostringstream desc;
1444 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1445 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001446 ret = false;
1447 }
1448 }
1449 }
1450 remote_streams_ = streams;
1451 return ret;
1452}
1453
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001454void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001455 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001456// Absolute Send Time extension id is used only with external auth,
1457// so do not bother searching for it and making asyncronious call to set
1458// something that is not used.
1459#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001460 const webrtc::RtpExtension* send_time_extension =
1461 FindHeaderExtension(extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001462 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001463 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001464 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001465 RTC_FROM_HERE, network_thread_,
1466 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1467 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001468#endif
1469}
1470
1471void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1472 int rtp_abs_sendtime_extn_id) {
1473 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001474}
1475
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001476void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001477 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001478 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001479 case MSG_SEND_RTP_PACKET:
1480 case MSG_SEND_RTCP_PACKET: {
1481 RTC_DCHECK(network_thread_->IsCurrent());
1482 SendPacketMessageData* data =
1483 static_cast<SendPacketMessageData*>(pmsg->pdata);
1484 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1485 SendPacket(rtcp, &data->packet, data->options);
1486 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001487 break;
1488 }
1489 case MSG_FIRSTPACKETRECEIVED: {
1490 SignalFirstPacketReceived(this);
1491 break;
1492 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001493 }
1494}
1495
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001496void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001497 // Flush all remaining RTCP messages. This should only be called in
1498 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001499 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001500 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001501 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1502 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001503 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1504 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001505 }
1506}
1507
johand89ab142016-10-25 10:50:32 -07001508void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001509 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001510 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001511 RTC_DCHECK(network_thread_->IsCurrent());
1512 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001513 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001514 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1515}
1516
1517void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1518 RTC_DCHECK(worker_thread_->IsCurrent());
1519 SignalSentPacket(sent_packet);
1520}
1521
1522VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1523 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001524 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001525 MediaEngineInterface* media_engine,
1526 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001527 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001528 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001529 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001530 : BaseChannel(worker_thread,
1531 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001532 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001533 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001534 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001535 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001536 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001537 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001538 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001539
1540VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001541 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001542 StopAudioMonitor();
1543 StopMediaMonitor();
1544 // this can't be done in the base class, since it calls a virtual
1545 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001546 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001547}
1548
Peter Boström0c4e06b2015-10-07 12:23:21 +02001549bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001550 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001551 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001552 AudioSource* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001553 return InvokeOnWorker(RTC_FROM_HERE,
1554 Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001555 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001556}
1557
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001558// TODO(juberti): Handle early media the right way. We should get an explicit
1559// ringing message telling us to start playing local ringback, which we cancel
1560// if any early media actually arrives. For now, we do the opposite, which is
1561// to wait 1 second for early media, and start playing local ringback if none
1562// arrives.
1563void VoiceChannel::SetEarlyMedia(bool enable) {
1564 if (enable) {
1565 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001566 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1567 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001568 } else {
1569 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001570 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001571 }
1572}
1573
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001574bool VoiceChannel::CanInsertDtmf() {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001575 return InvokeOnWorker(
1576 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001577}
1578
Peter Boström0c4e06b2015-10-07 12:23:21 +02001579bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1580 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001581 int duration) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001582 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceChannel::InsertDtmf_w, this,
1583 ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001584}
1585
solenberg4bac9c52015-10-09 02:32:53 -07001586bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001587 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetOutputVolume,
1588 media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001589}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001590
Tommif888bb52015-12-12 01:37:01 +01001591void VoiceChannel::SetRawAudioSink(
1592 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001593 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1594 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001595 // passing. So we invoke to our own little routine that gets a pointer to
1596 // our local variable. This is OK since we're synchronously invoking.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001597 InvokeOnWorker(RTC_FROM_HERE,
1598 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001599}
1600
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001601webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001602 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001603 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001604}
1605
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001606webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1607 uint32_t ssrc) const {
1608 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001609}
1610
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001611bool VoiceChannel::SetRtpSendParameters(
1612 uint32_t ssrc,
1613 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001614 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001615 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001616 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001617}
1618
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001619bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1620 webrtc::RtpParameters parameters) {
1621 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1622}
1623
1624webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1625 uint32_t ssrc) const {
1626 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001627 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001628 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1629}
1630
1631webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1632 uint32_t ssrc) const {
1633 return media_channel()->GetRtpReceiveParameters(ssrc);
1634}
1635
1636bool VoiceChannel::SetRtpReceiveParameters(
1637 uint32_t ssrc,
1638 const webrtc::RtpParameters& parameters) {
1639 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001640 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001641 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1642}
1643
1644bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1645 webrtc::RtpParameters parameters) {
1646 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001647}
1648
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001649bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001650 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1651 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001652}
1653
hbos8d609f62017-04-10 07:39:05 -07001654std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1655 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
1656 RTC_FROM_HERE,
1657 Bind(&WebRtcVoiceMediaChannel::GetSources,
1658 static_cast<WebRtcVoiceMediaChannel*>(media_channel()), ssrc));
1659}
1660
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001661void VoiceChannel::StartMediaMonitor(int cms) {
1662 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001663 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001664 media_monitor_->SignalUpdate.connect(
1665 this, &VoiceChannel::OnMediaMonitorUpdate);
1666 media_monitor_->Start(cms);
1667}
1668
1669void VoiceChannel::StopMediaMonitor() {
1670 if (media_monitor_) {
1671 media_monitor_->Stop();
1672 media_monitor_->SignalUpdate.disconnect(this);
1673 media_monitor_.reset();
1674 }
1675}
1676
1677void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001678 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001679 audio_monitor_
1680 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1681 audio_monitor_->Start(cms);
1682}
1683
1684void VoiceChannel::StopAudioMonitor() {
1685 if (audio_monitor_) {
1686 audio_monitor_->Stop();
1687 audio_monitor_.reset();
1688 }
1689}
1690
1691bool VoiceChannel::IsAudioMonitorRunning() const {
1692 return (audio_monitor_.get() != NULL);
1693}
1694
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001695int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001696 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001697}
1698
1699int VoiceChannel::GetOutputLevel_w() {
1700 return media_channel()->GetOutputLevel();
1701}
1702
1703void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1704 media_channel()->GetActiveStreams(actives);
1705}
1706
deadbeef5bd5ca32017-02-10 11:31:50 -08001707void VoiceChannel::OnPacketRead(rtc::PacketTransportInternal* transport,
johand89ab142016-10-25 10:50:32 -07001708 const char* data,
1709 size_t len,
1710 const rtc::PacketTime& packet_time,
wu@webrtc.orga9890802013-12-13 00:21:03 +00001711 int flags) {
johand89ab142016-10-25 10:50:32 -07001712 BaseChannel::OnPacketRead(transport, data, len, packet_time, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001713 // Set a flag when we've received an RTP packet. If we're waiting for early
1714 // media, this will disable the timeout.
johand89ab142016-10-25 10:50:32 -07001715 if (!received_media_ && !PacketIsRtcp(transport, data, len)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001716 received_media_ = true;
1717 }
1718}
1719
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001720void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001721 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001722 invoker_.AsyncInvoke<void>(
1723 RTC_FROM_HERE, worker_thread_,
1724 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001725}
1726
michaelt79e05882016-11-08 02:50:09 -08001727int BaseChannel::GetTransportOverheadPerPacket() const {
1728 RTC_DCHECK(network_thread_->IsCurrent());
1729
1730 if (!selected_candidate_pair_)
1731 return 0;
1732
1733 int transport_overhead_per_packet = 0;
1734
1735 constexpr int kIpv4Overhaed = 20;
1736 constexpr int kIpv6Overhaed = 40;
1737 transport_overhead_per_packet +=
1738 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1739 ? kIpv4Overhaed
1740 : kIpv6Overhaed;
1741
1742 constexpr int kUdpOverhaed = 8;
1743 constexpr int kTcpOverhaed = 20;
1744 transport_overhead_per_packet +=
1745 selected_candidate_pair_->local_candidate().protocol() ==
1746 TCP_PROTOCOL_NAME
1747 ? kTcpOverhaed
1748 : kUdpOverhaed;
1749
1750 if (secure()) {
1751 int srtp_overhead = 0;
1752 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1753 transport_overhead_per_packet += srtp_overhead;
1754 }
1755
1756 return transport_overhead_per_packet;
1757}
1758
1759void BaseChannel::UpdateTransportOverhead() {
1760 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1761 if (transport_overhead_per_packet)
1762 invoker_.AsyncInvoke<void>(
1763 RTC_FROM_HERE, worker_thread_,
1764 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1765 transport_overhead_per_packet));
1766}
1767
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001768void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001769 // Render incoming data if we're the active call, and we have the local
1770 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001771 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001772 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001773
1774 // Send outgoing data if we're the active call, we have the remote content,
1775 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001776 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001777 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001778
1779 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1780}
1781
1782const ContentInfo* VoiceChannel::GetFirstContent(
1783 const SessionDescription* sdesc) {
1784 return GetFirstAudioContent(sdesc);
1785}
1786
1787bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001788 ContentAction action,
1789 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001790 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001791 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001792 LOG(LS_INFO) << "Setting local voice description";
1793
1794 const AudioContentDescription* audio =
1795 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001796 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001797 if (!audio) {
1798 SafeSetError("Can't find audio content in local description.", error_desc);
1799 return false;
1800 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001801
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001802 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001803 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001804 }
1805
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001806 AudioRecvParameters recv_params = last_recv_params_;
1807 RtpParametersFromMediaDescription(audio, &recv_params);
1808 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001809 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001810 error_desc);
1811 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001812 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001813 for (const AudioCodec& codec : audio->codecs()) {
1814 bundle_filter()->AddPayloadType(codec.id);
1815 }
1816 last_recv_params_ = recv_params;
1817
1818 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1819 // only give it to the media channel once we have a remote
1820 // description too (without a remote description, we won't be able
1821 // to send them anyway).
1822 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1823 SafeSetError("Failed to set local audio description streams.", error_desc);
1824 return false;
1825 }
1826
1827 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001828 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001829 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001830}
1831
1832bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001833 ContentAction action,
1834 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001835 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001836 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001837 LOG(LS_INFO) << "Setting remote voice description";
1838
1839 const AudioContentDescription* audio =
1840 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001841 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001842 if (!audio) {
1843 SafeSetError("Can't find audio content in remote description.", error_desc);
1844 return false;
1845 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001846
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001847 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001848 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001849 }
1850
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001851 AudioSendParameters send_params = last_send_params_;
1852 RtpSendParametersFromMediaDescription(audio, &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001853 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001854 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001855 }
skvladdc1c62c2016-03-16 19:07:43 -07001856
1857 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1858 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001859 SafeSetError("Failed to set remote audio description send parameters.",
1860 error_desc);
1861 return false;
1862 }
1863 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001864
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001865 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1866 // and only give it to the media channel once we have a local
1867 // description too (without a local description, we won't be able to
1868 // recv them anyway).
1869 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1870 SafeSetError("Failed to set remote audio description streams.", error_desc);
1871 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001872 }
1873
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001874 if (audio->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001875 MaybeCacheRtpAbsSendTimeHeaderExtension_w(audio->rtp_header_extensions());
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001876 }
1877
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001878 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001879 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001880 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001881}
1882
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001883void VoiceChannel::HandleEarlyMediaTimeout() {
1884 // This occurs on the main thread, not the worker thread.
1885 if (!received_media_) {
1886 LOG(LS_INFO) << "No early media received before timeout";
1887 SignalEarlyMediaTimeout(this);
1888 }
1889}
1890
Peter Boström0c4e06b2015-10-07 12:23:21 +02001891bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1892 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001893 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001894 if (!enabled()) {
1895 return false;
1896 }
solenberg1d63dd02015-12-02 12:35:09 -08001897 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001898}
1899
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001900void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001901 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001902 case MSG_EARLYMEDIATIMEOUT:
1903 HandleEarlyMediaTimeout();
1904 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001905 case MSG_CHANNEL_ERROR: {
1906 VoiceChannelErrorMessageData* data =
1907 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001908 delete data;
1909 break;
1910 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001911 default:
1912 BaseChannel::OnMessage(pmsg);
1913 break;
1914 }
1915}
1916
1917void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001918 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001919 SignalConnectionMonitor(this, infos);
1920}
1921
1922void VoiceChannel::OnMediaMonitorUpdate(
1923 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001924 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001925 SignalMediaMonitor(this, info);
1926}
1927
1928void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1929 const AudioInfo& info) {
1930 SignalAudioMonitor(this, info);
1931}
1932
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001933VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1934 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001935 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001936 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001937 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001938 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001939 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001940 : BaseChannel(worker_thread,
1941 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001942 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001943 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001944 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001945 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001946 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001947
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001948VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001949 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001950 StopMediaMonitor();
1951 // this can't be done in the base class, since it calls a virtual
1952 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001953
1954 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001955}
1956
nisse08582ff2016-02-04 01:24:52 -08001957bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001958 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001959 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001960 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001961 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001962 return true;
1963}
1964
deadbeef5a4a75a2016-06-02 16:23:38 -07001965bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001966 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001967 bool mute,
1968 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001969 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001970 return InvokeOnWorker(RTC_FROM_HERE,
1971 Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
deadbeef5a4a75a2016-06-02 16:23:38 -07001972 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001973}
1974
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001975webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001976 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001977 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001978}
1979
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001980webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1981 uint32_t ssrc) const {
1982 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001983}
1984
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001985bool VideoChannel::SetRtpSendParameters(
1986 uint32_t ssrc,
1987 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001988 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001989 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001990 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001991}
1992
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001993bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1994 webrtc::RtpParameters parameters) {
1995 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1996}
1997
1998webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1999 uint32_t ssrc) const {
2000 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002001 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002002 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
2003}
2004
2005webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
2006 uint32_t ssrc) const {
2007 return media_channel()->GetRtpReceiveParameters(ssrc);
2008}
2009
2010bool VideoChannel::SetRtpReceiveParameters(
2011 uint32_t ssrc,
2012 const webrtc::RtpParameters& parameters) {
2013 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002014 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07002015 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
2016}
2017
2018bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
2019 webrtc::RtpParameters parameters) {
2020 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07002021}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002022
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002023void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002024 // Send outgoing data if we're the active call, we have the remote content,
2025 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002026 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002027 if (!media_channel()->SetSend(send)) {
2028 LOG(LS_ERROR) << "Failed to SetSend on video channel";
2029 // TODO(gangji): Report error back to server.
2030 }
2031
Peter Boström34fbfff2015-09-24 19:20:30 +02002032 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002033}
2034
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00002035bool VideoChannel::GetStats(VideoMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002036 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
2037 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002038}
2039
2040void VideoChannel::StartMediaMonitor(int cms) {
2041 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002042 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002043 media_monitor_->SignalUpdate.connect(
2044 this, &VideoChannel::OnMediaMonitorUpdate);
2045 media_monitor_->Start(cms);
2046}
2047
2048void VideoChannel::StopMediaMonitor() {
2049 if (media_monitor_) {
2050 media_monitor_->Stop();
2051 media_monitor_.reset();
2052 }
2053}
2054
2055const ContentInfo* VideoChannel::GetFirstContent(
2056 const SessionDescription* sdesc) {
2057 return GetFirstVideoContent(sdesc);
2058}
2059
2060bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002061 ContentAction action,
2062 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002063 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002064 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002065 LOG(LS_INFO) << "Setting local video description";
2066
2067 const VideoContentDescription* video =
2068 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002069 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002070 if (!video) {
2071 SafeSetError("Can't find video content in local description.", error_desc);
2072 return false;
2073 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002074
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002075 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002076 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002077 }
2078
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002079 VideoRecvParameters recv_params = last_recv_params_;
2080 RtpParametersFromMediaDescription(video, &recv_params);
2081 if (!media_channel()->SetRecvParameters(recv_params)) {
2082 SafeSetError("Failed to set local video description recv parameters.",
2083 error_desc);
2084 return false;
2085 }
2086 for (const VideoCodec& codec : video->codecs()) {
2087 bundle_filter()->AddPayloadType(codec.id);
2088 }
2089 last_recv_params_ = recv_params;
2090
2091 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2092 // only give it to the media channel once we have a remote
2093 // description too (without a remote description, we won't be able
2094 // to send them anyway).
2095 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2096 SafeSetError("Failed to set local video description streams.", error_desc);
2097 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002098 }
2099
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002100 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002101 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002102 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002103}
2104
2105bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002106 ContentAction action,
2107 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002108 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002109 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002110 LOG(LS_INFO) << "Setting remote video description";
2111
2112 const VideoContentDescription* video =
2113 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002114 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002115 if (!video) {
2116 SafeSetError("Can't find video content in remote description.", error_desc);
2117 return false;
2118 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002119
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002120 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002121 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002122 }
2123
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002124 VideoSendParameters send_params = last_send_params_;
2125 RtpSendParametersFromMediaDescription(video, &send_params);
2126 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002127 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002128 }
skvladdc1c62c2016-03-16 19:07:43 -07002129
2130 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2131
2132 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002133 SafeSetError("Failed to set remote video description send parameters.",
2134 error_desc);
2135 return false;
2136 }
2137 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002138
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002139 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2140 // and only give it to the media channel once we have a local
2141 // description too (without a local description, we won't be able to
2142 // recv them anyway).
2143 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2144 SafeSetError("Failed to set remote video description streams.", error_desc);
2145 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002146 }
2147
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002148 if (video->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002149 MaybeCacheRtpAbsSendTimeHeaderExtension_w(video->rtp_header_extensions());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002150 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002151
2152 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002153 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002154 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002155}
2156
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002157void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002158 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002159 case MSG_CHANNEL_ERROR: {
2160 const VideoChannelErrorMessageData* data =
2161 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002162 delete data;
2163 break;
2164 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002165 default:
2166 BaseChannel::OnMessage(pmsg);
2167 break;
2168 }
2169}
2170
2171void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002172 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002173 SignalConnectionMonitor(this, infos);
2174}
2175
2176// TODO(pthatcher): Look into removing duplicate code between
2177// audio, video, and data, perhaps by using templates.
2178void VideoChannel::OnMediaMonitorUpdate(
2179 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002180 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002181 SignalMediaMonitor(this, info);
2182}
2183
deadbeef953c2ce2017-01-09 14:53:41 -08002184RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2185 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002186 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002187 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002188 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002189 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002190 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002191 : BaseChannel(worker_thread,
2192 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002193 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002194 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002195 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002196 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002197 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002198
deadbeef953c2ce2017-01-09 14:53:41 -08002199RtpDataChannel::~RtpDataChannel() {
2200 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002201 StopMediaMonitor();
2202 // this can't be done in the base class, since it calls a virtual
2203 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002204
2205 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002206}
2207
deadbeeff5346592017-01-24 21:51:21 -08002208bool RtpDataChannel::Init_w(
2209 DtlsTransportInternal* rtp_dtls_transport,
2210 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002211 rtc::PacketTransportInternal* rtp_packet_transport,
2212 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -08002213 if (!BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2214 rtp_packet_transport, rtcp_packet_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002215 return false;
2216 }
deadbeef953c2ce2017-01-09 14:53:41 -08002217 media_channel()->SignalDataReceived.connect(this,
2218 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002219 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002220 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002221 return true;
2222}
2223
deadbeef953c2ce2017-01-09 14:53:41 -08002224bool RtpDataChannel::SendData(const SendDataParams& params,
2225 const rtc::CopyOnWriteBuffer& payload,
2226 SendDataResult* result) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002227 return InvokeOnWorker(
2228 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2229 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002230}
2231
deadbeef953c2ce2017-01-09 14:53:41 -08002232const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002233 const SessionDescription* sdesc) {
2234 return GetFirstDataContent(sdesc);
2235}
2236
deadbeef953c2ce2017-01-09 14:53:41 -08002237bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002238 const DataContentDescription* content,
2239 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002240 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2241 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002242 // It's been set before, but doesn't match. That's bad.
2243 if (is_sctp) {
2244 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2245 error_desc);
2246 return false;
2247 }
2248 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002249}
2250
deadbeef953c2ce2017-01-09 14:53:41 -08002251bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2252 ContentAction action,
2253 std::string* error_desc) {
2254 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002255 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002256 LOG(LS_INFO) << "Setting local data description";
2257
2258 const DataContentDescription* data =
2259 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002260 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002261 if (!data) {
2262 SafeSetError("Can't find data content in local description.", error_desc);
2263 return false;
2264 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002265
deadbeef953c2ce2017-01-09 14:53:41 -08002266 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002267 return false;
2268 }
2269
deadbeef953c2ce2017-01-09 14:53:41 -08002270 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
2271 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002272 }
2273
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002274 DataRecvParameters recv_params = last_recv_params_;
2275 RtpParametersFromMediaDescription(data, &recv_params);
2276 if (!media_channel()->SetRecvParameters(recv_params)) {
2277 SafeSetError("Failed to set remote data description recv parameters.",
2278 error_desc);
2279 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002280 }
deadbeef953c2ce2017-01-09 14:53:41 -08002281 for (const DataCodec& codec : data->codecs()) {
2282 bundle_filter()->AddPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002283 }
2284 last_recv_params_ = recv_params;
2285
2286 // TODO(pthatcher): Move local streams into DataSendParameters, and
2287 // only give it to the media channel once we have a remote
2288 // description too (without a remote description, we won't be able
2289 // to send them anyway).
2290 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2291 SafeSetError("Failed to set local data description streams.", error_desc);
2292 return false;
2293 }
2294
2295 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002296 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002297 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002298}
2299
deadbeef953c2ce2017-01-09 14:53:41 -08002300bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2301 ContentAction action,
2302 std::string* error_desc) {
2303 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002304 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002305
2306 const DataContentDescription* data =
2307 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002308 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002309 if (!data) {
2310 SafeSetError("Can't find data content in remote description.", error_desc);
2311 return false;
2312 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002313
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002314 // If the remote data doesn't have codecs and isn't an update, it
2315 // must be empty, so ignore it.
2316 if (!data->has_codecs() && action != CA_UPDATE) {
2317 return true;
2318 }
2319
deadbeef953c2ce2017-01-09 14:53:41 -08002320 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002321 return false;
2322 }
2323
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002324 LOG(LS_INFO) << "Setting remote data description";
deadbeef953c2ce2017-01-09 14:53:41 -08002325 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002326 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002327 }
2328
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002329 DataSendParameters send_params = last_send_params_;
2330 RtpSendParametersFromMediaDescription<DataCodec>(data, &send_params);
2331 if (!media_channel()->SetSendParameters(send_params)) {
2332 SafeSetError("Failed to set remote data description send parameters.",
2333 error_desc);
2334 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002335 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002336 last_send_params_ = send_params;
2337
2338 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2339 // and only give it to the media channel once we have a local
2340 // description too (without a local description, we won't be able to
2341 // recv them anyway).
2342 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2343 SafeSetError("Failed to set remote data description streams.",
2344 error_desc);
2345 return false;
2346 }
2347
2348 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002349 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002350 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002351}
2352
deadbeef953c2ce2017-01-09 14:53:41 -08002353void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002354 // Render incoming data if we're the active call, and we have the local
2355 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002356 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002357 if (!media_channel()->SetReceive(recv)) {
2358 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2359 }
2360
2361 // Send outgoing data if we're the active call, we have the remote content,
2362 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002363 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002364 if (!media_channel()->SetSend(send)) {
2365 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2366 }
2367
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002368 // Trigger SignalReadyToSendData asynchronously.
2369 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002370
2371 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2372}
2373
deadbeef953c2ce2017-01-09 14:53:41 -08002374void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002375 switch (pmsg->message_id) {
2376 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002377 DataChannelReadyToSendMessageData* data =
2378 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002379 ready_to_send_data_ = data->data();
2380 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002381 delete data;
2382 break;
2383 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002384 case MSG_DATARECEIVED: {
2385 DataReceivedMessageData* data =
2386 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002387 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002388 delete data;
2389 break;
2390 }
2391 case MSG_CHANNEL_ERROR: {
2392 const DataChannelErrorMessageData* data =
2393 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002394 delete data;
2395 break;
2396 }
2397 default:
2398 BaseChannel::OnMessage(pmsg);
2399 break;
2400 }
2401}
2402
deadbeef953c2ce2017-01-09 14:53:41 -08002403void RtpDataChannel::OnConnectionMonitorUpdate(
2404 ConnectionMonitor* monitor,
2405 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002406 SignalConnectionMonitor(this, infos);
2407}
2408
deadbeef953c2ce2017-01-09 14:53:41 -08002409void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002410 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002411 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002412 media_monitor_->SignalUpdate.connect(this,
2413 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002414 media_monitor_->Start(cms);
2415}
2416
deadbeef953c2ce2017-01-09 14:53:41 -08002417void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002418 if (media_monitor_) {
2419 media_monitor_->Stop();
2420 media_monitor_->SignalUpdate.disconnect(this);
2421 media_monitor_.reset();
2422 }
2423}
2424
deadbeef953c2ce2017-01-09 14:53:41 -08002425void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2426 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002427 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002428 SignalMediaMonitor(this, info);
2429}
2430
deadbeef953c2ce2017-01-09 14:53:41 -08002431void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2432 const char* data,
2433 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002434 DataReceivedMessageData* msg = new DataReceivedMessageData(
2435 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002436 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002437}
2438
deadbeef953c2ce2017-01-09 14:53:41 -08002439void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2440 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002441 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2442 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002443 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002444}
2445
deadbeef953c2ce2017-01-09 14:53:41 -08002446void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002447 // This is usded for congestion control to indicate that the stream is ready
2448 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2449 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002450 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002451 new DataChannelReadyToSendMessageData(writable));
2452}
2453
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002454} // namespace cricket