| //===-- SnippetGeneratorTest.cpp --------------------------------*- C++ -*-===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "../Common/AssemblerUtils.h" |
| #include "LlvmState.h" |
| #include "MCInstrDescView.h" |
| #include "MipsInstrInfo.h" |
| #include "ParallelSnippetGenerator.h" |
| #include "RegisterAliasing.h" |
| #include "SerialSnippetGenerator.h" |
| #include "TestBase.h" |
| |
| #include <unordered_set> |
| |
| namespace llvm { |
| namespace exegesis { |
| namespace { |
| |
| using testing::AnyOf; |
| using testing::ElementsAre; |
| using testing::HasSubstr; |
| using testing::SizeIs; |
| |
| MATCHER(IsInvalid, "") { return !arg.isValid(); } |
| MATCHER(IsReg, "") { return arg.isReg(); } |
| |
| template <typename SnippetGeneratorT> |
| class MipsSnippetGeneratorTest : public MipsTestBase { |
| protected: |
| MipsSnippetGeneratorTest() : Generator(State, SnippetGenerator::Options()) {} |
| |
| std::vector<CodeTemplate> checkAndGetCodeTemplates(unsigned Opcode) { |
| randomGenerator().seed(0); // Initialize seed. |
| const Instruction &Instr = State.getIC().getInstr(Opcode); |
| auto CodeTemplateOrError = Generator.generateCodeTemplates( |
| &Instr, State.getRATC().emptyRegisters()); |
| EXPECT_FALSE(CodeTemplateOrError.takeError()); // Valid configuration. |
| return std::move(CodeTemplateOrError.get()); |
| } |
| |
| SnippetGeneratorT Generator; |
| }; |
| |
| using MipsSerialSnippetGeneratorTest = MipsSnippetGeneratorTest<SerialSnippetGenerator>; |
| |
| using MipsParallelSnippetGeneratorTest = |
| MipsSnippetGeneratorTest<ParallelSnippetGenerator>; |
| |
| TEST_F(MipsSerialSnippetGeneratorTest, ImplicitSelfDependencyThroughExplicitRegs) { |
| // - ADD |
| // - Op0 Explicit Def RegClass(GPR32) |
| // - Op1 Explicit Use RegClass(GPR32) |
| // - Op2 Explicit Use RegClass(GPR32) |
| // - Var0 [Op0] |
| // - Var1 [Op1] |
| // - Var2 [Op2] |
| // - hasAliasingRegisters |
| const unsigned Opcode = Mips::ADD; |
| const auto CodeTemplates = checkAndGetCodeTemplates(Opcode); |
| ASSERT_THAT(CodeTemplates, SizeIs(1)); |
| const auto &CT = CodeTemplates[0]; |
| EXPECT_THAT(CT.Execution, ExecutionMode::SERIAL_VIA_EXPLICIT_REGS); |
| ASSERT_THAT(CT.Instructions, SizeIs(1)); |
| const InstructionTemplate &IT = CT.Instructions[0]; |
| EXPECT_THAT(IT.getOpcode(), Opcode); |
| ASSERT_THAT(IT.getVariableValues(), SizeIs(3)); |
| EXPECT_THAT(IT.getVariableValues(), |
| AnyOf(ElementsAre(IsReg(), IsInvalid(), IsReg()), |
| ElementsAre(IsReg(), IsReg(), IsInvalid()))) |
| << "Op0 is either set to Op1 or to Op2"; |
| } |
| |
| TEST_F(MipsSerialSnippetGeneratorTest, |
| ImplicitSelfDependencyThroughExplicitRegsForbidAll) { |
| // - XOR |
| // - Op0 Explicit Def RegClass(GPR32) |
| // - Op1 Explicit Use RegClass(GPR32) |
| // - Op2 Explicit Use RegClass(GPR32) |
| // - Var0 [Op0] |
| // - Var1 [Op1] |
| // - Var2 [Op2] |
| // - hasAliasingRegisters |
| randomGenerator().seed(0); // Initialize seed. |
| const Instruction &Instr = State.getIC().getInstr(Mips::XOR); |
| auto AllRegisters = State.getRATC().emptyRegisters(); |
| AllRegisters.flip(); |
| auto Error = |
| Generator.generateCodeTemplates(&Instr, AllRegisters).takeError(); |
| EXPECT_TRUE((bool)Error); |
| consumeError(std::move(Error)); |
| } |
| |
| TEST_F(MipsParallelSnippetGeneratorTest, MemoryUse) { |
| // LB reads from memory. |
| // - LB |
| // - Op0 Explicit Def RegClass(GPR32) |
| // - Op1 Explicit Use Memory RegClass(MSA128F16) |
| // - Op2 Explicit Use Memory |
| // - Var0 [Op0] |
| // - Var1 [Op1] |
| // - Var2 [Op2] |
| // - hasMemoryOperands |
| const unsigned Opcode = Mips::LB; |
| const auto CodeTemplates = checkAndGetCodeTemplates(Opcode); |
| ASSERT_THAT(CodeTemplates, SizeIs(1)); |
| const auto &CT = CodeTemplates[0]; |
| EXPECT_THAT(CT.Info, |
| HasSubstr("instruction is parallel, repeating a random one.")); |
| EXPECT_THAT(CT.Execution, ExecutionMode::UNKNOWN); |
| ASSERT_THAT(CT.Instructions, |
| SizeIs(ParallelSnippetGenerator::kMinNumDifferentAddresses)); |
| const InstructionTemplate &IT = CT.Instructions[0]; |
| EXPECT_THAT(IT.getOpcode(), Opcode); |
| ASSERT_THAT(IT.getVariableValues(), SizeIs(3)); |
| EXPECT_EQ(IT.getVariableValues()[0].getReg(), 0u); |
| EXPECT_EQ(IT.getVariableValues()[2].getImm(), 0); |
| } |
| |
| } // namespace |
| } // namespace exegesis |
| } // namespace llvm |