// 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 <initializer_list> #include <memory> #include <string> #include <utility> #include <vector> #include "gmock/gmock.h" #include "source/util/make_unique.h" #include "test/opt/module_utils.h" #include "test/opt/pass_fixture.h" namespace spvtools { namespace opt { namespace { using spvtest::GetIdBound; using ::testing::Eq; // A null pass whose construtors accept arguments class NullPassWithArgs : public NullPass { public: NullPassWithArgs(uint32_t) {} NullPassWithArgs(std::string) {} NullPassWithArgs(const std::vector<int>&) {} NullPassWithArgs(const std::vector<int>&, uint32_t) {} const char* name() const override { return "null-with-args"; } }; TEST(PassManager, Interface) { PassManager manager; EXPECT_EQ(0u, manager.NumPasses()); manager.AddPass<StripDebugInfoPass>(); EXPECT_EQ(1u, manager.NumPasses()); EXPECT_STREQ("strip-debug", manager.GetPass(0)->name()); manager.AddPass(MakeUnique<NullPass>()); EXPECT_EQ(2u, manager.NumPasses()); EXPECT_STREQ("strip-debug", manager.GetPass(0)->name()); EXPECT_STREQ("null", manager.GetPass(1)->name()); manager.AddPass<StripDebugInfoPass>(); EXPECT_EQ(3u, manager.NumPasses()); EXPECT_STREQ("strip-debug", manager.GetPass(0)->name()); EXPECT_STREQ("null", manager.GetPass(1)->name()); EXPECT_STREQ("strip-debug", manager.GetPass(2)->name()); manager.AddPass<NullPassWithArgs>(1u); manager.AddPass<NullPassWithArgs>("null pass args"); manager.AddPass<NullPassWithArgs>(std::initializer_list<int>{1, 2}); manager.AddPass<NullPassWithArgs>(std::initializer_list<int>{1, 2}, 3); EXPECT_EQ(7u, manager.NumPasses()); EXPECT_STREQ("strip-debug", manager.GetPass(0)->name()); EXPECT_STREQ("null", manager.GetPass(1)->name()); EXPECT_STREQ("strip-debug", manager.GetPass(2)->name()); EXPECT_STREQ("null-with-args", manager.GetPass(3)->name()); EXPECT_STREQ("null-with-args", manager.GetPass(4)->name()); EXPECT_STREQ("null-with-args", manager.GetPass(5)->name()); EXPECT_STREQ("null-with-args", manager.GetPass(6)->name()); } // A pass that appends an OpNop instruction to the debug1 section. class AppendOpNopPass : public Pass { public: const char* name() const override { return "AppendOpNop"; } Status Process() override { context()->AddDebug1Inst(MakeUnique<Instruction>(context())); return Status::SuccessWithChange; } }; // A pass that appends specified number of OpNop instructions to the debug1 // section. class AppendMultipleOpNopPass : public Pass { public: explicit AppendMultipleOpNopPass(uint32_t num_nop) : num_nop_(num_nop) {} const char* name() const override { return "AppendOpNop"; } Status Process() override { for (uint32_t i = 0; i < num_nop_; i++) { context()->AddDebug1Inst(MakeUnique<Instruction>(context())); } return Status::SuccessWithChange; } private: uint32_t num_nop_; }; // A pass that duplicates the last instruction in the debug1 section. class DuplicateInstPass : public Pass { public: const char* name() const override { return "DuplicateInst"; } Status Process() override { auto inst = MakeUnique<Instruction>(*(--context()->debug1_end())); context()->AddDebug1Inst(std::move(inst)); return Status::SuccessWithChange; } }; using PassManagerTest = PassTest<::testing::Test>; TEST_F(PassManagerTest, Run) { const std::string text = "OpMemoryModel Logical GLSL450\nOpSource ESSL 310\n"; AddPass<AppendOpNopPass>(); AddPass<AppendOpNopPass>(); RunAndCheck(text, text + "OpNop\nOpNop\n"); RenewPassManger(); AddPass<AppendOpNopPass>(); AddPass<DuplicateInstPass>(); RunAndCheck(text, text + "OpNop\nOpNop\n"); RenewPassManger(); AddPass<DuplicateInstPass>(); AddPass<AppendOpNopPass>(); RunAndCheck(text, text + "OpSource ESSL 310\nOpNop\n"); RenewPassManger(); AddPass<AppendMultipleOpNopPass>(3); RunAndCheck(text, text + "OpNop\nOpNop\nOpNop\n"); } // A pass that appends an OpTypeVoid instruction that uses a given id. class AppendTypeVoidInstPass : public Pass { public: explicit AppendTypeVoidInstPass(uint32_t result_id) : result_id_(result_id) {} const char* name() const override { return "AppendTypeVoidInstPass"; } Status Process() override { auto inst = MakeUnique<Instruction>(context(), SpvOpTypeVoid, 0, result_id_, std::vector<Operand>{}); context()->AddType(std::move(inst)); return Status::SuccessWithChange; } private: uint32_t result_id_; }; TEST(PassManager, RecomputeIdBoundAutomatically) { PassManager manager; std::unique_ptr<Module> module(new Module()); IRContext context(SPV_ENV_UNIVERSAL_1_2, std::move(module), manager.consumer()); EXPECT_THAT(GetIdBound(*context.module()), Eq(0u)); manager.Run(&context); manager.AddPass<AppendOpNopPass>(); // With no ID changes, the ID bound does not change. EXPECT_THAT(GetIdBound(*context.module()), Eq(0u)); // Now we force an Id of 100 to be used. manager.AddPass(MakeUnique<AppendTypeVoidInstPass>(100)); EXPECT_THAT(GetIdBound(*context.module()), Eq(0u)); manager.Run(&context); // The Id has been updated automatically, even though the pass // did not update it. EXPECT_THAT(GetIdBound(*context.module()), Eq(101u)); // Try one more time! manager.AddPass(MakeUnique<AppendTypeVoidInstPass>(200)); manager.Run(&context); EXPECT_THAT(GetIdBound(*context.module()), Eq(201u)); // Add another pass, but which uses a lower Id. manager.AddPass(MakeUnique<AppendTypeVoidInstPass>(10)); manager.Run(&context); // The Id stays high. EXPECT_THAT(GetIdBound(*context.module()), Eq(201u)); } } // anonymous namespace } // namespace opt } // namespace spvtools