forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			333 lines
		
	
	
	
		
			8.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			333 lines
		
	
	
	
		
			8.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- 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/. */
 | |
| 
 | |
| #include "UDPSocketChild.h"
 | |
| #include "mozilla/Unused.h"
 | |
| #include "mozilla/ipc/IPCStreamUtils.h"
 | |
| #include "mozilla/net/NeckoChild.h"
 | |
| #include "mozilla/dom/ContentChild.h"
 | |
| #include "mozilla/dom/PermissionMessageUtils.h"
 | |
| #include "mozilla/ipc/BackgroundChild.h"
 | |
| #include "mozilla/ipc/PBackgroundChild.h"
 | |
| #include "mozilla/ipc/BackgroundUtils.h"
 | |
| #include "mozilla/ipc/PBackgroundSharedTypes.h"
 | |
| 
 | |
| using mozilla::net::gNeckoChild;
 | |
| 
 | |
| namespace mozilla {
 | |
| namespace dom {
 | |
| 
 | |
| NS_IMPL_ISUPPORTS(UDPSocketChildBase, nsIUDPSocketChild)
 | |
| 
 | |
| UDPSocketChildBase::UDPSocketChildBase()
 | |
| : mIPCOpen(false)
 | |
| {
 | |
| }
 | |
| 
 | |
| UDPSocketChildBase::~UDPSocketChildBase()
 | |
| {
 | |
| }
 | |
| 
 | |
| void
 | |
| UDPSocketChildBase::ReleaseIPDLReference()
 | |
| {
 | |
|   MOZ_ASSERT(mIPCOpen);
 | |
|   mIPCOpen = false;
 | |
|   mSocket = nullptr;
 | |
|   this->Release();
 | |
| }
 | |
| 
 | |
| void
 | |
| UDPSocketChildBase::AddIPDLReference()
 | |
| {
 | |
|   MOZ_ASSERT(!mIPCOpen);
 | |
|   mIPCOpen = true;
 | |
|   this->AddRef();
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP_(MozExternalRefCountType) UDPSocketChild::Release(void)
 | |
| {
 | |
|   nsrefcnt refcnt = UDPSocketChildBase::Release();
 | |
|   if (refcnt == 1 && mIPCOpen) {
 | |
|     PUDPSocketChild::SendRequestDelete();
 | |
|     return 1;
 | |
|   }
 | |
|   return refcnt;
 | |
| }
 | |
| 
 | |
| UDPSocketChild::UDPSocketChild()
 | |
| :mBackgroundManager(nullptr)
 | |
| ,mLocalPort(0)
 | |
| {
 | |
| }
 | |
| 
 | |
| UDPSocketChild::~UDPSocketChild()
 | |
| {
 | |
| }
 | |
| 
 | |
| // nsIUDPSocketChild Methods
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::SetBackgroundSpinsEvents()
 | |
| {
 | |
|   using mozilla::ipc::BackgroundChild;
 | |
| 
 | |
|   mBackgroundManager =
 | |
|     BackgroundChild::GetOrCreateForCurrentThread();
 | |
|   if (NS_WARN_IF(!mBackgroundManager)) {
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::Bind(nsIUDPSocketInternal* aSocket,
 | |
|                      nsIPrincipal* aPrincipal,
 | |
|                      const nsACString& aHost,
 | |
|                      uint16_t aPort,
 | |
|                      bool aAddressReuse,
 | |
|                      bool aLoopback,
 | |
|                      uint32_t recvBufferSize,
 | |
|                      uint32_t sendBufferSize,
 | |
|                      nsIEventTarget* aMainThreadEventTarget)
 | |
| {
 | |
|   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, PromiseFlatCString(aHost).get(), aPort));
 | |
| 
 | |
|   NS_ENSURE_ARG(aSocket);
 | |
| 
 | |
|   mSocket = aSocket;
 | |
|   AddIPDLReference();
 | |
| 
 | |
|   if (mBackgroundManager) {
 | |
|     // If we want to support a passed-in principal here we'd need to
 | |
|     // convert it to a PrincipalInfo
 | |
|     MOZ_ASSERT(!aPrincipal);
 | |
|     mBackgroundManager->SendPUDPSocketConstructor(this, void_t(), mFilterName);
 | |
|   } else {
 | |
|     if (aMainThreadEventTarget) {
 | |
|       gNeckoChild->SetEventTargetForActor(this, aMainThreadEventTarget);
 | |
|     }
 | |
|     gNeckoChild->SendPUDPSocketConstructor(this, IPC::Principal(aPrincipal),
 | |
|                                            mFilterName);
 | |
|   }
 | |
| 
 | |
|   SendBind(UDPAddressInfo(nsCString(aHost), aPort), aAddressReuse, aLoopback,
 | |
|            recvBufferSize, sendBufferSize);
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::Connect(nsIUDPSocketInternal* aSocket, const nsACString & aHost, uint16_t aPort)
 | |
| {
 | |
|   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, PromiseFlatCString(aHost).get(), aPort));
 | |
| 
 | |
|   mSocket = aSocket;
 | |
| 
 | |
|   SendConnect(UDPAddressInfo(nsCString(aHost), aPort));
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::Close()
 | |
| {
 | |
|   SendClose();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::Send(const nsACString& aHost,
 | |
|                      uint16_t aPort,
 | |
|                      const uint8_t* aData,
 | |
|                      uint32_t aByteLength)
 | |
| {
 | |
|   NS_ENSURE_ARG(aData);
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %s:%u - %u bytes", __FUNCTION__, PromiseFlatCString(aHost).get(), aPort, aByteLength));
 | |
|   return SendDataInternal(UDPSocketAddr(UDPAddressInfo(nsCString(aHost), aPort)),
 | |
|                           aData, aByteLength);
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::SendWithAddr(nsINetAddr* aAddr,
 | |
|                              const uint8_t* aData,
 | |
|                              uint32_t aByteLength)
 | |
| {
 | |
|   NS_ENSURE_ARG(aAddr);
 | |
|   NS_ENSURE_ARG(aData);
 | |
| 
 | |
|   NetAddr addr;
 | |
|   aAddr->GetNetAddr(&addr);
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %u bytes", __FUNCTION__, aByteLength));
 | |
|   return SendDataInternal(UDPSocketAddr(addr), aData, aByteLength);
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::SendWithAddress(const NetAddr* aAddr,
 | |
|                                 const uint8_t* aData,
 | |
|                                 uint32_t aByteLength)
 | |
| {
 | |
|   NS_ENSURE_ARG(aAddr);
 | |
|   NS_ENSURE_ARG(aData);
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %u bytes", __FUNCTION__, aByteLength));
 | |
|   return SendDataInternal(UDPSocketAddr(*aAddr), aData, aByteLength);
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| UDPSocketChild::SendDataInternal(const UDPSocketAddr& aAddr,
 | |
|                                  const uint8_t* aData,
 | |
|                                  const uint32_t aByteLength)
 | |
| {
 | |
|   NS_ENSURE_ARG(aData);
 | |
| 
 | |
|   FallibleTArray<uint8_t> fallibleArray;
 | |
|   if (!fallibleArray.InsertElementsAt(0, aData, aByteLength, fallible)) {
 | |
|     return NS_ERROR_OUT_OF_MEMORY;
 | |
|   }
 | |
| 
 | |
|   InfallibleTArray<uint8_t> array;
 | |
|   array.SwapElements(fallibleArray);
 | |
| 
 | |
|   SendOutgoingData(array, aAddr);
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::SendBinaryStream(const nsACString& aHost,
 | |
|                                  uint16_t aPort,
 | |
|                                  nsIInputStream* aStream)
 | |
| {
 | |
|   NS_ENSURE_ARG(aStream);
 | |
| 
 | |
|   mozilla::ipc::AutoIPCStream autoStream;
 | |
|   autoStream.Serialize(aStream,
 | |
|                        static_cast<mozilla::dom::ContentChild*>(gNeckoChild->Manager()));
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, PromiseFlatCString(aHost).get(), aPort));
 | |
|   SendOutgoingData(UDPData(autoStream.TakeOptionalValue()),
 | |
|                    UDPSocketAddr(UDPAddressInfo(nsCString(aHost), aPort)));
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::JoinMulticast(const nsACString& aMulticastAddress,
 | |
|                               const nsACString& aInterface)
 | |
| {
 | |
|   SendJoinMulticast(nsCString(aMulticastAddress), nsCString(aInterface));
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::LeaveMulticast(const nsACString& aMulticastAddress,
 | |
|                                const nsACString& aInterface)
 | |
| {
 | |
|   SendLeaveMulticast(nsCString(aMulticastAddress), nsCString(aInterface));
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::GetLocalPort(uint16_t* aLocalPort)
 | |
| {
 | |
|   NS_ENSURE_ARG_POINTER(aLocalPort);
 | |
| 
 | |
|   *aLocalPort = mLocalPort;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::GetLocalAddress(nsACString& aLocalAddress)
 | |
| {
 | |
|   aLocalAddress = mLocalAddress;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::SetFilterName(const nsACString& aFilterName)
 | |
| {
 | |
|   if (!mFilterName.IsEmpty()) {
 | |
|     // filter name can only be set once.
 | |
|     return NS_ERROR_FAILURE;
 | |
|   }
 | |
|   mFilterName = aFilterName;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| NS_IMETHODIMP
 | |
| UDPSocketChild::GetFilterName(nsACString& aFilterName)
 | |
| {
 | |
|   aFilterName = mFilterName;
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| // PUDPSocketChild Methods
 | |
| mozilla::ipc::IPCResult
 | |
| UDPSocketChild::RecvCallbackOpened(const UDPAddressInfo& aAddressInfo)
 | |
| {
 | |
|   mLocalAddress = aAddressInfo.addr();
 | |
|   mLocalPort = aAddressInfo.port();
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, mLocalAddress.get(), mLocalPort));
 | |
|   nsresult rv = mSocket->CallListenerOpened();
 | |
|   mozilla::Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| // PUDPSocketChild Methods
 | |
| mozilla::ipc::IPCResult
 | |
| UDPSocketChild::RecvCallbackConnected(const UDPAddressInfo& aAddressInfo)
 | |
| {
 | |
|   mLocalAddress = aAddressInfo.addr();
 | |
|   mLocalPort = aAddressInfo.port();
 | |
| 
 | |
|   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, mLocalAddress.get(), mLocalPort));
 | |
|   nsresult rv = mSocket->CallListenerConnected();
 | |
|   mozilla::Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| UDPSocketChild::RecvCallbackClosed()
 | |
| {
 | |
|   nsresult rv = mSocket->CallListenerClosed();
 | |
|   mozilla::Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| UDPSocketChild::RecvCallbackReceivedData(const UDPAddressInfo& aAddressInfo,
 | |
|                                          InfallibleTArray<uint8_t>&& aData)
 | |
| {
 | |
|   UDPSOCKET_LOG(("%s: %s:%u length %zu", __FUNCTION__,
 | |
|                  aAddressInfo.addr().get(), aAddressInfo.port(), aData.Length()));
 | |
|   nsresult rv = mSocket->CallListenerReceivedData(aAddressInfo.addr(), aAddressInfo.port(),
 | |
|                                                   aData.Elements(), aData.Length());
 | |
|   mozilla::Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| UDPSocketChild::RecvCallbackError(const nsCString& aMessage,
 | |
|                                   const nsCString& aFilename,
 | |
|                                   const uint32_t& aLineNumber)
 | |
| {
 | |
|   UDPSOCKET_LOG(("%s: %s:%s:%u", __FUNCTION__, aMessage.get(), aFilename.get(), aLineNumber));
 | |
|   nsresult rv = mSocket->CallListenerError(aMessage, aFilename, aLineNumber);
 | |
|   mozilla::Unused << NS_WARN_IF(NS_FAILED(rv));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| } // namespace dom
 | |
| } // namespace mozilla
 | 
