/* * Copyright 2020 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. */ #ifndef PC_RTP_DATA_CHANNEL_H_ #define PC_RTP_DATA_CHANNEL_H_ #include #include #include "api/data_channel_interface.h" #include "api/priority.h" #include "api/scoped_refptr.h" #include "api/transport/data_channel_transport_interface.h" #include "media/base/media_channel.h" #include "pc/channel.h" #include "pc/data_channel_utils.h" #include "rtc_base/async_invoker.h" #include "rtc_base/third_party/sigslot/sigslot.h" namespace webrtc { class RtpDataChannel; // TODO(deadbeef): Once RTP data channels go away, get rid of this and have // DataChannel depend on SctpTransportInternal (pure virtual SctpTransport // interface) instead. class RtpDataChannelProviderInterface { public: // Sends the data to the transport. virtual bool SendData(const cricket::SendDataParams& params, const rtc::CopyOnWriteBuffer& payload, cricket::SendDataResult* result) = 0; // Connects to the transport signals. virtual bool ConnectDataChannel(RtpDataChannel* data_channel) = 0; // Disconnects from the transport signals. virtual void DisconnectDataChannel(RtpDataChannel* data_channel) = 0; // Returns true if the transport channel is ready to send data. virtual bool ReadyToSendData() const = 0; protected: virtual ~RtpDataChannelProviderInterface() {} }; // RtpDataChannel is an implementation of the DataChannelInterface based on // libjingle's data engine. It provides an implementation of unreliable data // channels. // DataChannel states: // kConnecting: The channel has been created the transport might not yet be // ready. // kOpen: The channel have a local SSRC set by a call to UpdateSendSsrc // and a remote SSRC set by call to UpdateReceiveSsrc and the transport // has been writable once. // kClosing: DataChannelInterface::Close has been called or UpdateReceiveSsrc // has been called with SSRC==0 // kClosed: Both UpdateReceiveSsrc and UpdateSendSsrc has been called with // SSRC==0. class RtpDataChannel : public DataChannelInterface, public sigslot::has_slots<> { public: static rtc::scoped_refptr Create( RtpDataChannelProviderInterface* provider, const std::string& label, const DataChannelInit& config, rtc::Thread* signaling_thread); // Instantiates an API proxy for a DataChannel instance that will be handed // out to external callers. static rtc::scoped_refptr CreateProxy( rtc::scoped_refptr channel); void RegisterObserver(DataChannelObserver* observer) override; void UnregisterObserver() override; std::string label() const override { return label_; } bool reliable() const override { return false; } bool ordered() const override { return config_.ordered; } // Backwards compatible accessors uint16_t maxRetransmitTime() const override { return config_.maxRetransmitTime ? *config_.maxRetransmitTime : static_cast(-1); } uint16_t maxRetransmits() const override { return config_.maxRetransmits ? *config_.maxRetransmits : static_cast(-1); } absl::optional maxPacketLifeTime() const override { return config_.maxRetransmitTime; } absl::optional maxRetransmitsOpt() const override { return config_.maxRetransmits; } std::string protocol() const override { return config_.protocol; } bool negotiated() const override { return config_.negotiated; } int id() const override { return config_.id; } Priority priority() const override { return config_.priority ? *config_.priority : Priority::kLow; } virtual int internal_id() const { return internal_id_; } uint64_t buffered_amount() const override { return 0; } void Close() override; DataState state() const override; RTCError error() const override; uint32_t messages_sent() const override; uint64_t bytes_sent() const override; uint32_t messages_received() const override; uint64_t bytes_received() const override; bool Send(const DataBuffer& buffer) override; // Close immediately, ignoring any queued data or closing procedure. // This is called when SDP indicates a channel should be removed. void CloseAbruptlyWithError(RTCError error); // Called when the channel's ready to use. That can happen when the // underlying DataMediaChannel becomes ready, or when this channel is a new // stream on an existing DataMediaChannel, and we've finished negotiation. void OnChannelReady(bool writable); // Slots for provider to connect signals to. void OnDataReceived(const cricket::ReceiveDataParams& params, const rtc::CopyOnWriteBuffer& payload); // Called when the transport channel is unusable. // This method makes sure the DataChannel is disconnected and changes state // to kClosed. void OnTransportChannelClosed(); DataChannelStats GetStats() const; // The remote peer requested that this channel should be closed. void RemotePeerRequestClose(); // Set the SSRC this channel should use to send data on the // underlying data engine. |send_ssrc| == 0 means that the channel is no // longer part of the session negotiation. void SetSendSsrc(uint32_t send_ssrc); // Set the SSRC this channel should use to receive data from the // underlying data engine. void SetReceiveSsrc(uint32_t receive_ssrc); // Emitted when state transitions to kOpen. sigslot::signal1 SignalOpened; // Emitted when state transitions to kClosed. sigslot::signal1 SignalClosed; // Reset the allocator for internal ID values for testing, so that // the internal IDs generated are predictable. Test only. static void ResetInternalIdAllocatorForTesting(int new_value); protected: RtpDataChannel(const DataChannelInit& config, RtpDataChannelProviderInterface* client, const std::string& label, rtc::Thread* signaling_thread); ~RtpDataChannel() override; private: bool Init(); void UpdateState(); void SetState(DataState state); void DisconnectFromProvider(); void DeliverQueuedReceivedData(); bool SendDataMessage(const DataBuffer& buffer); rtc::Thread* const signaling_thread_; const int internal_id_; const std::string label_; const DataChannelInit config_; DataChannelObserver* observer_ RTC_GUARDED_BY(signaling_thread_) = nullptr; DataState state_ RTC_GUARDED_BY(signaling_thread_) = kConnecting; RTCError error_ RTC_GUARDED_BY(signaling_thread_); uint32_t messages_sent_ RTC_GUARDED_BY(signaling_thread_) = 0; uint64_t bytes_sent_ RTC_GUARDED_BY(signaling_thread_) = 0; uint32_t messages_received_ RTC_GUARDED_BY(signaling_thread_) = 0; uint64_t bytes_received_ RTC_GUARDED_BY(signaling_thread_) = 0; RtpDataChannelProviderInterface* const provider_; bool connected_to_provider_ RTC_GUARDED_BY(signaling_thread_) = false; bool send_ssrc_set_ RTC_GUARDED_BY(signaling_thread_) = false; bool receive_ssrc_set_ RTC_GUARDED_BY(signaling_thread_) = false; bool writable_ RTC_GUARDED_BY(signaling_thread_) = false; uint32_t send_ssrc_ RTC_GUARDED_BY(signaling_thread_) = 0; uint32_t receive_ssrc_ RTC_GUARDED_BY(signaling_thread_) = 0; PacketQueue queued_received_data_ RTC_GUARDED_BY(signaling_thread_); }; } // namespace webrtc #endif // PC_RTP_DATA_CHANNEL_H_