blob: f556204c6c98ce7bb42b0ed6097c29ec8e11a6d5 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellanderb24317b2016-02-10 07:54:43 -08002 * Copyright 2012 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellanderb24317b2016-02-10 07:54:43 -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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "pc/webrtcsession.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000012
pbos@webrtc.org371243d2014-03-07 15:22:04 +000013#include <limits.h>
14
henrike@webrtc.org28e20752013-07-10 00:45:36 +000015#include <algorithm>
deadbeefcbecd352015-09-23 11:50:27 -070016#include <set>
Tommif888bb52015-12-12 01:37:01 +010017#include <utility>
18#include <vector>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000019
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "api/call/audio_sink.h"
21#include "api/jsepicecandidate.h"
22#include "api/jsepsessiondescription.h"
23#include "api/peerconnectioninterface.h"
24#include "call/call.h"
25#include "media/base/mediaconstants.h"
26#include "media/sctp/sctptransportinternal.h"
27#include "p2p/base/portallocator.h"
28#include "pc/channel.h"
29#include "pc/channelmanager.h"
30#include "pc/mediasession.h"
31#include "pc/sctputils.h"
32#include "pc/webrtcsessiondescriptionfactory.h"
33#include "rtc_base/basictypes.h"
34#include "rtc_base/bind.h"
35#include "rtc_base/checks.h"
36#include "rtc_base/helpers.h"
37#include "rtc_base/logging.h"
38#include "rtc_base/stringencode.h"
39#include "rtc_base/stringutils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000040
zhihuang9763d562016-08-05 11:14:50 -070041#ifdef HAVE_QUIC
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020042#include "p2p/quic/quictransportchannel.h"
zhihuang9763d562016-08-05 11:14:50 -070043#endif // HAVE_QUIC
44
henrike@webrtc.org28e20752013-07-10 00:45:36 +000045using cricket::ContentInfo;
46using cricket::ContentInfos;
47using cricket::MediaContentDescription;
48using cricket::SessionDescription;
49using cricket::TransportInfo;
50
Guo-wei Shieh3d564c12015-08-19 16:51:15 -070051using cricket::LOCAL_PORT_TYPE;
52using cricket::STUN_PORT_TYPE;
53using cricket::RELAY_PORT_TYPE;
54using cricket::PRFLX_PORT_TYPE;
55
henrike@webrtc.org28e20752013-07-10 00:45:36 +000056namespace webrtc {
57
henrike@webrtc.org28e20752013-07-10 00:45:36 +000058// Error messages
Olga Sharonovab49b6612017-10-20 12:56:51 +000059const char kBundleWithoutRtcpMux[] = "RTCP-MUX must be enabled when BUNDLE "
60 "is enabled.";
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000061const char kCreateChannelFailed[] = "Failed to create channels.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +000062const char kInvalidCandidates[] = "Description contains invalid candidates.";
63const char kInvalidSdp[] = "Invalid session description.";
Zhi Huang2a5e4262017-09-14 01:15:03 -070064const char kMlineMismatchInAnswer[] =
65 "The order of m-lines in answer doesn't match order in offer. Rejecting "
66 "answer.";
67const char kMlineMismatchInSubsequentOffer[] =
68 "The order of m-lines in subsequent offer doesn't match order from "
69 "previous offer/answer.";
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000070const char kPushDownTDFailed[] =
71 "Failed to push down transport description:";
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +000072const char kSdpWithoutDtlsFingerprint[] =
73 "Called with SDP without DTLS fingerprint.";
74const char kSdpWithoutSdesCrypto[] =
75 "Called with SDP without SDES crypto.";
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +000076const char kSdpWithoutIceUfragPwd[] =
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +000077 "Called with SDP without ice-ufrag and ice-pwd.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +000078const char kSessionError[] = "Session error code: ";
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +000079const char kSessionErrorDesc[] = "Session error description: ";
deadbeef953c2ce2017-01-09 14:53:41 -080080const char kDtlsSrtpSetupFailureRtp[] =
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +000081 "Couldn't set up DTLS-SRTP on RTP channel.";
deadbeef953c2ce2017-01-09 14:53:41 -080082const char kDtlsSrtpSetupFailureRtcp[] =
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +000083 "Couldn't set up DTLS-SRTP on RTCP channel.";
deadbeefcbecd352015-09-23 11:50:27 -070084const char kEnableBundleFailed[] = "Failed to enable BUNDLE.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +000085
Guo-wei Shieh3d564c12015-08-19 16:51:15 -070086IceCandidatePairType GetIceCandidatePairCounter(
87 const cricket::Candidate& local,
88 const cricket::Candidate& remote) {
89 const auto& l = local.type();
90 const auto& r = remote.type();
91 const auto& host = LOCAL_PORT_TYPE;
92 const auto& srflx = STUN_PORT_TYPE;
93 const auto& relay = RELAY_PORT_TYPE;
94 const auto& prflx = PRFLX_PORT_TYPE;
Guo-wei Shieh3cc834a2015-09-04 15:52:14 -070095 if (l == host && r == host) {
96 bool local_private = IPIsPrivate(local.address().ipaddr());
97 bool remote_private = IPIsPrivate(remote.address().ipaddr());
98 if (local_private) {
99 if (remote_private) {
100 return kIceCandidatePairHostPrivateHostPrivate;
101 } else {
102 return kIceCandidatePairHostPrivateHostPublic;
103 }
104 } else {
105 if (remote_private) {
106 return kIceCandidatePairHostPublicHostPrivate;
107 } else {
108 return kIceCandidatePairHostPublicHostPublic;
109 }
110 }
111 }
Guo-wei Shieh3d564c12015-08-19 16:51:15 -0700112 if (l == host && r == srflx)
113 return kIceCandidatePairHostSrflx;
114 if (l == host && r == relay)
115 return kIceCandidatePairHostRelay;
116 if (l == host && r == prflx)
117 return kIceCandidatePairHostPrflx;
118 if (l == srflx && r == host)
119 return kIceCandidatePairSrflxHost;
120 if (l == srflx && r == srflx)
121 return kIceCandidatePairSrflxSrflx;
122 if (l == srflx && r == relay)
123 return kIceCandidatePairSrflxRelay;
124 if (l == srflx && r == prflx)
125 return kIceCandidatePairSrflxPrflx;
126 if (l == relay && r == host)
127 return kIceCandidatePairRelayHost;
128 if (l == relay && r == srflx)
129 return kIceCandidatePairRelaySrflx;
130 if (l == relay && r == relay)
131 return kIceCandidatePairRelayRelay;
132 if (l == relay && r == prflx)
133 return kIceCandidatePairRelayPrflx;
134 if (l == prflx && r == host)
135 return kIceCandidatePairPrflxHost;
136 if (l == prflx && r == srflx)
137 return kIceCandidatePairPrflxSrflx;
138 if (l == prflx && r == relay)
139 return kIceCandidatePairPrflxRelay;
140 return kIceCandidatePairMax;
141}
142
Tommi589ae452017-10-15 21:20:46 +0000143// Verify that the order of media sections in |desc1| matches |desc2|. The
144// number of m= sections could be different.
145static bool MediaSectionsInSameOrder(const SessionDescription* desc1,
146 const SessionDescription* desc2) {
147 if (!desc1 || !desc2) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000148 return false;
Zhi Huang2a5e4262017-09-14 01:15:03 -0700149 }
Tommi589ae452017-10-15 21:20:46 +0000150 for (size_t i = 0;
151 i < desc1->contents().size() && i < desc2->contents().size(); ++i) {
152 if ((desc2->contents()[i].name) != desc1->contents()[i].name) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000153 return false;
154 }
Tommi589ae452017-10-15 21:20:46 +0000155 const MediaContentDescription* desc2_mdesc =
wu@webrtc.org4e393072014-04-07 17:04:35 +0000156 static_cast<const MediaContentDescription*>(
Tommi589ae452017-10-15 21:20:46 +0000157 desc2->contents()[i].description);
158 const MediaContentDescription* desc1_mdesc =
wu@webrtc.org4e393072014-04-07 17:04:35 +0000159 static_cast<const MediaContentDescription*>(
Tommi589ae452017-10-15 21:20:46 +0000160 desc1->contents()[i].description);
161 if (desc2_mdesc->type() != desc1_mdesc->type()) {
wu@webrtc.org4e393072014-04-07 17:04:35 +0000162 return false;
163 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000164 }
165 return true;
166}
167
Zhi Huang2a5e4262017-09-14 01:15:03 -0700168static bool MediaSectionsHaveSameCount(const SessionDescription* desc1,
169 const SessionDescription* desc2) {
170 if (!desc1 || !desc2) {
171 return false;
172 }
173 return desc1->contents().size() == desc2->contents().size();
174}
175
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000176// Checks that each non-rejected content has SDES crypto keys or a DTLS
deadbeefb7892532017-02-22 19:35:18 -0800177// fingerprint, unless it's in a BUNDLE group, in which case only the
178// BUNDLE-tag section (first media section/description in the BUNDLE group)
179// needs a ufrag and pwd. Mismatches, such as replying with a DTLS fingerprint
180// to SDES keys, will be caught in JsepTransport negotiation, and backstopped
181// by Channel's |srtp_required| check.
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000182static bool VerifyCrypto(const SessionDescription* desc,
183 bool dtls_enabled,
184 std::string* error) {
deadbeefb7892532017-02-22 19:35:18 -0800185 const cricket::ContentGroup* bundle =
186 desc->GetGroupByName(cricket::GROUP_TYPE_BUNDLE);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000187 const ContentInfos& contents = desc->contents();
188 for (size_t index = 0; index < contents.size(); ++index) {
189 const ContentInfo* cinfo = &contents[index];
190 if (cinfo->rejected) {
191 continue;
192 }
deadbeefb7892532017-02-22 19:35:18 -0800193 if (bundle && bundle->HasContentName(cinfo->name) &&
194 cinfo->name != *(bundle->FirstContentName())) {
195 // This isn't the first media section in the BUNDLE group, so it's not
196 // required to have crypto attributes, since only the crypto attributes
197 // from the first section actually get used.
198 continue;
199 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000200
deadbeefb7892532017-02-22 19:35:18 -0800201 // If the content isn't rejected or bundled into another m= section, crypto
202 // must be present.
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000203 const MediaContentDescription* media =
204 static_cast<const MediaContentDescription*>(cinfo->description);
205 const TransportInfo* tinfo = desc->GetTransportInfoByName(cinfo->name);
206 if (!media || !tinfo) {
207 // Something is not right.
208 LOG(LS_ERROR) << kInvalidSdp;
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000209 *error = kInvalidSdp;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000210 return false;
211 }
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000212 if (dtls_enabled) {
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000213 if (!tinfo->description.identity_fingerprint) {
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000214 LOG(LS_WARNING) <<
215 "Session description must have DTLS fingerprint if DTLS enabled.";
216 *error = kSdpWithoutDtlsFingerprint;
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000217 return false;
218 }
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000219 } else {
220 if (media->cryptos().empty()) {
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000221 LOG(LS_WARNING) <<
222 "Session description must have SDES when DTLS disabled.";
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000223 *error = kSdpWithoutSdesCrypto;
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000224 return false;
225 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000226 }
227 }
228
229 return true;
230}
231
deadbeefb7892532017-02-22 19:35:18 -0800232// Checks that each non-rejected content has ice-ufrag and ice-pwd set, unless
233// it's in a BUNDLE group, in which case only the BUNDLE-tag section (first
234// media section/description in the BUNDLE group) needs a ufrag and pwd.
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000235static bool VerifyIceUfragPwdPresent(const SessionDescription* desc) {
deadbeefb7892532017-02-22 19:35:18 -0800236 const cricket::ContentGroup* bundle =
237 desc->GetGroupByName(cricket::GROUP_TYPE_BUNDLE);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000238 const ContentInfos& contents = desc->contents();
239 for (size_t index = 0; index < contents.size(); ++index) {
240 const ContentInfo* cinfo = &contents[index];
241 if (cinfo->rejected) {
242 continue;
243 }
deadbeefb7892532017-02-22 19:35:18 -0800244 if (bundle && bundle->HasContentName(cinfo->name) &&
245 cinfo->name != *(bundle->FirstContentName())) {
246 // This isn't the first media section in the BUNDLE group, so it's not
247 // required to have ufrag/password, since only the ufrag/password from
248 // the first section actually get used.
249 continue;
250 }
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000251
deadbeefb7892532017-02-22 19:35:18 -0800252 // If the content isn't rejected or bundled into another m= section,
253 // ice-ufrag and ice-pwd must be present.
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +0000254 const TransportInfo* tinfo = desc->GetTransportInfoByName(cinfo->name);
255 if (!tinfo) {
256 // Something is not right.
257 LOG(LS_ERROR) << kInvalidSdp;
258 return false;
259 }
260 if (tinfo->description.ice_ufrag.empty() ||
261 tinfo->description.ice_pwd.empty()) {
262 LOG(LS_ERROR) << "Session description must have ice ufrag and pwd.";
263 return false;
264 }
265 }
266 return true;
267}
268
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000269static bool GetTrackIdBySsrc(const SessionDescription* session_description,
Peter Boström0c4e06b2015-10-07 12:23:21 +0200270 uint32_t ssrc,
271 std::string* track_id) {
nisseede5da42017-01-12 05:15:36 -0800272 RTC_DCHECK(track_id != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000273
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000274 const cricket::ContentInfo* audio_info =
275 cricket::GetFirstAudioContent(session_description);
jiayl@webrtc.orge21cc9a2014-08-28 22:21:34 +0000276 if (audio_info) {
277 const cricket::MediaContentDescription* audio_content =
278 static_cast<const cricket::MediaContentDescription*>(
279 audio_info->description);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000280
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000281 const auto* found =
282 cricket::GetStreamBySsrc(audio_content->streams(), ssrc);
283 if (found) {
284 *track_id = found->id;
jiayl@webrtc.orge21cc9a2014-08-28 22:21:34 +0000285 return true;
286 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000287 }
288
289 const cricket::ContentInfo* video_info =
290 cricket::GetFirstVideoContent(session_description);
jiayl@webrtc.orge21cc9a2014-08-28 22:21:34 +0000291 if (video_info) {
292 const cricket::MediaContentDescription* video_content =
293 static_cast<const cricket::MediaContentDescription*>(
294 video_info->description);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000295
tommi@webrtc.org586f2ed2015-01-22 23:00:41 +0000296 const auto* found =
297 cricket::GetStreamBySsrc(video_content->streams(), ssrc);
298 if (found) {
299 *track_id = found->id;
jiayl@webrtc.orge21cc9a2014-08-28 22:21:34 +0000300 return true;
301 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000302 }
303 return false;
304}
305
deadbeef953c2ce2017-01-09 14:53:41 -0800306// Get the SCTP port out of a SessionDescription.
307// Return -1 if not found.
308static int GetSctpPort(const SessionDescription* session_description) {
309 const ContentInfo* content_info = GetFirstDataContent(session_description);
310 RTC_DCHECK(content_info);
311 if (!content_info) {
312 return -1;
313 }
314 const cricket::DataContentDescription* data =
315 static_cast<const cricket::DataContentDescription*>(
316 (content_info->description));
317 std::string value;
318 cricket::DataCodec match_pattern(cricket::kGoogleSctpDataCodecPlType,
319 cricket::kGoogleSctpDataCodecName);
320 for (const cricket::DataCodec& codec : data->codecs()) {
321 if (!codec.Matches(match_pattern)) {
322 continue;
323 }
324 if (codec.GetParam(cricket::kCodecParamPort, &value)) {
325 return rtc::FromString<int>(value);
326 }
327 }
328 return -1;
329}
330
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000331static bool BadSdp(const std::string& source,
332 const std::string& type,
333 const std::string& reason,
334 std::string* err_desc) {
335 std::ostringstream desc;
deadbeefd59daf82015-10-14 15:02:44 -0700336 desc << "Failed to set " << source;
337 if (!type.empty()) {
338 desc << " " << type;
339 }
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000340 desc << " sdp: " << reason;
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000341
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000342 if (err_desc) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000343 *err_desc = desc.str();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000344 }
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000345 LOG(LS_ERROR) << desc.str();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000346 return false;
347}
348
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000349static bool BadSdp(cricket::ContentSource source,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000350 const std::string& type,
351 const std::string& reason,
352 std::string* err_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000353 if (source == cricket::CS_LOCAL) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000354 return BadSdp("local", type, reason, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000355 } else {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000356 return BadSdp("remote", type, reason, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000357 }
358}
359
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000360static bool BadLocalSdp(const std::string& type,
361 const std::string& reason,
362 std::string* err_desc) {
363 return BadSdp(cricket::CS_LOCAL, type, reason, err_desc);
364}
365
366static bool BadRemoteSdp(const std::string& type,
367 const std::string& reason,
368 std::string* err_desc) {
369 return BadSdp(cricket::CS_REMOTE, type, reason, err_desc);
370}
371
372static bool BadOfferSdp(cricket::ContentSource source,
373 const std::string& reason,
374 std::string* err_desc) {
375 return BadSdp(source, SessionDescriptionInterface::kOffer, reason, err_desc);
376}
377
378static bool BadPranswerSdp(cricket::ContentSource source,
379 const std::string& reason,
380 std::string* err_desc) {
381 return BadSdp(source, SessionDescriptionInterface::kPrAnswer,
382 reason, err_desc);
383}
384
385static bool BadAnswerSdp(cricket::ContentSource source,
386 const std::string& reason,
387 std::string* err_desc) {
388 return BadSdp(source, SessionDescriptionInterface::kAnswer, reason, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000389}
390
deadbeefd59daf82015-10-14 15:02:44 -0700391#define GET_STRING_OF_STATE(state) \
392 case webrtc::WebRtcSession::state: \
393 result = #state; \
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000394 break;
395
deadbeefd59daf82015-10-14 15:02:44 -0700396static std::string GetStateString(webrtc::WebRtcSession::State state) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000397 std::string result;
398 switch (state) {
399 GET_STRING_OF_STATE(STATE_INIT)
deadbeefd59daf82015-10-14 15:02:44 -0700400 GET_STRING_OF_STATE(STATE_SENTOFFER)
401 GET_STRING_OF_STATE(STATE_RECEIVEDOFFER)
402 GET_STRING_OF_STATE(STATE_SENTPRANSWER)
403 GET_STRING_OF_STATE(STATE_RECEIVEDPRANSWER)
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000404 GET_STRING_OF_STATE(STATE_INPROGRESS)
deadbeefd59daf82015-10-14 15:02:44 -0700405 GET_STRING_OF_STATE(STATE_CLOSED)
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000406 default:
nissec80e7412017-01-11 05:56:46 -0800407 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000408 break;
409 }
410 return result;
411}
412
deadbeefd59daf82015-10-14 15:02:44 -0700413#define GET_STRING_OF_ERROR_CODE(err) \
414 case webrtc::WebRtcSession::err: \
415 result = #err; \
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000416 break;
417
deadbeefd59daf82015-10-14 15:02:44 -0700418static std::string GetErrorCodeString(webrtc::WebRtcSession::Error err) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000419 std::string result;
420 switch (err) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000421 GET_STRING_OF_ERROR_CODE(ERROR_NONE)
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000422 GET_STRING_OF_ERROR_CODE(ERROR_CONTENT)
423 GET_STRING_OF_ERROR_CODE(ERROR_TRANSPORT)
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000424 default:
nisseeb4ca4e2017-01-12 02:24:27 -0800425 RTC_NOTREACHED();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000426 break;
427 }
428 return result;
429}
430
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000431static std::string MakeErrorString(const std::string& error,
432 const std::string& desc) {
433 std::ostringstream ret;
434 ret << error << " " << desc;
435 return ret.str();
436}
437
438static std::string MakeTdErrorString(const std::string& desc) {
439 return MakeErrorString(kPushDownTDFailed, desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000440}
441
deadbeef0ed85b22016-02-23 17:24:52 -0800442// Returns true if |new_desc| requests an ICE restart (i.e., new ufrag/pwd).
443bool CheckForRemoteIceRestart(const SessionDescriptionInterface* old_desc,
444 const SessionDescriptionInterface* new_desc,
445 const std::string& content_name) {
446 if (!old_desc) {
honghaiz503726c2015-07-31 12:37:38 -0700447 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000448 }
deadbeef0ed85b22016-02-23 17:24:52 -0800449 const SessionDescription* new_sd = new_desc->description();
450 const SessionDescription* old_sd = old_desc->description();
451 const ContentInfo* cinfo = new_sd->GetContentByName(content_name);
452 if (!cinfo || cinfo->rejected) {
453 return false;
454 }
455 // If the content isn't rejected, check if ufrag and password has changed.
456 const cricket::TransportDescription* new_transport_desc =
457 new_sd->GetTransportDescriptionByName(content_name);
458 const cricket::TransportDescription* old_transport_desc =
459 old_sd->GetTransportDescriptionByName(content_name);
460 if (!new_transport_desc || !old_transport_desc) {
461 // No transport description exists. This is not an ICE restart.
462 return false;
463 }
464 if (cricket::IceCredentialsChanged(
465 old_transport_desc->ice_ufrag, old_transport_desc->ice_pwd,
466 new_transport_desc->ice_ufrag, new_transport_desc->ice_pwd)) {
467 LOG(LS_INFO) << "Remote peer requests ICE restart for " << content_name
468 << ".";
469 return true;
470 }
471 return false;
472}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000473
zhihuang29ff8442016-07-27 11:07:25 -0700474WebRtcSession::WebRtcSession(
nisseeaabdf62017-05-05 02:23:02 -0700475 Call* call,
476 cricket::ChannelManager* channel_manager,
477 const cricket::MediaConfig& media_config,
478 RtcEventLog* event_log,
zhihuang29ff8442016-07-27 11:07:25 -0700479 rtc::Thread* network_thread,
480 rtc::Thread* worker_thread,
481 rtc::Thread* signaling_thread,
482 cricket::PortAllocator* port_allocator,
deadbeef953c2ce2017-01-09 14:53:41 -0800483 std::unique_ptr<cricket::TransportController> transport_controller,
484 std::unique_ptr<cricket::SctpTransportInternalFactory> sctp_factory)
zhihuang9763d562016-08-05 11:14:50 -0700485 : network_thread_(network_thread),
486 worker_thread_(worker_thread),
danilchape9021a32016-05-17 01:52:02 -0700487 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000488 // RFC 3264: The numeric value of the session id and version in the
489 // o line MUST be representable with a "64 bit signed integer".
490 // Due to this constraint session id |sid_| is max limited to LLONG_MAX.
deadbeefd59daf82015-10-14 15:02:44 -0700491 sid_(rtc::ToString(rtc::CreateRandomId64() & LLONG_MAX)),
zhihuang29ff8442016-07-27 11:07:25 -0700492 transport_controller_(std::move(transport_controller)),
deadbeef953c2ce2017-01-09 14:53:41 -0800493 sctp_factory_(std::move(sctp_factory)),
nisseeaabdf62017-05-05 02:23:02 -0700494 media_config_(media_config),
495 event_log_(event_log),
496 call_(call),
497 channel_manager_(channel_manager),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000498 ice_observer_(NULL),
499 ice_connection_state_(PeerConnectionInterface::kIceConnectionNew),
Peter Thatcher54360512015-07-08 11:08:35 -0700500 ice_connection_receiving_(true),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000501 older_version_remote_peer_(false),
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +0000502 dtls_enabled_(false),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000503 data_channel_type_(cricket::DCT_NONE),
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +0000504 metrics_observer_(NULL) {
deadbeefd59daf82015-10-14 15:02:44 -0700505 transport_controller_->SetIceRole(cricket::ICEROLE_CONTROLLED);
506 transport_controller_->SignalConnectionState.connect(
deadbeefcbecd352015-09-23 11:50:27 -0700507 this, &WebRtcSession::OnTransportControllerConnectionState);
deadbeefd59daf82015-10-14 15:02:44 -0700508 transport_controller_->SignalReceiving.connect(
deadbeefcbecd352015-09-23 11:50:27 -0700509 this, &WebRtcSession::OnTransportControllerReceiving);
deadbeefd59daf82015-10-14 15:02:44 -0700510 transport_controller_->SignalGatheringState.connect(
deadbeefcbecd352015-09-23 11:50:27 -0700511 this, &WebRtcSession::OnTransportControllerGatheringState);
deadbeefd59daf82015-10-14 15:02:44 -0700512 transport_controller_->SignalCandidatesGathered.connect(
deadbeefcbecd352015-09-23 11:50:27 -0700513 this, &WebRtcSession::OnTransportControllerCandidatesGathered);
Honghai Zhang7fb69db2016-03-14 11:59:18 -0700514 transport_controller_->SignalCandidatesRemoved.connect(
515 this, &WebRtcSession::OnTransportControllerCandidatesRemoved);
zhihuangd82eee02016-08-26 11:25:05 -0700516 transport_controller_->SignalDtlsHandshakeError.connect(
deadbeef953c2ce2017-01-09 14:53:41 -0800517 this, &WebRtcSession::OnTransportControllerDtlsHandshakeError);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000518}
519
520WebRtcSession::~WebRtcSession() {
nisseede5da42017-01-12 05:15:36 -0800521 RTC_DCHECK(signaling_thread()->IsCurrent());
Steve Anton169629a2017-08-30 17:36:36 -0700522 // Destroy video channels first since they may have a pointer to a voice
523 // channel.
524 for (auto* channel : video_channels_) {
525 DestroyVideoChannel(channel);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000526 }
Steve Anton169629a2017-08-30 17:36:36 -0700527 for (auto* channel : voice_channels_) {
528 DestroyVoiceChannel(channel);
buildbot@webrtc.orgb4c7b092014-08-25 12:11:58 +0000529 }
deadbeef953c2ce2017-01-09 14:53:41 -0800530 if (rtp_data_channel_) {
zhihuangf5b251b2017-01-12 19:37:48 -0800531 DestroyDataChannel();
deadbeef953c2ce2017-01-09 14:53:41 -0800532 }
533 if (sctp_transport_) {
534 SignalDataChannelDestroyed();
535 network_thread_->Invoke<void>(
536 RTC_FROM_HERE, rtc::Bind(&WebRtcSession::DestroySctpTransport_n, this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000537 }
zhihuang9763d562016-08-05 11:14:50 -0700538#ifdef HAVE_QUIC
539 if (quic_data_transport_) {
540 quic_data_transport_.reset();
541 }
542#endif
deadbeefd59daf82015-10-14 15:02:44 -0700543
544 LOG(LS_INFO) << "Session: " << id() << " is destroyed.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000545}
546
wu@webrtc.org91053e72013-08-10 07:18:04 +0000547bool WebRtcSession::Initialize(
wu@webrtc.org97077a32013-10-25 21:18:33 +0000548 const PeerConnectionFactoryInterface::Options& options,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200549 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
Henrik Lundin64dad832015-05-11 12:44:23 +0200550 const PeerConnectionInterface::RTCConfiguration& rtc_configuration) {
551 bundle_policy_ = rtc_configuration.bundle_policy;
Peter Thatcheraf55ccc2015-05-21 07:48:41 -0700552 rtcp_mux_policy_ = rtc_configuration.rtcp_mux_policy;
deadbeefd59daf82015-10-14 15:02:44 -0700553 transport_controller_->SetSslMaxProtocolVersion(options.ssl_max_version);
pthatcher@webrtc.org877ac762015-02-04 22:03:09 +0000554
Henrik Boström87713d02015-08-25 09:53:21 +0200555 // Obtain a certificate from RTCConfiguration if any were provided (optional).
556 rtc::scoped_refptr<rtc::RTCCertificate> certificate;
557 if (!rtc_configuration.certificates.empty()) {
558 // TODO(hbos,torbjorng): Decide on certificate-selection strategy instead of
559 // just picking the first one. The decision should be made based on the DTLS
560 // handshake. The DTLS negotiations need to know about all certificates.
561 certificate = rtc_configuration.certificates[0];
562 }
563
honghaiz1f429e32015-09-28 07:57:34 -0700564 SetIceConfig(ParseIceConfig(rtc_configuration));
honghaiz4edc39c2015-09-01 09:53:56 -0700565
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000566 if (options.disable_encryption) {
567 dtls_enabled_ = false;
568 } else {
Henrik Boström87713d02015-08-25 09:53:21 +0200569 // Enable DTLS by default if we have an identity store or a certificate.
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200570 dtls_enabled_ = (cert_generator || certificate);
htaa2a49d92016-03-04 02:51:39 -0800571 // |rtc_configuration| can override the default |dtls_enabled_| value.
572 if (rtc_configuration.enable_dtls_srtp) {
573 dtls_enabled_ = *(rtc_configuration.enable_dtls_srtp);
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000574 }
sergeyu@chromium.orga59696b2013-09-13 23:48:58 +0000575 }
576
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000577 // Enable creation of RTP data channels if the kEnableRtpDataChannels is set.
wu@webrtc.org97077a32013-10-25 21:18:33 +0000578 // It takes precendence over the disable_sctp_data_channels
579 // PeerConnectionFactoryInterface::Options.
htaa2a49d92016-03-04 02:51:39 -0800580 if (rtc_configuration.enable_rtp_data_channel) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000581 data_channel_type_ = cricket::DCT_RTP;
zhihuang9763d562016-08-05 11:14:50 -0700582 }
583#ifdef HAVE_QUIC
584 else if (rtc_configuration.enable_quic) {
585 // Use QUIC instead of DTLS when |enable_quic| is true.
586 data_channel_type_ = cricket::DCT_QUIC;
587 transport_controller_->use_quic();
588 if (dtls_enabled_) {
589 LOG(LS_INFO) << "Using QUIC instead of DTLS";
590 }
591 quic_data_transport_.reset(
592 new QuicDataTransport(signaling_thread(), worker_thread(),
593 network_thread(), transport_controller_.get()));
594 }
595#endif // HAVE_QUIC
596 else {
wu@webrtc.org91053e72013-08-10 07:18:04 +0000597 // DTLS has to be enabled to use SCTP.
wu@webrtc.org97077a32013-10-25 21:18:33 +0000598 if (!options.disable_sctp_data_channels && dtls_enabled_) {
wu@webrtc.org91053e72013-08-10 07:18:04 +0000599 data_channel_type_ = cricket::DCT_SCTP;
600 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000601 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000602
htaa2a49d92016-03-04 02:51:39 -0800603 video_options_.screencast_min_bitrate_kbps =
604 rtc_configuration.screencast_min_bitrate;
605 audio_options_.combined_audio_video_bwe =
606 rtc_configuration.combined_audio_video_bwe;
buildbot@webrtc.orgb4c7b092014-08-25 12:11:58 +0000607
kwiberg102c6a62015-10-30 02:47:38 -0700608 audio_options_.audio_jitter_buffer_max_packets =
Karl Wibergbe579832015-11-10 22:34:18 +0100609 rtc::Optional<int>(rtc_configuration.audio_jitter_buffer_max_packets);
Henrik Lundin64dad832015-05-11 12:44:23 +0200610
Karl Wibergbe579832015-11-10 22:34:18 +0100611 audio_options_.audio_jitter_buffer_fast_accelerate = rtc::Optional<bool>(
612 rtc_configuration.audio_jitter_buffer_fast_accelerate);
Henrik Lundin5263b3c2015-06-01 10:29:41 +0200613
Henrik Boström87713d02015-08-25 09:53:21 +0200614 if (!dtls_enabled_) {
615 // Construct with DTLS disabled.
616 webrtc_session_desc_factory_.reset(new WebRtcSessionDescriptionFactory(
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200617 signaling_thread(), channel_manager_, this, id(),
618 std::unique_ptr<rtc::RTCCertificateGeneratorInterface>()));
Henrik Boström87713d02015-08-25 09:53:21 +0200619 } else {
620 // Construct with DTLS enabled.
621 if (!certificate) {
Henrik Boström87713d02015-08-25 09:53:21 +0200622 webrtc_session_desc_factory_.reset(new WebRtcSessionDescriptionFactory(
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200623 signaling_thread(), channel_manager_, this, id(),
624 std::move(cert_generator)));
Henrik Boström87713d02015-08-25 09:53:21 +0200625 } else {
626 // Use the already generated certificate.
627 webrtc_session_desc_factory_.reset(new WebRtcSessionDescriptionFactory(
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200628 signaling_thread(), channel_manager_, this, id(), certificate));
Henrik Boström87713d02015-08-25 09:53:21 +0200629 }
630 }
wu@webrtc.org91053e72013-08-10 07:18:04 +0000631
Henrik Boströmd8281982015-08-27 10:12:24 +0200632 webrtc_session_desc_factory_->SignalCertificateReady.connect(
633 this, &WebRtcSession::OnCertificateReady);
mallinath@webrtc.org7e809c32013-09-30 18:59:08 +0000634
wu@webrtc.org97077a32013-10-25 21:18:33 +0000635 if (options.disable_encryption) {
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +0000636 webrtc_session_desc_factory_->SetSdesPolicy(cricket::SEC_DISABLED);
mallinath@webrtc.org7e809c32013-09-30 18:59:08 +0000637 }
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700638
jbauch5869f502017-06-29 12:31:36 -0700639 webrtc_session_desc_factory_->set_enable_encrypted_rtp_header_extensions(
640 options.crypto_options.enable_encrypted_rtp_header_extensions);
641
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000642 return true;
643}
644
deadbeefd59daf82015-10-14 15:02:44 -0700645void WebRtcSession::Close() {
646 SetState(STATE_CLOSED);
647 RemoveUnusedChannels(nullptr);
stefanf79ade12017-06-02 06:44:03 -0700648 call_ = nullptr;
Steve Anton169629a2017-08-30 17:36:36 -0700649 RTC_DCHECK(voice_channels_.empty());
650 RTC_DCHECK(video_channels_.empty());
deadbeef953c2ce2017-01-09 14:53:41 -0800651 RTC_DCHECK(!rtp_data_channel_);
652 RTC_DCHECK(!sctp_transport_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000653}
654
deadbeef0ed85b22016-02-23 17:24:52 -0800655cricket::BaseChannel* WebRtcSession::GetChannel(
656 const std::string& content_name) {
657 if (voice_channel() && voice_channel()->content_name() == content_name) {
658 return voice_channel();
659 }
660 if (video_channel() && video_channel()->content_name() == content_name) {
661 return video_channel();
662 }
deadbeef953c2ce2017-01-09 14:53:41 -0800663 if (rtp_data_channel() &&
664 rtp_data_channel()->content_name() == content_name) {
665 return rtp_data_channel();
deadbeef0ed85b22016-02-23 17:24:52 -0800666 }
667 return nullptr;
668}
669
deadbeef953c2ce2017-01-09 14:53:41 -0800670bool WebRtcSession::GetSctpSslRole(rtc::SSLRole* role) {
671 if (!local_description() || !remote_description()) {
672 LOG(LS_INFO) << "Local and Remote descriptions must be applied to get the "
673 << "SSL Role of the SCTP transport.";
674 return false;
675 }
676 if (!sctp_transport_) {
677 LOG(LS_INFO) << "Non-rejected SCTP m= section is needed to get the "
678 << "SSL Role of the SCTP transport.";
679 return false;
680 }
681
682 return transport_controller_->GetSslRole(*sctp_transport_name_, role);
683}
684
685bool WebRtcSession::GetSslRole(const std::string& content_name,
Taylor Brandstetterf475d362016-01-08 15:35:57 -0800686 rtc::SSLRole* role) {
deadbeeffe4a8a42016-12-20 17:56:17 -0800687 if (!local_description() || !remote_description()) {
deadbeef953c2ce2017-01-09 14:53:41 -0800688 LOG(LS_INFO) << "Local and Remote descriptions must be applied to get the "
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000689 << "SSL Role of the session.";
690 return false;
691 }
692
deadbeef953c2ce2017-01-09 14:53:41 -0800693 return transport_controller_->GetSslRole(GetTransportName(content_name),
694 role);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000695}
696
jiayl@webrtc.orgb18bf5e2014-08-04 18:34:16 +0000697void WebRtcSession::CreateOffer(
698 CreateSessionDescriptionObserver* observer,
deadbeefab9b2d12015-10-14 11:33:11 -0700699 const PeerConnectionInterface::RTCOfferAnswerOptions& options,
700 const cricket::MediaSessionOptions& session_options) {
701 webrtc_session_desc_factory_->CreateOffer(observer, options, session_options);
wu@webrtc.org91053e72013-08-10 07:18:04 +0000702}
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000703
deadbeefab9b2d12015-10-14 11:33:11 -0700704void WebRtcSession::CreateAnswer(
705 CreateSessionDescriptionObserver* observer,
deadbeefab9b2d12015-10-14 11:33:11 -0700706 const cricket::MediaSessionOptions& session_options) {
htaa2a49d92016-03-04 02:51:39 -0800707 webrtc_session_desc_factory_->CreateAnswer(observer, session_options);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000708}
709
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000710bool WebRtcSession::SetLocalDescription(SessionDescriptionInterface* desc,
711 std::string* err_desc) {
nisseede5da42017-01-12 05:15:36 -0800712 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700713
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000714 // Takes the ownership of |desc| regardless of the result.
715 std::unique_ptr<SessionDescriptionInterface> desc_temp(desc);
716
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000717 // Validate SDP.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000718 if (!ValidateSessionDescription(desc, cricket::CS_LOCAL, err_desc)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000719 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000720 }
721
deadbeefd59daf82015-10-14 15:02:44 -0700722 // Update the initial_offerer flag if this session is the initial_offerer.
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000723 Action action = GetAction(desc->type());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000724 if (state() == STATE_INIT && action == kOffer) {
deadbeefd59daf82015-10-14 15:02:44 -0700725 initial_offerer_ = true;
726 transport_controller_->SetIceRole(cricket::ICEROLE_CONTROLLING);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000727 }
728
deadbeeffe4a8a42016-12-20 17:56:17 -0800729 if (action == kAnswer) {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000730 current_local_description_.reset(desc_temp.release());
731 pending_local_description_.reset(nullptr);
732 current_remote_description_.reset(pending_remote_description_.release());
deadbeeffe4a8a42016-12-20 17:56:17 -0800733 } else {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000734 pending_local_description_.reset(desc_temp.release());
deadbeeffe4a8a42016-12-20 17:56:17 -0800735 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000736
737 // Transport and Media channels will be created only when offer is set.
deadbeeffe4a8a42016-12-20 17:56:17 -0800738 if (action == kOffer && !CreateChannels(local_description()->description())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000739 // TODO(mallinath) - Handle CreateChannel failure, as new local description
740 // is applied. Restore back to old description.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000741 return BadLocalSdp(desc->type(), kCreateChannelFailed, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000742 }
743
deadbeefcbecd352015-09-23 11:50:27 -0700744 // Remove unused channels if MediaContentDescription is rejected.
deadbeeffe4a8a42016-12-20 17:56:17 -0800745 RemoveUnusedChannels(local_description()->description());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000746
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000747 if (!UpdateSessionState(action, cricket::CS_LOCAL, err_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000748 return false;
749 }
deadbeeffe4a8a42016-12-20 17:56:17 -0800750 if (remote_description()) {
deadbeefd59daf82015-10-14 15:02:44 -0700751 // Now that we have a local description, we can push down remote candidates.
deadbeeffe4a8a42016-12-20 17:56:17 -0800752 UseCandidatesInSessionDescription(remote_description());
deadbeefcbecd352015-09-23 11:50:27 -0700753 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000754
deadbeef0ed85b22016-02-23 17:24:52 -0800755 pending_ice_restarts_.clear();
deadbeefd59daf82015-10-14 15:02:44 -0700756 if (error() != ERROR_NONE) {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000757 return BadLocalSdp(desc->type(), GetSessionErrorMsg(), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000758 }
759 return true;
760}
761
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000762bool WebRtcSession::SetRemoteDescription(SessionDescriptionInterface* desc,
763 std::string* err_desc) {
nisseede5da42017-01-12 05:15:36 -0800764 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700765
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000766 // Takes the ownership of |desc| regardless of the result.
767 std::unique_ptr<SessionDescriptionInterface> desc_temp(desc);
768
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000769 // Validate SDP.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000770 if (!ValidateSessionDescription(desc, cricket::CS_REMOTE, err_desc)) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +0000771 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000772 }
773
deadbeeffe4a8a42016-12-20 17:56:17 -0800774 const SessionDescriptionInterface* old_remote_description =
775 remote_description();
776 // Grab ownership of the description being replaced for the remainder of this
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000777 // method, since it's used below.
deadbeeffe4a8a42016-12-20 17:56:17 -0800778 std::unique_ptr<SessionDescriptionInterface> replaced_remote_description;
779 Action action = GetAction(desc->type());
780 if (action == kAnswer) {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000781 replaced_remote_description.reset(
782 pending_remote_description_ ? pending_remote_description_.release()
783 : current_remote_description_.release());
784 current_remote_description_.reset(desc_temp.release());
785 pending_remote_description_.reset(nullptr);
786 current_local_description_.reset(pending_local_description_.release());
deadbeeffe4a8a42016-12-20 17:56:17 -0800787 } else {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000788 replaced_remote_description.reset(pending_remote_description_.release());
789 pending_remote_description_.reset(desc_temp.release());
deadbeeffe4a8a42016-12-20 17:56:17 -0800790 }
deadbeefd59daf82015-10-14 15:02:44 -0700791
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000792 // Transport and Media channels will be created only when offer is set.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000793 if (action == kOffer && !CreateChannels(desc->description())) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000794 // TODO(mallinath) - Handle CreateChannel failure, as new local description
795 // is applied. Restore back to old description.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000796 return BadRemoteSdp(desc->type(), kCreateChannelFailed, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000797 }
798
deadbeefcbecd352015-09-23 11:50:27 -0700799 // Remove unused channels if MediaContentDescription is rejected.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000800 RemoveUnusedChannels(desc->description());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000801
802 // NOTE: Candidates allocation will be initiated only when SetLocalDescription
803 // is called.
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000804 if (!UpdateSessionState(action, cricket::CS_REMOTE, err_desc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000805 return false;
806 }
807
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000808 if (local_description() && !UseCandidatesInSessionDescription(desc)) {
809 return BadRemoteSdp(desc->type(), kInvalidCandidates, err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000810 }
811
deadbeeffe4a8a42016-12-20 17:56:17 -0800812 if (old_remote_description) {
deadbeef0ed85b22016-02-23 17:24:52 -0800813 for (const cricket::ContentInfo& content :
deadbeeffe4a8a42016-12-20 17:56:17 -0800814 old_remote_description->description()->contents()) {
deadbeef0ed85b22016-02-23 17:24:52 -0800815 // Check if this new SessionDescription contains new ICE ufrag and
816 // password that indicates the remote peer requests an ICE restart.
817 // TODO(deadbeef): When we start storing both the current and pending
818 // remote description, this should reset pending_ice_restarts and compare
819 // against the current description.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000820 if (CheckForRemoteIceRestart(old_remote_description, desc,
deadbeeffe4a8a42016-12-20 17:56:17 -0800821 content.name)) {
deadbeef0ed85b22016-02-23 17:24:52 -0800822 if (action == kOffer) {
823 pending_ice_restarts_.insert(content.name);
824 }
825 } else {
826 // We retain all received candidates only if ICE is not restarted.
827 // When ICE is restarted, all previous candidates belong to an old
828 // generation and should not be kept.
829 // TODO(deadbeef): This goes against the W3C spec which says the remote
830 // description should only contain candidates from the last set remote
831 // description plus any candidates added since then. We should remove
832 // this once we're sure it won't break anything.
833 WebRtcSessionDescriptionFactory::CopyCandidatesFromSessionDescription(
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000834 old_remote_description, content.name, desc);
deadbeef0ed85b22016-02-23 17:24:52 -0800835 }
836 }
honghaiz503726c2015-07-31 12:37:38 -0700837 }
838
deadbeefd59daf82015-10-14 15:02:44 -0700839 if (error() != ERROR_NONE) {
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000840 return BadRemoteSdp(desc->type(), GetSessionErrorMsg(), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000841 }
jiayl@webrtc.orgdacdd942015-01-23 17:33:34 +0000842
843 // Set the the ICE connection state to connecting since the connection may
844 // become writable with peer reflexive candidates before any remote candidate
845 // is signaled.
846 // TODO(pthatcher): This is a short-term solution for crbug/446908. A real fix
847 // is to have a new signal the indicates a change in checking state from the
848 // transport and expose a new checking() member from transport that can be
849 // read to determine the current checking state. The existing SignalConnecting
850 // actually means "gathering candidates", so cannot be be used here.
Olga Sharonovaf2662f02017-10-20 09:42:08 +0000851 if (desc->type() != SessionDescriptionInterface::kOffer &&
jiayl@webrtc.orgdacdd942015-01-23 17:33:34 +0000852 ice_connection_state_ == PeerConnectionInterface::kIceConnectionNew) {
853 SetIceConnectionState(PeerConnectionInterface::kIceConnectionChecking);
854 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000855 return true;
856}
857
Steve Anton18ee1d52017-09-11 11:32:35 -0700858// TODO(steveanton): Eventually it'd be nice to store the channels as a single
859// vector of BaseChannel pointers instead of separate voice and video channel
860// vectors. At that point, this will become a simple getter.
861std::vector<cricket::BaseChannel*> WebRtcSession::Channels() const {
862 std::vector<cricket::BaseChannel*> channels;
863 channels.insert(channels.end(), voice_channels_.begin(),
864 voice_channels_.end());
865 channels.insert(channels.end(), video_channels_.begin(),
866 video_channels_.end());
867 if (rtp_data_channel_) {
868 channels.push_back(rtp_data_channel_.get());
869 }
870 return channels;
871}
872
deadbeefd59daf82015-10-14 15:02:44 -0700873void WebRtcSession::LogState(State old_state, State new_state) {
874 LOG(LS_INFO) << "Session:" << id()
875 << " Old state:" << GetStateString(old_state)
876 << " New state:" << GetStateString(new_state);
877}
878
879void WebRtcSession::SetState(State state) {
nisseede5da42017-01-12 05:15:36 -0800880 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeefd59daf82015-10-14 15:02:44 -0700881 if (state != state_) {
882 LogState(state_, state);
883 state_ = state;
884 SignalState(this, state_);
885 }
886}
887
888void WebRtcSession::SetError(Error error, const std::string& error_desc) {
nisseede5da42017-01-12 05:15:36 -0800889 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeefd59daf82015-10-14 15:02:44 -0700890 if (error != error_) {
891 error_ = error;
892 error_desc_ = error_desc;
893 }
894}
895
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000896bool WebRtcSession::UpdateSessionState(
897 Action action, cricket::ContentSource source,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000898 std::string* err_desc) {
nisseede5da42017-01-12 05:15:36 -0800899 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -0700900
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000901 // If there's already a pending error then no state transition should happen.
902 // But all call-sites should be verifying this before calling us!
nisseede5da42017-01-12 05:15:36 -0800903 RTC_DCHECK(error() == ERROR_NONE);
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000904 std::string td_err;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000905 if (action == kOffer) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000906 if (!PushdownTransportDescription(source, cricket::CA_OFFER, &td_err)) {
907 return BadOfferSdp(source, MakeTdErrorString(td_err), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000908 }
deadbeefd59daf82015-10-14 15:02:44 -0700909 SetState(source == cricket::CS_LOCAL ? STATE_SENTOFFER
910 : STATE_RECEIVEDOFFER);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000911 if (!PushdownMediaDescription(cricket::CA_OFFER, source, err_desc)) {
deadbeefd59daf82015-10-14 15:02:44 -0700912 SetError(ERROR_CONTENT, *err_desc);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000913 }
deadbeefd59daf82015-10-14 15:02:44 -0700914 if (error() != ERROR_NONE) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000915 return BadOfferSdp(source, GetSessionErrorMsg(), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000916 }
917 } else if (action == kPrAnswer) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000918 if (!PushdownTransportDescription(source, cricket::CA_PRANSWER, &td_err)) {
919 return BadPranswerSdp(source, MakeTdErrorString(td_err), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000920 }
921 EnableChannels();
deadbeefd59daf82015-10-14 15:02:44 -0700922 SetState(source == cricket::CS_LOCAL ? STATE_SENTPRANSWER
923 : STATE_RECEIVEDPRANSWER);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000924 if (!PushdownMediaDescription(cricket::CA_PRANSWER, source, err_desc)) {
deadbeefd59daf82015-10-14 15:02:44 -0700925 SetError(ERROR_CONTENT, *err_desc);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000926 }
deadbeefd59daf82015-10-14 15:02:44 -0700927 if (error() != ERROR_NONE) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000928 return BadPranswerSdp(source, GetSessionErrorMsg(), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000929 }
930 } else if (action == kAnswer) {
deadbeefcbecd352015-09-23 11:50:27 -0700931 const cricket::ContentGroup* local_bundle =
deadbeeffe4a8a42016-12-20 17:56:17 -0800932 local_description()->description()->GetGroupByName(
933 cricket::GROUP_TYPE_BUNDLE);
deadbeefcbecd352015-09-23 11:50:27 -0700934 const cricket::ContentGroup* remote_bundle =
deadbeeffe4a8a42016-12-20 17:56:17 -0800935 remote_description()->description()->GetGroupByName(
936 cricket::GROUP_TYPE_BUNDLE);
deadbeefcbecd352015-09-23 11:50:27 -0700937 if (local_bundle && remote_bundle) {
Taylor Brandstetterf475d362016-01-08 15:35:57 -0800938 // The answerer decides the transport to bundle on.
deadbeefcbecd352015-09-23 11:50:27 -0700939 const cricket::ContentGroup* answer_bundle =
940 (source == cricket::CS_LOCAL ? local_bundle : remote_bundle);
941 if (!EnableBundle(*answer_bundle)) {
942 LOG(LS_WARNING) << "Failed to enable BUNDLE.";
943 return BadAnswerSdp(source, kEnableBundleFailed, err_desc);
944 }
945 }
Taylor Brandstetterf475d362016-01-08 15:35:57 -0800946 // Only push down the transport description after enabling BUNDLE; we don't
947 // want to push down a description on a transport about to be destroyed.
948 if (!PushdownTransportDescription(source, cricket::CA_ANSWER, &td_err)) {
949 return BadAnswerSdp(source, MakeTdErrorString(td_err), err_desc);
950 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000951 EnableChannels();
deadbeefd59daf82015-10-14 15:02:44 -0700952 SetState(STATE_INPROGRESS);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000953 if (!PushdownMediaDescription(cricket::CA_ANSWER, source, err_desc)) {
deadbeefd59daf82015-10-14 15:02:44 -0700954 SetError(ERROR_CONTENT, *err_desc);
pthatcher@webrtc.org592470b2015-03-16 21:15:37 +0000955 }
deadbeefd59daf82015-10-14 15:02:44 -0700956 if (error() != ERROR_NONE) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +0000957 return BadAnswerSdp(source, GetSessionErrorMsg(), err_desc);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000958 }
959 }
960 return true;
961}
962
963WebRtcSession::Action WebRtcSession::GetAction(const std::string& type) {
964 if (type == SessionDescriptionInterface::kOffer) {
965 return WebRtcSession::kOffer;
966 } else if (type == SessionDescriptionInterface::kPrAnswer) {
967 return WebRtcSession::kPrAnswer;
968 } else if (type == SessionDescriptionInterface::kAnswer) {
969 return WebRtcSession::kAnswer;
970 }
nisseede5da42017-01-12 05:15:36 -0800971 RTC_NOTREACHED() << "unknown action type";
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000972 return WebRtcSession::kOffer;
973}
974
deadbeefd59daf82015-10-14 15:02:44 -0700975bool WebRtcSession::PushdownMediaDescription(
976 cricket::ContentAction action,
977 cricket::ContentSource source,
978 std::string* err) {
Steve Anton18ee1d52017-09-11 11:32:35 -0700979 const SessionDescription* sdesc =
980 (source == cricket::CS_LOCAL ? local_description() : remote_description())
981 ->description();
982 RTC_DCHECK(sdesc);
983 bool all_success = true;
984 for (auto* channel : Channels()) {
985 // TODO(steveanton): Add support for multiple channels of the same type.
986 const ContentInfo* content_info =
987 cricket::GetFirstMediaContent(sdesc->contents(), channel->media_type());
988 if (!content_info) {
989 continue;
deadbeefd59daf82015-10-14 15:02:44 -0700990 }
Steve Anton18ee1d52017-09-11 11:32:35 -0700991 const MediaContentDescription* content_desc =
992 static_cast<const MediaContentDescription*>(content_info->description);
993 if (content_desc && !content_info->rejected) {
994 bool success = (source == cricket::CS_LOCAL)
995 ? channel->SetLocalContent(content_desc, action, err)
996 : channel->SetRemoteContent(content_desc, action, err);
997 if (!success) {
998 all_success = false;
999 break;
1000 }
1001 }
1002 }
deadbeef953c2ce2017-01-09 14:53:41 -08001003 // Need complete offer/answer with an SCTP m= section before starting SCTP,
1004 // according to https://tools.ietf.org/html/draft-ietf-mmusic-sctp-sdp-19
1005 if (sctp_transport_ && local_description() && remote_description() &&
1006 cricket::GetFirstDataContent(local_description()->description()) &&
1007 cricket::GetFirstDataContent(remote_description()->description())) {
Steve Anton18ee1d52017-09-11 11:32:35 -07001008 all_success &= network_thread_->Invoke<bool>(
deadbeef953c2ce2017-01-09 14:53:41 -08001009 RTC_FROM_HERE,
1010 rtc::Bind(&WebRtcSession::PushdownSctpParameters_n, this, source));
1011 }
Steve Anton18ee1d52017-09-11 11:32:35 -07001012 return all_success;
deadbeef953c2ce2017-01-09 14:53:41 -08001013}
1014
1015bool WebRtcSession::PushdownSctpParameters_n(cricket::ContentSource source) {
1016 RTC_DCHECK(network_thread_->IsCurrent());
1017 RTC_DCHECK(local_description());
1018 RTC_DCHECK(remote_description());
1019 // Apply the SCTP port (which is hidden inside a DataCodec structure...)
1020 // When we support "max-message-size", that would also be pushed down here.
1021 return sctp_transport_->Start(
1022 GetSctpPort(local_description()->description()),
1023 GetSctpPort(remote_description()->description()));
deadbeefd59daf82015-10-14 15:02:44 -07001024}
1025
1026bool WebRtcSession::PushdownTransportDescription(cricket::ContentSource source,
1027 cricket::ContentAction action,
1028 std::string* error_desc) {
1029 RTC_DCHECK(signaling_thread()->IsCurrent());
1030
1031 if (source == cricket::CS_LOCAL) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001032 return PushdownLocalTransportDescription(local_description()->description(),
1033 action, error_desc);
deadbeefd59daf82015-10-14 15:02:44 -07001034 }
deadbeeffe4a8a42016-12-20 17:56:17 -08001035 return PushdownRemoteTransportDescription(remote_description()->description(),
1036 action, error_desc);
deadbeefd59daf82015-10-14 15:02:44 -07001037}
1038
1039bool WebRtcSession::PushdownLocalTransportDescription(
1040 const SessionDescription* sdesc,
1041 cricket::ContentAction action,
1042 std::string* err) {
1043 RTC_DCHECK(signaling_thread()->IsCurrent());
1044
1045 if (!sdesc) {
1046 return false;
1047 }
1048
1049 for (const TransportInfo& tinfo : sdesc->transport_infos()) {
1050 if (!transport_controller_->SetLocalTransportDescription(
1051 tinfo.content_name, tinfo.description, action, err)) {
1052 return false;
1053 }
1054 }
1055
1056 return true;
1057}
1058
1059bool WebRtcSession::PushdownRemoteTransportDescription(
1060 const SessionDescription* sdesc,
1061 cricket::ContentAction action,
1062 std::string* err) {
1063 RTC_DCHECK(signaling_thread()->IsCurrent());
1064
1065 if (!sdesc) {
1066 return false;
1067 }
1068
1069 for (const TransportInfo& tinfo : sdesc->transport_infos()) {
1070 if (!transport_controller_->SetRemoteTransportDescription(
1071 tinfo.content_name, tinfo.description, action, err)) {
1072 return false;
1073 }
1074 }
1075
1076 return true;
1077}
1078
1079bool WebRtcSession::GetTransportDescription(
1080 const SessionDescription* description,
1081 const std::string& content_name,
1082 cricket::TransportDescription* tdesc) {
1083 if (!description || !tdesc) {
1084 return false;
1085 }
1086 const TransportInfo* transport_info =
1087 description->GetTransportInfoByName(content_name);
1088 if (!transport_info) {
1089 return false;
1090 }
1091 *tdesc = transport_info->description;
1092 return true;
1093}
1094
deadbeefcbecd352015-09-23 11:50:27 -07001095bool WebRtcSession::EnableBundle(const cricket::ContentGroup& bundle) {
1096 const std::string* first_content_name = bundle.FirstContentName();
1097 if (!first_content_name) {
1098 LOG(LS_WARNING) << "Tried to BUNDLE with no contents.";
1099 return false;
1100 }
1101 const std::string& transport_name = *first_content_name;
deadbeefcbecd352015-09-23 11:50:27 -07001102
zhihuang9763d562016-08-05 11:14:50 -07001103#ifdef HAVE_QUIC
1104 if (quic_data_transport_ &&
1105 bundle.HasContentName(quic_data_transport_->content_name()) &&
1106 quic_data_transport_->transport_name() != transport_name) {
1107 LOG(LS_ERROR) << "Unable to BUNDLE " << quic_data_transport_->content_name()
1108 << " on " << transport_name << "with QUIC.";
1109 }
1110#endif
deadbeef953c2ce2017-01-09 14:53:41 -08001111 auto maybe_set_transport = [this, bundle,
1112 transport_name](cricket::BaseChannel* ch) {
deadbeefcbecd352015-09-23 11:50:27 -07001113 if (!ch || !bundle.HasContentName(ch->content_name())) {
pthatcher@webrtc.orgc04a97f2015-03-16 19:31:40 +00001114 return true;
1115 }
1116
zhihuangf5b251b2017-01-12 19:37:48 -08001117 std::string old_transport_name = ch->transport_name();
1118 if (old_transport_name == transport_name) {
deadbeefcbecd352015-09-23 11:50:27 -07001119 LOG(LS_INFO) << "BUNDLE already enabled for " << ch->content_name()
1120 << " on " << transport_name << ".";
1121 return true;
deadbeef47ee2f32015-09-22 15:08:23 -07001122 }
torbjornga81a42f2015-09-23 02:16:58 -07001123
zhihuangb2cdd932017-01-19 16:54:25 -08001124 cricket::DtlsTransportInternal* rtp_dtls_transport =
1125 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001126 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001127 bool need_rtcp = (ch->rtcp_dtls_transport() != nullptr);
1128 cricket::DtlsTransportInternal* rtcp_dtls_transport = nullptr;
zhihuangf5b251b2017-01-12 19:37:48 -08001129 if (need_rtcp) {
deadbeefd8cf08f2017-07-10 20:06:59 -07001130 rtcp_dtls_transport = transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001131 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
1132 }
1133
zhihuangb2cdd932017-01-19 16:54:25 -08001134 ch->SetTransports(rtp_dtls_transport, rtcp_dtls_transport);
deadbeefcbecd352015-09-23 11:50:27 -07001135 LOG(LS_INFO) << "Enabled BUNDLE for " << ch->content_name() << " on "
1136 << transport_name << ".";
zhihuangb2cdd932017-01-19 16:54:25 -08001137 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08001138 old_transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangf5b251b2017-01-12 19:37:48 -08001139 // If the channel needs rtcp, it means that the channel used to have a
1140 // rtcp transport which needs to be deleted now.
1141 if (need_rtcp) {
zhihuangb2cdd932017-01-19 16:54:25 -08001142 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08001143 old_transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
zhihuangf5b251b2017-01-12 19:37:48 -08001144 }
pthatcher@webrtc.orgc04a97f2015-03-16 19:31:40 +00001145 return true;
1146 };
1147
deadbeefcbecd352015-09-23 11:50:27 -07001148 if (!maybe_set_transport(voice_channel()) ||
1149 !maybe_set_transport(video_channel()) ||
deadbeef953c2ce2017-01-09 14:53:41 -08001150 !maybe_set_transport(rtp_data_channel())) {
deadbeefcbecd352015-09-23 11:50:27 -07001151 return false;
pthatcher@webrtc.orgc04a97f2015-03-16 19:31:40 +00001152 }
deadbeef953c2ce2017-01-09 14:53:41 -08001153 // For SCTP, transport creation/deletion happens here instead of in the
1154 // object itself.
1155 if (sctp_transport_) {
1156 RTC_DCHECK(sctp_transport_name_);
1157 RTC_DCHECK(sctp_content_name_);
1158 if (transport_name != *sctp_transport_name_ &&
1159 bundle.HasContentName(*sctp_content_name_)) {
1160 network_thread_->Invoke<void>(
1161 RTC_FROM_HERE, rtc::Bind(&WebRtcSession::ChangeSctpTransport_n, this,
1162 transport_name));
1163 }
1164 }
deadbeefcbecd352015-09-23 11:50:27 -07001165
pthatcher@webrtc.orgc04a97f2015-03-16 19:31:40 +00001166 return true;
1167}
1168
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001169bool WebRtcSession::ProcessIceMessage(const IceCandidateInterface* candidate) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001170 if (!remote_description()) {
deadbeefd59daf82015-10-14 15:02:44 -07001171 LOG(LS_ERROR) << "ProcessIceMessage: ICE candidates can't be added "
1172 << "without any remote session description.";
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001173 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001174 }
1175
1176 if (!candidate) {
deadbeefd59daf82015-10-14 15:02:44 -07001177 LOG(LS_ERROR) << "ProcessIceMessage: Candidate is NULL.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001178 return false;
1179 }
1180
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00001181 bool valid = false;
deadbeefd59daf82015-10-14 15:02:44 -07001182 bool ready = ReadyToUseRemoteCandidate(candidate, NULL, &valid);
1183 if (!valid) {
1184 return false;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001185 }
1186
1187 // Add this candidate to the remote session description.
deadbeeffe4a8a42016-12-20 17:56:17 -08001188 if (!mutable_remote_description()->AddCandidate(candidate)) {
deadbeefd59daf82015-10-14 15:02:44 -07001189 LOG(LS_ERROR) << "ProcessIceMessage: Candidate cannot be used.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001190 return false;
1191 }
1192
deadbeefd59daf82015-10-14 15:02:44 -07001193 if (ready) {
1194 return UseCandidate(candidate);
1195 } else {
1196 LOG(LS_INFO) << "ProcessIceMessage: Not ready to use candidate.";
1197 return true;
1198 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001199}
1200
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001201bool WebRtcSession::RemoveRemoteIceCandidates(
1202 const std::vector<cricket::Candidate>& candidates) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001203 if (!remote_description()) {
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001204 LOG(LS_ERROR) << "RemoveRemoteIceCandidates: ICE candidates can't be "
1205 << "removed without any remote session description.";
1206 return false;
1207 }
1208
1209 if (candidates.empty()) {
1210 LOG(LS_ERROR) << "RemoveRemoteIceCandidates: candidates are empty.";
1211 return false;
1212 }
1213
deadbeeffe4a8a42016-12-20 17:56:17 -08001214 size_t number_removed =
1215 mutable_remote_description()->RemoveCandidates(candidates);
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001216 if (number_removed != candidates.size()) {
1217 LOG(LS_ERROR) << "RemoveRemoteIceCandidates: Failed to remove candidates. "
1218 << "Requested " << candidates.size() << " but only "
1219 << number_removed << " are removed.";
1220 }
1221
1222 // Remove the candidates from the transport controller.
1223 std::string error;
1224 bool res = transport_controller_->RemoveRemoteCandidates(candidates, &error);
1225 if (!res && !error.empty()) {
1226 LOG(LS_ERROR) << "Error when removing remote candidates: " << error;
1227 }
1228 return true;
1229}
1230
deadbeefd59daf82015-10-14 15:02:44 -07001231cricket::IceConfig WebRtcSession::ParseIceConfig(
1232 const PeerConnectionInterface::RTCConfiguration& config) const {
Honghai Zhang5622c5e2016-07-01 13:59:29 -07001233 cricket::ContinualGatheringPolicy gathering_policy;
1234 // TODO(honghaiz): Add the third continual gathering policy in
1235 // PeerConnectionInterface and map it to GATHER_CONTINUALLY_AND_RECOVER.
1236 switch (config.continual_gathering_policy) {
1237 case PeerConnectionInterface::GATHER_ONCE:
1238 gathering_policy = cricket::GATHER_ONCE;
1239 break;
1240 case PeerConnectionInterface::GATHER_CONTINUALLY:
1241 gathering_policy = cricket::GATHER_CONTINUALLY;
1242 break;
1243 default:
nisseeb4ca4e2017-01-12 02:24:27 -08001244 RTC_NOTREACHED();
Honghai Zhang5622c5e2016-07-01 13:59:29 -07001245 gathering_policy = cricket::GATHER_ONCE;
1246 }
deadbeefd59daf82015-10-14 15:02:44 -07001247 cricket::IceConfig ice_config;
Honghai Zhang049fbb12016-03-07 11:13:07 -08001248 ice_config.receiving_timeout = config.ice_connection_receiving_timeout;
guoweis36f01372016-03-02 18:02:40 -08001249 ice_config.prioritize_most_likely_candidate_pairs =
1250 config.prioritize_most_likely_ice_candidate_pairs;
Honghai Zhang381b4212015-12-04 12:24:03 -08001251 ice_config.backup_connection_ping_interval =
1252 config.ice_backup_candidate_pair_ping_interval;
Honghai Zhang5622c5e2016-07-01 13:59:29 -07001253 ice_config.continual_gathering_policy = gathering_policy;
Taylor Brandstettere9851112016-07-01 11:11:13 -07001254 ice_config.presume_writable_when_fully_relayed =
1255 config.presume_writable_when_fully_relayed;
skvlad51072462017-02-02 11:50:14 -08001256 ice_config.ice_check_min_interval = config.ice_check_min_interval;
Steve Anton300bf8e2017-07-14 10:13:10 -07001257 ice_config.regather_all_networks_interval_range =
1258 config.ice_regather_interval_range;
deadbeefd59daf82015-10-14 15:02:44 -07001259 return ice_config;
1260}
1261
1262void WebRtcSession::SetIceConfig(const cricket::IceConfig& config) {
1263 transport_controller_->SetIceConfig(config);
1264}
1265
1266void WebRtcSession::MaybeStartGathering() {
1267 transport_controller_->MaybeStartGathering();
1268}
1269
Peter Boström0c4e06b2015-10-07 12:23:21 +02001270bool WebRtcSession::GetLocalTrackIdBySsrc(uint32_t ssrc,
1271 std::string* track_id) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001272 if (!local_description()) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001273 return false;
deadbeefd59daf82015-10-14 15:02:44 -07001274 }
deadbeeffe4a8a42016-12-20 17:56:17 -08001275 return webrtc::GetTrackIdBySsrc(local_description()->description(), ssrc,
1276 track_id);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001277}
1278
Peter Boström0c4e06b2015-10-07 12:23:21 +02001279bool WebRtcSession::GetRemoteTrackIdBySsrc(uint32_t ssrc,
1280 std::string* track_id) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001281 if (!remote_description()) {
xians@webrtc.org4cb01282014-06-12 14:57:05 +00001282 return false;
deadbeefd59daf82015-10-14 15:02:44 -07001283 }
deadbeeffe4a8a42016-12-20 17:56:17 -08001284 return webrtc::GetTrackIdBySsrc(remote_description()->description(), ssrc,
1285 track_id);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001286}
1287
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001288std::string WebRtcSession::BadStateErrMsg(State state) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001289 std::ostringstream desc;
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00001290 desc << "Called in wrong state: " << GetStateString(state);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001291 return desc.str();
1292}
1293
wu@webrtc.org78187522013-10-07 23:32:02 +00001294bool WebRtcSession::SendData(const cricket::SendDataParams& params,
jbaucheec21bd2016-03-20 06:15:43 -07001295 const rtc::CopyOnWriteBuffer& payload,
wu@webrtc.org78187522013-10-07 23:32:02 +00001296 cricket::SendDataResult* result) {
deadbeef953c2ce2017-01-09 14:53:41 -08001297 if (!rtp_data_channel_ && !sctp_transport_) {
1298 LOG(LS_ERROR) << "SendData called when rtp_data_channel_ "
1299 << "and sctp_transport_ are NULL.";
wu@webrtc.org78187522013-10-07 23:32:02 +00001300 return false;
1301 }
deadbeef953c2ce2017-01-09 14:53:41 -08001302 return rtp_data_channel_
1303 ? rtp_data_channel_->SendData(params, payload, result)
1304 : network_thread_->Invoke<bool>(
1305 RTC_FROM_HERE,
1306 Bind(&cricket::SctpTransportInternal::SendData,
1307 sctp_transport_.get(), params, payload, result));
wu@webrtc.org78187522013-10-07 23:32:02 +00001308}
1309
1310bool WebRtcSession::ConnectDataChannel(DataChannel* webrtc_data_channel) {
deadbeef953c2ce2017-01-09 14:53:41 -08001311 if (!rtp_data_channel_ && !sctp_transport_) {
deadbeefdaf88b12016-10-05 22:29:30 -07001312 // Don't log an error here, because DataChannels are expected to call
1313 // ConnectDataChannel in this state. It's the only way to initially tell
1314 // whether or not the underlying transport is ready.
wu@webrtc.org78187522013-10-07 23:32:02 +00001315 return false;
1316 }
deadbeef953c2ce2017-01-09 14:53:41 -08001317 if (rtp_data_channel_) {
1318 rtp_data_channel_->SignalReadyToSendData.connect(
1319 webrtc_data_channel, &DataChannel::OnChannelReady);
1320 rtp_data_channel_->SignalDataReceived.connect(webrtc_data_channel,
1321 &DataChannel::OnDataReceived);
1322 } else {
1323 SignalSctpReadyToSendData.connect(webrtc_data_channel,
1324 &DataChannel::OnChannelReady);
1325 SignalSctpDataReceived.connect(webrtc_data_channel,
1326 &DataChannel::OnDataReceived);
1327 SignalSctpStreamClosedRemotely.connect(
1328 webrtc_data_channel, &DataChannel::OnStreamClosedRemotely);
1329 }
wu@webrtc.org78187522013-10-07 23:32:02 +00001330 return true;
1331}
1332
1333void WebRtcSession::DisconnectDataChannel(DataChannel* webrtc_data_channel) {
deadbeef953c2ce2017-01-09 14:53:41 -08001334 if (!rtp_data_channel_ && !sctp_transport_) {
1335 LOG(LS_ERROR) << "DisconnectDataChannel called when rtp_data_channel_ and "
1336 "sctp_transport_ are NULL.";
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001337 return;
1338 }
deadbeef953c2ce2017-01-09 14:53:41 -08001339 if (rtp_data_channel_) {
1340 rtp_data_channel_->SignalReadyToSendData.disconnect(webrtc_data_channel);
1341 rtp_data_channel_->SignalDataReceived.disconnect(webrtc_data_channel);
1342 } else {
1343 SignalSctpReadyToSendData.disconnect(webrtc_data_channel);
1344 SignalSctpDataReceived.disconnect(webrtc_data_channel);
1345 SignalSctpStreamClosedRemotely.disconnect(webrtc_data_channel);
1346 }
wu@webrtc.org78187522013-10-07 23:32:02 +00001347}
1348
bemasc@webrtc.org9b5467e2014-12-04 23:16:52 +00001349void WebRtcSession::AddSctpDataStream(int sid) {
deadbeef953c2ce2017-01-09 14:53:41 -08001350 if (!sctp_transport_) {
1351 LOG(LS_ERROR) << "AddSctpDataStream called when sctp_transport_ is NULL.";
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001352 return;
1353 }
deadbeef953c2ce2017-01-09 14:53:41 -08001354 network_thread_->Invoke<void>(
1355 RTC_FROM_HERE, rtc::Bind(&cricket::SctpTransportInternal::OpenStream,
1356 sctp_transport_.get(), sid));
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001357}
1358
bemasc@webrtc.org9b5467e2014-12-04 23:16:52 +00001359void WebRtcSession::RemoveSctpDataStream(int sid) {
deadbeef953c2ce2017-01-09 14:53:41 -08001360 if (!sctp_transport_) {
1361 LOG(LS_ERROR) << "RemoveSctpDataStream called when sctp_transport_ is "
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001362 << "NULL.";
1363 return;
1364 }
deadbeef953c2ce2017-01-09 14:53:41 -08001365 network_thread_->Invoke<void>(
1366 RTC_FROM_HERE, rtc::Bind(&cricket::SctpTransportInternal::ResetStream,
1367 sctp_transport_.get(), sid));
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001368}
1369
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00001370bool WebRtcSession::ReadyToSendData() const {
deadbeef953c2ce2017-01-09 14:53:41 -08001371 return (rtp_data_channel_ && rtp_data_channel_->ready_to_send_data()) ||
1372 sctp_ready_to_send_data_;
1373}
1374
1375std::unique_ptr<SessionStats> WebRtcSession::GetStats_s() {
nisseede5da42017-01-12 05:15:36 -08001376 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeef953c2ce2017-01-09 14:53:41 -08001377 ChannelNamePairs channel_name_pairs;
1378 if (voice_channel()) {
1379 channel_name_pairs.voice = rtc::Optional<ChannelNamePair>(ChannelNamePair(
1380 voice_channel()->content_name(), voice_channel()->transport_name()));
1381 }
1382 if (video_channel()) {
1383 channel_name_pairs.video = rtc::Optional<ChannelNamePair>(ChannelNamePair(
1384 video_channel()->content_name(), video_channel()->transport_name()));
1385 }
1386 if (rtp_data_channel()) {
1387 channel_name_pairs.data = rtc::Optional<ChannelNamePair>(
1388 ChannelNamePair(rtp_data_channel()->content_name(),
1389 rtp_data_channel()->transport_name()));
1390 }
1391 if (sctp_transport_) {
1392 RTC_DCHECK(sctp_content_name_);
1393 RTC_DCHECK(sctp_transport_name_);
1394 channel_name_pairs.data = rtc::Optional<ChannelNamePair>(
1395 ChannelNamePair(*sctp_content_name_, *sctp_transport_name_));
1396 }
1397 return GetStats(channel_name_pairs);
1398}
1399
1400std::unique_ptr<SessionStats> WebRtcSession::GetStats(
1401 const ChannelNamePairs& channel_name_pairs) {
1402 if (network_thread()->IsCurrent()) {
1403 return GetStats_n(channel_name_pairs);
1404 }
1405 return network_thread()->Invoke<std::unique_ptr<SessionStats>>(
1406 RTC_FROM_HERE,
1407 rtc::Bind(&WebRtcSession::GetStats_n, this, channel_name_pairs));
1408}
1409
1410bool WebRtcSession::GetLocalCertificate(
1411 const std::string& transport_name,
1412 rtc::scoped_refptr<rtc::RTCCertificate>* certificate) {
1413 return transport_controller_->GetLocalCertificate(transport_name,
1414 certificate);
1415}
1416
1417std::unique_ptr<rtc::SSLCertificate> WebRtcSession::GetRemoteSSLCertificate(
1418 const std::string& transport_name) {
1419 return transport_controller_->GetRemoteSSLCertificate(transport_name);
wu@webrtc.org07a6fbe2013-11-04 18:41:34 +00001420}
1421
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001422cricket::DataChannelType WebRtcSession::data_channel_type() const {
1423 return data_channel_type_;
1424}
1425
deadbeef0ed85b22016-02-23 17:24:52 -08001426bool WebRtcSession::IceRestartPending(const std::string& content_name) const {
1427 return pending_ice_restarts_.find(content_name) !=
1428 pending_ice_restarts_.end();
wu@webrtc.org91053e72013-08-10 07:18:04 +00001429}
1430
deadbeefd1a38b52016-12-10 13:15:33 -08001431void WebRtcSession::SetNeedsIceRestartFlag() {
1432 transport_controller_->SetNeedsIceRestartFlag();
1433}
1434
1435bool WebRtcSession::NeedsIceRestart(const std::string& content_name) const {
1436 return transport_controller_->NeedsIceRestart(content_name);
1437}
1438
Henrik Boströmd8281982015-08-27 10:12:24 +02001439void WebRtcSession::OnCertificateReady(
1440 const rtc::scoped_refptr<rtc::RTCCertificate>& certificate) {
deadbeefd59daf82015-10-14 15:02:44 -07001441 transport_controller_->SetLocalCertificate(certificate);
wu@webrtc.org91053e72013-08-10 07:18:04 +00001442}
1443
deadbeef953c2ce2017-01-09 14:53:41 -08001444void WebRtcSession::OnDtlsSrtpSetupFailure(cricket::BaseChannel*, bool rtcp) {
1445 SetError(ERROR_TRANSPORT,
1446 rtcp ? kDtlsSrtpSetupFailureRtcp : kDtlsSrtpSetupFailureRtp);
1447}
1448
Henrik Boströmd8281982015-08-27 10:12:24 +02001449bool WebRtcSession::waiting_for_certificate_for_testing() const {
Henrik Boström87713d02015-08-25 09:53:21 +02001450 return webrtc_session_desc_factory_->waiting_for_certificate_for_testing();
wu@webrtc.org91053e72013-08-10 07:18:04 +00001451}
1452
deadbeefcbecd352015-09-23 11:50:27 -07001453const rtc::scoped_refptr<rtc::RTCCertificate>&
1454WebRtcSession::certificate_for_testing() {
deadbeefd59daf82015-10-14 15:02:44 -07001455 return transport_controller_->certificate_for_testing();
deadbeefcbecd352015-09-23 11:50:27 -07001456}
1457
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001458void WebRtcSession::SetIceConnectionState(
1459 PeerConnectionInterface::IceConnectionState state) {
1460 if (ice_connection_state_ == state) {
1461 return;
1462 }
1463
deadbeefcbecd352015-09-23 11:50:27 -07001464 LOG(LS_INFO) << "Changing IceConnectionState " << ice_connection_state_
1465 << " => " << state;
Taylor Brandstetter6aefc632016-05-26 16:08:23 -07001466 RTC_DCHECK(ice_connection_state_ !=
1467 PeerConnectionInterface::kIceConnectionClosed);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001468 ice_connection_state_ = state;
1469 if (ice_observer_) {
zstein6dfd53a2017-03-06 13:49:03 -08001470 ice_observer_->OnIceConnectionStateChange(ice_connection_state_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001471 }
1472}
1473
deadbeefcbecd352015-09-23 11:50:27 -07001474void WebRtcSession::OnTransportControllerConnectionState(
1475 cricket::IceConnectionState state) {
1476 switch (state) {
1477 case cricket::kIceConnectionConnecting:
1478 // If the current state is Connected or Completed, then there were
1479 // writable channels but now there are not, so the next state must
1480 // be Disconnected.
1481 // kIceConnectionConnecting is currently used as the default,
1482 // un-connected state by the TransportController, so its only use is
1483 // detecting disconnections.
1484 if (ice_connection_state_ ==
1485 PeerConnectionInterface::kIceConnectionConnected ||
1486 ice_connection_state_ ==
1487 PeerConnectionInterface::kIceConnectionCompleted) {
1488 SetIceConnectionState(
1489 PeerConnectionInterface::kIceConnectionDisconnected);
1490 }
torbjornga81a42f2015-09-23 02:16:58 -07001491 break;
deadbeefcbecd352015-09-23 11:50:27 -07001492 case cricket::kIceConnectionFailed:
1493 SetIceConnectionState(PeerConnectionInterface::kIceConnectionFailed);
1494 break;
1495 case cricket::kIceConnectionConnected:
1496 LOG(LS_INFO) << "Changing to ICE connected state because "
1497 << "all transports are writable.";
1498 SetIceConnectionState(PeerConnectionInterface::kIceConnectionConnected);
1499 break;
1500 case cricket::kIceConnectionCompleted:
1501 LOG(LS_INFO) << "Changing to ICE completed state because "
1502 << "all transports are complete.";
1503 if (ice_connection_state_ !=
1504 PeerConnectionInterface::kIceConnectionConnected) {
1505 // If jumping directly from "checking" to "connected",
1506 // signal "connected" first.
1507 SetIceConnectionState(PeerConnectionInterface::kIceConnectionConnected);
1508 }
1509 SetIceConnectionState(PeerConnectionInterface::kIceConnectionCompleted);
1510 if (metrics_observer_) {
1511 ReportTransportStats();
1512 }
1513 break;
1514 default:
nissec80e7412017-01-11 05:56:46 -08001515 RTC_NOTREACHED();
torbjornga81a42f2015-09-23 02:16:58 -07001516 }
deadbeefcbecd352015-09-23 11:50:27 -07001517}
1518
1519void WebRtcSession::OnTransportControllerReceiving(bool receiving) {
Peter Thatcher54360512015-07-08 11:08:35 -07001520 SetIceConnectionReceiving(receiving);
1521}
1522
1523void WebRtcSession::SetIceConnectionReceiving(bool receiving) {
1524 if (ice_connection_receiving_ == receiving) {
1525 return;
1526 }
1527 ice_connection_receiving_ = receiving;
1528 if (ice_observer_) {
1529 ice_observer_->OnIceConnectionReceivingChange(receiving);
1530 }
1531}
1532
deadbeefcbecd352015-09-23 11:50:27 -07001533void WebRtcSession::OnTransportControllerCandidatesGathered(
1534 const std::string& transport_name,
1535 const cricket::Candidates& candidates) {
nisseede5da42017-01-12 05:15:36 -08001536 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -07001537 int sdp_mline_index;
1538 if (!GetLocalCandidateMediaIndex(transport_name, &sdp_mline_index)) {
1539 LOG(LS_ERROR) << "OnTransportControllerCandidatesGathered: content name "
1540 << transport_name << " not found";
1541 return;
1542 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001543
deadbeefcbecd352015-09-23 11:50:27 -07001544 for (cricket::Candidates::const_iterator citer = candidates.begin();
1545 citer != candidates.end(); ++citer) {
1546 // Use transport_name as the candidate media id.
jbauch81bf7b02017-03-25 08:31:12 -07001547 std::unique_ptr<JsepIceCandidate> candidate(
1548 new JsepIceCandidate(transport_name, sdp_mline_index, *citer));
deadbeeffe4a8a42016-12-20 17:56:17 -08001549 if (local_description()) {
jbauch81bf7b02017-03-25 08:31:12 -07001550 mutable_local_description()->AddCandidate(candidate.get());
1551 }
1552 if (ice_observer_) {
1553 ice_observer_->OnIceCandidate(std::move(candidate));
deadbeefcbecd352015-09-23 11:50:27 -07001554 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001555 }
1556}
1557
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001558void WebRtcSession::OnTransportControllerCandidatesRemoved(
1559 const std::vector<cricket::Candidate>& candidates) {
nisseede5da42017-01-12 05:15:36 -08001560 RTC_DCHECK(signaling_thread()->IsCurrent());
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001561 // Sanity check.
1562 for (const cricket::Candidate& candidate : candidates) {
1563 if (candidate.transport_name().empty()) {
1564 LOG(LS_ERROR) << "OnTransportControllerCandidatesRemoved: "
1565 << "empty content name in candidate "
1566 << candidate.ToString();
1567 return;
1568 }
1569 }
1570
deadbeeffe4a8a42016-12-20 17:56:17 -08001571 if (local_description()) {
1572 mutable_local_description()->RemoveCandidates(candidates);
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001573 }
1574 if (ice_observer_) {
1575 ice_observer_->OnIceCandidatesRemoved(candidates);
1576 }
1577}
1578
deadbeef953c2ce2017-01-09 14:53:41 -08001579void WebRtcSession::OnTransportControllerDtlsHandshakeError(
1580 rtc::SSLHandshakeError error) {
1581 if (metrics_observer_) {
1582 metrics_observer_->IncrementEnumCounter(
1583 webrtc::kEnumCounterDtlsHandshakeError, static_cast<int>(error),
1584 static_cast<int>(rtc::SSLHandshakeError::MAX_VALUE));
1585 }
1586}
1587
Steve Anton169629a2017-08-30 17:36:36 -07001588// Enabling voice and video (and RTP data) channels.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001589void WebRtcSession::EnableChannels() {
Steve Anton169629a2017-08-30 17:36:36 -07001590 for (cricket::VoiceChannel* voice_channel : voice_channels_) {
1591 if (!voice_channel->enabled()) {
1592 voice_channel->Enable(true);
1593 }
1594 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001595
Steve Anton169629a2017-08-30 17:36:36 -07001596 for (cricket::VideoChannel* video_channel : video_channels_) {
1597 if (!video_channel->enabled()) {
1598 video_channel->Enable(true);
1599 }
1600 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001601
deadbeef953c2ce2017-01-09 14:53:41 -08001602 if (rtp_data_channel_ && !rtp_data_channel_->enabled())
1603 rtp_data_channel_->Enable(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001604}
1605
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001606// Returns the media index for a local ice candidate given the content name.
1607bool WebRtcSession::GetLocalCandidateMediaIndex(const std::string& content_name,
1608 int* sdp_mline_index) {
deadbeeffe4a8a42016-12-20 17:56:17 -08001609 if (!local_description() || !sdp_mline_index) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001610 return false;
deadbeefd59daf82015-10-14 15:02:44 -07001611 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001612
1613 bool content_found = false;
deadbeeffe4a8a42016-12-20 17:56:17 -08001614 const ContentInfos& contents = local_description()->description()->contents();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001615 for (size_t index = 0; index < contents.size(); ++index) {
1616 if (contents[index].name == content_name) {
henrike@webrtc.org28654cb2013-07-22 21:07:49 +00001617 *sdp_mline_index = static_cast<int>(index);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001618 content_found = true;
1619 break;
1620 }
1621 }
1622 return content_found;
1623}
1624
1625bool WebRtcSession::UseCandidatesInSessionDescription(
1626 const SessionDescriptionInterface* remote_desc) {
deadbeefd59daf82015-10-14 15:02:44 -07001627 if (!remote_desc) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001628 return true;
deadbeefd59daf82015-10-14 15:02:44 -07001629 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001630 bool ret = true;
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00001631
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001632 for (size_t m = 0; m < remote_desc->number_of_mediasections(); ++m) {
1633 const IceCandidateCollection* candidates = remote_desc->candidates(m);
deadbeefd59daf82015-10-14 15:02:44 -07001634 for (size_t n = 0; n < candidates->count(); ++n) {
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00001635 const IceCandidateInterface* candidate = candidates->at(n);
1636 bool valid = false;
1637 if (!ReadyToUseRemoteCandidate(candidate, remote_desc, &valid)) {
1638 if (valid) {
deadbeefd59daf82015-10-14 15:02:44 -07001639 LOG(LS_INFO) << "UseCandidatesInSessionDescription: Not ready to use "
1640 << "candidate.";
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00001641 }
1642 continue;
1643 }
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00001644 ret = UseCandidate(candidate);
deadbeefd59daf82015-10-14 15:02:44 -07001645 if (!ret) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001646 break;
deadbeefd59daf82015-10-14 15:02:44 -07001647 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001648 }
1649 }
1650 return ret;
1651}
1652
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001653bool WebRtcSession::UseCandidate(const IceCandidateInterface* candidate) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001654 size_t mediacontent_index = static_cast<size_t>(candidate->sdp_mline_index());
deadbeeffe4a8a42016-12-20 17:56:17 -08001655 size_t remote_content_size =
1656 remote_description()->description()->contents().size();
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001657 if (mediacontent_index >= remote_content_size) {
Honghai Zhang7fb69db2016-03-14 11:59:18 -07001658 LOG(LS_ERROR) << "UseCandidate: Invalid candidate media index.";
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001659 return false;
1660 }
1661
1662 cricket::ContentInfo content =
deadbeeffe4a8a42016-12-20 17:56:17 -08001663 remote_description()->description()->contents()[mediacontent_index];
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001664 std::vector<cricket::Candidate> candidates;
1665 candidates.push_back(candidate->candidate());
1666 // Invoking BaseSession method to handle remote candidates.
1667 std::string error;
deadbeefd59daf82015-10-14 15:02:44 -07001668 if (transport_controller_->AddRemoteCandidates(content.name, candidates,
1669 &error)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001670 // Candidates successfully submitted for checking.
1671 if (ice_connection_state_ == PeerConnectionInterface::kIceConnectionNew ||
1672 ice_connection_state_ ==
1673 PeerConnectionInterface::kIceConnectionDisconnected) {
1674 // If state is New, then the session has just gotten its first remote ICE
1675 // candidates, so go to Checking.
1676 // If state is Disconnected, the session is re-using old candidates or
1677 // receiving additional ones, so go to Checking.
1678 // If state is Connected, stay Connected.
1679 // TODO(bemasc): If state is Connected, and the new candidates are for a
1680 // newly added transport, then the state actually _should_ move to
1681 // checking. Add a way to distinguish that case.
1682 SetIceConnectionState(PeerConnectionInterface::kIceConnectionChecking);
1683 }
1684 // TODO(bemasc): If state is Completed, go back to Connected.
1685 } else {
fischman@webrtc.org4f2bd682014-03-28 18:13:34 +00001686 if (!error.empty()) {
1687 LOG(LS_WARNING) << error;
1688 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001689 }
1690 return true;
1691}
1692
deadbeefcbecd352015-09-23 11:50:27 -07001693void WebRtcSession::RemoveUnusedChannels(const SessionDescription* desc) {
Steve Anton169629a2017-08-30 17:36:36 -07001694 // TODO(steveanton): Add support for multiple audio/video channels.
1695 // Destroy video channel first since it may have a pointer to the
1696 // voice channel.
1697 const cricket::ContentInfo* video_info = cricket::GetFirstVideoContent(desc);
1698 if ((!video_info || video_info->rejected) && video_channel()) {
1699 RemoveAndDestroyVideoChannel(video_channel());
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001700 }
1701
Steve Anton169629a2017-08-30 17:36:36 -07001702 const cricket::ContentInfo* voice_info = cricket::GetFirstAudioContent(desc);
1703 if ((!voice_info || voice_info->rejected) && voice_channel()) {
1704 RemoveAndDestroyVoiceChannel(voice_channel());
buildbot@webrtc.orgb4c7b092014-08-25 12:11:58 +00001705 }
1706
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001707 const cricket::ContentInfo* data_info =
1708 cricket::GetFirstDataContent(desc);
zhihuang9763d562016-08-05 11:14:50 -07001709 if (!data_info || data_info->rejected) {
deadbeef953c2ce2017-01-09 14:53:41 -08001710 if (rtp_data_channel_) {
zhihuangf5b251b2017-01-12 19:37:48 -08001711 DestroyDataChannel();
deadbeef953c2ce2017-01-09 14:53:41 -08001712 }
1713 if (sctp_transport_) {
1714 SignalDataChannelDestroyed();
1715 network_thread_->Invoke<void>(
1716 RTC_FROM_HERE,
1717 rtc::Bind(&WebRtcSession::DestroySctpTransport_n, this));
zhihuang9763d562016-08-05 11:14:50 -07001718 }
1719#ifdef HAVE_QUIC
1720 // Clean up the existing QuicDataTransport and its QuicTransportChannels.
1721 if (quic_data_transport_) {
1722 quic_data_transport_.reset();
1723 }
1724#endif
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001725 }
1726}
1727
skvlad6c87a672016-05-17 17:49:52 -07001728// Returns the name of the transport channel when BUNDLE is enabled, or nullptr
1729// if the channel is not part of any bundle.
1730const std::string* WebRtcSession::GetBundleTransportName(
1731 const cricket::ContentInfo* content,
1732 const cricket::ContentGroup* bundle) {
1733 if (!bundle) {
1734 return nullptr;
1735 }
1736 const std::string* first_content_name = bundle->FirstContentName();
1737 if (!first_content_name) {
1738 LOG(LS_WARNING) << "Tried to BUNDLE with no contents.";
1739 return nullptr;
1740 }
1741 if (!bundle->HasContentName(content->name)) {
1742 LOG(LS_WARNING) << content->name << " is not part of any bundle group";
1743 return nullptr;
1744 }
1745 LOG(LS_INFO) << "Bundling " << content->name << " on " << *first_content_name;
1746 return first_content_name;
1747}
1748
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001749bool WebRtcSession::CreateChannels(const SessionDescription* desc) {
Steve Anton169629a2017-08-30 17:36:36 -07001750 // TODO(steveanton): Add support for multiple audio/video channels.
skvlad6c87a672016-05-17 17:49:52 -07001751 const cricket::ContentGroup* bundle_group = nullptr;
1752 if (bundle_policy_ == PeerConnectionInterface::kBundlePolicyMaxBundle) {
1753 bundle_group = desc->GetGroupByName(cricket::GROUP_TYPE_BUNDLE);
1754 if (!bundle_group) {
1755 LOG(LS_WARNING) << "max-bundle specified without BUNDLE specified";
1756 return false;
1757 }
1758 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001759 // Creating the media channels and transport proxies.
1760 const cricket::ContentInfo* voice = cricket::GetFirstAudioContent(desc);
Steve Anton169629a2017-08-30 17:36:36 -07001761 if (voice && !voice->rejected && !voice_channel()) {
skvlad6c87a672016-05-17 17:49:52 -07001762 if (!CreateVoiceChannel(voice,
1763 GetBundleTransportName(voice, bundle_group))) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001764 LOG(LS_ERROR) << "Failed to create voice channel.";
1765 return false;
1766 }
1767 }
1768
1769 const cricket::ContentInfo* video = cricket::GetFirstVideoContent(desc);
Steve Anton169629a2017-08-30 17:36:36 -07001770 if (video && !video->rejected && !video_channel()) {
skvlad6c87a672016-05-17 17:49:52 -07001771 if (!CreateVideoChannel(video,
1772 GetBundleTransportName(video, bundle_group))) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001773 LOG(LS_ERROR) << "Failed to create video channel.";
1774 return false;
1775 }
1776 }
1777
1778 const cricket::ContentInfo* data = cricket::GetFirstDataContent(desc);
deadbeef953c2ce2017-01-09 14:53:41 -08001779 if (data_channel_type_ != cricket::DCT_NONE && data && !data->rejected &&
1780 !rtp_data_channel_ && !sctp_transport_) {
skvlad6c87a672016-05-17 17:49:52 -07001781 if (!CreateDataChannel(data, GetBundleTransportName(data, bundle_group))) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001782 LOG(LS_ERROR) << "Failed to create data channel.";
1783 return false;
1784 }
1785 }
1786
1787 return true;
1788}
1789
skvlad6c87a672016-05-17 17:49:52 -07001790bool WebRtcSession::CreateVoiceChannel(const cricket::ContentInfo* content,
1791 const std::string* bundle_transport) {
Steve Anton169629a2017-08-30 17:36:36 -07001792 // TODO(steveanton): Check to see if it's safe to create multiple voice
1793 // channels.
1794 RTC_DCHECK(voice_channels_.empty());
1795
skvlad6c87a672016-05-17 17:49:52 -07001796 bool require_rtcp_mux =
1797 rtcp_mux_policy_ == PeerConnectionInterface::kRtcpMuxPolicyRequire;
zhihuangf5b251b2017-01-12 19:37:48 -08001798
1799 std::string transport_name =
1800 bundle_transport ? *bundle_transport : content->name;
1801
zhihuangb2cdd932017-01-19 16:54:25 -08001802 cricket::DtlsTransportInternal* rtp_dtls_transport =
1803 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001804 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001805 cricket::DtlsTransportInternal* rtcp_dtls_transport = nullptr;
deadbeefac22f702017-01-12 21:59:29 -08001806 if (!require_rtcp_mux) {
zhihuangb2cdd932017-01-19 16:54:25 -08001807 rtcp_dtls_transport = transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001808 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
1809 }
1810
Steve Anton169629a2017-08-30 17:36:36 -07001811 cricket::VoiceChannel* voice_channel = channel_manager_->CreateVoiceChannel(
nisseeaabdf62017-05-05 02:23:02 -07001812 call_, media_config_, rtp_dtls_transport, rtcp_dtls_transport,
deadbeef1a2183d2017-02-10 23:44:49 -08001813 transport_controller_->signaling_thread(), content->name, SrtpRequired(),
Steve Anton169629a2017-08-30 17:36:36 -07001814 audio_options_);
1815 if (!voice_channel) {
zhihuangb2cdd932017-01-19 16:54:25 -08001816 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08001817 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001818 if (rtcp_dtls_transport) {
1819 transport_controller_->DestroyDtlsTransport(
Steve Anton24efa722017-08-29 18:12:51 -07001820 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
deadbeefbad5dad2017-01-17 18:32:35 -08001821 }
wu@webrtc.orgde305012013-10-31 15:40:38 +00001822 return false;
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001823 }
zhihuangf5b251b2017-01-12 19:37:48 -08001824
Steve Anton169629a2017-08-30 17:36:36 -07001825 voice_channels_.push_back(voice_channel);
1826
1827 voice_channel->SignalRtcpMuxFullyActive.connect(
deadbeefac22f702017-01-12 21:59:29 -08001828 this, &WebRtcSession::DestroyRtcpTransport_n);
Steve Anton169629a2017-08-30 17:36:36 -07001829 voice_channel->SignalDtlsSrtpSetupFailure.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001830 this, &WebRtcSession::OnDtlsSrtpSetupFailure);
deadbeefab9b2d12015-10-14 11:33:11 -07001831
Steve Anton169629a2017-08-30 17:36:36 -07001832 // TODO(steveanton): This should signal which voice channel was created since
1833 // we can have multiple.
deadbeefab9b2d12015-10-14 11:33:11 -07001834 SignalVoiceChannelCreated();
Steve Anton169629a2017-08-30 17:36:36 -07001835 voice_channel->SignalSentPacket.connect(this, &WebRtcSession::OnSentPacket_w);
wu@webrtc.orgde305012013-10-31 15:40:38 +00001836 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001837}
1838
skvlad6c87a672016-05-17 17:49:52 -07001839bool WebRtcSession::CreateVideoChannel(const cricket::ContentInfo* content,
1840 const std::string* bundle_transport) {
Steve Anton169629a2017-08-30 17:36:36 -07001841 // TODO(steveanton): Check to see if it's safe to create multiple video
1842 // channels.
1843 RTC_DCHECK(video_channels_.empty());
1844
skvlad6c87a672016-05-17 17:49:52 -07001845 bool require_rtcp_mux =
1846 rtcp_mux_policy_ == PeerConnectionInterface::kRtcpMuxPolicyRequire;
zhihuangf5b251b2017-01-12 19:37:48 -08001847
1848 std::string transport_name =
1849 bundle_transport ? *bundle_transport : content->name;
1850
zhihuangb2cdd932017-01-19 16:54:25 -08001851 cricket::DtlsTransportInternal* rtp_dtls_transport =
1852 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001853 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001854 cricket::DtlsTransportInternal* rtcp_dtls_transport = nullptr;
deadbeefac22f702017-01-12 21:59:29 -08001855 if (!require_rtcp_mux) {
zhihuangb2cdd932017-01-19 16:54:25 -08001856 rtcp_dtls_transport = transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001857 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
1858 }
1859
Steve Anton169629a2017-08-30 17:36:36 -07001860 cricket::VideoChannel* video_channel = channel_manager_->CreateVideoChannel(
nisseeaabdf62017-05-05 02:23:02 -07001861 call_, media_config_, rtp_dtls_transport, rtcp_dtls_transport,
deadbeef1a2183d2017-02-10 23:44:49 -08001862 transport_controller_->signaling_thread(), content->name, SrtpRequired(),
Steve Anton169629a2017-08-30 17:36:36 -07001863 video_options_);
zhihuangf5b251b2017-01-12 19:37:48 -08001864
Steve Anton169629a2017-08-30 17:36:36 -07001865 if (!video_channel) {
zhihuangb2cdd932017-01-19 16:54:25 -08001866 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08001867 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001868 if (rtcp_dtls_transport) {
1869 transport_controller_->DestroyDtlsTransport(
Steve Anton24efa722017-08-29 18:12:51 -07001870 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
deadbeefbad5dad2017-01-17 18:32:35 -08001871 }
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001872 return false;
1873 }
zhihuangf5b251b2017-01-12 19:37:48 -08001874
Steve Anton169629a2017-08-30 17:36:36 -07001875 video_channels_.push_back(video_channel);
1876
1877 video_channel->SignalRtcpMuxFullyActive.connect(
deadbeefac22f702017-01-12 21:59:29 -08001878 this, &WebRtcSession::DestroyRtcpTransport_n);
Steve Anton169629a2017-08-30 17:36:36 -07001879 video_channel->SignalDtlsSrtpSetupFailure.connect(
deadbeef953c2ce2017-01-09 14:53:41 -08001880 this, &WebRtcSession::OnDtlsSrtpSetupFailure);
deadbeefab9b2d12015-10-14 11:33:11 -07001881
Steve Anton169629a2017-08-30 17:36:36 -07001882 // TODO(steveanton): This should signal which video channel was created since
1883 // we can have multiple.
deadbeefab9b2d12015-10-14 11:33:11 -07001884 SignalVideoChannelCreated();
Steve Anton169629a2017-08-30 17:36:36 -07001885 video_channel->SignalSentPacket.connect(this, &WebRtcSession::OnSentPacket_w);
pthatcher@webrtc.org4eeef582015-03-16 19:34:23 +00001886 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001887}
1888
skvlad6c87a672016-05-17 17:49:52 -07001889bool WebRtcSession::CreateDataChannel(const cricket::ContentInfo* content,
1890 const std::string* bundle_transport) {
deadbeef953c2ce2017-01-09 14:53:41 -08001891 const std::string transport_name =
1892 bundle_transport ? *bundle_transport : content->name;
zhihuang9763d562016-08-05 11:14:50 -07001893#ifdef HAVE_QUIC
1894 if (data_channel_type_ == cricket::DCT_QUIC) {
1895 RTC_DCHECK(transport_controller_->quic());
zhihuangb2cdd932017-01-19 16:54:25 -08001896 quic_data_transport_->SetTransports(transport_name);
zhihuang9763d562016-08-05 11:14:50 -07001897 return true;
1898 }
1899#endif // HAVE_QUIC
wu@webrtc.orgcecfd182013-10-30 05:18:12 +00001900 bool sctp = (data_channel_type_ == cricket::DCT_SCTP);
deadbeefc0dad892017-01-04 20:28:21 -08001901 if (sctp) {
deadbeef953c2ce2017-01-09 14:53:41 -08001902 if (!sctp_factory_) {
1903 LOG(LS_ERROR)
1904 << "Trying to create SCTP transport, but didn't compile with "
1905 "SCTP support (HAVE_SCTP)";
1906 return false;
1907 }
1908 if (!network_thread_->Invoke<bool>(
1909 RTC_FROM_HERE, rtc::Bind(&WebRtcSession::CreateSctpTransport_n,
1910 this, content->name, transport_name))) {
1911 return false;
1912 };
1913 } else {
1914 bool require_rtcp_mux =
1915 rtcp_mux_policy_ == PeerConnectionInterface::kRtcpMuxPolicyRequire;
zhihuangf5b251b2017-01-12 19:37:48 -08001916
1917 std::string transport_name =
1918 bundle_transport ? *bundle_transport : content->name;
zhihuangb2cdd932017-01-19 16:54:25 -08001919 cricket::DtlsTransportInternal* rtp_dtls_transport =
1920 transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001921 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001922 cricket::DtlsTransportInternal* rtcp_dtls_transport = nullptr;
deadbeefac22f702017-01-12 21:59:29 -08001923 if (!require_rtcp_mux) {
zhihuangb2cdd932017-01-19 16:54:25 -08001924 rtcp_dtls_transport = transport_controller_->CreateDtlsTransport(
zhihuangf5b251b2017-01-12 19:37:48 -08001925 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
1926 }
1927
deadbeef953c2ce2017-01-09 14:53:41 -08001928 rtp_data_channel_.reset(channel_manager_->CreateRtpDataChannel(
nisseeaabdf62017-05-05 02:23:02 -07001929 media_config_, rtp_dtls_transport, rtcp_dtls_transport,
zhihuangf5b251b2017-01-12 19:37:48 -08001930 transport_controller_->signaling_thread(), content->name,
deadbeef1a2183d2017-02-10 23:44:49 -08001931 SrtpRequired()));
zhihuangf5b251b2017-01-12 19:37:48 -08001932
deadbeef953c2ce2017-01-09 14:53:41 -08001933 if (!rtp_data_channel_) {
zhihuangb2cdd932017-01-19 16:54:25 -08001934 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08001935 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangb2cdd932017-01-19 16:54:25 -08001936 if (rtcp_dtls_transport) {
1937 transport_controller_->DestroyDtlsTransport(
Steve Anton24efa722017-08-29 18:12:51 -07001938 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
deadbeefbad5dad2017-01-17 18:32:35 -08001939 }
deadbeef953c2ce2017-01-09 14:53:41 -08001940 return false;
1941 }
zhihuangf5b251b2017-01-12 19:37:48 -08001942
deadbeefac22f702017-01-12 21:59:29 -08001943 rtp_data_channel_->SignalRtcpMuxFullyActive.connect(
1944 this, &WebRtcSession::DestroyRtcpTransport_n);
deadbeef953c2ce2017-01-09 14:53:41 -08001945 rtp_data_channel_->SignalDtlsSrtpSetupFailure.connect(
1946 this, &WebRtcSession::OnDtlsSrtpSetupFailure);
1947 rtp_data_channel_->SignalSentPacket.connect(this,
1948 &WebRtcSession::OnSentPacket_w);
deadbeefc0dad892017-01-04 20:28:21 -08001949 }
1950
deadbeefab9b2d12015-10-14 11:33:11 -07001951 SignalDataChannelCreated();
zhihuangf5b251b2017-01-12 19:37:48 -08001952
wu@webrtc.org91053e72013-08-10 07:18:04 +00001953 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001954}
1955
stefanf79ade12017-06-02 06:44:03 -07001956Call::Stats WebRtcSession::GetCallStats() {
1957 if (!worker_thread()->IsCurrent()) {
1958 return worker_thread()->Invoke<Call::Stats>(
1959 RTC_FROM_HERE, rtc::Bind(&WebRtcSession::GetCallStats, this));
1960 }
zhihuang38ede132017-06-15 12:52:32 -07001961 if (!call_)
1962 return Call::Stats();
stefanf79ade12017-06-02 06:44:03 -07001963 return call_->GetStats();
1964}
1965
hbosdf6075a2016-12-19 04:58:02 -08001966std::unique_ptr<SessionStats> WebRtcSession::GetStats_n(
1967 const ChannelNamePairs& channel_name_pairs) {
nisseede5da42017-01-12 05:15:36 -08001968 RTC_DCHECK(network_thread()->IsCurrent());
hbosdf6075a2016-12-19 04:58:02 -08001969 std::unique_ptr<SessionStats> session_stats(new SessionStats());
1970 for (const auto channel_name_pair : { &channel_name_pairs.voice,
1971 &channel_name_pairs.video,
1972 &channel_name_pairs.data }) {
1973 if (*channel_name_pair) {
1974 cricket::TransportStats transport_stats;
1975 if (!transport_controller_->GetStats((*channel_name_pair)->transport_name,
1976 &transport_stats)) {
1977 return nullptr;
1978 }
1979 session_stats->proxy_to_transport[(*channel_name_pair)->content_name] =
1980 (*channel_name_pair)->transport_name;
1981 session_stats->transport_stats[(*channel_name_pair)->transport_name] =
1982 std::move(transport_stats);
1983 }
1984 }
1985 return session_stats;
1986}
1987
deadbeef953c2ce2017-01-09 14:53:41 -08001988bool WebRtcSession::CreateSctpTransport_n(const std::string& content_name,
1989 const std::string& transport_name) {
1990 RTC_DCHECK(network_thread_->IsCurrent());
1991 RTC_DCHECK(sctp_factory_);
zhihuangb2cdd932017-01-19 16:54:25 -08001992 cricket::DtlsTransportInternal* tc =
1993 transport_controller_->CreateDtlsTransport_n(
deadbeef953c2ce2017-01-09 14:53:41 -08001994 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
1995 sctp_transport_ = sctp_factory_->CreateSctpTransport(tc);
1996 RTC_DCHECK(sctp_transport_);
1997 sctp_invoker_.reset(new rtc::AsyncInvoker());
1998 sctp_transport_->SignalReadyToSendData.connect(
1999 this, &WebRtcSession::OnSctpTransportReadyToSendData_n);
2000 sctp_transport_->SignalDataReceived.connect(
2001 this, &WebRtcSession::OnSctpTransportDataReceived_n);
2002 sctp_transport_->SignalStreamClosedRemotely.connect(
2003 this, &WebRtcSession::OnSctpStreamClosedRemotely_n);
2004 sctp_transport_name_ = rtc::Optional<std::string>(transport_name);
2005 sctp_content_name_ = rtc::Optional<std::string>(content_name);
2006 return true;
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002007}
2008
deadbeef953c2ce2017-01-09 14:53:41 -08002009void WebRtcSession::ChangeSctpTransport_n(const std::string& transport_name) {
2010 RTC_DCHECK(network_thread_->IsCurrent());
2011 RTC_DCHECK(sctp_transport_);
2012 RTC_DCHECK(sctp_transport_name_);
2013 std::string old_sctp_transport_name = *sctp_transport_name_;
2014 sctp_transport_name_ = rtc::Optional<std::string>(transport_name);
zhihuangb2cdd932017-01-19 16:54:25 -08002015 cricket::DtlsTransportInternal* tc =
2016 transport_controller_->CreateDtlsTransport_n(
deadbeef953c2ce2017-01-09 14:53:41 -08002017 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
2018 sctp_transport_->SetTransportChannel(tc);
zhihuangb2cdd932017-01-19 16:54:25 -08002019 transport_controller_->DestroyDtlsTransport_n(
deadbeef953c2ce2017-01-09 14:53:41 -08002020 old_sctp_transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
2021}
2022
2023void WebRtcSession::DestroySctpTransport_n() {
2024 RTC_DCHECK(network_thread_->IsCurrent());
2025 sctp_transport_.reset(nullptr);
2026 sctp_content_name_.reset();
2027 sctp_transport_name_.reset();
2028 sctp_invoker_.reset(nullptr);
2029 sctp_ready_to_send_data_ = false;
2030}
2031
2032void WebRtcSession::OnSctpTransportReadyToSendData_n() {
2033 RTC_DCHECK(data_channel_type_ == cricket::DCT_SCTP);
2034 RTC_DCHECK(network_thread_->IsCurrent());
2035 sctp_invoker_->AsyncInvoke<void>(
2036 RTC_FROM_HERE, signaling_thread_,
2037 rtc::Bind(&WebRtcSession::OnSctpTransportReadyToSendData_s, this, true));
2038}
2039
2040void WebRtcSession::OnSctpTransportReadyToSendData_s(bool ready) {
2041 RTC_DCHECK(signaling_thread_->IsCurrent());
2042 sctp_ready_to_send_data_ = ready;
2043 SignalSctpReadyToSendData(ready);
2044}
2045
2046void WebRtcSession::OnSctpTransportDataReceived_n(
henrika@webrtc.orgaebb1ad2014-01-14 10:00:58 +00002047 const cricket::ReceiveDataParams& params,
jbaucheec21bd2016-03-20 06:15:43 -07002048 const rtc::CopyOnWriteBuffer& payload) {
deadbeefab9b2d12015-10-14 11:33:11 -07002049 RTC_DCHECK(data_channel_type_ == cricket::DCT_SCTP);
deadbeef953c2ce2017-01-09 14:53:41 -08002050 RTC_DCHECK(network_thread_->IsCurrent());
2051 sctp_invoker_->AsyncInvoke<void>(
2052 RTC_FROM_HERE, signaling_thread_,
2053 rtc::Bind(&WebRtcSession::OnSctpTransportDataReceived_s, this, params,
2054 payload));
2055}
2056
2057void WebRtcSession::OnSctpTransportDataReceived_s(
2058 const cricket::ReceiveDataParams& params,
2059 const rtc::CopyOnWriteBuffer& payload) {
2060 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeefab9b2d12015-10-14 11:33:11 -07002061 if (params.type == cricket::DMT_CONTROL && IsOpenMessage(payload)) {
2062 // Received OPEN message; parse and signal that a new data channel should
2063 // be created.
2064 std::string label;
2065 InternalDataChannelInit config;
2066 config.id = params.ssrc;
2067 if (!ParseDataChannelOpenMessage(payload, &label, &config)) {
2068 LOG(LS_WARNING) << "Failed to parse the OPEN message for sid "
2069 << params.ssrc;
2070 return;
2071 }
2072 config.open_handshake_role = InternalDataChannelInit::kAcker;
2073 SignalDataChannelOpenMessage(label, config);
deadbeef953c2ce2017-01-09 14:53:41 -08002074 } else {
2075 // Otherwise just forward the signal.
2076 SignalSctpDataReceived(params, payload);
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002077 }
deadbeef953c2ce2017-01-09 14:53:41 -08002078}
2079
2080void WebRtcSession::OnSctpStreamClosedRemotely_n(int sid) {
2081 RTC_DCHECK(data_channel_type_ == cricket::DCT_SCTP);
2082 RTC_DCHECK(network_thread_->IsCurrent());
2083 sctp_invoker_->AsyncInvoke<void>(
2084 RTC_FROM_HERE, signaling_thread_,
2085 rtc::Bind(&sigslot::signal1<int>::operator(),
2086 &SignalSctpStreamClosedRemotely, sid));
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002087}
2088
henrike@webrtc.org1e09a712013-07-26 19:17:59 +00002089// Returns false if bundle is enabled and rtcp_mux is disabled.
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002090bool WebRtcSession::ValidateBundleSettings(const SessionDescription* desc) {
henrike@webrtc.org1e09a712013-07-26 19:17:59 +00002091 bool bundle_enabled = desc->HasGroup(cricket::GROUP_TYPE_BUNDLE);
2092 if (!bundle_enabled)
2093 return true;
2094
2095 const cricket::ContentGroup* bundle_group =
2096 desc->GetGroupByName(cricket::GROUP_TYPE_BUNDLE);
nisseede5da42017-01-12 05:15:36 -08002097 RTC_DCHECK(bundle_group != NULL);
henrike@webrtc.org1e09a712013-07-26 19:17:59 +00002098
2099 const cricket::ContentInfos& contents = desc->contents();
2100 for (cricket::ContentInfos::const_iterator citer = contents.begin();
2101 citer != contents.end(); ++citer) {
2102 const cricket::ContentInfo* content = (&*citer);
nisseede5da42017-01-12 05:15:36 -08002103 RTC_DCHECK(content != NULL);
henrike@webrtc.org1e09a712013-07-26 19:17:59 +00002104 if (bundle_group->HasContentName(content->name) &&
2105 !content->rejected && content->type == cricket::NS_JINGLE_RTP) {
2106 if (!HasRtcpMuxEnabled(content))
2107 return false;
2108 }
2109 }
2110 // RTCP-MUX is enabled in all the contents.
2111 return true;
2112}
2113
2114bool WebRtcSession::HasRtcpMuxEnabled(
2115 const cricket::ContentInfo* content) {
2116 const cricket::MediaContentDescription* description =
2117 static_cast<cricket::MediaContentDescription*>(content->description);
2118 return description->rtcp_mux();
2119}
2120
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002121bool WebRtcSession::ValidateSessionDescription(
2122 const SessionDescriptionInterface* sdesc,
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002123 cricket::ContentSource source, std::string* err_desc) {
2124 std::string type;
deadbeefd59daf82015-10-14 15:02:44 -07002125 if (error() != ERROR_NONE) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002126 return BadSdp(source, type, GetSessionErrorMsg(), err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002127 }
2128
2129 if (!sdesc || !sdesc->description()) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002130 return BadSdp(source, type, kInvalidSdp, err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002131 }
2132
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002133 type = sdesc->type();
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002134 Action action = GetAction(sdesc->type());
2135 if (source == cricket::CS_LOCAL) {
2136 if (!ExpectSetLocalDescription(action))
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002137 return BadLocalSdp(type, BadStateErrMsg(state()), err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002138 } else {
2139 if (!ExpectSetRemoteDescription(action))
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002140 return BadRemoteSdp(type, BadStateErrMsg(state()), err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002141 }
2142
2143 // Verify crypto settings.
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +00002144 std::string crypto_error;
henrike@webrtc.orgb90991d2014-03-04 19:54:57 +00002145 if ((webrtc_session_desc_factory_->SdesPolicy() == cricket::SEC_REQUIRED ||
2146 dtls_enabled_) &&
mallinath@webrtc.orga27be8e2013-09-27 23:04:10 +00002147 !VerifyCrypto(sdesc->description(), dtls_enabled_, &crypto_error)) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002148 return BadSdp(source, type, crypto_error, err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002149 }
2150
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00002151 // Verify ice-ufrag and ice-pwd.
2152 if (!VerifyIceUfragPwdPresent(sdesc->description())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002153 return BadSdp(source, type, kSdpWithoutIceUfragPwd, err_desc);
mallinath@webrtc.org19f27e62013-10-13 17:18:27 +00002154 }
2155
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002156 if (!ValidateBundleSettings(sdesc->description())) {
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002157 return BadSdp(source, type, kBundleWithoutRtcpMux, err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002158 }
2159
skvlad6c87a672016-05-17 17:49:52 -07002160 // TODO(skvlad): When the local rtcp-mux policy is Require, reject any
2161 // m-lines that do not rtcp-mux enabled.
2162
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002163 // Verify m-lines in Answer when compared against Offer.
Zhi Huang2a5e4262017-09-14 01:15:03 -07002164 if (action == kAnswer || action == kPrAnswer) {
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002165 const cricket::SessionDescription* offer_desc =
deadbeeffe4a8a42016-12-20 17:56:17 -08002166 (source == cricket::CS_LOCAL) ? remote_description()->description()
2167 : local_description()->description();
Tommi589ae452017-10-15 21:20:46 +00002168 if (!MediaSectionsHaveSameCount(sdesc->description(), offer_desc) ||
2169 !MediaSectionsInSameOrder(sdesc->description(), offer_desc)) {
Zhi Huang2a5e4262017-09-14 01:15:03 -07002170 return BadAnswerSdp(source, kMlineMismatchInAnswer, err_desc);
2171 }
2172 } else {
Tommi589ae452017-10-15 21:20:46 +00002173 // The re-offers should respect the order of m= sections in current local
Zhi Huang2a5e4262017-09-14 01:15:03 -07002174 // description. See RFC3264 Section 8 paragraph 4 for more details.
Tommi589ae452017-10-15 21:20:46 +00002175 if (local_description() &&
2176 !MediaSectionsInSameOrder(sdesc->description(),
2177 local_description()->description())) {
Zhi Huang2a5e4262017-09-14 01:15:03 -07002178 return BadOfferSdp(source, kMlineMismatchInSubsequentOffer, err_desc);
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002179 }
2180 }
2181
2182 return true;
2183}
2184
2185bool WebRtcSession::ExpectSetLocalDescription(Action action) {
2186 return ((action == kOffer && state() == STATE_INIT) ||
2187 // update local offer
deadbeefd59daf82015-10-14 15:02:44 -07002188 (action == kOffer && state() == STATE_SENTOFFER) ||
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002189 // update the current ongoing session.
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002190 (action == kOffer && state() == STATE_INPROGRESS) ||
2191 // accept remote offer
deadbeefd59daf82015-10-14 15:02:44 -07002192 (action == kAnswer && state() == STATE_RECEIVEDOFFER) ||
2193 (action == kAnswer && state() == STATE_SENTPRANSWER) ||
2194 (action == kPrAnswer && state() == STATE_RECEIVEDOFFER) ||
2195 (action == kPrAnswer && state() == STATE_SENTPRANSWER));
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002196}
2197
2198bool WebRtcSession::ExpectSetRemoteDescription(Action action) {
2199 return ((action == kOffer && state() == STATE_INIT) ||
2200 // update remote offer
deadbeefd59daf82015-10-14 15:02:44 -07002201 (action == kOffer && state() == STATE_RECEIVEDOFFER) ||
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002202 // update the current ongoing session
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002203 (action == kOffer && state() == STATE_INPROGRESS) ||
2204 // accept local offer
deadbeefd59daf82015-10-14 15:02:44 -07002205 (action == kAnswer && state() == STATE_SENTOFFER) ||
2206 (action == kAnswer && state() == STATE_RECEIVEDPRANSWER) ||
2207 (action == kPrAnswer && state() == STATE_SENTOFFER) ||
2208 (action == kPrAnswer && state() == STATE_RECEIVEDPRANSWER));
sergeyu@chromium.org0be6aa02013-08-23 23:21:25 +00002209}
2210
sergeyu@chromium.org4b26e2e2014-01-15 23:15:54 +00002211std::string WebRtcSession::GetSessionErrorMsg() {
2212 std::ostringstream desc;
2213 desc << kSessionError << GetErrorCodeString(error()) << ". ";
2214 desc << kSessionErrorDesc << error_desc() << ".";
2215 return desc.str();
2216}
2217
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002218// We need to check the local/remote description for the Transport instead of
2219// the session, because a new Transport added during renegotiation may have
2220// them unset while the session has them set from the previous negotiation.
2221// Not doing so may trigger the auto generation of transport description and
2222// mess up DTLS identity information, ICE credential, etc.
2223bool WebRtcSession::ReadyToUseRemoteCandidate(
2224 const IceCandidateInterface* candidate,
2225 const SessionDescriptionInterface* remote_desc,
2226 bool* valid) {
zhihuang9763d562016-08-05 11:14:50 -07002227 *valid = true;
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002228
2229 const SessionDescriptionInterface* current_remote_desc =
deadbeeffe4a8a42016-12-20 17:56:17 -08002230 remote_desc ? remote_desc : remote_description();
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002231
deadbeefd59daf82015-10-14 15:02:44 -07002232 if (!current_remote_desc) {
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002233 return false;
deadbeefd59daf82015-10-14 15:02:44 -07002234 }
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002235
2236 size_t mediacontent_index =
2237 static_cast<size_t>(candidate->sdp_mline_index());
2238 size_t remote_content_size =
2239 current_remote_desc->description()->contents().size();
2240 if (mediacontent_index >= remote_content_size) {
Honghai Zhang7fb69db2016-03-14 11:59:18 -07002241 LOG(LS_ERROR) << "ReadyToUseRemoteCandidate: Invalid candidate media index "
2242 << mediacontent_index;
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002243
2244 *valid = false;
2245 return false;
2246 }
2247
2248 cricket::ContentInfo content =
2249 current_remote_desc->description()->contents()[mediacontent_index];
zhihuang9763d562016-08-05 11:14:50 -07002250
2251 const std::string transport_name = GetTransportName(content.name);
2252 if (transport_name.empty()) {
deadbeefcbecd352015-09-23 11:50:27 -07002253 return false;
2254 }
zhihuang9763d562016-08-05 11:14:50 -07002255 return transport_controller_->ReadyForRemoteCandidates(transport_name);
jiayl@webrtc.orge10d28c2014-07-17 17:07:49 +00002256}
2257
deadbeef7af91dd2016-12-13 11:29:11 -08002258bool WebRtcSession::SrtpRequired() const {
2259 return dtls_enabled_ ||
2260 webrtc_session_desc_factory_->SdesPolicy() == cricket::SEC_REQUIRED;
2261}
2262
deadbeefcbecd352015-09-23 11:50:27 -07002263void WebRtcSession::OnTransportControllerGatheringState(
2264 cricket::IceGatheringState state) {
nisseede5da42017-01-12 05:15:36 -08002265 RTC_DCHECK(signaling_thread()->IsCurrent());
deadbeefcbecd352015-09-23 11:50:27 -07002266 if (state == cricket::kIceGatheringGathering) {
2267 if (ice_observer_) {
2268 ice_observer_->OnIceGatheringChange(
2269 PeerConnectionInterface::kIceGatheringGathering);
2270 }
2271 } else if (state == cricket::kIceGatheringComplete) {
2272 if (ice_observer_) {
2273 ice_observer_->OnIceGatheringChange(
2274 PeerConnectionInterface::kIceGatheringComplete);
deadbeefcbecd352015-09-23 11:50:27 -07002275 }
2276 }
2277}
2278
2279void WebRtcSession::ReportTransportStats() {
2280 // Use a set so we don't report the same stats twice if two channels share
2281 // a transport.
2282 std::set<std::string> transport_names;
2283 if (voice_channel()) {
2284 transport_names.insert(voice_channel()->transport_name());
2285 }
2286 if (video_channel()) {
2287 transport_names.insert(video_channel()->transport_name());
2288 }
deadbeef953c2ce2017-01-09 14:53:41 -08002289 if (rtp_data_channel()) {
2290 transport_names.insert(rtp_data_channel()->transport_name());
2291 }
2292 if (sctp_transport_name_) {
2293 transport_names.insert(*sctp_transport_name_);
deadbeefcbecd352015-09-23 11:50:27 -07002294 }
2295 for (const auto& name : transport_names) {
2296 cricket::TransportStats stats;
deadbeefd59daf82015-10-14 15:02:44 -07002297 if (transport_controller_->GetStats(name, &stats)) {
deadbeefcbecd352015-09-23 11:50:27 -07002298 ReportBestConnectionState(stats);
2299 ReportNegotiatedCiphers(stats);
2300 }
2301 }
2302}
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +00002303// Walk through the ConnectionInfos to gather best connection usage
2304// for IPv4 and IPv6.
jbauchac8869e2015-07-03 01:36:14 -07002305void WebRtcSession::ReportBestConnectionState(
2306 const cricket::TransportStats& stats) {
henrikg91d6ede2015-09-17 00:24:34 -07002307 RTC_DCHECK(metrics_observer_ != NULL);
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +00002308 for (cricket::TransportChannelStatsList::const_iterator it =
2309 stats.channel_stats.begin();
2310 it != stats.channel_stats.end(); ++it) {
2311 for (cricket::ConnectionInfos::const_iterator it_info =
2312 it->connection_infos.begin();
2313 it_info != it->connection_infos.end(); ++it_info) {
2314 if (!it_info->best_connection) {
2315 continue;
2316 }
Guo-wei Shieh3d564c12015-08-19 16:51:15 -07002317
2318 PeerConnectionEnumCounterType type = kPeerConnectionEnumCounterMax;
2319 const cricket::Candidate& local = it_info->local_candidate;
2320 const cricket::Candidate& remote = it_info->remote_candidate;
2321
2322 // Increment the counter for IceCandidatePairType.
2323 if (local.protocol() == cricket::TCP_PROTOCOL_NAME ||
2324 (local.type() == RELAY_PORT_TYPE &&
2325 local.relay_protocol() == cricket::TCP_PROTOCOL_NAME)) {
2326 type = kEnumCounterIceCandidatePairTypeTcp;
2327 } else if (local.protocol() == cricket::UDP_PROTOCOL_NAME) {
2328 type = kEnumCounterIceCandidatePairTypeUdp;
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +00002329 } else {
henrikg91d6ede2015-09-17 00:24:34 -07002330 RTC_CHECK(0);
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +00002331 }
Guo-wei Shieh3d564c12015-08-19 16:51:15 -07002332 metrics_observer_->IncrementEnumCounter(
2333 type, GetIceCandidatePairCounter(local, remote),
2334 kIceCandidatePairMax);
2335
2336 // Increment the counter for IP type.
2337 if (local.address().family() == AF_INET) {
Guo-wei Shieh3d564c12015-08-19 16:51:15 -07002338 metrics_observer_->IncrementEnumCounter(
2339 kEnumCounterAddressFamily, kBestConnections_IPv4,
2340 kPeerConnectionAddressFamilyCounter_Max);
2341
2342 } else if (local.address().family() == AF_INET6) {
Guo-wei Shieh3d564c12015-08-19 16:51:15 -07002343 metrics_observer_->IncrementEnumCounter(
2344 kEnumCounterAddressFamily, kBestConnections_IPv6,
2345 kPeerConnectionAddressFamilyCounter_Max);
2346 } else {
henrikg91d6ede2015-09-17 00:24:34 -07002347 RTC_CHECK(0);
Guo-wei Shieh3d564c12015-08-19 16:51:15 -07002348 }
2349
guoweis@webrtc.org7169afd2014-12-04 17:59:29 +00002350 return;
2351 }
2352 }
2353}
2354
jbauchac8869e2015-07-03 01:36:14 -07002355void WebRtcSession::ReportNegotiatedCiphers(
2356 const cricket::TransportStats& stats) {
henrikg91d6ede2015-09-17 00:24:34 -07002357 RTC_DCHECK(metrics_observer_ != NULL);
jbauchac8869e2015-07-03 01:36:14 -07002358 if (!dtls_enabled_ || stats.channel_stats.empty()) {
2359 return;
2360 }
2361
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08002362 int srtp_crypto_suite = stats.channel_stats[0].srtp_crypto_suite;
2363 int ssl_cipher_suite = stats.channel_stats[0].ssl_cipher_suite;
2364 if (srtp_crypto_suite == rtc::SRTP_INVALID_CRYPTO_SUITE &&
2365 ssl_cipher_suite == rtc::TLS_NULL_WITH_NULL_NULL) {
jbauchac8869e2015-07-03 01:36:14 -07002366 return;
2367 }
2368
Guo-wei Shieh456696a2015-09-30 21:48:54 -07002369 PeerConnectionEnumCounterType srtp_counter_type;
2370 PeerConnectionEnumCounterType ssl_counter_type;
deadbeefcbecd352015-09-23 11:50:27 -07002371 if (stats.transport_name == cricket::CN_AUDIO) {
Guo-wei Shieh456696a2015-09-30 21:48:54 -07002372 srtp_counter_type = kEnumCounterAudioSrtpCipher;
2373 ssl_counter_type = kEnumCounterAudioSslCipher;
deadbeefcbecd352015-09-23 11:50:27 -07002374 } else if (stats.transport_name == cricket::CN_VIDEO) {
Guo-wei Shieh456696a2015-09-30 21:48:54 -07002375 srtp_counter_type = kEnumCounterVideoSrtpCipher;
2376 ssl_counter_type = kEnumCounterVideoSslCipher;
deadbeefcbecd352015-09-23 11:50:27 -07002377 } else if (stats.transport_name == cricket::CN_DATA) {
Guo-wei Shieh456696a2015-09-30 21:48:54 -07002378 srtp_counter_type = kEnumCounterDataSrtpCipher;
2379 ssl_counter_type = kEnumCounterDataSslCipher;
jbauchac8869e2015-07-03 01:36:14 -07002380 } else {
2381 RTC_NOTREACHED();
2382 return;
2383 }
2384
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08002385 if (srtp_crypto_suite != rtc::SRTP_INVALID_CRYPTO_SUITE) {
2386 metrics_observer_->IncrementSparseEnumCounter(srtp_counter_type,
2387 srtp_crypto_suite);
jbauchac8869e2015-07-03 01:36:14 -07002388 }
Guo-wei Shieh521ed7b2015-11-18 19:41:53 -08002389 if (ssl_cipher_suite != rtc::TLS_NULL_WITH_NULL_NULL) {
2390 metrics_observer_->IncrementSparseEnumCounter(ssl_counter_type,
2391 ssl_cipher_suite);
jbauchac8869e2015-07-03 01:36:14 -07002392 }
2393}
2394
Danil Chapovalov33b01f22016-05-11 19:55:27 +02002395void WebRtcSession::OnSentPacket_w(const rtc::SentPacket& sent_packet) {
stefanc1aeaf02015-10-15 07:26:07 -07002396 RTC_DCHECK(worker_thread()->IsCurrent());
stefanf79ade12017-06-02 06:44:03 -07002397 RTC_DCHECK(call_);
nisseeaabdf62017-05-05 02:23:02 -07002398 call_->OnSentPacket(sent_packet);
stefanc1aeaf02015-10-15 07:26:07 -07002399}
2400
zhihuang9763d562016-08-05 11:14:50 -07002401const std::string WebRtcSession::GetTransportName(
2402 const std::string& content_name) {
2403 cricket::BaseChannel* channel = GetChannel(content_name);
2404 if (!channel) {
2405#ifdef HAVE_QUIC
2406 if (data_channel_type_ == cricket::DCT_QUIC && quic_data_transport_ &&
2407 content_name == quic_data_transport_->transport_name()) {
2408 return quic_data_transport_->transport_name();
2409 }
2410#endif
deadbeef953c2ce2017-01-09 14:53:41 -08002411 if (sctp_transport_) {
2412 RTC_DCHECK(sctp_content_name_);
2413 RTC_DCHECK(sctp_transport_name_);
2414 if (content_name == *sctp_content_name_) {
2415 return *sctp_transport_name_;
2416 }
2417 }
zhihuang9763d562016-08-05 11:14:50 -07002418 // Return an empty string if failed to retrieve the transport name.
2419 return "";
2420 }
2421 return channel->transport_name();
2422}
zhihuangd82eee02016-08-26 11:25:05 -07002423
deadbeefac22f702017-01-12 21:59:29 -08002424void WebRtcSession::DestroyRtcpTransport_n(const std::string& transport_name) {
nissea9dd4a12017-01-13 07:08:34 -08002425 RTC_DCHECK(network_thread()->IsCurrent());
zhihuangb2cdd932017-01-19 16:54:25 -08002426 transport_controller_->DestroyDtlsTransport_n(
zhihuangf5b251b2017-01-12 19:37:48 -08002427 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
2428}
2429
Steve Anton169629a2017-08-30 17:36:36 -07002430void WebRtcSession::RemoveAndDestroyVideoChannel(
2431 cricket::VideoChannel* video_channel) {
2432 auto it =
2433 std::find(video_channels_.begin(), video_channels_.end(), video_channel);
2434 RTC_DCHECK(it != video_channels_.end());
2435 if (it == video_channels_.end()) {
2436 return;
2437 }
2438 video_channels_.erase(it);
2439 DestroyVideoChannel(video_channel);
2440}
2441
2442void WebRtcSession::DestroyVideoChannel(cricket::VideoChannel* video_channel) {
2443 // TODO(steveanton): This should take an identifier for the video channel
2444 // since we now support more than one.
zhihuangf5b251b2017-01-12 19:37:48 -08002445 SignalVideoChannelDestroyed();
Steve Anton169629a2017-08-30 17:36:36 -07002446 RTC_DCHECK(video_channel->rtp_dtls_transport());
2447 const std::string transport_name =
2448 video_channel->rtp_dtls_transport()->transport_name();
2449 const bool need_to_delete_rtcp =
2450 (video_channel->rtcp_dtls_transport() != nullptr);
2451 // The above need to be cached before destroying the video channel so that we
2452 // do not access uninitialized memory.
2453 channel_manager_->DestroyVideoChannel(video_channel);
zhihuangb2cdd932017-01-19 16:54:25 -08002454 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002455 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangf5b251b2017-01-12 19:37:48 -08002456 if (need_to_delete_rtcp) {
zhihuangb2cdd932017-01-19 16:54:25 -08002457 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002458 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
zhihuangf5b251b2017-01-12 19:37:48 -08002459 }
2460}
2461
Steve Anton169629a2017-08-30 17:36:36 -07002462void WebRtcSession::RemoveAndDestroyVoiceChannel(
2463 cricket::VoiceChannel* voice_channel) {
2464 auto it =
2465 std::find(voice_channels_.begin(), voice_channels_.end(), voice_channel);
2466 RTC_DCHECK(it != voice_channels_.end());
2467 if (it == voice_channels_.end()) {
2468 return;
2469 }
2470 voice_channels_.erase(it);
2471 DestroyVoiceChannel(voice_channel);
2472}
2473
2474void WebRtcSession::DestroyVoiceChannel(cricket::VoiceChannel* voice_channel) {
2475 // TODO(steveanton): This should take an identifier for the voice channel
2476 // since we now support more than one.
zhihuangf5b251b2017-01-12 19:37:48 -08002477 SignalVoiceChannelDestroyed();
Steve Anton169629a2017-08-30 17:36:36 -07002478 RTC_DCHECK(voice_channel->rtp_dtls_transport());
2479 const std::string transport_name =
2480 voice_channel->rtp_dtls_transport()->transport_name();
2481 const bool need_to_delete_rtcp =
2482 (voice_channel->rtcp_dtls_transport() != nullptr);
2483 // The above need to be cached before destroying the video channel so that we
2484 // do not access uninitialized memory.
2485 channel_manager_->DestroyVoiceChannel(voice_channel);
zhihuangb2cdd932017-01-19 16:54:25 -08002486 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002487 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangf5b251b2017-01-12 19:37:48 -08002488 if (need_to_delete_rtcp) {
zhihuangb2cdd932017-01-19 16:54:25 -08002489 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002490 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
zhihuangf5b251b2017-01-12 19:37:48 -08002491 }
2492}
2493
2494void WebRtcSession::DestroyDataChannel() {
2495 SignalDataChannelDestroyed();
zhihuangb2cdd932017-01-19 16:54:25 -08002496 RTC_DCHECK(rtp_data_channel_->rtp_dtls_transport());
zhihuangf5b251b2017-01-12 19:37:48 -08002497 std::string transport_name;
zhihuangb2cdd932017-01-19 16:54:25 -08002498 transport_name = rtp_data_channel_->rtp_dtls_transport()->transport_name();
2499 bool need_to_delete_rtcp =
2500 (rtp_data_channel_->rtcp_dtls_transport() != nullptr);
zhihuangf5b251b2017-01-12 19:37:48 -08002501 channel_manager_->DestroyRtpDataChannel(rtp_data_channel_.release());
zhihuangb2cdd932017-01-19 16:54:25 -08002502 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002503 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);
zhihuangf5b251b2017-01-12 19:37:48 -08002504 if (need_to_delete_rtcp) {
zhihuangb2cdd932017-01-19 16:54:25 -08002505 transport_controller_->DestroyDtlsTransport(
deadbeefbad5dad2017-01-17 18:32:35 -08002506 transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);
zhihuangf5b251b2017-01-12 19:37:48 -08002507 }
2508}
henrike@webrtc.org28e20752013-07-10 00:45:36 +00002509} // namespace webrtc