mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-06 01:50:38 +00:00
138 lines
6.8 KiB
Python
138 lines
6.8 KiB
Python
import rdtest
|
|
import struct
|
|
import renderdoc as rd
|
|
|
|
|
|
class VK_Descriptor_Indexing(rdtest.TestCase):
|
|
demos_test_name = 'VK_Descriptor_Indexing'
|
|
|
|
def check_capture(self):
|
|
|
|
action = self.find_action("Dispatch")
|
|
self.check(action is not None)
|
|
self.controller.SetFrameEvent(action.eventId, False)
|
|
|
|
pipe = self.controller.GetPipelineState()
|
|
vkpipe: rd.VKState = self.controller.GetVulkanPipelineState()
|
|
|
|
if len(vkpipe.compute.descriptorSets) != 1:
|
|
raise rdtest.TestFailureException("Wrong number of compute sets is bound: {}, not 1"
|
|
.format(len(vkpipe.compute.descriptorSets)))
|
|
|
|
rw = pipe.GetReadWriteResources(rd.ShaderStage.Compute)
|
|
|
|
self.check_eq(len(rw), 1)
|
|
self.check_eq(rw[0].access.index, 0)
|
|
self.check_eq(rw[0].access.stage, rd.ShaderStage.Compute)
|
|
self.check_eq(rw[0].access.type, rd.DescriptorType.ReadWriteBuffer)
|
|
self.check_eq(rw[0].access.staticallyUnused, False)
|
|
self.check_eq(rw[0].access.arrayElement, 15)
|
|
self.check_eq(rw[0].access.descriptorStore,
|
|
vkpipe.compute.descriptorSets[0].descriptorSetResourceId)
|
|
|
|
self.check_eq(
|
|
len(pipe.GetReadOnlyResources(rd.ShaderStage.Compute)), 0)
|
|
|
|
rw_used = pipe.GetReadWriteResources(rd.ShaderStage.Compute, True)
|
|
|
|
# should get the same results for dynamic array indexing, the 'only used' is only for
|
|
# statically unused or used bindings
|
|
self.check(rw == rw_used)
|
|
|
|
action = self.find_action("Draw")
|
|
self.check(action is not None)
|
|
self.controller.SetFrameEvent(action.eventId, False)
|
|
|
|
pipe = self.controller.GetPipelineState()
|
|
vkpipe = self.controller.GetVulkanPipelineState()
|
|
|
|
# Check bindings:
|
|
# - buffer 15 in bind 0 (the SSBO) should be use
|
|
# - buffer 6 in bind 1 (the first aliased SSBO) should be used
|
|
# - buffer 12 in bind 2 (the first aliased SSBO) should be used
|
|
# - images 19, 20, 21 in bind 0 should be used for the non-uniform index
|
|
# images 49 & 59 in bind 0 should be used for the first fixed index
|
|
# image 4 in bind 0 should be used for the global access from a function with no dynamic/patched parameters
|
|
# - images 381 & 386 in bind 1 should be used for the second fixed index
|
|
# - image 1 in bind 2 should be used
|
|
bind_info = {
|
|
(rd.DescriptorType.ReadWriteBuffer, 0): {'loc': (0, 0), 'elems': [15]},
|
|
(rd.DescriptorType.ReadWriteBuffer, 1): {'loc': (0, 3), 'elems': [6]},
|
|
(rd.DescriptorType.ReadWriteBuffer, 2): {'loc': (0, 3), 'elems': [12]},
|
|
(rd.DescriptorType.ImageSampler, 0): {'loc': (0, 1), 'elems': [4, 19, 20, 21, 49, 59]},
|
|
(rd.DescriptorType.ImageSampler, 1): {'loc': (0, 2), 'elems': [381, 386]},
|
|
(rd.DescriptorType.ImageSampler, 2): {'loc': (0, 4), 'elems': [1]},
|
|
}
|
|
|
|
if len(vkpipe.graphics.descriptorSets) != 1:
|
|
raise rdtest.TestFailureException("Wrong number of sets is bound: {}, not 1"
|
|
.format(len(vkpipe.graphics.descriptorSets)))
|
|
|
|
desc_set = vkpipe.graphics.descriptorSets[0]
|
|
|
|
rw = pipe.GetReadWriteResources(rd.ShaderStage.Fragment)
|
|
ro = pipe.GetReadOnlyResources(rd.ShaderStage.Fragment)
|
|
|
|
self.check_eq(len(ro), 1+6+2)
|
|
self.check_eq(len(rw), 1+1+1)
|
|
|
|
refl = pipe.GetShaderReflection(rd.ShaderStage.Fragment)
|
|
|
|
for a in ro + rw:
|
|
idx = (a.access.type, a.access.index)
|
|
if idx not in bind_info.keys():
|
|
raise rdtest.TestFailureException(
|
|
"Accessed bind {} of type {} doesn't exist in expected list".format(a.access.index, str(a.access.type)))
|
|
|
|
if rd.IsReadOnlyDescriptor(a.access.type):
|
|
res = refl.readOnlyResources[a.access.index]
|
|
else:
|
|
res = refl.readWriteResources[a.access.index]
|
|
|
|
if a.access.arrayElement not in bind_info[idx]['elems']:
|
|
raise rdtest.TestFailureException("Bind {} reports array element {} as used, which shouldn't be"
|
|
.format(res.name, a.access.arrayElement))
|
|
|
|
if a.access.descriptorStore != desc_set.descriptorSetResourceId:
|
|
raise rdtest.TestFailureException("Access is in descriptor store {} but expected set 0 {}"
|
|
.format(a.access.descriptorStore, desc_set))
|
|
|
|
if (res.fixedBindSetOrSpace, res.fixedBindNumber) != bind_info[idx]['loc']:
|
|
raise rdtest.TestFailureException("Bind {} expected to be {} but is {}, {}"
|
|
.format(res.name, bind_info[idx]['loc']), res.fixedBindSetOrSpace, res.fixedBindNumber)
|
|
|
|
# On vulkan the logical bind name is set-relative bind[idx]. The fixed bind number is the bind only
|
|
loc = self.controller.GetDescriptorLocations(
|
|
a.access.descriptorStore, [rd.DescriptorRange(a.access)])[0]
|
|
if loc.fixedBindNumber != bind_info[idx]['loc'][1]:
|
|
raise rdtest.TestFailureException("Bind {} not expected for set,bind {}"
|
|
.format(loc.fixedBindNumber, bind_info[idx]['loc']))
|
|
if loc.logicalBindName != "{}[{}]".format(bind_info[idx]['loc'][1], a.access.arrayElement):
|
|
raise rdtest.TestFailureException("Bind {} not expected for set,bind {} array element {}"
|
|
.format(loc.logicalBindName, bind_info[idx]['loc'], a.access.arrayElement))
|
|
|
|
bind_info[idx]['elems'].remove(a.access.arrayElement)
|
|
|
|
for a in rw:
|
|
res = refl.readWriteResources[a.access.index]
|
|
data = self.controller.GetBufferData(a.descriptor.resource, a.descriptor.byteOffset, a.descriptor.byteSize)
|
|
|
|
offset = 0
|
|
if res.name == "aliasbuf1":
|
|
self.check_eq(res.variableType.members[0].name, "Color")
|
|
self.check_eq(res.variableType.members[0].byteOffset, 0)
|
|
offset = res.variableType.members[0].byteOffset
|
|
elif res.name == "aliasbuf2":
|
|
self.check_eq(res.variableType.members[2].name, "Color")
|
|
self.check_eq(res.variableType.members[2].byteOffset, 32)
|
|
offset = res.variableType.members[2].byteOffset
|
|
else:
|
|
continue
|
|
|
|
col = struct.unpack_from("4f", data, offset)
|
|
if col[0] == 0.0 or col[1] == 0.0 or col[2] == 0.0 or col[3] == 0.0:
|
|
raise rdtest.TestFailureException(
|
|
"Color from aliased buffer {} doesn't match expected value: {}".format(res.name, col))
|
|
|
|
rdtest.log.success("Dynamic usage is as expected")
|