2018-11-30 19:52:05 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2018-06-13 17:43:48 +00:00
|
|
|
/* 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/. */
|
|
|
|
|
2019-12-10 17:07:18 +00:00
|
|
|
#include "js/ArrayBuffer.h"
|
|
|
|
#include "js/Value.h"
|
2020-02-24 22:54:54 +00:00
|
|
|
#include "mozilla/Logging.h"
|
2020-02-24 20:12:16 +00:00
|
|
|
#include "mozilla/ipc/Shmem.h"
|
|
|
|
#include "mozilla/dom/WebGPUBinding.h"
|
2018-06-13 17:43:48 +00:00
|
|
|
#include "Device.h"
|
|
|
|
|
|
|
|
#include "Adapter.h"
|
2020-02-24 20:12:16 +00:00
|
|
|
#include "Buffer.h"
|
|
|
|
#include "Sampler.h"
|
|
|
|
#include "Texture.h"
|
|
|
|
#include "TextureView.h"
|
2020-03-03 15:23:55 +00:00
|
|
|
#include "Queue.h"
|
2019-11-14 04:59:56 +00:00
|
|
|
#include "ipc/WebGPUChild.h"
|
2018-06-13 17:43:48 +00:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace webgpu {
|
|
|
|
|
2019-12-10 17:07:18 +00:00
|
|
|
mozilla::LazyLogModule gWebGPULog("WebGPU");
|
|
|
|
|
2020-01-22 07:31:51 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(Device, DOMEventTargetHelper, mBridge,
|
|
|
|
mQueue)
|
2019-10-02 16:46:03 +00:00
|
|
|
NS_IMPL_ISUPPORTS_CYCLE_COLLECTION_INHERITED_0(Device, DOMEventTargetHelper)
|
|
|
|
GPU_IMPL_JS_WRAP(Device)
|
2019-10-02 03:04:25 +00:00
|
|
|
|
2019-12-10 17:07:18 +00:00
|
|
|
static void mapFreeCallback(void* aContents, void* aUserData) {
|
|
|
|
Unused << aContents;
|
|
|
|
Unused << aUserData;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSObject* Device::CreateExternalArrayBuffer(JSContext* aCx, size_t aSize,
|
|
|
|
ipc::Shmem& aShmem) {
|
|
|
|
MOZ_ASSERT(aShmem.Size<uint8_t>() == aSize);
|
|
|
|
return JS::NewExternalArrayBuffer(aCx, aSize, aShmem.get<uint8_t>(),
|
|
|
|
&mapFreeCallback, nullptr);
|
|
|
|
}
|
|
|
|
|
2019-11-14 04:59:56 +00:00
|
|
|
Device::Device(Adapter* const aParent, RawId aId)
|
|
|
|
: DOMEventTargetHelper(aParent->GetParentObject()),
|
2020-01-24 16:27:09 +00:00
|
|
|
mBridge(aParent->mBridge),
|
2020-01-22 07:31:51 +00:00
|
|
|
mId(aId),
|
2020-01-24 16:27:09 +00:00
|
|
|
mQueue(new Queue(this, aParent->mBridge, aId)) {}
|
2019-11-14 04:59:56 +00:00
|
|
|
|
2020-01-22 07:31:51 +00:00
|
|
|
Device::~Device() { Cleanup(); }
|
|
|
|
|
|
|
|
void Device::Cleanup() {
|
|
|
|
if (mValid && mBridge && mBridge->IsOpen()) {
|
|
|
|
mValid = false;
|
2019-11-14 04:59:56 +00:00
|
|
|
mBridge->SendDeviceDestroy(mId);
|
|
|
|
}
|
|
|
|
}
|
2018-06-13 17:43:48 +00:00
|
|
|
|
2019-10-02 16:46:03 +00:00
|
|
|
void Device::GetLabel(nsAString& aValue) const { aValue = mLabel; }
|
|
|
|
void Device::SetLabel(const nsAString& aLabel) { mLabel = aLabel; }
|
2018-06-13 17:43:48 +00:00
|
|
|
|
2020-01-22 07:31:51 +00:00
|
|
|
Queue* Device::DefaultQueue() const { return mQueue; }
|
|
|
|
|
2019-12-10 17:07:18 +00:00
|
|
|
already_AddRefed<Buffer> Device::CreateBuffer(
|
|
|
|
const dom::GPUBufferDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateBuffer(mId, aDesc);
|
|
|
|
RefPtr<Buffer> buffer = new Buffer(this, id, aDesc.mSize);
|
|
|
|
return buffer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Device::CreateBufferMapped(JSContext* aCx,
|
|
|
|
const dom::GPUBufferDescriptor& aDesc,
|
|
|
|
nsTArray<JS::Value>& aSequence,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
const auto checked = CheckedInt<size_t>(aDesc.mSize);
|
|
|
|
if (!checked.isValid()) {
|
|
|
|
aRv.ThrowRangeError(u"Mapped size is too large");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const auto& size = checked.value();
|
|
|
|
|
|
|
|
// TODO: use `ShmemPool`
|
|
|
|
ipc::Shmem shmem;
|
|
|
|
if (!mBridge->AllocShmem(size, ipc::Shmem::SharedMemory::TYPE_BASIC,
|
|
|
|
&shmem)) {
|
2020-02-03 20:19:11 +00:00
|
|
|
aRv.ThrowAbortError(
|
2019-12-10 17:07:18 +00:00
|
|
|
nsPrintfCString("Unable to allocate shmem of size %" PRIuPTR, size));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// zero out memory
|
|
|
|
memset(shmem.get<uint8_t>(), 0, size);
|
|
|
|
|
|
|
|
JS::Rooted<JSObject*> arrayBuffer(
|
|
|
|
aCx, CreateExternalArrayBuffer(aCx, size, shmem));
|
|
|
|
if (!arrayBuffer) {
|
|
|
|
aRv.NoteJSContextException(aCx);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dom::GPUBufferDescriptor modifiedDesc(aDesc);
|
|
|
|
modifiedDesc.mUsage |= dom::GPUBufferUsage_Binding::MAP_WRITE;
|
|
|
|
RawId id = mBridge->DeviceCreateBuffer(mId, modifiedDesc);
|
|
|
|
RefPtr<Buffer> buffer = new Buffer(this, id, aDesc.mSize);
|
|
|
|
|
|
|
|
JS::Rooted<JS::Value> bufferValue(aCx);
|
|
|
|
if (!dom::ToJSValue(aCx, buffer, &bufferValue)) {
|
|
|
|
aRv.NoteJSContextException(aCx);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
aSequence.AppendElement(bufferValue);
|
|
|
|
aSequence.AppendElement(JS::ObjectValue(*arrayBuffer));
|
|
|
|
|
|
|
|
buffer->InitMapping(std::move(shmem), arrayBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
RefPtr<MappingPromise> Device::MapBufferForReadAsync(RawId aId, size_t aSize,
|
|
|
|
ErrorResult& aRv) {
|
|
|
|
ipc::Shmem shmem;
|
|
|
|
if (!mBridge->AllocShmem(aSize, ipc::Shmem::SharedMemory::TYPE_BASIC,
|
|
|
|
&shmem)) {
|
2020-02-03 20:19:11 +00:00
|
|
|
aRv.ThrowAbortError(
|
2019-12-10 17:07:18 +00:00
|
|
|
nsPrintfCString("Unable to allocate shmem of size %" PRIuPTR, aSize));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2020-01-22 07:31:51 +00:00
|
|
|
return mBridge->SendBufferMapRead(aId, std::move(shmem));
|
2019-12-10 17:07:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Device::UnmapBuffer(RawId aId, UniquePtr<ipc::Shmem> aShmem) {
|
|
|
|
mBridge->SendDeviceUnmapBuffer(mId, aId, std::move(*aShmem));
|
|
|
|
}
|
|
|
|
|
2020-02-24 20:12:16 +00:00
|
|
|
already_AddRefed<Texture> Device::CreateTexture(
|
|
|
|
const dom::GPUTextureDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateTexture(mId, aDesc);
|
|
|
|
RefPtr<Texture> texture = new Texture(this, id, aDesc);
|
|
|
|
return texture.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<Sampler> Device::CreateSampler(
|
|
|
|
const dom::GPUSamplerDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateSampler(mId, aDesc);
|
|
|
|
RefPtr<Sampler> sampler = new Sampler(this, id);
|
|
|
|
return sampler.forget();
|
|
|
|
}
|
|
|
|
|
2020-01-22 07:31:51 +00:00
|
|
|
already_AddRefed<CommandEncoder> Device::CreateCommandEncoder(
|
|
|
|
const dom::GPUCommandEncoderDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateCommandEncoder(mId, aDesc);
|
|
|
|
RefPtr<CommandEncoder> encoder = new CommandEncoder(this, mBridge, id);
|
|
|
|
return encoder.forget();
|
|
|
|
}
|
|
|
|
|
2020-01-24 16:27:09 +00:00
|
|
|
already_AddRefed<BindGroupLayout> Device::CreateBindGroupLayout(
|
|
|
|
const dom::GPUBindGroupLayoutDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateBindGroupLayout(mId, aDesc);
|
|
|
|
RefPtr<BindGroupLayout> object = new BindGroupLayout(this, id);
|
|
|
|
return object.forget();
|
|
|
|
}
|
|
|
|
already_AddRefed<PipelineLayout> Device::CreatePipelineLayout(
|
|
|
|
const dom::GPUPipelineLayoutDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreatePipelineLayout(mId, aDesc);
|
|
|
|
RefPtr<PipelineLayout> object = new PipelineLayout(this, id);
|
|
|
|
return object.forget();
|
|
|
|
}
|
|
|
|
already_AddRefed<BindGroup> Device::CreateBindGroup(
|
|
|
|
const dom::GPUBindGroupDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateBindGroup(mId, aDesc);
|
|
|
|
RefPtr<BindGroup> object = new BindGroup(this, id);
|
|
|
|
return object.forget();
|
2020-01-24 05:05:34 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 16:27:09 +00:00
|
|
|
already_AddRefed<ShaderModule> Device::CreateShaderModule(
|
|
|
|
const dom::GPUShaderModuleDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateShaderModule(mId, aDesc);
|
|
|
|
RefPtr<ShaderModule> object = new ShaderModule(this, id);
|
|
|
|
return object.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ComputePipeline> Device::CreateComputePipeline(
|
|
|
|
const dom::GPUComputePipelineDescriptor& aDesc) {
|
|
|
|
RawId id = mBridge->DeviceCreateComputePipeline(mId, aDesc);
|
|
|
|
RefPtr<ComputePipeline> object = new ComputePipeline(this, id);
|
|
|
|
return object.forget();
|
2019-12-10 17:07:18 +00:00
|
|
|
}
|
|
|
|
|
2018-06-13 17:43:48 +00:00
|
|
|
} // namespace webgpu
|
|
|
|
} // namespace mozilla
|