blob: 64ec58e12576b9b8a37614f28c35db5e0f9aa283 [file] [log] [blame] [edit]
// Copyright 2024 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#ifndef SRC_DAWN_NATIVE_SHAREDRESOURCEMEMORY_H_
#define SRC_DAWN_NATIVE_SHAREDRESOURCEMEMORY_H_
#include "dawn/common/StackContainer.h"
#include "dawn/common/WeakRef.h"
#include "dawn/common/WeakRefSupport.h"
#include "dawn/native/Error.h"
#include "dawn/native/Forward.h"
#include "dawn/native/IntegerTypes.h"
#include "dawn/native/ObjectBase.h"
#include "dawn/native/SharedFence.h"
#include "dawn/native/dawn_platform.h"
namespace dawn::native {
class SharedResourceMemoryContents;
enum SharedResourceAccessState { NotAccessed, ExclusiveRead, SimultaneousRead, Write };
class SharedResource : public ApiObjectBase {
public:
using ApiObjectBase::ApiObjectBase;
~SharedResource() override;
SharedResourceMemoryContents* GetSharedResourceMemoryContents() const;
virtual void SetHasAccess(bool hasAccess) = 0;
virtual bool HasAccess() const = 0;
virtual bool IsDestroyed() const = 0;
virtual void SetInitialized(bool initialized) = 0;
virtual bool IsInitialized() const = 0;
protected:
// The shared contents the resource was created from. May be null.
Ref<SharedResourceMemoryContents> mSharedResourceMemoryContents;
};
class SharedResourceMemory : public ApiObjectBase, public WeakRefSupport<SharedResourceMemory> {
public:
using PendingFenceList = StackVector<FenceAndSignalValue, 1>;
~SharedResourceMemory() override;
void Initialize();
void DestroyImpl() override;
// Returns true if access was acquired. If it returns true, then APIEndAccess must
// be called to release access. Other errors may occur even if `true` is returned.
// Use an error scope to catch them.
bool APIBeginAccess(TextureBase* texture,
const SharedTextureMemoryBeginAccessDescriptor* descriptor);
// Returns true if access was released.
bool APIEndAccess(TextureBase* texture, SharedTextureMemoryEndAccessState* state);
// Returns true if access was acquired. If it returns true, then APIEndAccess must
// be called to release access. Other errors may occur even if `true` is returned.
// Use an error scope to catch them.
bool APIBeginAccess(BufferBase* buffer,
const SharedBufferMemoryBeginAccessDescriptor* descriptor);
// Returns true if access was released.
bool APIEndAccess(BufferBase* buffer, SharedBufferMemoryEndAccessState* state);
// Returns true iff the device passed to this object on creation is now lost.
// TODO(crbug.com/1506468): Eliminate this API once Chromium has been
// transitioned away from using it in favor of observing device lost events.
bool APIIsDeviceLost() const;
SharedResourceMemoryContents* GetContents() const;
// Validate that the resource was created from this SharedResourceMemory.
MaybeError ValidateResourceCreatedFromSelf(SharedResource* resource);
protected:
SharedResourceMemory(DeviceBase* device, ObjectBase::ErrorTag, const char* label);
using ApiObjectBase::ApiObjectBase;
bool HasWriteAccess() const;
bool HasExclusiveReadAccess() const;
int GetReadAccessCount() const;
private:
virtual Ref<SharedResourceMemoryContents> CreateContents();
template <typename Resource, typename BeginAccessDescriptor>
MaybeError BeginAccess(Resource* resource, const BeginAccessDescriptor* rawDescriptor);
template <typename Resource, typename EndAccessState>
MaybeError EndAccess(Resource* resource, EndAccessState* state, bool* didEnd);
template <typename Resource, typename EndAccessState>
ResultOrError<FenceAndSignalValue> EndAccessInternal(Resource* resource,
EndAccessState* rawState);
// BeginAccessImpl validates the operation is valid on the backend, and performs any
// backend specific operations. It does NOT need to acquire begin fences; that is done in the
// frontend in BeginAccess.
virtual MaybeError BeginAccessImpl(
TextureBase* texture,
const UnpackedPtr<SharedTextureMemoryBeginAccessDescriptor>& descriptor);
virtual MaybeError BeginAccessImpl(
BufferBase* buffer,
const UnpackedPtr<SharedBufferMemoryBeginAccessDescriptor>& descriptor);
// EndAccessImpl validates the operation is valid on the backend, and returns the end fence.
// It should also write out any backend specific state in chained out structs of EndAccessState.
virtual ResultOrError<FenceAndSignalValue> EndAccessImpl(
TextureBase* texture,
UnpackedPtr<SharedTextureMemoryEndAccessState>& state);
virtual ResultOrError<FenceAndSignalValue> EndAccessImpl(
BufferBase* buffer,
UnpackedPtr<SharedBufferMemoryEndAccessState>& state);
Ref<SharedResource> mExclusiveAccess;
SharedResourceAccessState mSharedResourceAccessState = SharedResourceAccessState::NotAccessed;
int mReadAccessCount = 0;
Ref<SharedResourceMemoryContents> mContents;
};
// SharedResourceMemoryContents is a separate object because it needs to live as long as
// the SharedResourceMemory or any resources created from the SharedResourceMemory. This
// allows state and objects needed by the resource to persist after the
// SharedResourceMemory itself has been dropped.
class SharedResourceMemoryContents : public RefCounted {
public:
using PendingFenceList = SharedResourceMemory::PendingFenceList;
explicit SharedResourceMemoryContents(WeakRef<SharedResourceMemory> sharedResourceMemory);
void AcquirePendingFences(PendingFenceList* fences);
// Set the last usage serial. This indicates when the SharedFence exported
// from APIEndAccess will complete.
void SetLastUsageSerial(ExecutionSerial lastUsageSerial);
ExecutionSerial GetLastUsageSerial() const;
const WeakRef<SharedResourceMemory>& GetSharedResourceMemory() const;
private:
friend class SharedResourceMemory;
PendingFenceList mPendingFences;
ExecutionSerial mLastUsageSerial{0};
WeakRef<SharedResourceMemory> mSharedResourceMemory;
};
} // namespace dawn::native
#endif // SRC_DAWN_NATIVE_SHAREDRESOURCEMEMORY_H_