blob: c3e64aba95308c44482aa4873aad620ad4f7729b [file] [log] [blame]
Harald Alvestrand05e4d082019-12-03 14:04:21 +01001/*
2 * Copyright 2019 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_DATA_CHANNEL_CONTROLLER_H_
12#define PC_DATA_CHANNEL_CONTROLLER_H_
13
14#include <map>
15#include <memory>
16#include <string>
17#include <vector>
18
19#include "pc/channel.h"
20#include "pc/data_channel.h"
Harald Alvestrand246724b2019-12-03 22:31:42 +010021#include "rtc_base/weak_ptr.h"
Harald Alvestrand05e4d082019-12-03 14:04:21 +010022
23namespace webrtc {
24
25class PeerConnection;
26
27class DataChannelController : public DataChannelProviderInterface,
28 public DataChannelSink {
29 public:
30 explicit DataChannelController(PeerConnection* pc) : pc_(pc) {}
31
Harald Alvestrandab813162020-01-09 13:29:56 +010032 // Not copyable or movable.
33 DataChannelController(DataChannelController&) = delete;
34 DataChannelController& operator=(const DataChannelController& other) = delete;
35 DataChannelController(DataChannelController&&) = delete;
36 DataChannelController& operator=(DataChannelController&& other) = delete;
37
Harald Alvestrand05e4d082019-12-03 14:04:21 +010038 // Implements DataChannelProviderInterface.
39 bool SendData(const cricket::SendDataParams& params,
40 const rtc::CopyOnWriteBuffer& payload,
41 cricket::SendDataResult* result) override;
42 bool ConnectDataChannel(DataChannel* webrtc_data_channel) override;
43 void DisconnectDataChannel(DataChannel* webrtc_data_channel) override;
44 void AddSctpDataStream(int sid) override;
45 void RemoveSctpDataStream(int sid) override;
46 bool ReadyToSendData() const override;
47
48 // Implements DataChannelSink.
49 void OnDataReceived(int channel_id,
50 DataMessageType type,
51 const rtc::CopyOnWriteBuffer& buffer) override;
52 void OnChannelClosing(int channel_id) override;
53 void OnChannelClosed(int channel_id) override;
54 void OnReadyToSend() override;
Harald Alvestrand2697ac12019-12-16 10:37:04 +010055 void OnTransportClosed() override;
Harald Alvestrand05e4d082019-12-03 14:04:21 +010056
57 // Called from PeerConnection::SetupDataChannelTransport_n
58 void SetupDataChannelTransport_n();
59 // Called from PeerConnection::TeardownDataChannelTransport_n
60 void TeardownDataChannelTransport_n();
61
62 // Called from PeerConnection::OnTransportChanged
63 // to make required changes to datachannels' transports.
64 void OnTransportChanged(
65 DataChannelTransportInterface* data_channel_transport);
66
Tomas Gunnarsson2e94de52020-06-16 16:54:10 +020067 // Called from PeerConnection::GetDataChannelStats on the signaling thread.
68 std::vector<DataChannel::Stats> GetDataChannelStats() const;
69
Harald Alvestrand05e4d082019-12-03 14:04:21 +010070 // Creates channel and adds it to the collection of DataChannels that will
71 // be offered in a SessionDescription.
72 rtc::scoped_refptr<DataChannel> InternalCreateDataChannel(
73 const std::string& label,
74 const InternalDataChannelInit*
75 config) /* RTC_RUN_ON(signaling_thread()) */;
76 void AllocateSctpSids(rtc::SSLRole role);
77
78 DataChannel* FindDataChannelBySid(int sid) const;
79
80 // Checks if any data channel has been added.
81 bool HasDataChannels() const;
82 bool HasSctpDataChannels() const {
83 RTC_DCHECK_RUN_ON(signaling_thread());
84 return !sctp_data_channels_.empty();
85 }
86 bool HasRtpDataChannels() const {
87 RTC_DCHECK_RUN_ON(signaling_thread());
88 return !rtp_data_channels_.empty();
89 }
90
Harald Alvestrand05e4d082019-12-03 14:04:21 +010091 void UpdateLocalRtpDataChannels(const cricket::StreamParamsVec& streams);
92 void UpdateRemoteRtpDataChannels(const cricket::StreamParamsVec& streams);
93
94 // Accessors
Tomas Gunnarsson7d3cfbf2020-06-15 13:47:42 +020095 cricket::DataChannelType data_channel_type() const;
96 void set_data_channel_type(cricket::DataChannelType type);
Harald Alvestrand05e4d082019-12-03 14:04:21 +010097 cricket::RtpDataChannel* rtp_data_channel() const {
98 return rtp_data_channel_;
99 }
100 void set_rtp_data_channel(cricket::RtpDataChannel* channel) {
101 rtp_data_channel_ = channel;
102 }
Tomas Gunnarsson7d3cfbf2020-06-15 13:47:42 +0200103 DataChannelTransportInterface* data_channel_transport() const;
104 void set_data_channel_transport(DataChannelTransportInterface* transport);
Harald Alvestrand05e4d082019-12-03 14:04:21 +0100105 const std::map<std::string, rtc::scoped_refptr<DataChannel>>*
Tomas Gunnarsson7d3cfbf2020-06-15 13:47:42 +0200106 rtp_data_channels() const;
Harald Alvestrand05e4d082019-12-03 14:04:21 +0100107
108 sigslot::signal1<DataChannel*>& SignalDataChannelCreated() {
109 RTC_DCHECK_RUN_ON(signaling_thread());
110 return SignalDataChannelCreated_;
111 }
112 // Called when the transport for the data channels is closed or destroyed.
113 void OnTransportChannelClosed();
114
115 void OnSctpDataChannelClosed(DataChannel* channel);
116
117 private:
118 // Parses and handles open messages. Returns true if the message is an open
119 // message, false otherwise.
120 bool HandleOpenMessage_s(const cricket::ReceiveDataParams& params,
121 const rtc::CopyOnWriteBuffer& buffer)
122 RTC_RUN_ON(signaling_thread());
123 // Called when a valid data channel OPEN message is received.
124 void OnDataChannelOpenMessage(const std::string& label,
125 const InternalDataChannelInit& config)
126 RTC_RUN_ON(signaling_thread());
127
128 void CreateRemoteRtpDataChannel(const std::string& label,
129 uint32_t remote_ssrc)
130 RTC_RUN_ON(signaling_thread());
131
132 void UpdateClosingRtpDataChannels(
133 const std::vector<std::string>& active_channels,
134 bool is_local_update) RTC_RUN_ON(signaling_thread());
135
Tomas Gunnarsson7d3cfbf2020-06-15 13:47:42 +0200136 // Called from SendData when data_channel_transport() is true.
137 bool DataChannelSendData(const cricket::SendDataParams& params,
138 const rtc::CopyOnWriteBuffer& payload,
139 cricket::SendDataResult* result);
140
Tomas Gunnarsson2e94de52020-06-16 16:54:10 +0200141 // Called when all data channels need to be notified of a transport channel
142 // (calls OnTransportChannelCreated on the signaling thread).
143 void NotifyDataChannelsOfTransportCreated();
144
Harald Alvestrand05e4d082019-12-03 14:04:21 +0100145 rtc::Thread* network_thread() const;
146 rtc::Thread* signaling_thread() const;
147
148 // Specifies which kind of data channel is allowed. This is controlled
149 // by the chrome command-line flag and constraints:
150 // 1. If chrome command-line switch 'enable-sctp-data-channels' is enabled,
151 // constraint kEnableDtlsSrtp is true, and constaint kEnableRtpDataChannels is
152 // not set or false, SCTP is allowed (DCT_SCTP);
153 // 2. If constraint kEnableRtpDataChannels is true, RTP is allowed (DCT_RTP);
154 // 3. If both 1&2 are false, data channel is not allowed (DCT_NONE).
155 cricket::DataChannelType data_channel_type_ =
156 cricket::DCT_NONE; // TODO(bugs.webrtc.org/9987): Accessed on both
157 // signaling and network thread.
158
159 // Plugin transport used for data channels. Pointer may be accessed and
160 // checked from any thread, but the object may only be touched on the
161 // network thread.
162 // TODO(bugs.webrtc.org/9987): Accessed on both signaling and network
163 // thread.
164 DataChannelTransportInterface* data_channel_transport_ = nullptr;
165
166 // Cached value of whether the data channel transport is ready to send.
167 bool data_channel_transport_ready_to_send_
168 RTC_GUARDED_BY(signaling_thread()) = false;
169
170 // |rtp_data_channel_| is used if in RTP data channel mode,
171 // |data_channel_transport_| when using SCTP.
172 cricket::RtpDataChannel* rtp_data_channel_ = nullptr;
173 // TODO(bugs.webrtc.org/9987): Accessed on both
174 // signaling and some other thread.
175
176 SctpSidAllocator sid_allocator_ /* RTC_GUARDED_BY(signaling_thread()) */;
177 std::vector<rtc::scoped_refptr<DataChannel>> sctp_data_channels_
178 RTC_GUARDED_BY(signaling_thread());
179 std::vector<rtc::scoped_refptr<DataChannel>> sctp_data_channels_to_free_
180 RTC_GUARDED_BY(signaling_thread());
181
182 // Map of label -> DataChannel
183 std::map<std::string, rtc::scoped_refptr<DataChannel>> rtp_data_channels_
184 RTC_GUARDED_BY(signaling_thread());
185
186 // Signals from |data_channel_transport_|. These are invoked on the
187 // signaling thread.
Tomas Gunnarsson7d3cfbf2020-06-15 13:47:42 +0200188 // TODO(bugs.webrtc.org/11547): These '_s' signals likely all belong on the
189 // network thread.
Harald Alvestrand05e4d082019-12-03 14:04:21 +0100190 sigslot::signal1<bool> SignalDataChannelTransportWritable_s
191 RTC_GUARDED_BY(signaling_thread());
192 sigslot::signal2<const cricket::ReceiveDataParams&,
193 const rtc::CopyOnWriteBuffer&>
194 SignalDataChannelTransportReceivedData_s
195 RTC_GUARDED_BY(signaling_thread());
196 sigslot::signal1<int> SignalDataChannelTransportChannelClosing_s
197 RTC_GUARDED_BY(signaling_thread());
198 sigslot::signal1<int> SignalDataChannelTransportChannelClosed_s
199 RTC_GUARDED_BY(signaling_thread());
200
201 sigslot::signal1<DataChannel*> SignalDataChannelCreated_
202 RTC_GUARDED_BY(signaling_thread());
203
204 // Used to invoke data channel transport signals on the signaling thread.
205 std::unique_ptr<rtc::AsyncInvoker> data_channel_transport_invoker_
206 RTC_GUARDED_BY(network_thread());
207
208 // Owning PeerConnection.
209 PeerConnection* const pc_;
Harald Alvestrand246724b2019-12-03 22:31:42 +0100210 rtc::WeakPtrFactory<DataChannelController> weak_factory_{this};
Harald Alvestrand05e4d082019-12-03 14:04:21 +0100211};
212
213} // namespace webrtc
214
215#endif // PC_DATA_CHANNEL_CONTROLLER_H_