1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2025-04-05 16:40:17 +02:00

[d3d11] Reset dirty bindings on command submission

This commit is contained in:
Philip Rebohle 2025-02-19 15:41:39 +01:00 committed by Philip Rebohle
parent 3e6dfcfb15
commit c0983a32be
4 changed files with 86 additions and 1 deletions

View File

@ -3186,6 +3186,7 @@ namespace dxvk {
if (!bindMask) if (!bindMask)
return; return;
// Need to clear dirty bits before binding
const auto& state = m_state.cbv[Stage]; const auto& state = m_state.cbv[Stage];
DirtyMask.cbvMask -= bindMask; DirtyMask.cbvMask -= bindMask;
@ -3208,6 +3209,7 @@ namespace dxvk {
if (!bindMask) if (!bindMask)
return; return;
// Need to clear dirty bits before binding
const auto& state = m_state.samplers[Stage]; const auto& state = m_state.samplers[Stage];
DirtyMask.samplerMask -= bindMask; DirtyMask.samplerMask -= bindMask;
@ -3230,6 +3232,7 @@ namespace dxvk {
if (!bindMask) if (!bindMask)
continue; continue;
// Need to clear dirty bits before binding
DirtyMask.srvMask[maskIndex] -= bindMask; DirtyMask.srvMask[maskIndex] -= bindMask;
for (uint32_t slot : bit::BitMask(bindMask)) for (uint32_t slot : bit::BitMask(bindMask))
@ -4889,6 +4892,8 @@ namespace dxvk {
for (uint32_t i = 0; i < m_state.so.targets.size(); i++) for (uint32_t i = 0; i < m_state.so.targets.size(); i++)
BindXfbBuffer(i, m_state.so.targets[i].buffer.ptr(), ~0u); BindXfbBuffer(i, m_state.so.targets[i].buffer.ptr(), ~0u);
// Reset dirty binding and shader masks before applying
// bindings to avoid implicit null binding overrids.
ResetDirtyTracking(); ResetDirtyTracking();
for (uint32_t i = 0; i < uint32_t(DxbcProgramType::Count); i++) { for (uint32_t i = 0; i < uint32_t(DxbcProgramType::Count); i++) {

View File

@ -218,7 +218,12 @@ namespace dxvk {
D3D10DeviceLock lock = LockContext(); D3D10DeviceLock lock = LockContext();
auto commandList = static_cast<D3D11CommandList*>(pCommandList); auto commandList = static_cast<D3D11CommandList*>(pCommandList);
// Reset dirty binding tracking before submitting any CS chunks.
// This is needed so that any submission that might occur during
// this call does not disrupt bindings set by the deferred context.
ResetDirtyTracking();
// Clear state so that the command list can't observe any // Clear state so that the command list can't observe any
// current context state. The command list itself will clean // current context state. The command list itself will clean
// up after execution to ensure that no state changes done // up after execution to ensure that no state changes done
@ -979,6 +984,73 @@ namespace dxvk {
} }
void D3D11ImmediateContext::ApplyDirtyNullBindings() {
// At the end of a submission, set all bindings that have not been applied yet
// to null on the DXVK context. This way, we avoid keeping resources alive that
// are bound to the DXVK context but not to the immediate context.
//
// Note: This requires that all methods that may modify dirty bindings on the
// DXVK context also reset the corresponding dirty bits *before* performing the
// bind operation, or otherwise an implicit flush can potentially override them.
auto& dirtyState = m_state.lazy.bindingsDirty;
EmitCs<false>([
cDirtyState = dirtyState
] (DxvkContext* ctx) {
for (uint32_t i = 0; i < uint32_t(DxbcProgramType::Count); i++) {
auto dxStage = DxbcProgramType(i);
auto vkStage = GetShaderStage(dxStage);
// Unbind all dirty constant buffers
auto cbvSlot = computeConstantBufferBinding(dxStage, 0);
for (uint32_t index : bit::BitMask(cDirtyState[dxStage].cbvMask))
ctx->bindUniformBuffer(vkStage, cbvSlot + index, DxvkBufferSlice());
// Unbind all dirty samplers
auto samplerSlot = computeSamplerBinding(dxStage, 0);
for (uint32_t index : bit::BitMask(cDirtyState[dxStage].samplerMask))
ctx->bindResourceSampler(vkStage, samplerSlot + index, nullptr);
// Unbind all dirty shader resource views
auto srvSlot = computeSrvBinding(dxStage, 0);
for (uint32_t m = 0; m < cDirtyState[dxStage].srvMask.size(); m++) {
for (uint32_t index : bit::BitMask(cDirtyState[dxStage].srvMask[m]))
ctx->bindResourceImageView(vkStage, srvSlot + index + m * 64u, nullptr);
}
}
});
// Since we set the DXVK context bindings to null, any bindings that are null
// on the D3D context are no longer dirty, so we can clear the respective bits.
for (uint32_t i = 0; i < uint32_t(DxbcProgramType::Count); i++) {
auto stage = DxbcProgramType(i);
for (uint32_t index : bit::BitMask(dirtyState[stage].cbvMask)) {
if (!m_state.cbv[stage].buffers[index].buffer.ptr())
dirtyState[stage].cbvMask &= ~(1u << index);
}
for (uint32_t index : bit::BitMask(dirtyState[stage].samplerMask)) {
if (!m_state.samplers[stage].samplers[index])
dirtyState[stage].samplerMask &= ~(1u << index);
}
for (uint32_t m = 0; m < dirtyState[stage].srvMask.size(); m++) {
for (uint32_t index : bit::BitMask(dirtyState[stage].srvMask[m])) {
if (!m_state.srv[stage].views[index + m * 64u].ptr())
dirtyState[stage].srvMask[m] &= ~(uint64_t(1u) << index);
}
}
if (dirtyState[stage].empty())
m_state.lazy.shadersDirty.clr(stage);
}
}
void D3D11ImmediateContext::ConsiderFlush( void D3D11ImmediateContext::ConsiderFlush(
GpuFlushType FlushType) { GpuFlushType FlushType) {
uint64_t chunkId = GetCurrentSequenceNumber(); uint64_t chunkId = GetCurrentSequenceNumber();
@ -1002,6 +1074,9 @@ namespace dxvk {
if (!GetPendingCsChunks() && !hEvent) if (!GetPendingCsChunks() && !hEvent)
return; return;
// Unbind unused resources
ApplyDirtyNullBindings();
// Signal the submission fence and flush the command list // Signal the submission fence and flush the command list
uint64_t submissionId = ++m_submissionId; uint64_t submissionId = ++m_submissionId;

View File

@ -195,6 +195,8 @@ namespace dxvk {
uint64_t GetPendingCsChunks(); uint64_t GetPendingCsChunks();
void ApplyDirtyNullBindings();
void ConsiderFlush( void ConsiderFlush(
GpuFlushType FlushType); GpuFlushType FlushType);

View File

@ -1037,7 +1037,9 @@ namespace dxvk {
continue; continue;
if (!hasStreamsEnabled) { if (!hasStreamsEnabled) {
m_ctx->ResetDirtyTracking();
m_ctx->ResetCommandListState(); m_ctx->ResetCommandListState();
BindOutputView(pOutputView); BindOutputView(pOutputView);
hasStreamsEnabled = true; hasStreamsEnabled = true;
} }
@ -1047,6 +1049,7 @@ namespace dxvk {
if (hasStreamsEnabled) { if (hasStreamsEnabled) {
UnbindResources(); UnbindResources();
m_ctx->RestoreCommandListState(); m_ctx->RestoreCommandListState();
} }