forked from OSchip/llvm-project
[SystemZ] Match operands to fields by name rather than by order
The SystemZ port currently relies on the order of the instruction operands matching the order of the instruction field lists. This isn't desirable for disassembly, where the two are matched only by name. E.g. the R1 and R2 fields of an RR instruction should have corresponding R1 and R2 operands. The main complication is that addresses are compound operands, and as far as I know there is no mechanism to allow individual suboperands to be selected by name in "let Inst{...} = ..." assignments. Luckily it doesn't really matter though. The SystemZ instruction encoding groups all address fields together in a predictable order, so it's just as valid to see the entire compound address operand as a single field. That's the approach taken in this patch. Matching by name in turn means that the operands to COPY SIGN and CONVERT TO FIXED instructions can be given in natural order. (It was easier to do this at the same time as the rename, since otherwise the intermediate step was too confusing.) No functional change intended. llvm-svn: 181769
This commit is contained in:
parent
24db0f0afd
commit
d454ec0c31
|
@ -45,30 +45,43 @@ private:
|
|||
|
||||
// Called by the TableGen code to get the binary encoding of operand
|
||||
// MO in MI. Fixups is the list of fixups against MI.
|
||||
unsigned getMachineOpValue(const MCInst &MI, const MCOperand &MO,
|
||||
uint64_t getMachineOpValue(const MCInst &MI, const MCOperand &MO,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const;
|
||||
|
||||
// Called by the TableGen code to get the binary encoding of an address.
|
||||
// The index, if any, is encoded first, followed by the base,
|
||||
// followed by the displacement. In a 20-bit displacement,
|
||||
// the low 12 bits are encoded before the high 8 bits.
|
||||
uint64_t getBDAddr12Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const;
|
||||
uint64_t getBDAddr20Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const;
|
||||
uint64_t getBDXAddr12Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const;
|
||||
uint64_t getBDXAddr20Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const;
|
||||
|
||||
// Operand OpNum of MI needs a PC-relative fixup of kind Kind at
|
||||
// Offset bytes from the start of MI. Add the fixup to Fixups
|
||||
// and return the in-place addend, which since we're a RELA target
|
||||
// is always 0.
|
||||
unsigned getPCRelEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t getPCRelEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups,
|
||||
unsigned Kind, int64_t Offset) const;
|
||||
|
||||
unsigned getPC16DBLEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t getPC16DBLEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
return getPCRelEncoding(MI, OpNum, Fixups, SystemZ::FK_390_PC16DBL, 2);
|
||||
}
|
||||
unsigned getPC32DBLEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t getPC32DBLEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
return getPCRelEncoding(MI, OpNum, Fixups, SystemZ::FK_390_PC32DBL, 2);
|
||||
}
|
||||
unsigned getPLT16DBLEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t getPLT16DBLEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
return getPCRelEncoding(MI, OpNum, Fixups, SystemZ::FK_390_PLT16DBL, 2);
|
||||
}
|
||||
unsigned getPLT32DBLEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t getPLT32DBLEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
return getPCRelEncoding(MI, OpNum, Fixups, SystemZ::FK_390_PLT32DBL, 2);
|
||||
}
|
||||
|
@ -95,18 +108,57 @@ EncodeInstruction(const MCInst &MI, raw_ostream &OS,
|
|||
}
|
||||
}
|
||||
|
||||
unsigned SystemZMCCodeEmitter::
|
||||
uint64_t SystemZMCCodeEmitter::
|
||||
getMachineOpValue(const MCInst &MI, const MCOperand &MO,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
if (MO.isReg())
|
||||
return Ctx.getRegisterInfo().getEncodingValue(MO.getReg());
|
||||
if (MO.isImm())
|
||||
return static_cast<unsigned>(MO.getImm());
|
||||
return static_cast<uint64_t>(MO.getImm());
|
||||
llvm_unreachable("Unexpected operand type!");
|
||||
}
|
||||
|
||||
unsigned
|
||||
SystemZMCCodeEmitter::getPCRelEncoding(const MCInst &MI, unsigned int OpNum,
|
||||
uint64_t SystemZMCCodeEmitter::
|
||||
getBDAddr12Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
uint64_t Base = getMachineOpValue(MI, MI.getOperand(OpNum), Fixups);
|
||||
uint64_t Disp = getMachineOpValue(MI, MI.getOperand(OpNum + 1), Fixups);
|
||||
assert(isUInt<4>(Base) && isUInt<12>(Disp));
|
||||
return (Base << 12) | Disp;
|
||||
}
|
||||
|
||||
uint64_t SystemZMCCodeEmitter::
|
||||
getBDAddr20Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
uint64_t Base = getMachineOpValue(MI, MI.getOperand(OpNum), Fixups);
|
||||
uint64_t Disp = getMachineOpValue(MI, MI.getOperand(OpNum + 1), Fixups);
|
||||
assert(isUInt<4>(Base) && isInt<20>(Disp));
|
||||
return (Base << 20) | ((Disp & 0xfff) << 8) | ((Disp & 0xff000) >> 12);
|
||||
}
|
||||
|
||||
uint64_t SystemZMCCodeEmitter::
|
||||
getBDXAddr12Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
uint64_t Base = getMachineOpValue(MI, MI.getOperand(OpNum), Fixups);
|
||||
uint64_t Disp = getMachineOpValue(MI, MI.getOperand(OpNum + 1), Fixups);
|
||||
uint64_t Index = getMachineOpValue(MI, MI.getOperand(OpNum + 2), Fixups);
|
||||
assert(isUInt<4>(Base) && isUInt<12>(Disp) && isUInt<4>(Index));
|
||||
return (Index << 16) | (Base << 12) | Disp;
|
||||
}
|
||||
|
||||
uint64_t SystemZMCCodeEmitter::
|
||||
getBDXAddr20Encoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups) const {
|
||||
uint64_t Base = getMachineOpValue(MI, MI.getOperand(OpNum), Fixups);
|
||||
uint64_t Disp = getMachineOpValue(MI, MI.getOperand(OpNum + 1), Fixups);
|
||||
uint64_t Index = getMachineOpValue(MI, MI.getOperand(OpNum + 2), Fixups);
|
||||
assert(isUInt<4>(Base) && isInt<20>(Disp) && isUInt<4>(Index));
|
||||
return (Index << 24) | (Base << 20) | ((Disp & 0xfff) << 8)
|
||||
| ((Disp & 0xff000) >> 12);
|
||||
}
|
||||
|
||||
uint64_t
|
||||
SystemZMCCodeEmitter::getPCRelEncoding(const MCInst &MI, unsigned OpNum,
|
||||
SmallVectorImpl<MCFixup> &Fixups,
|
||||
unsigned Kind, int64_t Offset) const {
|
||||
const MCOperand &MO = MI.getOperand(OpNum);
|
||||
|
|
|
@ -40,24 +40,22 @@ def LDGR : UnaryRRE<"ldgr", 0xB3C1, bitconvert, FP64, GR64>;
|
|||
|
||||
// fcopysign with an FP32 result.
|
||||
let isCodeGenOnly = 1 in {
|
||||
def CPSDRss : BinaryRevRRF<"cpsdr", 0xB372, fcopysign, FP32, FP32>;
|
||||
def CPSDRsd : BinaryRevRRF<"cpsdr", 0xB372, fcopysign, FP32, FP64>;
|
||||
def CPSDRss : BinaryRRF<"cpsdr", 0xB372, fcopysign, FP32, FP32>;
|
||||
def CPSDRsd : BinaryRRF<"cpsdr", 0xB372, fcopysign, FP32, FP64>;
|
||||
}
|
||||
|
||||
// The sign of an FP128 is in the high register. Give the CPSDRsd
|
||||
// operands in R1, R2, R3 order.
|
||||
// The sign of an FP128 is in the high register.
|
||||
def : Pat<(fcopysign FP32:$src1, FP128:$src2),
|
||||
(CPSDRsd (EXTRACT_SUBREG FP128:$src2, subreg_high), FP32:$src1)>;
|
||||
(CPSDRsd FP32:$src1, (EXTRACT_SUBREG FP128:$src2, subreg_high))>;
|
||||
|
||||
// fcopysign with an FP64 result.
|
||||
let isCodeGenOnly = 1 in
|
||||
def CPSDRds : BinaryRevRRF<"cpsdr", 0xB372, fcopysign, FP64, FP32>;
|
||||
def CPSDRdd : BinaryRevRRF<"cpsdr", 0xB372, fcopysign, FP64, FP64>;
|
||||
def CPSDRds : BinaryRRF<"cpsdr", 0xB372, fcopysign, FP64, FP32>;
|
||||
def CPSDRdd : BinaryRRF<"cpsdr", 0xB372, fcopysign, FP64, FP64>;
|
||||
|
||||
// The sign of an FP128 is in the high register. Give the CPSDRdd
|
||||
// operands in R1, R2, R3 order.
|
||||
// The sign of an FP128 is in the high register.
|
||||
def : Pat<(fcopysign FP64:$src1, FP128:$src2),
|
||||
(CPSDRdd (EXTRACT_SUBREG FP128:$src2, subreg_high), FP64:$src1)>;
|
||||
(CPSDRdd FP64:$src1, (EXTRACT_SUBREG FP128:$src2, subreg_high))>;
|
||||
|
||||
// fcopysign with an FP128 result. Use "upper" as the high half and leave
|
||||
// the low half as-is.
|
||||
|
@ -65,13 +63,12 @@ class CopySign128<RegisterOperand cls, dag upper>
|
|||
: Pat<(fcopysign FP128:$src1, cls:$src2),
|
||||
(INSERT_SUBREG FP128:$src1, upper, subreg_high)>;
|
||||
|
||||
// Give the CPSDR* operands in R1, R2, R3 order.
|
||||
def : CopySign128<FP32, (CPSDRds FP32:$src2,
|
||||
(EXTRACT_SUBREG FP128:$src1, subreg_high))>;
|
||||
def : CopySign128<FP64, (CPSDRdd FP64:$src2,
|
||||
(EXTRACT_SUBREG FP128:$src1, subreg_high))>;
|
||||
def : CopySign128<FP128, (CPSDRdd (EXTRACT_SUBREG FP128:$src2, subreg_high),
|
||||
(EXTRACT_SUBREG FP128:$src1, subreg_high))>;
|
||||
def : CopySign128<FP32, (CPSDRds (EXTRACT_SUBREG FP128:$src1, subreg_high),
|
||||
FP32:$src2)>;
|
||||
def : CopySign128<FP64, (CPSDRdd (EXTRACT_SUBREG FP128:$src1, subreg_high),
|
||||
FP64:$src2)>;
|
||||
def : CopySign128<FP128, (CPSDRdd (EXTRACT_SUBREG FP128:$src1, subreg_high),
|
||||
(EXTRACT_SUBREG FP128:$src2, subreg_high))>;
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Load instructions
|
||||
|
@ -155,13 +152,13 @@ let Defs = [PSW] in {
|
|||
}
|
||||
|
||||
// fp_to_sint always rounds towards zero, which is modifier value 5.
|
||||
def : Pat<(i32 (fp_to_sint FP32:$src)), (CFEBR FP32:$src, 5)>;
|
||||
def : Pat<(i32 (fp_to_sint FP64:$src)), (CFDBR FP64:$src, 5)>;
|
||||
def : Pat<(i32 (fp_to_sint FP128:$src)), (CFXBR FP128:$src, 5)>;
|
||||
def : Pat<(i32 (fp_to_sint FP32:$src)), (CFEBR 5, FP32:$src)>;
|
||||
def : Pat<(i32 (fp_to_sint FP64:$src)), (CFDBR 5, FP64:$src)>;
|
||||
def : Pat<(i32 (fp_to_sint FP128:$src)), (CFXBR 5, FP128:$src)>;
|
||||
|
||||
def : Pat<(i64 (fp_to_sint FP32:$src)), (CGEBR FP32:$src, 5)>;
|
||||
def : Pat<(i64 (fp_to_sint FP64:$src)), (CGDBR FP64:$src, 5)>;
|
||||
def : Pat<(i64 (fp_to_sint FP128:$src)), (CGXBR FP128:$src, 5)>;
|
||||
def : Pat<(i64 (fp_to_sint FP32:$src)), (CGEBR 5, FP32:$src)>;
|
||||
def : Pat<(i64 (fp_to_sint FP64:$src)), (CGDBR 5, FP64:$src)>;
|
||||
def : Pat<(i64 (fp_to_sint FP128:$src)), (CGXBR 5, FP128:$src)>;
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Unary arithmetic
|
||||
|
@ -210,9 +207,9 @@ let Defs = [PSW] in {
|
|||
|
||||
// frint rounds according to the current mode (modifier 0) and detects
|
||||
// inexact conditions.
|
||||
def : Pat<(frint FP32:$src), (FIEBR FP32:$src, 0)>;
|
||||
def : Pat<(frint FP64:$src), (FIDBR FP64:$src, 0)>;
|
||||
def : Pat<(frint FP128:$src), (FIXBR FP128:$src, 0)>;
|
||||
def : Pat<(frint FP32:$src), (FIEBR 0, FP32:$src)>;
|
||||
def : Pat<(frint FP64:$src), (FIDBR 0, FP64:$src)>;
|
||||
def : Pat<(frint FP128:$src), (FIXBR 0, FP128:$src)>;
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Binary arithmetic
|
||||
|
|
|
@ -82,19 +82,17 @@ def getDisp20Opcode : InstrMapping {
|
|||
//
|
||||
// Formats are specified using operand field declarations of the form:
|
||||
//
|
||||
// bits<4> Rn : register input or output for operand n
|
||||
// bits<m> In : immediate value of width m for operand n
|
||||
// bits<4> Bn : base register for address operand n
|
||||
// bits<m> Dn : displacement value of width m for address operand n
|
||||
// bits<4> Xn : index register for address operand n
|
||||
// bits<4> Mn : mode value for operand n
|
||||
// bits<4> Rn : register input or output for operand n
|
||||
// bits<m> In : immediate value of width m for operand n
|
||||
// bits<4> BDn : address operand n, which has a base and a displacement
|
||||
// bits<m> XBDn : address operand n, which has an index, a base and a
|
||||
// displacement
|
||||
// bits<4> Xn : index register for address operand n
|
||||
// bits<4> Mn : mode value for operand n
|
||||
//
|
||||
// The operand numbers ("n" in the list above) follow the architecture manual,
|
||||
// but the fields are always declared in assembly order, so there are some
|
||||
// cases where operand "2" comes after operand "3". For address operands,
|
||||
// the base register field is declared first, followed by the displacement,
|
||||
// followed by the index (if any). This matches the bdaddr* and bdxaddr*
|
||||
// orders.
|
||||
// The operand numbers ("n" in the list above) follow the architecture manual.
|
||||
// Assembly operands sometimes have a different order; in particular, R3 often
|
||||
// is often written between operands 1 and 2.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
|
@ -203,15 +201,11 @@ class InstRX<bits<8> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
field bits<32> Inst;
|
||||
|
||||
bits<4> R1;
|
||||
bits<4> B2;
|
||||
bits<12> D2;
|
||||
bits<4> X2;
|
||||
bits<20> XBD2;
|
||||
|
||||
let Inst{31-24} = op;
|
||||
let Inst{23-20} = R1;
|
||||
let Inst{19-16} = X2;
|
||||
let Inst{15-12} = B2;
|
||||
let Inst{11-0} = D2;
|
||||
let Inst{19-0} = XBD2;
|
||||
|
||||
let HasIndex = 1;
|
||||
}
|
||||
|
@ -221,15 +215,11 @@ class InstRXE<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
field bits<48> Inst;
|
||||
|
||||
bits<4> R1;
|
||||
bits<4> B2;
|
||||
bits<12> D2;
|
||||
bits<4> X2;
|
||||
bits<20> XBD2;
|
||||
|
||||
let Inst{47-40} = op{15-8};
|
||||
let Inst{39-36} = R1;
|
||||
let Inst{35-32} = X2;
|
||||
let Inst{31-28} = B2;
|
||||
let Inst{27-16} = D2;
|
||||
let Inst{35-16} = XBD2;
|
||||
let Inst{15-8} = 0;
|
||||
let Inst{7-0} = op{7-0};
|
||||
|
||||
|
@ -242,15 +232,11 @@ class InstRXF<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
|
||||
bits<4> R1;
|
||||
bits<4> R3;
|
||||
bits<4> B2;
|
||||
bits<12> D2;
|
||||
bits<4> X2;
|
||||
bits<20> XBD2;
|
||||
|
||||
let Inst{47-40} = op{15-8};
|
||||
let Inst{39-36} = R3;
|
||||
let Inst{35-32} = X2;
|
||||
let Inst{31-28} = B2;
|
||||
let Inst{27-16} = D2;
|
||||
let Inst{35-16} = XBD2;
|
||||
let Inst{15-12} = R1;
|
||||
let Inst{11-8} = 0;
|
||||
let Inst{7-0} = op{7-0};
|
||||
|
@ -263,16 +249,11 @@ class InstRXY<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
field bits<48> Inst;
|
||||
|
||||
bits<4> R1;
|
||||
bits<4> B2;
|
||||
bits<20> D2;
|
||||
bits<4> X2;
|
||||
bits<28> XBD2;
|
||||
|
||||
let Inst{47-40} = op{15-8};
|
||||
let Inst{39-36} = R1;
|
||||
let Inst{35-32} = X2;
|
||||
let Inst{31-28} = B2;
|
||||
let Inst{27-16} = D2{11-0};
|
||||
let Inst{15-8} = D2{19-12};
|
||||
let Inst{35-8} = XBD2;
|
||||
let Inst{7-0} = op{7-0};
|
||||
|
||||
let Has20BitOffset = 1;
|
||||
|
@ -285,14 +266,12 @@ class InstRS<bits<8> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
|
||||
bits<4> R1;
|
||||
bits<4> R3;
|
||||
bits<4> B2;
|
||||
bits<12> D2;
|
||||
bits<16> BD2;
|
||||
|
||||
let Inst{31-24} = op;
|
||||
let Inst{23-20} = R1;
|
||||
let Inst{19-16} = R3;
|
||||
let Inst{15-12} = B2;
|
||||
let Inst{11-0} = D2;
|
||||
let Inst{15-0} = BD2;
|
||||
}
|
||||
|
||||
class InstRSY<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
||||
|
@ -301,15 +280,12 @@ class InstRSY<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
|
||||
bits<4> R1;
|
||||
bits<4> R3;
|
||||
bits<4> B2;
|
||||
bits<20> D2;
|
||||
bits<24> BD2;
|
||||
|
||||
let Inst{47-40} = op{15-8};
|
||||
let Inst{39-36} = R1;
|
||||
let Inst{35-32} = R3;
|
||||
let Inst{31-28} = B2;
|
||||
let Inst{27-16} = D2{11-0};
|
||||
let Inst{15-8} = D2{19-12};
|
||||
let Inst{31-8} = BD2;
|
||||
let Inst{7-0} = op{7-0};
|
||||
|
||||
let Has20BitOffset = 1;
|
||||
|
@ -319,27 +295,23 @@ class InstSI<bits<8> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
: InstSystemZ<4, outs, ins, asmstr, pattern> {
|
||||
field bits<32> Inst;
|
||||
|
||||
bits<4> B1;
|
||||
bits<12> D1;
|
||||
bits<16> BD1;
|
||||
bits<8> I2;
|
||||
|
||||
let Inst{31-24} = op;
|
||||
let Inst{23-16} = I2;
|
||||
let Inst{15-12} = B1;
|
||||
let Inst{11-0} = D1;
|
||||
let Inst{15-0} = BD1;
|
||||
}
|
||||
|
||||
class InstSIL<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
||||
: InstSystemZ<6, outs, ins, asmstr, pattern> {
|
||||
field bits<48> Inst;
|
||||
|
||||
bits<4> B1;
|
||||
bits<12> D1;
|
||||
bits<16> BD1;
|
||||
bits<16> I2;
|
||||
|
||||
let Inst{47-32} = op;
|
||||
let Inst{31-28} = B1;
|
||||
let Inst{27-16} = D1;
|
||||
let Inst{31-16} = BD1;
|
||||
let Inst{15-0} = I2;
|
||||
}
|
||||
|
||||
|
@ -347,15 +319,12 @@ class InstSIY<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
: InstSystemZ<6, outs, ins, asmstr, pattern> {
|
||||
field bits<48> Inst;
|
||||
|
||||
bits<4> B1;
|
||||
bits<20> D1;
|
||||
bits<24> BD1;
|
||||
bits<8> I2;
|
||||
|
||||
let Inst{47-40} = op{15-8};
|
||||
let Inst{39-32} = I2;
|
||||
let Inst{31-28} = B1;
|
||||
let Inst{27-16} = D1{11-0};
|
||||
let Inst{15-8} = D1{19-12};
|
||||
let Inst{31-8} = BD1;
|
||||
let Inst{7-0} = op{7-0};
|
||||
|
||||
let Has20BitOffset = 1;
|
||||
|
@ -432,23 +401,23 @@ class InstSIY<bits<16> op, dag outs, dag ins, string asmstr, list<dag> pattern>
|
|||
|
||||
class InherentRRE<string mnemonic, bits<16> opcode, RegisterOperand cls,
|
||||
dag src>
|
||||
: InstRRE<opcode, (outs cls:$dst), (ins),
|
||||
mnemonic#"\t$dst",
|
||||
[(set cls:$dst, src)]> {
|
||||
: InstRRE<opcode, (outs cls:$R1), (ins),
|
||||
mnemonic#"\t$R1",
|
||||
[(set cls:$R1, src)]> {
|
||||
let R2 = 0;
|
||||
}
|
||||
|
||||
class LoadMultipleRSY<string mnemonic, bits<16> opcode, RegisterOperand cls>
|
||||
: InstRSY<opcode, (outs cls:$dst1, cls:$dst2), (ins bdaddr20only:$addr),
|
||||
mnemonic#"\t$dst1, $dst2, $addr", []> {
|
||||
: InstRSY<opcode, (outs cls:$R1, cls:$R3), (ins bdaddr20only:$BD2),
|
||||
mnemonic#"\t$R1, $R3, $BD2", []> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class StoreRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls>
|
||||
: InstRIL<opcode, (outs), (ins cls:$src, pcrel32:$addr),
|
||||
mnemonic#"\t$src, $addr",
|
||||
[(operator cls:$src, pcrel32:$addr)]> {
|
||||
: InstRIL<opcode, (outs), (ins cls:$R1, pcrel32:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(operator cls:$R1, pcrel32:$I2)]> {
|
||||
let mayStore = 1;
|
||||
// We want PC-relative addresses to be tried ahead of BD and BDX addresses.
|
||||
// However, BDXs have two extra operands and are therefore 6 units more
|
||||
|
@ -458,17 +427,17 @@ class StoreRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
|||
|
||||
class StoreRX<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdxaddr12only>
|
||||
: InstRX<opcode, (outs), (ins cls:$src, mode:$addr),
|
||||
mnemonic#"\t$src, $addr",
|
||||
[(operator cls:$src, mode:$addr)]> {
|
||||
: InstRX<opcode, (outs), (ins cls:$R1, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(operator cls:$R1, mode:$XBD2)]> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class StoreRXY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdxaddr20only>
|
||||
: InstRXY<opcode, (outs), (ins cls:$src, mode:$addr),
|
||||
mnemonic#"\t$src, $addr",
|
||||
[(operator cls:$src, mode:$addr)]> {
|
||||
: InstRXY<opcode, (outs), (ins cls:$R1, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(operator cls:$R1, mode:$XBD2)]> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
|
@ -483,32 +452,32 @@ multiclass StoreRXPair<string mnemonic, bits<8> rxOpcode, bits<16> rxyOpcode,
|
|||
}
|
||||
|
||||
class StoreMultipleRSY<string mnemonic, bits<16> opcode, RegisterOperand cls>
|
||||
: InstRSY<opcode, (outs), (ins cls:$from, cls:$to, bdaddr20only:$addr),
|
||||
mnemonic#"\t$from, $to, $addr", []> {
|
||||
: InstRSY<opcode, (outs), (ins cls:$R1, cls:$R3, bdaddr20only:$BD2),
|
||||
mnemonic#"\t$R1, $R3, $BD2", []> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class StoreSI<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
Immediate imm, AddressingMode mode = bdaddr12only>
|
||||
: InstSI<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator imm:$src, mode:$addr)]> {
|
||||
: InstSI<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator imm:$I2, mode:$BD1)]> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class StoreSIY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
Immediate imm, AddressingMode mode = bdaddr20only>
|
||||
: InstSIY<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator imm:$src, mode:$addr)]> {
|
||||
: InstSIY<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator imm:$I2, mode:$BD1)]> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class StoreSIL<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
Immediate imm>
|
||||
: InstSIL<opcode, (outs), (ins bdaddr12only:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator imm:$src, bdaddr12only:$addr)]> {
|
||||
: InstSIL<opcode, (outs), (ins bdaddr12only:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator imm:$I2, bdaddr12only:$BD1)]> {
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
|
@ -524,38 +493,38 @@ multiclass StoreSIPair<string mnemonic, bits<8> siOpcode, bits<16> siyOpcode,
|
|||
|
||||
class UnaryRR<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRR<opcode, (outs cls1:$dst), (ins cls2:$src),
|
||||
mnemonic#"\t$dst, $src",
|
||||
[(set cls1:$dst, (operator cls2:$src))]>;
|
||||
: InstRR<opcode, (outs cls1:$R1), (ins cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(set cls1:$R1, (operator cls2:$R2))]>;
|
||||
|
||||
class UnaryRRE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRRE<opcode, (outs cls1:$dst), (ins cls2:$src),
|
||||
mnemonic#"\t$dst, $src",
|
||||
[(set cls1:$dst, (operator cls2:$src))]>;
|
||||
: InstRRE<opcode, (outs cls1:$R1), (ins cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(set cls1:$R1, (operator cls2:$R2))]>;
|
||||
|
||||
class UnaryRRF<string mnemonic, bits<16> opcode, RegisterOperand cls1,
|
||||
RegisterOperand cls2>
|
||||
: InstRRF<opcode, (outs cls1:$dst), (ins cls2:$src, uimm8zx4:$mode),
|
||||
mnemonic#"\t$dst, $mode, $src", []>;
|
||||
: InstRRF<opcode, (outs cls1:$R1), (ins uimm8zx4:$R3, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R3, $R2", []>;
|
||||
|
||||
class UnaryRI<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRI<opcode, (outs cls:$dst), (ins imm:$src),
|
||||
mnemonic#"\t$dst, $src",
|
||||
[(set cls:$dst, (operator imm:$src))]>;
|
||||
: InstRI<opcode, (outs cls:$R1), (ins imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(set cls:$R1, (operator imm:$I2))]>;
|
||||
|
||||
class UnaryRIL<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRIL<opcode, (outs cls:$dst), (ins imm:$src),
|
||||
mnemonic#"\t$dst, $src",
|
||||
[(set cls:$dst, (operator imm:$src))]>;
|
||||
: InstRIL<opcode, (outs cls:$R1), (ins imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(set cls:$R1, (operator imm:$I2))]>;
|
||||
|
||||
class UnaryRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls>
|
||||
: InstRIL<opcode, (outs cls:$dst), (ins pcrel32:$addr),
|
||||
mnemonic#"\t$dst, $addr",
|
||||
[(set cls:$dst, (operator pcrel32:$addr))]> {
|
||||
: InstRIL<opcode, (outs cls:$R1), (ins pcrel32:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(set cls:$R1, (operator pcrel32:$I2))]> {
|
||||
let mayLoad = 1;
|
||||
// We want PC-relative addresses to be tried ahead of BD and BDX addresses.
|
||||
// However, BDXs have two extra operands and are therefore 6 units more
|
||||
|
@ -565,25 +534,25 @@ class UnaryRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
|||
|
||||
class UnaryRX<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdxaddr12only>
|
||||
: InstRX<opcode, (outs cls:$dst), (ins mode:$addr),
|
||||
mnemonic#"\t$dst, $addr",
|
||||
[(set cls:$dst, (operator mode:$addr))]> {
|
||||
: InstRX<opcode, (outs cls:$R1), (ins mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator mode:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class UnaryRXE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls>
|
||||
: InstRXE<opcode, (outs cls:$dst), (ins bdxaddr12only:$addr),
|
||||
mnemonic#"\t$dst, $addr",
|
||||
[(set cls:$dst, (operator bdxaddr12only:$addr))]> {
|
||||
: InstRXE<opcode, (outs cls:$R1), (ins bdxaddr12only:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator bdxaddr12only:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class UnaryRXY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdxaddr20only>
|
||||
: InstRXY<opcode, (outs cls:$dst), (ins mode:$addr),
|
||||
mnemonic#"\t$dst, $addr",
|
||||
[(set cls:$dst, (operator mode:$addr))]> {
|
||||
: InstRXY<opcode, (outs cls:$R1), (ins mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator mode:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
|
@ -599,83 +568,76 @@ multiclass UnaryRXPair<string mnemonic, bits<8> rxOpcode, bits<16> rxyOpcode,
|
|||
|
||||
class BinaryRR<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRR<opcode, (outs cls1:$dst), (ins cls1:$src1, cls2:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls1:$dst, (operator cls1:$src1, cls2:$src2))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRR<opcode, (outs cls1:$R1), (ins cls1:$R1src, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(set cls1:$R1, (operator cls1:$R1src, cls2:$R2))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
class BinaryRRE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRRE<opcode, (outs cls1:$dst), (ins cls1:$src1, cls2:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls1:$dst, (operator cls1:$src1, cls2:$src2))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRRE<opcode, (outs cls1:$R1), (ins cls1:$R1src, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(set cls1:$R1, (operator cls1:$R1src, cls2:$R2))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
// Here the assembly and dag operands are in natural order,
|
||||
// but the first input operand maps to R3 and the second to R2.
|
||||
// This is used for "CPSDR R1, R3, R2", which is equivalent to
|
||||
// R1 = copysign (R3, R2).
|
||||
//
|
||||
// Direct uses of the instruction must pass operands in encoding order --
|
||||
// R1, R2, R3 -- so they must pass the source operands in reverse order.
|
||||
class BinaryRevRRF<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRRF<opcode, (outs cls1:$dst), (ins cls2:$src2, cls1:$src1),
|
||||
mnemonic#"\t$dst, $src1, $src2",
|
||||
[(set cls1:$dst, (operator cls1:$src1, cls2:$src2))]>;
|
||||
class BinaryRRF<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRRF<opcode, (outs cls1:$R1), (ins cls1:$R3, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R3, $R2",
|
||||
[(set cls1:$R1, (operator cls1:$R3, cls2:$R2))]>;
|
||||
|
||||
class BinaryRI<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRI<opcode, (outs cls:$dst), (ins cls:$src1, imm:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, imm:$src2))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRI<opcode, (outs cls:$R1), (ins cls:$R1src, imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(set cls:$R1, (operator cls:$R1src, imm:$I2))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
class BinaryRIL<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRIL<opcode, (outs cls:$dst), (ins cls:$src1, imm:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, imm:$src2))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRIL<opcode, (outs cls:$R1), (ins cls:$R1src, imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(set cls:$R1, (operator cls:$R1src, imm:$I2))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
class BinaryRX<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load,
|
||||
AddressingMode mode = bdxaddr12only>
|
||||
: InstRX<opcode, (outs cls:$dst), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, (load mode:$src2)))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRX<opcode, (outs cls:$R1), (ins cls:$R1src, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator cls:$R1src, (load mode:$XBD2)))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class BinaryRXE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load>
|
||||
: InstRXE<opcode, (outs cls:$dst), (ins cls:$src1, bdxaddr12only:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1,
|
||||
(load bdxaddr12only:$src2)))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRXE<opcode, (outs cls:$R1), (ins cls:$R1src, bdxaddr12only:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator cls:$R1src,
|
||||
(load bdxaddr12only:$XBD2)))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class BinaryRXY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load,
|
||||
AddressingMode mode = bdxaddr20only>
|
||||
: InstRXY<opcode, (outs cls:$dst), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, (load mode:$src2)))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRXY<opcode, (outs cls:$R1), (ins cls:$R1src, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(set cls:$R1, (operator cls:$R1src, (load mode:$XBD2)))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
|
@ -693,18 +655,18 @@ multiclass BinaryRXPair<string mnemonic, bits<8> rxOpcode, bits<16> rxyOpcode,
|
|||
|
||||
class BinarySI<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
Operand imm, AddressingMode mode = bdaddr12only>
|
||||
: InstSI<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(store (operator (load mode:$addr), imm:$src), mode:$addr)]> {
|
||||
: InstSI<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(store (operator (load mode:$BD1), imm:$I2), mode:$BD1)]> {
|
||||
let mayLoad = 1;
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class BinarySIY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
Operand imm, AddressingMode mode = bdaddr20only>
|
||||
: InstSIY<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(store (operator (load mode:$addr), imm:$src), mode:$addr)]> {
|
||||
: InstSIY<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(store (operator (load mode:$BD1), imm:$I2), mode:$BD1)]> {
|
||||
let mayLoad = 1;
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
@ -722,49 +684,49 @@ multiclass BinarySIPair<string mnemonic, bits<8> siOpcode,
|
|||
|
||||
class ShiftRS<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode>
|
||||
: InstRS<opcode, (outs cls:$dst), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$dst, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, mode:$src2))]> {
|
||||
: InstRS<opcode, (outs cls:$R1), (ins cls:$R1src, mode:$BD2),
|
||||
mnemonic#"\t$R1, $BD2",
|
||||
[(set cls:$R1, (operator cls:$R1src, mode:$BD2))]> {
|
||||
let R3 = 0;
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
class ShiftRSY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode>
|
||||
: InstRSY<opcode, (outs cls:$dst), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$dst, $src1, $src2",
|
||||
[(set cls:$dst, (operator cls:$src1, mode:$src2))]>;
|
||||
: InstRSY<opcode, (outs cls:$R1), (ins cls:$R3, mode:$BD2),
|
||||
mnemonic#"\t$R1, $R3, $BD2",
|
||||
[(set cls:$R1, (operator cls:$R3, mode:$BD2))]>;
|
||||
|
||||
class CompareRR<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRR<opcode, (outs), (ins cls1:$src1, cls2:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls1:$src1, cls2:$src2)]>;
|
||||
: InstRR<opcode, (outs), (ins cls1:$R1, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(operator cls1:$R1, cls2:$R2)]>;
|
||||
|
||||
class CompareRRE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls1, RegisterOperand cls2>
|
||||
: InstRRE<opcode, (outs), (ins cls1:$src1, cls2:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls1:$src1, cls2:$src2)]>;
|
||||
: InstRRE<opcode, (outs), (ins cls1:$R1, cls2:$R2),
|
||||
mnemonic#"\t$R1, $R2",
|
||||
[(operator cls1:$R1, cls2:$R2)]>;
|
||||
|
||||
class CompareRI<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRI<opcode, (outs), (ins cls:$src1, imm:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, imm:$src2)]>;
|
||||
: InstRI<opcode, (outs), (ins cls:$R1, imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(operator cls:$R1, imm:$I2)]>;
|
||||
|
||||
class CompareRIL<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, Immediate imm>
|
||||
: InstRIL<opcode, (outs), (ins cls:$src1, imm:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, imm:$src2)]>;
|
||||
: InstRIL<opcode, (outs), (ins cls:$R1, imm:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(operator cls:$R1, imm:$I2)]>;
|
||||
|
||||
class CompareRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load>
|
||||
: InstRIL<opcode, (outs), (ins cls:$src1, pcrel32:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, (load pcrel32:$src2))]> {
|
||||
: InstRIL<opcode, (outs), (ins cls:$R1, pcrel32:$I2),
|
||||
mnemonic#"\t$R1, $I2",
|
||||
[(operator cls:$R1, (load pcrel32:$I2))]> {
|
||||
let mayLoad = 1;
|
||||
// We want PC-relative addresses to be tried ahead of BD and BDX addresses.
|
||||
// However, BDXs have two extra operands and are therefore 6 units more
|
||||
|
@ -775,26 +737,26 @@ class CompareRILPC<string mnemonic, bits<12> opcode, SDPatternOperator operator,
|
|||
class CompareRX<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load,
|
||||
AddressingMode mode = bdxaddr12only>
|
||||
: InstRX<opcode, (outs), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, (load mode:$src2))]> {
|
||||
: InstRX<opcode, (outs), (ins cls:$R1, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(operator cls:$R1, (load mode:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class CompareRXE<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load>
|
||||
: InstRXE<opcode, (outs), (ins cls:$src1, bdxaddr12only:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, (load bdxaddr12only:$src2))]> {
|
||||
: InstRXE<opcode, (outs), (ins cls:$R1, bdxaddr12only:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(operator cls:$R1, (load bdxaddr12only:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class CompareRXY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load,
|
||||
AddressingMode mode = bdxaddr20only>
|
||||
: InstRXY<opcode, (outs), (ins cls:$src1, mode:$src2),
|
||||
mnemonic#"\t$src1, $src2",
|
||||
[(operator cls:$src1, (load mode:$src2))]> {
|
||||
: InstRXY<opcode, (outs), (ins cls:$R1, mode:$XBD2),
|
||||
mnemonic#"\t$R1, $XBD2",
|
||||
[(operator cls:$R1, (load mode:$XBD2))]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
|
@ -814,26 +776,26 @@ multiclass CompareRXPair<string mnemonic, bits<8> rxOpcode, bits<16> rxyOpcode,
|
|||
class CompareSI<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
SDPatternOperator load, Immediate imm,
|
||||
AddressingMode mode = bdaddr12only>
|
||||
: InstSI<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator (load mode:$addr), imm:$src)]> {
|
||||
: InstSI<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator (load mode:$BD1), imm:$I2)]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class CompareSIL<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
SDPatternOperator load, Immediate imm>
|
||||
: InstSIL<opcode, (outs), (ins bdaddr12only:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator (load bdaddr12only:$addr), imm:$src)]> {
|
||||
: InstSIL<opcode, (outs), (ins bdaddr12only:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator (load bdaddr12only:$BD1), imm:$I2)]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class CompareSIY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
SDPatternOperator load, Immediate imm,
|
||||
AddressingMode mode = bdaddr20only>
|
||||
: InstSIY<opcode, (outs), (ins mode:$addr, imm:$src),
|
||||
mnemonic#"\t$addr, $src",
|
||||
[(operator (load mode:$addr), imm:$src)]> {
|
||||
: InstSIY<opcode, (outs), (ins mode:$BD1, imm:$I2),
|
||||
mnemonic#"\t$BD1, $I2",
|
||||
[(operator (load mode:$BD1), imm:$I2)]> {
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
|
@ -851,43 +813,43 @@ multiclass CompareSIPair<string mnemonic, bits<8> siOpcode, bits<16> siyOpcode,
|
|||
|
||||
class TernaryRRD<string mnemonic, bits<16> opcode,
|
||||
SDPatternOperator operator, RegisterOperand cls>
|
||||
: InstRRD<opcode, (outs cls:$dst), (ins cls:$src1, cls:$src2, cls:$src3),
|
||||
mnemonic#"\t$dst, $src2, $src3",
|
||||
[(set cls:$dst, (operator cls:$src1, cls:$src2, cls:$src3))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRRD<opcode, (outs cls:$R1), (ins cls:$R1src, cls:$R3, cls:$R2),
|
||||
mnemonic#"\t$R1, $R3, $R2",
|
||||
[(set cls:$R1, (operator cls:$R1src, cls:$R3, cls:$R2))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
class TernaryRXF<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, SDPatternOperator load>
|
||||
: InstRXF<opcode, (outs cls:$dst),
|
||||
(ins cls:$src1, cls:$src2, bdxaddr12only:$src3),
|
||||
mnemonic#"\t$dst, $src2, $src3",
|
||||
[(set cls:$dst, (operator cls:$src1, cls:$src2,
|
||||
(load bdxaddr12only:$src3)))]> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRXF<opcode, (outs cls:$R1),
|
||||
(ins cls:$R1src, cls:$R3, bdxaddr12only:$XBD2),
|
||||
mnemonic#"\t$R1, $R3, $XBD2",
|
||||
[(set cls:$R1, (operator cls:$R1src, cls:$R3,
|
||||
(load bdxaddr12only:$XBD2)))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
}
|
||||
|
||||
class CmpSwapRS<string mnemonic, bits<8> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdaddr12only>
|
||||
: InstRS<opcode, (outs cls:$dst), (ins cls:$old, cls:$new, mode:$ptr),
|
||||
mnemonic#"\t$dst, $new, $ptr",
|
||||
[(set cls:$dst, (operator mode:$ptr, cls:$old, cls:$new))]> {
|
||||
let Constraints = "$old = $dst";
|
||||
let DisableEncoding = "$old";
|
||||
: InstRS<opcode, (outs cls:$R1), (ins cls:$R1src, cls:$R3, mode:$BD2),
|
||||
mnemonic#"\t$R1, $R3, $BD2",
|
||||
[(set cls:$R1, (operator mode:$BD2, cls:$R1src, cls:$R3))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
||||
class CmpSwapRSY<string mnemonic, bits<16> opcode, SDPatternOperator operator,
|
||||
RegisterOperand cls, AddressingMode mode = bdaddr20only>
|
||||
: InstRSY<opcode, (outs cls:$dst), (ins cls:$old, cls:$new, mode:$ptr),
|
||||
mnemonic#"\t$dst, $new, $ptr",
|
||||
[(set cls:$dst, (operator mode:$ptr, cls:$old, cls:$new))]> {
|
||||
let Constraints = "$old = $dst";
|
||||
let DisableEncoding = "$old";
|
||||
: InstRSY<opcode, (outs cls:$R1), (ins cls:$R1src, cls:$R3, mode:$BD2),
|
||||
mnemonic#"\t$R1, $R3, $BD2",
|
||||
[(set cls:$R1, (operator mode:$BD2, cls:$R1src, cls:$R3))]> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
let mayLoad = 1;
|
||||
let mayStore = 1;
|
||||
}
|
||||
|
@ -904,12 +866,12 @@ multiclass CmpSwapRSPair<string mnemonic, bits<8> rsOpcode, bits<16> rsyOpcode,
|
|||
|
||||
class RotateSelectRIEf<string mnemonic, bits<16> opcode, RegisterOperand cls1,
|
||||
RegisterOperand cls2>
|
||||
: InstRIEf<opcode, (outs cls1:$dst),
|
||||
(ins cls1:$src1, cls2:$src2,
|
||||
uimm8zx6:$imm1, uimm8zx6:$imm2, uimm8zx6:$imm3),
|
||||
mnemonic#"\t$dst, $src2, $imm1, $imm2, $imm3", []> {
|
||||
let Constraints = "$src1 = $dst";
|
||||
let DisableEncoding = "$src1";
|
||||
: InstRIEf<opcode, (outs cls1:$R1),
|
||||
(ins cls1:$R1src, cls2:$R2,
|
||||
uimm8zx6:$I3, uimm8zx6:$I4, uimm8zx6:$I5),
|
||||
mnemonic#"\t$R1, $R2, $I3, $I4, $I5", []> {
|
||||
let Constraints = "$R1 = $R1src";
|
||||
let DisableEncoding = "$R1src";
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
|
|
@ -42,20 +42,20 @@ let isReturn = 1, isTerminator = 1, isBarrier = 1, hasCtrlDep = 1,
|
|||
// Unconditional branches. R1 is the condition-code mask (all 1s).
|
||||
let isBranch = 1, isTerminator = 1, isBarrier = 1, R1 = 15 in {
|
||||
let isIndirectBranch = 1 in
|
||||
def BR : InstRR<0x07, (outs), (ins ADDR64:$dst),
|
||||
"br\t$dst", [(brind ADDR64:$dst)]>;
|
||||
def BR : InstRR<0x07, (outs), (ins ADDR64:$R2),
|
||||
"br\t$R2", [(brind ADDR64:$R2)]>;
|
||||
|
||||
// An assembler extended mnemonic for BRC. Use a separate instruction for
|
||||
// the asm parser, so that we don't relax Js to external symbols into JGs.
|
||||
let isCodeGenOnly = 1 in
|
||||
def J : InstRI<0xA74, (outs), (ins brtarget16:$dst), "j\t$dst", []>;
|
||||
def J : InstRI<0xA74, (outs), (ins brtarget16:$I2), "j\t$I2", []>;
|
||||
let isAsmParserOnly = 1 in
|
||||
def AsmJ : InstRI<0xA74, (outs), (ins brtarget16:$dst), "j\t$dst", []>;
|
||||
def AsmJ : InstRI<0xA74, (outs), (ins brtarget16:$I2), "j\t$I2", []>;
|
||||
|
||||
// An assembler extended mnemonic for BRCL. (The extension is "G"
|
||||
// rather than "L" because "JL" is "Jump if Less".)
|
||||
def JG : InstRIL<0xC04, (outs), (ins brtarget32:$dst),
|
||||
"jg\t$dst", [(br bb:$dst)]>;
|
||||
def JG : InstRIL<0xC04, (outs), (ins brtarget32:$I2),
|
||||
"jg\t$I2", [(br bb:$I2)]>;
|
||||
}
|
||||
|
||||
// Conditional branches. It's easier for LLVM to handle these branches
|
||||
|
@ -64,24 +64,24 @@ let isBranch = 1, isTerminator = 1, isBarrier = 1, R1 = 15 in {
|
|||
// JE and JLH when writing out the assembly though.
|
||||
multiclass CondBranches<Operand imm, string short, string long> {
|
||||
let isBranch = 1, isTerminator = 1, Uses = [PSW] in {
|
||||
def "" : InstRI<0xA74, (outs), (ins imm:$cond, brtarget16:$dst), short, []>;
|
||||
def L : InstRIL<0xC04, (outs), (ins imm:$cond, brtarget32:$dst), long, []>;
|
||||
def "" : InstRI<0xA74, (outs), (ins imm:$R1, brtarget16:$I2), short, []>;
|
||||
def L : InstRIL<0xC04, (outs), (ins imm:$R1, brtarget32:$I2), long, []>;
|
||||
}
|
||||
}
|
||||
let isCodeGenOnly = 1 in
|
||||
defm BRC : CondBranches<cond4, "j$cond\t$dst", "jg$cond\t$dst">;
|
||||
defm BRC : CondBranches<cond4, "j$R1\t$I2", "jg$R1\t$I2">;
|
||||
let isAsmParserOnly = 1 in
|
||||
defm AsmBRC : CondBranches<uimm8zx4, "brc\t$cond, $dst", "brcl\t$cond, $dst">;
|
||||
defm AsmBRC : CondBranches<uimm8zx4, "brc\t$R1, $I2", "brcl\t$R1, $I2">;
|
||||
|
||||
def : Pat<(z_br_ccmask cond4:$cond, bb:$dst), (BRCL cond4:$cond, bb:$dst)>;
|
||||
|
||||
// Define AsmParser mnemonics for each condition code.
|
||||
multiclass CondExtendedMnemonic<bits<4> Cond, string name> {
|
||||
let R1 = Cond in {
|
||||
def "" : InstRI<0xA74, (outs), (ins brtarget16:$dst),
|
||||
"j"##name##"\t$dst", []>;
|
||||
def L : InstRIL<0xC04, (outs), (ins brtarget32:$dst),
|
||||
"jg"##name##"\t$dst", []>;
|
||||
def "" : InstRI<0xA74, (outs), (ins brtarget16:$I2),
|
||||
"j"##name##"\t$I2", []>;
|
||||
def L : InstRIL<0xC04, (outs), (ins brtarget32:$I2),
|
||||
"jg"##name##"\t$I2", []>;
|
||||
}
|
||||
}
|
||||
let isAsmParserOnly = 1 in {
|
||||
|
@ -112,23 +112,23 @@ def Select64 : SelectWrapper<GR64>;
|
|||
let isCall = 1, Defs = [R0D, R1D, R2D, R3D, R4D, R5D, R14D,
|
||||
F0D, F1D, F2D, F3D, F4D, F5D, F6D, F7D],
|
||||
R1 = 14, isCodeGenOnly = 1 in {
|
||||
def BRAS : InstRI<0xA75, (outs), (ins pcrel16call:$dst, variable_ops),
|
||||
"bras\t%r14, $dst", []>;
|
||||
def BRASL : InstRIL<0xC05, (outs), (ins pcrel32call:$dst, variable_ops),
|
||||
"brasl\t%r14, $dst", [(z_call pcrel32call:$dst)]>;
|
||||
def BASR : InstRR<0x0D, (outs), (ins ADDR64:$dst, variable_ops),
|
||||
"basr\t%r14, $dst", [(z_call ADDR64:$dst)]>;
|
||||
def BRAS : InstRI<0xA75, (outs), (ins pcrel16call:$I2, variable_ops),
|
||||
"bras\t%r14, $I2", []>;
|
||||
def BRASL : InstRIL<0xC05, (outs), (ins pcrel32call:$I2, variable_ops),
|
||||
"brasl\t%r14, $I2", [(z_call pcrel32call:$I2)]>;
|
||||
def BASR : InstRR<0x0D, (outs), (ins ADDR64:$R2, variable_ops),
|
||||
"basr\t%r14, $R2", [(z_call ADDR64:$R2)]>;
|
||||
}
|
||||
|
||||
// Define the general form of the call instructions for the asm parser.
|
||||
// These instructions don't hard-code %r14 as the return address register.
|
||||
let isAsmParserOnly = 1 in {
|
||||
def AsmBRAS : InstRI<0xA75, (outs), (ins GR64:$save, brtarget16:$dst),
|
||||
"bras\t$save, $dst", []>;
|
||||
def AsmBRASL : InstRIL<0xC05, (outs), (ins GR64:$save, brtarget32:$dst),
|
||||
"brasl\t$save, $dst", []>;
|
||||
def AsmBASR : InstRR<0x0D, (outs), (ins GR64:$save, ADDR64:$dst),
|
||||
"basr\t$save, $dst", []>;
|
||||
def AsmBRAS : InstRI<0xA75, (outs), (ins GR64:$R1, brtarget16:$I2),
|
||||
"bras\t$R1, $I2", []>;
|
||||
def AsmBRASL : InstRIL<0xC05, (outs), (ins GR64:$R1, brtarget32:$I2),
|
||||
"brasl\t$R1, $I2", []>;
|
||||
def AsmBASR : InstRR<0x0D, (outs), (ins GR64:$R1, ADDR64:$R2),
|
||||
"basr\t$R1, $R2", []>;
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
@ -337,21 +337,21 @@ def STRVG : StoreRXY<"strvg", 0xE32F, storeu<bswap>, GR64>;
|
|||
// Load BDX-style addresses.
|
||||
let neverHasSideEffects = 1, Function = "la" in {
|
||||
let PairType = "12" in
|
||||
def LA : InstRX<0x41, (outs GR64:$dst), (ins laaddr12pair:$src),
|
||||
"la\t$dst, $src",
|
||||
[(set GR64:$dst, laaddr12pair:$src)]>;
|
||||
def LA : InstRX<0x41, (outs GR64:$R1), (ins laaddr12pair:$XBD2),
|
||||
"la\t$R1, $XBD2",
|
||||
[(set GR64:$R1, laaddr12pair:$XBD2)]>;
|
||||
let PairType = "20" in
|
||||
def LAY : InstRXY<0xE371, (outs GR64:$dst), (ins laaddr20pair:$src),
|
||||
"lay\t$dst, $src",
|
||||
[(set GR64:$dst, laaddr20pair:$src)]>;
|
||||
def LAY : InstRXY<0xE371, (outs GR64:$R1), (ins laaddr20pair:$XBD2),
|
||||
"lay\t$R1, $XBD2",
|
||||
[(set GR64:$R1, laaddr20pair:$XBD2)]>;
|
||||
}
|
||||
|
||||
// Load a PC-relative address. There's no version of this instruction
|
||||
// with a 16-bit offset, so there's no relaxation.
|
||||
let neverHasSideEffects = 1 in {
|
||||
def LARL : InstRIL<0xC00, (outs GR64:$dst), (ins pcrel32:$src),
|
||||
"larl\t$dst, $src",
|
||||
[(set GR64:$dst, pcrel32:$src)]>;
|
||||
def LARL : InstRIL<0xC00, (outs GR64:$R1), (ins pcrel32:$I2),
|
||||
"larl\t$R1, $I2",
|
||||
[(set GR64:$R1, pcrel32:$I2)]>;
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
@ -903,9 +903,9 @@ let Defs = [PSW] in {
|
|||
// Read a 32-bit access register into a GR32. As with all GR32 operations,
|
||||
// the upper 32 bits of the enclosing GR64 remain unchanged, which is useful
|
||||
// when a 64-bit address is stored in a pair of access registers.
|
||||
def EAR : InstRRE<0xB24F, (outs GR32:$dst), (ins access_reg:$src),
|
||||
"ear\t$dst, $src",
|
||||
[(set GR32:$dst, (z_extract_access access_reg:$src))]>;
|
||||
def EAR : InstRRE<0xB24F, (outs GR32:$R1), (ins access_reg:$R2),
|
||||
"ear\t$R1, $R2",
|
||||
[(set GR32:$R1, (z_extract_access access_reg:$R2))]>;
|
||||
|
||||
// Find leftmost one, AKA count leading zeros. The instruction actually
|
||||
// returns a pair of GR64s, the first giving the number of leading zeros
|
||||
|
|
|
@ -45,11 +45,13 @@ class AddressAsmOperand<string format, string bitsize, string dispsize>
|
|||
}
|
||||
|
||||
// Constructs both a DAG pattern and instruction operand for an addressing mode.
|
||||
// The mode is selected by custom code in selectTYPE...SUFFIX(). The address
|
||||
// registers have BITSIZE bits and displacements have DISPSIZE bits. NUMOPS is
|
||||
// the number of operands that make up an address and OPERANDS lists the types
|
||||
// of those operands using (ops ...). FORMAT is the type of addressing mode,
|
||||
// which needs to match the names used in AddressAsmOperand.
|
||||
// The mode is selected by custom code in select<TYPE><DISPSIZE><SUFFIX>()
|
||||
// and encoded by custom code in get<FORMAT><DISPSIZE>Encoding().
|
||||
// The address registers have BITSIZE bits and displacements have
|
||||
// DISPSIZE bits. NUMOPS is the number of operands that make up an
|
||||
// address and OPERANDS lists the types of those operands using (ops ...).
|
||||
// FORMAT is the type of addressing mode, which needs to match the names
|
||||
// used in AddressAsmOperand.
|
||||
class AddressingMode<string type, string bitsize, string dispsize,
|
||||
string suffix, int numops, string format, dag operands>
|
||||
: ComplexPattern<!cast<ValueType>("i"##bitsize), numops,
|
||||
|
@ -57,6 +59,7 @@ class AddressingMode<string type, string bitsize, string dispsize,
|
|||
[add, sub, or, frameindex, z_adjdynalloc]>,
|
||||
Operand<!cast<ValueType>("i"##bitsize)> {
|
||||
let PrintMethod = "print"##format##"Operand";
|
||||
let EncoderMethod = "get"##format##dispsize##"Encoding";
|
||||
let MIOperandInfo = operands;
|
||||
let ParserMatchClass =
|
||||
!cast<AddressAsmOperand>(format##bitsize##"Disp"##dispsize);
|
||||
|
|
Loading…
Reference in New Issue