Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Vulkan dump resources: Fix in vertex buffer size calculation #1915

Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
44 changes: 33 additions & 11 deletions framework/decode/vulkan_replay_dump_resources_common.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -304,10 +304,11 @@ uint32_t VkIndexTypeToBytes(VkIndexType type)
}
}

uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,
uint32_t index_count,
uint32_t first_index,
VkIndexType type)
std::pair<uint32_t, uint32_t> FindMinMaxVertexIndices(const std::vector<uint8_t>& index_data,
uint32_t index_count,
uint32_t first_index,
int32_t vertex_offset,
VkIndexType type)
{
switch (type)
{
Expand All @@ -323,9 +324,10 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,

if (i == index_count)
{
return 0;
return std::make_pair(0, 0);
}

uint8_t min = indices[first_index + i];
uint8_t max = indices[first_index + i];

for (; i < index_count; ++i)
Expand All @@ -339,9 +341,15 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,
{
max = indices[first_index + i];
}

if (indices[first_index + i] < min)
{
min = indices[first_index + i];
}
}

return max;
return std::make_pair(static_cast<uint32_t>(min) + vertex_offset,
static_cast<uint32_t>(max) + vertex_offset);
}
break;

Expand All @@ -357,9 +365,10 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,

if (i == index_count)
{
return 0;
return std::make_pair(0, 0);
}

uint16_t min = indices[first_index + i];
uint16_t max = indices[first_index + i];

for (; i < index_count; ++i)
Expand All @@ -373,9 +382,15 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,
{
max = indices[first_index + i];
}

if (indices[first_index + i] < min)
{
min = indices[first_index + i];
}
}

return max;
return std::make_pair(static_cast<uint32_t>(min) + vertex_offset,
static_cast<uint32_t>(max) + vertex_offset);
}
break;

Expand All @@ -391,9 +406,10 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,

if (i == index_count)
{
return 0;
return std::make_pair(0, 0);
}

uint32_t min = indices[first_index + i];
uint32_t max = indices[first_index + i];

for (; i < index_count; ++i)
Expand All @@ -407,17 +423,23 @@ uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,
{
max = indices[first_index + i];
}

if (indices[first_index + i] < min)
{
min = indices[first_index + i];
}
}

return max;
return std::make_pair(min + vertex_offset, max + vertex_offset);
}
break;

case VK_INDEX_TYPE_NONE_KHR:
default:
GFXRECON_LOG_ERROR("%s() Unrecognized/unhandled index type (%u)", __func__, static_cast<uint32_t>(type));
assert(0);
return 0;

return std::make_pair(0, 0);
break;
}
}
Expand Down
9 changes: 5 additions & 4 deletions framework/decode/vulkan_replay_dump_resources_common.h
Original file line number Diff line number Diff line change
Expand Up @@ -98,10 +98,11 @@ VkResult CloneBuffer(CommonObjectInfoTable& object_info_table,

uint32_t VkIndexTypeToBytes(VkIndexType type);

uint32_t FindGreatestVertexIndex(const std::vector<uint8_t>& index_data,
uint32_t index_count,
uint32_t first_index,
VkIndexType type);
std::pair<uint32_t, uint32_t> FindMinMaxVertexIndices(const std::vector<uint8_t>& index_data,
uint32_t index_count,
uint32_t first_index,
int32_t vertex_offset,
VkIndexType type);

VkResult DumpImageToFile(const VulkanImageInfo* image_info,
const VulkanDeviceInfo* device_info,
Expand Down
53 changes: 39 additions & 14 deletions framework/decode/vulkan_replay_dump_resources_draw_calls.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2360,17 +2360,18 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
assert(dc_params_entry != draw_call_params.end());
DrawCallParameters& dc_params = dc_params_entry->second;

uint32_t greatest_vertex_index = 0;
std::pair<uint32_t, uint32_t> min_max_vertex_indices(std::numeric_limits<uint32_t>::max(), 0);

// Dump index buffer
if (IsDrawCallIndexed(dc_params.type) && dc_params.referenced_index_buffer.buffer_info != nullptr)
{
// Store all (indexCount, firstIndex) pairs used by all associated with this index buffer.
// Latter we will parse the index buffer using all these pairs in order to detect the
// greatest index.
// Store all (indexCount, firstIndex) pairs used by all draw calls (in case of indirect)
// associated with this index buffer. Then we will parse the index buffer using all these pairs in order to
// detect the greatest index.
std::vector<std::pair<uint32_t, uint32_t>> index_count_first_index_pairs;

uint32_t abs_index_count = 0;
uint32_t abs_index_count = 0;
int32_t greatest_vertex_offset = 0;

if (IsDrawCallIndirect(dc_params.type))
{
Expand All @@ -2394,6 +2395,11 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
{
abs_index_count = indirect_index_count + indirect_first_index;
}

if (greatest_vertex_offset < ic_params.draw_indexed_params[d].vertexOffset)
{
greatest_vertex_offset = ic_params.draw_indexed_params[d].vertexOffset;
}
}
}
}
Expand All @@ -2417,6 +2423,11 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
{
abs_index_count = indirect_index_count + indirect_first_index;
}

if (greatest_vertex_offset < i_params.draw_indexed_params[d].vertexOffset)
{
greatest_vertex_offset = i_params.draw_indexed_params[d].vertexOffset;
}
}
}
}
Expand All @@ -2427,7 +2438,8 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
const uint32_t first_index = dc_params.dc_params_union.draw_indexed.firstIndex;

index_count_first_index_pairs.emplace_back(std::make_pair(index_count, first_index));
abs_index_count = index_count + first_index;
abs_index_count = index_count + first_index;
greatest_vertex_offset = dc_params.dc_params_union.draw_indexed.vertexOffset;
}

if (abs_index_count)
Expand Down Expand Up @@ -2468,12 +2480,19 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
std::string filename = GenerateIndexBufferFilename(qs_index, bcb_index, dc_index, index_type);
util::bufferwriter::WriteBuffer(filename, index_data.data(), index_data.size());

// Parse all indices in order to find the smallest and greatest index
for (const auto& pairs : index_count_first_index_pairs)
{
const uint32_t gvi = FindGreatestVertexIndex(index_data, pairs.first, pairs.second, index_type);
if (greatest_vertex_index < gvi)
const std::pair<uint32_t, uint32_t> min_max_indices =
FindMinMaxVertexIndices(index_data, pairs.first, pairs.second, greatest_vertex_offset, index_type);
if (min_max_indices.first < min_max_vertex_indices.first)
{
greatest_vertex_index = gvi;
min_max_vertex_indices.first = min_max_indices.first;
}

if (min_max_indices.second > min_max_vertex_indices.second)
{
min_max_vertex_indices.second = min_max_indices.second;
}
}
}
Expand All @@ -2488,7 +2507,8 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
if (IsDrawCallIndexed(dc_params.type))
{
// For indexed draw calls the greatest vertex index will be used as the max vertex count
vertex_count = greatest_vertex_index + 1;
GFXRECON_ASSERT(min_max_vertex_indices.second >= min_max_vertex_indices.first);
vertex_count = (min_max_vertex_indices.second - min_max_vertex_indices.first) + 1;

if (IsDrawCallIndirect(dc_params.type))
{
Expand Down Expand Up @@ -2615,16 +2635,18 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint

const uint32_t count =
vis.second.inputRate == VK_VERTEX_INPUT_RATE_VERTEX ? vertex_count : instance_count;
const uint32_t offset = vb_entry->second.offset;
uint32_t total_size = 0;
uint32_t total_size = 0;
uint32_t binding_stride;

if (vb_entry->second.size)
{
// Exact size was provided by vkCmdBindVertexBuffers2
total_size = vb_entry->second.size;
total_size = vb_entry->second.size;
binding_stride = vb_entry->second.stride;
}
else
{
const uint32_t binding_stride = vis.second.stride;
binding_stride = vis.second.stride;
if (binding_stride)
{
total_size = count * binding_stride;
Expand Down Expand Up @@ -2660,6 +2682,9 @@ VkResult DrawCallsDumpingContext::DumpVertexIndexBuffers(uint64_t qs_index, uint
}
}

// Calculate offset including vertexOffset
const uint32_t offset = vb_entry->second.offset + (min_max_vertex_indices.first * binding_stride);

assert(total_size <= vb_entry->second.buffer_info->size - offset);
// There is something wrong with the calculations if this is true
if (total_size > vb_entry->second.buffer_info->size - offset)
Expand Down
Loading