forked from mirrors/gecko-dev
		
	Bug 1750576 - WebGPU shader module asynchronous CompilationInfo API. r=aosmond,emilio
This patch is a lot of plumbing for not that much functionality. The goal is to align CreateShaderModule's error reporting with the spec. Creating a shader module is now a dedicated async IPDL message returning the compilation info so that it can be exposed as a promise by the WebGPU API. Differential Revision: https://phabricator.services.mozilla.com/D146817
This commit is contained in:
		
							parent
							
								
									21b693f361
								
							
						
					
					
						commit
						1d6db74e44
					
				
					 20 changed files with 255 additions and 57 deletions
				
			
		
							
								
								
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							|  | @ -6180,6 +6180,7 @@ version = "0.1.0" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "bincode", |  "bincode", | ||||||
|  "log", |  "log", | ||||||
|  |  "nsstring", | ||||||
|  "parking_lot 0.11.2", |  "parking_lot 0.11.2", | ||||||
|  "serde", |  "serde", | ||||||
|  "wgpu-core", |  "wgpu-core", | ||||||
|  |  | ||||||
|  | @ -16,4 +16,19 @@ GPU_IMPL_JS_WRAP(CompilationInfo) | ||||||
| CompilationInfo::CompilationInfo(ShaderModule* const aParent) | CompilationInfo::CompilationInfo(ShaderModule* const aParent) | ||||||
|     : ChildOf(aParent) {} |     : ChildOf(aParent) {} | ||||||
| 
 | 
 | ||||||
|  | void CompilationInfo::SetMessages( | ||||||
|  |     nsTArray<mozilla::webgpu::WebGPUCompilationMessage>& aMessages) { | ||||||
|  |   for (auto& msg : aMessages) { | ||||||
|  |     mMessages.AppendElement(MakeAndAddRef<mozilla::webgpu::CompilationMessage>( | ||||||
|  |         this, msg.lineNum, msg.linePos, msg.offset, std::move(msg.message))); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CompilationInfo::GetMessages( | ||||||
|  |     nsTArray<RefPtr<mozilla::webgpu::CompilationMessage>>& aMessages) { | ||||||
|  |   for (auto& msg : mMessages) { | ||||||
|  |     aMessages.AppendElement(msg); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
|  |  | ||||||
|  | @ -8,6 +8,7 @@ | ||||||
| 
 | 
 | ||||||
| #include "nsWrapperCache.h" | #include "nsWrapperCache.h" | ||||||
| #include "ObjectModel.h" | #include "ObjectModel.h" | ||||||
|  | #include "CompilationMessage.h" | ||||||
| 
 | 
 | ||||||
| namespace mozilla::webgpu { | namespace mozilla::webgpu { | ||||||
| class ShaderModule; | class ShaderModule; | ||||||
|  | @ -18,10 +19,19 @@ class CompilationInfo final : public nsWrapperCache, | ||||||
|   GPU_DECL_CYCLE_COLLECTION(CompilationInfo) |   GPU_DECL_CYCLE_COLLECTION(CompilationInfo) | ||||||
|   GPU_DECL_JS_WRAP(CompilationInfo) |   GPU_DECL_JS_WRAP(CompilationInfo) | ||||||
| 
 | 
 | ||||||
|  private: |  | ||||||
|   explicit CompilationInfo(ShaderModule* const aParent); |   explicit CompilationInfo(ShaderModule* const aParent); | ||||||
|  | 
 | ||||||
|  |   void SetMessages( | ||||||
|  |       nsTArray<mozilla::webgpu::WebGPUCompilationMessage>& aMessages); | ||||||
|  | 
 | ||||||
|  |   void GetMessages( | ||||||
|  |       nsTArray<RefPtr<mozilla::webgpu::CompilationMessage>>& aMessages); | ||||||
|  | 
 | ||||||
|  |  private: | ||||||
|   ~CompilationInfo() = default; |   ~CompilationInfo() = default; | ||||||
|   void Cleanup() {} |   void Cleanup() {} | ||||||
|  | 
 | ||||||
|  |   nsTArray<RefPtr<mozilla::webgpu::CompilationMessage>> mMessages; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
|  |  | ||||||
|  | @ -12,7 +12,13 @@ namespace mozilla::webgpu { | ||||||
| GPU_IMPL_CYCLE_COLLECTION(CompilationMessage, mParent) | GPU_IMPL_CYCLE_COLLECTION(CompilationMessage, mParent) | ||||||
| GPU_IMPL_JS_WRAP(CompilationMessage) | GPU_IMPL_JS_WRAP(CompilationMessage) | ||||||
| 
 | 
 | ||||||
| CompilationMessage::CompilationMessage(CompilationInfo* const aParent) | CompilationMessage::CompilationMessage(CompilationInfo* const aParent, | ||||||
|     : ChildOf(aParent) {} |                                        uint64_t aLineNum, uint64_t aLinePos, | ||||||
|  |                                        uint64_t aOffset, nsString&& aMessage) | ||||||
|  |     : ChildOf(aParent), | ||||||
|  |       mLineNum(aLineNum), | ||||||
|  |       mLinePos(aLinePos), | ||||||
|  |       mOffset(aOffset), | ||||||
|  |       mMessage(std::move(aMessage)) {} | ||||||
| 
 | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
|  |  | ||||||
|  | @ -24,12 +24,19 @@ class CompilationMessage final : public nsWrapperCache, | ||||||
|   uint64_t mLinePos = 0; |   uint64_t mLinePos = 0; | ||||||
|   uint64_t mOffset = 0; |   uint64_t mOffset = 0; | ||||||
|   uint64_t mLength = 0; |   uint64_t mLength = 0; | ||||||
|  |   nsString mMessage; | ||||||
| 
 | 
 | ||||||
|  public: |  public: | ||||||
|   GPU_DECL_CYCLE_COLLECTION(CompilationMessage) |   GPU_DECL_CYCLE_COLLECTION(CompilationMessage) | ||||||
|   GPU_DECL_JS_WRAP(CompilationMessage) |   GPU_DECL_JS_WRAP(CompilationMessage) | ||||||
| 
 | 
 | ||||||
|   void GetMessage(dom::DOMString& aMessage) {} |   explicit CompilationMessage(CompilationInfo* const aParent, uint64_t aLineNum, | ||||||
|  |                               uint64_t aLinePos, uint64_t aOffset, | ||||||
|  |                               nsString&& aMessage); | ||||||
|  | 
 | ||||||
|  |   void GetMessage(dom::DOMString& aMessage) { | ||||||
|  |     aMessage.AsAString().Assign(mMessage); | ||||||
|  |   } | ||||||
|   dom::GPUCompilationMessageType Type() const { return mType; } |   dom::GPUCompilationMessageType Type() const { return mType; } | ||||||
|   uint64_t LineNum() const { return mLineNum; } |   uint64_t LineNum() const { return mLineNum; } | ||||||
|   uint64_t LinePos() const { return mLinePos; } |   uint64_t LinePos() const { return mLinePos; } | ||||||
|  | @ -37,7 +44,6 @@ class CompilationMessage final : public nsWrapperCache, | ||||||
|   uint64_t Length() const { return mLength; } |   uint64_t Length() const { return mLength; } | ||||||
| 
 | 
 | ||||||
|  private: |  private: | ||||||
|   explicit CompilationMessage(CompilationInfo* const aParent); |  | ||||||
|   ~CompilationMessage() = default; |   ~CompilationMessage() = default; | ||||||
|   void Cleanup() {} |   void Cleanup() {} | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -266,12 +266,18 @@ already_AddRefed<BindGroup> Device::CreateBindGroup( | ||||||
| already_AddRefed<ShaderModule> Device::CreateShaderModule( | already_AddRefed<ShaderModule> Device::CreateShaderModule( | ||||||
|     JSContext* aCx, const dom::GPUShaderModuleDescriptor& aDesc) { |     JSContext* aCx, const dom::GPUShaderModuleDescriptor& aDesc) { | ||||||
|   Unused << aCx; |   Unused << aCx; | ||||||
|   RawId id = 0; | 
 | ||||||
|   if (mBridge->CanSend()) { |   if (!mBridge->CanSend()) { | ||||||
|     id = mBridge->DeviceCreateShaderModule(mId, aDesc); |     return nullptr; | ||||||
|   } |   } | ||||||
|   RefPtr<ShaderModule> object = new ShaderModule(this, id); | 
 | ||||||
|   return object.forget(); |   ErrorResult err; | ||||||
|  |   RefPtr<dom::Promise> promise = dom::Promise::Create(GetParentObject(), err); | ||||||
|  |   if (NS_WARN_IF(err.Failed())) { | ||||||
|  |     return nullptr; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return mBridge->DeviceCreateShaderModule(this, aDesc, promise); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| already_AddRefed<ComputePipeline> Device::CreateComputePipeline( | already_AddRefed<ComputePipeline> Device::CreateComputePipeline( | ||||||
|  |  | ||||||
|  | @ -4,7 +4,9 @@ | ||||||
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 |  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | ||||||
| 
 | 
 | ||||||
| #include "mozilla/dom/WebGPUBinding.h" | #include "mozilla/dom/WebGPUBinding.h" | ||||||
|  | #include "mozilla/dom/Promise.h" | ||||||
| #include "ShaderModule.h" | #include "ShaderModule.h" | ||||||
|  | #include "CompilationInfo.h" | ||||||
| #include "ipc/WebGPUChild.h" | #include "ipc/WebGPUChild.h" | ||||||
| 
 | 
 | ||||||
| #include "Device.h" | #include "Device.h" | ||||||
|  | @ -14,8 +16,9 @@ namespace mozilla::webgpu { | ||||||
| GPU_IMPL_CYCLE_COLLECTION(ShaderModule, mParent) | GPU_IMPL_CYCLE_COLLECTION(ShaderModule, mParent) | ||||||
| GPU_IMPL_JS_WRAP(ShaderModule) | GPU_IMPL_JS_WRAP(ShaderModule) | ||||||
| 
 | 
 | ||||||
| ShaderModule::ShaderModule(Device* const aParent, RawId aId) | ShaderModule::ShaderModule(Device* const aParent, RawId aId, | ||||||
|     : ChildOf(aParent), mId(aId) {} |                            const RefPtr<dom::Promise>& aCompilationInfo) | ||||||
|  |     : ChildOf(aParent), mId(aId), mCompilationInfo(aCompilationInfo) {} | ||||||
| 
 | 
 | ||||||
| ShaderModule::~ShaderModule() { Cleanup(); } | ShaderModule::~ShaderModule() { Cleanup(); } | ||||||
| 
 | 
 | ||||||
|  | @ -29,4 +32,9 @@ void ShaderModule::Cleanup() { | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | already_AddRefed<dom::Promise> ShaderModule::CompilationInfo(ErrorResult& aRv) { | ||||||
|  |   RefPtr<dom::Promise> tmp = mCompilationInfo; | ||||||
|  |   return tmp.forget(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
|  |  | ||||||
|  | @ -12,20 +12,27 @@ | ||||||
| 
 | 
 | ||||||
| namespace mozilla::webgpu { | namespace mozilla::webgpu { | ||||||
| 
 | 
 | ||||||
|  | class CompilationInfo; | ||||||
| class Device; | class Device; | ||||||
| 
 | 
 | ||||||
| class ShaderModule final : public ObjectBase, public ChildOf<Device> { | class ShaderModule final : public ObjectBase, public ChildOf<Device> { | ||||||
|  public: |  public: | ||||||
|   GPU_DECL_CYCLE_COLLECTION(ShaderModule) |   GPU_DECL_CYCLE_COLLECTION( | ||||||
|  |       ShaderModule)  // TODO: kvark's WIP patch was passing CompilationInfo as a
 | ||||||
|  |                      // second argument here.
 | ||||||
|   GPU_DECL_JS_WRAP(ShaderModule) |   GPU_DECL_JS_WRAP(ShaderModule) | ||||||
| 
 | 
 | ||||||
|   ShaderModule(Device* const aParent, RawId aId); |   ShaderModule(Device* const aParent, RawId aId, | ||||||
|  |                const RefPtr<dom::Promise>& aCompilationInfo); | ||||||
|  |   already_AddRefed<dom::Promise> CompilationInfo(ErrorResult& aRv); | ||||||
| 
 | 
 | ||||||
|   const RawId mId; |   const RawId mId; | ||||||
| 
 | 
 | ||||||
|  private: |  private: | ||||||
|   virtual ~ShaderModule(); |   virtual ~ShaderModule(); | ||||||
|   void Cleanup(); |   void Cleanup(); | ||||||
|  | 
 | ||||||
|  |   RefPtr<dom::Promise> mCompilationInfo; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
|  |  | ||||||
|  | @ -13,6 +13,7 @@ using dom::GPUCommandBufferDescriptor from "mozilla/dom/WebGPUBinding.h"; | ||||||
| using dom::GPUBufferDescriptor from "mozilla/dom/WebGPUBinding.h"; | using dom::GPUBufferDescriptor from "mozilla/dom/WebGPUBinding.h"; | ||||||
| using webgpu::ffi::WGPUHostMap from "mozilla/webgpu/ffi/wgpu.h"; | using webgpu::ffi::WGPUHostMap from "mozilla/webgpu/ffi/wgpu.h"; | ||||||
| using MaybeScopedError from "mozilla/webgpu/WebGPUTypes.h"; | using MaybeScopedError from "mozilla/webgpu/WebGPUTypes.h"; | ||||||
|  | using WebGPUCompilationMessage from "mozilla/webgpu/WebGPUTypes.h"; | ||||||
| 
 | 
 | ||||||
| include "mozilla/ipc/ByteBufUtils.h"; | include "mozilla/ipc/ByteBufUtils.h"; | ||||||
| include "mozilla/layers/LayersMessageUtils.h"; | include "mozilla/layers/LayersMessageUtils.h"; | ||||||
|  | @ -44,6 +45,8 @@ parent: | ||||||
|   async InstanceRequestAdapter(GPURequestAdapterOptions options, RawId[] ids) returns (ByteBuf byteBuf); |   async InstanceRequestAdapter(GPURequestAdapterOptions options, RawId[] ids) returns (ByteBuf byteBuf); | ||||||
|   async AdapterRequestDevice(RawId selfId, ByteBuf buf, RawId newId) returns (bool success); |   async AdapterRequestDevice(RawId selfId, ByteBuf buf, RawId newId) returns (bool success); | ||||||
|   async AdapterDestroy(RawId selfId); |   async AdapterDestroy(RawId selfId); | ||||||
|  |   // TODO: We want to return an array of compilation messages. | ||||||
|  |   async DeviceCreateShaderModule(RawId selfId, RawId bufferId, nsString label, nsCString code) returns (WebGPUCompilationMessage[] messages); | ||||||
|   async BufferReturnShmem(RawId selfId, Shmem shmem); |   async BufferReturnShmem(RawId selfId, Shmem shmem); | ||||||
|   async BufferMap(RawId selfId, WGPUHostMap hostMap, uint64_t offset, uint64_t size) returns (Shmem sm); |   async BufferMap(RawId selfId, WGPUHostMap hostMap, uint64_t offset, uint64_t size) returns (Shmem sm); | ||||||
|   async BufferUnmap(RawId selfId, Shmem shmem, bool flush, bool keepShmem); |   async BufferUnmap(RawId selfId, Shmem shmem, bool flush, bool keepShmem); | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ | ||||||
| #include "Adapter.h" | #include "Adapter.h" | ||||||
| #include "DeviceLostInfo.h" | #include "DeviceLostInfo.h" | ||||||
| #include "Sampler.h" | #include "Sampler.h" | ||||||
|  | #include "CompilationInfo.h" | ||||||
| 
 | 
 | ||||||
| namespace mozilla::webgpu { | namespace mozilla::webgpu { | ||||||
| 
 | 
 | ||||||
|  | @ -705,20 +706,34 @@ RawId WebGPUChild::DeviceCreateBindGroup( | ||||||
|   return id; |   return id; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| RawId WebGPUChild::DeviceCreateShaderModule( | already_AddRefed<ShaderModule> WebGPUChild::DeviceCreateShaderModule( | ||||||
|     RawId aSelfId, const dom::GPUShaderModuleDescriptor& aDesc) { |     Device* aDevice, const dom::GPUShaderModuleDescriptor& aDesc, | ||||||
|   ffi::WGPUShaderModuleDescriptor desc = {}; |     RefPtr<dom::Promise> aPromise) { | ||||||
|  |   RawId deviceId = aDevice->mId; | ||||||
|  |   RawId moduleId = | ||||||
|  |       ffi::wgpu_client_make_shader_module_id(mClient.get(), deviceId); | ||||||
| 
 | 
 | ||||||
|   desc.code = reinterpret_cast<const uint8_t*>(aDesc.mCode.get()); |   RefPtr<ShaderModule> shaderModule = | ||||||
|   desc.code_length = aDesc.mCode.Length(); |       new ShaderModule(aDevice, moduleId, aPromise); | ||||||
| 
 | 
 | ||||||
|   ByteBuf bb; |   nsString noLabel; | ||||||
|   RawId id = ffi::wgpu_client_create_shader_module(mClient.get(), aSelfId, |   const nsString& label = | ||||||
|                                                    &desc, ToFFI(&bb)); |       aDesc.mLabel.WasPassed() ? aDesc.mLabel.Value() : noLabel; | ||||||
|   if (!SendDeviceAction(aSelfId, std::move(bb))) { |   SendDeviceCreateShaderModule(deviceId, moduleId, label, aDesc.mCode) | ||||||
|     MOZ_CRASH("IPC failure"); |       ->Then( | ||||||
|   } |           GetCurrentSerialEventTarget(), __func__, | ||||||
|   return id; |           [aPromise, | ||||||
|  |            shaderModule](nsTArray<WebGPUCompilationMessage>&& messages) { | ||||||
|  |             RefPtr<CompilationInfo> infoObject( | ||||||
|  |                 new CompilationInfo(shaderModule)); | ||||||
|  |             infoObject->SetMessages(messages); | ||||||
|  |             aPromise->MaybeResolve(infoObject); | ||||||
|  |           }, | ||||||
|  |           [aPromise](const ipc::ResponseRejectReason& aReason) { | ||||||
|  |             aPromise->MaybeRejectWithNotSupportedError("IPC error"); | ||||||
|  |           }); | ||||||
|  | 
 | ||||||
|  |   return shaderModule.forget(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| RawId WebGPUChild::DeviceCreateComputePipelineImpl( | RawId WebGPUChild::DeviceCreateComputePipelineImpl( | ||||||
|  |  | ||||||
|  | @ -83,9 +83,6 @@ class WebGPUChild final : public PWebGPUChild, public SupportsWeakPtr { | ||||||
|       RawId aSelfId, const dom::GPUPipelineLayoutDescriptor& aDesc); |       RawId aSelfId, const dom::GPUPipelineLayoutDescriptor& aDesc); | ||||||
|   RawId DeviceCreateBindGroup(RawId aSelfId, |   RawId DeviceCreateBindGroup(RawId aSelfId, | ||||||
|                               const dom::GPUBindGroupDescriptor& aDesc); |                               const dom::GPUBindGroupDescriptor& aDesc); | ||||||
|   RawId DeviceCreateShaderModule(RawId aSelfId, |  | ||||||
|                                  const dom::GPUShaderModuleDescriptor& aDesc); |  | ||||||
| 
 |  | ||||||
|   RawId DeviceCreateComputePipeline( |   RawId DeviceCreateComputePipeline( | ||||||
|       PipelineCreationContext* const aContext, |       PipelineCreationContext* const aContext, | ||||||
|       const dom::GPUComputePipelineDescriptor& aDesc); |       const dom::GPUComputePipelineDescriptor& aDesc); | ||||||
|  | @ -98,6 +95,9 @@ class WebGPUChild final : public PWebGPUChild, public SupportsWeakPtr { | ||||||
|   RefPtr<PipelinePromise> DeviceCreateRenderPipelineAsync( |   RefPtr<PipelinePromise> DeviceCreateRenderPipelineAsync( | ||||||
|       PipelineCreationContext* const aContext, |       PipelineCreationContext* const aContext, | ||||||
|       const dom::GPURenderPipelineDescriptor& aDesc); |       const dom::GPURenderPipelineDescriptor& aDesc); | ||||||
|  |   already_AddRefed<ShaderModule> DeviceCreateShaderModule( | ||||||
|  |       Device* aDevice, const dom::GPUShaderModuleDescriptor& aDesc, | ||||||
|  |       RefPtr<dom::Promise> aPromise); | ||||||
| 
 | 
 | ||||||
|   void DeviceCreateSwapChain(RawId aSelfId, const RGBDescriptor& aRgbDesc, |   void DeviceCreateSwapChain(RawId aSelfId, const RGBDescriptor& aRgbDesc, | ||||||
|                              size_t maxBufferCount, |                              size_t maxBufferCount, | ||||||
|  |  | ||||||
|  | @ -633,6 +633,37 @@ ipc::IPCResult WebGPUParent::RecvDeviceCreateSwapChain( | ||||||
|   return IPC_OK(); |   return IPC_OK(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | ipc::IPCResult WebGPUParent::RecvDeviceCreateShaderModule( | ||||||
|  |     RawId aSelfId, RawId aBufferId, const nsString& aLabel, | ||||||
|  |     const nsCString& aCode, DeviceCreateShaderModuleResolver&& aOutMessage) { | ||||||
|  |   NS_ConvertUTF16toUTF8 label(aLabel); | ||||||
|  | 
 | ||||||
|  |   ffi::WGPUShaderModuleCompilationMessage message; | ||||||
|  | 
 | ||||||
|  |   bool ok = ffi::wgpu_server_device_create_shader_module( | ||||||
|  |       mContext.get(), aSelfId, aBufferId, label.get(), | ||||||
|  |       reinterpret_cast<const uint8_t*>(aCode.get()), aCode.Length(), &message); | ||||||
|  | 
 | ||||||
|  |   nsTArray<WebGPUCompilationMessage> messages; | ||||||
|  | 
 | ||||||
|  |   if (!ok) { | ||||||
|  |     WebGPUCompilationMessage msg; | ||||||
|  |     msg.lineNum = message.line_number; | ||||||
|  |     msg.linePos = message.line_pos; | ||||||
|  |     msg.offset = message.utf16_offset; | ||||||
|  |     msg.length = message.utf16_length; | ||||||
|  |     msg.message = message.message; | ||||||
|  |     // wgpu currently only returns errors.
 | ||||||
|  |     msg.messageType = WebGPUCompilationMessageType::Error; | ||||||
|  | 
 | ||||||
|  |     messages.AppendElement(msg); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   aOutMessage(messages); | ||||||
|  | 
 | ||||||
|  |   return IPC_OK(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| struct PresentRequest { | struct PresentRequest { | ||||||
|   const ffi::WGPUGlobal* mContext; |   const ffi::WGPUGlobal* mContext; | ||||||
|   RefPtr<PresentationData> mData; |   RefPtr<PresentationData> mData; | ||||||
|  |  | ||||||
|  | @ -69,6 +69,10 @@ class WebGPUParent final : public PWebGPUParent { | ||||||
|                                            const layers::RGBDescriptor& aDesc, |                                            const layers::RGBDescriptor& aDesc, | ||||||
|                                            const nsTArray<RawId>& aBufferIds, |                                            const nsTArray<RawId>& aBufferIds, | ||||||
|                                            const CompositableHandle& aHandle); |                                            const CompositableHandle& aHandle); | ||||||
|  |   ipc::IPCResult RecvDeviceCreateShaderModule( | ||||||
|  |       RawId aSelfId, RawId aModuleId, const nsString& aLabel, | ||||||
|  |       const nsCString& aCode, DeviceCreateShaderModuleResolver&& aOutMessage); | ||||||
|  | 
 | ||||||
|   ipc::IPCResult RecvSwapChainPresent(const CompositableHandle& aHandle, |   ipc::IPCResult RecvSwapChainPresent(const CompositableHandle& aHandle, | ||||||
|                                       RawId aTextureId, |                                       RawId aTextureId, | ||||||
|                                       RawId aCommandEncoderId); |                                       RawId aCommandEncoderId); | ||||||
|  |  | ||||||
|  | @ -39,6 +39,9 @@ DEFINE_IPC_SERIALIZER_WITH_FIELDS(mozilla::dom::GPUBufferDescriptor, mSize, | ||||||
| DEFINE_IPC_SERIALIZER_WITH_FIELDS(mozilla::webgpu::ScopedError, operationError, | DEFINE_IPC_SERIALIZER_WITH_FIELDS(mozilla::webgpu::ScopedError, operationError, | ||||||
|                                   validationMessage); |                                   validationMessage); | ||||||
| 
 | 
 | ||||||
|  | DEFINE_IPC_SERIALIZER_WITH_FIELDS(mozilla::webgpu::WebGPUCompilationMessage, | ||||||
|  |                                   message, lineNum, linePos); | ||||||
|  | 
 | ||||||
| #undef DEFINE_IPC_SERIALIZER_FFI_ENUM | #undef DEFINE_IPC_SERIALIZER_FFI_ENUM | ||||||
| #undef DEFINE_IPC_SERIALIZER_DOM_ENUM | #undef DEFINE_IPC_SERIALIZER_DOM_ENUM | ||||||
| #undef DEFINE_IPC_SERIALIZER_ENUM_GUARD | #undef DEFINE_IPC_SERIALIZER_ENUM_GUARD | ||||||
|  |  | ||||||
|  | @ -26,6 +26,22 @@ struct ScopedError { | ||||||
| }; | }; | ||||||
| using MaybeScopedError = Maybe<ScopedError>; | using MaybeScopedError = Maybe<ScopedError>; | ||||||
| 
 | 
 | ||||||
|  | enum class WebGPUCompilationMessageType { Error, Warning, Info }; | ||||||
|  | 
 | ||||||
|  | // TODO: Better name? CompilationMessage alread taken by the dom object.
 | ||||||
|  | /// The serializable counterpart of the dom object CompilationMessage.
 | ||||||
|  | struct WebGPUCompilationMessage { | ||||||
|  |   nsString message; | ||||||
|  |   uint64_t lineNum = 0; | ||||||
|  |   uint64_t linePos = 0; | ||||||
|  |   // In utf16 code units.
 | ||||||
|  |   uint64_t offset = 0; | ||||||
|  |   // In utf16 code units.
 | ||||||
|  |   uint64_t length = 0; | ||||||
|  |   WebGPUCompilationMessageType messageType = | ||||||
|  |       WebGPUCompilationMessageType::Error; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| }  // namespace mozilla::webgpu
 | }  // namespace mozilla::webgpu
 | ||||||
| 
 | 
 | ||||||
| #endif  // WEBGPU_TYPES_H_
 | #endif  // WEBGPU_TYPES_H_
 | ||||||
|  |  | ||||||
|  | @ -627,9 +627,8 @@ interface GPUCompilationMessage { | ||||||
| [Pref="dom.webgpu.enabled", | [Pref="dom.webgpu.enabled", | ||||||
|  Exposed=(Window,DedicatedWorker)] |  Exposed=(Window,DedicatedWorker)] | ||||||
| interface GPUCompilationInfo { | interface GPUCompilationInfo { | ||||||
|     //TODO: |     [Cached, Frozen, Pure] | ||||||
|     //[Cached, Frozen, Pure] |     readonly attribute sequence<GPUCompilationMessage> messages; | ||||||
|     //readonly attribute sequence<GPUCompilationMessage> messages; |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // ShaderModule | // ShaderModule | ||||||
|  | @ -643,8 +642,8 @@ dictionary GPUShaderModuleDescriptor : GPUObjectDescriptorBase { | ||||||
| [Pref="dom.webgpu.enabled", | [Pref="dom.webgpu.enabled", | ||||||
|  Exposed=(Window,DedicatedWorker)] |  Exposed=(Window,DedicatedWorker)] | ||||||
| interface GPUShaderModule { | interface GPUShaderModule { | ||||||
|     //TODO: |     [Throws] | ||||||
|     //Promise<GPUCompilationInfo> compilationInfo(); |     Promise<GPUCompilationInfo> compilationInfo(); | ||||||
| }; | }; | ||||||
| GPUShaderModule includes GPUObjectBase; | GPUShaderModule includes GPUObjectBase; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -37,3 +37,4 @@ bincode = "1" | ||||||
| log = "0.4" | log = "0.4" | ||||||
| parking_lot = "0.11" | parking_lot = "0.11" | ||||||
| serde = "1" | serde = "1" | ||||||
|  | nsstring = { path = "../../xpcom/rust/nsstring" } | ||||||
|  |  | ||||||
|  | @ -15,6 +15,8 @@ autogen_warning = """/* DO NOT MODIFY THIS MANUALLY! This file was generated usi | ||||||
|  * - $CBINDGEN is the path to the cbindgen executable provided by mozbuild (the exact version often matters) |  * - $CBINDGEN is the path to the cbindgen executable provided by mozbuild (the exact version often matters) | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
|  | #include "nsString.h" | ||||||
|  | 
 | ||||||
| struct WGPUByteBuf; | struct WGPUByteBuf; | ||||||
| typedef uint64_t WGPUNonZeroU64; | typedef uint64_t WGPUNonZeroU64; | ||||||
| typedef uint64_t WGPUOption_BufferSize; | typedef uint64_t WGPUOption_BufferSize; | ||||||
|  | @ -27,6 +29,7 @@ typedef uint64_t WGPUOption_BindGroupLayoutId; | ||||||
| typedef uint64_t WGPUOption_SamplerId; | typedef uint64_t WGPUOption_SamplerId; | ||||||
| typedef uint64_t WGPUOption_SurfaceId; | typedef uint64_t WGPUOption_SurfaceId; | ||||||
| typedef uint64_t WGPUOption_TextureViewId; | typedef uint64_t WGPUOption_TextureViewId; | ||||||
|  | typedef nsString WGPUnsString; | ||||||
| """ | """ | ||||||
| include_version = true | include_version = true | ||||||
| braces = "SameLine" | braces = "SameLine" | ||||||
|  |  | ||||||
|  | @ -35,13 +35,6 @@ fn make_byte_buf<T: serde::Serialize>(data: &T) -> ByteBuf { | ||||||
|     ByteBuf::from_vec(vec) |     ByteBuf::from_vec(vec) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[repr(C)] |  | ||||||
| pub struct ShaderModuleDescriptor { |  | ||||||
|     label: RawString, |  | ||||||
|     code: *const u8, |  | ||||||
|     code_length: usize, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[repr(C)] | #[repr(C)] | ||||||
| pub struct ProgrammableStageDescriptor { | pub struct ProgrammableStageDescriptor { | ||||||
|     module: id::ShaderModuleId, |     module: id::ShaderModuleId, | ||||||
|  | @ -894,30 +887,17 @@ pub unsafe extern "C" fn wgpu_client_create_bind_group( | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[no_mangle] | #[no_mangle] | ||||||
| pub unsafe extern "C" fn wgpu_client_create_shader_module( | pub extern "C" fn wgpu_client_make_shader_module_id( | ||||||
|     client: &Client, |     client: &Client, | ||||||
|     device_id: id::DeviceId, |     device_id: id::DeviceId, | ||||||
|     desc: &ShaderModuleDescriptor, |  | ||||||
|     bb: &mut ByteBuf, |  | ||||||
| ) -> id::ShaderModuleId { | ) -> id::ShaderModuleId { | ||||||
|     let backend = device_id.backend(); |     let backend = device_id.backend(); | ||||||
|     let id = client |     client | ||||||
|         .identities |         .identities | ||||||
|         .lock() |         .lock() | ||||||
|         .select(backend) |         .select(backend) | ||||||
|         .shader_modules |         .shader_modules | ||||||
|         .alloc(backend); |         .alloc(backend) | ||||||
| 
 |  | ||||||
|     let code = |  | ||||||
|         std::str::from_utf8_unchecked(std::slice::from_raw_parts(desc.code, desc.code_length)); |  | ||||||
|     let desc = wgc::pipeline::ShaderModuleDescriptor { |  | ||||||
|         label: cow_label(&desc.label), |  | ||||||
|         shader_bound_checks: wgt::ShaderBoundChecks::new(), |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     let action = DeviceAction::CreateShaderModule(id, desc, Cow::Borrowed(code)); |  | ||||||
|     *bb = make_byte_buf(&action); |  | ||||||
|     id |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[no_mangle] | #[no_mangle] | ||||||
|  |  | ||||||
|  | @ -7,10 +7,14 @@ use crate::{ | ||||||
|     CommandEncoderAction, DeviceAction, DropAction, QueueWriteAction, RawString, TextureAction, |     CommandEncoderAction, DeviceAction, DropAction, QueueWriteAction, RawString, TextureAction, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | use nsstring::nsString; | ||||||
|  | 
 | ||||||
| use wgc::{gfx_select, id}; | use wgc::{gfx_select, id}; | ||||||
|  | use wgc::pipeline::CreateShaderModuleError; | ||||||
| 
 | 
 | ||||||
| use std::sync::atomic::{AtomicU32, Ordering}; | use std::sync::atomic::{AtomicU32, Ordering}; | ||||||
| use std::{error::Error, os::raw::c_char, ptr, slice}; | use std::{error::Error, os::raw::c_char, ptr, slice}; | ||||||
|  | use std::borrow::Cow; | ||||||
| 
 | 
 | ||||||
| /// A fixed-capacity, null-terminated error buffer owned by C++.
 | /// A fixed-capacity, null-terminated error buffer owned by C++.
 | ||||||
| ///
 | ///
 | ||||||
|  | @ -207,6 +211,86 @@ pub extern "C" fn wgpu_server_device_drop(global: &Global, self_id: id::DeviceId | ||||||
|     gfx_select!(self_id => global.device_drop(self_id)) |     gfx_select!(self_id => global.device_drop(self_id)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | impl ShaderModuleCompilationMessage { | ||||||
|  |     fn set_error(&mut self, error: &CreateShaderModuleError, source: &str) { | ||||||
|  |         // The WebGPU spec says that if the message doesn't point to a particular position in
 | ||||||
|  |         // the source, the line number, position, offset and lengths should be zero.
 | ||||||
|  |         self.line_number = 0; | ||||||
|  |         self.line_pos = 0; | ||||||
|  |         self.utf16_offset = 0; | ||||||
|  |         self.utf16_length = 0; | ||||||
|  | 
 | ||||||
|  |         if let Some(location) = error.location(source) { | ||||||
|  |             self.line_number = location.line_number as u64; | ||||||
|  |             self.line_pos = location.line_position as u64; | ||||||
|  | 
 | ||||||
|  |             let start = location.offset as usize; | ||||||
|  |             let end = start + location.length as usize; | ||||||
|  |             self.utf16_offset = source[0..start].chars().map(|c| c.len_utf16() as u64).sum(); | ||||||
|  |             self.utf16_length = source[start..end].chars().map(|c| c.len_utf16() as u64).sum(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let error_string = error.to_string(); | ||||||
|  | 
 | ||||||
|  |         if !error_string.is_empty() { | ||||||
|  |             self.message = nsString::from(&error_string[..]); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A compilation message representation for the ffi boundary.
 | ||||||
|  | /// the message is immediately copied into an equivalent C++
 | ||||||
|  | /// structure that owns its strings.
 | ||||||
|  | #[repr(C)] | ||||||
|  | #[derive(Clone)] | ||||||
|  | pub struct ShaderModuleCompilationMessage { | ||||||
|  |     pub line_number: u64, | ||||||
|  |     pub line_pos: u64, | ||||||
|  |     pub utf16_offset: u64, | ||||||
|  |     pub utf16_length: u64, | ||||||
|  |     pub message: nsString, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Creates a shader module and returns an object describing the errors if any.
 | ||||||
|  | ///
 | ||||||
|  | /// If there was no error, the returned pointer is nil.
 | ||||||
|  | #[no_mangle] | ||||||
|  | pub extern "C" fn wgpu_server_device_create_shader_module( | ||||||
|  |     global: &Global, | ||||||
|  |     self_id: id::DeviceId, | ||||||
|  |     module_id: id::ShaderModuleId, | ||||||
|  |     label: RawString, | ||||||
|  |     code: *const u8, | ||||||
|  |     code_length: usize, | ||||||
|  |     out_message: &mut ShaderModuleCompilationMessage | ||||||
|  | ) -> bool { | ||||||
|  |     unsafe { | ||||||
|  |         let label = cow_label(&label); | ||||||
|  | 
 | ||||||
|  |         let source_str = std::str::from_utf8(std::slice::from_raw_parts(code, code_length)).unwrap(); | ||||||
|  |         let source = wgc::pipeline::ShaderModuleSource::Wgsl(Cow::from(source_str)); | ||||||
|  | 
 | ||||||
|  |         let desc = wgc::pipeline::ShaderModuleDescriptor { | ||||||
|  |             label, | ||||||
|  |             shader_bound_checks: wgt::ShaderBoundChecks::new(), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         let (_, error) = gfx_select!( | ||||||
|  |             self_id => global.device_create_shader_module( | ||||||
|  |                 self_id, &desc, source, module_id | ||||||
|  |             ) | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         if let Some(err) = error { | ||||||
|  |             out_message.set_error(&err, source_str); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Avoid allocating the structure that holds errors in the common case (no errors).
 | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[no_mangle] | #[no_mangle] | ||||||
| pub extern "C" fn wgpu_server_device_create_buffer( | pub extern "C" fn wgpu_server_device_create_buffer( | ||||||
|     global: &Global, |     global: &Global, | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Nicolas Silva
						Nicolas Silva