2017-10-10 23:32:13 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "dxvk_include.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
2018-06-16 10:40:30 +02:00
|
|
|
enum class DxvkAccess {
|
2017-11-26 13:24:01 +01:00
|
|
|
Read = 0,
|
|
|
|
Write = 1,
|
2019-09-19 19:35:52 +02:00
|
|
|
None = 2,
|
2017-11-26 13:24:01 +01:00
|
|
|
};
|
|
|
|
|
2018-06-16 10:40:30 +02:00
|
|
|
using DxvkAccessFlags = Flags<DxvkAccess>;
|
2017-11-26 13:24:01 +01:00
|
|
|
|
2017-10-10 23:32:13 +02:00
|
|
|
/**
|
|
|
|
* \brief DXVK resource
|
|
|
|
*
|
|
|
|
* Keeps track of whether the resource is currently in use
|
|
|
|
* by the GPU. As soon as a command that uses the resource
|
|
|
|
* is recorded, it will be marked as 'in use'.
|
|
|
|
*/
|
|
|
|
class DxvkResource : public RcObject {
|
2019-09-19 19:35:52 +02:00
|
|
|
constexpr static uint32_t UseCountIncrementW = 1 << 18;
|
|
|
|
constexpr static uint32_t UseCountIncrementR = 1;
|
|
|
|
constexpr static uint32_t UseCountMaskW = ~(UseCountIncrementW - 1);
|
|
|
|
constexpr static uint32_t UseCountMaskR = ~(UseCountIncrementR - 1);
|
2017-10-10 23:32:13 +02:00
|
|
|
public:
|
|
|
|
|
|
|
|
virtual ~DxvkResource();
|
|
|
|
|
2019-09-19 19:35:52 +02:00
|
|
|
/**
|
|
|
|
* \brief Checks whether resource is in use
|
|
|
|
*
|
|
|
|
* Returns \c true if there are pending accesses to
|
|
|
|
* the resource by the GPU matching the given access
|
|
|
|
* type. Note that checking for reads will also return
|
|
|
|
* \c true if the resource is being written to.
|
|
|
|
* \param [in] access Access type to check for
|
|
|
|
* \returns \c true if the resource is in use
|
|
|
|
*/
|
|
|
|
bool isInUse(DxvkAccess access = DxvkAccess::Read) const {
|
|
|
|
uint32_t mask = access == DxvkAccess::Write
|
|
|
|
? UseCountMaskW
|
|
|
|
: UseCountMaskR;
|
|
|
|
return m_useCount.load() & mask;
|
2017-10-10 23:32:13 +02:00
|
|
|
}
|
|
|
|
|
2019-09-19 19:35:52 +02:00
|
|
|
/**
|
|
|
|
* \brief Acquires resource
|
|
|
|
*
|
|
|
|
* Increments use count for the given access type.
|
|
|
|
* \param Access Resource access type
|
|
|
|
*/
|
|
|
|
void acquire(DxvkAccess access) {
|
|
|
|
if (access != DxvkAccess::None)
|
|
|
|
m_useCount += getIncrement(access);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Releases resource
|
|
|
|
*
|
|
|
|
* Decrements use count for the given access type.
|
|
|
|
* \param Access Resource access type
|
|
|
|
*/
|
|
|
|
void release(DxvkAccess access) {
|
|
|
|
if (access != DxvkAccess::None)
|
|
|
|
m_useCount -= getIncrement(access);
|
|
|
|
}
|
2017-10-10 23:32:13 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
std::atomic<uint32_t> m_useCount = { 0u };
|
2019-09-19 19:35:52 +02:00
|
|
|
|
|
|
|
static constexpr uint32_t getIncrement(DxvkAccess access) {
|
|
|
|
return access == DxvkAccess::Write
|
|
|
|
? UseCountIncrementW
|
|
|
|
: UseCountIncrementR;
|
|
|
|
}
|
|
|
|
|
2017-10-10 23:32:13 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|