Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions llvm/include/llvm/MC/MCInstrAnalysis.h
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,10 @@ class LLVM_ABI MCInstrAnalysis {
return Info->get(Inst.getOpcode()).isTerminator();
}

virtual bool isBarrier(const MCInst &Inst) const {
return Info->get(Inst.getOpcode()).isBarrier();
}

virtual bool mayAffectControlFlow(const MCInst &Inst,
const MCRegisterInfo &MCRI) const {
if (isBranch(Inst) || isCall(Inst) || isReturn(Inst) ||
Expand Down
1 change: 1 addition & 0 deletions llvm/unittests/Target/AArch64/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -34,4 +34,5 @@ add_llvm_target_unittest(AArch64Tests
AArch64SVESchedPseudoTest.cpp
AArch64SelectionDAGTest.cpp
Immediates.cpp
MCInstrAnalysisTest.cpp
)
142 changes: 142 additions & 0 deletions llvm/unittests/Target/AArch64/MCInstrAnalysisTest.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
//===- MCInstrAnalysisTest.cpp - AArch64MCInstrAnalysis unit tests --------===//
//
// 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 "llvm/MC/MCInstrAnalysis.h"
#include "MCTargetDesc/AArch64MCTargetDesc.h"
#include "Utils/AArch64BaseInfo.h"
#include "llvm/MC/MCInstBuilder.h"
#include "llvm/MC/TargetRegistry.h"
#include "llvm/Support/TargetSelect.h"

#include "gtest/gtest.h"

#include <memory>

using namespace llvm;

namespace {

class InstrAnalysisTest : public testing::TestWithParam<const char *> {
protected:
std::unique_ptr<const MCInstrInfo> Info;
std::unique_ptr<const MCInstrAnalysis> Analysis;

static void SetUpTestSuite() {
LLVMInitializeAArch64TargetInfo();
LLVMInitializeAArch64Target();
LLVMInitializeAArch64TargetMC();
}

InstrAnalysisTest() {
std::string Error;
const Target *TheTarget =
TargetRegistry::lookupTarget(Triple::normalize(GetParam()), Error);
Info = std::unique_ptr<const MCInstrInfo>(TheTarget->createMCInstrInfo());
Analysis = std::unique_ptr<const MCInstrAnalysis>(
TheTarget->createMCInstrAnalysis(Info.get()));
}
};

} // namespace

static MCInst beq() {
return MCInstBuilder(AArch64::Bcc).addImm(AArch64CC::EQ).addReg(AArch64::X0);
}
static MCInst tbz(unsigned Rt = AArch64::X0, unsigned Imm = 0,
unsigned Label = 32) {
return MCInstBuilder(AArch64::TBZX).addReg(Rt).addImm(Imm).addImm(Label);
}
static MCInst b() { return MCInstBuilder(AArch64::B).addImm(32); }
static MCInst bl() { return MCInstBuilder(AArch64::BL).addImm(32); }
static MCInst br(unsigned Rn = AArch64::X0) {
return MCInstBuilder(AArch64::BR).addReg(Rn);
}
static MCInst ret(unsigned Rn = AArch64::LR) {
return MCInstBuilder(AArch64::RET).addReg(Rn);
}
static MCInst hlt() { return MCInstBuilder(AArch64::HLT); }
static MCInst brk() { return MCInstBuilder(AArch64::BRK); }

TEST_P(InstrAnalysisTest, IsTerminator) {
EXPECT_TRUE(Analysis->isTerminator(beq()));
EXPECT_TRUE(Analysis->isTerminator(tbz()));
EXPECT_TRUE(Analysis->isTerminator(b()));
EXPECT_FALSE(Analysis->isTerminator(bl()));
EXPECT_TRUE(Analysis->isTerminator(br()));
EXPECT_TRUE(Analysis->isTerminator(ret()));
EXPECT_FALSE(Analysis->isTerminator(hlt()));
EXPECT_FALSE(Analysis->isTerminator(brk()));
}

TEST_P(InstrAnalysisTest, IsBarrier) {
EXPECT_FALSE(Analysis->isBarrier(beq()));
EXPECT_FALSE(Analysis->isBarrier(tbz()));
EXPECT_TRUE(Analysis->isBarrier(b()));
EXPECT_FALSE(Analysis->isBarrier(bl()));
EXPECT_TRUE(Analysis->isBarrier(br()));
EXPECT_TRUE(Analysis->isBarrier(ret()));
EXPECT_FALSE(Analysis->isBarrier(hlt()));
EXPECT_FALSE(Analysis->isBarrier(brk()));
}

TEST_P(InstrAnalysisTest, IsCall) {
EXPECT_FALSE(Analysis->isCall(beq()));
EXPECT_FALSE(Analysis->isCall(tbz()));
EXPECT_FALSE(Analysis->isCall(b()));
EXPECT_TRUE(Analysis->isCall(bl()));
EXPECT_FALSE(Analysis->isCall(br()));
EXPECT_FALSE(Analysis->isCall(ret()));
}

TEST_P(InstrAnalysisTest, IsReturn) {
EXPECT_FALSE(Analysis->isReturn(beq()));
EXPECT_FALSE(Analysis->isReturn(tbz()));
EXPECT_FALSE(Analysis->isReturn(b()));
EXPECT_FALSE(Analysis->isReturn(bl()));
EXPECT_FALSE(Analysis->isReturn(br()));
EXPECT_FALSE(Analysis->isReturn(br(AArch64::LR)));
EXPECT_TRUE(Analysis->isReturn(ret()));
}

TEST_P(InstrAnalysisTest, IsBranch) {
EXPECT_TRUE(Analysis->isBranch(beq()));
EXPECT_TRUE(Analysis->isBranch(tbz()));
EXPECT_TRUE(Analysis->isBranch(b()));
EXPECT_FALSE(Analysis->isBranch(bl()));
EXPECT_TRUE(Analysis->isBranch(br()));
EXPECT_FALSE(Analysis->isBranch(ret()));
}

TEST_P(InstrAnalysisTest, IsConditionalBranch) {
EXPECT_TRUE(Analysis->isConditionalBranch(beq()));
EXPECT_TRUE(Analysis->isConditionalBranch(tbz()));
EXPECT_FALSE(Analysis->isConditionalBranch(b()));
EXPECT_FALSE(Analysis->isConditionalBranch(bl()));
EXPECT_FALSE(Analysis->isConditionalBranch(ret()));
}

TEST_P(InstrAnalysisTest, IsUnconditionalBranch) {
EXPECT_FALSE(Analysis->isUnconditionalBranch(beq()));
EXPECT_FALSE(Analysis->isUnconditionalBranch(tbz()));
EXPECT_TRUE(Analysis->isUnconditionalBranch(b()));
EXPECT_FALSE(Analysis->isUnconditionalBranch(bl()));
EXPECT_FALSE(Analysis->isUnconditionalBranch(br()));
EXPECT_FALSE(Analysis->isUnconditionalBranch(ret()));
}

TEST_P(InstrAnalysisTest, IsIndirectBranch) {
EXPECT_FALSE(Analysis->isIndirectBranch(beq()));
EXPECT_FALSE(Analysis->isIndirectBranch(tbz()));
EXPECT_FALSE(Analysis->isIndirectBranch(b()));
EXPECT_FALSE(Analysis->isIndirectBranch(bl()));
EXPECT_TRUE(Analysis->isIndirectBranch(br()));
EXPECT_FALSE(Analysis->isIndirectBranch(ret()));
}

INSTANTIATE_TEST_SUITE_P(AArch64, InstrAnalysisTest,
testing::Values("aarch64"));