зеркало из https://github.com/mozilla/gecko-dev.git
554 строки
17 KiB
C++
554 строки
17 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 "DCPresentationChannelDescription.h"
|
|
#include "mozilla/dom/ContentProcessManager.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/ipc/InputStreamUtils.h"
|
|
#include "mozilla/Unused.h"
|
|
#include "nsIPresentationDeviceManager.h"
|
|
#include "nsIPresentationSessionTransport.h"
|
|
#include "nsIPresentationSessionTransportBuilder.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
#include "PresentationBuilderParent.h"
|
|
#include "PresentationParent.h"
|
|
#include "PresentationService.h"
|
|
#include "PresentationSessionInfo.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
namespace {
|
|
|
|
class PresentationTransportBuilderConstructorIPC final :
|
|
public nsIPresentationTransportBuilderConstructor
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIPRESENTATIONTRANSPORTBUILDERCONSTRUCTOR
|
|
|
|
explicit PresentationTransportBuilderConstructorIPC(PresentationParent* aParent)
|
|
: mParent(aParent)
|
|
{
|
|
}
|
|
|
|
private:
|
|
virtual ~PresentationTransportBuilderConstructorIPC() = default;
|
|
|
|
RefPtr<PresentationParent> mParent;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(PresentationTransportBuilderConstructorIPC,
|
|
nsIPresentationTransportBuilderConstructor)
|
|
|
|
NS_IMETHODIMP
|
|
PresentationTransportBuilderConstructorIPC::CreateTransportBuilder(
|
|
uint8_t aType,
|
|
nsIPresentationSessionTransportBuilder** aRetval)
|
|
{
|
|
if (NS_WARN_IF(!aRetval)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
*aRetval = nullptr;
|
|
|
|
if (NS_WARN_IF(aType != nsIPresentationChannelDescription::TYPE_TCP &&
|
|
aType != nsIPresentationChannelDescription::TYPE_DATACHANNEL)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
if (XRE_IsContentProcess()) {
|
|
MOZ_ASSERT(false,
|
|
"CreateTransportBuilder can only be invoked in parent process.");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsCOMPtr<nsIPresentationSessionTransportBuilder> builder;
|
|
if (aType == nsIPresentationChannelDescription::TYPE_TCP) {
|
|
builder = do_CreateInstance(PRESENTATION_TCP_SESSION_TRANSPORT_CONTRACTID);
|
|
} else {
|
|
builder = new PresentationBuilderParent(mParent);
|
|
}
|
|
|
|
if (NS_WARN_IF(!builder)) {
|
|
return NS_ERROR_DOM_OPERATION_ERR;
|
|
}
|
|
|
|
builder.forget(aRetval);
|
|
return NS_OK;
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
/*
|
|
* Implementation of PresentationParent
|
|
*/
|
|
|
|
NS_IMPL_ISUPPORTS(PresentationParent,
|
|
nsIPresentationAvailabilityListener,
|
|
nsIPresentationSessionListener,
|
|
nsIPresentationRespondingListener)
|
|
|
|
PresentationParent::PresentationParent()
|
|
{
|
|
}
|
|
|
|
/* virtual */ PresentationParent::~PresentationParent()
|
|
{
|
|
}
|
|
|
|
bool
|
|
PresentationParent::Init(ContentParentId aContentParentId)
|
|
{
|
|
MOZ_ASSERT(!mService);
|
|
mService = do_GetService(PRESENTATION_SERVICE_CONTRACTID);
|
|
mChildId = aContentParentId;
|
|
return NS_WARN_IF(!mService) ? false : true;
|
|
}
|
|
|
|
void
|
|
PresentationParent::ActorDestroy(ActorDestroyReason aWhy)
|
|
{
|
|
mActorDestroyed = true;
|
|
|
|
for (uint32_t i = 0; i < mSessionIdsAtController.Length(); i++) {
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->
|
|
UnregisterSessionListener(mSessionIdsAtController[i],
|
|
nsIPresentationService::ROLE_CONTROLLER)));
|
|
}
|
|
mSessionIdsAtController.Clear();
|
|
|
|
for (uint32_t i = 0; i < mSessionIdsAtReceiver.Length(); i++) {
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->
|
|
UnregisterSessionListener(mSessionIdsAtReceiver[i], nsIPresentationService::ROLE_RECEIVER)));
|
|
}
|
|
mSessionIdsAtReceiver.Clear();
|
|
|
|
for (uint32_t i = 0; i < mWindowIds.Length(); i++) {
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->
|
|
UnregisterRespondingListener(mWindowIds[i])));
|
|
}
|
|
mWindowIds.Clear();
|
|
|
|
if (!mContentAvailabilityUrls.IsEmpty()) {
|
|
mService->UnregisterAvailabilityListener(mContentAvailabilityUrls, this);
|
|
}
|
|
mService = nullptr;
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::RecvPPresentationRequestConstructor(
|
|
PPresentationRequestParent* aActor,
|
|
const PresentationIPCRequest& aRequest)
|
|
{
|
|
PresentationRequestParent* actor = static_cast<PresentationRequestParent*>(aActor);
|
|
|
|
nsresult rv = NS_ERROR_FAILURE;
|
|
switch (aRequest.type()) {
|
|
case PresentationIPCRequest::TStartSessionRequest:
|
|
rv = actor->DoRequest(aRequest.get_StartSessionRequest());
|
|
break;
|
|
case PresentationIPCRequest::TSendSessionMessageRequest:
|
|
rv = actor->DoRequest(aRequest.get_SendSessionMessageRequest());
|
|
break;
|
|
case PresentationIPCRequest::TCloseSessionRequest:
|
|
rv = actor->DoRequest(aRequest.get_CloseSessionRequest());
|
|
break;
|
|
case PresentationIPCRequest::TTerminateSessionRequest:
|
|
rv = actor->DoRequest(aRequest.get_TerminateSessionRequest());
|
|
break;
|
|
case PresentationIPCRequest::TReconnectSessionRequest:
|
|
rv = actor->DoRequest(aRequest.get_ReconnectSessionRequest());
|
|
break;
|
|
case PresentationIPCRequest::TBuildTransportRequest:
|
|
rv = actor->DoRequest(aRequest.get_BuildTransportRequest());
|
|
break;
|
|
default:
|
|
MOZ_CRASH("Unknown PresentationIPCRequest type");
|
|
}
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return IPC_FAIL_NO_REASON(this);
|
|
}
|
|
return IPC_OK();
|
|
}
|
|
|
|
PPresentationRequestParent*
|
|
PresentationParent::AllocPPresentationRequestParent(
|
|
const PresentationIPCRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
RefPtr<PresentationRequestParent> actor = new PresentationRequestParent(mService, mChildId);
|
|
return actor.forget().take();
|
|
}
|
|
|
|
bool
|
|
PresentationParent::DeallocPPresentationRequestParent(
|
|
PPresentationRequestParent* aActor)
|
|
{
|
|
RefPtr<PresentationRequestParent> actor =
|
|
dont_AddRef(static_cast<PresentationRequestParent*>(aActor));
|
|
return true;
|
|
}
|
|
|
|
PPresentationBuilderParent*
|
|
PresentationParent::AllocPPresentationBuilderParent(const nsString& aSessionId,
|
|
const uint8_t& aRole)
|
|
{
|
|
NS_NOTREACHED("We should never be manually allocating AllocPPresentationBuilderParent actors");
|
|
return nullptr;
|
|
}
|
|
|
|
bool
|
|
PresentationParent::DeallocPPresentationBuilderParent(
|
|
PPresentationBuilderParent* aActor)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::Recv__delete__()
|
|
{
|
|
return IPC_OK();
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::RecvRegisterAvailabilityHandler(
|
|
nsTArray<nsString>&& aAvailabilityUrls)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->RegisterAvailabilityListener(
|
|
aAvailabilityUrls,
|
|
this)));
|
|
mContentAvailabilityUrls.AppendElements(aAvailabilityUrls);
|
|
return IPC_OK();
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::RecvUnregisterAvailabilityHandler(
|
|
nsTArray<nsString>&& aAvailabilityUrls)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->UnregisterAvailabilityListener(
|
|
aAvailabilityUrls,
|
|
this)));
|
|
for (const auto& url : aAvailabilityUrls) {
|
|
mContentAvailabilityUrls.RemoveElement(url);
|
|
}
|
|
return IPC_OK();
|
|
}
|
|
|
|
/* virtual */ mozilla::ipc::IPCResult
|
|
PresentationParent::RecvRegisterSessionHandler(const nsString& aSessionId,
|
|
const uint8_t& aRole)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aSessionId, aRole, OtherPid()))) {
|
|
return IPC_OK();
|
|
}
|
|
|
|
if (nsIPresentationService::ROLE_CONTROLLER == aRole) {
|
|
mSessionIdsAtController.AppendElement(aSessionId);
|
|
} else {
|
|
mSessionIdsAtReceiver.AppendElement(aSessionId);
|
|
}
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->RegisterSessionListener(aSessionId, aRole, this)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
/* virtual */ mozilla::ipc::IPCResult
|
|
PresentationParent::RecvUnregisterSessionHandler(const nsString& aSessionId,
|
|
const uint8_t& aRole)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
if (nsIPresentationService::ROLE_CONTROLLER == aRole) {
|
|
mSessionIdsAtController.RemoveElement(aSessionId);
|
|
} else {
|
|
mSessionIdsAtReceiver.RemoveElement(aSessionId);
|
|
}
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->UnregisterSessionListener(aSessionId, aRole)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
/* virtual */ mozilla::ipc::IPCResult
|
|
PresentationParent::RecvRegisterRespondingHandler(const uint64_t& aWindowId)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
mWindowIds.AppendElement(aWindowId);
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->RegisterRespondingListener(aWindowId, this)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
/* virtual */ mozilla::ipc::IPCResult
|
|
PresentationParent::RecvUnregisterRespondingHandler(const uint64_t& aWindowId)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
mWindowIds.RemoveElement(aWindowId);
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->UnregisterRespondingListener(aWindowId)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationParent::NotifyAvailableChange(const nsTArray<nsString>& aAvailabilityUrls,
|
|
bool aAvailable)
|
|
{
|
|
if (NS_WARN_IF(mActorDestroyed ||
|
|
!SendNotifyAvailableChange(aAvailabilityUrls,
|
|
aAvailable))) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationParent::NotifyStateChange(const nsAString& aSessionId,
|
|
uint16_t aState,
|
|
nsresult aReason)
|
|
{
|
|
if (NS_WARN_IF(mActorDestroyed ||
|
|
!SendNotifySessionStateChange(nsString(aSessionId),
|
|
aState,
|
|
aReason))) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationParent::NotifyMessage(const nsAString& aSessionId,
|
|
const nsACString& aData,
|
|
bool aIsBinary)
|
|
{
|
|
if (NS_WARN_IF(mActorDestroyed ||
|
|
!SendNotifyMessage(nsString(aSessionId),
|
|
nsCString(aData),
|
|
aIsBinary))) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationParent::NotifySessionConnect(uint64_t aWindowId,
|
|
const nsAString& aSessionId)
|
|
{
|
|
if (NS_WARN_IF(mActorDestroyed ||
|
|
!SendNotifySessionConnect(aWindowId, nsString(aSessionId)))) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::RecvNotifyReceiverReady(const nsString& aSessionId,
|
|
const uint64_t& aWindowId,
|
|
const bool& aIsLoading)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
nsCOMPtr<nsIPresentationTransportBuilderConstructor> constructor =
|
|
new PresentationTransportBuilderConstructorIPC(this);
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->NotifyReceiverReady(aSessionId,
|
|
aWindowId,
|
|
aIsLoading,
|
|
constructor)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
mozilla::ipc::IPCResult
|
|
PresentationParent::RecvNotifyTransportClosed(const nsString& aSessionId,
|
|
const uint8_t& aRole,
|
|
const nsresult& aReason)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
Unused << NS_WARN_IF(NS_FAILED(mService->NotifyTransportClosed(aSessionId, aRole, aReason)));
|
|
return IPC_OK();
|
|
}
|
|
|
|
/*
|
|
* Implementation of PresentationRequestParent
|
|
*/
|
|
|
|
NS_IMPL_ISUPPORTS(PresentationRequestParent, nsIPresentationServiceCallback)
|
|
|
|
PresentationRequestParent::PresentationRequestParent(nsIPresentationService* aService,
|
|
ContentParentId aContentParentId)
|
|
: mService(aService)
|
|
, mChildId(aContentParentId)
|
|
{
|
|
}
|
|
|
|
PresentationRequestParent::~PresentationRequestParent()
|
|
{
|
|
}
|
|
|
|
void
|
|
PresentationRequestParent::ActorDestroy(ActorDestroyReason aWhy)
|
|
{
|
|
mActorDestroyed = true;
|
|
mService = nullptr;
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const StartSessionRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
mSessionId = aRequest.sessionId();
|
|
|
|
nsCOMPtr<nsIDOMEventTarget> eventTarget;
|
|
ContentProcessManager* cpm = ContentProcessManager::GetSingleton();
|
|
RefPtr<TabParent> tp =
|
|
cpm->GetTopLevelTabParentByProcessAndTabId(mChildId, aRequest.tabId());
|
|
if (tp) {
|
|
eventTarget = do_QueryInterface(tp->GetOwnerElement());
|
|
}
|
|
|
|
RefPtr<PresentationParent> parent = static_cast<PresentationParent*>(Manager());
|
|
nsCOMPtr<nsIPresentationTransportBuilderConstructor> constructor =
|
|
new PresentationTransportBuilderConstructorIPC(parent);
|
|
return mService->StartSession(aRequest.urls(), aRequest.sessionId(),
|
|
aRequest.origin(), aRequest.deviceId(),
|
|
aRequest.windowId(), eventTarget,
|
|
aRequest.principal(), this, constructor);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const SendSessionMessageRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aRequest.sessionId(), aRequest.role(), OtherPid()))) {
|
|
return SendResponse(NS_ERROR_DOM_SECURITY_ERR);
|
|
}
|
|
|
|
nsresult rv = mService->SendSessionMessage(aRequest.sessionId(),
|
|
aRequest.role(),
|
|
aRequest.data());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return SendResponse(rv);
|
|
}
|
|
return SendResponse(NS_OK);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const CloseSessionRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aRequest.sessionId(), aRequest.role(), OtherPid()))) {
|
|
return SendResponse(NS_ERROR_DOM_SECURITY_ERR);
|
|
}
|
|
|
|
nsresult rv = mService->CloseSession(aRequest.sessionId(),
|
|
aRequest.role(),
|
|
aRequest.closedReason());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return SendResponse(rv);
|
|
}
|
|
return SendResponse(NS_OK);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const TerminateSessionRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aRequest.sessionId(), aRequest.role(), OtherPid()))) {
|
|
return SendResponse(NS_ERROR_DOM_SECURITY_ERR);
|
|
}
|
|
|
|
nsresult rv = mService->TerminateSession(aRequest.sessionId(), aRequest.role());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return SendResponse(rv);
|
|
}
|
|
return SendResponse(NS_OK);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const ReconnectSessionRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aRequest.sessionId(), aRequest.role(), OtherPid()))) {
|
|
|
|
// NOTE: Return NS_ERROR_DOM_NOT_FOUND_ERR here to match the spec.
|
|
// https://w3c.github.io/presentation-api/#reconnecting-to-a-presentation
|
|
return SendResponse(NS_ERROR_DOM_NOT_FOUND_ERR);
|
|
}
|
|
|
|
mSessionId = aRequest.sessionId();
|
|
return mService->ReconnectSession(aRequest.urls(),
|
|
aRequest.sessionId(),
|
|
aRequest.role(),
|
|
this);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::DoRequest(const BuildTransportRequest& aRequest)
|
|
{
|
|
MOZ_ASSERT(mService);
|
|
|
|
// Validate the accessibility (primarily for receiver side) so that a
|
|
// compromised child process can't fake the ID.
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(mService.get())->
|
|
IsSessionAccessible(aRequest.sessionId(), aRequest.role(), OtherPid()))) {
|
|
return SendResponse(NS_ERROR_DOM_SECURITY_ERR);
|
|
}
|
|
|
|
nsresult rv = mService->BuildTransport(aRequest.sessionId(), aRequest.role());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return SendResponse(rv);
|
|
}
|
|
return SendResponse(NS_OK);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationRequestParent::NotifySuccess(const nsAString& aUrl)
|
|
{
|
|
Unused << SendNotifyRequestUrlSelected(nsString(aUrl));
|
|
return SendResponse(NS_OK);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
PresentationRequestParent::NotifyError(nsresult aError)
|
|
{
|
|
return SendResponse(aError);
|
|
}
|
|
|
|
nsresult
|
|
PresentationRequestParent::SendResponse(nsresult aResult)
|
|
{
|
|
if (NS_WARN_IF(mActorDestroyed || !Send__delete__(this, aResult))) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|