| /* |
| * Copyright 2018 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "pc/sctp_transport.h" |
| |
| #include <algorithm> |
| #include <optional> |
| #include <utility> |
| |
| #include "api/dtls_transport_interface.h" |
| #include "api/priority.h" |
| #include "api/sequence_checker.h" |
| #include "rtc_base/checks.h" |
| #include "rtc_base/logging.h" |
| |
| namespace webrtc { |
| |
| SctpTransport::SctpTransport( |
| std::unique_ptr<cricket::SctpTransportInternal> internal, |
| rtc::scoped_refptr<DtlsTransport> dtls_transport) |
| : owner_thread_(rtc::Thread::Current()), |
| info_(SctpTransportState::kConnecting, |
| dtls_transport, |
| /*max_message_size=*/std::nullopt, |
| /*max_channels=*/std::nullopt), |
| internal_sctp_transport_(std::move(internal)), |
| dtls_transport_(dtls_transport) { |
| RTC_DCHECK(internal_sctp_transport_.get()); |
| RTC_DCHECK(dtls_transport_.get()); |
| |
| dtls_transport_->internal()->SubscribeDtlsTransportState( |
| [this](cricket::DtlsTransportInternal* transport, |
| DtlsTransportState state) { |
| OnDtlsStateChange(transport, state); |
| }); |
| |
| internal_sctp_transport_->SetDtlsTransport(dtls_transport->internal()); |
| internal_sctp_transport_->SetOnConnectedCallback( |
| [this]() { OnAssociationChangeCommunicationUp(); }); |
| } |
| |
| SctpTransport::~SctpTransport() { |
| // We depend on the network thread to call Clear() before dropping |
| // its last reference to this object. |
| RTC_DCHECK(owner_thread_->IsCurrent() || !internal_sctp_transport_); |
| } |
| |
| SctpTransportInformation SctpTransport::Information() const { |
| // TODO(tommi): Update PeerConnection::GetSctpTransport to hand out a proxy |
| // to the transport so that we can be sure that methods get called on the |
| // expected thread. Chromium currently calls this method from |
| // TransceiverStateSurfacer. |
| if (!owner_thread_->IsCurrent()) { |
| return owner_thread_->BlockingCall([this] { return Information(); }); |
| } |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| return info_; |
| } |
| |
| void SctpTransport::RegisterObserver(SctpTransportObserverInterface* observer) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(observer); |
| RTC_DCHECK(!observer_); |
| observer_ = observer; |
| } |
| |
| void SctpTransport::UnregisterObserver() { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| observer_ = nullptr; |
| } |
| |
| RTCError SctpTransport::OpenChannel(int channel_id, PriorityValue priority) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| internal_sctp_transport_->OpenStream(channel_id, priority); |
| return RTCError::OK(); |
| } |
| |
| RTCError SctpTransport::SendData(int channel_id, |
| const SendDataParams& params, |
| const rtc::CopyOnWriteBuffer& buffer) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| return internal_sctp_transport_->SendData(channel_id, params, buffer); |
| } |
| |
| RTCError SctpTransport::CloseChannel(int channel_id) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| internal_sctp_transport_->ResetStream(channel_id); |
| return RTCError::OK(); |
| } |
| |
| void SctpTransport::SetDataSink(DataChannelSink* sink) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| internal_sctp_transport_->SetDataChannelSink(sink); |
| } |
| |
| bool SctpTransport::IsReadyToSend() const { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| return internal_sctp_transport_->ReadyToSendData(); |
| } |
| |
| size_t SctpTransport::buffered_amount(int channel_id) const { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| return internal_sctp_transport_->buffered_amount(channel_id); |
| } |
| |
| size_t SctpTransport::buffered_amount_low_threshold(int channel_id) const { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| return internal_sctp_transport_->buffered_amount_low_threshold(channel_id); |
| } |
| |
| void SctpTransport::SetBufferedAmountLowThreshold(int channel_id, |
| size_t bytes) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| internal_sctp_transport_->SetBufferedAmountLowThreshold(channel_id, bytes); |
| } |
| |
| rtc::scoped_refptr<DtlsTransportInterface> SctpTransport::dtls_transport() |
| const { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| return dtls_transport_; |
| } |
| |
| // Internal functions |
| void SctpTransport::Clear() { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal()); |
| // Note that we delete internal_sctp_transport_, but |
| // only drop the reference to dtls_transport_. |
| dtls_transport_ = nullptr; |
| internal_sctp_transport_ = nullptr; |
| UpdateInformation(SctpTransportState::kClosed); |
| } |
| |
| void SctpTransport::Start(int local_port, |
| int remote_port, |
| int max_message_size) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| info_ = SctpTransportInformation(info_.state(), info_.dtls_transport(), |
| max_message_size, info_.MaxChannels()); |
| |
| if (!internal()->Start(local_port, remote_port, max_message_size)) { |
| RTC_LOG(LS_ERROR) << "Failed to push down SCTP parameters, closing."; |
| UpdateInformation(SctpTransportState::kClosed); |
| } |
| } |
| |
| void SctpTransport::UpdateInformation(SctpTransportState state) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| bool must_send_update = (state != info_.state()); |
| // TODO(https://bugs.webrtc.org/10358): Update max channels from internal |
| // SCTP transport when available. |
| if (internal_sctp_transport_) { |
| info_ = SctpTransportInformation( |
| state, dtls_transport_, info_.MaxMessageSize(), info_.MaxChannels()); |
| } else { |
| info_ = SctpTransportInformation( |
| state, dtls_transport_, info_.MaxMessageSize(), info_.MaxChannels()); |
| } |
| |
| if (observer_ && must_send_update) { |
| observer_->OnStateChange(info_); |
| } |
| } |
| |
| void SctpTransport::OnAssociationChangeCommunicationUp() { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_DCHECK(internal_sctp_transport_); |
| if (internal_sctp_transport_->max_outbound_streams() && |
| internal_sctp_transport_->max_inbound_streams()) { |
| int max_channels = |
| std::min(*(internal_sctp_transport_->max_outbound_streams()), |
| *(internal_sctp_transport_->max_inbound_streams())); |
| // Record max channels. |
| info_ = SctpTransportInformation(info_.state(), info_.dtls_transport(), |
| info_.MaxMessageSize(), max_channels); |
| } |
| |
| UpdateInformation(SctpTransportState::kConnected); |
| } |
| |
| void SctpTransport::OnDtlsStateChange(cricket::DtlsTransportInternal* transport, |
| DtlsTransportState state) { |
| RTC_DCHECK_RUN_ON(owner_thread_); |
| RTC_CHECK(transport == dtls_transport_->internal()); |
| if (state == DtlsTransportState::kClosed || |
| state == DtlsTransportState::kFailed) { |
| UpdateInformation(SctpTransportState::kClosed); |
| // TODO(http://bugs.webrtc.org/11090): Close all the data channels |
| } |
| } |
| |
| } // namespace webrtc |