blob: 942e8267cbad87b6749e86c2cb526148da695db6 [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());
zhihuangf5b251b2017-01-12 19:37:48 -0800231 // const std::string& transport_name =
232 // (bundle_transport_name ? *bundle_transport_name : content_name());
233 if (!SetTransport_n(rtp_transport, rtcp_transport)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200234 return false;
235 }
236
zhihuangf5b251b2017-01-12 19:37:48 -0800237 if (!SetDtlsSrtpCryptoSuites_n(rtp_transport_, false)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200238 return false;
239 }
zhihuangf5b251b2017-01-12 19:37:48 -0800240 if (rtcp_transport_ && !SetDtlsSrtpCryptoSuites_n(rtcp_transport_, true)) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200241 return false;
242 }
deadbeefac22f702017-01-12 21:59:29 -0800243 if (rtcp_mux_required_) {
244 rtcp_mux_filter_.SetActive();
245 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200246 return true;
247}
248
wu@webrtc.org78187522013-10-07 23:32:02 +0000249void BaseChannel::Deinit() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200250 RTC_DCHECK(worker_thread_->IsCurrent());
wu@webrtc.org78187522013-10-07 23:32:02 +0000251 media_channel_->SetInterface(NULL);
Danil Chapovalovdae07ba2016-05-14 01:43:50 +0200252 // Packets arrive on the network thread, processing packets calls virtual
253 // functions, so need to stop this process in Deinit that is called in
254 // derived classes destructor.
255 network_thread_->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700256 RTC_FROM_HERE, Bind(&BaseChannel::DisconnectTransportChannels_n, this));
wu@webrtc.org78187522013-10-07 23:32:02 +0000257}
258
zhihuangf5b251b2017-01-12 19:37:48 -0800259bool BaseChannel::SetTransport(TransportChannel* rtp_transport,
260 TransportChannel* rtcp_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200261 return network_thread_->Invoke<bool>(
zhihuangf5b251b2017-01-12 19:37:48 -0800262 RTC_FROM_HERE,
263 Bind(&BaseChannel::SetTransport_n, this, rtp_transport, rtcp_transport));
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000264}
265
zhihuangf5b251b2017-01-12 19:37:48 -0800266bool BaseChannel::SetTransport_n(TransportChannel* rtp_transport,
267 TransportChannel* rtcp_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200268 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangf5b251b2017-01-12 19:37:48 -0800269 if (!rtp_transport && !rtcp_transport) {
270 LOG(LS_ERROR) << "Setting nullptr to RTP Transport and RTCP Transport.";
271 return false;
272 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000273
zhihuangf5b251b2017-01-12 19:37:48 -0800274 if (rtcp_transport) {
275 RTC_DCHECK(rtp_transport->transport_name() ==
276 rtcp_transport->transport_name());
277 RTC_DCHECK(NeedsRtcpTransport());
278 }
279
280 if (rtp_transport->transport_name() == transport_name_) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700281 // Nothing to do if transport name isn't changing.
deadbeefcbecd352015-09-23 11:50:27 -0700282 return true;
283 }
284
zhihuangf5b251b2017-01-12 19:37:48 -0800285 transport_name_ = rtp_transport->transport_name();
286
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800287 // When using DTLS-SRTP, we must reset the SrtpFilter every time the transport
288 // changes and wait until the DTLS handshake is complete to set the newly
289 // negotiated parameters.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200290 if (ShouldSetupDtlsSrtp_n()) {
guoweis46383312015-12-17 16:45:59 -0800291 // Set |writable_| to false such that UpdateWritableState_w can set up
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700292 // DTLS-SRTP when |writable_| becomes true again.
guoweis46383312015-12-17 16:45:59 -0800293 writable_ = false;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800294 srtp_filter_.ResetParams();
295 }
296
deadbeefac22f702017-01-12 21:59:29 -0800297 // If this BaseChannel doesn't require RTCP mux and we haven't fully
298 // negotiated RTCP mux, we need an RTCP transport.
zhihuangf5b251b2017-01-12 19:37:48 -0800299 if (NeedsRtcpTransport()) {
300 LOG(LS_INFO) << "Setting RTCP Transport for " << content_name() << " on "
301 << transport_name() << " transport " << rtcp_transport;
302 SetTransportChannel_n(true, rtcp_transport);
303 if (!rtcp_transport_) {
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000304 return false;
305 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000306 }
307
zhihuangf5b251b2017-01-12 19:37:48 -0800308 LOG(LS_INFO) << "Setting non-RTCP Transport for " << content_name() << " on "
309 << transport_name() << " transport " << rtp_transport;
310 SetTransportChannel_n(false, rtp_transport);
311 if (!rtp_transport_) {
guoweis46383312015-12-17 16:45:59 -0800312 return false;
313 }
314
deadbeefcbecd352015-09-23 11:50:27 -0700315 // Update aggregate writable/ready-to-send state between RTP and RTCP upon
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700316 // setting new transport channels.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200317 UpdateWritableState_n();
deadbeef062ce9f2016-08-26 21:42:15 -0700318 // We can only update ready-to-send after updating writability.
319 //
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700320 // On setting a new channel, assume it's ready to send if it's writable,
321 // because we have no way of knowing otherwise (the channel doesn't give us
322 // "was last send successful?").
323 //
324 // This won't always be accurate (the last SendPacket call from another
325 // BaseChannel could have resulted in an error), but even so, we'll just
326 // encounter the error again and update "ready to send" accordingly.
zhihuangf5b251b2017-01-12 19:37:48 -0800327 SetTransportChannelReadyToSend(false,
328 rtp_transport_ && rtp_transport_->writable());
deadbeef062ce9f2016-08-26 21:42:15 -0700329 SetTransportChannelReadyToSend(
zhihuangf5b251b2017-01-12 19:37:48 -0800330 true, rtcp_transport_ && rtcp_transport_->writable());
deadbeef062ce9f2016-08-26 21:42:15 -0700331 return true;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000332}
333
deadbeef062ce9f2016-08-26 21:42:15 -0700334void BaseChannel::SetTransportChannel_n(bool rtcp,
zhihuangf5b251b2017-01-12 19:37:48 -0800335 TransportChannel* new_transport) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200336 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangf5b251b2017-01-12 19:37:48 -0800337 TransportChannel*& old_transport = rtcp ? rtcp_transport_ : rtp_transport_;
338 if (!old_transport && !new_transport) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700339 // Nothing to do.
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000340 return;
341 }
zhihuangf5b251b2017-01-12 19:37:48 -0800342 RTC_DCHECK(old_transport != new_transport);
343 if (old_transport) {
344 DisconnectFromTransportChannel(old_transport);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000345 }
346
zhihuangf5b251b2017-01-12 19:37:48 -0800347 old_transport = new_transport;
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000348
zhihuangf5b251b2017-01-12 19:37:48 -0800349 if (new_transport) {
deadbeef062ce9f2016-08-26 21:42:15 -0700350 if (rtcp) {
351 RTC_CHECK(!(ShouldSetupDtlsSrtp_n() && srtp_filter_.IsActive()))
352 << "Setting RTCP for DTLS/SRTP after SrtpFilter is active "
353 << "should never happen.";
deadbeefcbecd352015-09-23 11:50:27 -0700354 }
zhihuangf5b251b2017-01-12 19:37:48 -0800355 ConnectToTransportChannel(new_transport);
deadbeef062ce9f2016-08-26 21:42:15 -0700356 auto& socket_options = rtcp ? rtcp_socket_options_ : socket_options_;
357 for (const auto& pair : socket_options) {
zhihuangf5b251b2017-01-12 19:37:48 -0800358 new_transport->SetOption(pair.first, pair.second);
deadbeef062ce9f2016-08-26 21:42:15 -0700359 }
guoweis46383312015-12-17 16:45:59 -0800360 }
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000361}
362
363void BaseChannel::ConnectToTransportChannel(TransportChannel* tc) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200364 RTC_DCHECK(network_thread_->IsCurrent());
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000365
366 tc->SignalWritableState.connect(this, &BaseChannel::OnWritableState);
johand89ab142016-10-25 10:50:32 -0700367 tc->SignalReadPacket.connect(this, &BaseChannel::OnPacketRead);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000368 tc->SignalReadyToSend.connect(this, &BaseChannel::OnReadyToSend);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800369 tc->SignalDtlsState.connect(this, &BaseChannel::OnDtlsState);
Honghai Zhangcc411c02016-03-29 17:27:21 -0700370 tc->SignalSelectedCandidatePairChanged.connect(
371 this, &BaseChannel::OnSelectedCandidatePairChanged);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200372 tc->SignalSentPacket.connect(this, &BaseChannel::SignalSentPacket_n);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000373}
374
375void BaseChannel::DisconnectFromTransportChannel(TransportChannel* tc) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200376 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800377 OnSelectedCandidatePairChanged(tc, nullptr, -1, false);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000378
379 tc->SignalWritableState.disconnect(this);
380 tc->SignalReadPacket.disconnect(this);
381 tc->SignalReadyToSend.disconnect(this);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800382 tc->SignalDtlsState.disconnect(this);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200383 tc->SignalSelectedCandidatePairChanged.disconnect(this);
384 tc->SignalSentPacket.disconnect(this);
pthatcher@webrtc.org6ad507a2015-03-16 20:19:12 +0000385}
386
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000387bool BaseChannel::Enable(bool enable) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700388 worker_thread_->Invoke<void>(
389 RTC_FROM_HERE,
390 Bind(enable ? &BaseChannel::EnableMedia_w : &BaseChannel::DisableMedia_w,
391 this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000392 return true;
393}
394
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000395bool BaseChannel::AddRecvStream(const StreamParams& sp) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700396 return InvokeOnWorker(RTC_FROM_HERE,
397 Bind(&BaseChannel::AddRecvStream_w, this, sp));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000398}
399
Peter Boström0c4e06b2015-10-07 12:23:21 +0200400bool BaseChannel::RemoveRecvStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700401 return InvokeOnWorker(RTC_FROM_HERE,
402 Bind(&BaseChannel::RemoveRecvStream_w, this, ssrc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000403}
404
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000405bool BaseChannel::AddSendStream(const StreamParams& sp) {
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +0000406 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700407 RTC_FROM_HERE, Bind(&MediaChannel::AddSendStream, media_channel(), sp));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000408}
409
Peter Boström0c4e06b2015-10-07 12:23:21 +0200410bool BaseChannel::RemoveSendStream(uint32_t ssrc) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700411 return InvokeOnWorker(RTC_FROM_HERE, Bind(&MediaChannel::RemoveSendStream,
412 media_channel(), ssrc));
wu@webrtc.orgcadf9042013-08-30 21:24:16 +0000413}
414
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000415bool BaseChannel::SetLocalContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000416 ContentAction action,
417 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100418 TRACE_EVENT0("webrtc", "BaseChannel::SetLocalContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700419 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetLocalContent_w,
420 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000421}
422
423bool BaseChannel::SetRemoteContent(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000424 ContentAction action,
425 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +0100426 TRACE_EVENT0("webrtc", "BaseChannel::SetRemoteContent");
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700427 return InvokeOnWorker(RTC_FROM_HERE, Bind(&BaseChannel::SetRemoteContent_w,
428 this, content, action, error_desc));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000429}
430
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000431void BaseChannel::StartConnectionMonitor(int cms) {
zhihuangf5b251b2017-01-12 19:37:48 -0800432 // We pass in the BaseChannel instead of the rtp_transport_
433 // because if the rtp_transport_ changes, the ConnectionMonitor
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000434 // would be pointing to the wrong TransportChannel.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200435 // We pass in the network thread because on that thread connection monitor
436 // will call BaseChannel::GetConnectionStats which must be called on the
437 // network thread.
438 connection_monitor_.reset(
439 new ConnectionMonitor(this, network_thread(), rtc::Thread::Current()));
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000440 connection_monitor_->SignalUpdate.connect(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000441 this, &BaseChannel::OnConnectionMonitorUpdate);
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000442 connection_monitor_->Start(cms);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000443}
444
445void BaseChannel::StopConnectionMonitor() {
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000446 if (connection_monitor_) {
447 connection_monitor_->Stop();
448 connection_monitor_.reset();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000449 }
450}
451
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000452bool BaseChannel::GetConnectionStats(ConnectionInfos* infos) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200453 RTC_DCHECK(network_thread_->IsCurrent());
zhihuangf5b251b2017-01-12 19:37:48 -0800454 return rtp_transport_->GetStats(infos);
455}
456
457bool BaseChannel::NeedsRtcpTransport() {
deadbeefac22f702017-01-12 21:59:29 -0800458 // If this BaseChannel doesn't require RTCP mux and we haven't fully
459 // negotiated RTCP mux, we need an RTCP transport.
460 return !rtcp_mux_required_ && !rtcp_mux_filter_.IsFullyActive();
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +0000461}
462
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700463bool BaseChannel::IsReadyToReceiveMedia_w() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000464 // Receive data if we are enabled and have local content,
465 return enabled() && IsReceiveContentDirection(local_content_direction_);
466}
467
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700468bool BaseChannel::IsReadyToSendMedia_w() const {
469 // Need to access some state updated on the network thread.
470 return network_thread_->Invoke<bool>(
471 RTC_FROM_HERE, Bind(&BaseChannel::IsReadyToSendMedia_n, this));
472}
473
474bool BaseChannel::IsReadyToSendMedia_n() const {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000475 // Send outgoing data if we are enabled, have local and remote content,
476 // and we have had some form of connectivity.
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800477 return enabled() && IsReceiveContentDirection(remote_content_direction_) &&
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000478 IsSendContentDirection(local_content_direction_) &&
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700479 was_ever_writable() &&
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200480 (srtp_filter_.IsActive() || !ShouldSetupDtlsSrtp_n());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000481}
482
jbaucheec21bd2016-03-20 06:15:43 -0700483bool BaseChannel::SendPacket(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700484 const rtc::PacketOptions& options) {
485 return SendPacket(false, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000486}
487
jbaucheec21bd2016-03-20 06:15:43 -0700488bool BaseChannel::SendRtcp(rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700489 const rtc::PacketOptions& options) {
490 return SendPacket(true, packet, options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000491}
492
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000493int BaseChannel::SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000494 int value) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200495 return network_thread_->Invoke<int>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700496 RTC_FROM_HERE, Bind(&BaseChannel::SetOption_n, this, type, opt, value));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200497}
498
499int BaseChannel::SetOption_n(SocketType type,
500 rtc::Socket::Option opt,
501 int value) {
502 RTC_DCHECK(network_thread_->IsCurrent());
503 TransportChannel* channel = nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000504 switch (type) {
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000505 case ST_RTP:
zhihuangf5b251b2017-01-12 19:37:48 -0800506 channel = rtp_transport_;
deadbeefcbecd352015-09-23 11:50:27 -0700507 socket_options_.push_back(
508 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000509 break;
510 case ST_RTCP:
zhihuangf5b251b2017-01-12 19:37:48 -0800511 channel = rtcp_transport_;
deadbeefcbecd352015-09-23 11:50:27 -0700512 rtcp_socket_options_.push_back(
513 std::pair<rtc::Socket::Option, int>(opt, value));
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000514 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000515 }
wu@webrtc.org9caf2762013-12-11 18:25:07 +0000516 return channel ? channel->SetOption(opt, value) : -1;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000517}
518
jbauchcb560652016-08-04 05:20:32 -0700519bool BaseChannel::SetCryptoOptions(const rtc::CryptoOptions& crypto_options) {
520 crypto_options_ = crypto_options;
521 return true;
522}
523
johand89ab142016-10-25 10:50:32 -0700524void BaseChannel::OnWritableState(rtc::PacketTransportInterface* transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800525 RTC_DCHECK(transport == rtp_transport_ || transport == rtcp_transport_);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200526 RTC_DCHECK(network_thread_->IsCurrent());
527 UpdateWritableState_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000528}
529
johand89ab142016-10-25 10:50:32 -0700530void BaseChannel::OnPacketRead(rtc::PacketTransportInterface* transport,
531 const char* data,
532 size_t len,
533 const rtc::PacketTime& packet_time,
534 int flags) {
535 TRACE_EVENT0("webrtc", "BaseChannel::OnPacketRead");
536 // OnPacketRead gets called from P2PSocket; now pass data to MediaEngine
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200537 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000538
539 // When using RTCP multiplexing we might get RTCP packets on the RTP
540 // transport. We feed RTP traffic into the demuxer to determine if it is RTCP.
johand89ab142016-10-25 10:50:32 -0700541 bool rtcp = PacketIsRtcp(transport, data, len);
jbaucheec21bd2016-03-20 06:15:43 -0700542 rtc::CopyOnWriteBuffer packet(data, len);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000543 HandlePacket(rtcp, &packet, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000544}
545
johand89ab142016-10-25 10:50:32 -0700546void BaseChannel::OnReadyToSend(rtc::PacketTransportInterface* transport) {
zhihuangf5b251b2017-01-12 19:37:48 -0800547 RTC_DCHECK(transport == rtp_transport_ || transport == rtcp_transport_);
548 SetTransportChannelReadyToSend(transport == rtcp_transport_, true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000549}
550
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800551void BaseChannel::OnDtlsState(TransportChannel* channel,
552 DtlsTransportState state) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200553 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800554 return;
555 }
556
557 // Reset the srtp filter if it's not the CONNECTED state. For the CONNECTED
558 // state, setting up DTLS-SRTP context is deferred to ChannelWritable_w to
559 // cover other scenarios like the whole channel is writable (not just this
560 // TransportChannel) or when TransportChannel is attached after DTLS is
561 // negotiated.
562 if (state != DTLS_TRANSPORT_CONNECTED) {
563 srtp_filter_.ResetParams();
564 }
565}
566
Honghai Zhangcc411c02016-03-29 17:27:21 -0700567void BaseChannel::OnSelectedCandidatePairChanged(
568 TransportChannel* channel,
Honghai Zhang52dce732016-03-31 12:37:31 -0700569 CandidatePairInterface* selected_candidate_pair,
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700570 int last_sent_packet_id,
571 bool ready_to_send) {
zhihuangf5b251b2017-01-12 19:37:48 -0800572 RTC_DCHECK(channel == rtp_transport_ || channel == rtcp_transport_);
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200573 RTC_DCHECK(network_thread_->IsCurrent());
michaelt79e05882016-11-08 02:50:09 -0800574 selected_candidate_pair_ = selected_candidate_pair;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200575 std::string transport_name = channel->transport_name();
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700576 rtc::NetworkRoute network_route;
Honghai Zhangcc411c02016-03-29 17:27:21 -0700577 if (selected_candidate_pair) {
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700578 network_route = rtc::NetworkRoute(
Taylor Brandstetter6bb1ef22016-06-27 18:09:03 -0700579 ready_to_send, selected_candidate_pair->local_candidate().network_id(),
Honghai Zhang0e533ef2016-04-19 15:41:36 -0700580 selected_candidate_pair->remote_candidate().network_id(),
581 last_sent_packet_id);
michaelt79e05882016-11-08 02:50:09 -0800582
583 UpdateTransportOverhead();
Honghai Zhangcc411c02016-03-29 17:27:21 -0700584 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200585 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700586 RTC_FROM_HERE, worker_thread_,
587 Bind(&MediaChannel::OnNetworkRouteChanged, media_channel_, transport_name,
588 network_route));
Honghai Zhangcc411c02016-03-29 17:27:21 -0700589}
590
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700591void BaseChannel::SetTransportChannelReadyToSend(bool rtcp, bool ready) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200592 RTC_DCHECK(network_thread_->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700593 if (rtcp) {
594 rtcp_ready_to_send_ = ready;
595 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000596 rtp_ready_to_send_ = ready;
597 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000598
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200599 bool ready_to_send =
600 (rtp_ready_to_send_ &&
zhihuangf5b251b2017-01-12 19:37:48 -0800601 // In the case of rtcp mux |rtcp_transport_| will be null.
602 (rtcp_ready_to_send_ || !rtcp_transport_));
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200603
604 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700605 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200606 Bind(&MediaChannel::OnReadyToSend, media_channel_, ready_to_send));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000607}
608
johand89ab142016-10-25 10:50:32 -0700609bool BaseChannel::PacketIsRtcp(const rtc::PacketTransportInterface* transport,
610 const char* data,
611 size_t len) {
zhihuangf5b251b2017-01-12 19:37:48 -0800612 return (transport == rtcp_transport_ ||
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000613 rtcp_mux_filter_.DemuxRtcp(data, static_cast<int>(len)));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000614}
615
stefanc1aeaf02015-10-15 07:26:07 -0700616bool BaseChannel::SendPacket(bool rtcp,
jbaucheec21bd2016-03-20 06:15:43 -0700617 rtc::CopyOnWriteBuffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700618 const rtc::PacketOptions& options) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200619 // SendPacket gets called from MediaEngine, on a pacer or an encoder thread.
620 // If the thread is not our network thread, we will post to our network
621 // so that the real work happens on our network. This avoids us having to
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000622 // synchronize access to all the pieces of the send path, including
623 // SRTP and the inner workings of the transport channels.
624 // The only downside is that we can't return a proper failure code if
625 // needed. Since UDP is unreliable anyway, this should be a non-issue.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200626 if (!network_thread_->IsCurrent()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000627 // Avoid a copy by transferring the ownership of the packet data.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200628 int message_id = rtcp ? MSG_SEND_RTCP_PACKET : MSG_SEND_RTP_PACKET;
629 SendPacketMessageData* data = new SendPacketMessageData;
kwiberg0eb15ed2015-12-17 03:04:15 -0800630 data->packet = std::move(*packet);
stefanc1aeaf02015-10-15 07:26:07 -0700631 data->options = options;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700632 network_thread_->Post(RTC_FROM_HERE, this, message_id, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000633 return true;
634 }
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200635 TRACE_EVENT0("webrtc", "BaseChannel::SendPacket");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000636
637 // Now that we are on the correct thread, ensure we have a place to send this
638 // packet before doing anything. (We might get RTCP packets that we don't
639 // intend to send.) If we've negotiated RTCP mux, send RTCP over the RTP
640 // transport.
zhihuangf5b251b2017-01-12 19:37:48 -0800641 TransportChannel* channel =
642 (!rtcp || rtcp_mux_filter_.IsActive()) ? rtp_transport_ : rtcp_transport_;
wu@webrtc.org97077a32013-10-25 21:18:33 +0000643 if (!channel || !channel->writable()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000644 return false;
645 }
646
647 // Protect ourselves against crazy data.
648 if (!ValidPacket(rtcp, packet)) {
649 LOG(LS_ERROR) << "Dropping outgoing " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000650 << PacketType(rtcp)
651 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000652 return false;
653 }
654
stefanc1aeaf02015-10-15 07:26:07 -0700655 rtc::PacketOptions updated_options;
656 updated_options = options;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000657 // Protect if needed.
658 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200659 TRACE_EVENT0("webrtc", "SRTP Encode");
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000660 bool res;
Karl Wibergc56ac1e2015-05-04 14:54:55 +0200661 uint8_t* data = packet->data();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000662 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000663 if (!rtcp) {
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000664 // If ENABLE_EXTERNAL_AUTH flag is on then packet authentication is not done
665 // inside libsrtp for a RTP packet. A external HMAC module will be writing
666 // a fake HMAC value. This is ONLY done for a RTP packet.
667 // Socket layer will update rtp sendtime extension header if present in
668 // packet with current time before updating the HMAC.
669#if !defined(ENABLE_EXTERNAL_AUTH)
670 res = srtp_filter_.ProtectRtp(
671 data, len, static_cast<int>(packet->capacity()), &len);
672#else
stefanc1aeaf02015-10-15 07:26:07 -0700673 updated_options.packet_time_params.rtp_sendtime_extension_id =
henrike@webrtc.org05376342014-03-10 15:53:12 +0000674 rtp_abs_sendtime_extn_id_;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000675 res = srtp_filter_.ProtectRtp(
676 data, len, static_cast<int>(packet->capacity()), &len,
stefanc1aeaf02015-10-15 07:26:07 -0700677 &updated_options.packet_time_params.srtp_packet_index);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000678 // If protection succeeds, let's get auth params from srtp.
679 if (res) {
Peter Boström0c4e06b2015-10-07 12:23:21 +0200680 uint8_t* auth_key = NULL;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000681 int key_len;
682 res = srtp_filter_.GetRtpAuthParams(
stefanc1aeaf02015-10-15 07:26:07 -0700683 &auth_key, &key_len,
684 &updated_options.packet_time_params.srtp_auth_tag_len);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000685 if (res) {
stefanc1aeaf02015-10-15 07:26:07 -0700686 updated_options.packet_time_params.srtp_auth_key.resize(key_len);
687 updated_options.packet_time_params.srtp_auth_key.assign(
688 auth_key, auth_key + key_len);
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +0000689 }
690 }
691#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000692 if (!res) {
693 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200694 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000695 GetRtpSeqNum(data, len, &seq_num);
696 GetRtpSsrc(data, len, &ssrc);
697 LOG(LS_ERROR) << "Failed to protect " << content_name_
698 << " RTP packet: size=" << len
699 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
700 return false;
701 }
702 } else {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +0000703 res = srtp_filter_.ProtectRtcp(data, len,
704 static_cast<int>(packet->capacity()),
705 &len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000706 if (!res) {
707 int type = -1;
708 GetRtcpType(data, len, &type);
709 LOG(LS_ERROR) << "Failed to protect " << content_name_
710 << " RTCP packet: size=" << len << ", type=" << type;
711 return false;
712 }
713 }
714
715 // Update the length of the packet now that we've added the auth tag.
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000716 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800717 } else if (srtp_required_) {
deadbeef8f425f92016-12-01 12:26:27 -0800718 // The audio/video engines may attempt to send RTCP packets as soon as the
719 // streams are created, so don't treat this as an error for RTCP.
720 // See: https://bugs.chromium.org/p/webrtc/issues/detail?id=6809
721 if (rtcp) {
722 return false;
723 }
724 // However, there shouldn't be any RTP packets sent before SRTP is set up
725 // (and SetSend(true) is called).
726 LOG(LS_ERROR) << "Can't send outgoing RTP packet when SRTP is inactive"
727 << " and crypto is required";
nisseeb4ca4e2017-01-12 02:24:27 -0800728 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000729 return false;
730 }
731
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000732 // Bon voyage.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200733 int flags = (secure() && secure_dtls()) ? PF_SRTP_BYPASS : PF_NORMAL;
734 int ret = channel->SendPacket(packet->data<char>(), packet->size(),
735 updated_options, flags);
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000736 if (ret != static_cast<int>(packet->size())) {
skvladc309e0e2016-07-28 17:15:20 -0700737 if (channel->GetError() == ENOTCONN) {
738 LOG(LS_WARNING) << "Got ENOTCONN from transport.";
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700739 SetTransportChannelReadyToSend(rtcp, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000740 }
741 return false;
742 }
743 return true;
744}
745
jbaucheec21bd2016-03-20 06:15:43 -0700746bool BaseChannel::WantsPacket(bool rtcp, const rtc::CopyOnWriteBuffer* packet) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000747 // Protect ourselves against crazy data.
748 if (!ValidPacket(rtcp, packet)) {
749 LOG(LS_ERROR) << "Dropping incoming " << content_name_ << " "
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000750 << PacketType(rtcp)
751 << " packet: wrong size=" << packet->size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000752 return false;
753 }
pbos482b12e2015-11-16 10:19:58 -0800754 if (rtcp) {
755 // Permit all (seemingly valid) RTCP packets.
756 return true;
757 }
758 // Check whether we handle this payload.
jbaucheec21bd2016-03-20 06:15:43 -0700759 return bundle_filter_.DemuxPacket(packet->data(), packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000760}
761
jbaucheec21bd2016-03-20 06:15:43 -0700762void BaseChannel::HandlePacket(bool rtcp, rtc::CopyOnWriteBuffer* packet,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000763 const rtc::PacketTime& packet_time) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200764 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000765 if (!WantsPacket(rtcp, packet)) {
766 return;
767 }
768
honghaiz@google.coma67ca1a2015-01-28 19:48:33 +0000769 // We are only interested in the first rtp packet because that
770 // indicates the media has started flowing.
771 if (!has_received_packet_ && !rtcp) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000772 has_received_packet_ = true;
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700773 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_FIRSTPACKETRECEIVED);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000774 }
775
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000776 // Unprotect the packet, if needed.
777 if (srtp_filter_.IsActive()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200778 TRACE_EVENT0("webrtc", "SRTP Decode");
Karl Wiberg94784372015-04-20 14:03:07 +0200779 char* data = packet->data<char>();
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000780 int len = static_cast<int>(packet->size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000781 bool res;
782 if (!rtcp) {
783 res = srtp_filter_.UnprotectRtp(data, len, &len);
784 if (!res) {
785 int seq_num = -1;
Peter Boström0c4e06b2015-10-07 12:23:21 +0200786 uint32_t ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000787 GetRtpSeqNum(data, len, &seq_num);
788 GetRtpSsrc(data, len, &ssrc);
789 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
790 << " RTP packet: size=" << len
791 << ", seqnum=" << seq_num << ", SSRC=" << ssrc;
792 return;
793 }
794 } else {
795 res = srtp_filter_.UnprotectRtcp(data, len, &len);
796 if (!res) {
797 int type = -1;
798 GetRtcpType(data, len, &type);
799 LOG(LS_ERROR) << "Failed to unprotect " << content_name_
800 << " RTCP packet: size=" << len << ", type=" << type;
801 return;
802 }
803 }
804
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000805 packet->SetSize(len);
deadbeef7af91dd2016-12-13 11:29:11 -0800806 } else if (srtp_required_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000807 // Our session description indicates that SRTP is required, but we got a
808 // packet before our SRTP filter is active. This means either that
809 // a) we got SRTP packets before we received the SDES keys, in which case
810 // we can't decrypt it anyway, or
811 // b) we got SRTP packets before DTLS completed on both the RTP and RTCP
812 // channels, so we haven't yet extracted keys, even if DTLS did complete
813 // on the channel that the packets are being sent on. It's really good
814 // practice to wait for both RTP and RTCP to be good to go before sending
815 // media, to prevent weird failure modes, so it's fine for us to just eat
816 // packets here. This is all sidestepped if RTCP mux is used anyway.
817 LOG(LS_WARNING) << "Can't process incoming " << PacketType(rtcp)
818 << " packet when SRTP is inactive and crypto is required";
819 return;
820 }
821
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200822 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700823 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200824 Bind(&BaseChannel::OnPacketReceived, this, rtcp, *packet, packet_time));
825}
826
827void BaseChannel::OnPacketReceived(bool rtcp,
828 const rtc::CopyOnWriteBuffer& packet,
829 const rtc::PacketTime& packet_time) {
830 RTC_DCHECK(worker_thread_->IsCurrent());
831 // Need to copy variable because OnRtcpReceived/OnPacketReceived
832 // requires non-const pointer to buffer. This doesn't memcpy the actual data.
833 rtc::CopyOnWriteBuffer data(packet);
834 if (rtcp) {
835 media_channel_->OnRtcpReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000836 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200837 media_channel_->OnPacketReceived(&data, packet_time);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000838 }
839}
840
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000841bool BaseChannel::PushdownLocalDescription(
842 const SessionDescription* local_desc, ContentAction action,
843 std::string* error_desc) {
844 const ContentInfo* content_info = GetFirstContent(local_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000845 const MediaContentDescription* content_desc =
846 GetContentDescription(content_info);
847 if (content_desc && content_info && !content_info->rejected &&
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000848 !SetLocalContent(content_desc, action, error_desc)) {
849 LOG(LS_ERROR) << "Failure in SetLocalContent with action " << action;
850 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000851 }
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000852 return true;
853}
854
855bool BaseChannel::PushdownRemoteDescription(
856 const SessionDescription* remote_desc, ContentAction action,
857 std::string* error_desc) {
858 const ContentInfo* content_info = GetFirstContent(remote_desc);
859 const MediaContentDescription* content_desc =
860 GetContentDescription(content_info);
861 if (content_desc && content_info && !content_info->rejected &&
862 !SetRemoteContent(content_desc, action, error_desc)) {
863 LOG(LS_ERROR) << "Failure in SetRemoteContent with action " << action;
864 return false;
865 }
866 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000867}
868
869void BaseChannel::EnableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700870 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000871 if (enabled_)
872 return;
873
874 LOG(LS_INFO) << "Channel enabled";
875 enabled_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700876 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000877}
878
879void BaseChannel::DisableMedia_w() {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700880 RTC_DCHECK(worker_thread_ == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000881 if (!enabled_)
882 return;
883
884 LOG(LS_INFO) << "Channel disabled";
885 enabled_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700886 UpdateMediaSendRecvState_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000887}
888
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200889void BaseChannel::UpdateWritableState_n() {
zhihuangf5b251b2017-01-12 19:37:48 -0800890 if (rtp_transport_ && rtp_transport_->writable() &&
891 (!rtcp_transport_ || rtcp_transport_->writable())) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200892 ChannelWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700893 } else {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200894 ChannelNotWritable_n();
deadbeefcbecd352015-09-23 11:50:27 -0700895 }
896}
897
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200898void BaseChannel::ChannelWritable_n() {
899 RTC_DCHECK(network_thread_->IsCurrent());
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800900 if (writable_) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000901 return;
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800902 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000903
deadbeefcbecd352015-09-23 11:50:27 -0700904 LOG(LS_INFO) << "Channel writable (" << content_name_ << ")"
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905 << (was_ever_writable_ ? "" : " for the first time");
906
michaelt79e05882016-11-08 02:50:09 -0800907 if (selected_candidate_pair_)
908 LOG(LS_INFO)
909 << "Using "
910 << selected_candidate_pair_->local_candidate().ToSensitiveString()
911 << "->"
912 << selected_candidate_pair_->remote_candidate().ToSensitiveString();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000913
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000914 was_ever_writable_ = true;
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200915 MaybeSetupDtlsSrtp_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000916 writable_ = true;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700917 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000918}
919
deadbeef953c2ce2017-01-09 14:53:41 -0800920void BaseChannel::SignalDtlsSrtpSetupFailure_n(bool rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200921 RTC_DCHECK(network_thread_->IsCurrent());
922 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -0700923 RTC_FROM_HERE, signaling_thread(),
deadbeef953c2ce2017-01-09 14:53:41 -0800924 Bind(&BaseChannel::SignalDtlsSrtpSetupFailure_s, this, rtcp));
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000925}
926
deadbeef953c2ce2017-01-09 14:53:41 -0800927void BaseChannel::SignalDtlsSrtpSetupFailure_s(bool rtcp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -0700928 RTC_DCHECK(signaling_thread() == rtc::Thread::Current());
deadbeef953c2ce2017-01-09 14:53:41 -0800929 SignalDtlsSrtpSetupFailure(this, rtcp);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +0000930}
931
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200932bool BaseChannel::SetDtlsSrtpCryptoSuites_n(TransportChannel* tc, bool rtcp) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800933 std::vector<int> crypto_suites;
934 // We always use the default SRTP crypto suites for RTCP, but we may use
935 // different crypto suites for RTP depending on the media type.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000936 if (!rtcp) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200937 GetSrtpCryptoSuites_n(&crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000938 } else {
jbauchcb560652016-08-04 05:20:32 -0700939 GetDefaultSrtpCryptoSuites(crypto_options(), &crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000940 }
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800941 return tc->SetSrtpCryptoSuites(crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000942}
943
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200944bool BaseChannel::ShouldSetupDtlsSrtp_n() const {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800945 // Since DTLS is applied to all channels, checking RTP should be enough.
zhihuangf5b251b2017-01-12 19:37:48 -0800946 return rtp_transport_ && rtp_transport_->IsDtlsActive();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000947}
948
949// This function returns true if either DTLS-SRTP is not in use
950// *or* DTLS-SRTP is successfully set up.
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200951bool BaseChannel::SetupDtlsSrtp_n(bool rtcp_channel) {
952 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000953 bool ret = false;
954
zhihuangf5b251b2017-01-12 19:37:48 -0800955 TransportChannel* channel = rtcp_channel ? rtcp_transport_ : rtp_transport_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000956
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -0800957 RTC_DCHECK(channel->IsDtlsActive());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000958
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800959 int selected_crypto_suite;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000960
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -0800961 if (!channel->GetSrtpCryptoSuite(&selected_crypto_suite)) {
962 LOG(LS_ERROR) << "No DTLS-SRTP selected crypto suite";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000963 return false;
964 }
965
966 LOG(LS_INFO) << "Installing keys from DTLS-SRTP on "
967 << content_name() << " "
968 << PacketType(rtcp_channel);
969
jbauchcb560652016-08-04 05:20:32 -0700970 int key_len;
971 int salt_len;
972 if (!rtc::GetSrtpKeyAndSaltLengths(selected_crypto_suite, &key_len,
973 &salt_len)) {
974 LOG(LS_ERROR) << "Unknown DTLS-SRTP crypto suite" << selected_crypto_suite;
975 return false;
976 }
977
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000978 // OK, we're now doing DTLS (RFC 5764)
jbauchcb560652016-08-04 05:20:32 -0700979 std::vector<unsigned char> dtls_buffer(key_len * 2 + salt_len * 2);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000980
981 // RFC 5705 exporter using the RFC 5764 parameters
982 if (!channel->ExportKeyingMaterial(
983 kDtlsSrtpExporterLabel,
984 NULL, 0, false,
985 &dtls_buffer[0], dtls_buffer.size())) {
986 LOG(LS_WARNING) << "DTLS-SRTP key export failed";
nisseeb4ca4e2017-01-12 02:24:27 -0800987 RTC_NOTREACHED(); // This should never happen
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000988 return false;
989 }
990
991 // Sync up the keys with the DTLS-SRTP interface
jbauchcb560652016-08-04 05:20:32 -0700992 std::vector<unsigned char> client_write_key(key_len + salt_len);
993 std::vector<unsigned char> server_write_key(key_len + salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000994 size_t offset = 0;
jbauchcb560652016-08-04 05:20:32 -0700995 memcpy(&client_write_key[0], &dtls_buffer[offset], key_len);
996 offset += key_len;
997 memcpy(&server_write_key[0], &dtls_buffer[offset], key_len);
998 offset += key_len;
999 memcpy(&client_write_key[key_len], &dtls_buffer[offset], salt_len);
1000 offset += salt_len;
1001 memcpy(&server_write_key[key_len], &dtls_buffer[offset], salt_len);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001002
1003 std::vector<unsigned char> *send_key, *recv_key;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001004 rtc::SSLRole role;
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00001005 if (!channel->GetSslRole(&role)) {
1006 LOG(LS_WARNING) << "GetSslRole failed";
1007 return false;
1008 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001009
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001010 if (role == rtc::SSL_SERVER) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001011 send_key = &server_write_key;
1012 recv_key = &client_write_key;
1013 } else {
1014 send_key = &client_write_key;
1015 recv_key = &server_write_key;
1016 }
1017
1018 if (rtcp_channel) {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001019 ret = srtp_filter_.SetRtcpParams(selected_crypto_suite, &(*send_key)[0],
1020 static_cast<int>(send_key->size()),
1021 selected_crypto_suite, &(*recv_key)[0],
1022 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001023 } else {
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08001024 ret = srtp_filter_.SetRtpParams(selected_crypto_suite, &(*send_key)[0],
1025 static_cast<int>(send_key->size()),
1026 selected_crypto_suite, &(*recv_key)[0],
1027 static_cast<int>(recv_key->size()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001028 }
1029
michaelt79e05882016-11-08 02:50:09 -08001030 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001031 LOG(LS_WARNING) << "DTLS-SRTP key installation failed";
michaelt79e05882016-11-08 02:50:09 -08001032 } else {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001033 dtls_keyed_ = true;
michaelt79e05882016-11-08 02:50:09 -08001034 UpdateTransportOverhead();
1035 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001036 return ret;
1037}
1038
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001039void BaseChannel::MaybeSetupDtlsSrtp_n() {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001040 if (srtp_filter_.IsActive()) {
1041 return;
1042 }
1043
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001044 if (!ShouldSetupDtlsSrtp_n()) {
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001045 return;
1046 }
1047
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001048 if (!SetupDtlsSrtp_n(false)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001049 SignalDtlsSrtpSetupFailure_n(false);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001050 return;
1051 }
1052
zhihuangf5b251b2017-01-12 19:37:48 -08001053 if (rtcp_transport_) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001054 if (!SetupDtlsSrtp_n(true)) {
deadbeef953c2ce2017-01-09 14:53:41 -08001055 SignalDtlsSrtpSetupFailure_n(true);
Guo-wei Shieh1218d7a2015-12-05 09:59:56 -08001056 return;
1057 }
1058 }
1059}
1060
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001061void BaseChannel::ChannelNotWritable_n() {
1062 RTC_DCHECK(network_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001063 if (!writable_)
1064 return;
1065
deadbeefcbecd352015-09-23 11:50:27 -07001066 LOG(LS_INFO) << "Channel not writable (" << content_name_ << ")";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001067 writable_ = false;
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001068 UpdateMediaSendRecvState();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001069}
1070
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001071bool BaseChannel::SetRtpTransportParameters(
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001072 const MediaContentDescription* content,
1073 ContentAction action,
1074 ContentSource src,
1075 std::string* error_desc) {
1076 if (action == CA_UPDATE) {
1077 // These parameters never get changed by a CA_UDPATE.
1078 return true;
1079 }
1080
deadbeef7af91dd2016-12-13 11:29:11 -08001081 // Cache srtp_required_ for belt and suspenders check on SendPacket
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001082 return network_thread_->Invoke<bool>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001083 RTC_FROM_HERE, Bind(&BaseChannel::SetRtpTransportParameters_n, this,
1084 content, action, src, error_desc));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001085}
1086
1087bool BaseChannel::SetRtpTransportParameters_n(
1088 const MediaContentDescription* content,
1089 ContentAction action,
1090 ContentSource src,
1091 std::string* error_desc) {
1092 RTC_DCHECK(network_thread_->IsCurrent());
1093
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001094 if (!SetSrtp_n(content->cryptos(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001095 return false;
1096 }
1097
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001098 if (!SetRtcpMux_n(content->rtcp_mux(), action, src, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001099 return false;
1100 }
1101
1102 return true;
1103}
1104
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001105// |dtls| will be set to true if DTLS is active for transport channel and
1106// crypto is empty.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001107bool BaseChannel::CheckSrtpConfig_n(const std::vector<CryptoParams>& cryptos,
1108 bool* dtls,
1109 std::string* error_desc) {
zhihuangf5b251b2017-01-12 19:37:48 -08001110 *dtls = rtp_transport_->IsDtlsActive();
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001111 if (*dtls && !cryptos.empty()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001112 SafeSetError("Cryptos must be empty when DTLS is active.", error_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001113 return false;
1114 }
1115 return true;
1116}
1117
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001118bool BaseChannel::SetSrtp_n(const std::vector<CryptoParams>& cryptos,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001119 ContentAction action,
1120 ContentSource src,
1121 std::string* error_desc) {
Peter Boströmca8b4042016-03-08 14:24:13 -08001122 TRACE_EVENT0("webrtc", "BaseChannel::SetSrtp_w");
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001123 if (action == CA_UPDATE) {
1124 // no crypto params.
1125 return true;
1126 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001127 bool ret = false;
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001128 bool dtls = false;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001129 ret = CheckSrtpConfig_n(cryptos, &dtls, error_desc);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001130 if (!ret) {
1131 return false;
1132 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001133 switch (action) {
1134 case CA_OFFER:
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001135 // If DTLS is already active on the channel, we could be renegotiating
1136 // here. We don't update the srtp filter.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001137 if (!dtls) {
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00001138 ret = srtp_filter_.SetOffer(cryptos, src);
1139 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001140 break;
1141 case CA_PRANSWER:
1142 // If we're doing DTLS-SRTP, we don't want to update the filter
1143 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001144 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001145 ret = srtp_filter_.SetProvisionalAnswer(cryptos, src);
1146 }
1147 break;
1148 case CA_ANSWER:
1149 // If we're doing DTLS-SRTP, we don't want to update the filter
1150 // with an answer, because we already have SRTP parameters.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001151 if (!dtls) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001152 ret = srtp_filter_.SetAnswer(cryptos, src);
1153 }
1154 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001155 default:
1156 break;
1157 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001158 if (!ret) {
1159 SafeSetError("Failed to setup SRTP filter.", error_desc);
1160 return false;
1161 }
1162 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001163}
1164
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001165bool BaseChannel::SetRtcpMux_n(bool enable,
1166 ContentAction action,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001167 ContentSource src,
1168 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001169 bool ret = false;
1170 switch (action) {
1171 case CA_OFFER:
1172 ret = rtcp_mux_filter_.SetOffer(enable, src);
1173 break;
1174 case CA_PRANSWER:
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001175 // This may activate RTCP muxing, but we don't yet destroy the channel
1176 // because the final answer may deactivate it.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001177 ret = rtcp_mux_filter_.SetProvisionalAnswer(enable, src);
1178 break;
1179 case CA_ANSWER:
1180 ret = rtcp_mux_filter_.SetAnswer(enable, src);
1181 if (ret && rtcp_mux_filter_.IsActive()) {
1182 // We activated RTCP mux, close down the RTCP transport.
deadbeefcbecd352015-09-23 11:50:27 -07001183 LOG(LS_INFO) << "Enabling rtcp-mux for " << content_name()
1184 << " by destroying RTCP transport channel for "
1185 << transport_name();
deadbeefac22f702017-01-12 21:59:29 -08001186 if (rtcp_transport()) {
1187 SetTransportChannel_n(true, nullptr);
1188 SignalRtcpMuxFullyActive(rtp_transport()->transport_name());
zhihuangf5b251b2017-01-12 19:37:48 -08001189 }
deadbeef062ce9f2016-08-26 21:42:15 -07001190 UpdateWritableState_n();
1191 SetTransportChannelReadyToSend(true, false);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001192 }
1193 break;
1194 case CA_UPDATE:
1195 // No RTCP mux info.
1196 ret = true;
Henrik Kjellander7c027b62015-04-22 13:21:30 +02001197 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001198 default:
1199 break;
1200 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001201 if (!ret) {
1202 SafeSetError("Failed to setup RTCP mux filter.", error_desc);
1203 return false;
1204 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001205 // |rtcp_mux_filter_| can be active if |action| is CA_PRANSWER or
1206 // CA_ANSWER, but we only want to tear down the RTCP transport channel if we
1207 // received a final answer.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001208 if (rtcp_mux_filter_.IsActive()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001209 // If the RTP transport is already writable, then so are we.
zhihuangf5b251b2017-01-12 19:37:48 -08001210 if (rtp_transport_->writable()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001211 ChannelWritable_n();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001212 }
1213 }
1214
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001215 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001216}
1217
1218bool BaseChannel::AddRecvStream_w(const StreamParams& sp) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001219 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
pbos482b12e2015-11-16 10:19:58 -08001220 return media_channel()->AddRecvStream(sp);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001221}
1222
Peter Boström0c4e06b2015-10-07 12:23:21 +02001223bool BaseChannel::RemoveRecvStream_w(uint32_t ssrc) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001224 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001225 return media_channel()->RemoveRecvStream(ssrc);
1226}
1227
1228bool BaseChannel::UpdateLocalStreams_w(const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001229 ContentAction action,
1230 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001231 if (!VERIFY(action == CA_OFFER || action == CA_ANSWER ||
1232 action == CA_PRANSWER || action == CA_UPDATE))
1233 return false;
1234
1235 // If this is an update, streams only contain streams that have changed.
1236 if (action == CA_UPDATE) {
1237 for (StreamParamsVec::const_iterator it = streams.begin();
1238 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001239 const StreamParams* existing_stream =
1240 GetStreamByIds(local_streams_, it->groupid, it->id);
1241 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001242 if (media_channel()->AddSendStream(*it)) {
1243 local_streams_.push_back(*it);
1244 LOG(LS_INFO) << "Add send stream ssrc: " << it->first_ssrc();
1245 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001246 std::ostringstream desc;
1247 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1248 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001249 return false;
1250 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001251 } else if (existing_stream && !it->has_ssrcs()) {
1252 if (!media_channel()->RemoveSendStream(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001253 std::ostringstream desc;
1254 desc << "Failed to remove send stream with ssrc "
1255 << it->first_ssrc() << ".";
1256 SafeSetError(desc.str(), error_desc);
1257 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001258 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001259 RemoveStreamBySsrc(&local_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001260 } else {
1261 LOG(LS_WARNING) << "Ignore unsupported stream update";
1262 }
1263 }
1264 return true;
1265 }
1266 // Else streams are all the streams we want to send.
1267
1268 // Check for streams that have been removed.
1269 bool ret = true;
1270 for (StreamParamsVec::const_iterator it = local_streams_.begin();
1271 it != local_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001272 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001273 if (!media_channel()->RemoveSendStream(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001274 std::ostringstream desc;
1275 desc << "Failed to remove send stream with ssrc "
1276 << it->first_ssrc() << ".";
1277 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001278 ret = false;
1279 }
1280 }
1281 }
1282 // Check for new streams.
1283 for (StreamParamsVec::const_iterator it = streams.begin();
1284 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001285 if (!GetStreamBySsrc(local_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001286 if (media_channel()->AddSendStream(*it)) {
stefanc1aeaf02015-10-15 07:26:07 -07001287 LOG(LS_INFO) << "Add send stream ssrc: " << it->ssrcs[0];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001288 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001289 std::ostringstream desc;
1290 desc << "Failed to add send stream ssrc: " << it->first_ssrc();
1291 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001292 ret = false;
1293 }
1294 }
1295 }
1296 local_streams_ = streams;
1297 return ret;
1298}
1299
1300bool BaseChannel::UpdateRemoteStreams_w(
1301 const std::vector<StreamParams>& streams,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001302 ContentAction action,
1303 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001304 if (!VERIFY(action == CA_OFFER || action == CA_ANSWER ||
1305 action == CA_PRANSWER || action == CA_UPDATE))
1306 return false;
1307
1308 // If this is an update, streams only contain streams that have changed.
1309 if (action == CA_UPDATE) {
1310 for (StreamParamsVec::const_iterator it = streams.begin();
1311 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001312 const StreamParams* existing_stream =
1313 GetStreamByIds(remote_streams_, it->groupid, it->id);
1314 if (!existing_stream && it->has_ssrcs()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001315 if (AddRecvStream_w(*it)) {
1316 remote_streams_.push_back(*it);
1317 LOG(LS_INFO) << "Add remote stream ssrc: " << it->first_ssrc();
1318 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001319 std::ostringstream desc;
1320 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1321 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001322 return false;
1323 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001324 } else if (existing_stream && !it->has_ssrcs()) {
1325 if (!RemoveRecvStream_w(existing_stream->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001326 std::ostringstream desc;
1327 desc << "Failed to remove remote stream with ssrc "
1328 << it->first_ssrc() << ".";
1329 SafeSetError(desc.str(), error_desc);
1330 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001331 }
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001332 RemoveStreamBySsrc(&remote_streams_, existing_stream->first_ssrc());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001333 } else {
1334 LOG(LS_WARNING) << "Ignore unsupported stream update."
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001335 << " Stream exists? " << (existing_stream != nullptr)
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001336 << " new stream = " << it->ToString();
1337 }
1338 }
1339 return true;
1340 }
1341 // Else streams are all the streams we want to receive.
1342
1343 // Check for streams that have been removed.
1344 bool ret = true;
1345 for (StreamParamsVec::const_iterator it = remote_streams_.begin();
1346 it != remote_streams_.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001347 if (!GetStreamBySsrc(streams, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001348 if (!RemoveRecvStream_w(it->first_ssrc())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001349 std::ostringstream desc;
1350 desc << "Failed to remove remote stream with ssrc "
1351 << it->first_ssrc() << ".";
1352 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001353 ret = false;
1354 }
1355 }
1356 }
1357 // Check for new streams.
1358 for (StreamParamsVec::const_iterator it = streams.begin();
1359 it != streams.end(); ++it) {
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +00001360 if (!GetStreamBySsrc(remote_streams_, it->first_ssrc())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001361 if (AddRecvStream_w(*it)) {
1362 LOG(LS_INFO) << "Add remote ssrc: " << it->ssrcs[0];
1363 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001364 std::ostringstream desc;
1365 desc << "Failed to add remote stream ssrc: " << it->first_ssrc();
1366 SafeSetError(desc.str(), error_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001367 ret = false;
1368 }
1369 }
1370 }
1371 remote_streams_ = streams;
1372 return ret;
1373}
1374
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001375void BaseChannel::MaybeCacheRtpAbsSendTimeHeaderExtension_w(
isheriff6f8d6862016-05-26 11:24:55 -07001376 const std::vector<webrtc::RtpExtension>& extensions) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001377// Absolute Send Time extension id is used only with external auth,
1378// so do not bother searching for it and making asyncronious call to set
1379// something that is not used.
1380#if defined(ENABLE_EXTERNAL_AUTH)
isheriff6f8d6862016-05-26 11:24:55 -07001381 const webrtc::RtpExtension* send_time_extension =
1382 FindHeaderExtension(extensions, webrtc::RtpExtension::kAbsSendTimeUri);
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001383 int rtp_abs_sendtime_extn_id =
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001384 send_time_extension ? send_time_extension->id : -1;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001385 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001386 RTC_FROM_HERE, network_thread_,
1387 Bind(&BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n, this,
1388 rtp_abs_sendtime_extn_id));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001389#endif
1390}
1391
1392void BaseChannel::CacheRtpAbsSendTimeHeaderExtension_n(
1393 int rtp_abs_sendtime_extn_id) {
1394 rtp_abs_sendtime_extn_id_ = rtp_abs_sendtime_extn_id;
henrike@webrtc.orgd43aa9d2014-02-21 23:43:24 +00001395}
1396
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001397void BaseChannel::OnMessage(rtc::Message *pmsg) {
Peter Boström6f28cf02015-12-07 23:17:15 +01001398 TRACE_EVENT0("webrtc", "BaseChannel::OnMessage");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001399 switch (pmsg->message_id) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001400 case MSG_SEND_RTP_PACKET:
1401 case MSG_SEND_RTCP_PACKET: {
1402 RTC_DCHECK(network_thread_->IsCurrent());
1403 SendPacketMessageData* data =
1404 static_cast<SendPacketMessageData*>(pmsg->pdata);
1405 bool rtcp = pmsg->message_id == MSG_SEND_RTCP_PACKET;
1406 SendPacket(rtcp, &data->packet, data->options);
1407 delete data;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001408 break;
1409 }
1410 case MSG_FIRSTPACKETRECEIVED: {
1411 SignalFirstPacketReceived(this);
1412 break;
1413 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001414 }
1415}
1416
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001417void BaseChannel::FlushRtcpMessages_n() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001418 // Flush all remaining RTCP messages. This should only be called in
1419 // destructor.
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001420 RTC_DCHECK(network_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001421 rtc::MessageList rtcp_messages;
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001422 network_thread_->Clear(this, MSG_SEND_RTCP_PACKET, &rtcp_messages);
1423 for (const auto& message : rtcp_messages) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001424 network_thread_->Send(RTC_FROM_HERE, this, MSG_SEND_RTCP_PACKET,
1425 message.pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001426 }
1427}
1428
johand89ab142016-10-25 10:50:32 -07001429void BaseChannel::SignalSentPacket_n(
1430 rtc::PacketTransportInterface* /* transport */,
1431 const rtc::SentPacket& sent_packet) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001432 RTC_DCHECK(network_thread_->IsCurrent());
1433 invoker_.AsyncInvoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001434 RTC_FROM_HERE, worker_thread_,
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001435 rtc::Bind(&BaseChannel::SignalSentPacket_w, this, sent_packet));
1436}
1437
1438void BaseChannel::SignalSentPacket_w(const rtc::SentPacket& sent_packet) {
1439 RTC_DCHECK(worker_thread_->IsCurrent());
1440 SignalSentPacket(sent_packet);
1441}
1442
1443VoiceChannel::VoiceChannel(rtc::Thread* worker_thread,
1444 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001445 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001446 MediaEngineInterface* media_engine,
1447 VoiceMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001448 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001449 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001450 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001451 : BaseChannel(worker_thread,
1452 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001453 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001454 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001455 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001456 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001457 srtp_required),
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001458 media_engine_(media_engine),
deadbeefcbecd352015-09-23 11:50:27 -07001459 received_media_(false) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001460
1461VoiceChannel::~VoiceChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001462 TRACE_EVENT0("webrtc", "VoiceChannel::~VoiceChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001463 StopAudioMonitor();
1464 StopMediaMonitor();
1465 // this can't be done in the base class, since it calls a virtual
1466 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001467 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001468}
1469
zhihuangf5b251b2017-01-12 19:37:48 -08001470bool VoiceChannel::Init_w(TransportChannel* rtp_transport,
1471 TransportChannel* rtcp_transport) {
1472 return BaseChannel::Init_w(rtp_transport, rtcp_transport);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001473}
1474
Peter Boström0c4e06b2015-10-07 12:23:21 +02001475bool VoiceChannel::SetAudioSend(uint32_t ssrc,
solenbergdfc8f4f2015-10-01 02:31:10 -07001476 bool enable,
solenberg1dd98f32015-09-10 01:57:14 -07001477 const AudioOptions* options,
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001478 AudioSource* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001479 return InvokeOnWorker(RTC_FROM_HERE,
1480 Bind(&VoiceMediaChannel::SetAudioSend, media_channel(),
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001481 ssrc, enable, options, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001482}
1483
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001484// TODO(juberti): Handle early media the right way. We should get an explicit
1485// ringing message telling us to start playing local ringback, which we cancel
1486// if any early media actually arrives. For now, we do the opposite, which is
1487// to wait 1 second for early media, and start playing local ringback if none
1488// arrives.
1489void VoiceChannel::SetEarlyMedia(bool enable) {
1490 if (enable) {
1491 // Start the early media timeout
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001492 worker_thread()->PostDelayed(RTC_FROM_HERE, kEarlyMediaTimeout, this,
1493 MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001494 } else {
1495 // Stop the timeout if currently going.
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001496 worker_thread()->Clear(this, MSG_EARLYMEDIATIMEOUT);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001497 }
1498}
1499
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001500bool VoiceChannel::CanInsertDtmf() {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001501 return InvokeOnWorker(
1502 RTC_FROM_HERE, Bind(&VoiceMediaChannel::CanInsertDtmf, media_channel()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001503}
1504
Peter Boström0c4e06b2015-10-07 12:23:21 +02001505bool VoiceChannel::InsertDtmf(uint32_t ssrc,
1506 int event_code,
solenberg1d63dd02015-12-02 12:35:09 -08001507 int duration) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001508 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceChannel::InsertDtmf_w, this,
1509 ssrc, event_code, duration));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001510}
1511
solenberg4bac9c52015-10-09 02:32:53 -07001512bool VoiceChannel::SetOutputVolume(uint32_t ssrc, double volume) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001513 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::SetOutputVolume,
1514 media_channel(), ssrc, volume));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001515}
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00001516
Tommif888bb52015-12-12 01:37:01 +01001517void VoiceChannel::SetRawAudioSink(
1518 uint32_t ssrc,
kwiberg31022942016-03-11 14:18:21 -08001519 std::unique_ptr<webrtc::AudioSinkInterface> sink) {
1520 // We need to work around Bind's lack of support for unique_ptr and ownership
deadbeef2d110be2016-01-13 12:00:26 -08001521 // passing. So we invoke to our own little routine that gets a pointer to
1522 // our local variable. This is OK since we're synchronously invoking.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001523 InvokeOnWorker(RTC_FROM_HERE,
1524 Bind(&SetRawAudioSink_w, media_channel(), ssrc, &sink));
Tommif888bb52015-12-12 01:37:01 +01001525}
1526
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001527webrtc::RtpParameters VoiceChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001528 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001529 RTC_FROM_HERE, Bind(&VoiceChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001530}
1531
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001532webrtc::RtpParameters VoiceChannel::GetRtpSendParameters_w(
1533 uint32_t ssrc) const {
1534 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001535}
1536
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001537bool VoiceChannel::SetRtpSendParameters(
1538 uint32_t ssrc,
1539 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001540 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001541 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001542 Bind(&VoiceChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001543}
1544
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001545bool VoiceChannel::SetRtpSendParameters_w(uint32_t ssrc,
1546 webrtc::RtpParameters parameters) {
1547 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1548}
1549
1550webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters(
1551 uint32_t ssrc) const {
1552 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001553 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001554 Bind(&VoiceChannel::GetRtpReceiveParameters_w, this, ssrc));
1555}
1556
1557webrtc::RtpParameters VoiceChannel::GetRtpReceiveParameters_w(
1558 uint32_t ssrc) const {
1559 return media_channel()->GetRtpReceiveParameters(ssrc);
1560}
1561
1562bool VoiceChannel::SetRtpReceiveParameters(
1563 uint32_t ssrc,
1564 const webrtc::RtpParameters& parameters) {
1565 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001566 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001567 Bind(&VoiceChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1568}
1569
1570bool VoiceChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1571 webrtc::RtpParameters parameters) {
1572 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001573}
1574
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001575bool VoiceChannel::GetStats(VoiceMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001576 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VoiceMediaChannel::GetStats,
1577 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001578}
1579
1580void VoiceChannel::StartMediaMonitor(int cms) {
1581 media_monitor_.reset(new VoiceMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001582 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001583 media_monitor_->SignalUpdate.connect(
1584 this, &VoiceChannel::OnMediaMonitorUpdate);
1585 media_monitor_->Start(cms);
1586}
1587
1588void VoiceChannel::StopMediaMonitor() {
1589 if (media_monitor_) {
1590 media_monitor_->Stop();
1591 media_monitor_->SignalUpdate.disconnect(this);
1592 media_monitor_.reset();
1593 }
1594}
1595
1596void VoiceChannel::StartAudioMonitor(int cms) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001597 audio_monitor_.reset(new AudioMonitor(this, rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001598 audio_monitor_
1599 ->SignalUpdate.connect(this, &VoiceChannel::OnAudioMonitorUpdate);
1600 audio_monitor_->Start(cms);
1601}
1602
1603void VoiceChannel::StopAudioMonitor() {
1604 if (audio_monitor_) {
1605 audio_monitor_->Stop();
1606 audio_monitor_.reset();
1607 }
1608}
1609
1610bool VoiceChannel::IsAudioMonitorRunning() const {
1611 return (audio_monitor_.get() != NULL);
1612}
1613
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001614int VoiceChannel::GetInputLevel_w() {
Fredrik Solenberg0c022642015-08-05 12:25:22 +02001615 return media_engine_->GetInputLevel();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001616}
1617
1618int VoiceChannel::GetOutputLevel_w() {
1619 return media_channel()->GetOutputLevel();
1620}
1621
1622void VoiceChannel::GetActiveStreams_w(AudioInfo::StreamList* actives) {
1623 media_channel()->GetActiveStreams(actives);
1624}
1625
johand89ab142016-10-25 10:50:32 -07001626void VoiceChannel::OnPacketRead(rtc::PacketTransportInterface* transport,
1627 const char* data,
1628 size_t len,
1629 const rtc::PacketTime& packet_time,
wu@webrtc.orga9890802013-12-13 00:21:03 +00001630 int flags) {
johand89ab142016-10-25 10:50:32 -07001631 BaseChannel::OnPacketRead(transport, data, len, packet_time, flags);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001632 // Set a flag when we've received an RTP packet. If we're waiting for early
1633 // media, this will disable the timeout.
johand89ab142016-10-25 10:50:32 -07001634 if (!received_media_ && !PacketIsRtcp(transport, data, len)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001635 received_media_ = true;
1636 }
1637}
1638
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001639void BaseChannel::UpdateMediaSendRecvState() {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001640 RTC_DCHECK(network_thread_->IsCurrent());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001641 invoker_.AsyncInvoke<void>(
1642 RTC_FROM_HERE, worker_thread_,
1643 Bind(&BaseChannel::UpdateMediaSendRecvState_w, this));
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001644}
1645
michaelt79e05882016-11-08 02:50:09 -08001646int BaseChannel::GetTransportOverheadPerPacket() const {
1647 RTC_DCHECK(network_thread_->IsCurrent());
1648
1649 if (!selected_candidate_pair_)
1650 return 0;
1651
1652 int transport_overhead_per_packet = 0;
1653
1654 constexpr int kIpv4Overhaed = 20;
1655 constexpr int kIpv6Overhaed = 40;
1656 transport_overhead_per_packet +=
1657 selected_candidate_pair_->local_candidate().address().family() == AF_INET
1658 ? kIpv4Overhaed
1659 : kIpv6Overhaed;
1660
1661 constexpr int kUdpOverhaed = 8;
1662 constexpr int kTcpOverhaed = 20;
1663 transport_overhead_per_packet +=
1664 selected_candidate_pair_->local_candidate().protocol() ==
1665 TCP_PROTOCOL_NAME
1666 ? kTcpOverhaed
1667 : kUdpOverhaed;
1668
1669 if (secure()) {
1670 int srtp_overhead = 0;
1671 if (srtp_filter_.GetSrtpOverhead(&srtp_overhead))
1672 transport_overhead_per_packet += srtp_overhead;
1673 }
1674
1675 return transport_overhead_per_packet;
1676}
1677
1678void BaseChannel::UpdateTransportOverhead() {
1679 int transport_overhead_per_packet = GetTransportOverheadPerPacket();
1680 if (transport_overhead_per_packet)
1681 invoker_.AsyncInvoke<void>(
1682 RTC_FROM_HERE, worker_thread_,
1683 Bind(&MediaChannel::OnTransportOverheadChanged, media_channel_,
1684 transport_overhead_per_packet));
1685}
1686
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001687void VoiceChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001688 // Render incoming data if we're the active call, and we have the local
1689 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001690 bool recv = IsReadyToReceiveMedia_w();
solenberg5b14b422015-10-01 04:10:31 -07001691 media_channel()->SetPlayout(recv);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001692
1693 // Send outgoing data if we're the active call, we have the remote content,
1694 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001695 bool send = IsReadyToSendMedia_w();
Taylor Brandstetter1a018dc2016-03-08 12:37:39 -08001696 media_channel()->SetSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001697
1698 LOG(LS_INFO) << "Changing voice state, recv=" << recv << " send=" << send;
1699}
1700
1701const ContentInfo* VoiceChannel::GetFirstContent(
1702 const SessionDescription* sdesc) {
1703 return GetFirstAudioContent(sdesc);
1704}
1705
1706bool VoiceChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001707 ContentAction action,
1708 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001709 TRACE_EVENT0("webrtc", "VoiceChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001710 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001711 LOG(LS_INFO) << "Setting local voice description";
1712
1713 const AudioContentDescription* audio =
1714 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001715 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001716 if (!audio) {
1717 SafeSetError("Can't find audio content in local description.", error_desc);
1718 return false;
1719 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001720
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001721 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001722 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001723 }
1724
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001725 AudioRecvParameters recv_params = last_recv_params_;
1726 RtpParametersFromMediaDescription(audio, &recv_params);
1727 if (!media_channel()->SetRecvParameters(recv_params)) {
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001728 SafeSetError("Failed to set local audio description recv parameters.",
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001729 error_desc);
1730 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001731 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001732 for (const AudioCodec& codec : audio->codecs()) {
1733 bundle_filter()->AddPayloadType(codec.id);
1734 }
1735 last_recv_params_ = recv_params;
1736
1737 // TODO(pthatcher): Move local streams into AudioSendParameters, and
1738 // only give it to the media channel once we have a remote
1739 // description too (without a remote description, we won't be able
1740 // to send them anyway).
1741 if (!UpdateLocalStreams_w(audio->streams(), action, error_desc)) {
1742 SafeSetError("Failed to set local audio description streams.", error_desc);
1743 return false;
1744 }
1745
1746 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001747 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001748 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001749}
1750
1751bool VoiceChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001752 ContentAction action,
1753 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001754 TRACE_EVENT0("webrtc", "VoiceChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001755 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001756 LOG(LS_INFO) << "Setting remote voice description";
1757
1758 const AudioContentDescription* audio =
1759 static_cast<const AudioContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001760 RTC_DCHECK(audio != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001761 if (!audio) {
1762 SafeSetError("Can't find audio content in remote description.", error_desc);
1763 return false;
1764 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001765
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001766 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001767 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001768 }
1769
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001770 AudioSendParameters send_params = last_send_params_;
1771 RtpSendParametersFromMediaDescription(audio, &send_params);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001772 if (audio->agc_minus_10db()) {
Karl Wibergbe579832015-11-10 22:34:18 +01001773 send_params.options.adjust_agc_delta = rtc::Optional<int>(kAgcMinus10db);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001774 }
skvladdc1c62c2016-03-16 19:07:43 -07001775
1776 bool parameters_applied = media_channel()->SetSendParameters(send_params);
1777 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001778 SafeSetError("Failed to set remote audio description send parameters.",
1779 error_desc);
1780 return false;
1781 }
1782 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001783
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001784 // TODO(pthatcher): Move remote streams into AudioRecvParameters,
1785 // and only give it to the media channel once we have a local
1786 // description too (without a local description, we won't be able to
1787 // recv them anyway).
1788 if (!UpdateRemoteStreams_w(audio->streams(), action, error_desc)) {
1789 SafeSetError("Failed to set remote audio description streams.", error_desc);
1790 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001791 }
1792
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001793 if (audio->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001794 MaybeCacheRtpAbsSendTimeHeaderExtension_w(audio->rtp_header_extensions());
Peter Thatcherbfab5cb2015-08-20 17:40:24 -07001795 }
1796
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001797 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001798 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07001799 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001800}
1801
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001802void VoiceChannel::HandleEarlyMediaTimeout() {
1803 // This occurs on the main thread, not the worker thread.
1804 if (!received_media_) {
1805 LOG(LS_INFO) << "No early media received before timeout";
1806 SignalEarlyMediaTimeout(this);
1807 }
1808}
1809
Peter Boström0c4e06b2015-10-07 12:23:21 +02001810bool VoiceChannel::InsertDtmf_w(uint32_t ssrc,
1811 int event,
solenberg1d63dd02015-12-02 12:35:09 -08001812 int duration) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001813 if (!enabled()) {
1814 return false;
1815 }
solenberg1d63dd02015-12-02 12:35:09 -08001816 return media_channel()->InsertDtmf(ssrc, event, duration);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001817}
1818
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001819void VoiceChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001820 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001821 case MSG_EARLYMEDIATIMEOUT:
1822 HandleEarlyMediaTimeout();
1823 break;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001824 case MSG_CHANNEL_ERROR: {
1825 VoiceChannelErrorMessageData* data =
1826 static_cast<VoiceChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001827 delete data;
1828 break;
1829 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001830 default:
1831 BaseChannel::OnMessage(pmsg);
1832 break;
1833 }
1834}
1835
1836void VoiceChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00001837 ConnectionMonitor* monitor, const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001838 SignalConnectionMonitor(this, infos);
1839}
1840
1841void VoiceChannel::OnMediaMonitorUpdate(
1842 VoiceMediaChannel* media_channel, const VoiceMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001843 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001844 SignalMediaMonitor(this, info);
1845}
1846
1847void VoiceChannel::OnAudioMonitorUpdate(AudioMonitor* monitor,
1848 const AudioInfo& info) {
1849 SignalAudioMonitor(this, info);
1850}
1851
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001852void VoiceChannel::GetSrtpCryptoSuites_n(
1853 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07001854 GetSupportedAudioCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001855}
1856
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001857VideoChannel::VideoChannel(rtc::Thread* worker_thread,
1858 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001859 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001860 VideoMediaChannel* media_channel,
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001861 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08001862 bool rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001863 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001864 : BaseChannel(worker_thread,
1865 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08001866 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07001867 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07001868 content_name,
deadbeefac22f702017-01-12 21:59:29 -08001869 rtcp_mux_required,
deadbeef7af91dd2016-12-13 11:29:11 -08001870 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001871
zhihuangf5b251b2017-01-12 19:37:48 -08001872bool VideoChannel::Init_w(TransportChannel* rtp_transport,
1873 TransportChannel* rtcp_transport) {
1874 return BaseChannel::Init_w(rtp_transport, rtcp_transport);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001875}
1876
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001877VideoChannel::~VideoChannel() {
Peter Boströmca8b4042016-03-08 14:24:13 -08001878 TRACE_EVENT0("webrtc", "VideoChannel::~VideoChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001879 StopMediaMonitor();
1880 // this can't be done in the base class, since it calls a virtual
1881 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00001882
1883 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001884}
1885
nisse08582ff2016-02-04 01:24:52 -08001886bool VideoChannel::SetSink(uint32_t ssrc,
nisseacd935b2016-11-11 03:55:13 -08001887 rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) {
nisse08582ff2016-02-04 01:24:52 -08001888 worker_thread()->Invoke<void>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001889 RTC_FROM_HERE,
nisse08582ff2016-02-04 01:24:52 -08001890 Bind(&VideoMediaChannel::SetSink, media_channel(), ssrc, sink));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001891 return true;
1892}
1893
deadbeef5a4a75a2016-06-02 16:23:38 -07001894bool VideoChannel::SetVideoSend(
nisse2ded9b12016-04-08 02:23:55 -07001895 uint32_t ssrc,
deadbeef5a4a75a2016-06-02 16:23:38 -07001896 bool mute,
1897 const VideoOptions* options,
nisseacd935b2016-11-11 03:55:13 -08001898 rtc::VideoSourceInterface<webrtc::VideoFrame>* source) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001899 return InvokeOnWorker(RTC_FROM_HERE,
1900 Bind(&VideoMediaChannel::SetVideoSend, media_channel(),
deadbeef5a4a75a2016-06-02 16:23:38 -07001901 ssrc, mute, options, source));
solenberg1dd98f32015-09-10 01:57:14 -07001902}
1903
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001904webrtc::RtpParameters VideoChannel::GetRtpSendParameters(uint32_t ssrc) const {
skvladdc1c62c2016-03-16 19:07:43 -07001905 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001906 RTC_FROM_HERE, Bind(&VideoChannel::GetRtpSendParameters_w, this, ssrc));
skvladdc1c62c2016-03-16 19:07:43 -07001907}
1908
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001909webrtc::RtpParameters VideoChannel::GetRtpSendParameters_w(
1910 uint32_t ssrc) const {
1911 return media_channel()->GetRtpSendParameters(ssrc);
skvladdc1c62c2016-03-16 19:07:43 -07001912}
1913
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001914bool VideoChannel::SetRtpSendParameters(
1915 uint32_t ssrc,
1916 const webrtc::RtpParameters& parameters) {
skvladdc1c62c2016-03-16 19:07:43 -07001917 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001918 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001919 Bind(&VideoChannel::SetRtpSendParameters_w, this, ssrc, parameters));
skvladdc1c62c2016-03-16 19:07:43 -07001920}
1921
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001922bool VideoChannel::SetRtpSendParameters_w(uint32_t ssrc,
1923 webrtc::RtpParameters parameters) {
1924 return media_channel()->SetRtpSendParameters(ssrc, parameters);
1925}
1926
1927webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters(
1928 uint32_t ssrc) const {
1929 return worker_thread()->Invoke<webrtc::RtpParameters>(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001930 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001931 Bind(&VideoChannel::GetRtpReceiveParameters_w, this, ssrc));
1932}
1933
1934webrtc::RtpParameters VideoChannel::GetRtpReceiveParameters_w(
1935 uint32_t ssrc) const {
1936 return media_channel()->GetRtpReceiveParameters(ssrc);
1937}
1938
1939bool VideoChannel::SetRtpReceiveParameters(
1940 uint32_t ssrc,
1941 const webrtc::RtpParameters& parameters) {
1942 return InvokeOnWorker(
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001943 RTC_FROM_HERE,
Taylor Brandstetterdb0cd9e2016-05-16 11:40:30 -07001944 Bind(&VideoChannel::SetRtpReceiveParameters_w, this, ssrc, parameters));
1945}
1946
1947bool VideoChannel::SetRtpReceiveParameters_w(uint32_t ssrc,
1948 webrtc::RtpParameters parameters) {
1949 return media_channel()->SetRtpReceiveParameters(ssrc, parameters);
skvladdc1c62c2016-03-16 19:07:43 -07001950}
Danil Chapovalov33b01f22016-05-11 19:55:27 +02001951
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001952void VideoChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001953 // Send outgoing data if we're the active call, we have the remote content,
1954 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001955 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001956 if (!media_channel()->SetSend(send)) {
1957 LOG(LS_ERROR) << "Failed to SetSend on video channel";
1958 // TODO(gangji): Report error back to server.
1959 }
1960
Peter Boström34fbfff2015-09-24 19:20:30 +02001961 LOG(LS_INFO) << "Changing video state, send=" << send;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001962}
1963
pbos@webrtc.org058b1f12015-03-04 08:54:32 +00001964bool VideoChannel::GetStats(VideoMediaInfo* stats) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07001965 return InvokeOnWorker(RTC_FROM_HERE, Bind(&VideoMediaChannel::GetStats,
1966 media_channel(), stats));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001967}
1968
1969void VideoChannel::StartMediaMonitor(int cms) {
1970 media_monitor_.reset(new VideoMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00001971 rtc::Thread::Current()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001972 media_monitor_->SignalUpdate.connect(
1973 this, &VideoChannel::OnMediaMonitorUpdate);
1974 media_monitor_->Start(cms);
1975}
1976
1977void VideoChannel::StopMediaMonitor() {
1978 if (media_monitor_) {
1979 media_monitor_->Stop();
1980 media_monitor_.reset();
1981 }
1982}
1983
1984const ContentInfo* VideoChannel::GetFirstContent(
1985 const SessionDescription* sdesc) {
1986 return GetFirstVideoContent(sdesc);
1987}
1988
1989bool VideoChannel::SetLocalContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001990 ContentAction action,
1991 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01001992 TRACE_EVENT0("webrtc", "VideoChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001993 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001994 LOG(LS_INFO) << "Setting local video description";
1995
1996 const VideoContentDescription* video =
1997 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07001998 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001999 if (!video) {
2000 SafeSetError("Can't find video content in local description.", error_desc);
2001 return false;
2002 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002003
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002004 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002005 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002006 }
2007
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002008 VideoRecvParameters recv_params = last_recv_params_;
2009 RtpParametersFromMediaDescription(video, &recv_params);
2010 if (!media_channel()->SetRecvParameters(recv_params)) {
2011 SafeSetError("Failed to set local video description recv parameters.",
2012 error_desc);
2013 return false;
2014 }
2015 for (const VideoCodec& codec : video->codecs()) {
2016 bundle_filter()->AddPayloadType(codec.id);
2017 }
2018 last_recv_params_ = recv_params;
2019
2020 // TODO(pthatcher): Move local streams into VideoSendParameters, and
2021 // only give it to the media channel once we have a remote
2022 // description too (without a remote description, we won't be able
2023 // to send them anyway).
2024 if (!UpdateLocalStreams_w(video->streams(), action, error_desc)) {
2025 SafeSetError("Failed to set local video description streams.", error_desc);
2026 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002027 }
2028
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002029 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002030 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002031 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002032}
2033
2034bool VideoChannel::SetRemoteContent_w(const MediaContentDescription* content,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002035 ContentAction action,
2036 std::string* error_desc) {
Peter Boström9f45a452015-12-08 13:25:57 +01002037 TRACE_EVENT0("webrtc", "VideoChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002038 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002039 LOG(LS_INFO) << "Setting remote video description";
2040
2041 const VideoContentDescription* video =
2042 static_cast<const VideoContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002043 RTC_DCHECK(video != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002044 if (!video) {
2045 SafeSetError("Can't find video content in remote description.", error_desc);
2046 return false;
2047 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002048
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002049 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002050 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002051 }
2052
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002053 VideoSendParameters send_params = last_send_params_;
2054 RtpSendParametersFromMediaDescription(video, &send_params);
2055 if (video->conference_mode()) {
nisse4b4dc862016-02-17 05:25:36 -08002056 send_params.conference_mode = true;
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002057 }
skvladdc1c62c2016-03-16 19:07:43 -07002058
2059 bool parameters_applied = media_channel()->SetSendParameters(send_params);
2060
2061 if (!parameters_applied) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002062 SafeSetError("Failed to set remote video description send parameters.",
2063 error_desc);
2064 return false;
2065 }
2066 last_send_params_ = send_params;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002067
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002068 // TODO(pthatcher): Move remote streams into VideoRecvParameters,
2069 // and only give it to the media channel once we have a local
2070 // description too (without a local description, we won't be able to
2071 // recv them anyway).
2072 if (!UpdateRemoteStreams_w(video->streams(), action, error_desc)) {
2073 SafeSetError("Failed to set remote video description streams.", error_desc);
2074 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002075 }
2076
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002077 if (video->rtp_header_extensions_set()) {
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002078 MaybeCacheRtpAbsSendTimeHeaderExtension_w(video->rtp_header_extensions());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002079 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002080
2081 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002082 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002083 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002084}
2085
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002086void VideoChannel::OnMessage(rtc::Message *pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002087 switch (pmsg->message_id) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002088 case MSG_CHANNEL_ERROR: {
2089 const VideoChannelErrorMessageData* data =
2090 static_cast<VideoChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002091 delete data;
2092 break;
2093 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002094 default:
2095 BaseChannel::OnMessage(pmsg);
2096 break;
2097 }
2098}
2099
2100void VideoChannel::OnConnectionMonitorUpdate(
pthatcher@webrtc.orgb4aac132015-03-13 18:25:21 +00002101 ConnectionMonitor* monitor, const std::vector<ConnectionInfo> &infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002102 SignalConnectionMonitor(this, infos);
2103}
2104
2105// TODO(pthatcher): Look into removing duplicate code between
2106// audio, video, and data, perhaps by using templates.
2107void VideoChannel::OnMediaMonitorUpdate(
2108 VideoMediaChannel* media_channel, const VideoMediaInfo &info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002109 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002110 SignalMediaMonitor(this, info);
2111}
2112
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002113void VideoChannel::GetSrtpCryptoSuites_n(
2114 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002115 GetSupportedVideoCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002116}
2117
deadbeef953c2ce2017-01-09 14:53:41 -08002118RtpDataChannel::RtpDataChannel(rtc::Thread* worker_thread,
2119 rtc::Thread* network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002120 rtc::Thread* signaling_thread,
deadbeef953c2ce2017-01-09 14:53:41 -08002121 DataMediaChannel* media_channel,
deadbeef953c2ce2017-01-09 14:53:41 -08002122 const std::string& content_name,
deadbeefac22f702017-01-12 21:59:29 -08002123 bool rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002124 bool srtp_required)
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002125 : BaseChannel(worker_thread,
2126 network_thread,
zhihuangf5b251b2017-01-12 19:37:48 -08002127 signaling_thread,
deadbeefcbecd352015-09-23 11:50:27 -07002128 media_channel,
deadbeefcbecd352015-09-23 11:50:27 -07002129 content_name,
deadbeefac22f702017-01-12 21:59:29 -08002130 rtcp_mux_required,
deadbeef953c2ce2017-01-09 14:53:41 -08002131 srtp_required) {}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002132
deadbeef953c2ce2017-01-09 14:53:41 -08002133RtpDataChannel::~RtpDataChannel() {
2134 TRACE_EVENT0("webrtc", "RtpDataChannel::~RtpDataChannel");
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002135 StopMediaMonitor();
2136 // this can't be done in the base class, since it calls a virtual
2137 DisableMedia_w();
wu@webrtc.org78187522013-10-07 23:32:02 +00002138
2139 Deinit();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002140}
2141
zhihuangf5b251b2017-01-12 19:37:48 -08002142bool RtpDataChannel::Init_w(TransportChannel* rtp_transport,
2143 TransportChannel* rtcp_transport) {
2144 if (!BaseChannel::Init_w(rtp_transport, rtcp_transport)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002145 return false;
2146 }
deadbeef953c2ce2017-01-09 14:53:41 -08002147 media_channel()->SignalDataReceived.connect(this,
2148 &RtpDataChannel::OnDataReceived);
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002149 media_channel()->SignalReadyToSend.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08002150 this, &RtpDataChannel::OnDataChannelReadyToSend);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002151 return true;
2152}
2153
deadbeef953c2ce2017-01-09 14:53:41 -08002154bool RtpDataChannel::SendData(const SendDataParams& params,
2155 const rtc::CopyOnWriteBuffer& payload,
2156 SendDataResult* result) {
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002157 return InvokeOnWorker(
2158 RTC_FROM_HERE, Bind(&DataMediaChannel::SendData, media_channel(), params,
2159 payload, result));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002160}
2161
deadbeef953c2ce2017-01-09 14:53:41 -08002162const ContentInfo* RtpDataChannel::GetFirstContent(
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002163 const SessionDescription* sdesc) {
2164 return GetFirstDataContent(sdesc);
2165}
2166
deadbeef953c2ce2017-01-09 14:53:41 -08002167bool RtpDataChannel::CheckDataChannelTypeFromContent(
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002168 const DataContentDescription* content,
2169 std::string* error_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002170 bool is_sctp = ((content->protocol() == kMediaProtocolSctp) ||
2171 (content->protocol() == kMediaProtocolDtlsSctp));
deadbeef953c2ce2017-01-09 14:53:41 -08002172 // It's been set before, but doesn't match. That's bad.
2173 if (is_sctp) {
2174 SafeSetError("Data channel type mismatch. Expected RTP, got SCTP.",
2175 error_desc);
2176 return false;
2177 }
2178 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002179}
2180
deadbeef953c2ce2017-01-09 14:53:41 -08002181bool RtpDataChannel::SetLocalContent_w(const MediaContentDescription* content,
2182 ContentAction action,
2183 std::string* error_desc) {
2184 TRACE_EVENT0("webrtc", "RtpDataChannel::SetLocalContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002185 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002186 LOG(LS_INFO) << "Setting local data description";
2187
2188 const DataContentDescription* data =
2189 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002190 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002191 if (!data) {
2192 SafeSetError("Can't find data content in local description.", error_desc);
2193 return false;
2194 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002195
deadbeef953c2ce2017-01-09 14:53:41 -08002196 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002197 return false;
2198 }
2199
deadbeef953c2ce2017-01-09 14:53:41 -08002200 if (!SetRtpTransportParameters(content, action, CS_LOCAL, error_desc)) {
2201 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002202 }
2203
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002204 DataRecvParameters recv_params = last_recv_params_;
2205 RtpParametersFromMediaDescription(data, &recv_params);
2206 if (!media_channel()->SetRecvParameters(recv_params)) {
2207 SafeSetError("Failed to set remote data description recv parameters.",
2208 error_desc);
2209 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002210 }
deadbeef953c2ce2017-01-09 14:53:41 -08002211 for (const DataCodec& codec : data->codecs()) {
2212 bundle_filter()->AddPayloadType(codec.id);
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002213 }
2214 last_recv_params_ = recv_params;
2215
2216 // TODO(pthatcher): Move local streams into DataSendParameters, and
2217 // only give it to the media channel once we have a remote
2218 // description too (without a remote description, we won't be able
2219 // to send them anyway).
2220 if (!UpdateLocalStreams_w(data->streams(), action, error_desc)) {
2221 SafeSetError("Failed to set local data description streams.", error_desc);
2222 return false;
2223 }
2224
2225 set_local_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002226 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002227 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002228}
2229
deadbeef953c2ce2017-01-09 14:53:41 -08002230bool RtpDataChannel::SetRemoteContent_w(const MediaContentDescription* content,
2231 ContentAction action,
2232 std::string* error_desc) {
2233 TRACE_EVENT0("webrtc", "RtpDataChannel::SetRemoteContent_w");
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002234 RTC_DCHECK(worker_thread() == rtc::Thread::Current());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002235
2236 const DataContentDescription* data =
2237 static_cast<const DataContentDescription*>(content);
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002238 RTC_DCHECK(data != NULL);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002239 if (!data) {
2240 SafeSetError("Can't find data content in remote description.", error_desc);
2241 return false;
2242 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002243
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002244 // If the remote data doesn't have codecs and isn't an update, it
2245 // must be empty, so ignore it.
2246 if (!data->has_codecs() && action != CA_UPDATE) {
2247 return true;
2248 }
2249
deadbeef953c2ce2017-01-09 14:53:41 -08002250 if (!CheckDataChannelTypeFromContent(data, error_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002251 return false;
2252 }
2253
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002254 LOG(LS_INFO) << "Setting remote data description";
deadbeef953c2ce2017-01-09 14:53:41 -08002255 if (!SetRtpTransportParameters(content, action, CS_REMOTE, error_desc)) {
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002256 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002257 }
2258
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002259 DataSendParameters send_params = last_send_params_;
2260 RtpSendParametersFromMediaDescription<DataCodec>(data, &send_params);
2261 if (!media_channel()->SetSendParameters(send_params)) {
2262 SafeSetError("Failed to set remote data description send parameters.",
2263 error_desc);
2264 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002265 }
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002266 last_send_params_ = send_params;
2267
2268 // TODO(pthatcher): Move remote streams into DataRecvParameters,
2269 // and only give it to the media channel once we have a local
2270 // description too (without a local description, we won't be able to
2271 // recv them anyway).
2272 if (!UpdateRemoteStreams_w(data->streams(), action, error_desc)) {
2273 SafeSetError("Failed to set remote data description streams.",
2274 error_desc);
2275 return false;
2276 }
2277
2278 set_remote_content_direction(content->direction());
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002279 UpdateMediaSendRecvState_w();
Peter Thatcherc2ee2c82015-08-07 16:05:34 -07002280 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002281}
2282
deadbeef953c2ce2017-01-09 14:53:41 -08002283void RtpDataChannel::UpdateMediaSendRecvState_w() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002284 // Render incoming data if we're the active call, and we have the local
2285 // content. We receive data on the default channel and multiplexed streams.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002286 bool recv = IsReadyToReceiveMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002287 if (!media_channel()->SetReceive(recv)) {
2288 LOG(LS_ERROR) << "Failed to SetReceive on data channel";
2289 }
2290
2291 // Send outgoing data if we're the active call, we have the remote content,
2292 // and we have had some form of connectivity.
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002293 bool send = IsReadyToSendMedia_w();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002294 if (!media_channel()->SetSend(send)) {
2295 LOG(LS_ERROR) << "Failed to SetSend on data channel";
2296 }
2297
sergeyu@chromium.org9cf037b2014-02-07 19:03:26 +00002298 // Trigger SignalReadyToSendData asynchronously.
2299 OnDataChannelReadyToSend(send);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002300
2301 LOG(LS_INFO) << "Changing data state, recv=" << recv << " send=" << send;
2302}
2303
deadbeef953c2ce2017-01-09 14:53:41 -08002304void RtpDataChannel::OnMessage(rtc::Message* pmsg) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002305 switch (pmsg->message_id) {
2306 case MSG_READYTOSENDDATA: {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002307 DataChannelReadyToSendMessageData* data =
2308 static_cast<DataChannelReadyToSendMessageData*>(pmsg->pdata);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00002309 ready_to_send_data_ = data->data();
2310 SignalReadyToSendData(ready_to_send_data_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002311 delete data;
2312 break;
2313 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002314 case MSG_DATARECEIVED: {
2315 DataReceivedMessageData* data =
2316 static_cast<DataReceivedMessageData*>(pmsg->pdata);
deadbeef953c2ce2017-01-09 14:53:41 -08002317 SignalDataReceived(data->params, data->payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002318 delete data;
2319 break;
2320 }
2321 case MSG_CHANNEL_ERROR: {
2322 const DataChannelErrorMessageData* data =
2323 static_cast<DataChannelErrorMessageData*>(pmsg->pdata);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002324 delete data;
2325 break;
2326 }
2327 default:
2328 BaseChannel::OnMessage(pmsg);
2329 break;
2330 }
2331}
2332
deadbeef953c2ce2017-01-09 14:53:41 -08002333void RtpDataChannel::OnConnectionMonitorUpdate(
2334 ConnectionMonitor* monitor,
2335 const std::vector<ConnectionInfo>& infos) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002336 SignalConnectionMonitor(this, infos);
2337}
2338
deadbeef953c2ce2017-01-09 14:53:41 -08002339void RtpDataChannel::StartMediaMonitor(int cms) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002340 media_monitor_.reset(new DataMediaMonitor(media_channel(), worker_thread(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +00002341 rtc::Thread::Current()));
deadbeef953c2ce2017-01-09 14:53:41 -08002342 media_monitor_->SignalUpdate.connect(this,
2343 &RtpDataChannel::OnMediaMonitorUpdate);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002344 media_monitor_->Start(cms);
2345}
2346
deadbeef953c2ce2017-01-09 14:53:41 -08002347void RtpDataChannel::StopMediaMonitor() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002348 if (media_monitor_) {
2349 media_monitor_->Stop();
2350 media_monitor_->SignalUpdate.disconnect(this);
2351 media_monitor_.reset();
2352 }
2353}
2354
deadbeef953c2ce2017-01-09 14:53:41 -08002355void RtpDataChannel::OnMediaMonitorUpdate(DataMediaChannel* media_channel,
2356 const DataMediaInfo& info) {
Taylor Brandstetterbad33bf2016-08-25 13:31:14 -07002357 RTC_DCHECK(media_channel == this->media_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002358 SignalMediaMonitor(this, info);
2359}
2360
deadbeef953c2ce2017-01-09 14:53:41 -08002361void RtpDataChannel::OnDataReceived(const ReceiveDataParams& params,
2362 const char* data,
2363 size_t len) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002364 DataReceivedMessageData* msg = new DataReceivedMessageData(
2365 params, data, len);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002366 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_DATARECEIVED, msg);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002367}
2368
deadbeef953c2ce2017-01-09 14:53:41 -08002369void RtpDataChannel::OnDataChannelError(uint32_t ssrc,
2370 DataMediaChannel::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002371 DataChannelErrorMessageData* data = new DataChannelErrorMessageData(
2372 ssrc, err);
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002373 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_CHANNEL_ERROR, data);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002374}
2375
deadbeef953c2ce2017-01-09 14:53:41 -08002376void RtpDataChannel::OnDataChannelReadyToSend(bool writable) {
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002377 // This is usded for congestion control to indicate that the stream is ready
2378 // to send by the MediaChannel, as opposed to OnReadyToSend, which indicates
2379 // that the transport channel is ready.
Taylor Brandstetter5d97a9a2016-06-10 14:17:27 -07002380 signaling_thread()->Post(RTC_FROM_HERE, this, MSG_READYTOSENDDATA,
wu@webrtc.orgd64719d2013-08-01 00:00:07 +00002381 new DataChannelReadyToSendMessageData(writable));
2382}
2383
deadbeef953c2ce2017-01-09 14:53:41 -08002384void RtpDataChannel::GetSrtpCryptoSuites_n(
2385 std::vector<int>* crypto_suites) const {
jbauchcb560652016-08-04 05:20:32 -07002386 GetSupportedDataCryptoSuites(crypto_options(), crypto_suites);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002387}
2388
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002389} // namespace cricket