blob: f00fa5e01d7acdb9d77f51b39bfa2c8b6fad24ff [file] [log] [blame]
/* Copyright (c) 2024 The Khronos Group Inc.
* Copyright (c) 2024 Valve Corporation
* Copyright (c) 2024 LunarG, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "utils/vk_struct_compare.h"
#include "utils/vk_layer_utils.h"
#include <vulkan/utility/vk_struct_helper.hpp>
static inline bool ComparePipelineSampleLocationsStateCreateInfo(const VkPipelineSampleLocationsStateCreateInfoEXT &a,
const VkPipelineSampleLocationsStateCreateInfoEXT &b) {
// Having VkSampleLocationEXT not confirmed to matter for the VU this is being used for, so just check the Count is good enough
return (a.sampleLocationsEnable == b.sampleLocationsEnable) &&
(a.sampleLocationsInfo.sampleLocationsPerPixel == b.sampleLocationsInfo.sampleLocationsPerPixel) &&
(a.sampleLocationsInfo.sampleLocationGridSize.height == b.sampleLocationsInfo.sampleLocationGridSize.height) &&
(a.sampleLocationsInfo.sampleLocationGridSize.width == b.sampleLocationsInfo.sampleLocationGridSize.width) &&
(a.sampleLocationsInfo.sampleLocationsCount == b.sampleLocationsInfo.sampleLocationsCount);
}
bool ComparePipelineMultisampleStateCreateInfo(const VkPipelineMultisampleStateCreateInfo &a,
const VkPipelineMultisampleStateCreateInfo &b) {
bool valid_mask = true;
if (a.pSampleMask && b.pSampleMask && (a.rasterizationSamples == b.rasterizationSamples)) {
uint32_t length = (SampleCountSize(a.rasterizationSamples) + 31) / 32;
for (uint32_t i = 0; i < length; i++) {
if (a.pSampleMask[i] != b.pSampleMask[i]) {
valid_mask = false;
break;
}
}
} else if (a.pSampleMask || b.pSampleMask) {
valid_mask = false; // one is not null
}
bool valid_pNext = true;
if (a.pNext && b.pNext) {
auto *a_sample_location = vku::FindStructInPNextChain<VkPipelineSampleLocationsStateCreateInfoEXT>(a.pNext);
auto *b_sample_location = vku::FindStructInPNextChain<VkPipelineSampleLocationsStateCreateInfoEXT>(b.pNext);
if (a_sample_location && b_sample_location) {
if (!ComparePipelineSampleLocationsStateCreateInfo(*a_sample_location, *b_sample_location)) {
valid_pNext = false;
}
} else if (a_sample_location != b_sample_location) {
valid_pNext = false; // both are not null
}
} else if (a.pNext != b.pNext) {
valid_pNext = false; // both are not null
}
return (a.sType == b.sType) && (valid_pNext) && (a.flags == b.flags) && (a.rasterizationSamples == b.rasterizationSamples) &&
(a.sampleShadingEnable == b.sampleShadingEnable) && (a.minSampleShading == b.minSampleShading) && (valid_mask) &&
(a.alphaToCoverageEnable == b.alphaToCoverageEnable) && (a.alphaToOneEnable == b.alphaToOneEnable);
}
bool CompareDescriptorSetLayoutBinding(const VkDescriptorSetLayoutBinding &a, const VkDescriptorSetLayoutBinding &b) {
return (a.binding == b.binding) && (a.descriptorType == b.descriptorType) && (a.descriptorCount == b.descriptorCount) &&
(a.stageFlags == b.stageFlags) && (a.pImmutableSamplers == b.pImmutableSamplers);
}
bool ComparePipelineColorBlendAttachmentState(const VkPipelineColorBlendAttachmentState &a,
const VkPipelineColorBlendAttachmentState &b) {
return (a.blendEnable == b.blendEnable) && (a.srcColorBlendFactor == b.srcColorBlendFactor) &&
(a.dstColorBlendFactor == b.dstColorBlendFactor) && (a.colorBlendOp == b.colorBlendOp) &&
(a.srcAlphaBlendFactor == b.srcAlphaBlendFactor) && (a.dstAlphaBlendFactor == b.dstAlphaBlendFactor) &&
(a.alphaBlendOp == b.alphaBlendOp) && (a.colorWriteMask == b.colorWriteMask);
}
bool ComparePipelineFragmentShadingRateStateCreateInfo(const VkPipelineFragmentShadingRateStateCreateInfoKHR &a,
const VkPipelineFragmentShadingRateStateCreateInfoKHR &b) {
// Since this is chained in a pnext, we don't want to check the pNext/sType
return (a.fragmentSize.width == b.fragmentSize.width) && (a.fragmentSize.height == b.fragmentSize.height) &&
(a.combinerOps[0] == b.combinerOps[0]) && (a.combinerOps[1] == b.combinerOps[1]);
}
static inline bool CompareSamplerYcbcrConversionInfo(const VkSamplerYcbcrConversionInfo &a, const VkSamplerYcbcrConversionInfo &b) {
return a.conversion == b.conversion;
}
static inline bool CompareSamplerReductionModeCreateInfo(const VkSamplerReductionModeCreateInfo &a,
const VkSamplerReductionModeCreateInfo &b) {
return a.reductionMode == b.reductionMode;
}
static inline bool CompareSamplerBorderColorComponentMappingCreateInfo(const VkSamplerBorderColorComponentMappingCreateInfoEXT &a,
const VkSamplerBorderColorComponentMappingCreateInfoEXT &b) {
return (a.components.r == b.components.r) && (a.components.g == b.components.g) && (a.components.b == b.components.b) &&
(a.components.a == b.components.a) && (a.srgb == b.srgb);
}
static inline bool CompareSamplerCustomBorderColorCreateInfo(const VkSamplerCustomBorderColorCreateInfoEXT &a,
const VkSamplerCustomBorderColorCreateInfoEXT &b) {
return (a.format == b.format);
}
bool CompareSamplerCreateInfo(const VkSamplerCreateInfo &a, const VkSamplerCreateInfo &b) {
if (a.pNext && b.pNext) {
auto *a_ycbcr_conversion = vku::FindStructInPNextChain<VkSamplerYcbcrConversionInfo>(a.pNext);
auto *b_ycbcr_conversion = vku::FindStructInPNextChain<VkSamplerYcbcrConversionInfo>(b.pNext);
if (a_ycbcr_conversion && b_ycbcr_conversion) {
if (!CompareSamplerYcbcrConversionInfo(*a_ycbcr_conversion, *b_ycbcr_conversion)) {
return false;
}
} else if (a_ycbcr_conversion != b_ycbcr_conversion) {
return false; // both are not null
}
auto *a_reduction_mode = vku::FindStructInPNextChain<VkSamplerReductionModeCreateInfo>(a.pNext);
auto *b_reduction_mode = vku::FindStructInPNextChain<VkSamplerReductionModeCreateInfo>(b.pNext);
if (a_reduction_mode && b_reduction_mode) {
if (!CompareSamplerReductionModeCreateInfo(*a_reduction_mode, *b_reduction_mode)) {
return false;
}
} else if (a_reduction_mode != b_reduction_mode) {
return false; // both are not null
}
auto *a_component_mapping = vku::FindStructInPNextChain<VkSamplerBorderColorComponentMappingCreateInfoEXT>(a.pNext);
auto *b_component_mapping = vku::FindStructInPNextChain<VkSamplerBorderColorComponentMappingCreateInfoEXT>(b.pNext);
if (a_component_mapping && b_component_mapping) {
if (!CompareSamplerBorderColorComponentMappingCreateInfo(*a_component_mapping, *b_component_mapping)) {
return false;
}
} else if (a_component_mapping != b_component_mapping) {
return false; // both are not null
}
auto *a_border_color = vku::FindStructInPNextChain<VkSamplerCustomBorderColorCreateInfoEXT>(a.pNext);
auto *b_border_color = vku::FindStructInPNextChain<VkSamplerCustomBorderColorCreateInfoEXT>(b.pNext);
if (a_border_color && b_border_color) {
if (!CompareSamplerCustomBorderColorCreateInfo(*a_border_color, *b_border_color)) {
return false;
}
} else if (a_border_color != b_border_color) {
return false; // both are not null
}
} else if (a.pNext != b.pNext) {
return false; // both are not null
}
return (a.flags == b.flags) && (a.magFilter == b.magFilter) && (a.minFilter == b.minFilter) && (a.mipmapMode == b.mipmapMode) &&
(a.addressModeU == b.addressModeU) && (a.addressModeV == b.addressModeV) && (a.addressModeW == b.addressModeW) &&
(a.mipLodBias == b.mipLodBias) && (a.anisotropyEnable == b.anisotropyEnable) && (a.maxAnisotropy == b.maxAnisotropy) &&
(a.compareEnable == b.compareEnable) && (a.compareOp == b.compareOp) && (a.minLod == b.minLod) &&
(a.maxLod == b.maxLod) && (a.borderColor == b.borderColor) && (a.unnormalizedCoordinates == b.unnormalizedCoordinates);
}