1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2024-12-12 22:08:59 +01:00

[dxvk] Use existing bit mask iterator when updating descriptor sets

And clean up the code a little.
This commit is contained in:
Philip Rebohle 2022-08-05 21:13:26 +02:00
parent 9e1ecec79f
commit 2fe61675bf
No known key found for this signature in database
GPG Key ID: C8CC613427A31C99

View File

@ -4675,17 +4675,12 @@ namespace dxvk {
: m_descriptorState.getDirtyComputeSets(); : m_descriptorState.getDirtyComputeSets();
dirtySetMask &= layoutSetMask; dirtySetMask &= layoutSetMask;
uint32_t bindCount = 0;
uint32_t k = 0;
std::array<VkDescriptorSet, DxvkDescriptorSets::SetCount> sets; std::array<VkDescriptorSet, DxvkDescriptorSets::SetCount> sets;
m_descriptorPool->alloc(layout, dirtySetMask, sets.data()); m_descriptorPool->alloc(layout, dirtySetMask, sets.data());
while (dirtySetMask) { uint32_t descriptorCount = 0;
uint32_t setIndex = bit::tzcnt(dirtySetMask);
// Initialize binding mask for the current set, only for (auto setIndex : bit::BitMask(dirtySetMask)) {
// clear bits if certain resources are actually unbound.
uint32_t bindingCount = bindings.getBindingCount(setIndex); uint32_t bindingCount = bindings.getBindingCount(setIndex);
VkDescriptorSet set = sets[setIndex]; VkDescriptorSet set = sets[setIndex];
@ -4693,26 +4688,29 @@ namespace dxvk {
const auto& binding = bindings.getBinding(setIndex, j); const auto& binding = bindings.getBinding(setIndex, j);
if (!useDescriptorTemplates) { if (!useDescriptorTemplates) {
m_descriptorWrites[k].dstSet = set; auto& descriptorWrite = m_descriptorWrites[descriptorCount];
m_descriptorWrites[k].dstBinding = j; descriptorWrite.dstSet = set;
m_descriptorWrites[k].descriptorType = binding.descriptorType; descriptorWrite.dstBinding = j;
descriptorWrite.descriptorType = binding.descriptorType;
} }
auto& descriptorInfo = m_descriptors[descriptorCount++];
switch (binding.descriptorType) { switch (binding.descriptorType) {
case VK_DESCRIPTOR_TYPE_SAMPLER: { case VK_DESCRIPTOR_TYPE_SAMPLER: {
const auto& res = m_rc[binding.resourceBinding]; const auto& res = m_rc[binding.resourceBinding];
if (res.sampler != nullptr) { if (res.sampler != nullptr) {
m_descriptors[k].image.sampler = res.sampler->handle(); descriptorInfo.image.sampler = res.sampler->handle();
m_descriptors[k].image.imageView = VK_NULL_HANDLE; descriptorInfo.image.imageView = VK_NULL_HANDLE;
m_descriptors[k].image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED; descriptorInfo.image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
if (m_rcTracked.set(binding.resourceBinding)) if (m_rcTracked.set(binding.resourceBinding))
m_cmd->trackResource<DxvkAccess::None>(res.sampler); m_cmd->trackResource<DxvkAccess::None>(res.sampler);
} else { } else {
m_descriptors[k].image.sampler = m_common->dummyResources().samplerHandle(); descriptorInfo.image.sampler = m_common->dummyResources().samplerHandle();
m_descriptors[k].image.imageView = VK_NULL_HANDLE; descriptorInfo.image.imageView = VK_NULL_HANDLE;
m_descriptors[k].image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED; descriptorInfo.image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
} }
} break; } break;
@ -4720,18 +4718,18 @@ namespace dxvk {
const auto& res = m_rc[binding.resourceBinding]; const auto& res = m_rc[binding.resourceBinding];
if (res.imageView != nullptr && res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) { if (res.imageView != nullptr && res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) {
m_descriptors[k].image.sampler = VK_NULL_HANDLE; descriptorInfo.image.sampler = VK_NULL_HANDLE;
m_descriptors[k].image.imageView = res.imageView->handle(binding.viewType); descriptorInfo.image.imageView = res.imageView->handle(binding.viewType);
m_descriptors[k].image.imageLayout = res.imageView->imageInfo().layout; descriptorInfo.image.imageLayout = res.imageView->imageInfo().layout;
if (m_rcTracked.set(binding.resourceBinding)) { if (m_rcTracked.set(binding.resourceBinding)) {
m_cmd->trackResource<DxvkAccess::None>(res.imageView); m_cmd->trackResource<DxvkAccess::None>(res.imageView);
m_cmd->trackResource<DxvkAccess::Read>(res.imageView->image()); m_cmd->trackResource<DxvkAccess::Read>(res.imageView->image());
} }
} else { } else {
m_descriptors[k].image.sampler = VK_NULL_HANDLE; descriptorInfo.image.sampler = VK_NULL_HANDLE;
m_descriptors[k].image.imageView = VK_NULL_HANDLE; descriptorInfo.image.imageView = VK_NULL_HANDLE;
m_descriptors[k].image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED; descriptorInfo.image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
} }
} break; } break;
@ -4739,18 +4737,18 @@ namespace dxvk {
const auto& res = m_rc[binding.resourceBinding]; const auto& res = m_rc[binding.resourceBinding];
if (res.imageView != nullptr && res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) { if (res.imageView != nullptr && res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) {
m_descriptors[k].image.sampler = VK_NULL_HANDLE; descriptorInfo.image.sampler = VK_NULL_HANDLE;
m_descriptors[k].image.imageView = res.imageView->handle(binding.viewType); descriptorInfo.image.imageView = res.imageView->handle(binding.viewType);
m_descriptors[k].image.imageLayout = res.imageView->imageInfo().layout; descriptorInfo.image.imageLayout = res.imageView->imageInfo().layout;
if (m_rcTracked.set(binding.resourceBinding)) { if (m_rcTracked.set(binding.resourceBinding)) {
m_cmd->trackResource<DxvkAccess::None>(res.imageView); m_cmd->trackResource<DxvkAccess::None>(res.imageView);
m_cmd->trackResource<DxvkAccess::Write>(res.imageView->image()); m_cmd->trackResource<DxvkAccess::Write>(res.imageView->image());
} }
} else { } else {
m_descriptors[k].image.sampler = VK_NULL_HANDLE; descriptorInfo.image.sampler = VK_NULL_HANDLE;
m_descriptors[k].image.imageView = VK_NULL_HANDLE; descriptorInfo.image.imageView = VK_NULL_HANDLE;
m_descriptors[k].image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED; descriptorInfo.image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
} }
} break; } break;
@ -4759,9 +4757,9 @@ namespace dxvk {
if (res.sampler != nullptr && res.imageView != nullptr if (res.sampler != nullptr && res.imageView != nullptr
&& res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) { && res.imageView->handle(binding.viewType) != VK_NULL_HANDLE) {
m_descriptors[k].image.sampler = res.sampler->handle(); descriptorInfo.image.sampler = res.sampler->handle();
m_descriptors[k].image.imageView = res.imageView->handle(binding.viewType); descriptorInfo.image.imageView = res.imageView->handle(binding.viewType);
m_descriptors[k].image.imageLayout = res.imageView->imageInfo().layout; descriptorInfo.image.imageLayout = res.imageView->imageInfo().layout;
if (m_rcTracked.set(binding.resourceBinding)) { if (m_rcTracked.set(binding.resourceBinding)) {
m_cmd->trackResource<DxvkAccess::None>(res.sampler); m_cmd->trackResource<DxvkAccess::None>(res.sampler);
@ -4769,9 +4767,9 @@ namespace dxvk {
m_cmd->trackResource<DxvkAccess::Read>(res.imageView->image()); m_cmd->trackResource<DxvkAccess::Read>(res.imageView->image());
} }
} else { } else {
m_descriptors[k].image.sampler = m_common->dummyResources().samplerHandle(); descriptorInfo.image.sampler = m_common->dummyResources().samplerHandle();
m_descriptors[k].image.imageView = VK_NULL_HANDLE; descriptorInfo.image.imageView = VK_NULL_HANDLE;
m_descriptors[k].image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED; descriptorInfo.image.imageLayout = VK_IMAGE_LAYOUT_UNDEFINED;
} }
} break; } break;
@ -4780,14 +4778,14 @@ namespace dxvk {
if (res.bufferView != nullptr) { if (res.bufferView != nullptr) {
res.bufferView->updateView(); res.bufferView->updateView();
m_descriptors[k].texelBuffer = res.bufferView->handle(); descriptorInfo.texelBuffer = res.bufferView->handle();
if (m_rcTracked.set(binding.resourceBinding)) { if (m_rcTracked.set(binding.resourceBinding)) {
m_cmd->trackResource<DxvkAccess::None>(res.bufferView); m_cmd->trackResource<DxvkAccess::None>(res.bufferView);
m_cmd->trackResource<DxvkAccess::Read>(res.bufferView->buffer()); m_cmd->trackResource<DxvkAccess::Read>(res.bufferView->buffer());
} }
} else { } else {
m_descriptors[k].texelBuffer = VK_NULL_HANDLE; descriptorInfo.texelBuffer = VK_NULL_HANDLE;
} }
} break; } break;
@ -4796,14 +4794,14 @@ namespace dxvk {
if (res.bufferView != nullptr) { if (res.bufferView != nullptr) {
res.bufferView->updateView(); res.bufferView->updateView();
m_descriptors[k].texelBuffer = res.bufferView->handle(); descriptorInfo.texelBuffer = res.bufferView->handle();
if (m_rcTracked.set(binding.resourceBinding)) { if (m_rcTracked.set(binding.resourceBinding)) {
m_cmd->trackResource<DxvkAccess::None>(res.bufferView); m_cmd->trackResource<DxvkAccess::None>(res.bufferView);
m_cmd->trackResource<DxvkAccess::Write>(res.bufferView->buffer()); m_cmd->trackResource<DxvkAccess::Write>(res.bufferView->buffer());
} }
} else { } else {
m_descriptors[k].texelBuffer = VK_NULL_HANDLE; descriptorInfo.texelBuffer = VK_NULL_HANDLE;
} }
} break; } break;
@ -4811,14 +4809,14 @@ namespace dxvk {
const auto& res = m_rc[binding.resourceBinding]; const auto& res = m_rc[binding.resourceBinding];
if (res.bufferSlice.defined()) { if (res.bufferSlice.defined()) {
m_descriptors[k] = res.bufferSlice.getDescriptor(); descriptorInfo = res.bufferSlice.getDescriptor();
if (m_rcTracked.set(binding.resourceBinding)) if (m_rcTracked.set(binding.resourceBinding))
m_cmd->trackResource<DxvkAccess::Read>(res.bufferSlice.buffer()); m_cmd->trackResource<DxvkAccess::Read>(res.bufferSlice.buffer());
} else { } else {
m_descriptors[k].buffer.buffer = VK_NULL_HANDLE; descriptorInfo.buffer.buffer = VK_NULL_HANDLE;
m_descriptors[k].buffer.offset = 0; descriptorInfo.buffer.offset = 0;
m_descriptors[k].buffer.range = VK_WHOLE_SIZE; descriptorInfo.buffer.range = VK_WHOLE_SIZE;
} }
} break; } break;
@ -4826,51 +4824,48 @@ namespace dxvk {
const auto& res = m_rc[binding.resourceBinding]; const auto& res = m_rc[binding.resourceBinding];
if (res.bufferSlice.defined()) { if (res.bufferSlice.defined()) {
m_descriptors[k] = res.bufferSlice.getDescriptor(); descriptorInfo = res.bufferSlice.getDescriptor();
if (m_rcTracked.set(binding.resourceBinding)) if (m_rcTracked.set(binding.resourceBinding))
m_cmd->trackResource<DxvkAccess::Write>(res.bufferSlice.buffer()); m_cmd->trackResource<DxvkAccess::Write>(res.bufferSlice.buffer());
} else { } else {
m_descriptors[k].buffer.buffer = VK_NULL_HANDLE; descriptorInfo.buffer.buffer = VK_NULL_HANDLE;
m_descriptors[k].buffer.offset = 0; descriptorInfo.buffer.offset = 0;
m_descriptors[k].buffer.range = VK_WHOLE_SIZE; descriptorInfo.buffer.range = VK_WHOLE_SIZE;
} }
} break; } break;
default: default:
break; break;
} }
k += 1;
} }
if (useDescriptorTemplates) { if (useDescriptorTemplates) {
m_cmd->updateDescriptorSetWithTemplate(set, m_cmd->updateDescriptorSetWithTemplate(set,
layout->getSetUpdateTemplate(setIndex), layout->getSetUpdateTemplate(setIndex),
&m_descriptors[k - bindingCount]); &m_descriptors[0]);
descriptorCount = 0;
} }
bindCount += 1;
// If the next set is not dirty, update and bind all previously // If the next set is not dirty, update and bind all previously
// updated sets in one go in order to reduce api call overhead. // updated sets in one go in order to reduce api call overhead.
if (!(dirtySetMask & (1u << (setIndex + 1)))) { if (!(((dirtySetMask >> 1) >> setIndex) & 1u)) {
if (!useDescriptorTemplates) { if (!useDescriptorTemplates) {
m_cmd->updateDescriptorSets(k, m_descriptorWrites.data()); m_cmd->updateDescriptorSets(descriptorCount,
k = 0; m_descriptorWrites.data());
descriptorCount = 0;
} }
uint32_t firstSet = setIndex + 1 - bindCount; // Find first dirty set in the mask and clear bits
// for all sets that we're going to update here.
uint32_t firstSet = bit::tzcnt(dirtySetMask);
dirtySetMask &= (~1u) << setIndex;
m_cmd->cmdBindDescriptorSets(BindPoint, m_cmd->cmdBindDescriptorSets(BindPoint,
layout->getPipelineLayout(independentSets), layout->getPipelineLayout(independentSets),
firstSet, bindCount, &sets[firstSet], firstSet, setIndex - firstSet + 1, &sets[firstSet],
0, nullptr); 0, nullptr);
bindCount = 0;
} }
dirtySetMask &= dirtySetMask - 1;
} }
} }