diff options
Diffstat (limited to 'dom/messagechannel/SharedMessagePortMessage.cpp')
-rw-r--r-- | dom/messagechannel/SharedMessagePortMessage.cpp | 176 |
1 files changed, 176 insertions, 0 deletions
diff --git a/dom/messagechannel/SharedMessagePortMessage.cpp b/dom/messagechannel/SharedMessagePortMessage.cpp new file mode 100644 index 000000000..f7ace9dd5 --- /dev/null +++ b/dom/messagechannel/SharedMessagePortMessage.cpp @@ -0,0 +1,176 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* 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 "SharedMessagePortMessage.h" +#include "MessagePort.h" +#include "MessagePortChild.h" +#include "MessagePortParent.h" +#include "mozilla/dom/ipc/BlobChild.h" +#include "mozilla/dom/ipc/BlobParent.h" +#include "mozilla/dom/File.h" +#include "mozilla/dom/PMessagePort.h" +#include "mozilla/ipc/BackgroundChild.h" +#include "mozilla/ipc/BackgroundParent.h" + +namespace mozilla { + +using namespace ipc; + +namespace dom { + +/* static */ void +SharedMessagePortMessage::FromSharedToMessagesChild( + MessagePortChild* aActor, + const nsTArray<RefPtr<SharedMessagePortMessage>>& aData, + nsTArray<MessagePortMessage>& aArray) +{ + MOZ_ASSERT(aActor); + MOZ_ASSERT(aArray.IsEmpty()); + aArray.SetCapacity(aData.Length()); + + PBackgroundChild* backgroundManager = aActor->Manager(); + MOZ_ASSERT(backgroundManager); + + for (auto& data : aData) { + MessagePortMessage* message = aArray.AppendElement(); + data->mBuffer->abandon(); + data->mBuffer->steal(&message->data().data); + + const nsTArray<RefPtr<BlobImpl>>& blobImpls = data->BlobImpls(); + if (!blobImpls.IsEmpty()) { + message->blobsChild().SetCapacity(blobImpls.Length()); + + for (uint32_t i = 0, len = blobImpls.Length(); i < len; ++i) { + PBlobChild* blobChild = + BackgroundChild::GetOrCreateActorForBlobImpl(backgroundManager, + blobImpls[i]); + message->blobsChild().AppendElement(blobChild); + } + } + + message->transferredPorts().AppendElements(data->PortIdentifiers()); + } +} + +/* static */ bool +SharedMessagePortMessage::FromMessagesToSharedChild( + nsTArray<MessagePortMessage>& aArray, + FallibleTArray<RefPtr<SharedMessagePortMessage>>& aData) +{ + MOZ_ASSERT(aData.IsEmpty()); + + if (NS_WARN_IF(!aData.SetCapacity(aArray.Length(), mozilla::fallible))) { + return false; + } + + for (auto& message : aArray) { + RefPtr<SharedMessagePortMessage> data = new SharedMessagePortMessage(); + + data->mBuffer = MakeUnique<JSAutoStructuredCloneBuffer>( + JS::StructuredCloneScope::DifferentProcess, nullptr, nullptr); + data->mBuffer->adopt(Move(message.data().data), JS_STRUCTURED_CLONE_VERSION, + &StructuredCloneHolder::sCallbacks, data.get()); + + const nsTArray<PBlobChild*>& blobs = message.blobsChild(); + if (!blobs.IsEmpty()) { + data->BlobImpls().SetCapacity(blobs.Length()); + + for (uint32_t i = 0, len = blobs.Length(); i < len; ++i) { + RefPtr<BlobImpl> impl = + static_cast<BlobChild*>(blobs[i])->GetBlobImpl(); + data->BlobImpls().AppendElement(impl); + } + } + + data->PortIdentifiers().AppendElements(message.transferredPorts()); + + if (!aData.AppendElement(data, mozilla::fallible)) { + return false; + } + } + + return true; +} + +/* static */ bool +SharedMessagePortMessage::FromSharedToMessagesParent( + MessagePortParent* aActor, + const nsTArray<RefPtr<SharedMessagePortMessage>>& aData, + FallibleTArray<MessagePortMessage>& aArray) +{ + MOZ_ASSERT(aArray.IsEmpty()); + + if (NS_WARN_IF(!aArray.SetCapacity(aData.Length(), mozilla::fallible))) { + return false; + } + + PBackgroundParent* backgroundManager = aActor->Manager(); + MOZ_ASSERT(backgroundManager); + + for (auto& data : aData) { + MessagePortMessage* message = aArray.AppendElement(mozilla::fallible); + data->mBuffer->abandon(); + data->mBuffer->steal(&message->data().data); + + const nsTArray<RefPtr<BlobImpl>>& blobImpls = data->BlobImpls(); + if (!blobImpls.IsEmpty()) { + message->blobsParent().SetCapacity(blobImpls.Length()); + + for (uint32_t i = 0, len = blobImpls.Length(); i < len; ++i) { + PBlobParent* blobParent = + BackgroundParent::GetOrCreateActorForBlobImpl(backgroundManager, + blobImpls[i]); + message->blobsParent().AppendElement(blobParent); + } + } + + message->transferredPorts().AppendElements(data->PortIdentifiers()); + } + + return true; +} + +/* static */ bool +SharedMessagePortMessage::FromMessagesToSharedParent( + nsTArray<MessagePortMessage>& aArray, + FallibleTArray<RefPtr<SharedMessagePortMessage>>& aData) +{ + MOZ_ASSERT(aData.IsEmpty()); + + if (NS_WARN_IF(!aData.SetCapacity(aArray.Length(), mozilla::fallible))) { + return false; + } + + for (auto& message : aArray) { + RefPtr<SharedMessagePortMessage> data = new SharedMessagePortMessage(); + + data->mBuffer = MakeUnique<JSAutoStructuredCloneBuffer>( + JS::StructuredCloneScope::DifferentProcess, nullptr, nullptr); + data->mBuffer->adopt(Move(message.data().data), JS_STRUCTURED_CLONE_VERSION, + &StructuredCloneHolder::sCallbacks, data.get()); + + const nsTArray<PBlobParent*>& blobs = message.blobsParent(); + if (!blobs.IsEmpty()) { + data->BlobImpls().SetCapacity(blobs.Length()); + + for (uint32_t i = 0, len = blobs.Length(); i < len; ++i) { + RefPtr<BlobImpl> impl = + static_cast<BlobParent*>(blobs[i])->GetBlobImpl(); + data->BlobImpls().AppendElement(impl); + } + } + + data->PortIdentifiers().AppendElements(message.transferredPorts()); + + if (!aData.AppendElement(data, mozilla::fallible)) { + return false; + } + } + + return true; +} + +} // namespace dom +} // namespace mozilla |