blob: 872ee7fcb600acc727bc036666d152cddc585702 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellander65c7f672016-02-12 00:05:01 -08002 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellander65c7f672016-02-12 00:05:01 -08004 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
kwiberg0eb15ed2015-12-17 03:04:15 -080011#include <utility>
12
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010013#include "webrtc/pc/channel.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000014
kjellandera69d9732016-08-31 07:33:05 -070015#include "webrtc/api/call/audio_sink.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000016#include "webrtc/base/bind.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000017#include "webrtc/base/byteorder.h"
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -070018#include "webrtc/base/checks.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000019#include "webrtc/base/common.h"
jbaucheec21bd2016-03-20 06:15:43 -070020#include "webrtc/base/copyonwritebuffer.h"
buildbot@webrtc.org65b98d12014-08-07 22:09:08 +000021#include "webrtc/base/dscp.h"
22#include "webrtc/base/logging.h"
Honghai Zhangcc411c02016-03-29 17:27:21 -070023#include "webrtc/base/networkroute.h"
Peter Boström6f28cf02015-12-07 23:17:15 +010024#include "webrtc/base/trace_event.h"
kjellanderf4752772016-03-02 05:42:30 -080025#include "webrtc/media/base/mediaconstants.h"
kjellandera96e2d72016-02-04 23:52:28 -080026#include "webrtc/media/base/rtputils.h"
johand89ab142016-10-25 10:50:32 -070027#include "webrtc/p2p/base/packettransportinterface.h"
Peter Boström6f28cf02015-12-07 23:17:15 +010028#include "webrtc/p2p/base/transportchannel.h"
kjellander@webrtc.org9b8df252016-02-12 06:47:59 +010029#include "webrtc/pc/channelmanager.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000030
31namespace cricket {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000032using rtc::Bind;
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000033
deadbeef2d110be2016-01-13 12:00:26 -080034namespace {
kwiberg31022942016-03-11 14:18:21 -080035// See comment below for why we need to use a pointer to a unique_ptr.
deadbeef2d110be2016-01-13 12:00:26 -080036bool SetRawAudioSink_w(VoiceMediaChannel* channel,
37 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -080038 std::unique_ptr<webrtc::AudioSinkInterface>* sink) {
39 channel->SetRawAudioSink(ssrc, std::move(*sink));
deadbeef2d110be2016-01-13 12:00:26 -080040 return true;
41}
Danil Chapovalov33b01f22016-05-11 19:55:27 +020042
43struct SendPacketMessageData : public rtc::MessageData {
44 rtc::CopyOnWriteBuffer packet;
45 rtc::PacketOptions options;
46};
47
isheriff6f8d6862016-05-26 11:24:55 -070048#if defined(ENABLE_EXTERNAL_AUTH)
49// Returns the named header extension if found among all extensions,
50// nullptr otherwise.
51const webrtc::RtpExtension* FindHeaderExtension(
52 const std::vector<webrtc::RtpExtension>& extensions,
53 const std::string& uri) {
54 for (const auto& extension : extensions) {
55 if (extension.uri == uri)
56 return &extension;
57 }
58 return nullptr;
59}
60#endif
61
deadbeef2d110be2016-01-13 12:00:26 -080062} // namespace
63
henrike@webrtc.org28e20752013-07-10 00:45:36 +000064enum {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +000065 MSG_EARLYMEDIATIMEOUT = 1,
Danil Chapovalov33b01f22016-05-11 19:55:27 +020066 MSG_SEND_RTP_PACKET,
67 MSG_SEND_RTCP_PACKET,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000068 MSG_CHANNEL_ERROR,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000069 MSG_READYTOSENDDATA,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000070 MSG_DATARECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000071 MSG_FIRSTPACKETRECEIVED,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000072};
73
74// Value specified in RFC 5764.
75static const char kDtlsSrtpExporterLabel[] = "EXTRACTOR-dtls_srtp";
76
77static const int kAgcMinus10db = -10;
78
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000079static void SafeSetError(const std::string& message, std::string* error_desc) {
80 if (error_desc) {
81 *error_desc = message;
82 }
83}
84
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000085struct VoiceChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020086 VoiceChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000087 VoiceMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020088 : ssrc(in_ssrc), error(in_error) {}
89 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000090 VoiceMediaChannel::Error error;
91};
92
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000093struct VideoChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +020094 VideoChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000095 VideoMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +020096 : ssrc(in_ssrc), error(in_error) {}
97 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000098 VideoMediaChannel::Error error;
99};
100
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000101struct DataChannelErrorMessageData : public rtc::MessageData {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200102 DataChannelErrorMessageData(uint32_t in_ssrc,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000103 DataMediaChannel::Error in_error)
Peter Boström0c4e06b2015-10-07 12:23:21 +0200104 : ssrc(in_ssrc), error(in_error) {}
105 uint32_t ssrc;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000106 DataMediaChannel::Error error;
107};
108
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000109static const char* PacketType(bool rtcp) {
110 return (!rtcp) ? "RTP" : "RTCP";
111}
112
jbaucheec21bd2016-03-20 06:15:43 -0700113static bool ValidPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000114 // Check the packet size. We could check the header too if needed.
115 return (packet &&
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000116 packet->size() >= (!rtcp ? kMinRtpPacketLen : kMinRtcpPacketLen) &&
117 packet->size() <= kMaxRtpPacketLen);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000118}
119
120static bool IsReceiveContentDirection(MediaContentDirection direction) {
121 return direction == MD_SENDRECV || direction == MD_RECVONLY;
122}
123
124static bool IsSendContentDirection(MediaContentDirection direction) {
125 return direction == MD_SENDRECV || direction == MD_SENDONLY;
126}
127
128static const MediaContentDescription* GetContentDescription(
129 const ContentInfo* cinfo) {
130 if (cinfo == NULL)
131 return NULL;
132 return static_cast<const MediaContentDescription*>(cinfo->description);
133}
134
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700135template <class Codec>
136void RtpParametersFromMediaDescription(
137 const MediaContentDescriptionImpl<Codec>* desc,
138 RtpParameters<Codec>* params) {
139 // TODO(pthatcher): Remove this once we're sure no one will give us
140 // a description without codecs (currently a CA_UPDATE with just
141 // streams can).
142 if (desc->has_codecs()) {
143 params->codecs = desc->codecs();
144 }
145 // TODO(pthatcher): See if we really need
146 // rtp_header_extensions_set() and remove it if we don't.
147 if (desc->rtp_header_extensions_set()) {
148 params->extensions = desc->rtp_header_extensions();
149 }
deadbeef13871492015-12-09 12:37:51 -0800150 params->rtcp.reduced_size = desc->rtcp_reduced_size();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700151}
152
nisse05103312016-03-16 02:22:50 -0700153template <class Codec>
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700154void RtpSendParametersFromMediaDescription(
155 const MediaContentDescriptionImpl<Codec>* desc,
nisse05103312016-03-16 02:22:50 -0700156 RtpSendParameters<Codec>* send_params) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -0700157 RtpParametersFromMediaDescription(desc, send_params);
158 send_params->max_bandwidth_bps = desc->bandwidth();
159}
160
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200161BaseChannel::BaseChannel(rtc::Thread* worker_thread,
162 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -0800163 rtc::Thread* signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -0700164 MediaChannel* media_channel,
deadbeefcbecd352015-09-23 11:50:27 -0700165 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -0800166 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -0800167 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200168 : worker_thread_(worker_thread),
169 network_thread_(network_thread),
zhihuangf5b251b2017-01-12 19:37:48 -0800170 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000171 content_name_(content_name),
deadbeefac22f702017-01-12 21:59:29 -0800172 rtcp_mux_required_(rtcp_mux_required),
deadbeef7af91dd2016-12-13 11:29:11 -0800173 srtp_required_(srtp_required),
michaelt79e05882016-11-08 02:50:09 -0800174 media_channel_(media_channel),
175 selected_candidate_pair_(nullptr) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700176 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000177 LOG(LS_INFO) << "Created channel for " << content_name;
178}
179
180BaseChannel::~BaseChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -0800181 TRACE_EVENT0("webrtc", "BaseChannel::~BaseChannel");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700182 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
wu@webrtc.org78187522013-10-07 23:32:02 +0000183 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000184 StopConnectionMonitor();
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200185 // Eats any outstanding messages or packets.
186 worker_thread_->Clear(&invoker_);
187 worker_thread_->Clear(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000188 // We must destroy the media channel before the transport channel, otherwise
189 // the media channel may try to send on the dead transport channel. NULLing
190 // is not an effective strategy since the sends will come on another thread.
191 delete media_channel_;
zhihuangf5b251b2017-01-12 19:37:48 -0800192 LOG(LS_INFO) << "Destroyed channel: " << content_name_;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200193}
194
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200195void BaseChannel::DisconnectTransportChannels_n() {
196 // Send any outstanding RTCP packets.
197 FlushRtcpMessages_n();
198
199 // Stop signals from transport channels, but keep them alive because
200 // media_channel may use them from a different thread.
zhihuangf5b251b2017-01-12 19:37:48 -0800201 if (rtp_transport_) {
202 DisconnectFromTransportChannel(rtp_transport_);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200203 }
zhihuangf5b251b2017-01-12 19:37:48 -0800204 if (rtcp_transport_) {
205 DisconnectFromTransportChannel(rtcp_transport_);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200206 }
207
208 // Clear pending read packets/messages.
209 network_thread_->Clear(&invoker_);
210 network_thread_->Clear(this);
211}
212
zhihuangf5b251b2017-01-12 19:37:48 -0800213bool BaseChannel::Init_w(TransportChannel* rtp_transport,
214 TransportChannel* rtcp_transport) {
skvlad6c87a672016-05-17 17:49:52 -0700215 if (!network_thread_->Invoke<bool>(
zhihuangf5b251b2017-01-12 19:37:48 -0800216 RTC_FROM_HERE, Bind(&BaseChannel::InitNetwork_n, this, rtp_transport,
217 rtcp_transport))) {
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000218 return false;
219 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000220
wu@webrtc.orgde305012013-10-31 15:40:38 +0000221 // Both RTP and RTCP channels are set, we can call SetInterface on
222 // media channel and it can set network options.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200223 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.orgde305012013-10-31 15:40:38 +0000224 media_channel_->SetInterface(this);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000225 return true;
226}
227
zhihuangf5b251b2017-01-12 19:37:48 -0800228bool BaseChannel::InitNetwork_n(TransportChannel* rtp_transport,
229 TransportChannel* rtcp_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200230 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefbad5dad2017-01-17 18:32:35 -0800231 SetTransports_n(rtp_transport, rtcp_transport);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200232
zhihuangf5b251b2017-01-12 19:37:48 -0800233 if (!SetDtlsSrtpCryptoSuites_n(rtp_transport_, false)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200234 return false;
235 }
zhihuangf5b251b2017-01-12 19:37:48 -0800236 if (rtcp_transport_ && !SetDtlsSrtpCryptoSuites_n(rtcp_transport_, true)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200237 return false;
238 }
deadbeefac22f702017-01-12 21:59:29 -0800239 if (rtcp_mux_required_) {
240 rtcp_mux_filter_.SetActive();
241 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200242 return true;
243}
244
wu@webrtc.org78187522013-10-07 23:32:02 +0000245void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200246 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000247 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200248 // Packets arrive on the network thread, processing packets calls virtual
249 // functions, so need to stop this process in Deinit that is called in
250 // derived classes destructor.
251 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700252 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000253}
254
deadbeefbad5dad2017-01-17 18:32:35 -0800255void BaseChannel::SetTransports(TransportChannel* rtp_transport,
256 TransportChannel* rtcp_transport) {
257 network_thread_->Invoke<void>(
zhihuangf5b251b2017-01-12 19:37:48 -0800258 RTC_FROM_HERE,
deadbeefbad5dad2017-01-17 18:32:35 -0800259 Bind(&BaseChannel::SetTransports_n, this, rtp_transport, rtcp_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000260}
261
deadbeefbad5dad2017-01-17 18:32:35 -0800262void BaseChannel::SetTransports_n(TransportChannel* rtp_transport,
263 TransportChannel* rtcp_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200264 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefbad5dad2017-01-17 18:32:35 -0800265 // Verify some assumptions (as described in the comment above SetTransport).
266 RTC_DCHECK(rtp_transport);
267 RTC_DCHECK(NeedsRtcpTransport() == (rtcp_transport != nullptr));
zhihuangf5b251b2017-01-12 19:37:48 -0800268 if (rtcp_transport) {
269 RTC_DCHECK(rtp_transport->transport_name() ==
270 rtcp_transport->transport_name());
zhihuangf5b251b2017-01-12 19:37:48 -0800271 }
272
273 if (rtp_transport->transport_name() == transport_name_) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700274 // Nothing to do if transport name isn't changing.
deadbeefbad5dad2017-01-17 18:32:35 -0800275 return;
deadbeefcbecd352015-09-23 11:50:27 -0700276 }
277
zhihuangf5b251b2017-01-12 19:37:48 -0800278 transport_name_ = rtp_transport->transport_name();
279
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800280 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
281 // changes and wait until the DTLS handshake is complete to set the newly
282 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200283 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800284 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700285 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800286 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800287 srtp_filter_.ResetParams();
288 }
289
deadbeefac22f702017-01-12 21:59:29 -0800290 // If this BaseChannel doesn't require RTCP mux and we haven't fully
291 // negotiated RTCP mux, we need an RTCP transport.
zhihuangf5b251b2017-01-12 19:37:48 -0800292 if (NeedsRtcpTransport()) {
293 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
294 << transport_name() << " transport " << rtcp_transport;
295 SetTransportChannel_n(true, rtcp_transport);
deadbeefbad5dad2017-01-17 18:32:35 -0800296 RTC_DCHECK(rtcp_transport_);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000297 }
298
zhihuangf5b251b2017-01-12 19:37:48 -0800299 LOG(LS_INFO) << "Setting non-RTCP Transport for " << content_name() << " on "
300 << transport_name() << " transport " << rtp_transport;
301 SetTransportChannel_n(false, rtp_transport);
deadbeefbad5dad2017-01-17 18:32:35 -0800302 RTC_DCHECK(rtp_transport_);
guoweis46383312015-12-17 16:45:59 -0800303
deadbeefcbecd352015-09-23 11:50:27 -0700304 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700305 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200306 UpdateWritableState_n();
deadbeef062ce9f2016-08-26 21:42:15 -0700307 // We can only update ready-to-send after updating writability.
308 //
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700309 // On setting a new channel, assume it's ready to send if it's writable,
310 // because we have no way of knowing otherwise (the channel doesn't give us
311 // "was last send successful?").
312 //
313 // This won't always be accurate (the last SendPacket call from another
314 // BaseChannel could have resulted in an error), but even so, we'll just
315 // encounter the error again and update "ready to send" accordingly.
zhihuangf5b251b2017-01-12 19:37:48 -0800316 SetTransportChannelReadyToSend(false,
317 rtp_transport_ && rtp_transport_->writable());
deadbeef062ce9f2016-08-26 21:42:15 -0700318 SetTransportChannelReadyToSend(
zhihuangf5b251b2017-01-12 19:37:48 -0800319 true, rtcp_transport_ && rtcp_transport_->writable());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000320}
321
deadbeef062ce9f2016-08-26 21:42:15 -0700322void BaseChannel::SetTransportChannel_n(bool rtcp,
zhihuangf5b251b2017-01-12 19:37:48 -0800323 TransportChannel* new_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200324 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangf5b251b2017-01-12 19:37:48 -0800325 TransportChannel*& old_transport = rtcp ? rtcp_transport_ : rtp_transport_;
326 if (!old_transport && !new_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700327 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000328 return;
329 }
zhihuangf5b251b2017-01-12 19:37:48 -0800330 RTC_DCHECK(old_transport != new_transport);
331 if (old_transport) {
332 DisconnectFromTransportChannel(old_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000333 }
334
zhihuangf5b251b2017-01-12 19:37:48 -0800335 old_transport = new_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000336
zhihuangf5b251b2017-01-12 19:37:48 -0800337 if (new_transport) {
deadbeef062ce9f2016-08-26 21:42:15 -0700338 if (rtcp) {
339 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
340 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
341 << "should never happen.";
deadbeefcbecd352015-09-23 11:50:27 -0700342 }
zhihuangf5b251b2017-01-12 19:37:48 -0800343 ConnectToTransportChannel(new_transport);
deadbeef062ce9f2016-08-26 21:42:15 -0700344 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
345 for (const auto& pair : socket_options) {
zhihuangf5b251b2017-01-12 19:37:48 -0800346 new_transport->SetOption(pair.first, pair.second);
deadbeef062ce9f2016-08-26 21:42:15 -0700347 }
guoweis46383312015-12-17 16:45:59 -0800348 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000349}
350
351void BaseChannel::ConnectToTransportChannel(TransportChannel* tc) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200352 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000353
354 tc->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
johand89ab142016-10-25 10:50:32 -0700355 tc->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000356 tc->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800357 tc->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
Honghai Zhangcc411c02016-03-29 17:27:21 -0700358 tc->SignalSelectedCandidatePairChanged.connect(
359 this, &BaseChannel::OnSelectedCandidatePairChanged);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200360 tc->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000361}
362
363void BaseChannel::DisconnectFromTransportChannel(TransportChannel* tc) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200364 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800365 OnSelectedCandidatePairChanged(tc, nullptr, -1, false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000366
367 tc->SignalWritableState.disconnect(this);
368 tc->SignalReadPacket.disconnect(this);
369 tc->SignalReadyToSend.disconnect(this);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800370 tc->SignalDtlsState.disconnect(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200371 tc->SignalSelectedCandidatePairChanged.disconnect(this);
372 tc->SignalSentPacket.disconnect(this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000373}
374
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000375bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700376 worker_thread_->Invoke<void>(
377 RTC_FROM_HERE,
378 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
379 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000380 return true;
381}
382
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000383bool BaseChannel::AddRecvStream(const StreamParams& sp) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700384 return InvokeOnWorker(RTC_FROM_HERE,
385 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000386}
387
Peter Boström0c4e06b2015-10-07 12:23:21 +0200388bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700389 return InvokeOnWorker(RTC_FROM_HERE,
390 Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000391}
392
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000393bool BaseChannel::AddSendStream(const StreamParams& sp) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +0000394 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700395 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000396}
397
Peter Boström0c4e06b2015-10-07 12:23:21 +0200398bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700399 return InvokeOnWorker(RTC_FROM_HERE, Bind(&MediaChannel::RemoveSendStream,
400 media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000401}
402
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000403bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000404 ContentAction action,
405 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100406 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700407 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetLocalContent_w,
408 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000409}
410
411bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000412 ContentAction action,
413 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100414 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700415 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w,
416 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000417}
418
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000419void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangf5b251b2017-01-12 19:37:48 -0800420 // We pass in the BaseChannel instead of the rtp_transport_
421 // because if the rtp_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000422 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200423 // We pass in the network thread because on that thread connection monitor
424 // will call BaseChannel::GetConnectionStats which must be called on the
425 // network thread.
426 connection_monitor_.reset(
427 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000428 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000429 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000430 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000431}
432
433void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000434 if (connection_monitor_) {
435 connection_monitor_->Stop();
436 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000437 }
438}
439
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000440bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200441 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangf5b251b2017-01-12 19:37:48 -0800442 return rtp_transport_->GetStats(infos);
443}
444
445bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800446 // If this BaseChannel doesn't require RTCP mux and we haven't fully
447 // negotiated RTCP mux, we need an RTCP transport.
448 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000449}
450
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700451bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000452 // Receive data if we are enabled and have local content,
453 return enabled() && IsReceiveContentDirection(local_content_direction_);
454}
455
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700456bool BaseChannel::IsReadyToSendMedia_w() const {
457 // Need to access some state updated on the network thread.
458 return network_thread_->Invoke<bool>(
459 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
460}
461
462bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000463 // Send outgoing data if we are enabled, have local and remote content,
464 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800465 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000466 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700467 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200468 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000469}
470
jbaucheec21bd2016-03-20 06:15:43 -0700471bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700472 const rtc::PacketOptions& options) {
473 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000474}
475
jbaucheec21bd2016-03-20 06:15:43 -0700476bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700477 const rtc::PacketOptions& options) {
478 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000479}
480
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000481int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000482 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200483 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700484 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200485}
486
487int BaseChannel::SetOption_n(SocketType type,
488 rtc::Socket::Option opt,
489 int value) {
490 RTC_DCHECK(network_thread_->IsCurrent());
491 TransportChannel* channel = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000492 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000493 case ST_RTP:
zhihuangf5b251b2017-01-12 19:37:48 -0800494 channel = rtp_transport_;
deadbeefcbecd352015-09-23 11:50:27 -0700495 socket_options_.push_back(
496 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000497 break;
498 case ST_RTCP:
zhihuangf5b251b2017-01-12 19:37:48 -0800499 channel = rtcp_transport_;
deadbeefcbecd352015-09-23 11:50:27 -0700500 rtcp_socket_options_.push_back(
501 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000502 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000503 }
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000504 return channel ? channel->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000505}
506
jbauchcb560652016-08-04 05:20:32 -0700507bool BaseChannel::SetCryptoOptions(const rtc::CryptoOptions& crypto_options) {
508 crypto_options_ = crypto_options;
509 return true;
510}
511
johand89ab142016-10-25 10:50:32 -0700512void BaseChannel::OnWritableState(rtc::PacketTransportInterface* transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800513 RTC_DCHECK(transport == rtp_transport_ || transport == rtcp_transport_);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200514 RTC_DCHECK(network_thread_->IsCurrent());
515 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000516}
517
johand89ab142016-10-25 10:50:32 -0700518void BaseChannel::OnPacketRead(rtc::PacketTransportInterface* transport,
519 const char* data,
520 size_t len,
521 const rtc::PacketTime& packet_time,
522 int flags) {
523 TRACE_EVENT0("webrtc", "BaseChannel::OnPacketRead");
524 // OnPacketRead gets called from P2PSocket; now pass data to MediaEngine
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200525 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000526
527 // When using RTCP multiplexing we might get RTCP packets on the RTP
528 // transport. We feed RTP traffic into the demuxer to determine if it is RTCP.
johand89ab142016-10-25 10:50:32 -0700529 bool rtcp = PacketIsRtcp(transport, data, len);
jbaucheec21bd2016-03-20 06:15:43 -0700530 rtc::CopyOnWriteBuffer packet(data, len);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000531 HandlePacket(rtcp, &packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000532}
533
johand89ab142016-10-25 10:50:32 -0700534void BaseChannel::OnReadyToSend(rtc::PacketTransportInterface* transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800535 RTC_DCHECK(transport == rtp_transport_ || transport == rtcp_transport_);
536 SetTransportChannelReadyToSend(transport == rtcp_transport_, true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000537}
538
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800539void BaseChannel::OnDtlsState(TransportChannel* channel,
540 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200541 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800542 return;
543 }
544
545 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
546 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
547 // cover other scenarios like the whole channel is writable (not just this
548 // TransportChannel) or when TransportChannel is attached after DTLS is
549 // negotiated.
550 if (state != DTLS_TRANSPORT_CONNECTED) {
551 srtp_filter_.ResetParams();
552 }
553}
554
Honghai Zhangcc411c02016-03-29 17:27:21 -0700555void BaseChannel::OnSelectedCandidatePairChanged(
556 TransportChannel* channel,
Honghai Zhang52dce732016-03-31 12:37:31 -0700557 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700558 int last_sent_packet_id,
559 bool ready_to_send) {
zhihuangf5b251b2017-01-12 19:37:48 -0800560 RTC_DCHECK(channel == rtp_transport_ || channel == rtcp_transport_);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200561 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800562 selected_candidate_pair_ = selected_candidate_pair;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200563 std::string transport_name = channel->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700564 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700565 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700566 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700567 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700568 selected_candidate_pair->remote_candidate().network_id(),
569 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800570
571 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700572 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200573 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700574 RTC_FROM_HERE, worker_thread_,
575 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
576 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700577}
578
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700579void BaseChannel::SetTransportChannelReadyToSend(bool rtcp, bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200580 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700581 if (rtcp) {
582 rtcp_ready_to_send_ = ready;
583 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000584 rtp_ready_to_send_ = ready;
585 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000586
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200587 bool ready_to_send =
588 (rtp_ready_to_send_ &&
zhihuangf5b251b2017-01-12 19:37:48 -0800589 // In the case of rtcp mux |rtcp_transport_| will be null.
590 (rtcp_ready_to_send_ || !rtcp_transport_));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200591
592 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700593 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200594 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready_to_send));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000595}
596
johand89ab142016-10-25 10:50:32 -0700597bool BaseChannel::PacketIsRtcp(const rtc::PacketTransportInterface* transport,
598 const char* data,
599 size_t len) {
zhihuangf5b251b2017-01-12 19:37:48 -0800600 return (transport == rtcp_transport_ ||
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000601 rtcp_mux_filter_.DemuxRtcp(data, static_cast<int>(len)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000602}
603
stefanc1aeaf02015-10-15 07:26:07 -0700604bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700605 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700606 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200607 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
608 // If the thread is not our network thread, we will post to our network
609 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000610 // synchronize access to all the pieces of the send path, including
611 // SRTP and the inner workings of the transport channels.
612 // The only downside is that we can't return a proper failure code if
613 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200614 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000615 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200616 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
617 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800618 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700619 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700620 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000621 return true;
622 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200623 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000624
625 // Now that we are on the correct thread, ensure we have a place to send this
626 // packet before doing anything. (We might get RTCP packets that we don't
627 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
628 // transport.
zhihuangf5b251b2017-01-12 19:37:48 -0800629 TransportChannel* channel =
630 (!rtcp || rtcp_mux_filter_.IsActive()) ? rtp_transport_ : rtcp_transport_;
wu@webrtc.org97077a32013-10-25 21:18:33 +0000631 if (!channel || !channel->writable()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000632 return false;
633 }
634
635 // Protect ourselves against crazy data.
636 if (!ValidPacket(rtcp, packet)) {
637 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000638 << PacketType(rtcp)
639 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000640 return false;
641 }
642
stefanc1aeaf02015-10-15 07:26:07 -0700643 rtc::PacketOptions updated_options;
644 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000645 // Protect if needed.
646 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200647 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000648 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200649 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000650 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000651 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000652 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
653 // inside libsrtp for a RTP packet. A external HMAC module will be writing
654 // a fake HMAC value. This is ONLY done for a RTP packet.
655 // Socket layer will update rtp sendtime extension header if present in
656 // packet with current time before updating the HMAC.
657#if !defined(ENABLE_EXTERNAL_AUTH)
658 res = srtp_filter_.ProtectRtp(
659 data, len, static_cast<int>(packet->capacity()), &len);
660#else
stefanc1aeaf02015-10-15 07:26:07 -0700661 updated_options.packet_time_params.rtp_sendtime_extension_id =
henrike@webrtc.org05376342014-03-10 15:53:12 +0000662 rtp_abs_sendtime_extn_id_;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000663 res = srtp_filter_.ProtectRtp(
664 data, len, static_cast<int>(packet->capacity()), &len,
stefanc1aeaf02015-10-15 07:26:07 -0700665 &updated_options.packet_time_params.srtp_packet_index);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000666 // If protection succeeds, let's get auth params from srtp.
667 if (res) {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200668 uint8_t* auth_key = NULL;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000669 int key_len;
670 res = srtp_filter_.GetRtpAuthParams(
stefanc1aeaf02015-10-15 07:26:07 -0700671 &auth_key, &key_len,
672 &updated_options.packet_time_params.srtp_auth_tag_len);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000673 if (res) {
stefanc1aeaf02015-10-15 07:26:07 -0700674 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
675 updated_options.packet_time_params.srtp_auth_key.assign(
676 auth_key, auth_key + key_len);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000677 }
678 }
679#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000680 if (!res) {
681 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200682 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000683 GetRtpSeqNum(data, len, &seq_num);
684 GetRtpSsrc(data, len, &ssrc);
685 LOG(LS_ERROR) << "Failed to protect " << content_name_
686 << " RTP packet: size=" << len
687 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
688 return false;
689 }
690 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000691 res = srtp_filter_.ProtectRtcp(data, len,
692 static_cast<int>(packet->capacity()),
693 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000694 if (!res) {
695 int type = -1;
696 GetRtcpType(data, len, &type);
697 LOG(LS_ERROR) << "Failed to protect " << content_name_
698 << " RTCP packet: size=" << len << ", type=" << type;
699 return false;
700 }
701 }
702
703 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000704 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800705 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800706 // The audio/video engines may attempt to send RTCP packets as soon as the
707 // streams are created, so don't treat this as an error for RTCP.
708 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
709 if (rtcp) {
710 return false;
711 }
712 // However, there shouldn't be any RTP packets sent before SRTP is set up
713 // (and SetSend(true) is called).
714 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
715 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800716 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000717 return false;
718 }
719
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000720 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200721 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
722 int ret = channel->SendPacket(packet->data<char>(), packet->size(),
723 updated_options, flags);
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000724 if (ret != static_cast<int>(packet->size())) {
skvladc309e0e2016-07-28 17:15:20 -0700725 if (channel->GetError() == ENOTCONN) {
726 LOG(LS_WARNING) << "Got ENOTCONN from transport.";
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700727 SetTransportChannelReadyToSend(rtcp, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000728 }
729 return false;
730 }
731 return true;
732}
733
jbaucheec21bd2016-03-20 06:15:43 -0700734bool BaseChannel::WantsPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000735 // Protect ourselves against crazy data.
736 if (!ValidPacket(rtcp, packet)) {
737 LOG(LS_ERROR) << "Dropping incoming " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000738 << PacketType(rtcp)
739 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000740 return false;
741 }
pbos482b12e2015-11-16 10:19:58 -0800742 if (rtcp) {
743 // Permit all (seemingly valid) RTCP packets.
744 return true;
745 }
746 // Check whether we handle this payload.
jbaucheec21bd2016-03-20 06:15:43 -0700747 return bundle_filter_.DemuxPacket(packet->data(), packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000748}
749
jbaucheec21bd2016-03-20 06:15:43 -0700750void BaseChannel::HandlePacket(bool rtcp, rtc::CopyOnWriteBuffer* packet,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000751 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200752 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000753 if (!WantsPacket(rtcp, packet)) {
754 return;
755 }
756
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000757 // We are only interested in the first rtp packet because that
758 // indicates the media has started flowing.
759 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000760 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700761 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000762 }
763
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000764 // Unprotect the packet, if needed.
765 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200766 TRACE_EVENT0("webrtc", "SRTP Decode");
Karl Wiberg94784372015-04-20 14:03:07 +0200767 char* data = packet->data<char>();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000768 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000769 bool res;
770 if (!rtcp) {
771 res = srtp_filter_.UnprotectRtp(data, len, &len);
772 if (!res) {
773 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200774 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000775 GetRtpSeqNum(data, len, &seq_num);
776 GetRtpSsrc(data, len, &ssrc);
777 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
778 << " RTP packet: size=" << len
779 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
780 return;
781 }
782 } else {
783 res = srtp_filter_.UnprotectRtcp(data, len, &len);
784 if (!res) {
785 int type = -1;
786 GetRtcpType(data, len, &type);
787 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
788 << " RTCP packet: size=" << len << ", type=" << type;
789 return;
790 }
791 }
792
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000793 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800794 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000795 // Our session description indicates that SRTP is required, but we got a
796 // packet before our SRTP filter is active. This means either that
797 // a) we got SRTP packets before we received the SDES keys, in which case
798 // we can't decrypt it anyway, or
799 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
800 // channels, so we haven't yet extracted keys, even if DTLS did complete
801 // on the channel that the packets are being sent on. It's really good
802 // practice to wait for both RTP and RTCP to be good to go before sending
803 // media, to prevent weird failure modes, so it's fine for us to just eat
804 // packets here. This is all sidestepped if RTCP mux is used anyway.
805 LOG(LS_WARNING) << "Can't process incoming " << PacketType(rtcp)
806 << " packet when SRTP is inactive and crypto is required";
807 return;
808 }
809
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200810 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700811 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200812 Bind(&BaseChannel::OnPacketReceived, this, rtcp, *packet, packet_time));
813}
814
815void BaseChannel::OnPacketReceived(bool rtcp,
816 const rtc::CopyOnWriteBuffer& packet,
817 const rtc::PacketTime& packet_time) {
818 RTC_DCHECK(worker_thread_->IsCurrent());
819 // Need to copy variable because OnRtcpReceived/OnPacketReceived
820 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
821 rtc::CopyOnWriteBuffer data(packet);
822 if (rtcp) {
823 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000824 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200825 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000826 }
827}
828
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000829bool BaseChannel::PushdownLocalDescription(
830 const SessionDescription* local_desc, ContentAction action,
831 std::string* error_desc) {
832 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000833 const MediaContentDescription* content_desc =
834 GetContentDescription(content_info);
835 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000836 !SetLocalContent(content_desc, action, error_desc)) {
837 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
838 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000839 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000840 return true;
841}
842
843bool BaseChannel::PushdownRemoteDescription(
844 const SessionDescription* remote_desc, ContentAction action,
845 std::string* error_desc) {
846 const ContentInfo* content_info = GetFirstContent(remote_desc);
847 const MediaContentDescription* content_desc =
848 GetContentDescription(content_info);
849 if (content_desc && content_info && !content_info->rejected &&
850 !SetRemoteContent(content_desc, action, error_desc)) {
851 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
852 return false;
853 }
854 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000855}
856
857void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700858 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000859 if (enabled_)
860 return;
861
862 LOG(LS_INFO) << "Channel enabled";
863 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700864 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000865}
866
867void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700868 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000869 if (!enabled_)
870 return;
871
872 LOG(LS_INFO) << "Channel disabled";
873 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700874 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000875}
876
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200877void BaseChannel::UpdateWritableState_n() {
zhihuangf5b251b2017-01-12 19:37:48 -0800878 if (rtp_transport_ && rtp_transport_->writable() &&
879 (!rtcp_transport_ || rtcp_transport_->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200880 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700881 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200882 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700883 }
884}
885
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200886void BaseChannel::ChannelWritable_n() {
887 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800888 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000889 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800890 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000891
deadbeefcbecd352015-09-23 11:50:27 -0700892 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000893 << (was_ever_writable_ ? "" : " for the first time");
894
michaelt79e05882016-11-08 02:50:09 -0800895 if (selected_candidate_pair_)
896 LOG(LS_INFO)
897 << "Using "
898 << selected_candidate_pair_->local_candidate().ToSensitiveString()
899 << "->"
900 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000901
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000902 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200903 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000904 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700905 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000906}
907
deadbeef953c2ce2017-01-09 14:53:41 -0800908void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200909 RTC_DCHECK(network_thread_->IsCurrent());
910 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700911 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800912 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000913}
914
deadbeef953c2ce2017-01-09 14:53:41 -0800915void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700916 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800917 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000918}
919
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200920bool BaseChannel::SetDtlsSrtpCryptoSuites_n(TransportChannel* tc, bool rtcp) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800921 std::vector<int> crypto_suites;
922 // We always use the default SRTP crypto suites for RTCP, but we may use
923 // different crypto suites for RTP depending on the media type.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000924 if (!rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200925 GetSrtpCryptoSuites_n(&crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000926 } else {
jbauchcb560652016-08-04 05:20:32 -0700927 GetDefaultSrtpCryptoSuites(crypto_options(), &crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000928 }
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800929 return tc->SetSrtpCryptoSuites(crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000930}
931
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200932bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800933 // Since DTLS is applied to all channels, checking RTP should be enough.
zhihuangf5b251b2017-01-12 19:37:48 -0800934 return rtp_transport_ && rtp_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000935}
936
937// This function returns true if either DTLS-SRTP is not in use
938// *or* DTLS-SRTP is successfully set up.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200939bool BaseChannel::SetupDtlsSrtp_n(bool rtcp_channel) {
940 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000941 bool ret = false;
942
zhihuangf5b251b2017-01-12 19:37:48 -0800943 TransportChannel* channel = rtcp_channel ? rtcp_transport_ : rtp_transport_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000944
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800945 RTC_DCHECK(channel->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000946
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800947 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000948
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800949 if (!channel->GetSrtpCryptoSuite(&selected_crypto_suite)) {
950 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000951 return false;
952 }
953
954 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on "
955 << content_name() << " "
956 << PacketType(rtcp_channel);
957
jbauchcb560652016-08-04 05:20:32 -0700958 int key_len;
959 int salt_len;
960 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
961 &salt_len)) {
962 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
963 return false;
964 }
965
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000966 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700967 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000968
969 // RFC 5705 exporter using the RFC 5764 parameters
970 if (!channel->ExportKeyingMaterial(
971 kDtlsSrtpExporterLabel,
972 NULL, 0, false,
973 &dtls_buffer[0], dtls_buffer.size())) {
974 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800975 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000976 return false;
977 }
978
979 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700980 std::vector<unsigned char> client_write_key(key_len + salt_len);
981 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000982 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700983 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
984 offset += key_len;
985 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
986 offset += key_len;
987 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
988 offset += salt_len;
989 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000990
991 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000992 rtc::SSLRole role;
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000993 if (!channel->GetSslRole(&role)) {
994 LOG(LS_WARNING) << "GetSslRole failed";
995 return false;
996 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000997
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000998 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000999 send_key = &server_write_key;
1000 recv_key = &client_write_key;
1001 } else {
1002 send_key = &client_write_key;
1003 recv_key = &server_write_key;
1004 }
1005
1006 if (rtcp_channel) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001007 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
1008 static_cast<int>(send_key->size()),
1009 selected_crypto_suite, &(*recv_key)[0],
1010 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001011 } else {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001012 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
1013 static_cast<int>(send_key->size()),
1014 selected_crypto_suite, &(*recv_key)[0],
1015 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001016 }
1017
michaelt79e05882016-11-08 02:50:09 -08001018 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001019 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001020 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001021 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001022 UpdateTransportOverhead();
1023 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001024 return ret;
1025}
1026
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001027void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001028 if (srtp_filter_.IsActive()) {
1029 return;
1030 }
1031
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001032 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001033 return;
1034 }
1035
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001036 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001037 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001038 return;
1039 }
1040
zhihuangf5b251b2017-01-12 19:37:48 -08001041 if (rtcp_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001042 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001043 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001044 return;
1045 }
1046 }
1047}
1048
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001049void BaseChannel::ChannelNotWritable_n() {
1050 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001051 if (!writable_)
1052 return;
1053
deadbeefcbecd352015-09-23 11:50:27 -07001054 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001055 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001056 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001057}
1058
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001059bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001060 const MediaContentDescription* content,
1061 ContentAction action,
1062 ContentSource src,
1063 std::string* error_desc) {
1064 if (action == CA_UPDATE) {
1065 // These parameters never get changed by a CA_UDPATE.
1066 return true;
1067 }
1068
deadbeef7af91dd2016-12-13 11:29:11 -08001069 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001070 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001071 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
1072 content, action, src, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001073}
1074
1075bool BaseChannel::SetRtpTransportParameters_n(
1076 const MediaContentDescription* content,
1077 ContentAction action,
1078 ContentSource src,
1079 std::string* error_desc) {
1080 RTC_DCHECK(network_thread_->IsCurrent());
1081
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001082 if (!SetSrtp_n(content->cryptos(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001083 return false;
1084 }
1085
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001086 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001087 return false;
1088 }
1089
1090 return true;
1091}
1092
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001093// |dtls| will be set to true if DTLS is active for transport channel and
1094// crypto is empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001095bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1096 bool* dtls,
1097 std::string* error_desc) {
zhihuangf5b251b2017-01-12 19:37:48 -08001098 *dtls = rtp_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001099 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001100 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001101 return false;
1102 }
1103 return true;
1104}
1105
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001106bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001107 ContentAction action,
1108 ContentSource src,
1109 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001110 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001111 if (action == CA_UPDATE) {
1112 // no crypto params.
1113 return true;
1114 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001115 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001116 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001117 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001118 if (!ret) {
1119 return false;
1120 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001121 switch (action) {
1122 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001123 // If DTLS is already active on the channel, we could be renegotiating
1124 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001125 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001126 ret = srtp_filter_.SetOffer(cryptos, src);
1127 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001128 break;
1129 case CA_PRANSWER:
1130 // If we're doing DTLS-SRTP, we don't want to update the filter
1131 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001132 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001133 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1134 }
1135 break;
1136 case CA_ANSWER:
1137 // If we're doing DTLS-SRTP, we don't want to update the filter
1138 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001139 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001140 ret = srtp_filter_.SetAnswer(cryptos, src);
1141 }
1142 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001143 default:
1144 break;
1145 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001146 if (!ret) {
1147 SafeSetError("Failed to setup SRTP filter.", error_desc);
1148 return false;
1149 }
1150 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001151}
1152
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001153bool BaseChannel::SetRtcpMux_n(bool enable,
1154 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001155 ContentSource src,
1156 std::string* error_desc) {
deadbeef8e814d72017-01-13 11:34:39 -08001157 // Provide a more specific error message for the RTCP mux "require" policy
1158 // case.
1159 if (rtcp_mux_required_ && !enable) {
1160 SafeSetError(
1161 "rtcpMuxPolicy is 'require', but media description does not "
1162 "contain 'a=rtcp-mux'.",
1163 error_desc);
1164 return false;
1165 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001166 bool ret = false;
1167 switch (action) {
1168 case CA_OFFER:
1169 ret = rtcp_mux_filter_.SetOffer(enable, src);
1170 break;
1171 case CA_PRANSWER:
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001172 // This may activate RTCP muxing, but we don't yet destroy the channel
1173 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001174 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1175 break;
1176 case CA_ANSWER:
1177 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1178 if (ret && rtcp_mux_filter_.IsActive()) {
1179 // We activated RTCP mux, close down the RTCP transport.
deadbeefcbecd352015-09-23 11:50:27 -07001180 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1181 << " by destroying RTCP transport channel for "
1182 << transport_name();
deadbeefac22f702017-01-12 21:59:29 -08001183 if (rtcp_transport()) {
1184 SetTransportChannel_n(true, nullptr);
1185 SignalRtcpMuxFullyActive(rtp_transport()->transport_name());
zhihuangf5b251b2017-01-12 19:37:48 -08001186 }
deadbeef062ce9f2016-08-26 21:42:15 -07001187 UpdateWritableState_n();
1188 SetTransportChannelReadyToSend(true, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001189 }
1190 break;
1191 case CA_UPDATE:
1192 // No RTCP mux info.
1193 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001194 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001195 default:
1196 break;
1197 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001198 if (!ret) {
1199 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1200 return false;
1201 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001202 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
1203 // CA_ANSWER, but we only want to tear down the RTCP transport channel if we
1204 // received a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001205 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001206 // If the RTP transport is already writable, then so are we.
zhihuangf5b251b2017-01-12 19:37:48 -08001207 if (rtp_transport_->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001208 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001209 }
1210 }
1211
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001212 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001213}
1214
1215bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001216 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001217 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001218}
1219
Peter Boström0c4e06b2015-10-07 12:23:21 +02001220bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001221 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001222 return media_channel()->RemoveRecvStream(ssrc);
1223}
1224
1225bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001226 ContentAction action,
1227 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001228 if (!VERIFY(action == CA_OFFER || action == CA_ANSWER ||
1229 action == CA_PRANSWER || action == CA_UPDATE))
1230 return false;
1231
1232 // If this is an update, streams only contain streams that have changed.
1233 if (action == CA_UPDATE) {
1234 for (StreamParamsVec::const_iterator it = streams.begin();
1235 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001236 const StreamParams* existing_stream =
1237 GetStreamByIds(local_streams_, it->groupid, it->id);
1238 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001239 if (media_channel()->AddSendStream(*it)) {
1240 local_streams_.push_back(*it);
1241 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1242 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001243 std::ostringstream desc;
1244 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1245 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001246 return false;
1247 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001248 } else if (existing_stream && !it->has_ssrcs()) {
1249 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001250 std::ostringstream desc;
1251 desc << "Failed to remove send stream with ssrc "
1252 << it->first_ssrc() << ".";
1253 SafeSetError(desc.str(), error_desc);
1254 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001255 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001256 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001257 } else {
1258 LOG(LS_WARNING) << "Ignore unsupported stream update";
1259 }
1260 }
1261 return true;
1262 }
1263 // Else streams are all the streams we want to send.
1264
1265 // Check for streams that have been removed.
1266 bool ret = true;
1267 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1268 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001269 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001270 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001271 std::ostringstream desc;
1272 desc << "Failed to remove send stream with ssrc "
1273 << it->first_ssrc() << ".";
1274 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001275 ret = false;
1276 }
1277 }
1278 }
1279 // Check for new streams.
1280 for (StreamParamsVec::const_iterator it = streams.begin();
1281 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001282 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001283 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001284 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001285 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001286 std::ostringstream desc;
1287 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1288 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001289 ret = false;
1290 }
1291 }
1292 }
1293 local_streams_ = streams;
1294 return ret;
1295}
1296
1297bool BaseChannel::UpdateRemoteStreams_w(
1298 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001299 ContentAction action,
1300 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001301 if (!VERIFY(action == CA_OFFER || action == CA_ANSWER ||
1302 action == CA_PRANSWER || action == CA_UPDATE))
1303 return false;
1304
1305 // If this is an update, streams only contain streams that have changed.
1306 if (action == CA_UPDATE) {
1307 for (StreamParamsVec::const_iterator it = streams.begin();
1308 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001309 const StreamParams* existing_stream =
1310 GetStreamByIds(remote_streams_, it->groupid, it->id);
1311 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001312 if (AddRecvStream_w(*it)) {
1313 remote_streams_.push_back(*it);
1314 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1315 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001316 std::ostringstream desc;
1317 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1318 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001319 return false;
1320 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001321 } else if (existing_stream && !it->has_ssrcs()) {
1322 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001323 std::ostringstream desc;
1324 desc << "Failed to remove remote stream with ssrc "
1325 << it->first_ssrc() << ".";
1326 SafeSetError(desc.str(), error_desc);
1327 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001328 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001329 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001330 } else {
1331 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001332 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001333 << " new stream = " << it->ToString();
1334 }
1335 }
1336 return true;
1337 }
1338 // Else streams are all the streams we want to receive.
1339
1340 // Check for streams that have been removed.
1341 bool ret = true;
1342 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1343 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001344 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001345 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001346 std::ostringstream desc;
1347 desc << "Failed to remove remote stream with ssrc "
1348 << it->first_ssrc() << ".";
1349 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001350 ret = false;
1351 }
1352 }
1353 }
1354 // Check for new streams.
1355 for (StreamParamsVec::const_iterator it = streams.begin();
1356 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001357 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001358 if (AddRecvStream_w(*it)) {
1359 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1360 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001361 std::ostringstream desc;
1362 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1363 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001364 ret = false;
1365 }
1366 }
1367 }
1368 remote_streams_ = streams;
1369 return ret;
1370}
1371
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001372void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001373 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001374// Absolute Send Time extension id is used only with external auth,
1375// so do not bother searching for it and making asyncronious call to set
1376// something that is not used.
1377#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001378 const webrtc::RtpExtension* send_time_extension =
1379 FindHeaderExtension(extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001380 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001381 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001382 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001383 RTC_FROM_HERE, network_thread_,
1384 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1385 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001386#endif
1387}
1388
1389void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1390 int rtp_abs_sendtime_extn_id) {
1391 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001392}
1393
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001394void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001395 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001396 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001397 case MSG_SEND_RTP_PACKET:
1398 case MSG_SEND_RTCP_PACKET: {
1399 RTC_DCHECK(network_thread_->IsCurrent());
1400 SendPacketMessageData* data =
1401 static_cast<SendPacketMessageData*>(pmsg->pdata);
1402 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1403 SendPacket(rtcp, &data->packet, data->options);
1404 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001405 break;
1406 }
1407 case MSG_FIRSTPACKETRECEIVED: {
1408 SignalFirstPacketReceived(this);
1409 break;
1410 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001411 }
1412}
1413
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001414void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001415 // Flush all remaining RTCP messages. This should only be called in
1416 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001417 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001418 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001419 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1420 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001421 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1422 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001423 }
1424}
1425
johand89ab142016-10-25 10:50:32 -07001426void BaseChannel::SignalSentPacket_n(
1427 rtc::PacketTransportInterface* /* transport */,
1428 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001429 RTC_DCHECK(network_thread_->IsCurrent());
1430 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001431 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001432 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1433}
1434
1435void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1436 RTC_DCHECK(worker_thread_->IsCurrent());
1437 SignalSentPacket(sent_packet);
1438}
1439
1440VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1441 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001442 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001443 MediaEngineInterface* media_engine,
1444 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001445 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001446 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001447 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001448 : BaseChannel(worker_thread,
1449 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001450 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001451 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001452 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001453 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001454 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001455 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001456 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001457
1458VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001459 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001460 StopAudioMonitor();
1461 StopMediaMonitor();
1462 // this can't be done in the base class, since it calls a virtual
1463 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001464 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001465}
1466
zhihuangf5b251b2017-01-12 19:37:48 -08001467bool VoiceChannel::Init_w(TransportChannel* rtp_transport,
1468 TransportChannel* rtcp_transport) {
1469 return BaseChannel::Init_w(rtp_transport, rtcp_transport);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001470}
1471
Peter Boström0c4e06b2015-10-07 12:23:21 +02001472bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001473 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001474 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001475 AudioSource* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001476 return InvokeOnWorker(RTC_FROM_HERE,
1477 Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001478 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001479}
1480
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001481// TODO(juberti): Handle early media the right way. We should get an explicit
1482// ringing message telling us to start playing local ringback, which we cancel
1483// if any early media actually arrives. For now, we do the opposite, which is
1484// to wait 1 second for early media, and start playing local ringback if none
1485// arrives.
1486void VoiceChannel::SetEarlyMedia(bool enable) {
1487 if (enable) {
1488 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001489 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1490 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001491 } else {
1492 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001493 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494 }
1495}
1496
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001497bool VoiceChannel::CanInsertDtmf() {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001498 return InvokeOnWorker(
1499 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001500}
1501
Peter Boström0c4e06b2015-10-07 12:23:21 +02001502bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1503 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001504 int duration) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001505 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceChannel::InsertDtmf_w, this,
1506 ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001507}
1508
solenberg4bac9c52015-10-09 02:32:53 -07001509bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001510 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetOutputVolume,
1511 media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001512}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001513
Tommif888bb52015-12-12 01:37:01 +01001514void VoiceChannel::SetRawAudioSink(
1515 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001516 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1517 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001518 // passing. So we invoke to our own little routine that gets a pointer to
1519 // our local variable. This is OK since we're synchronously invoking.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001520 InvokeOnWorker(RTC_FROM_HERE,
1521 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001522}
1523
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001524webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001525 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001526 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001527}
1528
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001529webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1530 uint32_t ssrc) const {
1531 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001532}
1533
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001534bool VoiceChannel::SetRtpSendParameters(
1535 uint32_t ssrc,
1536 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001537 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001538 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001539 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001540}
1541
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001542bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1543 webrtc::RtpParameters parameters) {
1544 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1545}
1546
1547webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1548 uint32_t ssrc) const {
1549 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001550 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001551 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1552}
1553
1554webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1555 uint32_t ssrc) const {
1556 return media_channel()->GetRtpReceiveParameters(ssrc);
1557}
1558
1559bool VoiceChannel::SetRtpReceiveParameters(
1560 uint32_t ssrc,
1561 const webrtc::RtpParameters& parameters) {
1562 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001563 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001564 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1565}
1566
1567bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1568 webrtc::RtpParameters parameters) {
1569 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001570}
1571
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001572bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001573 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1574 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001575}
1576
1577void VoiceChannel::StartMediaMonitor(int cms) {
1578 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001579 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001580 media_monitor_->SignalUpdate.connect(
1581 this, &VoiceChannel::OnMediaMonitorUpdate);
1582 media_monitor_->Start(cms);
1583}
1584
1585void VoiceChannel::StopMediaMonitor() {
1586 if (media_monitor_) {
1587 media_monitor_->Stop();
1588 media_monitor_->SignalUpdate.disconnect(this);
1589 media_monitor_.reset();
1590 }
1591}
1592
1593void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001594 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001595 audio_monitor_
1596 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1597 audio_monitor_->Start(cms);
1598}
1599
1600void VoiceChannel::StopAudioMonitor() {
1601 if (audio_monitor_) {
1602 audio_monitor_->Stop();
1603 audio_monitor_.reset();
1604 }
1605}
1606
1607bool VoiceChannel::IsAudioMonitorRunning() const {
1608 return (audio_monitor_.get() != NULL);
1609}
1610
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001611int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001612 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001613}
1614
1615int VoiceChannel::GetOutputLevel_w() {
1616 return media_channel()->GetOutputLevel();
1617}
1618
1619void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1620 media_channel()->GetActiveStreams(actives);
1621}
1622
johand89ab142016-10-25 10:50:32 -07001623void VoiceChannel::OnPacketRead(rtc::PacketTransportInterface* transport,
1624 const char* data,
1625 size_t len,
1626 const rtc::PacketTime& packet_time,
wu@webrtc.orga9890802013-12-13 00:21:03 +00001627 int flags) {
johand89ab142016-10-25 10:50:32 -07001628 BaseChannel::OnPacketRead(transport, data, len, packet_time, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001629 // Set a flag when we've received an RTP packet. If we're waiting for early
1630 // media, this will disable the timeout.
johand89ab142016-10-25 10:50:32 -07001631 if (!received_media_ && !PacketIsRtcp(transport, data, len)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001632 received_media_ = true;
1633 }
1634}
1635
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001636void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001637 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001638 invoker_.AsyncInvoke<void>(
1639 RTC_FROM_HERE, worker_thread_,
1640 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001641}
1642
michaelt79e05882016-11-08 02:50:09 -08001643int BaseChannel::GetTransportOverheadPerPacket() const {
1644 RTC_DCHECK(network_thread_->IsCurrent());
1645
1646 if (!selected_candidate_pair_)
1647 return 0;
1648
1649 int transport_overhead_per_packet = 0;
1650
1651 constexpr int kIpv4Overhaed = 20;
1652 constexpr int kIpv6Overhaed = 40;
1653 transport_overhead_per_packet +=
1654 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1655 ? kIpv4Overhaed
1656 : kIpv6Overhaed;
1657
1658 constexpr int kUdpOverhaed = 8;
1659 constexpr int kTcpOverhaed = 20;
1660 transport_overhead_per_packet +=
1661 selected_candidate_pair_->local_candidate().protocol() ==
1662 TCP_PROTOCOL_NAME
1663 ? kTcpOverhaed
1664 : kUdpOverhaed;
1665
1666 if (secure()) {
1667 int srtp_overhead = 0;
1668 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1669 transport_overhead_per_packet += srtp_overhead;
1670 }
1671
1672 return transport_overhead_per_packet;
1673}
1674
1675void BaseChannel::UpdateTransportOverhead() {
1676 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1677 if (transport_overhead_per_packet)
1678 invoker_.AsyncInvoke<void>(
1679 RTC_FROM_HERE, worker_thread_,
1680 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1681 transport_overhead_per_packet));
1682}
1683
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001684void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001685 // Render incoming data if we're the active call, and we have the local
1686 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001687 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001688 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001689
1690 // Send outgoing data if we're the active call, we have the remote content,
1691 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001692 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001693 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001694
1695 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1696}
1697
1698const ContentInfo* VoiceChannel::GetFirstContent(
1699 const SessionDescription* sdesc) {
1700 return GetFirstAudioContent(sdesc);
1701}
1702
1703bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001704 ContentAction action,
1705 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001706 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001707 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001708 LOG(LS_INFO) << "Setting local voice description";
1709
1710 const AudioContentDescription* audio =
1711 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001712 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001713 if (!audio) {
1714 SafeSetError("Can't find audio content in local description.", error_desc);
1715 return false;
1716 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001717
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001718 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001719 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001720 }
1721
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001722 AudioRecvParameters recv_params = last_recv_params_;
1723 RtpParametersFromMediaDescription(audio, &recv_params);
1724 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001725 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001726 error_desc);
1727 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001728 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001729 for (const AudioCodec& codec : audio->codecs()) {
1730 bundle_filter()->AddPayloadType(codec.id);
1731 }
1732 last_recv_params_ = recv_params;
1733
1734 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1735 // only give it to the media channel once we have a remote
1736 // description too (without a remote description, we won't be able
1737 // to send them anyway).
1738 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1739 SafeSetError("Failed to set local audio description streams.", error_desc);
1740 return false;
1741 }
1742
1743 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001744 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001745 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001746}
1747
1748bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001749 ContentAction action,
1750 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001751 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001752 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001753 LOG(LS_INFO) << "Setting remote voice description";
1754
1755 const AudioContentDescription* audio =
1756 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001757 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001758 if (!audio) {
1759 SafeSetError("Can't find audio content in remote description.", error_desc);
1760 return false;
1761 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001762
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001763 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001764 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001765 }
1766
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001767 AudioSendParameters send_params = last_send_params_;
1768 RtpSendParametersFromMediaDescription(audio, &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001769 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001770 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001771 }
skvladdc1c62c2016-03-16 19:07:43 -07001772
1773 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1774 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001775 SafeSetError("Failed to set remote audio description send parameters.",
1776 error_desc);
1777 return false;
1778 }
1779 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001780
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001781 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1782 // and only give it to the media channel once we have a local
1783 // description too (without a local description, we won't be able to
1784 // recv them anyway).
1785 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1786 SafeSetError("Failed to set remote audio description streams.", error_desc);
1787 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001788 }
1789
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001790 if (audio->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001791 MaybeCacheRtpAbsSendTimeHeaderExtension_w(audio->rtp_header_extensions());
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001792 }
1793
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001794 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001795 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001796 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001797}
1798
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001799void VoiceChannel::HandleEarlyMediaTimeout() {
1800 // This occurs on the main thread, not the worker thread.
1801 if (!received_media_) {
1802 LOG(LS_INFO) << "No early media received before timeout";
1803 SignalEarlyMediaTimeout(this);
1804 }
1805}
1806
Peter Boström0c4e06b2015-10-07 12:23:21 +02001807bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1808 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001809 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001810 if (!enabled()) {
1811 return false;
1812 }
solenberg1d63dd02015-12-02 12:35:09 -08001813 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001814}
1815
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001816void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001817 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001818 case MSG_EARLYMEDIATIMEOUT:
1819 HandleEarlyMediaTimeout();
1820 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001821 case MSG_CHANNEL_ERROR: {
1822 VoiceChannelErrorMessageData* data =
1823 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001824 delete data;
1825 break;
1826 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827 default:
1828 BaseChannel::OnMessage(pmsg);
1829 break;
1830 }
1831}
1832
1833void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001834 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001835 SignalConnectionMonitor(this, infos);
1836}
1837
1838void VoiceChannel::OnMediaMonitorUpdate(
1839 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001840 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001841 SignalMediaMonitor(this, info);
1842}
1843
1844void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1845 const AudioInfo& info) {
1846 SignalAudioMonitor(this, info);
1847}
1848
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001849void VoiceChannel::GetSrtpCryptoSuites_n(
1850 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07001851 GetSupportedAudioCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001852}
1853
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001854VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1855 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001856 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001857 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001858 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001859 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001860 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001861 : BaseChannel(worker_thread,
1862 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001863 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001864 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001865 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001866 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001867 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001868
zhihuangf5b251b2017-01-12 19:37:48 -08001869bool VideoChannel::Init_w(TransportChannel* rtp_transport,
1870 TransportChannel* rtcp_transport) {
1871 return BaseChannel::Init_w(rtp_transport, rtcp_transport);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001872}
1873
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001874VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001875 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001876 StopMediaMonitor();
1877 // this can't be done in the base class, since it calls a virtual
1878 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001879
1880 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001881}
1882
nisse08582ff2016-02-04 01:24:52 -08001883bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001884 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001885 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001886 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001887 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001888 return true;
1889}
1890
deadbeef5a4a75a2016-06-02 16:23:38 -07001891bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001892 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001893 bool mute,
1894 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001895 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001896 return InvokeOnWorker(RTC_FROM_HERE,
1897 Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
deadbeef5a4a75a2016-06-02 16:23:38 -07001898 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001899}
1900
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001901webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001902 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001903 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001904}
1905
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001906webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1907 uint32_t ssrc) const {
1908 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001909}
1910
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001911bool VideoChannel::SetRtpSendParameters(
1912 uint32_t ssrc,
1913 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001914 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001915 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001916 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001917}
1918
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001919bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1920 webrtc::RtpParameters parameters) {
1921 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1922}
1923
1924webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1925 uint32_t ssrc) const {
1926 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001927 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001928 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1929}
1930
1931webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1932 uint32_t ssrc) const {
1933 return media_channel()->GetRtpReceiveParameters(ssrc);
1934}
1935
1936bool VideoChannel::SetRtpReceiveParameters(
1937 uint32_t ssrc,
1938 const webrtc::RtpParameters& parameters) {
1939 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001940 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001941 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1942}
1943
1944bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1945 webrtc::RtpParameters parameters) {
1946 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001947}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001948
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001949void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001950 // Send outgoing data if we're the active call, we have the remote content,
1951 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001952 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001953 if (!media_channel()->SetSend(send)) {
1954 LOG(LS_ERROR) << "Failed to SetSend on video channel";
1955 // TODO(gangji): Report error back to server.
1956 }
1957
Peter Boström34fbfff2015-09-24 19:20:30 +02001958 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001959}
1960
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001961bool VideoChannel::GetStats(VideoMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001962 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1963 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001964}
1965
1966void VideoChannel::StartMediaMonitor(int cms) {
1967 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001968 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001969 media_monitor_->SignalUpdate.connect(
1970 this, &VideoChannel::OnMediaMonitorUpdate);
1971 media_monitor_->Start(cms);
1972}
1973
1974void VideoChannel::StopMediaMonitor() {
1975 if (media_monitor_) {
1976 media_monitor_->Stop();
1977 media_monitor_.reset();
1978 }
1979}
1980
1981const ContentInfo* VideoChannel::GetFirstContent(
1982 const SessionDescription* sdesc) {
1983 return GetFirstVideoContent(sdesc);
1984}
1985
1986bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001987 ContentAction action,
1988 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001989 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001990 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001991 LOG(LS_INFO) << "Setting local video description";
1992
1993 const VideoContentDescription* video =
1994 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001995 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001996 if (!video) {
1997 SafeSetError("Can't find video content in local description.", error_desc);
1998 return false;
1999 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002000
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002001 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002002 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002003 }
2004
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002005 VideoRecvParameters recv_params = last_recv_params_;
2006 RtpParametersFromMediaDescription(video, &recv_params);
2007 if (!media_channel()->SetRecvParameters(recv_params)) {
2008 SafeSetError("Failed to set local video description recv parameters.",
2009 error_desc);
2010 return false;
2011 }
2012 for (const VideoCodec& codec : video->codecs()) {
2013 bundle_filter()->AddPayloadType(codec.id);
2014 }
2015 last_recv_params_ = recv_params;
2016
2017 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2018 // only give it to the media channel once we have a remote
2019 // description too (without a remote description, we won't be able
2020 // to send them anyway).
2021 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2022 SafeSetError("Failed to set local video description streams.", error_desc);
2023 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002024 }
2025
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002026 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002027 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002028 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002029}
2030
2031bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002032 ContentAction action,
2033 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002034 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002035 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002036 LOG(LS_INFO) << "Setting remote video description";
2037
2038 const VideoContentDescription* video =
2039 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002040 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002041 if (!video) {
2042 SafeSetError("Can't find video content in remote description.", error_desc);
2043 return false;
2044 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002045
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002046 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002047 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002048 }
2049
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002050 VideoSendParameters send_params = last_send_params_;
2051 RtpSendParametersFromMediaDescription(video, &send_params);
2052 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002053 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002054 }
skvladdc1c62c2016-03-16 19:07:43 -07002055
2056 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2057
2058 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002059 SafeSetError("Failed to set remote video description send parameters.",
2060 error_desc);
2061 return false;
2062 }
2063 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002064
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002065 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2066 // and only give it to the media channel once we have a local
2067 // description too (without a local description, we won't be able to
2068 // recv them anyway).
2069 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2070 SafeSetError("Failed to set remote video description streams.", error_desc);
2071 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002072 }
2073
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002074 if (video->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002075 MaybeCacheRtpAbsSendTimeHeaderExtension_w(video->rtp_header_extensions());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002076 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002077
2078 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002079 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002080 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002081}
2082
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002083void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002084 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002085 case MSG_CHANNEL_ERROR: {
2086 const VideoChannelErrorMessageData* data =
2087 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002088 delete data;
2089 break;
2090 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002091 default:
2092 BaseChannel::OnMessage(pmsg);
2093 break;
2094 }
2095}
2096
2097void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002098 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002099 SignalConnectionMonitor(this, infos);
2100}
2101
2102// TODO(pthatcher): Look into removing duplicate code between
2103// audio, video, and data, perhaps by using templates.
2104void VideoChannel::OnMediaMonitorUpdate(
2105 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002106 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002107 SignalMediaMonitor(this, info);
2108}
2109
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002110void VideoChannel::GetSrtpCryptoSuites_n(
2111 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002112 GetSupportedVideoCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002113}
2114
deadbeef953c2ce2017-01-09 14:53:41 -08002115RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2116 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002117 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002118 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002119 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002120 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002121 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002122 : BaseChannel(worker_thread,
2123 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002124 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002125 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002126 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002127 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002128 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002129
deadbeef953c2ce2017-01-09 14:53:41 -08002130RtpDataChannel::~RtpDataChannel() {
2131 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002132 StopMediaMonitor();
2133 // this can't be done in the base class, since it calls a virtual
2134 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002135
2136 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002137}
2138
zhihuangf5b251b2017-01-12 19:37:48 -08002139bool RtpDataChannel::Init_w(TransportChannel* rtp_transport,
2140 TransportChannel* rtcp_transport) {
2141 if (!BaseChannel::Init_w(rtp_transport, rtcp_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002142 return false;
2143 }
deadbeef953c2ce2017-01-09 14:53:41 -08002144 media_channel()->SignalDataReceived.connect(this,
2145 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002146 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002147 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002148 return true;
2149}
2150
deadbeef953c2ce2017-01-09 14:53:41 -08002151bool RtpDataChannel::SendData(const SendDataParams& params,
2152 const rtc::CopyOnWriteBuffer& payload,
2153 SendDataResult* result) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002154 return InvokeOnWorker(
2155 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2156 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002157}
2158
deadbeef953c2ce2017-01-09 14:53:41 -08002159const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002160 const SessionDescription* sdesc) {
2161 return GetFirstDataContent(sdesc);
2162}
2163
deadbeef953c2ce2017-01-09 14:53:41 -08002164bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002165 const DataContentDescription* content,
2166 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002167 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2168 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002169 // It's been set before, but doesn't match. That's bad.
2170 if (is_sctp) {
2171 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2172 error_desc);
2173 return false;
2174 }
2175 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002176}
2177
deadbeef953c2ce2017-01-09 14:53:41 -08002178bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2179 ContentAction action,
2180 std::string* error_desc) {
2181 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002182 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002183 LOG(LS_INFO) << "Setting local data description";
2184
2185 const DataContentDescription* data =
2186 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002187 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002188 if (!data) {
2189 SafeSetError("Can't find data content in local description.", error_desc);
2190 return false;
2191 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002192
deadbeef953c2ce2017-01-09 14:53:41 -08002193 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002194 return false;
2195 }
2196
deadbeef953c2ce2017-01-09 14:53:41 -08002197 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
2198 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002199 }
2200
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002201 DataRecvParameters recv_params = last_recv_params_;
2202 RtpParametersFromMediaDescription(data, &recv_params);
2203 if (!media_channel()->SetRecvParameters(recv_params)) {
2204 SafeSetError("Failed to set remote data description recv parameters.",
2205 error_desc);
2206 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002207 }
deadbeef953c2ce2017-01-09 14:53:41 -08002208 for (const DataCodec& codec : data->codecs()) {
2209 bundle_filter()->AddPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002210 }
2211 last_recv_params_ = recv_params;
2212
2213 // TODO(pthatcher): Move local streams into DataSendParameters, and
2214 // only give it to the media channel once we have a remote
2215 // description too (without a remote description, we won't be able
2216 // to send them anyway).
2217 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2218 SafeSetError("Failed to set local data description streams.", error_desc);
2219 return false;
2220 }
2221
2222 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002223 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002224 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002225}
2226
deadbeef953c2ce2017-01-09 14:53:41 -08002227bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2228 ContentAction action,
2229 std::string* error_desc) {
2230 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002231 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002232
2233 const DataContentDescription* data =
2234 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002235 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002236 if (!data) {
2237 SafeSetError("Can't find data content in remote description.", error_desc);
2238 return false;
2239 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002240
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002241 // If the remote data doesn't have codecs and isn't an update, it
2242 // must be empty, so ignore it.
2243 if (!data->has_codecs() && action != CA_UPDATE) {
2244 return true;
2245 }
2246
deadbeef953c2ce2017-01-09 14:53:41 -08002247 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002248 return false;
2249 }
2250
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002251 LOG(LS_INFO) << "Setting remote data description";
deadbeef953c2ce2017-01-09 14:53:41 -08002252 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002253 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002254 }
2255
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002256 DataSendParameters send_params = last_send_params_;
2257 RtpSendParametersFromMediaDescription<DataCodec>(data, &send_params);
2258 if (!media_channel()->SetSendParameters(send_params)) {
2259 SafeSetError("Failed to set remote data description send parameters.",
2260 error_desc);
2261 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002262 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002263 last_send_params_ = send_params;
2264
2265 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2266 // and only give it to the media channel once we have a local
2267 // description too (without a local description, we won't be able to
2268 // recv them anyway).
2269 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2270 SafeSetError("Failed to set remote data description streams.",
2271 error_desc);
2272 return false;
2273 }
2274
2275 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002276 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002277 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002278}
2279
deadbeef953c2ce2017-01-09 14:53:41 -08002280void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002281 // Render incoming data if we're the active call, and we have the local
2282 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002283 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002284 if (!media_channel()->SetReceive(recv)) {
2285 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2286 }
2287
2288 // Send outgoing data if we're the active call, we have the remote content,
2289 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002290 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002291 if (!media_channel()->SetSend(send)) {
2292 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2293 }
2294
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002295 // Trigger SignalReadyToSendData asynchronously.
2296 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002297
2298 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2299}
2300
deadbeef953c2ce2017-01-09 14:53:41 -08002301void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002302 switch (pmsg->message_id) {
2303 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002304 DataChannelReadyToSendMessageData* data =
2305 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002306 ready_to_send_data_ = data->data();
2307 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002308 delete data;
2309 break;
2310 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002311 case MSG_DATARECEIVED: {
2312 DataReceivedMessageData* data =
2313 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002314 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002315 delete data;
2316 break;
2317 }
2318 case MSG_CHANNEL_ERROR: {
2319 const DataChannelErrorMessageData* data =
2320 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002321 delete data;
2322 break;
2323 }
2324 default:
2325 BaseChannel::OnMessage(pmsg);
2326 break;
2327 }
2328}
2329
deadbeef953c2ce2017-01-09 14:53:41 -08002330void RtpDataChannel::OnConnectionMonitorUpdate(
2331 ConnectionMonitor* monitor,
2332 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002333 SignalConnectionMonitor(this, infos);
2334}
2335
deadbeef953c2ce2017-01-09 14:53:41 -08002336void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002337 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002338 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002339 media_monitor_->SignalUpdate.connect(this,
2340 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002341 media_monitor_->Start(cms);
2342}
2343
deadbeef953c2ce2017-01-09 14:53:41 -08002344void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002345 if (media_monitor_) {
2346 media_monitor_->Stop();
2347 media_monitor_->SignalUpdate.disconnect(this);
2348 media_monitor_.reset();
2349 }
2350}
2351
deadbeef953c2ce2017-01-09 14:53:41 -08002352void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2353 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002354 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002355 SignalMediaMonitor(this, info);
2356}
2357
deadbeef953c2ce2017-01-09 14:53:41 -08002358void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2359 const char* data,
2360 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002361 DataReceivedMessageData* msg = new DataReceivedMessageData(
2362 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002363 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002364}
2365
deadbeef953c2ce2017-01-09 14:53:41 -08002366void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2367 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002368 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2369 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002370 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002371}
2372
deadbeef953c2ce2017-01-09 14:53:41 -08002373void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002374 // This is usded for congestion control to indicate that the stream is ready
2375 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2376 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002377 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002378 new DataChannelReadyToSendMessageData(writable));
2379}
2380
deadbeef953c2ce2017-01-09 14:53:41 -08002381void RtpDataChannel::GetSrtpCryptoSuites_n(
2382 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002383 GetSupportedDataCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002384}
2385
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002386} // namespace cricket