зеркало из https://github.com/mozilla/gecko-dev.git
110 строки
3.8 KiB
C++
110 строки
3.8 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
/* 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 "mozilla/dom/JSProcessActorBinding.h"
|
|
#include "mozilla/dom/JSProcessActorParent.h"
|
|
#include "mozilla/dom/InProcessChild.h"
|
|
#include "mozilla/dom/InProcessParent.h"
|
|
|
|
namespace mozilla::dom {
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(JSProcessActorParent, JSActor, mManager)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(JSProcessActorParent, JSActor)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(JSProcessActorParent)
|
|
NS_INTERFACE_MAP_END_INHERITING(JSActor)
|
|
|
|
NS_IMPL_ADDREF_INHERITED(JSProcessActorParent, JSActor)
|
|
NS_IMPL_RELEASE_INHERITED(JSProcessActorParent, JSActor)
|
|
|
|
JSObject* JSProcessActorParent::WrapObject(JSContext* aCx,
|
|
JS::Handle<JSObject*> aGivenProto) {
|
|
return JSProcessActorParent_Binding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
void JSProcessActorParent::Init(const nsACString& aName,
|
|
nsIDOMProcessParent* aManager) {
|
|
MOZ_ASSERT(!mManager, "Cannot Init() a JSProcessActorParent twice!");
|
|
SetName(aName);
|
|
mManager = aManager;
|
|
|
|
InvokeCallback(CallbackFunction::ActorCreated);
|
|
}
|
|
|
|
JSProcessActorParent::~JSProcessActorParent() { MOZ_ASSERT(!mManager); }
|
|
|
|
void JSProcessActorParent::SendRawMessage(
|
|
const JSActorMessageMeta& aMeta, Maybe<ipc::StructuredCloneData>&& aData,
|
|
Maybe<ipc::StructuredCloneData>&& aStack, ErrorResult& aRv) {
|
|
if (NS_WARN_IF(!CanSend() || !mManager || !mManager->GetCanSend())) {
|
|
aRv.ThrowInvalidStateError(
|
|
nsPrintfCString("Actor '%s' cannot send message '%s' during shutdown.",
|
|
PromiseFlatCString(aMeta.actorName()).get(),
|
|
NS_ConvertUTF16toUTF8(aMeta.messageName()).get()));
|
|
return;
|
|
}
|
|
|
|
size_t length = 0;
|
|
if (aData) {
|
|
length += aData->DataLength();
|
|
}
|
|
if (aStack) {
|
|
length += aStack->DataLength();
|
|
}
|
|
if (NS_WARN_IF(!AllowMessage(aMeta, length))) {
|
|
aRv.ThrowDataError(nsPrintfCString(
|
|
"Actor '%s' cannot send message '%s': message too long.",
|
|
PromiseFlatCString(aMeta.actorName()).get(),
|
|
NS_ConvertUTF16toUTF8(aMeta.messageName()).get()));
|
|
return;
|
|
}
|
|
|
|
// If the parent side is in the same process, we have a PInProcess manager,
|
|
// and can dispatch the message directly to the event loop.
|
|
ContentParent* contentParent = mManager->AsContentParent();
|
|
if (!contentParent) {
|
|
SendRawMessageInProcess(aMeta, std::move(aData), std::move(aStack), []() {
|
|
return do_AddRef(InProcessChild::Singleton());
|
|
});
|
|
return;
|
|
}
|
|
|
|
// Cross-process case - send data over ContentParent to other side.
|
|
Maybe<ClonedMessageData> msgData;
|
|
if (aData) {
|
|
msgData.emplace();
|
|
if (NS_WARN_IF(
|
|
!aData->BuildClonedMessageDataForParent(contentParent, *msgData))) {
|
|
aRv.ThrowDataCloneError(
|
|
nsPrintfCString("Actor '%s' cannot send message '%s': cannot clone.",
|
|
PromiseFlatCString(aMeta.actorName()).get(),
|
|
NS_ConvertUTF16toUTF8(aMeta.messageName()).get()));
|
|
return;
|
|
}
|
|
}
|
|
|
|
Maybe<ClonedMessageData> stackData;
|
|
if (aStack) {
|
|
stackData.emplace();
|
|
if (!aStack->BuildClonedMessageDataForParent(contentParent, *stackData)) {
|
|
stackData.reset();
|
|
}
|
|
}
|
|
|
|
if (NS_WARN_IF(!contentParent->SendRawMessage(aMeta, msgData, stackData))) {
|
|
aRv.ThrowOperationError(
|
|
nsPrintfCString("JSProcessActorParent send error in actor '%s'",
|
|
PromiseFlatCString(aMeta.actorName()).get()));
|
|
return;
|
|
}
|
|
}
|
|
|
|
void JSProcessActorParent::ClearManager() { mManager = nullptr; }
|
|
|
|
} // namespace mozilla::dom
|