diff options
Diffstat (limited to 'dom/media/webrtc/transport/WebrtcTCPSocketWrapper.cpp')
-rw-r--r-- | dom/media/webrtc/transport/WebrtcTCPSocketWrapper.cpp | 123 |
1 files changed, 123 insertions, 0 deletions
diff --git a/dom/media/webrtc/transport/WebrtcTCPSocketWrapper.cpp b/dom/media/webrtc/transport/WebrtcTCPSocketWrapper.cpp new file mode 100644 index 0000000000..c126e6b791 --- /dev/null +++ b/dom/media/webrtc/transport/WebrtcTCPSocketWrapper.cpp @@ -0,0 +1,123 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set sw=2 ts=8 et tw=80 ft=cpp : */ +/* 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/. */ + +#include "WebrtcTCPSocketWrapper.h" + +#include "mozilla/net/WebrtcTCPSocketChild.h" + +#include "nsNetCID.h" +#include "nsProxyRelease.h" +#include "nsServiceManagerUtils.h" + +#include "nr_socket_proxy_config.h" + +namespace mozilla::net { + +using std::shared_ptr; + +WebrtcTCPSocketWrapper::WebrtcTCPSocketWrapper( + WebrtcTCPSocketCallback* aCallbacks) + : mProxyCallbacks(aCallbacks), + mWebrtcTCPSocket(nullptr), + mMainThread(nullptr), + mSocketThread(nullptr) { + mMainThread = GetMainThreadEventTarget(); + mSocketThread = do_GetService(NS_SOCKETTRANSPORTSERVICE_CONTRACTID); + MOZ_RELEASE_ASSERT(mMainThread, "no main thread"); + MOZ_RELEASE_ASSERT(mSocketThread, "no socket thread"); +} + +WebrtcTCPSocketWrapper::~WebrtcTCPSocketWrapper() { + MOZ_ASSERT(!mWebrtcTCPSocket, "webrtc TCP socket non-null"); + + // If we're never opened then we never get an OnClose from our parent process. + // We need to release our callbacks here safely. + NS_ProxyRelease("WebrtcTCPSocketWrapper::CleanUpCallbacks", mSocketThread, + mProxyCallbacks.forget()); +} + +void WebrtcTCPSocketWrapper::AsyncOpen( + const nsCString& aHost, const int& aPort, const nsCString& aLocalAddress, + const int& aLocalPort, bool aUseTls, + const shared_ptr<NrSocketProxyConfig>& aConfig) { + if (!NS_IsMainThread()) { + MOZ_ALWAYS_SUCCEEDS(mMainThread->Dispatch( + NewRunnableMethod<const nsCString, const int, const nsCString, + const int, bool, + const shared_ptr<NrSocketProxyConfig>>( + "WebrtcTCPSocketWrapper::AsyncOpen", this, + &WebrtcTCPSocketWrapper::AsyncOpen, aHost, aPort, aLocalAddress, + aLocalPort, aUseTls, aConfig))); + return; + } + + MOZ_ASSERT(!mWebrtcTCPSocket, "wrapper already open"); + mWebrtcTCPSocket = new WebrtcTCPSocketChild(this); + mWebrtcTCPSocket->AsyncOpen(aHost, aPort, aLocalAddress, aLocalPort, aUseTls, + aConfig); +} + +void WebrtcTCPSocketWrapper::SendWrite(nsTArray<uint8_t>&& aReadData) { + if (!NS_IsMainThread()) { + MOZ_ALWAYS_SUCCEEDS( + mMainThread->Dispatch(NewRunnableMethod<nsTArray<uint8_t>&&>( + "WebrtcTCPSocketWrapper::SendWrite", this, + &WebrtcTCPSocketWrapper::SendWrite, std::move(aReadData)))); + return; + } + + MOZ_ASSERT(mWebrtcTCPSocket, "webrtc TCP socket should be non-null"); + mWebrtcTCPSocket->SendWrite(aReadData); +} + +void WebrtcTCPSocketWrapper::Close() { + if (!NS_IsMainThread()) { + MOZ_ALWAYS_SUCCEEDS(mMainThread->Dispatch( + NewRunnableMethod("WebrtcTCPSocketWrapper::Close", this, + &WebrtcTCPSocketWrapper::Close))); + return; + } + + // We're only open if we have a channel. Also Close() should be idempotent. + if (mWebrtcTCPSocket) { + RefPtr<WebrtcTCPSocketChild> child = mWebrtcTCPSocket; + mWebrtcTCPSocket = nullptr; + child->SendClose(); + } +} + +void WebrtcTCPSocketWrapper::OnClose(nsresult aReason) { + MOZ_ASSERT(NS_IsMainThread(), "not on main thread"); + MOZ_ASSERT(mProxyCallbacks, "webrtc TCP callbacks should be non-null"); + + MOZ_ALWAYS_SUCCEEDS(mSocketThread->Dispatch(NewRunnableMethod<nsresult>( + "WebrtcTCPSocketWrapper::OnClose", mProxyCallbacks, + &WebrtcTCPSocketCallback::OnClose, aReason))); + + NS_ProxyRelease("WebrtcTCPSocketWrapper::CleanUpCallbacks", mSocketThread, + mProxyCallbacks.forget()); +} + +void WebrtcTCPSocketWrapper::OnRead(nsTArray<uint8_t>&& aReadData) { + MOZ_ASSERT(NS_IsMainThread(), "not on main thread"); + MOZ_ASSERT(mProxyCallbacks, "webrtc TCP callbacks should be non-null"); + + MOZ_ALWAYS_SUCCEEDS( + mSocketThread->Dispatch(NewRunnableMethod<nsTArray<uint8_t>&&>( + "WebrtcTCPSocketWrapper::OnRead", mProxyCallbacks, + &WebrtcTCPSocketCallback::OnRead, std::move(aReadData)))); +} + +void WebrtcTCPSocketWrapper::OnConnected(const nsACString& aProxyType) { + MOZ_ASSERT(NS_IsMainThread(), "not on main thread"); + MOZ_ASSERT(mProxyCallbacks, "webrtc TCP callbacks should be non-null"); + + MOZ_ALWAYS_SUCCEEDS(mSocketThread->Dispatch(NewRunnableMethod<nsCString>( + "WebrtcTCPSocketWrapper::OnConnected", mProxyCallbacks, + &WebrtcTCPSocketCallback::OnConnected, aProxyType))); +} + +} // namespace mozilla::net |