// Copyright 2017 The Dawn Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef DAWNNATIVE_DEVICE_H_ #define DAWNNATIVE_DEVICE_H_ #include "common/Serial.h" #include "dawn_native/Error.h" #include "dawn_native/Forward.h" #include "dawn_native/ObjectBase.h" #include "dawn_native/DawnNative.h" #include "dawn_native/dawn_platform.h" #include namespace dawn_native { using ErrorCallback = void (*)(const char* errorMessage, void* userData); class AdapterBase; class FenceSignalTracker; class DynamicUploader; class StagingBufferBase; class DeviceBase { public: DeviceBase(AdapterBase* adapter); virtual ~DeviceBase(); void HandleError(const char* message); bool ConsumedError(MaybeError maybeError) { if (DAWN_UNLIKELY(maybeError.IsError())) { ConsumeError(maybeError.AcquireError()); return true; } return false; } MaybeError ValidateObject(const ObjectBase* object) const; AdapterBase* GetAdapter() const; // Used by autogenerated code, returns itself DeviceBase* GetDevice(); FenceSignalTracker* GetFenceSignalTracker() const; virtual CommandBufferBase* CreateCommandBuffer(CommandBufferBuilder* builder) = 0; virtual InputStateBase* CreateInputState(InputStateBuilder* builder) = 0; virtual RenderPassDescriptorBase* CreateRenderPassDescriptor( RenderPassDescriptorBuilder* builder) = 0; virtual SwapChainBase* CreateSwapChain(SwapChainBuilder* builder) = 0; virtual Serial GetCompletedCommandSerial() const = 0; virtual Serial GetLastSubmittedCommandSerial() const = 0; virtual Serial GetPendingCommandSerial() const = 0; virtual void TickImpl() = 0; // Many Dawn objects are completely immutable once created which means that if two // builders are given the same arguments, they can return the same object. Reusing // objects will help make comparisons between objects by a single pointer comparison. // // Technically no object is immutable as they have a reference count, and an // application with reference-counting issues could "see" that objects are reused. // This is solved by automatic-reference counting, and also the fact that when using // the client-server wire every creation will get a different proxy object, with a // different reference count. // // When trying to create an object, we give both the builder and an example of what // the built object will be, the "blueprint". The blueprint is just a FooBase object // instead of a backend Foo object. If the blueprint doesn't match an object in the // cache, then the builder is used to make a new object. ResultOrError GetOrCreateBindGroupLayout( const BindGroupLayoutDescriptor* descriptor); void UncacheBindGroupLayout(BindGroupLayoutBase* obj); // Dawn API BindGroupBase* CreateBindGroup(const BindGroupDescriptor* descriptor); BindGroupLayoutBase* CreateBindGroupLayout(const BindGroupLayoutDescriptor* descriptor); BufferBase* CreateBuffer(const BufferDescriptor* descriptor); CommandBufferBuilder* CreateCommandBufferBuilder(); ComputePipelineBase* CreateComputePipeline(const ComputePipelineDescriptor* descriptor); FenceBase* CreateFence(const FenceDescriptor* descriptor); InputStateBuilder* CreateInputStateBuilder(); PipelineLayoutBase* CreatePipelineLayout(const PipelineLayoutDescriptor* descriptor); QueueBase* CreateQueue(); RenderPassDescriptorBuilder* CreateRenderPassDescriptorBuilder(); RenderPipelineBase* CreateRenderPipeline(const RenderPipelineDescriptor* descriptor); SamplerBase* CreateSampler(const SamplerDescriptor* descriptor); ShaderModuleBase* CreateShaderModule(const ShaderModuleDescriptor* descriptor); SwapChainBuilder* CreateSwapChainBuilder(); TextureBase* CreateTexture(const TextureDescriptor* descriptor); TextureViewBase* CreateTextureView(TextureBase* texture, const TextureViewDescriptor* descriptor); void Tick(); void SetErrorCallback(dawn::DeviceErrorCallback callback, dawn::CallbackUserdata userdata); void Reference(); void Release(); BufferBuilder* CreateBufferBuilderForTesting() { return nullptr; } virtual ResultOrError> CreateStagingBuffer( size_t size) = 0; virtual MaybeError CopyFromStagingToBuffer(StagingBufferBase* source, uint32_t sourceOffset, BufferBase* destination, uint32_t destinationOffset, uint32_t size) = 0; private: virtual ResultOrError CreateBindGroupImpl( const BindGroupDescriptor* descriptor) = 0; virtual ResultOrError CreateBindGroupLayoutImpl( const BindGroupLayoutDescriptor* descriptor) = 0; virtual ResultOrError CreateBufferImpl(const BufferDescriptor* descriptor) = 0; virtual ResultOrError CreateComputePipelineImpl( const ComputePipelineDescriptor* descriptor) = 0; virtual ResultOrError CreatePipelineLayoutImpl( const PipelineLayoutDescriptor* descriptor) = 0; virtual ResultOrError CreateQueueImpl() = 0; virtual ResultOrError CreateRenderPipelineImpl( const RenderPipelineDescriptor* descriptor) = 0; virtual ResultOrError CreateSamplerImpl( const SamplerDescriptor* descriptor) = 0; virtual ResultOrError CreateShaderModuleImpl( const ShaderModuleDescriptor* descriptor) = 0; virtual ResultOrError CreateTextureImpl( const TextureDescriptor* descriptor) = 0; virtual ResultOrError CreateTextureViewImpl( TextureBase* texture, const TextureViewDescriptor* descriptor) = 0; MaybeError CreateBindGroupInternal(BindGroupBase** result, const BindGroupDescriptor* descriptor); MaybeError CreateBindGroupLayoutInternal(BindGroupLayoutBase** result, const BindGroupLayoutDescriptor* descriptor); MaybeError CreateBufferInternal(BufferBase** result, const BufferDescriptor* descriptor); MaybeError CreateComputePipelineInternal(ComputePipelineBase** result, const ComputePipelineDescriptor* descriptor); MaybeError CreateFenceInternal(FenceBase** result, const FenceDescriptor* descriptor); MaybeError CreatePipelineLayoutInternal(PipelineLayoutBase** result, const PipelineLayoutDescriptor* descriptor); MaybeError CreateQueueInternal(QueueBase** result); MaybeError CreateRenderPipelineInternal(RenderPipelineBase** result, const RenderPipelineDescriptor* descriptor); MaybeError CreateSamplerInternal(SamplerBase** result, const SamplerDescriptor* descriptor); MaybeError CreateShaderModuleInternal(ShaderModuleBase** result, const ShaderModuleDescriptor* descriptor); MaybeError CreateTextureInternal(TextureBase** result, const TextureDescriptor* descriptor); MaybeError CreateTextureViewInternal(TextureViewBase** result, TextureBase* texture, const TextureViewDescriptor* descriptor); void ConsumeError(ErrorData* error); AdapterBase* mAdapter = nullptr; // The object caches aren't exposed in the header as they would require a lot of // additional includes. struct Caches; std::unique_ptr mCaches; std::unique_ptr mFenceSignalTracker; dawn::DeviceErrorCallback mErrorCallback = nullptr; dawn::CallbackUserdata mErrorUserdata = 0; uint32_t mRefCount = 1; }; } // namespace dawn_native #endif // DAWNNATIVE_DEVICE_H_