mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-14 10:20:19 +00:00
b4c4da3e76
* No longer blindly add global non-semantic info instructions to global types and values * functions now have a list of non-semantic instructions that succeed them in the global scope * global non-semantic instructions go in global types and values if they appear before any function, otherwise they are attached to the immediate function predecessor in the module * changed ADCE to use the function removal utility * Modified EliminateFunction to have special handling for non-semantic instructions in the global scope * non-semantic instructions are moved to an earlier function (or full global set) if the function they are attached to is eliminated * Added IRContext::KillNonSemanticInfo to remove the tree of non-semantic instructions that use an instruction * this is used in function elimination * There is still significant work in the optimizer to handle non-semantic instructions fully in the optimizer
342 lines
9.1 KiB
C++
342 lines
9.1 KiB
C++
// Copyright (c) 2016 Google 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 <memory>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
#include "source/opt/build_module.h"
|
|
#include "source/opt/module.h"
|
|
#include "source/opt/pass.h"
|
|
#include "spirv-tools/libspirv.hpp"
|
|
#include "test/opt/module_utils.h"
|
|
|
|
namespace spvtools {
|
|
namespace opt {
|
|
namespace {
|
|
|
|
using ::testing::Eq;
|
|
using spvtest::GetIdBound;
|
|
|
|
TEST(ModuleTest, SetIdBound) {
|
|
Module m;
|
|
// It's initialized to 0.
|
|
EXPECT_EQ(0u, GetIdBound(m));
|
|
|
|
m.SetIdBound(19);
|
|
EXPECT_EQ(19u, GetIdBound(m));
|
|
|
|
m.SetIdBound(102);
|
|
EXPECT_EQ(102u, GetIdBound(m));
|
|
}
|
|
|
|
// Returns an IRContext owning the module formed by assembling the given text,
|
|
// then loading the result.
|
|
inline std::unique_ptr<IRContext> BuildModule(std::string text) {
|
|
return spvtools::BuildModule(SPV_ENV_UNIVERSAL_1_1, nullptr, text,
|
|
SPV_TEXT_TO_BINARY_OPTION_PRESERVE_NUMERIC_IDS);
|
|
}
|
|
|
|
TEST(ModuleTest, ComputeIdBound) {
|
|
// Emtpy module case.
|
|
EXPECT_EQ(1u, BuildModule("")->module()->ComputeIdBound());
|
|
// Sensitive to result id
|
|
EXPECT_EQ(2u, BuildModule("%void = OpTypeVoid")->module()->ComputeIdBound());
|
|
// Sensitive to type id
|
|
EXPECT_EQ(1000u,
|
|
BuildModule("%a = OpTypeArray !999 3")->module()->ComputeIdBound());
|
|
// Sensitive to a regular Id parameter
|
|
EXPECT_EQ(2000u,
|
|
BuildModule("OpDecorate !1999 0")->module()->ComputeIdBound());
|
|
// Sensitive to a scope Id parameter.
|
|
EXPECT_EQ(3000u,
|
|
BuildModule("%f = OpFunction %void None %fntype %a = OpLabel "
|
|
"OpMemoryBarrier !2999 %b\n")
|
|
->module()
|
|
->ComputeIdBound());
|
|
// Sensitive to a semantics Id parameter
|
|
EXPECT_EQ(4000u,
|
|
BuildModule("%f = OpFunction %void None %fntype %a = OpLabel "
|
|
"OpMemoryBarrier %b !3999\n")
|
|
->module()
|
|
->ComputeIdBound());
|
|
}
|
|
|
|
TEST(ModuleTest, OstreamOperator) {
|
|
const std::string text = R"(OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
OpName %7 "restrict"
|
|
OpDecorate %8 Restrict
|
|
%9 = OpTypeVoid
|
|
%10 = OpTypeInt 32 0
|
|
%11 = OpTypeStruct %10 %10
|
|
%12 = OpTypePointer Function %10
|
|
%13 = OpTypePointer Function %11
|
|
%14 = OpConstant %10 0
|
|
%15 = OpConstant %10 1
|
|
%7 = OpTypeFunction %9
|
|
%1 = OpFunction %9 None %7
|
|
%2 = OpLabel
|
|
%8 = OpVariable %13 Function
|
|
%3 = OpAccessChain %12 %8 %14
|
|
%4 = OpLoad %10 %3
|
|
%5 = OpAccessChain %12 %8 %15
|
|
%6 = OpLoad %10 %5
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::string s;
|
|
std::ostringstream str(s);
|
|
str << *BuildModule(text)->module();
|
|
EXPECT_EQ(text, str.str());
|
|
}
|
|
|
|
TEST(ModuleTest, OstreamOperatorInt64) {
|
|
const std::string text = R"(OpCapability Shader
|
|
OpCapability Linkage
|
|
OpCapability Int64
|
|
OpMemoryModel Logical GLSL450
|
|
OpName %7 "restrict"
|
|
OpDecorate %5 Restrict
|
|
%9 = OpTypeVoid
|
|
%10 = OpTypeInt 64 0
|
|
%11 = OpTypeStruct %10 %10
|
|
%12 = OpTypePointer Function %10
|
|
%13 = OpTypePointer Function %11
|
|
%14 = OpConstant %10 0
|
|
%15 = OpConstant %10 1
|
|
%16 = OpConstant %10 4294967297
|
|
%7 = OpTypeFunction %9
|
|
%1 = OpFunction %9 None %7
|
|
%2 = OpLabel
|
|
%5 = OpVariable %12 Function
|
|
%6 = OpLoad %10 %5
|
|
OpSelectionMerge %3 None
|
|
OpSwitch %6 %3 4294967297 %4
|
|
%4 = OpLabel
|
|
OpBranch %3
|
|
%3 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::string s;
|
|
std::ostringstream str(s);
|
|
str << *BuildModule(text)->module();
|
|
EXPECT_EQ(text, str.str());
|
|
}
|
|
|
|
TEST(ModuleTest, IdBoundTestAtLimit) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%1 = OpTypeVoid
|
|
%2 = OpTypeFunction %1
|
|
%3 = OpFunction %1 None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
uint32_t current_bound = context->module()->id_bound();
|
|
context->set_max_id_bound(current_bound);
|
|
uint32_t next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, 0);
|
|
EXPECT_EQ(current_bound, context->module()->id_bound());
|
|
next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, 0);
|
|
}
|
|
|
|
TEST(ModuleTest, IdBoundTestBelowLimit) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%1 = OpTypeVoid
|
|
%2 = OpTypeFunction %1
|
|
%3 = OpFunction %1 None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
uint32_t current_bound = context->module()->id_bound();
|
|
context->set_max_id_bound(current_bound + 100);
|
|
uint32_t next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, current_bound);
|
|
EXPECT_EQ(current_bound + 1, context->module()->id_bound());
|
|
next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, current_bound + 1);
|
|
}
|
|
|
|
TEST(ModuleTest, IdBoundTestNearLimit) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%1 = OpTypeVoid
|
|
%2 = OpTypeFunction %1
|
|
%3 = OpFunction %1 None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
uint32_t current_bound = context->module()->id_bound();
|
|
context->set_max_id_bound(current_bound + 1);
|
|
uint32_t next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, current_bound);
|
|
EXPECT_EQ(current_bound + 1, context->module()->id_bound());
|
|
next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, 0);
|
|
}
|
|
|
|
TEST(ModuleTest, IdBoundTestUIntMax) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%1 = OpTypeVoid
|
|
%2 = OpTypeFunction %1
|
|
%3 = OpFunction %1 None %2
|
|
%4294967294 = OpLabel ; ID is UINT_MAX-1
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
uint32_t current_bound = context->module()->id_bound();
|
|
|
|
// Expecting |BuildModule| to preserve the numeric ids.
|
|
EXPECT_EQ(current_bound, std::numeric_limits<uint32_t>::max());
|
|
|
|
context->set_max_id_bound(current_bound);
|
|
uint32_t next_id_bound = context->module()->TakeNextIdBound();
|
|
EXPECT_EQ(next_id_bound, 0);
|
|
EXPECT_EQ(current_bound, context->module()->id_bound());
|
|
}
|
|
|
|
// Tests that "text" does not change when it is assembled, converted into a
|
|
// module, converted back to a binary, and then disassembled.
|
|
void AssembleAndDisassemble(const std::string& text) {
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
std::vector<uint32_t> binary;
|
|
|
|
context->module()->ToBinary(&binary, false);
|
|
|
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_1);
|
|
std::string s;
|
|
tools.Disassemble(binary, &s);
|
|
EXPECT_EQ(s, text);
|
|
}
|
|
|
|
TEST(ModuleTest, TrailingOpLine) {
|
|
const std::string text = R"(OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%5 = OpString "file.ext"
|
|
%void = OpTypeVoid
|
|
%2 = OpTypeFunction %void
|
|
%3 = OpFunction %void None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
OpLine %5 1 0
|
|
)";
|
|
|
|
AssembleAndDisassemble(text);
|
|
}
|
|
|
|
TEST(ModuleTest, TrailingOpNoLine) {
|
|
const std::string text = R"(OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%void = OpTypeVoid
|
|
%2 = OpTypeFunction %void
|
|
%3 = OpFunction %void None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
OpNoLine
|
|
)";
|
|
|
|
AssembleAndDisassemble(text);
|
|
}
|
|
|
|
TEST(ModuleTest, MulitpleTrailingOpLine) {
|
|
const std::string text = R"(OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
%5 = OpString "file.ext"
|
|
%void = OpTypeVoid
|
|
%2 = OpTypeFunction %void
|
|
%3 = OpFunction %void None %2
|
|
%4 = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
OpLine %5 1 0
|
|
OpNoLine
|
|
OpLine %5 1 1
|
|
)";
|
|
|
|
AssembleAndDisassemble(text);
|
|
}
|
|
|
|
TEST(ModuleTest, NonSemanticInfoIteration) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpExtension "SPV_KHR_non_semantic_info"
|
|
%1 = OpExtInstImport "NonSemantic.Test"
|
|
OpMemoryModel Logical GLSL450
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeFunction %2
|
|
%4 = OpExtInst %2 %1 1
|
|
%5 = OpFunction %2 None %3
|
|
%6 = OpLabel
|
|
%7 = OpExtInst %2 %1 1
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%8 = OpExtInst %2 %1 1
|
|
%9 = OpFunction %2 None %3
|
|
%10 = OpLabel
|
|
%11 = OpExtInst %2 %1 1
|
|
OpReturn
|
|
OpFunctionEnd
|
|
%12 = OpExtInst %2 %1 1
|
|
)";
|
|
|
|
std::unique_ptr<IRContext> context = BuildModule(text);
|
|
std::unordered_set<uint32_t> non_semantic_ids;
|
|
context->module()->ForEachInst(
|
|
[&non_semantic_ids](const Instruction* inst) {
|
|
if (inst->opcode() == SpvOpExtInst) {
|
|
non_semantic_ids.insert(inst->result_id());
|
|
}
|
|
},
|
|
false);
|
|
|
|
EXPECT_EQ(1, non_semantic_ids.count(4));
|
|
EXPECT_EQ(1, non_semantic_ids.count(7));
|
|
EXPECT_EQ(1, non_semantic_ids.count(8));
|
|
EXPECT_EQ(1, non_semantic_ids.count(11));
|
|
EXPECT_EQ(1, non_semantic_ids.count(12));
|
|
}
|
|
} // namespace
|
|
} // namespace opt
|
|
} // namespace spvtools
|