mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-02-01 13:11:39 +01:00
ff724f5a95
getARMVPTBlockMask was an outdated function that only handled basic block masks: T, TT, TTT and TTTT. This worked fine before the MVE VPT Block Insertion Pass improvements as it was the only kind of masks that it could generate, but now it can generate more complex masks that uses E predicates, so it's dangerous to use that function to calculate VPT/VPST block masks. I replaced it with 2 different functions: - expandPredBlockMask, in ARMBaseInfo. This adds an "E" or "T" at the end of an existing PredBlockMask. - recomputeVPTBlockMask, in Thumb2InstrInfo. This takes an iterator to a VPT/VPST instruction and recomputes its block mask by looking at the predicated instructions that follows it. This should be used to recompute a block mask after removing/adding a predicated instruction to the block. The expandPredBlockMask function is pretty much imported from the MVE VPT Blocks pass. I had to change the ARMLowOverheadLoops and MVEVPTBlocks passes as well so they could use these new functions. Differential Revision: https://reviews.llvm.org/D78201
238 lines
7.4 KiB
C++
238 lines
7.4 KiB
C++
//===-- ARMBaseInfo.h - Top level definitions for ARM ---*- 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains small standalone helper functions and enum definitions for
|
|
// the ARM target useful for the compiler back-end and the MC libraries.
|
|
// As such, it deliberately does not include references to LLVM core
|
|
// code gen types, passes, etc..
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_LIB_TARGET_ARM_UTILS_ARMBASEINFO_H
|
|
#define LLVM_LIB_TARGET_ARM_UTILS_ARMBASEINFO_H
|
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include "llvm/MC/SubtargetFeature.h"
|
|
#include "MCTargetDesc/ARMMCTargetDesc.h"
|
|
|
|
namespace llvm {
|
|
|
|
// Enums corresponding to ARM condition codes
|
|
namespace ARMCC {
|
|
// The CondCodes constants map directly to the 4-bit encoding of the
|
|
// condition field for predicated instructions.
|
|
enum CondCodes { // Meaning (integer) Meaning (floating-point)
|
|
EQ, // Equal Equal
|
|
NE, // Not equal Not equal, or unordered
|
|
HS, // Carry set >, ==, or unordered
|
|
LO, // Carry clear Less than
|
|
MI, // Minus, negative Less than
|
|
PL, // Plus, positive or zero >, ==, or unordered
|
|
VS, // Overflow Unordered
|
|
VC, // No overflow Not unordered
|
|
HI, // Unsigned higher Greater than, or unordered
|
|
LS, // Unsigned lower or same Less than or equal
|
|
GE, // Greater than or equal Greater than or equal
|
|
LT, // Less than Less than, or unordered
|
|
GT, // Greater than Greater than
|
|
LE, // Less than or equal <, ==, or unordered
|
|
AL // Always (unconditional) Always (unconditional)
|
|
};
|
|
|
|
inline static CondCodes getOppositeCondition(CondCodes CC) {
|
|
switch (CC) {
|
|
default: llvm_unreachable("Unknown condition code");
|
|
case EQ: return NE;
|
|
case NE: return EQ;
|
|
case HS: return LO;
|
|
case LO: return HS;
|
|
case MI: return PL;
|
|
case PL: return MI;
|
|
case VS: return VC;
|
|
case VC: return VS;
|
|
case HI: return LS;
|
|
case LS: return HI;
|
|
case GE: return LT;
|
|
case LT: return GE;
|
|
case GT: return LE;
|
|
case LE: return GT;
|
|
}
|
|
}
|
|
|
|
/// getSwappedCondition - assume the flags are set by MI(a,b), return
|
|
/// the condition code if we modify the instructions such that flags are
|
|
/// set by MI(b,a).
|
|
inline static ARMCC::CondCodes getSwappedCondition(ARMCC::CondCodes CC) {
|
|
switch (CC) {
|
|
default: return ARMCC::AL;
|
|
case ARMCC::EQ: return ARMCC::EQ;
|
|
case ARMCC::NE: return ARMCC::NE;
|
|
case ARMCC::HS: return ARMCC::LS;
|
|
case ARMCC::LO: return ARMCC::HI;
|
|
case ARMCC::HI: return ARMCC::LO;
|
|
case ARMCC::LS: return ARMCC::HS;
|
|
case ARMCC::GE: return ARMCC::LE;
|
|
case ARMCC::LT: return ARMCC::GT;
|
|
case ARMCC::GT: return ARMCC::LT;
|
|
case ARMCC::LE: return ARMCC::GE;
|
|
}
|
|
}
|
|
} // end namespace ARMCC
|
|
|
|
namespace ARMVCC {
|
|
enum VPTCodes {
|
|
None = 0,
|
|
Then,
|
|
Else
|
|
};
|
|
} // namespace ARMVCC
|
|
|
|
namespace ARM {
|
|
/// Mask values for IT and VPT Blocks, to be used by MCOperands.
|
|
/// Note that this is different from the "real" encoding used by the
|
|
/// instructions. In this encoding, the lowest set bit indicates the end of
|
|
/// the encoding, and above that, "1" indicates an else, while "0" indicates
|
|
/// a then.
|
|
/// Tx = x100
|
|
/// Txy = xy10
|
|
/// Txyz = xyz1
|
|
enum class PredBlockMask {
|
|
T = 0b1000,
|
|
TT = 0b0100,
|
|
TE = 0b1100,
|
|
TTT = 0b0010,
|
|
TTE = 0b0110,
|
|
TEE = 0b1110,
|
|
TET = 0b1010,
|
|
TTTT = 0b0001,
|
|
TTTE = 0b0011,
|
|
TTEE = 0b0111,
|
|
TTET = 0b0101,
|
|
TEEE = 0b1111,
|
|
TEET = 0b1101,
|
|
TETT = 0b1001,
|
|
TETE = 0b1011
|
|
};
|
|
} // namespace ARM
|
|
|
|
// Expands a PredBlockMask by adding an E or a T at the end, depending on Kind.
|
|
// e.g ExpandPredBlockMask(T, Then) = TT, ExpandPredBlockMask(TT, Else) = TTE,
|
|
// and so on.
|
|
ARM::PredBlockMask expandPredBlockMask(ARM::PredBlockMask BlockMask,
|
|
ARMVCC::VPTCodes Kind);
|
|
|
|
inline static const char *ARMVPTPredToString(ARMVCC::VPTCodes CC) {
|
|
switch (CC) {
|
|
case ARMVCC::None: return "none";
|
|
case ARMVCC::Then: return "t";
|
|
case ARMVCC::Else: return "e";
|
|
}
|
|
llvm_unreachable("Unknown VPT code");
|
|
}
|
|
|
|
inline static unsigned ARMVectorCondCodeFromString(StringRef CC) {
|
|
return StringSwitch<unsigned>(CC.lower())
|
|
.Case("t", ARMVCC::Then)
|
|
.Case("e", ARMVCC::Else)
|
|
.Default(~0U);
|
|
}
|
|
|
|
inline static const char *ARMCondCodeToString(ARMCC::CondCodes CC) {
|
|
switch (CC) {
|
|
case ARMCC::EQ: return "eq";
|
|
case ARMCC::NE: return "ne";
|
|
case ARMCC::HS: return "hs";
|
|
case ARMCC::LO: return "lo";
|
|
case ARMCC::MI: return "mi";
|
|
case ARMCC::PL: return "pl";
|
|
case ARMCC::VS: return "vs";
|
|
case ARMCC::VC: return "vc";
|
|
case ARMCC::HI: return "hi";
|
|
case ARMCC::LS: return "ls";
|
|
case ARMCC::GE: return "ge";
|
|
case ARMCC::LT: return "lt";
|
|
case ARMCC::GT: return "gt";
|
|
case ARMCC::LE: return "le";
|
|
case ARMCC::AL: return "al";
|
|
}
|
|
llvm_unreachable("Unknown condition code");
|
|
}
|
|
|
|
inline static unsigned ARMCondCodeFromString(StringRef CC) {
|
|
return StringSwitch<unsigned>(CC.lower())
|
|
.Case("eq", ARMCC::EQ)
|
|
.Case("ne", ARMCC::NE)
|
|
.Case("hs", ARMCC::HS)
|
|
.Case("cs", ARMCC::HS)
|
|
.Case("lo", ARMCC::LO)
|
|
.Case("cc", ARMCC::LO)
|
|
.Case("mi", ARMCC::MI)
|
|
.Case("pl", ARMCC::PL)
|
|
.Case("vs", ARMCC::VS)
|
|
.Case("vc", ARMCC::VC)
|
|
.Case("hi", ARMCC::HI)
|
|
.Case("ls", ARMCC::LS)
|
|
.Case("ge", ARMCC::GE)
|
|
.Case("lt", ARMCC::LT)
|
|
.Case("gt", ARMCC::GT)
|
|
.Case("le", ARMCC::LE)
|
|
.Case("al", ARMCC::AL)
|
|
.Default(~0U);
|
|
}
|
|
|
|
// System Registers
|
|
namespace ARMSysReg {
|
|
struct MClassSysReg {
|
|
const char *Name;
|
|
uint16_t M1Encoding12;
|
|
uint16_t M2M3Encoding8;
|
|
uint16_t Encoding;
|
|
FeatureBitset FeaturesRequired;
|
|
|
|
// return true if FeaturesRequired are all present in ActiveFeatures
|
|
bool hasRequiredFeatures(FeatureBitset ActiveFeatures) const {
|
|
return (FeaturesRequired & ActiveFeatures) == FeaturesRequired;
|
|
}
|
|
|
|
// returns true if TestFeatures are all present in FeaturesRequired
|
|
bool isInRequiredFeatures(FeatureBitset TestFeatures) const {
|
|
return (FeaturesRequired & TestFeatures) == TestFeatures;
|
|
}
|
|
};
|
|
|
|
#define GET_MCLASSSYSREG_DECL
|
|
#include "ARMGenSystemRegister.inc"
|
|
|
|
// lookup system register using 12-bit SYSm value.
|
|
// Note: the search is uniqued using M1 mask
|
|
const MClassSysReg *lookupMClassSysRegBy12bitSYSmValue(unsigned SYSm);
|
|
|
|
// returns APSR with _<bits> qualifier.
|
|
// Note: ARMv7-M deprecates using MSR APSR without a _<bits> qualifier
|
|
const MClassSysReg *lookupMClassSysRegAPSRNonDeprecated(unsigned SYSm);
|
|
|
|
// lookup system registers using 8-bit SYSm value
|
|
const MClassSysReg *lookupMClassSysRegBy8bitSYSmValue(unsigned SYSm);
|
|
|
|
} // end namespace ARMSysReg
|
|
|
|
// Banked Registers
|
|
namespace ARMBankedReg {
|
|
struct BankedReg {
|
|
const char *Name;
|
|
uint16_t Encoding;
|
|
};
|
|
#define GET_BANKEDREG_DECL
|
|
#include "ARMGenSystemRegister.inc"
|
|
} // end namespace ARMBankedReg
|
|
|
|
} // end namespace llvm
|
|
|
|
#endif // LLVM_LIB_TARGET_ARM_UTILS_ARMBASEINFO_H
|