summaryrefslogtreecommitdiffstats
path: root/dom/messagechannel/MessagePortParent.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
commit2aa4a82499d4becd2284cdb482213d541b8804dd (patch)
treeb80bf8bf13c3766139fbacc530efd0dd9d54394c /dom/messagechannel/MessagePortParent.cpp
parentInitial commit. (diff)
downloadfirefox-2aa4a82499d4becd2284cdb482213d541b8804dd.tar.xz
firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.zip
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'dom/messagechannel/MessagePortParent.cpp')
-rw-r--r--dom/messagechannel/MessagePortParent.cpp160
1 files changed, 160 insertions, 0 deletions
diff --git a/dom/messagechannel/MessagePortParent.cpp b/dom/messagechannel/MessagePortParent.cpp
new file mode 100644
index 0000000000..a2f69ea815
--- /dev/null
+++ b/dom/messagechannel/MessagePortParent.cpp
@@ -0,0 +1,160 @@
+/* -*- 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 "MessagePortParent.h"
+#include "MessagePortService.h"
+#include "mozilla/dom/RefMessageBodyService.h"
+#include "mozilla/dom/SharedMessageBody.h"
+#include "mozilla/Unused.h"
+
+namespace mozilla::dom {
+
+MessagePortParent::MessagePortParent(const nsID& aUUID)
+ : mService(MessagePortService::GetOrCreate()),
+ mUUID(aUUID),
+ mEntangled(false),
+ mCanSendData(true) {
+ MOZ_ASSERT(mService);
+}
+
+MessagePortParent::~MessagePortParent() {
+ MOZ_ASSERT(!mService);
+ MOZ_ASSERT(!mEntangled);
+}
+
+bool MessagePortParent::Entangle(const nsID& aDestinationUUID,
+ const uint32_t& aSequenceID) {
+ if (!mService) {
+ NS_WARNING("Entangle is called after a shutdown!");
+ return false;
+ }
+
+ MOZ_ASSERT(!mEntangled);
+
+ return mService->RequestEntangling(this, aDestinationUUID, aSequenceID);
+}
+
+mozilla::ipc::IPCResult MessagePortParent::RecvPostMessages(
+ nsTArray<MessageData>&& aMessages) {
+ // This converts the object in a data struct where we have BlobImpls.
+ FallibleTArray<RefPtr<SharedMessageBody>> messages;
+ if (NS_WARN_IF(!SharedMessageBody::FromMessagesToSharedParent(aMessages,
+ messages))) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mEntangled) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mService) {
+ NS_WARNING("Entangle is called after a shutdown!");
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (messages.IsEmpty()) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mService->PostMessages(this, std::move(messages))) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+ return IPC_OK();
+}
+
+mozilla::ipc::IPCResult MessagePortParent::RecvDisentangle(
+ nsTArray<MessageData>&& aMessages) {
+ // This converts the object in a data struct where we have BlobImpls.
+ FallibleTArray<RefPtr<SharedMessageBody>> messages;
+ if (NS_WARN_IF(!SharedMessageBody::FromMessagesToSharedParent(aMessages,
+ messages))) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mEntangled) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mService) {
+ NS_WARNING("Entangle is called after a shutdown!");
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ if (!mService->DisentanglePort(this, std::move(messages))) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ CloseAndDelete();
+ return IPC_OK();
+}
+
+mozilla::ipc::IPCResult MessagePortParent::RecvStopSendingData() {
+ if (!mEntangled) {
+ return IPC_OK();
+ }
+
+ mCanSendData = false;
+ Unused << SendStopSendingDataConfirmed();
+ return IPC_OK();
+}
+
+mozilla::ipc::IPCResult MessagePortParent::RecvClose() {
+ if (mService) {
+ MOZ_ASSERT(mEntangled);
+
+ if (!mService->ClosePort(this)) {
+ return IPC_FAIL_NO_REASON(this);
+ }
+
+ Close();
+ }
+
+ MOZ_ASSERT(!mEntangled);
+
+ Unused << Send__delete__(this);
+ return IPC_OK();
+}
+
+void MessagePortParent::ActorDestroy(ActorDestroyReason aWhy) {
+ if (mService && mEntangled) {
+ // When the last parent is deleted, this service is freed but this cannot
+ // be done when the hashtables are written by CloseAll.
+ RefPtr<MessagePortService> kungFuDeathGrip = mService;
+ kungFuDeathGrip->ParentDestroy(this);
+ }
+}
+
+bool MessagePortParent::Entangled(nsTArray<MessageData>&& aMessages) {
+ MOZ_ASSERT(!mEntangled);
+ mEntangled = true;
+ return SendEntangled(aMessages);
+}
+
+void MessagePortParent::CloseAndDelete() {
+ Close();
+ Unused << Send__delete__(this);
+}
+
+void MessagePortParent::Close() {
+ mService = nullptr;
+ mEntangled = false;
+}
+
+/* static */
+bool MessagePortParent::ForceClose(const nsID& aUUID,
+ const nsID& aDestinationUUID,
+ const uint32_t& aSequenceID) {
+ MessagePortService* service = MessagePortService::Get();
+ if (!service) {
+ NS_WARNING(
+ "The service must exist if we want to close an existing MessagePort.");
+ return false;
+ }
+
+ return service->ForceClose(aUUID, aDestinationUUID, aSequenceID);
+}
+
+} // namespace mozilla::dom