/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef DOM_WEBTRANSPORT_API_WEBTRANSPORT__H_ #define DOM_WEBTRANSPORT_API_WEBTRANSPORT__H_ #include "nsCOMPtr.h" #include "nsTArray.h" #include "nsISupports.h" #include "nsTHashMap.h" #include "nsWrapperCache.h" #include "nsPIDOMWindow.h" #include "mozilla/dom/Promise.h" #include "mozilla/dom/WebTransportBinding.h" #include "mozilla/dom/WebTransportChild.h" #include "mozilla/dom/WebTransportSendStream.h" #include "mozilla/dom/WebTransportReceiveStream.h" #include "mozilla/dom/WebTransportStreams.h" #include "mozilla/ipc/DataPipe.h" namespace mozilla::dom { class WebTransportError; class WebTransportDatagramDuplexStream; class WebTransportIncomingStreamsAlgorithms; class ReadableStream; class WritableStream; using BidirectionalPair = std::pair, RefPtr>; struct DatagramEntry { DatagramEntry(nsTArray&& aData, const mozilla::TimeStamp& aTimeStamp) : mBuffer(std::move(aData)), mTimeStamp(aTimeStamp) {} DatagramEntry(Span& aData, const mozilla::TimeStamp& aTimeStamp) : mBuffer(aData), mTimeStamp(aTimeStamp) {} nsTArray mBuffer; mozilla::TimeStamp mTimeStamp; }; class WebTransport final : public nsISupports, public nsWrapperCache { friend class WebTransportIncomingStreamsAlgorithms; // For mSendStreams/mReceiveStreams friend class WebTransportSendStream; friend class WebTransportReceiveStream; public: explicit WebTransport(nsIGlobalObject* aGlobal); NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_WRAPPERCACHE_CLASS(WebTransport) enum class WebTransportState { CONNECTING, CONNECTED, CLOSED, FAILED }; static void NotifyBFCacheOnMainThread(nsPIDOMWindowInner* aInner, bool aCreated); void NotifyToWindow(bool aCreated) const; void Init(const GlobalObject& aGlobal, const nsAString& aUrl, const WebTransportOptions& aOptions, ErrorResult& aError); void ResolveWaitingConnection(WebTransportReliabilityMode aReliability); void RejectWaitingConnection(nsresult aRv); bool ParseURL(const nsAString& aURL) const; // this calls CloseNative(), which doesn't actually run script. See bug // 1810942 MOZ_CAN_RUN_SCRIPT_BOUNDARY void Cleanup( WebTransportError* aError, const WebTransportCloseInfo* aCloseInfo, ErrorResult& aRv); // From Parent void NewBidirectionalStream( uint64_t aStreamId, const RefPtr& aIncoming, const RefPtr& aOutgoing); void NewUnidirectionalStream( uint64_t aStreamId, const RefPtr& aStream); void NewDatagramReceived(nsTArray&& aData, const mozilla::TimeStamp& aTimeStamp); void RemoteClosed(bool aCleanly, const uint32_t& aCode, const nsACString& aReason); void OnStreamResetOrStopSending(uint64_t aStreamId, const StreamResetOrStopSendingError& aError); // WebIDL Boilerplate nsIGlobalObject* GetParentObject() const; JSObject* WrapObject(JSContext* aCx, JS::Handle aGivenProto) override; // WebIDL Interface static already_AddRefed Constructor( const GlobalObject& aGlobal, const nsAString& aUrl, const WebTransportOptions& aOptions, ErrorResult& aError); already_AddRefed GetStats(ErrorResult& aError); already_AddRefed Ready() { return do_AddRef(mReady); } WebTransportReliabilityMode Reliability(); WebTransportCongestionControl CongestionControl(); already_AddRefed Closed() { return do_AddRef(mClosed); } MOZ_CAN_RUN_SCRIPT void Close(const WebTransportCloseInfo& aOptions, ErrorResult& aRv); already_AddRefed GetDatagrams( ErrorResult& aRv); already_AddRefed CreateBidirectionalStream( const WebTransportSendStreamOptions& aOptions, ErrorResult& aRv); already_AddRefed CreateUnidirectionalStream( const WebTransportSendStreamOptions& aOptions, ErrorResult& aRv); MOZ_CAN_RUN_SCRIPT_BOUNDARY already_AddRefed IncomingBidirectionalStreams(); MOZ_CAN_RUN_SCRIPT_BOUNDARY already_AddRefed IncomingUnidirectionalStreams(); void Shutdown() {} private: ~WebTransport(); template MOZ_CAN_RUN_SCRIPT_BOUNDARY void PropagateError(Stream* aStream, WebTransportError* aError); nsCOMPtr mGlobal; // We are the owner of WebTransportChild. We must call Shutdown() on it // before we're destroyed. RefPtr mChild; // Spec in 5.8 says it can't be GC'd while CONNECTING or CONNECTED. We won't // hold ref which we drop on CLOSED or FAILED because a reference is also held // by IPC. We drop the IPC connection and by proxy the reference when it goes // to FAILED or CLOSED. // Spec-defined slots: // ordered sets, but we can't have duplicates, and this spec only appends. // Order is visible due to // https://w3c.github.io/webtransport/#webtransport-procedures step 10: "For // each sendStream in sendStreams, error sendStream with error." nsTHashMap> mSendStreams; nsTHashMap> mReceiveStreams; WebTransportState mState; RefPtr mReady; // XXX may not need to be a RefPtr, since we own it through the Streams RefPtr mIncomingBidirectionalAlgorithm; RefPtr mIncomingUnidirectionalAlgorithm; WebTransportReliabilityMode mReliability; // Incoming streams get queued here. Use a TArray though it's working as // a FIFO - rarely will there be more than one entry in these arrays, so // the overhead of mozilla::Queue is unneeded nsTArray>> mUnidirectionalStreams; nsTArray>> mBidirectionalStreams; // These are created in the constructor RefPtr mIncomingUnidirectionalStreams; RefPtr mIncomingBidirectionalStreams; RefPtr mDatagrams; RefPtr mClosed; }; } // namespace mozilla::dom #endif // DOM_WEBTRANSPORT_API_WEBTRANSPORT__H_