diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 14:29:10 +0000 |
commit | 2aa4a82499d4becd2284cdb482213d541b8804dd (patch) | |
tree | b80bf8bf13c3766139fbacc530efd0dd9d54394c /dom/webgpu/Queue.cpp | |
parent | Initial commit. (diff) | |
download | firefox-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 '')
-rw-r--r-- | dom/webgpu/Queue.cpp | 126 |
1 files changed, 126 insertions, 0 deletions
diff --git a/dom/webgpu/Queue.cpp b/dom/webgpu/Queue.cpp new file mode 100644 index 0000000000..052b40b49a --- /dev/null +++ b/dom/webgpu/Queue.cpp @@ -0,0 +1,126 @@ +/* -*- Mode: C++; tab-width: 4; 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 "mozilla/dom/WebGPUBinding.h" +#include "Queue.h" + +#include "CommandBuffer.h" +#include "CommandEncoder.h" +#include "ipc/WebGPUChild.h" +#include "mozilla/ErrorResult.h" + +namespace mozilla { +namespace webgpu { + +GPU_IMPL_CYCLE_COLLECTION(Queue, mParent, mBridge) +GPU_IMPL_JS_WRAP(Queue) + +Queue::Queue(Device* const aParent, WebGPUChild* aBridge, RawId aId) + : ChildOf(aParent), mBridge(aBridge), mId(aId) {} + +Queue::~Queue() { Cleanup(); } + +void Queue::Submit( + const dom::Sequence<OwningNonNull<CommandBuffer>>& aCommandBuffers) { + nsTArray<RawId> list(aCommandBuffers.Length()); + for (uint32_t i = 0; i < aCommandBuffers.Length(); ++i) { + auto idMaybe = aCommandBuffers[i]->Commit(); + if (idMaybe) { + list.AppendElement(*idMaybe); + } + } + + mBridge->SendQueueSubmit(mId, list); +} + +void Queue::WriteBuffer(const Buffer& aBuffer, uint64_t aBufferOffset, + const dom::ArrayBuffer& aData, uint64_t aDataOffset, + const dom::Optional<uint64_t>& aSize, + ErrorResult& aRv) { + aData.ComputeState(); + const auto checkedSize = + aSize.WasPassed() ? CheckedInt<size_t>(aSize.Value()) + : CheckedInt<size_t>(aData.Length()) - aDataOffset; + if (!checkedSize.isValid()) { + aRv.ThrowRangeError("Mapped size is too large"); + return; + } + + const auto& size = checkedSize.value(); + if (aDataOffset + size > aData.Length()) { + aRv.ThrowAbortError(nsPrintfCString("Wrong data size %" PRIuPTR, size)); + return; + } + + ipc::Shmem shmem; + if (!mBridge->AllocShmem(size, ipc::Shmem::SharedMemory::TYPE_BASIC, + &shmem)) { + aRv.ThrowAbortError( + nsPrintfCString("Unable to allocate shmem of size %" PRIuPTR, size)); + return; + } + + memcpy(shmem.get<uint8_t>(), aData.Data() + aDataOffset, size); + mBridge->SendQueueWriteBuffer(mId, aBuffer.mId, aBufferOffset, + std::move(shmem)); +} + +void Queue::WriteTexture(const dom::GPUTextureCopyView& aDestination, + const dom::ArrayBuffer& aData, + const dom::GPUTextureDataLayout& aDataLayout, + const dom::GPUExtent3D& aSize, ErrorResult& aRv) { + ffi::WGPUTextureCopyView copyView = {}; + CommandEncoder::ConvertTextureCopyViewToFFI(aDestination, ©View); + ffi::WGPUTextureDataLayout dataLayout = {}; + CommandEncoder::ConvertTextureDataLayoutToFFI(aDataLayout, &dataLayout); + dataLayout.offset = 0; // our Shmem has the contents starting from 0. + ffi::WGPUExtent3d extent = {}; + CommandEncoder::ConvertExtent3DToFFI(aSize, &extent); + + const auto bpb = aDestination.mTexture->mBytesPerBlock; + if (!bpb) { + aRv.ThrowAbortError(nsPrintfCString("Invalid texture format")); + return; + } + if (extent.width == 0 || extent.height == 0 || extent.depth == 0) { + aRv.ThrowAbortError(nsPrintfCString("Invalid copy size")); + return; + } + + // TODO: support block-compressed formats + aData.ComputeState(); + const auto fullRows = + (CheckedInt<size_t>(extent.depth - 1) * aDataLayout.mRowsPerImage + + extent.height - 1); + const auto checkedSize = fullRows * aDataLayout.mBytesPerRow + + CheckedInt<size_t>(extent.width) * bpb.value(); + if (!checkedSize.isValid()) { + aRv.ThrowRangeError("Mapped size is too large"); + return; + } + + const auto& size = checkedSize.value(); + auto availableSize = aData.Length(); + if (availableSize < aDataLayout.mOffset || + size > (availableSize - aDataLayout.mOffset)) { + aRv.ThrowAbortError(nsPrintfCString("Wrong data size %" PRIuPTR, size)); + return; + } + + ipc::Shmem shmem; + if (!mBridge->AllocShmem(size, ipc::Shmem::SharedMemory::TYPE_BASIC, + &shmem)) { + aRv.ThrowAbortError( + nsPrintfCString("Unable to allocate shmem of size %" PRIuPTR, size)); + return; + } + + memcpy(shmem.get<uint8_t>(), aData.Data() + aDataLayout.mOffset, size); + mBridge->SendQueueWriteTexture(mId, copyView, std::move(shmem), dataLayout, + extent); +} + +} // namespace webgpu +} // namespace mozilla |