| // 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 |