blob: 8d89795ce39fd290d54f5445bc442c6a705f1428 [file] [log] [blame]
Zhi Huange818b6e2018-02-22 15:26:27 -08001/*
2 * Copyright 2017 The WebRTC Project Authors. All rights reserved.
3 *
4 * 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.
9 */
10
11#ifndef PC_JSEPTRANSPORTCONTROLLER_H_
12#define PC_JSEPTRANSPORTCONTROLLER_H_
13
14#include <map>
15#include <memory>
16#include <string>
17#include <utility>
18#include <vector>
19
20#include "api/candidate.h"
Benjamin Wrighta54daf12018-10-11 15:33:17 -070021#include "api/crypto/cryptooptions.h"
Anton Sukhanov7940da02018-10-10 10:34:49 -070022#include "api/media_transport_interface.h"
Zhi Huange818b6e2018-02-22 15:26:27 -080023#include "api/peerconnectioninterface.h"
Qingsi Wang7685e862018-06-11 20:15:46 -070024#include "logging/rtc_event_log/rtc_event_log.h"
Zhi Huange818b6e2018-02-22 15:26:27 -080025#include "media/sctp/sctptransportinternal.h"
26#include "p2p/base/dtlstransport.h"
27#include "p2p/base/p2ptransportchannel.h"
28#include "p2p/base/transportfactoryinterface.h"
29#include "pc/channel.h"
30#include "pc/dtlssrtptransport.h"
Zhi Huang365381f2018-04-13 16:44:34 -070031#include "pc/jseptransport.h"
Zhi Huange818b6e2018-02-22 15:26:27 -080032#include "pc/rtptransport.h"
33#include "pc/srtptransport.h"
34#include "rtc_base/asyncinvoker.h"
35#include "rtc_base/constructormagic.h"
36#include "rtc_base/refcountedobject.h"
Artem Titove41c4332018-07-25 15:04:28 +020037#include "rtc_base/third_party/sigslot/sigslot.h"
Zhi Huange818b6e2018-02-22 15:26:27 -080038
39namespace rtc {
40class Thread;
41class PacketTransportInternal;
42} // namespace rtc
43
44namespace webrtc {
45
Steve Antond25828a2018-08-31 13:06:05 -070046class JsepTransportController : public sigslot::has_slots<> {
Zhi Huange818b6e2018-02-22 15:26:27 -080047 public:
Zhi Huang365381f2018-04-13 16:44:34 -070048 // Used when the RtpTransport/DtlsTransport of the m= section is changed
49 // because the section is rejected or BUNDLE is enabled.
50 class Observer {
51 public:
52 virtual ~Observer() {}
53
54 // Returns true if media associated with |mid| was successfully set up to be
55 // demultiplexed on |rtp_transport|. Could return false if two bundled m=
56 // sections use the same SSRC, for example.
Taylor Brandstettercbaa2542018-04-16 16:42:14 -070057 virtual bool OnTransportChanged(
Zhi Huang365381f2018-04-13 16:44:34 -070058 const std::string& mid,
Taylor Brandstettercbaa2542018-04-16 16:42:14 -070059 RtpTransportInternal* rtp_transport,
Zhi Huang365381f2018-04-13 16:44:34 -070060 cricket::DtlsTransportInternal* dtls_transport) = 0;
61 };
62
Zhi Huange818b6e2018-02-22 15:26:27 -080063 struct Config {
64 // If |redetermine_role_on_ice_restart| is true, ICE role is redetermined
65 // upon setting a local transport description that indicates an ICE
66 // restart.
67 bool redetermine_role_on_ice_restart = true;
68 rtc::SSLProtocolVersion ssl_max_version = rtc::SSL_PROTOCOL_DTLS_12;
69 // |crypto_options| is used to determine if created DTLS transports
70 // negotiate GCM crypto suites or not.
Benjamin Wrighta54daf12018-10-11 15:33:17 -070071 webrtc::CryptoOptions crypto_options;
Zhi Huange818b6e2018-02-22 15:26:27 -080072 PeerConnectionInterface::BundlePolicy bundle_policy =
73 PeerConnectionInterface::kBundlePolicyBalanced;
74 PeerConnectionInterface::RtcpMuxPolicy rtcp_mux_policy =
75 PeerConnectionInterface::kRtcpMuxPolicyRequire;
76 bool disable_encryption = false;
77 bool enable_external_auth = false;
78 // Used to inject the ICE/DTLS transports created externally.
79 cricket::TransportFactoryInterface* external_transport_factory = nullptr;
Zhi Huang365381f2018-04-13 16:44:34 -070080 Observer* transport_observer = nullptr;
Zhi Huangb57e1692018-06-12 11:41:11 -070081 bool active_reset_srtp_params = false;
Qingsi Wang7685e862018-06-11 20:15:46 -070082 RtcEventLog* event_log = nullptr;
Anton Sukhanov7940da02018-10-10 10:34:49 -070083
84 // Optional media transport factory (experimental). If provided it will be
85 // used to create media_transport and will be used to send / receive
86 // audio and video frames instead of RTP. Note that currently
87 // media_transport co-exists with RTP / RTCP transports and uses the same
88 // underlying ICE transport.
89 MediaTransportFactory* media_transport_factory = nullptr;
Zhi Huange818b6e2018-02-22 15:26:27 -080090 };
91
92 // The ICE related events are signaled on the |signaling_thread|.
93 // All the transport related methods are called on the |network_thread|.
94 JsepTransportController(rtc::Thread* signaling_thread,
95 rtc::Thread* network_thread,
96 cricket::PortAllocator* port_allocator,
Zach Steine20867f2018-08-02 13:20:15 -070097 AsyncResolverFactory* async_resolver_factory,
Zhi Huange818b6e2018-02-22 15:26:27 -080098 Config config);
99 virtual ~JsepTransportController();
100
101 // The main method to be called; applies a description at the transport
102 // level, creating/destroying transport objects as needed and updating their
103 // properties. This includes RTP, DTLS, and ICE (but not SCTP). At least not
104 // yet? May make sense to in the future.
105 RTCError SetLocalDescription(SdpType type,
106 const cricket::SessionDescription* description);
107
108 RTCError SetRemoteDescription(SdpType type,
109 const cricket::SessionDescription* description);
110
111 // Get transports to be used for the provided |mid|. If bundling is enabled,
112 // calling GetRtpTransport for multiple MIDs may yield the same object.
113 RtpTransportInternal* GetRtpTransport(const std::string& mid) const;
114 cricket::DtlsTransportInternal* GetDtlsTransport(
115 const std::string& mid) const;
116 cricket::DtlsTransportInternal* GetRtcpDtlsTransport(
117 const std::string& mid) const;
118
Anton Sukhanov7940da02018-10-10 10:34:49 -0700119 MediaTransportInterface* GetMediaTransport(const std::string& mid) const;
Bjorn Mellem175aa2e2018-11-08 11:23:22 -0800120 MediaTransportState GetMediaTransportState(const std::string& mid) const;
Anton Sukhanov7940da02018-10-10 10:34:49 -0700121
Zhi Huange818b6e2018-02-22 15:26:27 -0800122 /*********************
123 * ICE-related methods
124 ********************/
125 // This method is public to allow PeerConnection to update it from
126 // SetConfiguration.
127 void SetIceConfig(const cricket::IceConfig& config);
128 // Set the "needs-ice-restart" flag as described in JSEP. After the flag is
129 // set, offers should generate new ufrags/passwords until an ICE restart
130 // occurs.
131 void SetNeedsIceRestartFlag();
132 // Returns true if the ICE restart flag above was set, and no ICE restart has
133 // occurred yet for this transport (by applying a local description with
134 // changed ufrag/password). If the transport has been deleted as a result of
135 // bundling, returns false.
136 bool NeedsIceRestart(const std::string& mid) const;
137 // Start gathering candidates for any new transports, or transports doing an
138 // ICE restart.
139 void MaybeStartGathering();
140 RTCError AddRemoteCandidates(
141 const std::string& mid,
142 const std::vector<cricket::Candidate>& candidates);
143 RTCError RemoveRemoteCandidates(
144 const std::vector<cricket::Candidate>& candidates);
145
146 /**********************
147 * DTLS-related methods
148 *********************/
149 // Specifies the identity to use in this session.
150 // Can only be called once.
151 bool SetLocalCertificate(
152 const rtc::scoped_refptr<rtc::RTCCertificate>& certificate);
153 rtc::scoped_refptr<rtc::RTCCertificate> GetLocalCertificate(
154 const std::string& mid) const;
Taylor Brandstetterc3928662018-02-23 13:04:51 -0800155 // Caller owns returned certificate chain. This method mainly exists for
156 // stats reporting.
157 std::unique_ptr<rtc::SSLCertChain> GetRemoteSSLCertChain(
Zhi Huange818b6e2018-02-22 15:26:27 -0800158 const std::string& mid) const;
159 // Get negotiated role, if one has been negotiated.
Danil Chapovalov66cadcc2018-06-19 16:47:43 +0200160 absl::optional<rtc::SSLRole> GetDtlsRole(const std::string& mid) const;
Zhi Huange818b6e2018-02-22 15:26:27 -0800161
162 // TODO(deadbeef): GetStats isn't const because all the way down to
163 // OpenSSLStreamAdapter, GetSslCipherSuite and GetDtlsSrtpCryptoSuite are not
164 // const. Fix this.
165 bool GetStats(const std::string& mid, cricket::TransportStats* stats);
Zhi Huange818b6e2018-02-22 15:26:27 -0800166
Zhi Huange830e682018-03-30 10:48:35 -0700167 bool initial_offerer() const { return initial_offerer_ && *initial_offerer_; }
Zhi Huang365381f2018-04-13 16:44:34 -0700168
Zhi Huangb57e1692018-06-12 11:41:11 -0700169 void SetActiveResetSrtpParams(bool active_reset_srtp_params);
170
Piotr (Peter) Slatala97fc11f2018-10-18 12:57:59 -0700171 // Allows to overwrite the settings from config. You may set or reset the
172 // media transport factory on the jsep transport controller, as long as you
173 // did not call 'GetMediaTransport' or 'MaybeCreateJsepTransport'. Once Jsep
174 // transport is created, you can't change this setting.
175 void SetMediaTransportFactory(MediaTransportFactory* media_transport_factory);
176
Zhi Huange818b6e2018-02-22 15:26:27 -0800177 // All of these signals are fired on the signaling thread.
178
179 // If any transport failed => failed,
180 // Else if all completed => completed,
181 // Else if all connected => connected,
182 // Else => connecting
183 sigslot::signal1<cricket::IceConnectionState> SignalIceConnectionState;
184
Jonas Olsson635474e2018-10-18 15:58:17 +0200185 sigslot::signal1<PeerConnectionInterface::PeerConnectionState>
186 SignalConnectionState;
187 sigslot::signal1<PeerConnectionInterface::IceConnectionState>
188 SignalStandardizedIceConnectionState;
189
Zhi Huange818b6e2018-02-22 15:26:27 -0800190 // If all transports done gathering => complete,
191 // Else if any are gathering => gathering,
192 // Else => new
193 sigslot::signal1<cricket::IceGatheringState> SignalIceGatheringState;
194
195 // (mid, candidates)
196 sigslot::signal2<const std::string&, const std::vector<cricket::Candidate>&>
197 SignalIceCandidatesGathered;
198
199 sigslot::signal1<const std::vector<cricket::Candidate>&>
200 SignalIceCandidatesRemoved;
201
202 sigslot::signal1<rtc::SSLHandshakeError> SignalDtlsHandshakeError;
203
Bjorn Mellem175aa2e2018-11-08 11:23:22 -0800204 sigslot::signal<> SignalMediaTransportStateChanged;
205
Zhi Huange818b6e2018-02-22 15:26:27 -0800206 private:
Zhi Huange818b6e2018-02-22 15:26:27 -0800207 RTCError ApplyDescription_n(bool local,
208 SdpType type,
209 const cricket::SessionDescription* description);
Zhi Huangd2248f82018-04-10 14:41:03 -0700210 RTCError ValidateAndMaybeUpdateBundleGroup(
211 bool local,
212 SdpType type,
213 const cricket::SessionDescription* description);
Zhi Huange830e682018-03-30 10:48:35 -0700214 RTCError ValidateContent(const cricket::ContentInfo& content_info);
Zhi Huange818b6e2018-02-22 15:26:27 -0800215
Taylor Brandstettercbaa2542018-04-16 16:42:14 -0700216 void HandleRejectedContent(const cricket::ContentInfo& content_info,
Zhi Huangd2248f82018-04-10 14:41:03 -0700217 const cricket::SessionDescription* description);
Zhi Huang365381f2018-04-13 16:44:34 -0700218 bool HandleBundledContent(const cricket::ContentInfo& content_info);
Zhi Huange818b6e2018-02-22 15:26:27 -0800219
Zhi Huang365381f2018-04-13 16:44:34 -0700220 bool SetTransportForMid(const std::string& mid,
Taylor Brandstettercbaa2542018-04-16 16:42:14 -0700221 cricket::JsepTransport* jsep_transport);
222 void RemoveTransportForMid(const std::string& mid);
Zhi Huangd2248f82018-04-10 14:41:03 -0700223
Zhi Huange818b6e2018-02-22 15:26:27 -0800224 cricket::JsepTransportDescription CreateJsepTransportDescription(
225 cricket::ContentInfo content_info,
226 cricket::TransportInfo transport_info,
Zhi Huange830e682018-03-30 10:48:35 -0700227 const std::vector<int>& encrypted_extension_ids,
228 int rtp_abs_sendtime_extn_id);
Zhi Huange818b6e2018-02-22 15:26:27 -0800229
Danil Chapovalov66cadcc2018-06-19 16:47:43 +0200230 absl::optional<std::string> bundled_mid() const {
231 absl::optional<std::string> bundled_mid;
Taylor Brandstetter0ab56512018-04-12 10:30:48 -0700232 if (bundle_group_ && bundle_group_->FirstContentName()) {
233 bundled_mid = *(bundle_group_->FirstContentName());
Zhi Huange818b6e2018-02-22 15:26:27 -0800234 }
235 return bundled_mid;
236 }
237
238 bool IsBundled(const std::string& mid) const {
239 return bundle_group_ && bundle_group_->HasContentName(mid);
240 }
241
242 bool ShouldUpdateBundleGroup(SdpType type,
243 const cricket::SessionDescription* description);
244
245 std::vector<int> MergeEncryptedHeaderExtensionIdsForBundle(
246 const cricket::SessionDescription* description);
Zhi Huange818b6e2018-02-22 15:26:27 -0800247 std::vector<int> GetEncryptedHeaderExtensionIds(
248 const cricket::ContentInfo& content_info);
249
Zhi Huange830e682018-03-30 10:48:35 -0700250 int GetRtpAbsSendTimeHeaderExtensionId(
251 const cricket::ContentInfo& content_info);
Zhi Huange818b6e2018-02-22 15:26:27 -0800252
Zhi Huange830e682018-03-30 10:48:35 -0700253 // This method takes the BUNDLE group into account. If the JsepTransport is
254 // destroyed because of BUNDLE, it would return the transport which other
255 // transports are bundled on (In current implementation, it is the first
256 // content in the BUNDLE group).
Zhi Huang365381f2018-04-13 16:44:34 -0700257 const cricket::JsepTransport* GetJsepTransportForMid(
Zhi Huange830e682018-03-30 10:48:35 -0700258 const std::string& mid) const;
Zhi Huang365381f2018-04-13 16:44:34 -0700259 cricket::JsepTransport* GetJsepTransportForMid(const std::string& mid);
Zhi Huange830e682018-03-30 10:48:35 -0700260
261 // Get the JsepTransport without considering the BUNDLE group. Return nullptr
262 // if the JsepTransport is destroyed.
Zhi Huang365381f2018-04-13 16:44:34 -0700263 const cricket::JsepTransport* GetJsepTransportByName(
Zhi Huange830e682018-03-30 10:48:35 -0700264 const std::string& transport_name) const;
Zhi Huang365381f2018-04-13 16:44:34 -0700265 cricket::JsepTransport* GetJsepTransportByName(
Zhi Huange830e682018-03-30 10:48:35 -0700266 const std::string& transport_name);
267
Anton Sukhanov7940da02018-10-10 10:34:49 -0700268 // Creates jsep transport. Noop if transport is already created.
269 // Transport is created either during SetLocalDescription (|local| == true) or
270 // during SetRemoteDescription (|local| == false). Passing |local| helps to
271 // differentiate initiator (caller) from answerer (callee).
272 RTCError MaybeCreateJsepTransport(bool local,
273 const cricket::ContentInfo& content_info);
Zhi Huange818b6e2018-02-22 15:26:27 -0800274 void MaybeDestroyJsepTransport(const std::string& mid);
275 void DestroyAllJsepTransports_n();
276
277 void SetIceRole_n(cricket::IceRole ice_role);
278
279 cricket::IceRole DetermineIceRole(
Zhi Huang365381f2018-04-13 16:44:34 -0700280 cricket::JsepTransport* jsep_transport,
Zhi Huange818b6e2018-02-22 15:26:27 -0800281 const cricket::TransportInfo& transport_info,
282 SdpType type,
283 bool local);
284
285 std::unique_ptr<cricket::DtlsTransportInternal> CreateDtlsTransport(
286 const std::string& transport_name,
287 bool rtcp);
288
289 std::unique_ptr<webrtc::RtpTransport> CreateUnencryptedRtpTransport(
290 const std::string& transport_name,
291 rtc::PacketTransportInternal* rtp_packet_transport,
292 rtc::PacketTransportInternal* rtcp_packet_transport);
293 std::unique_ptr<webrtc::SrtpTransport> CreateSdesTransport(
294 const std::string& transport_name,
Zhi Huange830e682018-03-30 10:48:35 -0700295 cricket::DtlsTransportInternal* rtp_dtls_transport,
296 cricket::DtlsTransportInternal* rtcp_dtls_transport);
Zhi Huange818b6e2018-02-22 15:26:27 -0800297 std::unique_ptr<webrtc::DtlsSrtpTransport> CreateDtlsSrtpTransport(
298 const std::string& transport_name,
299 cricket::DtlsTransportInternal* rtp_dtls_transport,
300 cricket::DtlsTransportInternal* rtcp_dtls_transport);
301
302 // Collect all the DtlsTransports, including RTP and RTCP, from the
303 // JsepTransports. JsepTransportController can iterate all the DtlsTransports
304 // and update the aggregate states.
305 std::vector<cricket::DtlsTransportInternal*> GetDtlsTransports();
306
307 // Handlers for signals from Transport.
308 void OnTransportWritableState_n(rtc::PacketTransportInternal* transport);
309 void OnTransportReceivingState_n(rtc::PacketTransportInternal* transport);
310 void OnTransportGatheringState_n(cricket::IceTransportInternal* transport);
311 void OnTransportCandidateGathered_n(cricket::IceTransportInternal* transport,
312 const cricket::Candidate& candidate);
Zhi Huange818b6e2018-02-22 15:26:27 -0800313 void OnTransportCandidatesRemoved_n(cricket::IceTransportInternal* transport,
314 const cricket::Candidates& candidates);
315 void OnTransportRoleConflict_n(cricket::IceTransportInternal* transport);
316 void OnTransportStateChanged_n(cricket::IceTransportInternal* transport);
Bjorn Mellem175aa2e2018-11-08 11:23:22 -0800317 void OnMediaTransportStateChanged_n();
Zhi Huange818b6e2018-02-22 15:26:27 -0800318
319 void UpdateAggregateStates_n();
320
321 void OnDtlsHandshakeError(rtc::SSLHandshakeError error);
322
323 rtc::Thread* const signaling_thread_ = nullptr;
324 rtc::Thread* const network_thread_ = nullptr;
325 cricket::PortAllocator* const port_allocator_ = nullptr;
Zach Steine20867f2018-08-02 13:20:15 -0700326 AsyncResolverFactory* const async_resolver_factory_ = nullptr;
Zhi Huange818b6e2018-02-22 15:26:27 -0800327
Zhi Huang365381f2018-04-13 16:44:34 -0700328 std::map<std::string, std::unique_ptr<cricket::JsepTransport>>
Zhi Huange830e682018-03-30 10:48:35 -0700329 jsep_transports_by_name_;
Zhi Huangd2248f82018-04-10 14:41:03 -0700330 // This keeps track of the mapping between media section
Zhi Huang365381f2018-04-13 16:44:34 -0700331 // (BaseChannel/SctpTransport) and the JsepTransport underneath.
332 std::map<std::string, cricket::JsepTransport*> mid_to_transport_;
Zhi Huange818b6e2018-02-22 15:26:27 -0800333
Jonas Olsson635474e2018-10-18 15:58:17 +0200334 // Aggregate states for Transports.
335 // standardized_ice_connection_state_ is intended to replace
336 // ice_connection_state, see bugs.webrtc.org/9308
Zhi Huange818b6e2018-02-22 15:26:27 -0800337 cricket::IceConnectionState ice_connection_state_ =
338 cricket::kIceConnectionConnecting;
Jonas Olsson635474e2018-10-18 15:58:17 +0200339 PeerConnectionInterface::IceConnectionState
340 standardized_ice_connection_state_ =
341 PeerConnectionInterface::kIceConnectionNew;
342 PeerConnectionInterface::PeerConnectionState combined_connection_state_ =
343 PeerConnectionInterface::PeerConnectionState::kNew;
Zhi Huange818b6e2018-02-22 15:26:27 -0800344 cricket::IceGatheringState ice_gathering_state_ = cricket::kIceGatheringNew;
345
346 Config config_;
347 const cricket::SessionDescription* local_desc_ = nullptr;
348 const cricket::SessionDescription* remote_desc_ = nullptr;
Danil Chapovalov66cadcc2018-06-19 16:47:43 +0200349 absl::optional<bool> initial_offerer_;
Zhi Huange818b6e2018-02-22 15:26:27 -0800350
Danil Chapovalov66cadcc2018-06-19 16:47:43 +0200351 absl::optional<cricket::ContentGroup> bundle_group_;
Zhi Huange818b6e2018-02-22 15:26:27 -0800352
353 cricket::IceConfig ice_config_;
354 cricket::IceRole ice_role_ = cricket::ICEROLE_CONTROLLING;
355 uint64_t ice_tiebreaker_ = rtc::CreateRandomId64();
356 rtc::scoped_refptr<rtc::RTCCertificate> certificate_;
357 rtc::AsyncInvoker invoker_;
358
Zhi Huange818b6e2018-02-22 15:26:27 -0800359 RTC_DISALLOW_COPY_AND_ASSIGN(JsepTransportController);
360};
361
362} // namespace webrtc
363
364#endif // PC_JSEPTRANSPORTCONTROLLER_H_