1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2025-01-21 02:52:10 +01:00
dxvk/src/d3d9/d3d9_mem.h
Tatsuyuki Ishi e2a46a347d [meta] Declare bool conversion operators as explicit
Non-explicit conversion operators in general can participate in very
surprising conversion chains. Explicit bool operator is a good place to
start with, because even with explicit they do get automatic contextual
conversion in a lot of places, e.g., if conditions.
2024-01-27 11:44:51 +01:00

177 lines
4.3 KiB
C++

#pragma once
#include "../util/thread.h"
#if defined(_WIN32) && !defined(_WIN64)
#define D3D9_ALLOW_UNMAPPING
#endif
#ifdef D3D9_ALLOW_UNMAPPING
#define WIN32_LEAN_AND_MEAN
#include <winbase.h>
#endif
#include <vector>
namespace dxvk {
class D3D9MemoryAllocator;
class D3D9Memory;
#ifdef D3D9_ALLOW_UNMAPPING
class D3D9MemoryChunk;
constexpr uint32_t D3D9ChunkSize = 64 << 20;
struct D3D9MemoryRange {
uint32_t offset;
uint32_t length;
};
struct D3D9MappingRange {
uint32_t refCount = 0;
void* ptr = nullptr;
};
class D3D9MemoryChunk {
friend D3D9MemoryAllocator;
public:
~D3D9MemoryChunk();
D3D9MemoryChunk (const D3D9MemoryChunk&) = delete;
D3D9MemoryChunk& operator = (const D3D9MemoryChunk&) = delete;
D3D9MemoryChunk (D3D9MemoryChunk&& other) = delete;
D3D9MemoryChunk& operator = (D3D9MemoryChunk&& other) = delete;
D3D9Memory Alloc(uint32_t Size);
void Free(D3D9Memory* Memory);
bool IsEmpty();
uint32_t Size() const { return m_size; }
D3D9MemoryAllocator* Allocator() const;
HANDLE FileHandle() const;
void* Map(D3D9Memory* memory);
void Unmap(D3D9Memory* memory);
private:
D3D9MemoryChunk(D3D9MemoryAllocator* Allocator, uint32_t Size);
dxvk::mutex m_mutex;
D3D9MemoryAllocator* m_allocator;
HANDLE m_mapping;
uint32_t m_size;
uint32_t m_mappingGranularity;
std::vector<D3D9MemoryRange> m_freeRanges;
std::vector<D3D9MappingRange> m_mappingRanges;
};
class D3D9Memory {
friend D3D9MemoryChunk;
public:
D3D9Memory() {}
~D3D9Memory();
D3D9Memory (const D3D9Memory&) = delete;
D3D9Memory& operator = (const D3D9Memory&) = delete;
D3D9Memory (D3D9Memory&& other);
D3D9Memory& operator = (D3D9Memory&& other);
explicit operator bool() const { return m_chunk != nullptr; }
void Map();
void Unmap();
void* Ptr();
D3D9MemoryChunk* GetChunk() const { return m_chunk; }
size_t GetOffset() const { return m_offset; }
size_t GetSize() const { return m_size; }
private:
D3D9Memory(D3D9MemoryChunk* Chunk, size_t Offset, size_t Size);
void Free();
D3D9MemoryChunk* m_chunk = nullptr;
void* m_ptr = nullptr;
size_t m_offset = 0;
size_t m_size = 0;
};
class D3D9MemoryAllocator {
friend D3D9MemoryChunk;
public:
D3D9MemoryAllocator();
~D3D9MemoryAllocator() = default;
D3D9Memory Alloc(uint32_t Size);
void FreeChunk(D3D9MemoryChunk* Chunk);
void NotifyMapped(uint32_t Size);
void NotifyUnmapped(uint32_t Size);
void NotifyFreed(uint32_t Size);
uint32_t MappedMemory();
uint32_t UsedMemory();
uint32_t AllocatedMemory();
uint32_t MemoryGranularity() { return m_allocationGranularity; }
private:
dxvk::mutex m_mutex;
std::vector<std::unique_ptr<D3D9MemoryChunk>> m_chunks;
std::atomic<size_t> m_mappedMemory = 0;
std::atomic<size_t> m_allocatedMemory = 0;
std::atomic<size_t> m_usedMemory = 0;
uint32_t m_allocationGranularity;
};
#else
class D3D9Memory {
friend D3D9MemoryAllocator;
public:
D3D9Memory() {}
~D3D9Memory();
D3D9Memory (const D3D9Memory&) = delete;
D3D9Memory& operator = (const D3D9Memory&) = delete;
D3D9Memory (D3D9Memory&& other);
D3D9Memory& operator = (D3D9Memory&& other);
explicit operator bool() const { return m_ptr != nullptr; }
void Map() {}
void Unmap() {}
void* Ptr() { return m_ptr; }
size_t GetSize() const { return m_size; }
private:
D3D9Memory(D3D9MemoryAllocator* pAllocator, size_t Size);
void Free();
D3D9MemoryAllocator* m_allocator = nullptr;
void* m_ptr = nullptr;
size_t m_size = 0;
};
class D3D9MemoryAllocator {
public:
D3D9Memory Alloc(uint32_t Size);
uint32_t MappedMemory();
uint32_t UsedMemory();
uint32_t AllocatedMemory();
void NotifyFreed(uint32_t Size) {
m_allocatedMemory -= Size;
}
private:
std::atomic<size_t> m_allocatedMemory = 0;
};
#endif
}