mirror of
https://github.com/doitsujin/dxvk.git
synced 2025-01-18 20:52:10 +01:00
[d3d9] Limit amount of staging memory in flight
This commit is contained in:
parent
9d981ec1a8
commit
02f653fdd2
@ -4018,12 +4018,91 @@ namespace dxvk {
|
|||||||
|
|
||||||
|
|
||||||
D3D9BufferSlice D3D9DeviceEx::AllocStagingBuffer(VkDeviceSize size) {
|
D3D9BufferSlice D3D9DeviceEx::AllocStagingBuffer(VkDeviceSize size) {
|
||||||
|
m_stagingBufferAllocated += size;
|
||||||
|
|
||||||
D3D9BufferSlice result;
|
D3D9BufferSlice result;
|
||||||
result.slice = m_stagingBuffer.alloc(256, size);
|
result.slice = m_stagingBuffer.alloc(256, size);
|
||||||
result.mapPtr = result.slice.mapPtr(0);
|
result.mapPtr = result.slice.mapPtr(0);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void D3D9DeviceEx::EmitStagingBufferMarker() {
|
||||||
|
if (m_stagingBufferLastAllocated == m_stagingBufferAllocated)
|
||||||
|
return;
|
||||||
|
|
||||||
|
D3D9StagingBufferMarkerPayload payload;
|
||||||
|
payload.sequenceNumber = GetCurrentSequenceNumber();
|
||||||
|
payload.allocated = m_stagingBufferAllocated;
|
||||||
|
m_stagingBufferLastAllocated = m_stagingBufferAllocated;
|
||||||
|
|
||||||
|
Rc<D3D9StagingBufferMarker> marker = new D3D9StagingBufferMarker(payload);
|
||||||
|
m_stagingBufferMarkers.push(marker);
|
||||||
|
|
||||||
|
EmitCs([
|
||||||
|
cMarker = std::move(marker)
|
||||||
|
] (DxvkContext* ctx) {
|
||||||
|
ctx->insertMarker(cMarker);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void D3D9DeviceEx::WaitStagingBuffer() {
|
||||||
|
// The number below is not a hard limit, however we can be reasonably
|
||||||
|
// sure that there will never be more than two additional staging buffers
|
||||||
|
// in flight in addition to the number of staging buffers specified here.
|
||||||
|
constexpr VkDeviceSize maxStagingMemoryInFlight = env::is32BitHostPlatform()
|
||||||
|
? StagingBufferSize * 4
|
||||||
|
: StagingBufferSize * 16;
|
||||||
|
|
||||||
|
// If the game uploads a significant amount of data at once, it's
|
||||||
|
// possible that we exceed the limit while the queue is empty. In
|
||||||
|
// that case, enforce a flush early to populate the marker queue.
|
||||||
|
bool didFlush = false;
|
||||||
|
|
||||||
|
if (m_stagingBufferLastSignaled + maxStagingMemoryInFlight < m_stagingBufferAllocated
|
||||||
|
&& m_stagingBufferMarkers.empty()) {
|
||||||
|
Flush();
|
||||||
|
didFlush = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Process the marker queue. We'll remove as many markers as we
|
||||||
|
// can without stalling, and will stall until we're below the
|
||||||
|
// allocation limit again.
|
||||||
|
uint64_t lastSequenceNumber = m_csThread.lastSequenceNumber();
|
||||||
|
|
||||||
|
while (!m_stagingBufferMarkers.empty()) {
|
||||||
|
const auto& marker = m_stagingBufferMarkers.front();
|
||||||
|
const auto& payload = marker->payload();
|
||||||
|
|
||||||
|
bool needsStall = m_stagingBufferLastSignaled + maxStagingMemoryInFlight < m_stagingBufferAllocated;
|
||||||
|
|
||||||
|
if (payload.sequenceNumber > lastSequenceNumber) {
|
||||||
|
if (!needsStall)
|
||||||
|
break;
|
||||||
|
|
||||||
|
m_csThread.synchronize(payload.sequenceNumber);
|
||||||
|
lastSequenceNumber = payload.sequenceNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (marker->isInUse(DxvkAccess::Read)) {
|
||||||
|
if (!needsStall)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (!didFlush) {
|
||||||
|
Flush();
|
||||||
|
didFlush = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_dxvkDevice->waitForResource(marker, DxvkAccess::Read);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_stagingBufferLastSignaled = marker->payload().allocated;
|
||||||
|
m_stagingBufferMarkers.pop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
bool D3D9DeviceEx::ShouldRecord() {
|
bool D3D9DeviceEx::ShouldRecord() {
|
||||||
return m_recorder != nullptr && !m_recorder->IsApplying();
|
return m_recorder != nullptr && !m_recorder->IsApplying();
|
||||||
}
|
}
|
||||||
@ -4419,8 +4498,9 @@ namespace dxvk {
|
|||||||
UINT SrcSubresource,
|
UINT SrcSubresource,
|
||||||
VkOffset3D SrcOffset,
|
VkOffset3D SrcOffset,
|
||||||
VkExtent3D SrcExtent,
|
VkExtent3D SrcExtent,
|
||||||
VkOffset3D DestOffset
|
VkOffset3D DestOffset) {
|
||||||
) {
|
WaitStagingBuffer();
|
||||||
|
|
||||||
const Rc<DxvkImage> image = pDestTexture->GetImage();
|
const Rc<DxvkImage> image = pDestTexture->GetImage();
|
||||||
|
|
||||||
// Now that data has been written into the buffer,
|
// Now that data has been written into the buffer,
|
||||||
@ -4662,6 +4742,8 @@ namespace dxvk {
|
|||||||
|
|
||||||
HRESULT D3D9DeviceEx::FlushBuffer(
|
HRESULT D3D9DeviceEx::FlushBuffer(
|
||||||
D3D9CommonBuffer* pResource) {
|
D3D9CommonBuffer* pResource) {
|
||||||
|
WaitStagingBuffer();
|
||||||
|
|
||||||
auto dstBuffer = pResource->GetBufferSlice<D3D9_COMMON_BUFFER_TYPE_REAL>();
|
auto dstBuffer = pResource->GetBufferSlice<D3D9_COMMON_BUFFER_TYPE_REAL>();
|
||||||
auto srcSlice = pResource->GetMappedSlice();
|
auto srcSlice = pResource->GetMappedSlice();
|
||||||
|
|
||||||
@ -5096,6 +5178,8 @@ namespace dxvk {
|
|||||||
m_initializer->Flush();
|
m_initializer->Flush();
|
||||||
m_converter->Flush();
|
m_converter->Flush();
|
||||||
|
|
||||||
|
EmitStagingBufferMarker();
|
||||||
|
|
||||||
if (m_csIsBusy || !m_csChunk->empty()) {
|
if (m_csIsBusy || !m_csChunk->empty()) {
|
||||||
// Add commands to flush the threaded
|
// Add commands to flush the threaded
|
||||||
// context, then flush the command list
|
// context, then flush the command list
|
||||||
|
@ -96,6 +96,13 @@ namespace dxvk {
|
|||||||
void* mapPtr = nullptr;
|
void* mapPtr = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct D3D9StagingBufferMarkerPayload {
|
||||||
|
uint64_t sequenceNumber;
|
||||||
|
VkDeviceSize allocated;
|
||||||
|
};
|
||||||
|
|
||||||
|
using D3D9StagingBufferMarker = DxvkMarker<D3D9StagingBufferMarkerPayload>;
|
||||||
|
|
||||||
class D3D9DeviceEx final : public ComObjectClamp<IDirect3DDevice9Ex> {
|
class D3D9DeviceEx final : public ComObjectClamp<IDirect3DDevice9Ex> {
|
||||||
constexpr static uint32_t DefaultFrameLatency = 3;
|
constexpr static uint32_t DefaultFrameLatency = 3;
|
||||||
constexpr static uint32_t MaxFrameLatency = 20;
|
constexpr static uint32_t MaxFrameLatency = 20;
|
||||||
@ -977,6 +984,10 @@ namespace dxvk {
|
|||||||
|
|
||||||
D3D9BufferSlice AllocStagingBuffer(VkDeviceSize size);
|
D3D9BufferSlice AllocStagingBuffer(VkDeviceSize size);
|
||||||
|
|
||||||
|
void EmitStagingBufferMarker();
|
||||||
|
|
||||||
|
void WaitStagingBuffer();
|
||||||
|
|
||||||
bool ShouldRecord();
|
bool ShouldRecord();
|
||||||
|
|
||||||
HRESULT CreateShaderModule(
|
HRESULT CreateShaderModule(
|
||||||
@ -1188,6 +1199,10 @@ namespace dxvk {
|
|||||||
void* m_upBufferMapPtr = nullptr;
|
void* m_upBufferMapPtr = nullptr;
|
||||||
|
|
||||||
DxvkStagingBuffer m_stagingBuffer;
|
DxvkStagingBuffer m_stagingBuffer;
|
||||||
|
VkDeviceSize m_stagingBufferAllocated = 0ull;
|
||||||
|
VkDeviceSize m_stagingBufferLastAllocated = 0ull;
|
||||||
|
VkDeviceSize m_stagingBufferLastSignaled = 0ull;
|
||||||
|
std::queue<Rc<D3D9StagingBufferMarker>> m_stagingBufferMarkers;
|
||||||
|
|
||||||
D3D9Cursor m_cursor;
|
D3D9Cursor m_cursor;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user