mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-11-26 21:30:07 +00:00
Opt: Make DecorationManager::HaveTheSameDecorations symmetric
Fixes https://github.com/KhronosGroup/SPIRV-Tools/issues/1112 Also: Add SpvOpDecorateId to IsAnnotationInst()
This commit is contained in:
parent
a376b197ae
commit
702852bd22
@ -15,7 +15,7 @@
|
||||
#include "decoration_manager.h"
|
||||
|
||||
#include <algorithm>
|
||||
#include <iostream>
|
||||
#include <set>
|
||||
#include <stack>
|
||||
|
||||
namespace spvtools {
|
||||
@ -39,25 +39,59 @@ std::vector<const ir::Instruction*> DecorationManager::GetDecorationsFor(
|
||||
->InternalGetDecorationsFor<const ir::Instruction*>(id, include_linkage);
|
||||
}
|
||||
|
||||
// TODO(pierremoreau): The code will return true for { deco1, deco1 }, { deco1,
|
||||
// deco2 } when it should return false.
|
||||
bool DecorationManager::HaveTheSameDecorations(uint32_t id1,
|
||||
uint32_t id2) const {
|
||||
const auto decorationsFor1 = GetDecorationsFor(id1, false);
|
||||
const auto decorationsFor2 = GetDecorationsFor(id2, false);
|
||||
if (decorationsFor1.size() != decorationsFor2.size()) return false;
|
||||
using InstructionList = std::vector<const ir::Instruction*>;
|
||||
using DecorationSet = std::set<std::u32string>;
|
||||
|
||||
for (const ir::Instruction* inst1 : decorationsFor1) {
|
||||
bool didFindAMatch = false;
|
||||
for (const ir::Instruction* inst2 : decorationsFor2) {
|
||||
if (AreDecorationsTheSame(inst1, inst2, true)) {
|
||||
didFindAMatch = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!didFindAMatch) return false;
|
||||
}
|
||||
return true;
|
||||
const InstructionList decorationsFor1 = GetDecorationsFor(id1, false);
|
||||
const InstructionList decorationsFor2 = GetDecorationsFor(id2, false);
|
||||
|
||||
// This function splits the decoration instructions into different sets,
|
||||
// based on their opcode; only OpDecorate, OpDecorateId and OpMemberDecorate
|
||||
// are considered, the other opcodes are ignored.
|
||||
const auto fillDecorationSets =
|
||||
[](const InstructionList& decorationList, DecorationSet* decorateSet,
|
||||
DecorationSet* decorateIdSet, DecorationSet* memberDecorateSet) {
|
||||
for (const ir::Instruction* inst : decorationList) {
|
||||
std::u32string decorationPayload;
|
||||
// Ignore the opcode and the target as we do not want them to be
|
||||
// compared.
|
||||
for (uint32_t i = 1u; i < inst->NumInOperands(); ++i)
|
||||
for (uint32_t word : inst->GetInOperand(i).words)
|
||||
decorationPayload.push_back(word);
|
||||
|
||||
switch (inst->opcode()) {
|
||||
case SpvOpDecorate:
|
||||
decorateSet->emplace(std::move(decorationPayload));
|
||||
break;
|
||||
case SpvOpMemberDecorate:
|
||||
memberDecorateSet->emplace(std::move(decorationPayload));
|
||||
break;
|
||||
case SpvOpDecorateId:
|
||||
decorateIdSet->emplace(std::move(decorationPayload));
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
DecorationSet decorateSetFor1;
|
||||
DecorationSet decorateIdSetFor1;
|
||||
DecorationSet memberDecorateSetFor1;
|
||||
fillDecorationSets(decorationsFor1, &decorateSetFor1, &decorateIdSetFor1,
|
||||
&memberDecorateSetFor1);
|
||||
|
||||
DecorationSet decorateSetFor2;
|
||||
DecorationSet decorateIdSetFor2;
|
||||
DecorationSet memberDecorateSetFor2;
|
||||
fillDecorationSets(decorationsFor2, &decorateSetFor2, &decorateIdSetFor2,
|
||||
&memberDecorateSetFor2);
|
||||
|
||||
return decorateSetFor1 == decorateSetFor2 &&
|
||||
decorateIdSetFor1 == decorateIdSetFor2 &&
|
||||
memberDecorateSetFor1 == memberDecorateSetFor2;
|
||||
}
|
||||
|
||||
// TODO(pierremoreau): If OpDecorateId is referencing an OpConstant, one could
|
||||
|
@ -38,7 +38,8 @@ inline bool IsDebugLineInst(SpvOp opcode) {
|
||||
return opcode == SpvOpLine || opcode == SpvOpNoLine;
|
||||
}
|
||||
inline bool IsAnnotationInst(SpvOp opcode) {
|
||||
return opcode >= SpvOpDecorate && opcode <= SpvOpGroupMemberDecorate;
|
||||
return (opcode >= SpvOpDecorate && opcode <= SpvOpGroupMemberDecorate) ||
|
||||
opcode == SpvOpDecorateId;
|
||||
}
|
||||
inline bool IsTypeInst(SpvOp opcode) {
|
||||
return (opcode >= SpvOpTypeVoid && opcode <= SpvOpTypeForwardPointer) ||
|
||||
|
@ -301,4 +301,160 @@ OpGroupDecorate %4 %1 %2
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsDuplicateDecorations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorate %1 Constant
|
||||
OpDecorate %2 Constant
|
||||
OpDecorate %2 Constant
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%2 = OpVariable %u32 Uniform
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_TRUE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsDifferentVariations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorate %1 Location 0
|
||||
OpDecorate %2 Location 1
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%2 = OpVariable %u32 Uniform
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest,
|
||||
HaveTheSameDecorationsDuplicateMemberDecorations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpMemberDecorate %1 0 Location 0
|
||||
OpMemberDecorate %2 0 Location 0
|
||||
OpMemberDecorate %2 0 Location 0
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpTypeStruct %u32 %u32
|
||||
%2 = OpTypeStruct %u32 %u32
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_TRUE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest,
|
||||
HaveTheSameDecorationsDifferentMemberSameDecoration) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpMemberDecorate %1 0 Location 0
|
||||
OpMemberDecorate %2 1 Location 0
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpTypeStruct %u32 %u32
|
||||
%2 = OpTypeStruct %u32 %u32
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsDifferentMemberVariations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpMemberDecorate %1 0 Location 0
|
||||
OpMemberDecorate %2 0 Location 1
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpTypeStruct %u32 %u32
|
||||
%2 = OpTypeStruct %u32 %u32
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsDuplicateIdDecorations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorateId %1 AlignmentId %2
|
||||
OpDecorateId %3 AlignmentId %2
|
||||
OpDecorateId %3 AlignmentId %2
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%3 = OpVariable %u32 Uniform
|
||||
%2 = OpSpecConstant %u32 0
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_TRUE(decoManager->HaveTheSameDecorations(1u, 3u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsDifferentIdVariations) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorateId %1 AlignmentId %2
|
||||
OpDecorateId %3 AlignmentId %4
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%3 = OpVariable %u32 Uniform
|
||||
%2 = OpSpecConstant %u32 0
|
||||
%4 = OpSpecConstant %u32 0
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsLeftSymmetry) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorate %1 Constant
|
||||
OpDecorate %1 Constant
|
||||
OpDecorate %2 Constant
|
||||
OpDecorate %2 Restrict
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%2 = OpVariable %u32 Uniform
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
TEST_F(DecorationManagerTest, HaveTheSameDecorationsRightSymmetry) {
|
||||
const std::string spirv = R"(
|
||||
OpCapability Shader
|
||||
OpCapability Linkage
|
||||
OpMemoryModel Logical GLSL450
|
||||
OpDecorate %1 Constant
|
||||
OpDecorate %1 Restrict
|
||||
OpDecorate %2 Constant
|
||||
OpDecorate %2 Constant
|
||||
%u32 = OpTypeInt 32 0
|
||||
%1 = OpVariable %u32 Uniform
|
||||
%2 = OpVariable %u32 Uniform
|
||||
)";
|
||||
DecorationManager* decoManager = GetDecorationManager(spirv);
|
||||
EXPECT_THAT(GetErrorMessage(), "");
|
||||
EXPECT_FALSE(decoManager->HaveTheSameDecorations(1u, 2u));
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
Loading…
Reference in New Issue
Block a user