blob: e11647f2cab3d0f0ece967b6c9a506eb8fdbb69a [file] [log] [blame]
Harald Alvestrand00cf34c2019-12-02 08:56:021/*
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
Harald Alvestrand05e4d082019-12-03 13:04:2111#include "pc/data_channel_controller.h"
12
Harald Alvestrand5761e7b2021-01-29 14:45:0813#include <algorithm>
Harald Alvestrand05e4d082019-12-03 13:04:2114#include <utility>
Harald Alvestrand00cf34c2019-12-02 08:56:0215
Harald Alvestrand5761e7b2021-01-29 14:45:0816#include "absl/algorithm/container.h"
17#include "absl/types/optional.h"
18#include "api/peer_connection_interface.h"
19#include "api/rtc_error.h"
Harald Alvestrand00cf34c2019-12-02 08:56:0220#include "pc/peer_connection.h"
21#include "pc/sctp_utils.h"
Harald Alvestrand5761e7b2021-01-29 14:45:0822#include "rtc_base/location.h"
23#include "rtc_base/logging.h"
Niels Möller236e36c2021-03-23 08:23:1024#include "rtc_base/task_utils/to_queued_task.h"
Harald Alvestrand00cf34c2019-12-02 08:56:0225
26namespace webrtc {
27
Harald Alvestrand05e4d082019-12-03 13:04:2128bool DataChannelController::HasDataChannels() const {
29 RTC_DCHECK_RUN_ON(signaling_thread());
Harald Alvestrand7af57c62021-04-16 11:12:1430 return !sctp_data_channels_.empty();
Harald Alvestrand05e4d082019-12-03 13:04:2131}
32
Florent Castellid95b1492021-05-10 09:29:5633bool DataChannelController::SendData(int sid,
34 const SendDataParams& params,
Harald Alvestrand05e4d082019-12-03 13:04:2135 const rtc::CopyOnWriteBuffer& payload,
36 cricket::SendDataResult* result) {
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:4237 if (data_channel_transport())
Florent Castellid95b1492021-05-10 09:29:5638 return DataChannelSendData(sid, params, payload, result);
Harald Alvestrand00cf34c2019-12-02 08:56:0239 RTC_LOG(LS_ERROR) << "SendData called before transport is ready";
40 return false;
41}
42
Harald Alvestrand05e4d082019-12-03 13:04:2143bool DataChannelController::ConnectDataChannel(
Taylor Brandstetter3a034e12020-07-09 22:32:3444 SctpDataChannel* webrtc_data_channel) {
45 RTC_DCHECK_RUN_ON(signaling_thread());
46 if (!data_channel_transport()) {
47 // Don't log an error here, because DataChannels are expected to call
48 // ConnectDataChannel in this state. It's the only way to initially tell
49 // whether or not the underlying transport is ready.
50 return false;
Harald Alvestrand00cf34c2019-12-02 08:56:0251 }
Taylor Brandstetter3a034e12020-07-09 22:32:3452 SignalDataChannelTransportWritable_s.connect(
53 webrtc_data_channel, &SctpDataChannel::OnTransportReady);
54 SignalDataChannelTransportReceivedData_s.connect(
55 webrtc_data_channel, &SctpDataChannel::OnDataReceived);
56 SignalDataChannelTransportChannelClosing_s.connect(
57 webrtc_data_channel, &SctpDataChannel::OnClosingProcedureStartedRemotely);
58 SignalDataChannelTransportChannelClosed_s.connect(
59 webrtc_data_channel, &SctpDataChannel::OnClosingProcedureComplete);
60 return true;
61}
62
63void DataChannelController::DisconnectDataChannel(
64 SctpDataChannel* webrtc_data_channel) {
65 RTC_DCHECK_RUN_ON(signaling_thread());
66 if (!data_channel_transport()) {
67 RTC_LOG(LS_ERROR)
68 << "DisconnectDataChannel called when sctp_transport_ is NULL.";
69 return;
Harald Alvestrand00cf34c2019-12-02 08:56:0270 }
Taylor Brandstetter3a034e12020-07-09 22:32:3471 SignalDataChannelTransportWritable_s.disconnect(webrtc_data_channel);
72 SignalDataChannelTransportReceivedData_s.disconnect(webrtc_data_channel);
73 SignalDataChannelTransportChannelClosing_s.disconnect(webrtc_data_channel);
74 SignalDataChannelTransportChannelClosed_s.disconnect(webrtc_data_channel);
Harald Alvestrand00cf34c2019-12-02 08:56:0275}
76
Harald Alvestrand05e4d082019-12-03 13:04:2177void DataChannelController::AddSctpDataStream(int sid) {
Harald Alvestrand00cf34c2019-12-02 08:56:0278 if (data_channel_transport()) {
79 network_thread()->Invoke<void>(RTC_FROM_HERE, [this, sid] {
80 if (data_channel_transport()) {
81 data_channel_transport()->OpenChannel(sid);
82 }
83 });
84 }
85}
86
Harald Alvestrand05e4d082019-12-03 13:04:2187void DataChannelController::RemoveSctpDataStream(int sid) {
Harald Alvestrand00cf34c2019-12-02 08:56:0288 if (data_channel_transport()) {
89 network_thread()->Invoke<void>(RTC_FROM_HERE, [this, sid] {
90 if (data_channel_transport()) {
91 data_channel_transport()->CloseChannel(sid);
92 }
93 });
94 }
95}
96
Harald Alvestrand05e4d082019-12-03 13:04:2197bool DataChannelController::ReadyToSendData() const {
Harald Alvestrand00cf34c2019-12-02 08:56:0298 RTC_DCHECK_RUN_ON(signaling_thread());
Harald Alvestrand7af57c62021-04-16 11:12:1499 return (data_channel_transport() && data_channel_transport_ready_to_send_);
Harald Alvestrand00cf34c2019-12-02 08:56:02100}
101
Harald Alvestrand05e4d082019-12-03 13:04:21102void DataChannelController::OnDataReceived(
Harald Alvestrand00cf34c2019-12-02 08:56:02103 int channel_id,
104 DataMessageType type,
105 const rtc::CopyOnWriteBuffer& buffer) {
106 RTC_DCHECK_RUN_ON(network_thread());
107 cricket::ReceiveDataParams params;
108 params.sid = channel_id;
Florent Castellid95b1492021-05-10 09:29:56109 params.type = type;
Niels Möller236e36c2021-03-23 08:23:10110 signaling_thread()->PostTask(
111 ToQueuedTask([self = weak_factory_.GetWeakPtr(), params, buffer] {
112 if (self) {
113 RTC_DCHECK_RUN_ON(self->signaling_thread());
114 // TODO(bugs.webrtc.org/11547): The data being received should be
115 // delivered on the network thread. The way HandleOpenMessage_s works
116 // right now is that it's called for all types of buffers and operates
117 // as a selector function. Change this so that it's only called for
118 // buffers that it should be able to handle. Once we do that, we can
119 // deliver all other buffers on the network thread (change
120 // SignalDataChannelTransportReceivedData_s to
121 // SignalDataChannelTransportReceivedData_n).
122 if (!self->HandleOpenMessage_s(params, buffer)) {
123 self->SignalDataChannelTransportReceivedData_s(params, buffer);
124 }
Harald Alvestrand00cf34c2019-12-02 08:56:02125 }
Niels Möller236e36c2021-03-23 08:23:10126 }));
Harald Alvestrand00cf34c2019-12-02 08:56:02127}
128
Harald Alvestrand05e4d082019-12-03 13:04:21129void DataChannelController::OnChannelClosing(int channel_id) {
Harald Alvestrand00cf34c2019-12-02 08:56:02130 RTC_DCHECK_RUN_ON(network_thread());
Niels Möller236e36c2021-03-23 08:23:10131 signaling_thread()->PostTask(
132 ToQueuedTask([self = weak_factory_.GetWeakPtr(), channel_id] {
133 if (self) {
134 RTC_DCHECK_RUN_ON(self->signaling_thread());
135 self->SignalDataChannelTransportChannelClosing_s(channel_id);
136 }
137 }));
Harald Alvestrand00cf34c2019-12-02 08:56:02138}
139
Harald Alvestrand05e4d082019-12-03 13:04:21140void DataChannelController::OnChannelClosed(int channel_id) {
Harald Alvestrand00cf34c2019-12-02 08:56:02141 RTC_DCHECK_RUN_ON(network_thread());
Niels Möller236e36c2021-03-23 08:23:10142 signaling_thread()->PostTask(
143 ToQueuedTask([self = weak_factory_.GetWeakPtr(), channel_id] {
144 if (self) {
145 RTC_DCHECK_RUN_ON(self->signaling_thread());
146 self->SignalDataChannelTransportChannelClosed_s(channel_id);
147 }
148 }));
Harald Alvestrand00cf34c2019-12-02 08:56:02149}
150
Harald Alvestrand05e4d082019-12-03 13:04:21151void DataChannelController::OnReadyToSend() {
Harald Alvestrand00cf34c2019-12-02 08:56:02152 RTC_DCHECK_RUN_ON(network_thread());
Niels Möller236e36c2021-03-23 08:23:10153 signaling_thread()->PostTask(
154 ToQueuedTask([self = weak_factory_.GetWeakPtr()] {
155 if (self) {
156 RTC_DCHECK_RUN_ON(self->signaling_thread());
157 self->data_channel_transport_ready_to_send_ = true;
158 self->SignalDataChannelTransportWritable_s(
159 self->data_channel_transport_ready_to_send_);
160 }
161 }));
Harald Alvestrand00cf34c2019-12-02 08:56:02162}
163
Florent Castellidcb9ffc2021-06-29 12:58:23164void DataChannelController::OnTransportClosed(RTCError error) {
Harald Alvestrand2697ac12019-12-16 09:37:04165 RTC_DCHECK_RUN_ON(network_thread());
Niels Möller236e36c2021-03-23 08:23:10166 signaling_thread()->PostTask(
Florent Castellidcb9ffc2021-06-29 12:58:23167 ToQueuedTask([self = weak_factory_.GetWeakPtr(), error] {
Niels Möller236e36c2021-03-23 08:23:10168 if (self) {
169 RTC_DCHECK_RUN_ON(self->signaling_thread());
Florent Castellidcb9ffc2021-06-29 12:58:23170 self->OnTransportChannelClosed(error);
Niels Möller236e36c2021-03-23 08:23:10171 }
172 }));
Harald Alvestrand2697ac12019-12-16 09:37:04173}
174
Harald Alvestrand05e4d082019-12-03 13:04:21175void DataChannelController::SetupDataChannelTransport_n() {
Harald Alvestrand00cf34c2019-12-02 08:56:02176 RTC_DCHECK_RUN_ON(network_thread());
Tomas Gunnarsson2e94de52020-06-16 14:54:10177
178 // There's a new data channel transport. This needs to be signaled to the
Artem Titov880fa812021-07-30 20:30:23179 // `sctp_data_channels_` so that they can reopen and reconnect. This is
Tomas Gunnarsson2e94de52020-06-16 14:54:10180 // necessary when bundling is applied.
181 NotifyDataChannelsOfTransportCreated();
Harald Alvestrand00cf34c2019-12-02 08:56:02182}
183
Harald Alvestrand05e4d082019-12-03 13:04:21184void DataChannelController::TeardownDataChannelTransport_n() {
Harald Alvestrand00cf34c2019-12-02 08:56:02185 RTC_DCHECK_RUN_ON(network_thread());
Harald Alvestrand00cf34c2019-12-02 08:56:02186 if (data_channel_transport()) {
187 data_channel_transport()->SetDataSink(nullptr);
188 }
189 set_data_channel_transport(nullptr);
190}
191
Harald Alvestrand05e4d082019-12-03 13:04:21192void DataChannelController::OnTransportChanged(
Harald Alvestrand00cf34c2019-12-02 08:56:02193 DataChannelTransportInterface* new_data_channel_transport) {
194 RTC_DCHECK_RUN_ON(network_thread());
195 if (data_channel_transport() &&
196 data_channel_transport() != new_data_channel_transport) {
Artem Titov880fa812021-07-30 20:30:23197 // Changed which data channel transport is used for `sctp_mid_` (eg. now
Harald Alvestrand00cf34c2019-12-02 08:56:02198 // it's bundled).
199 data_channel_transport()->SetDataSink(nullptr);
200 set_data_channel_transport(new_data_channel_transport);
201 if (new_data_channel_transport) {
202 new_data_channel_transport->SetDataSink(this);
203
204 // There's a new data channel transport. This needs to be signaled to the
Artem Titov880fa812021-07-30 20:30:23205 // `sctp_data_channels_` so that they can reopen and reconnect. This is
Harald Alvestrand00cf34c2019-12-02 08:56:02206 // necessary when bundling is applied.
Tomas Gunnarsson2e94de52020-06-16 14:54:10207 NotifyDataChannelsOfTransportCreated();
Harald Alvestrand00cf34c2019-12-02 08:56:02208 }
209 }
210}
211
Taylor Brandstetter3a034e12020-07-09 22:32:34212std::vector<DataChannelStats> DataChannelController::GetDataChannelStats()
Tomas Gunnarsson2e94de52020-06-16 14:54:10213 const {
214 RTC_DCHECK_RUN_ON(signaling_thread());
Taylor Brandstetter3a034e12020-07-09 22:32:34215 std::vector<DataChannelStats> stats;
Tomas Gunnarsson2e94de52020-06-16 14:54:10216 stats.reserve(sctp_data_channels_.size());
217 for (const auto& channel : sctp_data_channels_)
218 stats.push_back(channel->GetStats());
219 return stats;
220}
221
Harald Alvestrand05e4d082019-12-03 13:04:21222bool DataChannelController::HandleOpenMessage_s(
Harald Alvestrand00cf34c2019-12-02 08:56:02223 const cricket::ReceiveDataParams& params,
224 const rtc::CopyOnWriteBuffer& buffer) {
Florent Castellid95b1492021-05-10 09:29:56225 if (params.type == DataMessageType::kControl && IsOpenMessage(buffer)) {
Harald Alvestrand00cf34c2019-12-02 08:56:02226 // Received OPEN message; parse and signal that a new data channel should
227 // be created.
228 std::string label;
229 InternalDataChannelInit config;
Harald Alvestrand7af57c62021-04-16 11:12:14230 config.id = params.sid;
Harald Alvestrand00cf34c2019-12-02 08:56:02231 if (!ParseDataChannelOpenMessage(buffer, &label, &config)) {
Harald Alvestrand7af57c62021-04-16 11:12:14232 RTC_LOG(LS_WARNING) << "Failed to parse the OPEN message for sid "
233 << params.sid;
Harald Alvestrand00cf34c2019-12-02 08:56:02234 return true;
235 }
236 config.open_handshake_role = InternalDataChannelInit::kAcker;
237 OnDataChannelOpenMessage(label, config);
238 return true;
239 }
240 return false;
241}
242
Harald Alvestrand05e4d082019-12-03 13:04:21243void DataChannelController::OnDataChannelOpenMessage(
Harald Alvestrand00cf34c2019-12-02 08:56:02244 const std::string& label,
245 const InternalDataChannelInit& config) {
Taylor Brandstetter3a034e12020-07-09 22:32:34246 rtc::scoped_refptr<DataChannelInterface> channel(
247 InternalCreateDataChannelWithProxy(label, &config));
Harald Alvestrand00cf34c2019-12-02 08:56:02248 if (!channel.get()) {
249 RTC_LOG(LS_ERROR) << "Failed to create DataChannel from the OPEN message.";
250 return;
251 }
252
Taylor Brandstetter3a034e12020-07-09 22:32:34253 pc_->Observer()->OnDataChannel(std::move(channel));
Harald Alvestrand05e4d082019-12-03 13:04:21254 pc_->NoteDataAddedEvent();
Harald Alvestrand00cf34c2019-12-02 08:56:02255}
256
Taylor Brandstetter3a034e12020-07-09 22:32:34257rtc::scoped_refptr<DataChannelInterface>
258DataChannelController::InternalCreateDataChannelWithProxy(
Harald Alvestrand00cf34c2019-12-02 08:56:02259 const std::string& label,
260 const InternalDataChannelInit* config) {
Harald Alvestrand05e4d082019-12-03 13:04:21261 RTC_DCHECK_RUN_ON(signaling_thread());
Harald Alvestrand00cf34c2019-12-02 08:56:02262 if (pc_->IsClosed()) {
263 return nullptr;
264 }
Florent Castelli516e2842021-04-19 13:29:50265
266 rtc::scoped_refptr<SctpDataChannel> channel =
267 InternalCreateSctpDataChannel(label, config);
268 if (channel) {
269 return SctpDataChannel::CreateProxy(channel);
Harald Alvestrand00cf34c2019-12-02 08:56:02270 }
271
Taylor Brandstetter3a034e12020-07-09 22:32:34272 return nullptr;
273}
274
Taylor Brandstetter3a034e12020-07-09 22:32:34275rtc::scoped_refptr<SctpDataChannel>
276DataChannelController::InternalCreateSctpDataChannel(
277 const std::string& label,
278 const InternalDataChannelInit* config) {
279 RTC_DCHECK_RUN_ON(signaling_thread());
280 InternalDataChannelInit new_config =
281 config ? (*config) : InternalDataChannelInit();
282 if (new_config.id < 0) {
283 rtc::SSLRole role;
284 if ((pc_->GetSctpSslRole(&role)) &&
285 !sid_allocator_.AllocateSid(role, &new_config.id)) {
286 RTC_LOG(LS_ERROR) << "No id can be allocated for the SCTP data channel.";
287 return nullptr;
288 }
289 } else if (!sid_allocator_.ReserveSid(new_config.id)) {
290 RTC_LOG(LS_ERROR) << "Failed to create a SCTP data channel "
291 "because the id is already in use or out of range.";
292 return nullptr;
293 }
294 rtc::scoped_refptr<SctpDataChannel> channel(SctpDataChannel::Create(
295 this, label, new_config, signaling_thread(), network_thread()));
Harald Alvestrand00cf34c2019-12-02 08:56:02296 if (!channel) {
297 sid_allocator_.ReleaseSid(new_config.id);
298 return nullptr;
299 }
Taylor Brandstetter3a034e12020-07-09 22:32:34300 sctp_data_channels_.push_back(channel);
301 channel->SignalClosed.connect(pc_, &PeerConnection::OnSctpDataChannelClosed);
302 SignalSctpDataChannelCreated_(channel.get());
Harald Alvestrand00cf34c2019-12-02 08:56:02303 return channel;
304}
305
Harald Alvestrand05e4d082019-12-03 13:04:21306void DataChannelController::AllocateSctpSids(rtc::SSLRole role) {
307 RTC_DCHECK_RUN_ON(signaling_thread());
Taylor Brandstetter3a034e12020-07-09 22:32:34308 std::vector<rtc::scoped_refptr<SctpDataChannel>> channels_to_close;
Harald Alvestrand05e4d082019-12-03 13:04:21309 for (const auto& channel : sctp_data_channels_) {
Harald Alvestrand00cf34c2019-12-02 08:56:02310 if (channel->id() < 0) {
311 int sid;
312 if (!sid_allocator_.AllocateSid(role, &sid)) {
313 RTC_LOG(LS_ERROR) << "Failed to allocate SCTP sid, closing channel.";
314 channels_to_close.push_back(channel);
315 continue;
316 }
317 channel->SetSctpSid(sid);
318 }
319 }
320 // Since closing modifies the list of channels, we have to do the actual
321 // closing outside the loop.
322 for (const auto& channel : channels_to_close) {
Harald Alvestranddfbfb462019-12-08 04:55:43323 channel->CloseAbruptlyWithDataChannelFailure("Failed to allocate SCTP SID");
Harald Alvestrand00cf34c2019-12-02 08:56:02324 }
325}
326
Taylor Brandstetter3a034e12020-07-09 22:32:34327void DataChannelController::OnSctpDataChannelClosed(SctpDataChannel* channel) {
Harald Alvestrand05e4d082019-12-03 13:04:21328 RTC_DCHECK_RUN_ON(signaling_thread());
329 for (auto it = sctp_data_channels_.begin(); it != sctp_data_channels_.end();
330 ++it) {
Harald Alvestrand00cf34c2019-12-02 08:56:02331 if (it->get() == channel) {
332 if (channel->id() >= 0) {
333 // After the closing procedure is done, it's safe to use this ID for
334 // another data channel.
335 sid_allocator_.ReleaseSid(channel->id());
336 }
337 // Since this method is triggered by a signal from the DataChannel,
338 // we can't free it directly here; we need to free it asynchronously.
Harald Alvestrand05e4d082019-12-03 13:04:21339 sctp_data_channels_to_free_.push_back(*it);
340 sctp_data_channels_.erase(it);
Harald Alvestrand246724b2019-12-03 21:31:42341 signaling_thread()->PostTask(
Niels Möller236e36c2021-03-23 08:23:10342 ToQueuedTask([self = weak_factory_.GetWeakPtr()] {
Harald Alvestrand246724b2019-12-03 21:31:42343 if (self) {
344 RTC_DCHECK_RUN_ON(self->signaling_thread());
345 self->sctp_data_channels_to_free_.clear();
346 }
Niels Möller236e36c2021-03-23 08:23:10347 }));
Harald Alvestrand00cf34c2019-12-02 08:56:02348 return;
349 }
350 }
351}
352
Florent Castellidcb9ffc2021-06-29 12:58:23353void DataChannelController::OnTransportChannelClosed(RTCError error) {
Harald Alvestrand05e4d082019-12-03 13:04:21354 RTC_DCHECK_RUN_ON(signaling_thread());
Harald Alvestrand7af57c62021-04-16 11:12:14355 // Use a temporary copy of the SCTP DataChannel list because the
Harald Alvestrand05e4d082019-12-03 13:04:21356 // DataChannel may callback to us and try to modify the list.
Taylor Brandstetter3a034e12020-07-09 22:32:34357 std::vector<rtc::scoped_refptr<SctpDataChannel>> temp_sctp_dcs;
Harald Alvestrand05e4d082019-12-03 13:04:21358 temp_sctp_dcs.swap(sctp_data_channels_);
359 for (const auto& channel : temp_sctp_dcs) {
Florent Castellidcb9ffc2021-06-29 12:58:23360 channel->OnTransportChannelClosed(error);
Harald Alvestrand05e4d082019-12-03 13:04:21361 }
362}
363
Taylor Brandstetter3a034e12020-07-09 22:32:34364SctpDataChannel* DataChannelController::FindDataChannelBySid(int sid) const {
Harald Alvestrand05e4d082019-12-03 13:04:21365 RTC_DCHECK_RUN_ON(signaling_thread());
366 for (const auto& channel : sctp_data_channels_) {
367 if (channel->id() == sid) {
368 return channel;
369 }
370 }
371 return nullptr;
372}
373
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:42374DataChannelTransportInterface* DataChannelController::data_channel_transport()
375 const {
376 // TODO(bugs.webrtc.org/11547): Only allow this accessor to be called on the
377 // network thread.
378 // RTC_DCHECK_RUN_ON(network_thread());
379 return data_channel_transport_;
380}
381
382void DataChannelController::set_data_channel_transport(
383 DataChannelTransportInterface* transport) {
384 RTC_DCHECK_RUN_ON(network_thread());
385 data_channel_transport_ = transport;
386}
387
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:42388bool DataChannelController::DataChannelSendData(
Florent Castellid95b1492021-05-10 09:29:56389 int sid,
390 const SendDataParams& params,
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:42391 const rtc::CopyOnWriteBuffer& payload,
392 cricket::SendDataResult* result) {
393 // TODO(bugs.webrtc.org/11547): Expect method to be called on the network
394 // thread instead. Remove the Invoke() below and move assocated state to
395 // the network thread.
396 RTC_DCHECK_RUN_ON(signaling_thread());
397 RTC_DCHECK(data_channel_transport());
398
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:42399 RTCError error = network_thread()->Invoke<RTCError>(
Florent Castellid95b1492021-05-10 09:29:56400 RTC_FROM_HERE, [this, sid, params, payload] {
401 return data_channel_transport()->SendData(sid, params, payload);
Tomas Gunnarsson7d3cfbf2020-06-15 11:47:42402 });
403
404 if (error.ok()) {
405 *result = cricket::SendDataResult::SDR_SUCCESS;
406 return true;
407 } else if (error.type() == RTCErrorType::RESOURCE_EXHAUSTED) {
408 // SCTP transport uses RESOURCE_EXHAUSTED when it's blocked.
409 // TODO(mellem): Stop using RTCError here and get rid of the mapping.
410 *result = cricket::SendDataResult::SDR_BLOCK;
411 return false;
412 }
413 *result = cricket::SendDataResult::SDR_ERROR;
414 return false;
415}
416
Tomas Gunnarsson2e94de52020-06-16 14:54:10417void DataChannelController::NotifyDataChannelsOfTransportCreated() {
418 RTC_DCHECK_RUN_ON(network_thread());
Niels Möller236e36c2021-03-23 08:23:10419 signaling_thread()->PostTask(
420 ToQueuedTask([self = weak_factory_.GetWeakPtr()] {
421 if (self) {
422 RTC_DCHECK_RUN_ON(self->signaling_thread());
423 for (const auto& channel : self->sctp_data_channels_) {
424 channel->OnTransportChannelCreated();
425 }
Tomas Gunnarsson2e94de52020-06-16 14:54:10426 }
Niels Möller236e36c2021-03-23 08:23:10427 }));
Tomas Gunnarsson2e94de52020-06-16 14:54:10428}
429
Harald Alvestrand05e4d082019-12-03 13:04:21430rtc::Thread* DataChannelController::network_thread() const {
431 return pc_->network_thread();
432}
433rtc::Thread* DataChannelController::signaling_thread() const {
434 return pc_->signaling_thread();
435}
436
Harald Alvestrand00cf34c2019-12-02 08:56:02437} // namespace webrtc