blob: e612aaece60d7dccb851b20c5814c06f71c2cf33 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellander65c7f672016-02-12 00:05:01 -08002 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellander65c7f672016-02-12 00:05:01 -08004 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
jbauch5869f502017-06-29 12:31:36 -070011#include <algorithm>
12#include <iterator>
kwiberg0eb15ed2015-12-17 03:04:15 -080013#include <utility>
14
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010015#include "webrtc/pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000016
kjellandera69d9732016-08-31 07:33:05 -070017#include "webrtc/api/call/audio_sink.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000018#include "webrtc/base/bind.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000019#include "webrtc/base/byteorder.h"
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -070020#include "webrtc/base/checks.h"
jbaucheec21bd2016-03-20 06:15:43 -070021#include "webrtc/base/copyonwritebuffer.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000022#include "webrtc/base/dscp.h"
23#include "webrtc/base/logging.h"
Honghai Zhangcc411c02016-03-29 17:27:21 -070024#include "webrtc/base/networkroute.h"
Peter Boström6f28cf02015-12-07 23:17:15 +010025#include "webrtc/base/trace_event.h"
kjellanderf4752772016-03-02 05:42:30 -080026#include "webrtc/media/base/mediaconstants.h"
kjellandera96e2d72016-02-04 23:52:28 -080027#include "webrtc/media/base/rtputils.h"
zhihuang38ede132017-06-15 12:52:32 -070028// Adding 'nogncheck' to disable the gn include headers check to support modular
29// WebRTC build targets.
30#include "webrtc/media/engine/webrtcvoiceengine.h" // nogncheck
deadbeef5bd5ca32017-02-10 11:31:50 -080031#include "webrtc/p2p/base/packettransportinternal.h"
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010032#include "webrtc/pc/channelmanager.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000033
34namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000035using rtc::Bind;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000036
deadbeef2d110be2016-01-13 12:00:26 -080037namespace {
kwiberg31022942016-03-11 14:18:21 -080038// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080039bool SetRawAudioSink_w(VoiceMediaChannel* channel,
40 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080041 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
42 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080043 return true;
44}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020045
46struct SendPacketMessageData : public rtc::MessageData {
47 rtc::CopyOnWriteBuffer packet;
48 rtc::PacketOptions options;
49};
50
deadbeef2d110be2016-01-13 12:00:26 -080051} // namespace
52
henrike@webrtc.org28e20752013-07-10 00:45:36 +000053enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000054 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020055 MSG_SEND_RTP_PACKET,
56 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000057 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000058 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000059 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000060 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000061};
62
63// Value specified in RFC 5764.
64static const char kDtlsSrtpExporterLabel[] = "EXTRACTOR-dtls_srtp";
65
66static const int kAgcMinus10db = -10;
67
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000068static void SafeSetError(const std::string& message, std::string* error_desc) {
69 if (error_desc) {
70 *error_desc = message;
71 }
72}
73
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000074struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020075 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000076 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020077 : ssrc(in_ssrc), error(in_error) {}
78 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000079 VoiceMediaChannel::Error error;
80};
81
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000082struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020083 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000084 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020085 : ssrc(in_ssrc), error(in_error) {}
86 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 VideoMediaChannel::Error error;
88};
89
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000090struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020091 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000092 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020093 : ssrc(in_ssrc), error(in_error) {}
94 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 DataMediaChannel::Error error;
96};
97
jbaucheec21bd2016-03-20 06:15:43 -070098static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000099 // Check the packet size. We could check the header too if needed.
zstein3dcf0e92017-06-01 13:22:42 -0700100 return packet && IsValidRtpRtcpPacketSize(rtcp, packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000101}
102
103static bool IsReceiveContentDirection(MediaContentDirection direction) {
104 return direction == MD_SENDRECV || direction == MD_RECVONLY;
105}
106
107static bool IsSendContentDirection(MediaContentDirection direction) {
108 return direction == MD_SENDRECV || direction == MD_SENDONLY;
109}
110
111static const MediaContentDescription* GetContentDescription(
112 const ContentInfo* cinfo) {
113 if (cinfo == NULL)
114 return NULL;
115 return static_cast<const MediaContentDescription*>(cinfo->description);
116}
117
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700118template <class Codec>
119void RtpParametersFromMediaDescription(
120 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700121 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700122 RtpParameters<Codec>* params) {
123 // TODO(pthatcher): Remove this once we're sure no one will give us
124 // a description without codecs (currently a CA_UPDATE with just
125 // streams can).
126 if (desc->has_codecs()) {
127 params->codecs = desc->codecs();
128 }
129 // TODO(pthatcher): See if we really need
130 // rtp_header_extensions_set() and remove it if we don't.
131 if (desc->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -0700132 params->extensions = extensions;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700133 }
deadbeef13871492015-12-09 12:37:51 -0800134 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700135}
136
nisse05103312016-03-16 02:22:50 -0700137template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700138void RtpSendParametersFromMediaDescription(
139 const MediaContentDescriptionImpl<Codec>* desc,
jbauch5869f502017-06-29 12:31:36 -0700140 const RtpHeaderExtensions& extensions,
nisse05103312016-03-16 02:22:50 -0700141 RtpSendParameters<Codec>* send_params) {
jbauch5869f502017-06-29 12:31:36 -0700142 RtpParametersFromMediaDescription(desc, extensions, send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700143 send_params->max_bandwidth_bps = desc->bandwidth();
144}
145
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200146BaseChannel::BaseChannel(rtc::Thread* worker_thread,
147 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800148 rtc::Thread* signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -0700149 MediaChannel* media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700150 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800151 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800152 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200153 : worker_thread_(worker_thread),
154 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800155 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000156 content_name_(content_name),
zstein56162b92017-04-24 16:54:35 -0700157 rtcp_mux_required_(rtcp_mux_required),
zsteind48dbda2017-04-04 19:45:57 -0700158 rtp_transport_(rtcp_mux_required),
deadbeef7af91dd2016-12-13 11:29:11 -0800159 srtp_required_(srtp_required),
michaelt79e05882016-11-08 02:50:09 -0800160 media_channel_(media_channel),
161 selected_candidate_pair_(nullptr) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700162 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
jbauchdfcab722017-03-06 00:14:10 -0800163#if defined(ENABLE_EXTERNAL_AUTH)
164 srtp_filter_.EnableExternalAuth();
165#endif
zstein56162b92017-04-24 16:54:35 -0700166 rtp_transport_.SignalReadyToSend.connect(
167 this, &BaseChannel::OnTransportReadyToSend);
zstein3dcf0e92017-06-01 13:22:42 -0700168 // TODO(zstein): RtpTransport::SignalPacketReceived will probably be replaced
169 // with a callback interface later so that the demuxer can select which
170 // channel to signal.
171 rtp_transport_.SignalPacketReceived.connect(this,
172 &BaseChannel::OnPacketReceived);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000173 LOG(LS_INFO) << "Created channel for " << content_name;
174}
175
176BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800177 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700178 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
wu@webrtc.org78187522013-10-07 23:32:02 +0000179 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000180 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200181 // Eats any outstanding messages or packets.
182 worker_thread_->Clear(&invoker_);
183 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000184 // We must destroy the media channel before the transport channel, otherwise
185 // the media channel may try to send on the dead transport channel. NULLing
186 // is not an effective strategy since the sends will come on another thread.
187 delete media_channel_;
zhihuangf5b251b2017-01-12 19:37:48 -0800188 LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200189}
190
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200191void BaseChannel::DisconnectTransportChannels_n() {
192 // Send any outstanding RTCP packets.
193 FlushRtcpMessages_n();
194
195 // Stop signals from transport channels, but keep them alive because
196 // media_channel may use them from a different thread.
zhihuangb2cdd932017-01-19 16:54:25 -0800197 if (rtp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800198 DisconnectFromDtlsTransport(rtp_dtls_transport_);
zsteind48dbda2017-04-04 19:45:57 -0700199 } else if (rtp_transport_.rtp_packet_transport()) {
200 DisconnectFromPacketTransport(rtp_transport_.rtp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200201 }
zhihuangb2cdd932017-01-19 16:54:25 -0800202 if (rtcp_dtls_transport_) {
deadbeeff5346592017-01-24 21:51:21 -0800203 DisconnectFromDtlsTransport(rtcp_dtls_transport_);
zsteind48dbda2017-04-04 19:45:57 -0700204 } else if (rtp_transport_.rtcp_packet_transport()) {
205 DisconnectFromPacketTransport(rtp_transport_.rtcp_packet_transport());
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200206 }
207
zstein3dcf0e92017-06-01 13:22:42 -0700208 rtp_transport_.SetRtpPacketTransport(nullptr);
209 rtp_transport_.SetRtcpPacketTransport(nullptr);
210
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200211 // Clear pending read packets/messages.
212 network_thread_->Clear(&invoker_);
213 network_thread_->Clear(this);
214}
215
zhihuangb2cdd932017-01-19 16:54:25 -0800216bool BaseChannel::Init_w(DtlsTransportInternal* rtp_dtls_transport,
deadbeeff5346592017-01-24 21:51:21 -0800217 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800218 rtc::PacketTransportInternal* rtp_packet_transport,
219 rtc::PacketTransportInternal* rtcp_packet_transport) {
skvlad6c87a672016-05-17 17:49:52 -0700220 if (!network_thread_->Invoke<bool>(
zhihuangb2cdd932017-01-19 16:54:25 -0800221 RTC_FROM_HERE, Bind(&BaseChannel::InitNetwork_n, this,
deadbeeff5346592017-01-24 21:51:21 -0800222 rtp_dtls_transport, rtcp_dtls_transport,
223 rtp_packet_transport, rtcp_packet_transport))) {
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000224 return false;
225 }
deadbeeff5346592017-01-24 21:51:21 -0800226 // Both RTP and RTCP channels should be set, we can call SetInterface on
227 // the media channel and it can set network options.
228 RTC_DCHECK_RUN_ON(worker_thread_);
wu@webrtc.orgde305012013-10-31 15:40:38 +0000229 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000230 return true;
231}
232
deadbeeff5346592017-01-24 21:51:21 -0800233bool BaseChannel::InitNetwork_n(
234 DtlsTransportInternal* rtp_dtls_transport,
235 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800236 rtc::PacketTransportInternal* rtp_packet_transport,
237 rtc::PacketTransportInternal* rtcp_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200238 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800239 SetTransports_n(rtp_dtls_transport, rtcp_dtls_transport, rtp_packet_transport,
240 rtcp_packet_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200241
zstein56162b92017-04-24 16:54:35 -0700242 if (rtcp_mux_required_) {
deadbeefac22f702017-01-12 21:59:29 -0800243 rtcp_mux_filter_.SetActive();
244 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200245 return true;
246}
247
wu@webrtc.org78187522013-10-07 23:32:02 +0000248void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200249 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000250 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200251 // Packets arrive on the network thread, processing packets calls virtual
252 // functions, so need to stop this process in Deinit that is called in
253 // derived classes destructor.
254 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700255 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000256}
257
zhihuangb2cdd932017-01-19 16:54:25 -0800258void BaseChannel::SetTransports(DtlsTransportInternal* rtp_dtls_transport,
259 DtlsTransportInternal* rtcp_dtls_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800260 network_thread_->Invoke<void>(
261 RTC_FROM_HERE,
262 Bind(&BaseChannel::SetTransports_n, this, rtp_dtls_transport,
263 rtcp_dtls_transport, rtp_dtls_transport, rtcp_dtls_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000264}
265
deadbeeff5346592017-01-24 21:51:21 -0800266void BaseChannel::SetTransports(
deadbeef5bd5ca32017-02-10 11:31:50 -0800267 rtc::PacketTransportInternal* rtp_packet_transport,
268 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800269 network_thread_->Invoke<void>(
270 RTC_FROM_HERE, Bind(&BaseChannel::SetTransports_n, this, nullptr, nullptr,
271 rtp_packet_transport, rtcp_packet_transport));
272}
zhihuangf5b251b2017-01-12 19:37:48 -0800273
deadbeeff5346592017-01-24 21:51:21 -0800274void BaseChannel::SetTransports_n(
275 DtlsTransportInternal* rtp_dtls_transport,
276 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800277 rtc::PacketTransportInternal* rtp_packet_transport,
278 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -0800279 RTC_DCHECK(network_thread_->IsCurrent());
280 // Validate some assertions about the input.
281 RTC_DCHECK(rtp_packet_transport);
282 RTC_DCHECK_EQ(NeedsRtcpTransport(), rtcp_packet_transport != nullptr);
283 if (rtp_dtls_transport || rtcp_dtls_transport) {
284 // DTLS/non-DTLS pointers should be to the same object.
285 RTC_DCHECK(rtp_dtls_transport == rtp_packet_transport);
286 RTC_DCHECK(rtcp_dtls_transport == rtcp_packet_transport);
287 // Can't go from non-DTLS to DTLS.
zsteind48dbda2017-04-04 19:45:57 -0700288 RTC_DCHECK(!rtp_transport_.rtp_packet_transport() || rtp_dtls_transport_);
deadbeeff5346592017-01-24 21:51:21 -0800289 } else {
290 // Can't go from DTLS to non-DTLS.
291 RTC_DCHECK(!rtp_dtls_transport_);
292 }
293 // Transport names should be the same.
zhihuangb2cdd932017-01-19 16:54:25 -0800294 if (rtp_dtls_transport && rtcp_dtls_transport) {
295 RTC_DCHECK(rtp_dtls_transport->transport_name() ==
296 rtcp_dtls_transport->transport_name());
zhihuangb2cdd932017-01-19 16:54:25 -0800297 }
deadbeeff5346592017-01-24 21:51:21 -0800298 std::string debug_name;
299 if (rtp_dtls_transport) {
300 transport_name_ = rtp_dtls_transport->transport_name();
301 debug_name = transport_name_;
302 } else {
303 debug_name = rtp_packet_transport->debug_name();
304 }
zsteind48dbda2017-04-04 19:45:57 -0700305 if (rtp_packet_transport == rtp_transport_.rtp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -0800306 // Nothing to do if transport isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800307 return;
deadbeefcbecd352015-09-23 11:50:27 -0700308 }
309
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800310 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
311 // changes and wait until the DTLS handshake is complete to set the newly
312 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200313 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800314 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700315 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800316 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800317 srtp_filter_.ResetParams();
318 }
319
deadbeefac22f702017-01-12 21:59:29 -0800320 // If this BaseChannel doesn't require RTCP mux and we haven't fully
321 // negotiated RTCP mux, we need an RTCP transport.
deadbeeff5346592017-01-24 21:51:21 -0800322 if (rtcp_packet_transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800323 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
deadbeeff5346592017-01-24 21:51:21 -0800324 << debug_name << " transport " << rtcp_packet_transport;
325 SetTransport_n(true, rtcp_dtls_transport, rtcp_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000326 }
327
deadbeeff5346592017-01-24 21:51:21 -0800328 LOG(LS_INFO) << "Setting RTP Transport for " << content_name() << " on "
329 << debug_name << " transport " << rtp_packet_transport;
330 SetTransport_n(false, rtp_dtls_transport, rtp_packet_transport);
guoweis46383312015-12-17 16:45:59 -0800331
deadbeefcbecd352015-09-23 11:50:27 -0700332 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700333 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200334 UpdateWritableState_n();
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000335}
336
deadbeeff5346592017-01-24 21:51:21 -0800337void BaseChannel::SetTransport_n(
338 bool rtcp,
339 DtlsTransportInternal* new_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -0800340 rtc::PacketTransportInternal* new_packet_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200341 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800342 DtlsTransportInternal*& old_dtls_transport =
zhihuangb2cdd932017-01-19 16:54:25 -0800343 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
zsteind48dbda2017-04-04 19:45:57 -0700344 rtc::PacketTransportInternal* old_packet_transport =
345 rtcp ? rtp_transport_.rtcp_packet_transport()
346 : rtp_transport_.rtp_packet_transport();
zhihuangb2cdd932017-01-19 16:54:25 -0800347
deadbeeff5346592017-01-24 21:51:21 -0800348 if (!old_packet_transport && !new_packet_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700349 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000350 return;
351 }
zhihuangb2cdd932017-01-19 16:54:25 -0800352
deadbeeff5346592017-01-24 21:51:21 -0800353 RTC_DCHECK(old_packet_transport != new_packet_transport);
354 if (old_dtls_transport) {
355 DisconnectFromDtlsTransport(old_dtls_transport);
356 } else if (old_packet_transport) {
357 DisconnectFromPacketTransport(old_packet_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000358 }
359
zsteind48dbda2017-04-04 19:45:57 -0700360 if (rtcp) {
zstein56162b92017-04-24 16:54:35 -0700361 rtp_transport_.SetRtcpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700362 } else {
zstein56162b92017-04-24 16:54:35 -0700363 rtp_transport_.SetRtpPacketTransport(new_packet_transport);
zsteind48dbda2017-04-04 19:45:57 -0700364 }
deadbeeff5346592017-01-24 21:51:21 -0800365 old_dtls_transport = new_dtls_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000366
deadbeeff5346592017-01-24 21:51:21 -0800367 // If there's no new transport, we're done after disconnecting from old one.
368 if (!new_packet_transport) {
369 return;
370 }
371
372 if (rtcp && new_dtls_transport) {
373 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
374 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
375 << "should never happen.";
376 }
zstein56162b92017-04-24 16:54:35 -0700377
deadbeeff5346592017-01-24 21:51:21 -0800378 if (new_dtls_transport) {
379 ConnectToDtlsTransport(new_dtls_transport);
380 } else {
381 ConnectToPacketTransport(new_packet_transport);
382 }
383 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
384 for (const auto& pair : socket_options) {
385 new_packet_transport->SetOption(pair.first, pair.second);
guoweis46383312015-12-17 16:45:59 -0800386 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000387}
388
deadbeeff5346592017-01-24 21:51:21 -0800389void BaseChannel::ConnectToDtlsTransport(DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200390 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000391
zstein56162b92017-04-24 16:54:35 -0700392 // TODO(zstein): de-dup with ConnectToPacketTransport
zhihuangb2cdd932017-01-19 16:54:25 -0800393 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
zhihuangb2cdd932017-01-19 16:54:25 -0800394 transport->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
395 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
396 transport->ice_transport()->SignalSelectedCandidatePairChanged.connect(
Honghai Zhangcc411c02016-03-29 17:27:21 -0700397 this, &BaseChannel::OnSelectedCandidatePairChanged);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000398}
399
deadbeeff5346592017-01-24 21:51:21 -0800400void BaseChannel::DisconnectFromDtlsTransport(
401 DtlsTransportInternal* transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200402 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -0800403 OnSelectedCandidatePairChanged(transport->ice_transport(), nullptr, -1,
404 false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000405
zhihuangb2cdd932017-01-19 16:54:25 -0800406 transport->SignalWritableState.disconnect(this);
zhihuangb2cdd932017-01-19 16:54:25 -0800407 transport->SignalDtlsState.disconnect(this);
408 transport->SignalSentPacket.disconnect(this);
409 transport->ice_transport()->SignalSelectedCandidatePairChanged.disconnect(
410 this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000411}
412
deadbeeff5346592017-01-24 21:51:21 -0800413void BaseChannel::ConnectToPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800414 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800415 RTC_DCHECK_RUN_ON(network_thread_);
416 transport->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
deadbeeff5346592017-01-24 21:51:21 -0800417 transport->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
418}
419
420void BaseChannel::DisconnectFromPacketTransport(
deadbeef5bd5ca32017-02-10 11:31:50 -0800421 rtc::PacketTransportInternal* transport) {
deadbeeff5346592017-01-24 21:51:21 -0800422 RTC_DCHECK_RUN_ON(network_thread_);
423 transport->SignalWritableState.disconnect(this);
deadbeeff5346592017-01-24 21:51:21 -0800424 transport->SignalSentPacket.disconnect(this);
425}
426
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000427bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700428 worker_thread_->Invoke<void>(
429 RTC_FROM_HERE,
430 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
431 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000432 return true;
433}
434
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000435bool BaseChannel::AddRecvStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700436 return InvokeOnWorker<bool>(RTC_FROM_HERE,
437 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000438}
439
Peter Boström0c4e06b2015-10-07 12:23:21 +0200440bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700441 return InvokeOnWorker<bool>(
442 RTC_FROM_HERE, Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000443}
444
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000445bool BaseChannel::AddSendStream(const StreamParams& sp) {
stefanf79ade12017-06-02 06:44:03 -0700446 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700447 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000448}
449
Peter Boström0c4e06b2015-10-07 12:23:21 +0200450bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
stefanf79ade12017-06-02 06:44:03 -0700451 return InvokeOnWorker<bool>(
452 RTC_FROM_HERE,
453 Bind(&MediaChannel::RemoveSendStream, media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000454}
455
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000456bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000457 ContentAction action,
458 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100459 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
stefanf79ade12017-06-02 06:44:03 -0700460 return InvokeOnWorker<bool>(
461 RTC_FROM_HERE,
462 Bind(&BaseChannel::SetLocalContent_w, this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000463}
464
465bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000466 ContentAction action,
467 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100468 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
stefanf79ade12017-06-02 06:44:03 -0700469 return InvokeOnWorker<bool>(
470 RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w, this, content,
471 action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000472}
473
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000474void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangb2cdd932017-01-19 16:54:25 -0800475 // We pass in the BaseChannel instead of the rtp_dtls_transport_
476 // because if the rtp_dtls_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000477 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200478 // We pass in the network thread because on that thread connection monitor
479 // will call BaseChannel::GetConnectionStats which must be called on the
480 // network thread.
481 connection_monitor_.reset(
482 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000483 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000484 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000485 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000486}
487
488void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000489 if (connection_monitor_) {
490 connection_monitor_->Stop();
491 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000492 }
493}
494
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000495bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200496 RTC_DCHECK(network_thread_->IsCurrent());
deadbeeff5346592017-01-24 21:51:21 -0800497 if (!rtp_dtls_transport_) {
498 return false;
499 }
zhihuangb2cdd932017-01-19 16:54:25 -0800500 return rtp_dtls_transport_->ice_transport()->GetStats(infos);
zhihuangf5b251b2017-01-12 19:37:48 -0800501}
502
503bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800504 // If this BaseChannel doesn't require RTCP mux and we haven't fully
505 // negotiated RTCP mux, we need an RTCP transport.
zstein56162b92017-04-24 16:54:35 -0700506 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000507}
508
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700509bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000510 // Receive data if we are enabled and have local content,
511 return enabled() && IsReceiveContentDirection(local_content_direction_);
512}
513
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700514bool BaseChannel::IsReadyToSendMedia_w() const {
515 // Need to access some state updated on the network thread.
516 return network_thread_->Invoke<bool>(
517 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
518}
519
520bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000521 // Send outgoing data if we are enabled, have local and remote content,
522 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800523 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000524 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700525 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200526 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000527}
528
jbaucheec21bd2016-03-20 06:15:43 -0700529bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700530 const rtc::PacketOptions& options) {
531 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000532}
533
jbaucheec21bd2016-03-20 06:15:43 -0700534bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700535 const rtc::PacketOptions& options) {
536 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000537}
538
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000539int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000540 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200541 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700542 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200543}
544
545int BaseChannel::SetOption_n(SocketType type,
546 rtc::Socket::Option opt,
547 int value) {
548 RTC_DCHECK(network_thread_->IsCurrent());
deadbeef5bd5ca32017-02-10 11:31:50 -0800549 rtc::PacketTransportInternal* transport = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000550 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000551 case ST_RTP:
zsteind48dbda2017-04-04 19:45:57 -0700552 transport = rtp_transport_.rtp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700553 socket_options_.push_back(
554 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000555 break;
556 case ST_RTCP:
zsteind48dbda2017-04-04 19:45:57 -0700557 transport = rtp_transport_.rtcp_packet_transport();
deadbeefcbecd352015-09-23 11:50:27 -0700558 rtcp_socket_options_.push_back(
559 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000560 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000561 }
deadbeeff5346592017-01-24 21:51:21 -0800562 return transport ? transport->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000563}
564
deadbeef5bd5ca32017-02-10 11:31:50 -0800565void BaseChannel::OnWritableState(rtc::PacketTransportInternal* transport) {
zsteind48dbda2017-04-04 19:45:57 -0700566 RTC_DCHECK(transport == rtp_transport_.rtp_packet_transport() ||
567 transport == rtp_transport_.rtcp_packet_transport());
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200568 RTC_DCHECK(network_thread_->IsCurrent());
569 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000570}
571
zhihuangb2cdd932017-01-19 16:54:25 -0800572void BaseChannel::OnDtlsState(DtlsTransportInternal* transport,
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800573 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200574 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800575 return;
576 }
577
578 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
579 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
zhihuangb2cdd932017-01-19 16:54:25 -0800580 // cover other scenarios like the whole transport is writable (not just this
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800581 // TransportChannel) or when TransportChannel is attached after DTLS is
582 // negotiated.
583 if (state != DTLS_TRANSPORT_CONNECTED) {
584 srtp_filter_.ResetParams();
585 }
586}
587
Honghai Zhangcc411c02016-03-29 17:27:21 -0700588void BaseChannel::OnSelectedCandidatePairChanged(
zhihuangb2cdd932017-01-19 16:54:25 -0800589 IceTransportInternal* ice_transport,
Honghai Zhang52dce732016-03-31 12:37:31 -0700590 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700591 int last_sent_packet_id,
592 bool ready_to_send) {
deadbeeff5346592017-01-24 21:51:21 -0800593 RTC_DCHECK((rtp_dtls_transport_ &&
594 ice_transport == rtp_dtls_transport_->ice_transport()) ||
595 (rtcp_dtls_transport_ &&
596 ice_transport == rtcp_dtls_transport_->ice_transport()));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200597 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800598 selected_candidate_pair_ = selected_candidate_pair;
zhihuangb2cdd932017-01-19 16:54:25 -0800599 std::string transport_name = ice_transport->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700600 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700601 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700602 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700603 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700604 selected_candidate_pair->remote_candidate().network_id(),
605 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800606
607 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700608 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200609 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700610 RTC_FROM_HERE, worker_thread_,
611 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
612 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700613}
614
zstein56162b92017-04-24 16:54:35 -0700615void BaseChannel::OnTransportReadyToSend(bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200616 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700617 RTC_FROM_HERE, worker_thread_,
zstein56162b92017-04-24 16:54:35 -0700618 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000619}
620
stefanc1aeaf02015-10-15 07:26:07 -0700621bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700622 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700623 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200624 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
625 // If the thread is not our network thread, we will post to our network
626 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000627 // synchronize access to all the pieces of the send path, including
628 // SRTP and the inner workings of the transport channels.
629 // The only downside is that we can't return a proper failure code if
630 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200631 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000632 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200633 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
634 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800635 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700636 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700637 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000638 return true;
639 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200640 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000641
642 // Now that we are on the correct thread, ensure we have a place to send this
643 // packet before doing anything. (We might get RTCP packets that we don't
644 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
645 // transport.
zstein56162b92017-04-24 16:54:35 -0700646 if (!rtp_transport_.IsWritable(rtcp)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000647 return false;
648 }
649
650 // Protect ourselves against crazy data.
651 if (!ValidPacket(rtcp, packet)) {
652 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
zstein3dcf0e92017-06-01 13:22:42 -0700653 << RtpRtcpStringLiteral(rtcp)
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000654 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000655 return false;
656 }
657
stefanc1aeaf02015-10-15 07:26:07 -0700658 rtc::PacketOptions updated_options;
659 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000660 // Protect if needed.
661 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200662 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000663 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200664 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000665 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000666 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000667 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
668 // inside libsrtp for a RTP packet. A external HMAC module will be writing
669 // a fake HMAC value. This is ONLY done for a RTP packet.
670 // Socket layer will update rtp sendtime extension header if present in
671 // packet with current time before updating the HMAC.
672#if !defined(ENABLE_EXTERNAL_AUTH)
673 res = srtp_filter_.ProtectRtp(
674 data, len, static_cast<int>(packet->capacity()), &len);
675#else
jbauchd48f4882017-03-01 15:34:36 -0800676 if (!srtp_filter_.IsExternalAuthActive()) {
677 res = srtp_filter_.ProtectRtp(
678 data, len, static_cast<int>(packet->capacity()), &len);
679 } else {
680 updated_options.packet_time_params.rtp_sendtime_extension_id =
681 rtp_abs_sendtime_extn_id_;
682 res = srtp_filter_.ProtectRtp(
683 data, len, static_cast<int>(packet->capacity()), &len,
684 &updated_options.packet_time_params.srtp_packet_index);
685 // If protection succeeds, let's get auth params from srtp.
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000686 if (res) {
jbauchd48f4882017-03-01 15:34:36 -0800687 uint8_t* auth_key = NULL;
688 int key_len;
689 res = srtp_filter_.GetRtpAuthParams(
690 &auth_key, &key_len,
691 &updated_options.packet_time_params.srtp_auth_tag_len);
692 if (res) {
693 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
694 updated_options.packet_time_params.srtp_auth_key.assign(
695 auth_key, auth_key + key_len);
696 }
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000697 }
698 }
699#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000700 if (!res) {
701 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200702 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000703 GetRtpSeqNum(data, len, &seq_num);
704 GetRtpSsrc(data, len, &ssrc);
705 LOG(LS_ERROR) << "Failed to protect " << content_name_
706 << " RTP packet: size=" << len
707 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
708 return false;
709 }
710 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000711 res = srtp_filter_.ProtectRtcp(data, len,
712 static_cast<int>(packet->capacity()),
713 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000714 if (!res) {
715 int type = -1;
716 GetRtcpType(data, len, &type);
717 LOG(LS_ERROR) << "Failed to protect " << content_name_
718 << " RTCP packet: size=" << len << ", type=" << type;
719 return false;
720 }
721 }
722
723 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000724 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800725 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800726 // The audio/video engines may attempt to send RTCP packets as soon as the
727 // streams are created, so don't treat this as an error for RTCP.
728 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
729 if (rtcp) {
730 return false;
731 }
732 // However, there shouldn't be any RTP packets sent before SRTP is set up
733 // (and SetSend(true) is called).
734 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
735 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800736 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000737 return false;
738 }
739
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000740 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200741 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
zstein56162b92017-04-24 16:54:35 -0700742 return rtp_transport_.SendPacket(rtcp, packet, updated_options, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000743}
744
zstein3dcf0e92017-06-01 13:22:42 -0700745bool BaseChannel::HandlesPayloadType(int packet_type) const {
746 return rtp_transport_.HandlesPayloadType(packet_type);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000747}
748
zstein3dcf0e92017-06-01 13:22:42 -0700749void BaseChannel::OnPacketReceived(bool rtcp,
750 rtc::CopyOnWriteBuffer& packet,
751 const rtc::PacketTime& packet_time) {
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000752 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000753 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700754 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000755 }
756
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000757 // Unprotect the packet, if needed.
758 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200759 TRACE_EVENT0("webrtc", "SRTP Decode");
zstein3dcf0e92017-06-01 13:22:42 -0700760 char* data = packet.data<char>();
761 int len = static_cast<int>(packet.size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000762 bool res;
763 if (!rtcp) {
764 res = srtp_filter_.UnprotectRtp(data, len, &len);
765 if (!res) {
766 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200767 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000768 GetRtpSeqNum(data, len, &seq_num);
769 GetRtpSsrc(data, len, &ssrc);
770 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
zstein3dcf0e92017-06-01 13:22:42 -0700771 << " RTP packet: size=" << len << ", seqnum=" << seq_num
772 << ", SSRC=" << ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000773 return;
774 }
775 } else {
776 res = srtp_filter_.UnprotectRtcp(data, len, &len);
777 if (!res) {
778 int type = -1;
779 GetRtcpType(data, len, &type);
780 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
781 << " RTCP packet: size=" << len << ", type=" << type;
782 return;
783 }
784 }
785
zstein3dcf0e92017-06-01 13:22:42 -0700786 packet.SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800787 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000788 // Our session description indicates that SRTP is required, but we got a
789 // packet before our SRTP filter is active. This means either that
790 // a) we got SRTP packets before we received the SDES keys, in which case
791 // we can't decrypt it anyway, or
792 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
zhihuangb2cdd932017-01-19 16:54:25 -0800793 // transports, so we haven't yet extracted keys, even if DTLS did
794 // complete on the transport that the packets are being sent on. It's
795 // really good practice to wait for both RTP and RTCP to be good to go
796 // before sending media, to prevent weird failure modes, so it's fine
797 // for us to just eat packets here. This is all sidestepped if RTCP mux
798 // is used anyway.
zstein3dcf0e92017-06-01 13:22:42 -0700799 LOG(LS_WARNING) << "Can't process incoming " << RtpRtcpStringLiteral(rtcp)
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000800 << " packet when SRTP is inactive and crypto is required";
801 return;
802 }
803
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200804 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700805 RTC_FROM_HERE, worker_thread_,
zstein3dcf0e92017-06-01 13:22:42 -0700806 Bind(&BaseChannel::ProcessPacket, this, rtcp, packet, packet_time));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200807}
808
zstein3dcf0e92017-06-01 13:22:42 -0700809void BaseChannel::ProcessPacket(bool rtcp,
810 const rtc::CopyOnWriteBuffer& packet,
811 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200812 RTC_DCHECK(worker_thread_->IsCurrent());
zstein3dcf0e92017-06-01 13:22:42 -0700813
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200814 // Need to copy variable because OnRtcpReceived/OnPacketReceived
815 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
816 rtc::CopyOnWriteBuffer data(packet);
817 if (rtcp) {
818 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000819 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200820 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000821 }
822}
823
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000824bool BaseChannel::PushdownLocalDescription(
825 const SessionDescription* local_desc, ContentAction action,
826 std::string* error_desc) {
827 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000828 const MediaContentDescription* content_desc =
829 GetContentDescription(content_info);
830 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000831 !SetLocalContent(content_desc, action, error_desc)) {
832 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
833 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000834 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000835 return true;
836}
837
838bool BaseChannel::PushdownRemoteDescription(
839 const SessionDescription* remote_desc, ContentAction action,
840 std::string* error_desc) {
841 const ContentInfo* content_info = GetFirstContent(remote_desc);
842 const MediaContentDescription* content_desc =
843 GetContentDescription(content_info);
844 if (content_desc && content_info && !content_info->rejected &&
845 !SetRemoteContent(content_desc, action, error_desc)) {
846 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
847 return false;
848 }
849 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000850}
851
852void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700853 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000854 if (enabled_)
855 return;
856
857 LOG(LS_INFO) << "Channel enabled";
858 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700859 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000860}
861
862void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700863 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000864 if (!enabled_)
865 return;
866
867 LOG(LS_INFO) << "Channel disabled";
868 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700869 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000870}
871
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200872void BaseChannel::UpdateWritableState_n() {
zsteind48dbda2017-04-04 19:45:57 -0700873 rtc::PacketTransportInternal* rtp_packet_transport =
874 rtp_transport_.rtp_packet_transport();
875 rtc::PacketTransportInternal* rtcp_packet_transport =
876 rtp_transport_.rtcp_packet_transport();
877 if (rtp_packet_transport && rtp_packet_transport->writable() &&
878 (!rtcp_packet_transport || rtcp_packet_transport->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200879 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700880 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200881 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700882 }
883}
884
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200885void BaseChannel::ChannelWritable_n() {
886 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800887 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000888 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800889 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000890
deadbeefcbecd352015-09-23 11:50:27 -0700891 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000892 << (was_ever_writable_ ? "" : " for the first time");
893
michaelt79e05882016-11-08 02:50:09 -0800894 if (selected_candidate_pair_)
895 LOG(LS_INFO)
896 << "Using "
897 << selected_candidate_pair_->local_candidate().ToSensitiveString()
898 << "->"
899 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000900
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000901 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200902 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000903 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700904 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905}
906
deadbeef953c2ce2017-01-09 14:53:41 -0800907void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200908 RTC_DCHECK(network_thread_->IsCurrent());
909 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700910 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800911 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000912}
913
deadbeef953c2ce2017-01-09 14:53:41 -0800914void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700915 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800916 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000917}
918
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200919bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
zhihuangb2cdd932017-01-19 16:54:25 -0800920 // Since DTLS is applied to all transports, checking RTP should be enough.
921 return rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000922}
923
924// This function returns true if either DTLS-SRTP is not in use
925// *or* DTLS-SRTP is successfully set up.
zhihuangb2cdd932017-01-19 16:54:25 -0800926bool BaseChannel::SetupDtlsSrtp_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200927 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000928 bool ret = false;
929
zhihuangb2cdd932017-01-19 16:54:25 -0800930 DtlsTransportInternal* transport =
931 rtcp ? rtcp_dtls_transport_ : rtp_dtls_transport_;
deadbeeff5346592017-01-24 21:51:21 -0800932 RTC_DCHECK(transport);
zhihuangb2cdd932017-01-19 16:54:25 -0800933 RTC_DCHECK(transport->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000934
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800935 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000936
zhihuangb2cdd932017-01-19 16:54:25 -0800937 if (!transport->GetSrtpCryptoSuite(&selected_crypto_suite)) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800938 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000939 return false;
940 }
941
zhihuangb2cdd932017-01-19 16:54:25 -0800942 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on " << content_name() << " "
zstein3dcf0e92017-06-01 13:22:42 -0700943 << RtpRtcpStringLiteral(rtcp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000944
jbauchcb560652016-08-04 05:20:32 -0700945 int key_len;
946 int salt_len;
947 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
948 &salt_len)) {
949 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
950 return false;
951 }
952
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000953 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700954 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000955
956 // RFC 5705 exporter using the RFC 5764 parameters
zhihuangb2cdd932017-01-19 16:54:25 -0800957 if (!transport->ExportKeyingMaterial(kDtlsSrtpExporterLabel, NULL, 0, false,
958 &dtls_buffer[0], dtls_buffer.size())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000959 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800960 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000961 return false;
962 }
963
964 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700965 std::vector<unsigned char> client_write_key(key_len + salt_len);
966 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000967 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700968 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
969 offset += key_len;
970 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
971 offset += key_len;
972 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
973 offset += salt_len;
974 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000975
976 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000977 rtc::SSLRole role;
zhihuangb2cdd932017-01-19 16:54:25 -0800978 if (!transport->GetSslRole(&role)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000979 LOG(LS_WARNING) << "GetSslRole failed";
980 return false;
981 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000982
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000983 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000984 send_key = &server_write_key;
985 recv_key = &client_write_key;
986 } else {
987 send_key = &client_write_key;
988 recv_key = &server_write_key;
989 }
990
jbauch5869f502017-06-29 12:31:36 -0700991 if (!srtp_filter_.IsActive()) {
992 if (rtcp) {
993 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
994 static_cast<int>(send_key->size()),
995 selected_crypto_suite, &(*recv_key)[0],
996 static_cast<int>(recv_key->size()));
997 } else {
998 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
999 static_cast<int>(send_key->size()),
1000 selected_crypto_suite, &(*recv_key)[0],
1001 static_cast<int>(recv_key->size()));
1002 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001003 } else {
jbauch5869f502017-06-29 12:31:36 -07001004 if (rtcp) {
1005 // RTCP doesn't need to be updated because UpdateRtpParams is only used
1006 // to update the set of encrypted RTP header extension IDs.
1007 ret = true;
1008 } else {
1009 ret = srtp_filter_.UpdateRtpParams(
1010 selected_crypto_suite,
1011 &(*send_key)[0], static_cast<int>(send_key->size()),
1012 selected_crypto_suite,
1013 &(*recv_key)[0], static_cast<int>(recv_key->size()));
1014 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001015 }
1016
michaelt79e05882016-11-08 02:50:09 -08001017 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001018 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001019 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001020 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001021 UpdateTransportOverhead();
1022 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001023 return ret;
1024}
1025
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001026void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001027 if (srtp_filter_.IsActive()) {
1028 return;
1029 }
1030
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001031 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001032 return;
1033 }
1034
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001035 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001036 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001037 return;
1038 }
1039
zhihuangb2cdd932017-01-19 16:54:25 -08001040 if (rtcp_dtls_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001041 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001042 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001043 return;
1044 }
1045 }
1046}
1047
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001048void BaseChannel::ChannelNotWritable_n() {
1049 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001050 if (!writable_)
1051 return;
1052
deadbeefcbecd352015-09-23 11:50:27 -07001053 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001054 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001055 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001056}
1057
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001058bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001059 const MediaContentDescription* content,
1060 ContentAction action,
1061 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001062 const RtpHeaderExtensions& extensions,
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001063 std::string* error_desc) {
1064 if (action == CA_UPDATE) {
1065 // These parameters never get changed by a CA_UDPATE.
1066 return true;
1067 }
1068
jbauch5869f502017-06-29 12:31:36 -07001069 std::vector<int> encrypted_extension_ids;
1070 for (const webrtc::RtpExtension& extension : extensions) {
1071 if (extension.encrypt) {
1072 LOG(LS_INFO) << "Using " << (src == CS_LOCAL ? "local" : "remote")
1073 << " encrypted extension: " << extension.ToString();
1074 encrypted_extension_ids.push_back(extension.id);
1075 }
1076 }
1077
deadbeef7af91dd2016-12-13 11:29:11 -08001078 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001079 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001080 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
jbauch5869f502017-06-29 12:31:36 -07001081 content, action, src, encrypted_extension_ids,
1082 error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001083}
1084
1085bool BaseChannel::SetRtpTransportParameters_n(
1086 const MediaContentDescription* content,
1087 ContentAction action,
1088 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001089 const std::vector<int>& encrypted_extension_ids,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001090 std::string* error_desc) {
1091 RTC_DCHECK(network_thread_->IsCurrent());
1092
jbauch5869f502017-06-29 12:31:36 -07001093 if (!SetSrtp_n(content->cryptos(), action, src, encrypted_extension_ids,
1094 error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001095 return false;
1096 }
1097
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001098 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001099 return false;
1100 }
1101
1102 return true;
1103}
1104
zhihuangb2cdd932017-01-19 16:54:25 -08001105// |dtls| will be set to true if DTLS is active for transport and crypto is
1106// empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001107bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1108 bool* dtls,
1109 std::string* error_desc) {
deadbeeff5346592017-01-24 21:51:21 -08001110 *dtls = rtp_dtls_transport_ && rtp_dtls_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001111 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001112 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001113 return false;
1114 }
1115 return true;
1116}
1117
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001118bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001119 ContentAction action,
1120 ContentSource src,
jbauch5869f502017-06-29 12:31:36 -07001121 const std::vector<int>& encrypted_extension_ids,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001122 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001123 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001124 if (action == CA_UPDATE) {
1125 // no crypto params.
1126 return true;
1127 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001128 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001129 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001130 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001131 if (!ret) {
1132 return false;
1133 }
jbauch5869f502017-06-29 12:31:36 -07001134 srtp_filter_.SetEncryptedHeaderExtensionIds(src, encrypted_extension_ids);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001135 switch (action) {
1136 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001137 // If DTLS is already active on the channel, we could be renegotiating
1138 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001139 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001140 ret = srtp_filter_.SetOffer(cryptos, src);
1141 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001142 break;
1143 case CA_PRANSWER:
1144 // If we're doing DTLS-SRTP, we don't want to update the filter
1145 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001146 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001147 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1148 }
1149 break;
1150 case CA_ANSWER:
1151 // If we're doing DTLS-SRTP, we don't want to update the filter
1152 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001153 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001154 ret = srtp_filter_.SetAnswer(cryptos, src);
1155 }
1156 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001157 default:
1158 break;
1159 }
jbauch5869f502017-06-29 12:31:36 -07001160 // Only update SRTP filter if using DTLS. SDES is handled internally
1161 // by the SRTP filter.
1162 // TODO(jbauch): Only update if encrypted extension ids have changed.
1163 if (ret && dtls_keyed_ && rtp_dtls_transport_ &&
1164 rtp_dtls_transport_->dtls_state() == DTLS_TRANSPORT_CONNECTED) {
1165 bool rtcp = false;
1166 ret = SetupDtlsSrtp_n(rtcp);
1167 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001168 if (!ret) {
1169 SafeSetError("Failed to setup SRTP filter.", error_desc);
1170 return false;
1171 }
1172 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001173}
1174
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001175bool BaseChannel::SetRtcpMux_n(bool enable,
1176 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001177 ContentSource src,
1178 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001179 // Provide a more specific error message for the RTCP mux "require" policy
1180 // case.
zstein56162b92017-04-24 16:54:35 -07001181 if (rtcp_mux_required_ && !enable) {
deadbeef8e814d72017-01-13 11:34:39 -08001182 SafeSetError(
1183 "rtcpMuxPolicy is 'require', but media description does not "
1184 "contain 'a=rtcp-mux'.",
1185 error_desc);
1186 return false;
1187 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001188 bool ret = false;
1189 switch (action) {
1190 case CA_OFFER:
1191 ret = rtcp_mux_filter_.SetOffer(enable, src);
1192 break;
1193 case CA_PRANSWER:
zhihuangb2cdd932017-01-19 16:54:25 -08001194 // This may activate RTCP muxing, but we don't yet destroy the transport
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001195 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001196 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1197 break;
1198 case CA_ANSWER:
1199 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1200 if (ret && rtcp_mux_filter_.IsActive()) {
deadbeefe814a0d2017-02-25 18:15:09 -08001201 // We permanently activated RTCP muxing; signal that we no longer need
1202 // the RTCP transport.
zsteind48dbda2017-04-04 19:45:57 -07001203 std::string debug_name =
1204 transport_name_.empty()
1205 ? rtp_transport_.rtp_packet_transport()->debug_name()
1206 : transport_name_;
deadbeefe814a0d2017-02-25 18:15:09 -08001207 ;
deadbeefcbecd352015-09-23 11:50:27 -07001208 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
deadbeefe814a0d2017-02-25 18:15:09 -08001209 << "; no longer need RTCP transport for " << debug_name;
zsteind48dbda2017-04-04 19:45:57 -07001210 if (rtp_transport_.rtcp_packet_transport()) {
deadbeeff5346592017-01-24 21:51:21 -08001211 SetTransport_n(true, nullptr, nullptr);
1212 SignalRtcpMuxFullyActive(transport_name_);
zhihuangf5b251b2017-01-12 19:37:48 -08001213 }
deadbeef062ce9f2016-08-26 21:42:15 -07001214 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001215 }
1216 break;
1217 case CA_UPDATE:
1218 // No RTCP mux info.
1219 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001220 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001221 default:
1222 break;
1223 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001224 if (!ret) {
1225 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1226 return false;
1227 }
zstein56162b92017-04-24 16:54:35 -07001228 rtp_transport_.SetRtcpMuxEnabled(rtcp_mux_filter_.IsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001229 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
zhihuangb2cdd932017-01-19 16:54:25 -08001230 // CA_ANSWER, but we only want to tear down the RTCP transport if we received
1231 // a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001232 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001233 // If the RTP transport is already writable, then so are we.
zsteind48dbda2017-04-04 19:45:57 -07001234 if (rtp_transport_.rtp_packet_transport()->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001235 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001236 }
1237 }
1238
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001239 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001240}
1241
1242bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001243 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001244 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001245}
1246
Peter Boström0c4e06b2015-10-07 12:23:21 +02001247bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001248 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001249 return media_channel()->RemoveRecvStream(ssrc);
1250}
1251
1252bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001253 ContentAction action,
1254 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001255 if (!(action == CA_OFFER || action == CA_ANSWER ||
1256 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001257 return false;
1258
1259 // If this is an update, streams only contain streams that have changed.
1260 if (action == CA_UPDATE) {
1261 for (StreamParamsVec::const_iterator it = streams.begin();
1262 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001263 const StreamParams* existing_stream =
1264 GetStreamByIds(local_streams_, it->groupid, it->id);
1265 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001266 if (media_channel()->AddSendStream(*it)) {
1267 local_streams_.push_back(*it);
1268 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1269 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001270 std::ostringstream desc;
1271 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1272 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001273 return false;
1274 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001275 } else if (existing_stream && !it->has_ssrcs()) {
1276 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001277 std::ostringstream desc;
1278 desc << "Failed to remove send stream with ssrc "
1279 << it->first_ssrc() << ".";
1280 SafeSetError(desc.str(), error_desc);
1281 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001282 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001283 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001284 } else {
1285 LOG(LS_WARNING) << "Ignore unsupported stream update";
1286 }
1287 }
1288 return true;
1289 }
1290 // Else streams are all the streams we want to send.
1291
1292 // Check for streams that have been removed.
1293 bool ret = true;
1294 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1295 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001296 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001297 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001298 std::ostringstream desc;
1299 desc << "Failed to remove send stream with ssrc "
1300 << it->first_ssrc() << ".";
1301 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001302 ret = false;
1303 }
1304 }
1305 }
1306 // Check for new streams.
1307 for (StreamParamsVec::const_iterator it = streams.begin();
1308 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001309 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001310 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001311 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001312 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001313 std::ostringstream desc;
1314 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1315 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001316 ret = false;
1317 }
1318 }
1319 }
1320 local_streams_ = streams;
1321 return ret;
1322}
1323
1324bool BaseChannel::UpdateRemoteStreams_w(
1325 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001326 ContentAction action,
1327 std::string* error_desc) {
nisse7ce109a2017-01-31 00:57:56 -08001328 if (!(action == CA_OFFER || action == CA_ANSWER ||
1329 action == CA_PRANSWER || action == CA_UPDATE))
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001330 return false;
1331
1332 // If this is an update, streams only contain streams that have changed.
1333 if (action == CA_UPDATE) {
1334 for (StreamParamsVec::const_iterator it = streams.begin();
1335 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001336 const StreamParams* existing_stream =
1337 GetStreamByIds(remote_streams_, it->groupid, it->id);
1338 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001339 if (AddRecvStream_w(*it)) {
1340 remote_streams_.push_back(*it);
1341 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1342 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001343 std::ostringstream desc;
1344 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1345 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001346 return false;
1347 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001348 } else if (existing_stream && !it->has_ssrcs()) {
1349 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001350 std::ostringstream desc;
1351 desc << "Failed to remove remote stream with ssrc "
1352 << it->first_ssrc() << ".";
1353 SafeSetError(desc.str(), error_desc);
1354 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001355 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001356 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001357 } else {
1358 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001359 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001360 << " new stream = " << it->ToString();
1361 }
1362 }
1363 return true;
1364 }
1365 // Else streams are all the streams we want to receive.
1366
1367 // Check for streams that have been removed.
1368 bool ret = true;
1369 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1370 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001371 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001372 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001373 std::ostringstream desc;
1374 desc << "Failed to remove remote stream with ssrc "
1375 << it->first_ssrc() << ".";
1376 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001377 ret = false;
1378 }
1379 }
1380 }
1381 // Check for new streams.
1382 for (StreamParamsVec::const_iterator it = streams.begin();
1383 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001384 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001385 if (AddRecvStream_w(*it)) {
1386 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1387 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001388 std::ostringstream desc;
1389 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1390 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001391 ret = false;
1392 }
1393 }
1394 }
1395 remote_streams_ = streams;
1396 return ret;
1397}
1398
jbauch5869f502017-06-29 12:31:36 -07001399RtpHeaderExtensions BaseChannel::GetFilteredRtpHeaderExtensions(
1400 const RtpHeaderExtensions& extensions) {
1401 if (!rtp_dtls_transport_ ||
1402 !rtp_dtls_transport_->crypto_options()
1403 .enable_encrypted_rtp_header_extensions) {
1404 RtpHeaderExtensions filtered;
1405 auto pred = [](const webrtc::RtpExtension& extension) {
1406 return !extension.encrypt;
1407 };
1408 std::copy_if(extensions.begin(), extensions.end(),
1409 std::back_inserter(filtered), pred);
1410 return filtered;
1411 }
1412
1413 return webrtc::RtpExtension::FilterDuplicateNonEncrypted(extensions);
1414}
1415
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001416void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001417 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001418// Absolute Send Time extension id is used only with external auth,
1419// so do not bother searching for it and making asyncronious call to set
1420// something that is not used.
1421#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001422 const webrtc::RtpExtension* send_time_extension =
jbauch5869f502017-06-29 12:31:36 -07001423 webrtc::RtpExtension::FindHeaderExtensionByUri(
1424 extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001425 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001426 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001427 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001428 RTC_FROM_HERE, network_thread_,
1429 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1430 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001431#endif
1432}
1433
1434void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1435 int rtp_abs_sendtime_extn_id) {
1436 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001437}
1438
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001439void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001440 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001441 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001442 case MSG_SEND_RTP_PACKET:
1443 case MSG_SEND_RTCP_PACKET: {
1444 RTC_DCHECK(network_thread_->IsCurrent());
1445 SendPacketMessageData* data =
1446 static_cast<SendPacketMessageData*>(pmsg->pdata);
1447 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1448 SendPacket(rtcp, &data->packet, data->options);
1449 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001450 break;
1451 }
1452 case MSG_FIRSTPACKETRECEIVED: {
1453 SignalFirstPacketReceived(this);
1454 break;
1455 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001456 }
1457}
1458
zstein3dcf0e92017-06-01 13:22:42 -07001459void BaseChannel::AddHandledPayloadType(int payload_type) {
1460 rtp_transport_.AddHandledPayloadType(payload_type);
1461}
1462
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001463void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001464 // Flush all remaining RTCP messages. This should only be called in
1465 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001466 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001467 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001468 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1469 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001470 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1471 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001472 }
1473}
1474
johand89ab142016-10-25 10:50:32 -07001475void BaseChannel::SignalSentPacket_n(
deadbeef5bd5ca32017-02-10 11:31:50 -08001476 rtc::PacketTransportInternal* /* transport */,
johand89ab142016-10-25 10:50:32 -07001477 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001478 RTC_DCHECK(network_thread_->IsCurrent());
1479 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001480 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001481 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1482}
1483
1484void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1485 RTC_DCHECK(worker_thread_->IsCurrent());
1486 SignalSentPacket(sent_packet);
1487}
1488
1489VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1490 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001491 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001492 MediaEngineInterface* media_engine,
1493 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001495 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001496 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001497 : BaseChannel(worker_thread,
1498 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001499 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001500 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001501 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001502 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001503 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001504 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001505 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001506
1507VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001508 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001509 StopAudioMonitor();
1510 StopMediaMonitor();
1511 // this can't be done in the base class, since it calls a virtual
1512 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001513 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001514}
1515
Peter Boström0c4e06b2015-10-07 12:23:21 +02001516bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001517 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001518 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001519 AudioSource* source) {
stefanf79ade12017-06-02 06:44:03 -07001520 return InvokeOnWorker<bool>(
1521 RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
1522 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001523}
1524
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001525// TODO(juberti): Handle early media the right way. We should get an explicit
1526// ringing message telling us to start playing local ringback, which we cancel
1527// if any early media actually arrives. For now, we do the opposite, which is
1528// to wait 1 second for early media, and start playing local ringback if none
1529// arrives.
1530void VoiceChannel::SetEarlyMedia(bool enable) {
1531 if (enable) {
1532 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001533 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1534 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001535 } else {
1536 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001537 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001538 }
1539}
1540
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001541bool VoiceChannel::CanInsertDtmf() {
stefanf79ade12017-06-02 06:44:03 -07001542 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001543 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001544}
1545
Peter Boström0c4e06b2015-10-07 12:23:21 +02001546bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1547 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001548 int duration) {
stefanf79ade12017-06-02 06:44:03 -07001549 return InvokeOnWorker<bool>(
1550 RTC_FROM_HERE,
1551 Bind(&VoiceChannel::InsertDtmf_w, this, ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001552}
1553
solenberg4bac9c52015-10-09 02:32:53 -07001554bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
stefanf79ade12017-06-02 06:44:03 -07001555 return InvokeOnWorker<bool>(
1556 RTC_FROM_HERE,
1557 Bind(&VoiceMediaChannel::SetOutputVolume, media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001558}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001559
Tommif888bb52015-12-12 01:37:01 +01001560void VoiceChannel::SetRawAudioSink(
1561 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001562 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1563 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001564 // passing. So we invoke to our own little routine that gets a pointer to
1565 // our local variable. This is OK since we're synchronously invoking.
stefanf79ade12017-06-02 06:44:03 -07001566 InvokeOnWorker<bool>(RTC_FROM_HERE,
1567 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001568}
1569
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001570webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001571 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001572 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001573}
1574
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001575webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1576 uint32_t ssrc) const {
1577 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001578}
1579
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001580bool VoiceChannel::SetRtpSendParameters(
1581 uint32_t ssrc,
1582 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001583 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001584 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001585 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001586}
1587
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001588bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1589 webrtc::RtpParameters parameters) {
1590 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1591}
1592
1593webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1594 uint32_t ssrc) const {
1595 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001596 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001597 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1598}
1599
1600webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1601 uint32_t ssrc) const {
1602 return media_channel()->GetRtpReceiveParameters(ssrc);
1603}
1604
1605bool VoiceChannel::SetRtpReceiveParameters(
1606 uint32_t ssrc,
1607 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001608 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001609 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001610 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1611}
1612
1613bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1614 webrtc::RtpParameters parameters) {
1615 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001616}
1617
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001618bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07001619 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1620 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001621}
1622
hbos8d609f62017-04-10 07:39:05 -07001623std::vector<webrtc::RtpSource> VoiceChannel::GetSources(uint32_t ssrc) const {
1624 return worker_thread()->Invoke<std::vector<webrtc::RtpSource>>(
zhihuang38ede132017-06-15 12:52:32 -07001625 RTC_FROM_HERE, Bind(&VoiceChannel::GetSources_w, this, ssrc));
1626}
1627
1628std::vector<webrtc::RtpSource> VoiceChannel::GetSources_w(uint32_t ssrc) const {
1629 RTC_DCHECK(worker_thread()->IsCurrent());
1630 return media_channel()->GetSources(ssrc);
hbos8d609f62017-04-10 07:39:05 -07001631}
1632
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001633void VoiceChannel::StartMediaMonitor(int cms) {
1634 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001635 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001636 media_monitor_->SignalUpdate.connect(
1637 this, &VoiceChannel::OnMediaMonitorUpdate);
1638 media_monitor_->Start(cms);
1639}
1640
1641void VoiceChannel::StopMediaMonitor() {
1642 if (media_monitor_) {
1643 media_monitor_->Stop();
1644 media_monitor_->SignalUpdate.disconnect(this);
1645 media_monitor_.reset();
1646 }
1647}
1648
1649void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001650 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001651 audio_monitor_
1652 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1653 audio_monitor_->Start(cms);
1654}
1655
1656void VoiceChannel::StopAudioMonitor() {
1657 if (audio_monitor_) {
1658 audio_monitor_->Stop();
1659 audio_monitor_.reset();
1660 }
1661}
1662
1663bool VoiceChannel::IsAudioMonitorRunning() const {
1664 return (audio_monitor_.get() != NULL);
1665}
1666
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001667int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001668 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001669}
1670
1671int VoiceChannel::GetOutputLevel_w() {
1672 return media_channel()->GetOutputLevel();
1673}
1674
1675void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1676 media_channel()->GetActiveStreams(actives);
1677}
1678
zstein3dcf0e92017-06-01 13:22:42 -07001679void VoiceChannel::OnPacketReceived(bool rtcp,
1680 rtc::CopyOnWriteBuffer& packet,
1681 const rtc::PacketTime& packet_time) {
1682 BaseChannel::OnPacketReceived(rtcp, packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001683 // Set a flag when we've received an RTP packet. If we're waiting for early
1684 // media, this will disable the timeout.
zstein3dcf0e92017-06-01 13:22:42 -07001685 if (!received_media_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001686 received_media_ = true;
1687 }
1688}
1689
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001690void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001691 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001692 invoker_.AsyncInvoke<void>(
1693 RTC_FROM_HERE, worker_thread_,
1694 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001695}
1696
michaelt79e05882016-11-08 02:50:09 -08001697int BaseChannel::GetTransportOverheadPerPacket() const {
1698 RTC_DCHECK(network_thread_->IsCurrent());
1699
1700 if (!selected_candidate_pair_)
1701 return 0;
1702
1703 int transport_overhead_per_packet = 0;
1704
1705 constexpr int kIpv4Overhaed = 20;
1706 constexpr int kIpv6Overhaed = 40;
1707 transport_overhead_per_packet +=
1708 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1709 ? kIpv4Overhaed
1710 : kIpv6Overhaed;
1711
1712 constexpr int kUdpOverhaed = 8;
1713 constexpr int kTcpOverhaed = 20;
1714 transport_overhead_per_packet +=
1715 selected_candidate_pair_->local_candidate().protocol() ==
1716 TCP_PROTOCOL_NAME
1717 ? kTcpOverhaed
1718 : kUdpOverhaed;
1719
1720 if (secure()) {
1721 int srtp_overhead = 0;
1722 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1723 transport_overhead_per_packet += srtp_overhead;
1724 }
1725
1726 return transport_overhead_per_packet;
1727}
1728
1729void BaseChannel::UpdateTransportOverhead() {
1730 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1731 if (transport_overhead_per_packet)
1732 invoker_.AsyncInvoke<void>(
1733 RTC_FROM_HERE, worker_thread_,
1734 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1735 transport_overhead_per_packet));
1736}
1737
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001738void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001739 // Render incoming data if we're the active call, and we have the local
1740 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001741 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001742 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001743
1744 // Send outgoing data if we're the active call, we have the remote content,
1745 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001746 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001747 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001748
1749 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1750}
1751
1752const ContentInfo* VoiceChannel::GetFirstContent(
1753 const SessionDescription* sdesc) {
1754 return GetFirstAudioContent(sdesc);
1755}
1756
1757bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001758 ContentAction action,
1759 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001760 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001761 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001762 LOG(LS_INFO) << "Setting local voice description";
1763
1764 const AudioContentDescription* audio =
1765 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001766 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001767 if (!audio) {
1768 SafeSetError("Can't find audio content in local description.", error_desc);
1769 return false;
1770 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001771
jbauch5869f502017-06-29 12:31:36 -07001772 RtpHeaderExtensions rtp_header_extensions =
1773 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1774
1775 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
1776 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001777 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001778 }
1779
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001780 AudioRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07001781 RtpParametersFromMediaDescription(audio, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001782 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001783 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001784 error_desc);
1785 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001786 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001787 for (const AudioCodec& codec : audio->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07001788 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001789 }
1790 last_recv_params_ = recv_params;
1791
1792 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1793 // only give it to the media channel once we have a remote
1794 // description too (without a remote description, we won't be able
1795 // to send them anyway).
1796 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1797 SafeSetError("Failed to set local audio description streams.", error_desc);
1798 return false;
1799 }
1800
1801 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001802 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001803 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001804}
1805
1806bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001807 ContentAction action,
1808 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001809 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001810 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001811 LOG(LS_INFO) << "Setting remote voice description";
1812
1813 const AudioContentDescription* audio =
1814 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001815 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001816 if (!audio) {
1817 SafeSetError("Can't find audio content in remote description.", error_desc);
1818 return false;
1819 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001820
jbauch5869f502017-06-29 12:31:36 -07001821 RtpHeaderExtensions rtp_header_extensions =
1822 GetFilteredRtpHeaderExtensions(audio->rtp_header_extensions());
1823
1824 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
1825 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001826 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827 }
1828
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001829 AudioSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07001830 RtpSendParametersFromMediaDescription(audio, rtp_header_extensions,
1831 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001832 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001833 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001834 }
skvladdc1c62c2016-03-16 19:07:43 -07001835
1836 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1837 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001838 SafeSetError("Failed to set remote audio description send parameters.",
1839 error_desc);
1840 return false;
1841 }
1842 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001843
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001844 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1845 // and only give it to the media channel once we have a local
1846 // description too (without a local description, we won't be able to
1847 // recv them anyway).
1848 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1849 SafeSetError("Failed to set remote audio description streams.", error_desc);
1850 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001851 }
1852
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001853 if (audio->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07001854 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001855 }
1856
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001857 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001858 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001859 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001860}
1861
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001862void VoiceChannel::HandleEarlyMediaTimeout() {
1863 // This occurs on the main thread, not the worker thread.
1864 if (!received_media_) {
1865 LOG(LS_INFO) << "No early media received before timeout";
1866 SignalEarlyMediaTimeout(this);
1867 }
1868}
1869
Peter Boström0c4e06b2015-10-07 12:23:21 +02001870bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1871 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001872 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001873 if (!enabled()) {
1874 return false;
1875 }
solenberg1d63dd02015-12-02 12:35:09 -08001876 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001877}
1878
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001879void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001880 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001881 case MSG_EARLYMEDIATIMEOUT:
1882 HandleEarlyMediaTimeout();
1883 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001884 case MSG_CHANNEL_ERROR: {
1885 VoiceChannelErrorMessageData* data =
1886 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001887 delete data;
1888 break;
1889 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001890 default:
1891 BaseChannel::OnMessage(pmsg);
1892 break;
1893 }
1894}
1895
1896void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001897 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001898 SignalConnectionMonitor(this, infos);
1899}
1900
1901void VoiceChannel::OnMediaMonitorUpdate(
1902 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001903 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001904 SignalMediaMonitor(this, info);
1905}
1906
1907void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1908 const AudioInfo& info) {
1909 SignalAudioMonitor(this, info);
1910}
1911
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001912VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1913 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001914 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001915 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001916 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001917 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001918 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001919 : BaseChannel(worker_thread,
1920 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001921 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001922 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001923 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001924 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001925 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001926
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001927VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001928 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001929 StopMediaMonitor();
1930 // this can't be done in the base class, since it calls a virtual
1931 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001932
1933 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001934}
1935
nisse08582ff2016-02-04 01:24:52 -08001936bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001937 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001938 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001939 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001940 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001941 return true;
1942}
1943
deadbeef5a4a75a2016-06-02 16:23:38 -07001944bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001945 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001946 bool mute,
1947 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001948 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
stefanf79ade12017-06-02 06:44:03 -07001949 return InvokeOnWorker<bool>(
1950 RTC_FROM_HERE, Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
1951 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001952}
1953
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001954webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001955 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001956 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001957}
1958
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001959webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1960 uint32_t ssrc) const {
1961 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001962}
1963
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001964bool VideoChannel::SetRtpSendParameters(
1965 uint32_t ssrc,
1966 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001967 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001968 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001969 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001970}
1971
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001972bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1973 webrtc::RtpParameters parameters) {
1974 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1975}
1976
1977webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1978 uint32_t ssrc) const {
1979 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001980 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001981 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1982}
1983
1984webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1985 uint32_t ssrc) const {
1986 return media_channel()->GetRtpReceiveParameters(ssrc);
1987}
1988
1989bool VideoChannel::SetRtpReceiveParameters(
1990 uint32_t ssrc,
1991 const webrtc::RtpParameters& parameters) {
stefanf79ade12017-06-02 06:44:03 -07001992 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001993 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001994 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1995}
1996
1997bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1998 webrtc::RtpParameters parameters) {
1999 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07002000}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002001
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002002void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002003 // Send outgoing data if we're the active call, we have the remote content,
2004 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002005 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002006 if (!media_channel()->SetSend(send)) {
2007 LOG(LS_ERROR) << "Failed to SetSend on video channel";
2008 // TODO(gangji): Report error back to server.
2009 }
2010
Peter Boström34fbfff2015-09-24 19:20:30 +02002011 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002012}
2013
stefanf79ade12017-06-02 06:44:03 -07002014void VideoChannel::FillBitrateInfo(BandwidthEstimationInfo* bwe_info) {
2015 InvokeOnWorker<void>(RTC_FROM_HERE, Bind(&VideoMediaChannel::FillBitrateInfo,
2016 media_channel(), bwe_info));
2017}
2018
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00002019bool VideoChannel::GetStats(VideoMediaInfo* stats) {
stefanf79ade12017-06-02 06:44:03 -07002020 return InvokeOnWorker<bool>(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
2021 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002022}
2023
2024void VideoChannel::StartMediaMonitor(int cms) {
2025 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002026 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002027 media_monitor_->SignalUpdate.connect(
2028 this, &VideoChannel::OnMediaMonitorUpdate);
2029 media_monitor_->Start(cms);
2030}
2031
2032void VideoChannel::StopMediaMonitor() {
2033 if (media_monitor_) {
2034 media_monitor_->Stop();
2035 media_monitor_.reset();
2036 }
2037}
2038
2039const ContentInfo* VideoChannel::GetFirstContent(
2040 const SessionDescription* sdesc) {
2041 return GetFirstVideoContent(sdesc);
2042}
2043
2044bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002045 ContentAction action,
2046 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002047 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002048 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002049 LOG(LS_INFO) << "Setting local video description";
2050
2051 const VideoContentDescription* video =
2052 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002053 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002054 if (!video) {
2055 SafeSetError("Can't find video content in local description.", error_desc);
2056 return false;
2057 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002058
jbauch5869f502017-06-29 12:31:36 -07002059 RtpHeaderExtensions rtp_header_extensions =
2060 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2061
2062 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2063 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002064 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002065 }
2066
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002067 VideoRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002068 RtpParametersFromMediaDescription(video, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002069 if (!media_channel()->SetRecvParameters(recv_params)) {
2070 SafeSetError("Failed to set local video description recv parameters.",
2071 error_desc);
2072 return false;
2073 }
2074 for (const VideoCodec& codec : video->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002075 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002076 }
2077 last_recv_params_ = recv_params;
2078
2079 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2080 // only give it to the media channel once we have a remote
2081 // description too (without a remote description, we won't be able
2082 // to send them anyway).
2083 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2084 SafeSetError("Failed to set local video description streams.", error_desc);
2085 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002086 }
2087
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002088 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002089 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002090 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002091}
2092
2093bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002094 ContentAction action,
2095 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002096 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002097 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002098 LOG(LS_INFO) << "Setting remote video description";
2099
2100 const VideoContentDescription* video =
2101 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002102 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002103 if (!video) {
2104 SafeSetError("Can't find video content in remote description.", error_desc);
2105 return false;
2106 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002107
jbauch5869f502017-06-29 12:31:36 -07002108 RtpHeaderExtensions rtp_header_extensions =
2109 GetFilteredRtpHeaderExtensions(video->rtp_header_extensions());
2110
2111 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2112 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002113 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002114 }
2115
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002116 VideoSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002117 RtpSendParametersFromMediaDescription(video, rtp_header_extensions,
2118 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002119 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002120 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002121 }
skvladdc1c62c2016-03-16 19:07:43 -07002122
2123 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2124
2125 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002126 SafeSetError("Failed to set remote video description send parameters.",
2127 error_desc);
2128 return false;
2129 }
2130 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002131
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002132 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2133 // and only give it to the media channel once we have a local
2134 // description too (without a local description, we won't be able to
2135 // recv them anyway).
2136 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2137 SafeSetError("Failed to set remote video description streams.", error_desc);
2138 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002139 }
2140
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002141 if (video->rtp_header_extensions_set()) {
jbauch5869f502017-06-29 12:31:36 -07002142 MaybeCacheRtpAbsSendTimeHeaderExtension_w(rtp_header_extensions);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002143 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002144
2145 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002146 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002147 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002148}
2149
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002150void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002151 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002152 case MSG_CHANNEL_ERROR: {
2153 const VideoChannelErrorMessageData* data =
2154 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002155 delete data;
2156 break;
2157 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002158 default:
2159 BaseChannel::OnMessage(pmsg);
2160 break;
2161 }
2162}
2163
2164void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002165 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002166 SignalConnectionMonitor(this, infos);
2167}
2168
2169// TODO(pthatcher): Look into removing duplicate code between
2170// audio, video, and data, perhaps by using templates.
2171void VideoChannel::OnMediaMonitorUpdate(
2172 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002173 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002174 SignalMediaMonitor(this, info);
2175}
2176
deadbeef953c2ce2017-01-09 14:53:41 -08002177RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2178 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002179 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002180 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002181 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002182 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002183 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002184 : BaseChannel(worker_thread,
2185 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002186 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002187 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002188 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002189 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002190 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002191
deadbeef953c2ce2017-01-09 14:53:41 -08002192RtpDataChannel::~RtpDataChannel() {
2193 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002194 StopMediaMonitor();
2195 // this can't be done in the base class, since it calls a virtual
2196 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002197
2198 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002199}
2200
deadbeeff5346592017-01-24 21:51:21 -08002201bool RtpDataChannel::Init_w(
2202 DtlsTransportInternal* rtp_dtls_transport,
2203 DtlsTransportInternal* rtcp_dtls_transport,
deadbeef5bd5ca32017-02-10 11:31:50 -08002204 rtc::PacketTransportInternal* rtp_packet_transport,
2205 rtc::PacketTransportInternal* rtcp_packet_transport) {
deadbeeff5346592017-01-24 21:51:21 -08002206 if (!BaseChannel::Init_w(rtp_dtls_transport, rtcp_dtls_transport,
2207 rtp_packet_transport, rtcp_packet_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002208 return false;
2209 }
deadbeef953c2ce2017-01-09 14:53:41 -08002210 media_channel()->SignalDataReceived.connect(this,
2211 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002212 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002213 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002214 return true;
2215}
2216
deadbeef953c2ce2017-01-09 14:53:41 -08002217bool RtpDataChannel::SendData(const SendDataParams& params,
2218 const rtc::CopyOnWriteBuffer& payload,
2219 SendDataResult* result) {
stefanf79ade12017-06-02 06:44:03 -07002220 return InvokeOnWorker<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002221 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2222 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002223}
2224
deadbeef953c2ce2017-01-09 14:53:41 -08002225const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002226 const SessionDescription* sdesc) {
2227 return GetFirstDataContent(sdesc);
2228}
2229
deadbeef953c2ce2017-01-09 14:53:41 -08002230bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002231 const DataContentDescription* content,
2232 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002233 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2234 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002235 // It's been set before, but doesn't match. That's bad.
2236 if (is_sctp) {
2237 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2238 error_desc);
2239 return false;
2240 }
2241 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002242}
2243
deadbeef953c2ce2017-01-09 14:53:41 -08002244bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2245 ContentAction action,
2246 std::string* error_desc) {
2247 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002248 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002249 LOG(LS_INFO) << "Setting local data description";
2250
2251 const DataContentDescription* data =
2252 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002253 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002254 if (!data) {
2255 SafeSetError("Can't find data content in local description.", error_desc);
2256 return false;
2257 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002258
deadbeef953c2ce2017-01-09 14:53:41 -08002259 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002260 return false;
2261 }
2262
jbauch5869f502017-06-29 12:31:36 -07002263 RtpHeaderExtensions rtp_header_extensions =
2264 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2265
2266 if (!SetRtpTransportParameters(content, action, CS_LOCAL,
2267 rtp_header_extensions, error_desc)) {
deadbeef953c2ce2017-01-09 14:53:41 -08002268 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002269 }
2270
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002271 DataRecvParameters recv_params = last_recv_params_;
jbauch5869f502017-06-29 12:31:36 -07002272 RtpParametersFromMediaDescription(data, rtp_header_extensions, &recv_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002273 if (!media_channel()->SetRecvParameters(recv_params)) {
2274 SafeSetError("Failed to set remote data description recv parameters.",
2275 error_desc);
2276 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002277 }
deadbeef953c2ce2017-01-09 14:53:41 -08002278 for (const DataCodec& codec : data->codecs()) {
zstein3dcf0e92017-06-01 13:22:42 -07002279 AddHandledPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002280 }
2281 last_recv_params_ = recv_params;
2282
2283 // TODO(pthatcher): Move local streams into DataSendParameters, and
2284 // only give it to the media channel once we have a remote
2285 // description too (without a remote description, we won't be able
2286 // to send them anyway).
2287 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2288 SafeSetError("Failed to set local data description streams.", error_desc);
2289 return false;
2290 }
2291
2292 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002293 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002294 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002295}
2296
deadbeef953c2ce2017-01-09 14:53:41 -08002297bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2298 ContentAction action,
2299 std::string* error_desc) {
2300 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002301 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002302
2303 const DataContentDescription* data =
2304 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002305 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002306 if (!data) {
2307 SafeSetError("Can't find data content in remote description.", error_desc);
2308 return false;
2309 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002310
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002311 // If the remote data doesn't have codecs and isn't an update, it
2312 // must be empty, so ignore it.
2313 if (!data->has_codecs() && action != CA_UPDATE) {
2314 return true;
2315 }
2316
deadbeef953c2ce2017-01-09 14:53:41 -08002317 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002318 return false;
2319 }
2320
jbauch5869f502017-06-29 12:31:36 -07002321 RtpHeaderExtensions rtp_header_extensions =
2322 GetFilteredRtpHeaderExtensions(data->rtp_header_extensions());
2323
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002324 LOG(LS_INFO) << "Setting remote data description";
jbauch5869f502017-06-29 12:31:36 -07002325 if (!SetRtpTransportParameters(content, action, CS_REMOTE,
2326 rtp_header_extensions, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002327 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002328 }
2329
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002330 DataSendParameters send_params = last_send_params_;
jbauch5869f502017-06-29 12:31:36 -07002331 RtpSendParametersFromMediaDescription<DataCodec>(data, rtp_header_extensions,
2332 &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002333 if (!media_channel()->SetSendParameters(send_params)) {
2334 SafeSetError("Failed to set remote data description send parameters.",
2335 error_desc);
2336 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002337 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002338 last_send_params_ = send_params;
2339
2340 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2341 // and only give it to the media channel once we have a local
2342 // description too (without a local description, we won't be able to
2343 // recv them anyway).
2344 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2345 SafeSetError("Failed to set remote data description streams.",
2346 error_desc);
2347 return false;
2348 }
2349
2350 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002351 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002352 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002353}
2354
deadbeef953c2ce2017-01-09 14:53:41 -08002355void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002356 // Render incoming data if we're the active call, and we have the local
2357 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002358 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002359 if (!media_channel()->SetReceive(recv)) {
2360 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2361 }
2362
2363 // Send outgoing data if we're the active call, we have the remote content,
2364 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002365 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002366 if (!media_channel()->SetSend(send)) {
2367 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2368 }
2369
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002370 // Trigger SignalReadyToSendData asynchronously.
2371 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002372
2373 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2374}
2375
deadbeef953c2ce2017-01-09 14:53:41 -08002376void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002377 switch (pmsg->message_id) {
2378 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002379 DataChannelReadyToSendMessageData* data =
2380 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002381 ready_to_send_data_ = data->data();
2382 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002383 delete data;
2384 break;
2385 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002386 case MSG_DATARECEIVED: {
2387 DataReceivedMessageData* data =
2388 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002389 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002390 delete data;
2391 break;
2392 }
2393 case MSG_CHANNEL_ERROR: {
2394 const DataChannelErrorMessageData* data =
2395 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002396 delete data;
2397 break;
2398 }
2399 default:
2400 BaseChannel::OnMessage(pmsg);
2401 break;
2402 }
2403}
2404
deadbeef953c2ce2017-01-09 14:53:41 -08002405void RtpDataChannel::OnConnectionMonitorUpdate(
2406 ConnectionMonitor* monitor,
2407 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002408 SignalConnectionMonitor(this, infos);
2409}
2410
deadbeef953c2ce2017-01-09 14:53:41 -08002411void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002412 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002413 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002414 media_monitor_->SignalUpdate.connect(this,
2415 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002416 media_monitor_->Start(cms);
2417}
2418
deadbeef953c2ce2017-01-09 14:53:41 -08002419void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002420 if (media_monitor_) {
2421 media_monitor_->Stop();
2422 media_monitor_->SignalUpdate.disconnect(this);
2423 media_monitor_.reset();
2424 }
2425}
2426
deadbeef953c2ce2017-01-09 14:53:41 -08002427void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2428 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002429 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002430 SignalMediaMonitor(this, info);
2431}
2432
deadbeef953c2ce2017-01-09 14:53:41 -08002433void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2434 const char* data,
2435 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002436 DataReceivedMessageData* msg = new DataReceivedMessageData(
2437 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002438 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002439}
2440
deadbeef953c2ce2017-01-09 14:53:41 -08002441void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2442 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002443 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2444 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002445 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002446}
2447
deadbeef953c2ce2017-01-09 14:53:41 -08002448void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002449 // This is usded for congestion control to indicate that the stream is ready
2450 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2451 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002452 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002453 new DataChannelReadyToSendMessageData(writable));
2454}
2455
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002456} // namespace cricket