2012-02-18 12:03:15 +00:00
|
|
|
//===-- ARMInstrThumb2.td - Thumb2 support for ARM ---------*- tablegen -*-===//
|
2009-05-29 23:41:08 +00:00
|
|
|
//
|
2019-01-19 08:50:56 +00:00
|
|
|
// 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
|
2009-05-29 23:41:08 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file describes the Thumb2 instruction set.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2009-07-10 01:54:42 +00:00
|
|
|
// IT block predicate field
|
2011-08-26 21:43:41 +00:00
|
|
|
def it_pred_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "ITCondCode";
|
|
|
|
let ParserMethod = "parseITCondCode";
|
|
|
|
}
|
2009-07-10 01:54:42 +00:00
|
|
|
def it_pred : Operand<i32> {
|
2010-03-02 17:57:15 +00:00
|
|
|
let PrintMethod = "printMandatoryPredicateOperand";
|
2011-08-26 21:43:41 +00:00
|
|
|
let ParserMatchClass = it_pred_asmoperand;
|
2009-07-10 01:54:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// IT block condition mask
|
2011-08-26 21:43:41 +00:00
|
|
|
def it_mask_asmoperand : AsmOperandClass { let Name = "ITMask"; }
|
2009-07-10 01:54:42 +00:00
|
|
|
def it_mask : Operand<i32> {
|
|
|
|
let PrintMethod = "printThumbITMask";
|
2011-08-26 21:43:41 +00:00
|
|
|
let ParserMatchClass = it_mask_asmoperand;
|
[ARM] Refactor handling of IT mask operands.
During assembly, the mask operand to an IT instruction (storing the
sequence of T/E for 'Then' and 'Else') is parsed out of the mnemonic
into a representation that encodes 'Then' and 'Else' in the same way
regardless of the condition code. At some point during encoding it has
to be converted into the instruction encoding used in the
architecture, in which the mask encodes a sequence of replacement
low-order bits for the condition code, so that which bit value means
'then' and which 'else' depends on whether the original condition code
had its low bit set.
Previously, that transformation was done by processInstruction(), half
way through assembly. So an MCOperand storing an IT mask would
sometimes store it in one format, and sometimes in the other,
depending on where in the assembly pipeline you were. You can see this
in diagnostics from `llvm-mc -debug -triple=thumbv8a -show-inst`, for
example: if you give it an instruction such as `itete eq`, you'd see
an `<MCOperand Imm:5>` in a diagnostic become `<MCOperand Imm:11>` in
the final output.
Having the same data structure store values with time-dependent
semantics is confusing already, and it will get more confusing when we
introduce the MVE VPT instruction which reuses the Then/Else bitmask
idea in a different context. So I'm refactoring: now, all `ARMOperand`
and `MCOperand` representations of an IT mask work exactly the same
way, namely, 0 means 'Then' and 1 means 'Else', regardless of what
original predicate is being referred to. The architectural encoding of
IT that depends on the original condition is now constructed at the
point when we turn the `MCOperand` into the final instruction bit
pattern, and decoded similarly in the disassembler.
The previous condition-independent parse-time format used 0 for Else
and 1 for Then. I've taken the opportunity to flip the sense of it
while I'm changing all of this anyway, because it seems to me more
natural to use 0 for 'leave the starting condition unchanged' and 1
for 'invert it', as if those bits were an XOR mask.
Reviewers: ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63219
llvm-svn: 363244
2019-06-13 10:01:52 +00:00
|
|
|
let EncoderMethod = "getITMaskOpValue";
|
2009-07-10 01:54:42 +00:00
|
|
|
}
|
|
|
|
|
2011-09-26 21:06:22 +00:00
|
|
|
// t2_shift_imm: An integer that encodes a shift amount and the type of shift
|
|
|
|
// (asr or lsl). The 6-bit immediate encodes as:
|
|
|
|
// {5} 0 ==> lsl
|
|
|
|
// 1 asr
|
|
|
|
// {4-0} imm5 shift amount.
|
|
|
|
// asr #32 not allowed
|
|
|
|
def t2_shift_imm : Operand<i32> {
|
|
|
|
let PrintMethod = "printShiftImmOperand";
|
|
|
|
let ParserMatchClass = ShifterImmAsmOperand;
|
|
|
|
let DecoderMethod = "DecodeT2ShifterImmOperand";
|
|
|
|
}
|
|
|
|
|
2019-06-11 12:04:32 +00:00
|
|
|
def mve_shift_imm : AsmOperandClass {
|
|
|
|
let Name = "MVELongShift";
|
|
|
|
let RenderMethod = "addImmOperands";
|
|
|
|
let DiagnosticString = "operand must be an immediate in the range [1,32]";
|
|
|
|
}
|
2019-09-25 10:16:48 +00:00
|
|
|
def long_shift : Operand<i32>,
|
|
|
|
ImmLeaf<i32, [{ return Imm > 0 && Imm <= 32; }]> {
|
2019-06-11 12:04:32 +00:00
|
|
|
let ParserMatchClass = mve_shift_imm;
|
|
|
|
let DecoderMethod = "DecodeLongShiftOperand";
|
|
|
|
}
|
|
|
|
|
2009-06-17 18:13:58 +00:00
|
|
|
// Shifted operands. No register controlled shifts for Thumb2.
|
|
|
|
// Note: We do not support rrx shifted operands yet.
|
|
|
|
def t2_so_reg : Operand<i32>, // reg imm
|
2015-09-07 11:45:18 +00:00
|
|
|
ComplexPattern<i32, 2, "SelectShiftImmShifterOperand",
|
2009-06-17 18:13:58 +00:00
|
|
|
[shl,srl,sra,rotr]> {
|
2010-11-15 05:19:05 +00:00
|
|
|
let EncoderMethod = "getT2SORegOpValue";
|
2009-06-27 02:26:13 +00:00
|
|
|
let PrintMethod = "printT2SOOperand";
|
2011-08-22 23:10:16 +00:00
|
|
|
let DecoderMethod = "DecodeSORegImmOperand";
|
2011-08-31 18:23:08 +00:00
|
|
|
let ParserMatchClass = ShiftedImmAsmOperand;
|
|
|
|
let MIOperandInfo = (ops rGPR, i32imm);
|
2009-06-17 18:13:58 +00:00
|
|
|
}
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
// t2_so_imm_not_XFORM - Return the complement of a t2_so_imm value
|
|
|
|
def t2_so_imm_not_XFORM : SDNodeXForm<imm, [{
|
2015-04-28 14:05:47 +00:00
|
|
|
return CurDAG->getTargetConstant(~((uint32_t)N->getZExtValue()), SDLoc(N),
|
|
|
|
MVT::i32);
|
2009-06-17 18:13:58 +00:00
|
|
|
}]>;
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
// t2_so_imm_neg_XFORM - Return the negation of a t2_so_imm value
|
|
|
|
def t2_so_imm_neg_XFORM : SDNodeXForm<imm, [{
|
2015-04-28 14:05:47 +00:00
|
|
|
return CurDAG->getTargetConstant(-((int)N->getZExtValue()), SDLoc(N),
|
|
|
|
MVT::i32);
|
2009-06-23 17:48:47 +00:00
|
|
|
}]>;
|
|
|
|
|
2012-06-18 14:51:32 +00:00
|
|
|
// so_imm_notSext_XFORM - Return a so_imm value packed into the format
|
|
|
|
// described for so_imm_notSext def below, with sign extension from 16
|
|
|
|
// bits.
|
|
|
|
def t2_so_imm_notSext16_XFORM : SDNodeXForm<imm, [{
|
|
|
|
APInt apIntN = N->getAPIntValue();
|
|
|
|
unsigned N16bitSignExt = apIntN.trunc(16).sext(32).getZExtValue();
|
2015-04-28 14:05:47 +00:00
|
|
|
return CurDAG->getTargetConstant(~N16bitSignExt, SDLoc(N), MVT::i32);
|
2012-06-18 14:51:32 +00:00
|
|
|
}]>;
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
// t2_so_imm - Match a 32-bit immediate operand, which is an
|
|
|
|
// 8-bit immediate rotated by an arbitrary number of bits, or an 8-bit
|
2011-02-07 17:43:06 +00:00
|
|
|
// immediate splatted into multiple bytes of the word.
|
2017-04-03 14:50:04 +00:00
|
|
|
def t2_so_imm_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "T2SOImm";
|
|
|
|
let RenderMethod = "addImmOperands";
|
|
|
|
|
|
|
|
}
|
2011-04-29 22:48:03 +00:00
|
|
|
def t2_so_imm : Operand<i32>, ImmLeaf<i32, [{
|
|
|
|
return ARM_AM::getT2SOImmVal(Imm) != -1;
|
|
|
|
}]> {
|
2011-06-27 23:54:06 +00:00
|
|
|
let ParserMatchClass = t2_so_imm_asmoperand;
|
2010-11-15 05:19:05 +00:00
|
|
|
let EncoderMethod = "getT2SOImmOpValue";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2SOImm";
|
2010-11-12 21:12:40 +00:00
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2010-02-16 21:07:46 +00:00
|
|
|
// t2_so_imm_not - Match an immediate that is a complement
|
2009-06-23 17:48:47 +00:00
|
|
|
// of a t2_so_imm.
|
2011-10-28 22:36:30 +00:00
|
|
|
// Note: this pattern doesn't require an encoder method and such, as it's
|
|
|
|
// only used on aliases (Pat<> and InstAlias<>). The actual encoding
|
|
|
|
// is handled by the destination instructions, which use t2_so_imm.
|
|
|
|
def t2_so_imm_not_asmoperand : AsmOperandClass { let Name = "T2SOImmNot"; }
|
2011-12-08 00:31:07 +00:00
|
|
|
def t2_so_imm_not : Operand<i32>, PatLeaf<(imm), [{
|
2009-07-08 21:03:57 +00:00
|
|
|
return ARM_AM::getT2SOImmVal(~((uint32_t)N->getZExtValue())) != -1;
|
2011-10-28 22:36:30 +00:00
|
|
|
}], t2_so_imm_not_XFORM> {
|
|
|
|
let ParserMatchClass = t2_so_imm_not_asmoperand;
|
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2012-06-18 14:51:32 +00:00
|
|
|
// t2_so_imm_notSext - match an immediate that is a complement of a t2_so_imm
|
|
|
|
// if the upper 16 bits are zero.
|
|
|
|
def t2_so_imm_notSext : Operand<i32>, PatLeaf<(imm), [{
|
|
|
|
APInt apIntN = N->getAPIntValue();
|
|
|
|
if (!apIntN.isIntN(16)) return false;
|
|
|
|
unsigned N16bitSignExt = apIntN.trunc(16).sext(32).getZExtValue();
|
|
|
|
return ARM_AM::getT2SOImmVal(~N16bitSignExt) != -1;
|
|
|
|
}], t2_so_imm_notSext16_XFORM> {
|
|
|
|
let ParserMatchClass = t2_so_imm_not_asmoperand;
|
|
|
|
}
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
// t2_so_imm_neg - Match an immediate that is a negation of a t2_so_imm.
|
2011-12-08 00:31:07 +00:00
|
|
|
def t2_so_imm_neg_asmoperand : AsmOperandClass { let Name = "T2SOImmNeg"; }
|
2017-03-22 23:12:59 +00:00
|
|
|
def t2_so_imm_neg : Operand<i32>, ImmLeaf<i32, [{
|
|
|
|
return Imm && ARM_AM::getT2SOImmVal(-(uint32_t)Imm) != -1;
|
2011-12-08 00:31:07 +00:00
|
|
|
}], t2_so_imm_neg_XFORM> {
|
|
|
|
let ParserMatchClass = t2_so_imm_neg_asmoperand;
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2017-04-03 14:50:04 +00:00
|
|
|
/// imm0_4095 predicate - True if the 32-bit immediate is in the range [0,4095].
|
|
|
|
def imm0_4095_asmoperand: ImmAsmOperand<0,4095> { let Name = "Imm0_4095"; }
|
2012-04-05 20:57:13 +00:00
|
|
|
def imm0_4095 : Operand<i32>, ImmLeaf<i32, [{
|
2011-04-28 05:49:04 +00:00
|
|
|
return Imm >= 0 && Imm < 4096;
|
2012-04-05 20:57:13 +00:00
|
|
|
}]> {
|
|
|
|
let ParserMatchClass = imm0_4095_asmoperand;
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2012-04-05 20:57:13 +00:00
|
|
|
def imm0_4095_neg_asmoperand: AsmOperandClass { let Name = "Imm0_4095Neg"; }
|
|
|
|
def imm0_4095_neg : Operand<i32>, PatLeaf<(i32 imm), [{
|
2010-02-16 21:07:46 +00:00
|
|
|
return (uint32_t)(-N->getZExtValue()) < 4096;
|
2012-04-05 20:57:13 +00:00
|
|
|
}], imm_neg_XFORM> {
|
|
|
|
let ParserMatchClass = imm0_4095_neg_asmoperand;
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2012-11-14 19:39:15 +00:00
|
|
|
def imm1_255_neg : PatLeaf<(i32 imm), [{
|
|
|
|
uint32_t Val = -N->getZExtValue();
|
|
|
|
return (Val > 0 && Val < 255);
|
2010-02-16 21:07:46 +00:00
|
|
|
}], imm_neg_XFORM>;
|
2009-08-04 01:41:15 +00:00
|
|
|
|
2010-07-14 17:45:16 +00:00
|
|
|
def imm0_255_not : PatLeaf<(i32 imm), [{
|
|
|
|
return (uint32_t)(~N->getZExtValue()) < 255;
|
2017-03-10 13:21:12 +00:00
|
|
|
}], imm_not_XFORM>;
|
2010-07-14 17:45:16 +00:00
|
|
|
|
2011-04-29 14:18:15 +00:00
|
|
|
def lo5AllOne : PatLeaf<(i32 imm), [{
|
|
|
|
// Returns true if all low 5-bits are 1.
|
|
|
|
return (((uint32_t)N->getZExtValue()) & 0x1FUL) == 0x1FUL;
|
|
|
|
}]>;
|
|
|
|
|
2009-06-29 07:51:04 +00:00
|
|
|
// Define Thumb2 specific addressing modes.
|
|
|
|
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
// t2_addr_offset_none := reg
|
|
|
|
def MemNoOffsetT2AsmOperand
|
|
|
|
: AsmOperandClass { let Name = "MemNoOffsetT2"; }
|
|
|
|
def t2_addr_offset_none : MemOperand {
|
|
|
|
let PrintMethod = "printAddrMode7Operand";
|
|
|
|
let DecoderMethod = "DecodeGPRnopcRegisterClass";
|
|
|
|
let ParserMatchClass = MemNoOffsetT2AsmOperand;
|
|
|
|
let MIOperandInfo = (ops GPRnopc:$base);
|
|
|
|
}
|
|
|
|
|
|
|
|
// t2_nosp_addr_offset_none := reg
|
|
|
|
def MemNoOffsetT2NoSpAsmOperand
|
|
|
|
: AsmOperandClass { let Name = "MemNoOffsetT2NoSp"; }
|
|
|
|
def t2_nosp_addr_offset_none : MemOperand {
|
|
|
|
let PrintMethod = "printAddrMode7Operand";
|
2019-06-24 10:00:39 +00:00
|
|
|
let DecoderMethod = "DecoderGPRRegisterClass";
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
let ParserMatchClass = MemNoOffsetT2NoSpAsmOperand;
|
2019-06-24 10:00:39 +00:00
|
|
|
let MIOperandInfo = (ops rGPR:$base);
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
|
|
|
|
2009-06-29 07:51:04 +00:00
|
|
|
// t2addrmode_imm12 := reg + imm12
|
2011-09-07 20:58:57 +00:00
|
|
|
def t2addrmode_imm12_asmoperand : AsmOperandClass {let Name="MemUImm12Offset";}
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2addrmode_imm12 : MemOperand,
|
2009-06-29 07:51:04 +00:00
|
|
|
ComplexPattern<i32, 2, "SelectT2AddrModeImm12", []> {
|
2013-04-12 18:47:25 +00:00
|
|
|
let PrintMethod = "printAddrModeImm12Operand<false>";
|
2010-12-10 20:53:44 +00:00
|
|
|
let EncoderMethod = "getAddrModeImm12OpValue";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm12";
|
2011-09-07 20:58:57 +00:00
|
|
|
let ParserMatchClass = t2addrmode_imm12_asmoperand;
|
2009-06-29 07:51:04 +00:00
|
|
|
let MIOperandInfo = (ops GPR:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2011-03-18 17:42:55 +00:00
|
|
|
// t2ldrlabel := imm12
|
|
|
|
def t2ldrlabel : Operand<i32> {
|
|
|
|
let EncoderMethod = "getAddrModeImm12OpValue";
|
2012-10-30 01:04:51 +00:00
|
|
|
let PrintMethod = "printThumbLdrLabelOperand";
|
2011-03-18 17:42:55 +00:00
|
|
|
}
|
|
|
|
|
2012-01-18 22:46:46 +00:00
|
|
|
def t2ldr_pcrel_imm12_asmoperand : AsmOperandClass {let Name = "MemPCRelImm12";}
|
|
|
|
def t2ldr_pcrel_imm12 : Operand<i32> {
|
|
|
|
let ParserMatchClass = t2ldr_pcrel_imm12_asmoperand;
|
|
|
|
// used for assembler pseudo instruction and maps to t2ldrlabel, so
|
|
|
|
// doesn't need encoder or print methods of its own.
|
|
|
|
}
|
2011-03-18 17:42:55 +00:00
|
|
|
|
2010-12-14 00:36:49 +00:00
|
|
|
// ADR instruction labels.
|
|
|
|
def t2adrlabel : Operand<i32> {
|
|
|
|
let EncoderMethod = "getT2AdrLabelOpValue";
|
2013-07-03 09:21:44 +00:00
|
|
|
let PrintMethod = "printAdrLabelOperand<0>";
|
2010-12-14 00:36:49 +00:00
|
|
|
}
|
|
|
|
|
2011-09-07 23:39:14 +00:00
|
|
|
// t2addrmode_posimm8 := reg + imm8
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
def MemPosImm8OffsetAsmOperand : AsmOperandClass {
|
|
|
|
let Name="MemPosImm8Offset";
|
|
|
|
let RenderMethod = "addMemImmOffsetOperands";
|
|
|
|
}
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2addrmode_posimm8 : MemOperand {
|
2013-06-18 08:12:51 +00:00
|
|
|
let PrintMethod = "printT2AddrModeImm8Operand<false>";
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeImmOpValue<8,0>";
|
2011-09-07 23:39:14 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm8";
|
|
|
|
let ParserMatchClass = MemPosImm8OffsetAsmOperand;
|
|
|
|
let MIOperandInfo = (ops GPR:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2011-09-07 20:58:57 +00:00
|
|
|
// t2addrmode_negimm8 := reg - imm8
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
def MemNegImm8OffsetAsmOperand : AsmOperandClass {
|
|
|
|
let Name="MemNegImm8Offset";
|
|
|
|
let RenderMethod = "addMemImmOffsetOperands";
|
|
|
|
}
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2addrmode_negimm8 : MemOperand,
|
2011-09-07 20:58:57 +00:00
|
|
|
ComplexPattern<i32, 2, "SelectT2AddrModeImm8", []> {
|
2013-06-18 08:12:51 +00:00
|
|
|
let PrintMethod = "printT2AddrModeImm8Operand<false>";
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeImmOpValue<8,0>";
|
2011-09-07 20:58:57 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm8";
|
|
|
|
let ParserMatchClass = MemNegImm8OffsetAsmOperand;
|
|
|
|
let MIOperandInfo = (ops GPR:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2010-03-04 17:40:44 +00:00
|
|
|
// t2addrmode_imm8 := reg +/- imm8
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
def MemImm8OffsetAsmOperand : AsmOperandClass {
|
|
|
|
let Name = "MemImm8Offset";
|
|
|
|
let RenderMethod = "addMemImmOffsetOperands";
|
|
|
|
}
|
2015-04-07 20:31:16 +00:00
|
|
|
class T2AddrMode_Imm8 : MemOperand,
|
2013-06-18 08:12:51 +00:00
|
|
|
ComplexPattern<i32, 2, "SelectT2AddrModeImm8", []> {
|
[ARM] Add MVE vector load/store instructions.
This adds the rest of the vector memory access instructions. It
includes contiguous loads/stores, with an ordinary addressing mode
such as [r0,#offset] (plus writeback variants); gather loads and
scatter stores with a scalar base address register and a vector of
offsets from it (written [r0,q1] or similar); and gather/scatters with
a vector of base addresses (written [q0,#offset], again with
writeback). Additionally, some of the loads can widen each loaded
value into a larger vector lane, and the corresponding stores narrow
them again.
To implement these, we also have to add the addressing modes they
need. Also, in AsmParser, the `isMem` query function now has
subqueries `isGPRMem` and `isMVEMem`, according to which kind of base
register is used by a given memory access operand.
I've also had to add an extra check in `checkTargetMatchPredicate` in
the AsmParser, without which our last-minute check of `rGPR` register
operands against SP and PC was failing an assertion because Tablegen
had inserted an immediate 0 in place of one of a pair of tied register
operands. (This matches the way the corresponding check for `MCK_rGPR`
in `validateTargetOperandClass` is guarded.) Apparently the MVE load
instructions were the first to have ever triggered this assertion, but
I think only because they were the first to have a combination of the
usual Arm pre/post writeback system and the `rGPR` class in particular.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62680
llvm-svn: 364291
2019-06-25 11:24:18 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeImmOpValue<8,0>";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm8";
|
2011-08-03 23:50:40 +00:00
|
|
|
let ParserMatchClass = MemImm8OffsetAsmOperand;
|
2009-06-29 07:51:04 +00:00
|
|
|
let MIOperandInfo = (ops GPR:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2013-06-18 08:12:51 +00:00
|
|
|
def t2addrmode_imm8 : T2AddrMode_Imm8 {
|
|
|
|
let PrintMethod = "printT2AddrModeImm8Operand<false>";
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2addrmode_imm8_pre : T2AddrMode_Imm8 {
|
|
|
|
let PrintMethod = "printT2AddrModeImm8Operand<true>";
|
|
|
|
}
|
|
|
|
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2am_imm8_offset : MemOperand,
|
2010-09-21 20:31:19 +00:00
|
|
|
ComplexPattern<i32, 1, "SelectT2AddrModeImm8Offset",
|
|
|
|
[], [SDNPWantRoot]> {
|
2009-07-02 07:28:31 +00:00
|
|
|
let PrintMethod = "printT2AddrModeImm8OffsetOperand";
|
2010-12-10 20:53:44 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeImm8OffsetOpValue";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2Imm8";
|
2009-07-02 07:28:31 +00:00
|
|
|
}
|
|
|
|
|
2009-07-09 22:21:59 +00:00
|
|
|
// t2addrmode_imm8s4 := reg +/- (imm8 << 2)
|
2011-09-08 22:07:06 +00:00
|
|
|
def MemImm8s4OffsetAsmOperand : AsmOperandClass {let Name = "MemImm8s4Offset";}
|
2020-03-09 13:29:37 +00:00
|
|
|
class T2AddrMode_Imm8s4 : MemOperand,
|
|
|
|
ComplexPattern<i32, 2, "SelectT2AddrModeImm8<2>", []> {
|
2010-12-10 20:53:44 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeImm8s4OpValue";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm8s4";
|
2011-09-08 22:07:06 +00:00
|
|
|
let ParserMatchClass = MemImm8s4OffsetAsmOperand;
|
2009-06-30 22:50:01 +00:00
|
|
|
let MIOperandInfo = (ops GPR:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2013-06-18 08:12:51 +00:00
|
|
|
def t2addrmode_imm8s4 : T2AddrMode_Imm8s4 {
|
|
|
|
let PrintMethod = "printT2AddrModeImm8s4Operand<false>";
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2addrmode_imm8s4_pre : T2AddrMode_Imm8s4 {
|
|
|
|
let PrintMethod = "printT2AddrModeImm8s4Operand<true>";
|
|
|
|
}
|
|
|
|
|
2011-09-08 22:07:06 +00:00
|
|
|
def t2am_imm8s4_offset_asmoperand : AsmOperandClass { let Name = "Imm8s4"; }
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2am_imm8s4_offset : MemOperand {
|
2010-03-11 01:13:36 +00:00
|
|
|
let PrintMethod = "printT2AddrModeImm8s4OffsetOperand";
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
let EncoderMethod = "getT2ScaledImmOpValue<8,2>";
|
2011-08-04 23:18:05 +00:00
|
|
|
let DecoderMethod = "DecodeT2Imm8S4";
|
2010-03-11 01:13:36 +00:00
|
|
|
}
|
|
|
|
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
// t2addrmode_imm7s4 := reg +/- (imm7 << 2)
|
|
|
|
def MemImm7s4OffsetAsmOperand : AsmOperandClass {let Name = "MemImm7s4Offset";}
|
|
|
|
class T2AddrMode_Imm7s4 : MemOperand {
|
|
|
|
let EncoderMethod = "getT2AddrModeImm7s4OpValue";
|
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm7<2,0>";
|
|
|
|
let ParserMatchClass = MemImm7s4OffsetAsmOperand;
|
|
|
|
let MIOperandInfo = (ops GPRnopc:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2addrmode_imm7s4 : T2AddrMode_Imm7s4 {
|
|
|
|
// They are printed the same way as the imm8 version
|
|
|
|
let PrintMethod = "printT2AddrModeImm8s4Operand<false>";
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2addrmode_imm7s4_pre : T2AddrMode_Imm7s4 {
|
|
|
|
// They are printed the same way as the imm8 version
|
|
|
|
let PrintMethod = "printT2AddrModeImm8s4Operand<true>";
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2am_imm7s4_offset_asmoperand : AsmOperandClass { let Name = "Imm7s4"; }
|
|
|
|
def t2am_imm7s4_offset : MemOperand {
|
|
|
|
// They are printed the same way as the imm8 version
|
|
|
|
let PrintMethod = "printT2AddrModeImm8s4OffsetOperand";
|
|
|
|
let ParserMatchClass = t2am_imm7s4_offset_asmoperand;
|
|
|
|
let EncoderMethod = "getT2ScaledImmOpValue<7,2>";
|
|
|
|
let DecoderMethod = "DecodeT2Imm7S4";
|
|
|
|
}
|
|
|
|
|
2011-09-09 18:37:27 +00:00
|
|
|
// t2addrmode_imm0_1020s4 := reg + (imm8 << 2)
|
|
|
|
def MemImm0_1020s4OffsetAsmOperand : AsmOperandClass {
|
|
|
|
let Name = "MemImm0_1020s4Offset";
|
|
|
|
}
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2addrmode_imm0_1020s4 : MemOperand,
|
2013-07-16 09:46:55 +00:00
|
|
|
ComplexPattern<i32, 2, "SelectT2AddrModeExclusive"> {
|
2011-09-09 18:37:27 +00:00
|
|
|
let PrintMethod = "printT2AddrModeImm0_1020s4Operand";
|
|
|
|
let EncoderMethod = "getT2AddrModeImm0_1020s4OpValue";
|
|
|
|
let DecoderMethod = "DecodeT2AddrModeImm0_1020s4";
|
|
|
|
let ParserMatchClass = MemImm0_1020s4OffsetAsmOperand;
|
|
|
|
let MIOperandInfo = (ops GPRnopc:$base, i32imm:$offsimm);
|
|
|
|
}
|
|
|
|
|
2009-07-09 20:40:44 +00:00
|
|
|
// t2addrmode_so_reg := reg + (reg << imm2)
|
2011-09-07 23:10:15 +00:00
|
|
|
def t2addrmode_so_reg_asmoperand : AsmOperandClass {let Name="T2MemRegOffset";}
|
2015-04-07 20:31:16 +00:00
|
|
|
def t2addrmode_so_reg : MemOperand,
|
2009-06-29 07:51:04 +00:00
|
|
|
ComplexPattern<i32, 3, "SelectT2AddrModeSoReg", []> {
|
|
|
|
let PrintMethod = "printT2AddrModeSoRegOperand";
|
2010-12-10 20:53:44 +00:00
|
|
|
let EncoderMethod = "getT2AddrModeSORegOpValue";
|
2011-08-09 20:55:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2AddrModeSOReg";
|
2011-09-07 23:10:15 +00:00
|
|
|
let ParserMatchClass = t2addrmode_so_reg_asmoperand;
|
2016-02-25 16:54:52 +00:00
|
|
|
let MIOperandInfo = (ops GPRnopc:$base, rGPR:$offsreg, i32imm:$offsimm);
|
2009-06-29 07:51:04 +00:00
|
|
|
}
|
|
|
|
|
2011-09-19 22:21:13 +00:00
|
|
|
// Addresses for the TBB/TBH instructions.
|
|
|
|
def addrmode_tbb_asmoperand : AsmOperandClass { let Name = "MemTBB"; }
|
2015-04-07 20:31:16 +00:00
|
|
|
def addrmode_tbb : MemOperand {
|
2011-09-19 22:21:13 +00:00
|
|
|
let PrintMethod = "printAddrModeTBB";
|
|
|
|
let ParserMatchClass = addrmode_tbb_asmoperand;
|
|
|
|
let MIOperandInfo = (ops GPR:$Rn, rGPR:$Rm);
|
|
|
|
}
|
|
|
|
def addrmode_tbh_asmoperand : AsmOperandClass { let Name = "MemTBH"; }
|
2015-04-07 20:31:16 +00:00
|
|
|
def addrmode_tbh : MemOperand {
|
2011-09-19 22:21:13 +00:00
|
|
|
let PrintMethod = "printAddrModeTBH";
|
|
|
|
let ParserMatchClass = addrmode_tbh_asmoperand;
|
|
|
|
let MIOperandInfo = (ops GPR:$Rn, rGPR:$Rm);
|
|
|
|
}
|
|
|
|
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
// Define ARMv8.1-M specific addressing modes.
|
|
|
|
|
|
|
|
// Label operands for BF/BFL/WLS/DLS/LE
|
[ARM] Fix handling of zero offsets in LOB instructions.
The BF and WLS/WLSTP instructions have various branch-offset fields
occupying different positions and lengths in the instruction encoding,
and all of them were decoded at disassembly time by the function
DecodeBFLabelOffset() which returned SoftFail if the offset was zero.
In fact, it's perfectly fine and not even a SoftFail for most of those
offset fields to be zero. The only one that can't be zero is the 4-bit
field labelled `boff` in the architecture spec, occupying bits {26-23}
of the BF instruction family. If that one is zero, the encoding
overlaps other instructions (WLS, DLS, LETP, VCTP), so it ought to be
a full Fail.
Fixed by adding an extra template parameter to DecodeBFLabelOffset
which controls whether a zero offset is accepted or rejected. Adjusted
existing tests (only in error messages for bad disassemblies); added
extra tests to demonstrate zero offsets being accepted in all the
right places, and a few demonstrating rejection of zero `boff`.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63864
llvm-svn: 364533
2019-06-27 12:41:07 +00:00
|
|
|
class BFLabelOp<string signed, string isNeg, string zeroPermitted, string size,
|
|
|
|
string fixup>
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
: Operand<OtherVT> {
|
|
|
|
let EncoderMethod = !strconcat("getBFTargetOpValue<", isNeg, ", ",
|
|
|
|
fixup, ">");
|
|
|
|
let OperandType = "OPERAND_PCREL";
|
|
|
|
let DecoderMethod = !strconcat("DecodeBFLabelOperand<", signed, ", ",
|
[ARM] Fix handling of zero offsets in LOB instructions.
The BF and WLS/WLSTP instructions have various branch-offset fields
occupying different positions and lengths in the instruction encoding,
and all of them were decoded at disassembly time by the function
DecodeBFLabelOffset() which returned SoftFail if the offset was zero.
In fact, it's perfectly fine and not even a SoftFail for most of those
offset fields to be zero. The only one that can't be zero is the 4-bit
field labelled `boff` in the architecture spec, occupying bits {26-23}
of the BF instruction family. If that one is zero, the encoding
overlaps other instructions (WLS, DLS, LETP, VCTP), so it ought to be
a full Fail.
Fixed by adding an extra template parameter to DecodeBFLabelOffset
which controls whether a zero offset is accepted or rejected. Adjusted
existing tests (only in error messages for bad disassemblies); added
extra tests to demonstrate zero offsets being accepted in all the
right places, and a few demonstrating rejection of zero `boff`.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63864
llvm-svn: 364533
2019-06-27 12:41:07 +00:00
|
|
|
isNeg, ", ", zeroPermitted, ", ", size, ">");
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
[ARM] Fix handling of zero offsets in LOB instructions.
The BF and WLS/WLSTP instructions have various branch-offset fields
occupying different positions and lengths in the instruction encoding,
and all of them were decoded at disassembly time by the function
DecodeBFLabelOffset() which returned SoftFail if the offset was zero.
In fact, it's perfectly fine and not even a SoftFail for most of those
offset fields to be zero. The only one that can't be zero is the 4-bit
field labelled `boff` in the architecture spec, occupying bits {26-23}
of the BF instruction family. If that one is zero, the encoding
overlaps other instructions (WLS, DLS, LETP, VCTP), so it ought to be
a full Fail.
Fixed by adding an extra template parameter to DecodeBFLabelOffset
which controls whether a zero offset is accepted or rejected. Adjusted
existing tests (only in error messages for bad disassemblies); added
extra tests to demonstrate zero offsets being accepted in all the
right places, and a few demonstrating rejection of zero `boff`.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63864
llvm-svn: 364533
2019-06-27 12:41:07 +00:00
|
|
|
def bflabel_u4 : BFLabelOp<"false", "false", "false", "4", "ARM::fixup_bf_branch">;
|
|
|
|
def bflabel_s12 : BFLabelOp<"true", "false", "true", "12", "ARM::fixup_bfc_target">;
|
|
|
|
def bflabel_s16 : BFLabelOp<"true", "false", "true", "16", "ARM::fixup_bf_target">;
|
|
|
|
def bflabel_s18 : BFLabelOp<"true", "false", "true", "18", "ARM::fixup_bfl_target">;
|
2019-06-25 11:24:33 +00:00
|
|
|
|
|
|
|
def wlslabel_u11_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "WLSLabel";
|
|
|
|
let RenderMethod = "addImmOperands";
|
|
|
|
let PredicateMethod = "isUnsignedOffset<11, 1>";
|
|
|
|
let DiagnosticString =
|
|
|
|
"loop end is out of range or not a positive multiple of 2";
|
|
|
|
}
|
[ARM] Fix handling of zero offsets in LOB instructions.
The BF and WLS/WLSTP instructions have various branch-offset fields
occupying different positions and lengths in the instruction encoding,
and all of them were decoded at disassembly time by the function
DecodeBFLabelOffset() which returned SoftFail if the offset was zero.
In fact, it's perfectly fine and not even a SoftFail for most of those
offset fields to be zero. The only one that can't be zero is the 4-bit
field labelled `boff` in the architecture spec, occupying bits {26-23}
of the BF instruction family. If that one is zero, the encoding
overlaps other instructions (WLS, DLS, LETP, VCTP), so it ought to be
a full Fail.
Fixed by adding an extra template parameter to DecodeBFLabelOffset
which controls whether a zero offset is accepted or rejected. Adjusted
existing tests (only in error messages for bad disassemblies); added
extra tests to demonstrate zero offsets being accepted in all the
right places, and a few demonstrating rejection of zero `boff`.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63864
llvm-svn: 364533
2019-06-27 12:41:07 +00:00
|
|
|
def wlslabel_u11 : BFLabelOp<"false", "false", "true", "11", "ARM::fixup_wls"> {
|
2019-06-25 11:24:33 +00:00
|
|
|
let ParserMatchClass = wlslabel_u11_asmoperand;
|
|
|
|
}
|
|
|
|
def lelabel_u11_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "LELabel";
|
|
|
|
let RenderMethod = "addImmOperands";
|
|
|
|
let PredicateMethod = "isLEOffset";
|
|
|
|
let DiagnosticString =
|
|
|
|
"loop start is out of range or not a negative multiple of 2";
|
|
|
|
}
|
[ARM] Fix handling of zero offsets in LOB instructions.
The BF and WLS/WLSTP instructions have various branch-offset fields
occupying different positions and lengths in the instruction encoding,
and all of them were decoded at disassembly time by the function
DecodeBFLabelOffset() which returned SoftFail if the offset was zero.
In fact, it's perfectly fine and not even a SoftFail for most of those
offset fields to be zero. The only one that can't be zero is the 4-bit
field labelled `boff` in the architecture spec, occupying bits {26-23}
of the BF instruction family. If that one is zero, the encoding
overlaps other instructions (WLS, DLS, LETP, VCTP), so it ought to be
a full Fail.
Fixed by adding an extra template parameter to DecodeBFLabelOffset
which controls whether a zero offset is accepted or rejected. Adjusted
existing tests (only in error messages for bad disassemblies); added
extra tests to demonstrate zero offsets being accepted in all the
right places, and a few demonstrating rejection of zero `boff`.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63864
llvm-svn: 364533
2019-06-27 12:41:07 +00:00
|
|
|
def lelabel_u11 : BFLabelOp<"false", "true", "true", "11", "ARM::fixup_le"> {
|
2019-06-25 11:24:33 +00:00
|
|
|
let ParserMatchClass = lelabel_u11_asmoperand;
|
|
|
|
}
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
|
|
|
|
def bfafter_target : Operand<OtherVT> {
|
|
|
|
let EncoderMethod = "getBFAfterTargetOpValue";
|
|
|
|
let OperandType = "OPERAND_PCREL";
|
|
|
|
let DecoderMethod = "DecodeBFAfterTargetOperand";
|
|
|
|
}
|
|
|
|
|
|
|
|
// pred operand excluding AL
|
|
|
|
def pred_noal_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "CondCodeNoAL";
|
|
|
|
let RenderMethod = "addITCondCodeOperands";
|
|
|
|
let PredicateMethod = "isITCondCodeNoAL";
|
|
|
|
let ParserMethod = "parseITCondCode";
|
|
|
|
}
|
|
|
|
def pred_noal : Operand<i32> {
|
|
|
|
let PrintMethod = "printMandatoryPredicateOperand";
|
|
|
|
let ParserMatchClass = pred_noal_asmoperand;
|
|
|
|
let DecoderMethod = "DecodePredNoALOperand";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// CSEL aliases inverted predicate
|
|
|
|
def pred_noal_inv_asmoperand : AsmOperandClass {
|
|
|
|
let Name = "CondCodeNoALInv";
|
|
|
|
let RenderMethod = "addITCondCodeInvOperands";
|
|
|
|
let PredicateMethod = "isITCondCodeNoAL";
|
|
|
|
let ParserMethod = "parseITCondCode";
|
|
|
|
}
|
|
|
|
def pred_noal_inv : Operand<i32> {
|
|
|
|
let PrintMethod = "printMandatoryInvertedPredicateOperand";
|
|
|
|
let ParserMatchClass = pred_noal_inv_asmoperand;
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2009-06-27 02:26:13 +00:00
|
|
|
// Multiclass helpers...
|
2009-06-17 18:13:58 +00:00
|
|
|
//
|
|
|
|
|
2010-11-15 18:45:17 +00:00
|
|
|
|
|
|
|
class T2OneRegImm<dag oops, dag iops, InstrItinClass itin,
|
2010-11-14 05:37:38 +00:00
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
2010-11-15 18:45:17 +00:00
|
|
|
bits<12> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-15 18:45:17 +00:00
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
|
|
|
}
|
|
|
|
|
2010-11-15 19:58:36 +00:00
|
|
|
|
2010-11-15 18:45:17 +00:00
|
|
|
class T2sOneRegImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
2010-11-14 05:37:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-14 05:37:38 +00:00
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
|
|
|
}
|
|
|
|
|
2010-11-15 19:58:36 +00:00
|
|
|
class T2OneRegCmpImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2010-11-15 19:58:36 +00:00
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-15 18:45:17 +00:00
|
|
|
class T2OneRegShiftedReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<12> ShiftedRm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-15 18:45:17 +00:00
|
|
|
let Inst{3-0} = ShiftedRm{3-0};
|
|
|
|
let Inst{5-4} = ShiftedRm{6-5};
|
|
|
|
let Inst{14-12} = ShiftedRm{11-9};
|
|
|
|
let Inst{7-6} = ShiftedRm{8-7};
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2sOneRegShiftedReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-12-07 20:50:15 +00:00
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
2010-11-15 18:45:17 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<12> ShiftedRm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-15 18:45:17 +00:00
|
|
|
let Inst{3-0} = ShiftedRm{3-0};
|
|
|
|
let Inst{5-4} = ShiftedRm{6-5};
|
|
|
|
let Inst{14-12} = ShiftedRm{11-9};
|
|
|
|
let Inst{7-6} = ShiftedRm{8-7};
|
|
|
|
}
|
|
|
|
|
2010-11-15 19:58:36 +00:00
|
|
|
class T2OneRegCmpShiftedReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> ShiftedRm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2010-11-15 19:58:36 +00:00
|
|
|
let Inst{3-0} = ShiftedRm{3-0};
|
|
|
|
let Inst{5-4} = ShiftedRm{6-5};
|
|
|
|
let Inst{14-12} = ShiftedRm{11-9};
|
|
|
|
let Inst{7-6} = ShiftedRm{8-7};
|
|
|
|
}
|
|
|
|
|
2010-11-15 18:45:17 +00:00
|
|
|
class T2TwoReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
2010-11-15 18:45:17 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-15 18:45:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class T2sTwoReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
2010-11-15 18:45:17 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-15 18:45:17 +00:00
|
|
|
}
|
|
|
|
|
2010-11-15 19:58:36 +00:00
|
|
|
class T2TwoRegCmp<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
2010-11-15 19:58:36 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-15 19:58:36 +00:00
|
|
|
}
|
|
|
|
|
2010-11-15 18:45:17 +00:00
|
|
|
|
|
|
|
class T2TwoRegImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
2010-12-08 23:04:16 +00:00
|
|
|
bits<4> Rn;
|
2010-12-08 23:24:29 +00:00
|
|
|
bits<12> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-12-08 23:24:29 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
2010-11-15 18:45:17 +00:00
|
|
|
}
|
|
|
|
|
2010-11-14 05:37:38 +00:00
|
|
|
class T2sTwoRegImm<dag oops, dag iops, InstrItinClass itin,
|
2010-11-12 21:12:40 +00:00
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
2010-11-12 21:12:40 +00:00
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
|
|
|
}
|
|
|
|
|
2010-11-15 19:58:36 +00:00
|
|
|
class T2TwoRegShiftImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rm;
|
|
|
|
bits<5> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-15 19:58:36 +00:00
|
|
|
let Inst{14-12} = imm{4-2};
|
|
|
|
let Inst{7-6} = imm{1-0};
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2sTwoRegShiftImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rm;
|
|
|
|
bits<5> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-15 19:58:36 +00:00
|
|
|
let Inst{14-12} = imm{4-2};
|
|
|
|
let Inst{7-6} = imm{1-0};
|
|
|
|
}
|
|
|
|
|
2010-11-12 21:12:40 +00:00
|
|
|
class T2ThreeReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
2010-11-14 05:37:38 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-14 05:37:38 +00:00
|
|
|
}
|
|
|
|
|
2013-09-18 09:45:55 +00:00
|
|
|
class T2ThreeRegNoP<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string asm, list<dag> pattern>
|
|
|
|
: T2XI<oops, iops, itin, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
|
|
|
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{3-0} = Rm;
|
|
|
|
}
|
|
|
|
|
2010-11-14 05:37:38 +00:00
|
|
|
class T2sThreeReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
2010-11-12 21:12:40 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-12 21:12:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class T2TwoRegShiftedReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-14 05:37:38 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> ShiftedRm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
2010-11-14 05:37:38 +00:00
|
|
|
let Inst{3-0} = ShiftedRm{3-0};
|
|
|
|
let Inst{5-4} = ShiftedRm{6-5};
|
|
|
|
let Inst{14-12} = ShiftedRm{11-9};
|
|
|
|
let Inst{7-6} = ShiftedRm{8-7};
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2sTwoRegShiftedReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-12 21:12:40 +00:00
|
|
|
: T2sI<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> ShiftedRm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
2010-11-12 21:12:40 +00:00
|
|
|
let Inst{3-0} = ShiftedRm{3-0};
|
|
|
|
let Inst{5-4} = ShiftedRm{6-5};
|
|
|
|
let Inst{14-12} = ShiftedRm{11-9};
|
|
|
|
let Inst{7-6} = ShiftedRm{8-7};
|
|
|
|
}
|
|
|
|
|
2010-11-18 01:08:42 +00:00
|
|
|
class T2FourReg<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
2010-11-18 01:08:42 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
|
|
|
bits<4> Ra;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = Ra;
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{3-0} = Rm;
|
2010-11-18 01:08:42 +00:00
|
|
|
}
|
|
|
|
|
2010-12-08 22:38:41 +00:00
|
|
|
class T2MulLong<bits<3> opc22_20, bits<4> opc7_4,
|
2016-07-19 14:44:05 +00:00
|
|
|
string opc, list<dag> pattern>
|
|
|
|
: T2I<(outs rGPR:$RdLo, rGPR:$RdHi), (ins rGPR:$Rn, rGPR:$Rm), IIC_iMUL64,
|
2017-02-22 07:22:57 +00:00
|
|
|
opc, "\t$RdLo, $RdHi, $Rn, $Rm", pattern>,
|
|
|
|
Sched<[WriteMUL64Lo, WriteMUL64Hi, ReadMUL, ReadMUL]> {
|
2010-12-08 22:29:28 +00:00
|
|
|
bits<4> RdLo;
|
|
|
|
bits<4> RdHi;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
|
|
|
|
2010-12-08 22:38:41 +00:00
|
|
|
let Inst{31-23} = 0b111110111;
|
|
|
|
let Inst{22-20} = opc22_20;
|
2010-12-08 22:29:28 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = RdLo;
|
|
|
|
let Inst{11-8} = RdHi;
|
2010-12-08 22:38:41 +00:00
|
|
|
let Inst{7-4} = opc7_4;
|
2010-12-08 22:29:28 +00:00
|
|
|
let Inst{3-0} = Rm;
|
|
|
|
}
|
2016-07-19 14:44:05 +00:00
|
|
|
class T2MlaLong<bits<3> opc22_20, bits<4> opc7_4, string opc>
|
|
|
|
: T2I<(outs rGPR:$RdLo, rGPR:$RdHi),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi), IIC_iMAC64,
|
|
|
|
opc, "\t$RdLo, $RdHi, $Rn, $Rm", []>,
|
2017-02-22 07:22:57 +00:00
|
|
|
RegConstraint<"$RLo = $RdLo, $RHi = $RdHi">,
|
|
|
|
Sched<[WriteMAC64Lo, WriteMAC64Hi, ReadMUL, ReadMUL, ReadMAC, ReadMAC]> {
|
2012-09-04 14:37:49 +00:00
|
|
|
bits<4> RdLo;
|
|
|
|
bits<4> RdHi;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
|
|
|
|
|
|
|
let Inst{31-23} = 0b111110111;
|
|
|
|
let Inst{22-20} = opc22_20;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = RdLo;
|
|
|
|
let Inst{11-8} = RdHi;
|
|
|
|
let Inst{7-4} = opc7_4;
|
|
|
|
let Inst{3-0} = Rm;
|
|
|
|
}
|
2010-12-08 22:29:28 +00:00
|
|
|
|
2010-11-18 01:08:42 +00:00
|
|
|
|
2009-06-23 19:39:13 +00:00
|
|
|
/// T2I_bin_irs - Defines a set of (op reg, {so_imm|r|so_reg}) patterns for a
|
2010-05-25 04:43:08 +00:00
|
|
|
/// binary operation that produces a value. These are predicable and can be
|
2009-06-25 02:08:06 +00:00
|
|
|
/// changed to modify CPSR.
|
2010-09-29 00:27:46 +00:00
|
|
|
multiclass T2I_bin_irs<bits<4> opcod, string opc,
|
|
|
|
InstrItinClass iii, InstrItinClass iir, InstrItinClass iis,
|
2016-03-08 16:23:54 +00:00
|
|
|
SDPatternOperator opnode, bit Commutable = 0,
|
|
|
|
string wide = ""> {
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted imm
|
2010-11-14 05:37:38 +00:00
|
|
|
def ri : T2sTwoRegImm<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_imm:$imm), iii,
|
|
|
|
opc, "\t$Rd, $Rn, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rn, t2_so_imm:$imm))]>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rr : T2sThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), iir,
|
|
|
|
opc, !strconcat(wide, "\t$Rd, $Rn, $Rm"),
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rn, rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU, ReadALU, ReadALU]> {
|
2009-06-26 00:19:44 +00:00
|
|
|
let isCommutable = Commutable;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
[ARM] Turn some undefined encoding bits into 0s.
The family of 32-bit Thumb instruction encodings that include t2ORR,
t2AND and t2EOR are all listed in the ArmARM as having (0) in bit 15.
The Tablegen descriptions of those instructions listed them as ?. This
change tightens that up by making them into 0 + Unpredictable.
In the specific case of t2ORR, we tighten it up still further by
making the zero bit mandatory. This change comes from Arm v8.1-M, in
which encodings with that bit equal to 1 will now be used for
different instructions.
Reviewers: dmgreen, samparker, SjoerdMeijer, efriedma
Reviewed By: dmgreen, efriedma
Subscribers: efriedma, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60705
llvm-svn: 362470
2019-06-04 08:28:48 +00:00
|
|
|
let Inst{15} = 0b0;
|
|
|
|
// In most of these instructions, and most versions of the Arm
|
|
|
|
// architecture, bit 15 of this encoding is listed as (0) rather
|
|
|
|
// than 0, i.e. setting it to 1 is UNPREDICTABLE or a soft-fail
|
|
|
|
// rather than a hard failure. In v8.1-M, this requirement is
|
|
|
|
// upgraded to a hard one for ORR, so that the encodings with 1
|
|
|
|
// in this bit can be reused for other instructions (such as
|
|
|
|
// CSEL). Setting Unpredictable{15} = 1 here would reintroduce
|
|
|
|
// that encoding clash in the auto- generated MC decoder, so I
|
|
|
|
// comment it out.
|
|
|
|
let Unpredictable{15} = !if(!eq(opcod, 0b0010), 0b0, 0b1);
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
2009-06-26 00:19:44 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rs : T2sTwoRegShiftedReg<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_reg:$ShiftedRm), iis,
|
|
|
|
opc, !strconcat(wide, "\t$Rd, $Rn, $ShiftedRm"),
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rn, t2_so_reg:$ShiftedRm))]>,
|
|
|
|
Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
[ARM] Turn some undefined encoding bits into 0s.
The family of 32-bit Thumb instruction encodings that include t2ORR,
t2AND and t2EOR are all listed in the ArmARM as having (0) in bit 15.
The Tablegen descriptions of those instructions listed them as ?. This
change tightens that up by making them into 0 + Unpredictable.
In the specific case of t2ORR, we tighten it up still further by
making the zero bit mandatory. This change comes from Arm v8.1-M, in
which encodings with that bit equal to 1 will now be used for
different instructions.
Reviewers: dmgreen, samparker, SjoerdMeijer, efriedma
Reviewed By: dmgreen, efriedma
Subscribers: efriedma, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60705
llvm-svn: 362470
2019-06-04 08:28:48 +00:00
|
|
|
let Inst{15} = 0;
|
|
|
|
let Unpredictable{15} = !if(!eq(opcod, 0b0010), 0b0, 0b1); // see above
|
2010-08-30 01:47:35 +00:00
|
|
|
}
|
2011-06-28 00:19:13 +00:00
|
|
|
// Assembly aliases for optional destination operand when it's the same
|
|
|
|
// as the source operand.
|
2011-08-22 18:04:24 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p} $Rdn, $imm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rdn, rGPR:$Rdn,
|
2011-06-28 00:19:13 +00:00
|
|
|
t2_so_imm:$imm, pred:$p,
|
2011-08-22 18:04:24 +00:00
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", wide, " $Rdn, $Rm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rdn, rGPR:$Rdn,
|
2011-06-28 00:19:13 +00:00
|
|
|
rGPR:$Rm, pred:$p,
|
2011-08-22 18:04:24 +00:00
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", wide, " $Rdn, $shift"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rs") rGPR:$Rdn, rGPR:$Rdn,
|
2011-06-28 00:19:13 +00:00
|
|
|
t2_so_reg:$shift, pred:$p,
|
2011-08-22 18:04:24 +00:00
|
|
|
cc_out:$s)>;
|
2010-08-30 01:47:35 +00:00
|
|
|
}
|
|
|
|
|
2009-07-27 23:34:12 +00:00
|
|
|
/// T2I_bin_w_irs - Same as T2I_bin_irs except these operations need
|
2011-06-28 00:19:13 +00:00
|
|
|
// the ".w" suffix to indicate that they are wide.
|
2010-09-29 00:27:46 +00:00
|
|
|
multiclass T2I_bin_w_irs<bits<4> opcod, string opc,
|
|
|
|
InstrItinClass iii, InstrItinClass iir, InstrItinClass iis,
|
2016-03-08 16:23:54 +00:00
|
|
|
SDPatternOperator opnode, bit Commutable = 0> :
|
2012-08-02 21:50:41 +00:00
|
|
|
T2I_bin_irs<opcod, opc, iii, iir, iis, opnode, Commutable, ".w"> {
|
2012-02-24 19:06:05 +00:00
|
|
|
// Assembler aliases w/ the ".w" suffix.
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}.w", " $Rd, $Rn, $imm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rd, rGPR:$Rn, t2_so_imm:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
// Assembler aliases w/o the ".w" suffix.
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rd, $Rn, $Rm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rd, $Rn, $shift"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rs") rGPR:$Rd, rGPR:$Rn, t2_so_reg:$shift,
|
|
|
|
pred:$p, cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
|
|
|
|
// and with the optional destination operand, too.
|
2012-03-16 22:18:29 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}.w", " $Rdn, $imm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rdn, rGPR:$Rdn, t2_so_imm:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rdn, $Rm"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rdn, rGPR:$Rdn, rGPR:$Rm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rdn, $shift"),
|
2012-08-02 21:50:41 +00:00
|
|
|
(!cast<Instruction>(NAME#"rs") rGPR:$Rdn, rGPR:$Rdn, t2_so_reg:$shift,
|
|
|
|
pred:$p, cc_out:$s)>;
|
2011-09-02 18:41:35 +00:00
|
|
|
}
|
2010-08-29 03:55:31 +00:00
|
|
|
|
2009-06-25 20:59:23 +00:00
|
|
|
/// T2I_rbin_is - Same as T2I_bin_irs except the order of operands are
|
2010-08-13 23:24:25 +00:00
|
|
|
/// reversed. The 'rr' form is only defined for the disassembler; for codegen
|
|
|
|
/// it is equivalent to the T2I_bin_irs counterpart.
|
2016-03-08 16:23:54 +00:00
|
|
|
multiclass T2I_rbin_irs<bits<4> opcod, string opc, SDNode opnode> {
|
2009-06-17 18:13:58 +00:00
|
|
|
// shifted imm
|
2010-11-14 05:37:38 +00:00
|
|
|
def ri : T2sTwoRegImm<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_imm:$imm), IIC_iALUi,
|
|
|
|
opc, ".w\t$Rd, $Rn, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode t2_so_imm:$imm, rGPR:$Rn))]>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2010-08-13 23:24:25 +00:00
|
|
|
// register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rr : T2sThreeReg<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iALUr,
|
|
|
|
opc, "\t$Rd, $Rn, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[/* For disassembly only; pattern left blank */]>,
|
|
|
|
Sched<[WriteALU, ReadALU, ReadALU]> {
|
2010-08-13 23:24:25 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
// shifted register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rs : T2sTwoRegShiftedReg<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_reg:$ShiftedRm),
|
|
|
|
IIC_iALUsir, opc, "\t$Rd, $Rn, $ShiftedRm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode t2_so_reg:$ShiftedRm, rGPR:$Rn))]>,
|
|
|
|
Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
}
|
|
|
|
|
2009-06-23 19:39:13 +00:00
|
|
|
/// T2I_bin_s_irs - Similar to T2I_bin_irs except it sets the 's' bit so the
|
2009-06-17 18:13:58 +00:00
|
|
|
/// instruction modifies the CPSR register.
|
2011-09-21 02:20:46 +00:00
|
|
|
///
|
|
|
|
/// These opcodes will be converted to the real non-S opcodes by
|
|
|
|
/// AdjustInstrPostInstrSelection after giving then an optional CPSR operand.
|
2011-10-18 19:18:52 +00:00
|
|
|
let hasPostISelHook = 1, Defs = [CPSR] in {
|
|
|
|
multiclass T2I_bin_s_irs<InstrItinClass iii, InstrItinClass iir,
|
2016-03-08 16:23:54 +00:00
|
|
|
InstrItinClass iis, SDNode opnode,
|
2011-10-18 19:18:52 +00:00
|
|
|
bit Commutable = 0> {
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted imm
|
2011-10-18 19:18:52 +00:00
|
|
|
def ri : t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins GPRnopc:$Rn, t2_so_imm:$imm, pred:$p),
|
|
|
|
4, iii,
|
|
|
|
[(set rGPR:$Rd, CPSR, (opnode GPRnopc:$Rn,
|
2013-06-06 16:35:25 +00:00
|
|
|
t2_so_imm:$imm))]>,
|
|
|
|
Sched<[WriteALU, ReadALU]>;
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2011-10-18 19:18:52 +00:00
|
|
|
def rr : t2PseudoInst<(outs rGPR:$Rd), (ins GPRnopc:$Rn, rGPR:$Rm, pred:$p),
|
|
|
|
4, iir,
|
|
|
|
[(set rGPR:$Rd, CPSR, (opnode GPRnopc:$Rn,
|
2013-06-06 16:35:25 +00:00
|
|
|
rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU, ReadALU, ReadALU]> {
|
2011-10-18 19:18:52 +00:00
|
|
|
let isCommutable = Commutable;
|
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted register
|
2011-10-18 19:18:52 +00:00
|
|
|
def rs : t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins GPRnopc:$Rn, t2_so_reg:$ShiftedRm, pred:$p),
|
|
|
|
4, iis,
|
|
|
|
[(set rGPR:$Rd, CPSR, (opnode GPRnopc:$Rn,
|
2013-06-06 16:35:25 +00:00
|
|
|
t2_so_reg:$ShiftedRm))]>,
|
|
|
|
Sched<[WriteALUsi, ReadALUsr]>;
|
2011-10-18 19:18:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// T2I_rbin_s_is - Same as T2I_bin_s_irs, except selection DAG
|
|
|
|
/// operands are reversed.
|
|
|
|
let hasPostISelHook = 1, Defs = [CPSR] in {
|
2016-03-08 16:23:54 +00:00
|
|
|
multiclass T2I_rbin_s_is<SDNode opnode> {
|
2011-10-18 19:18:52 +00:00
|
|
|
// shifted imm
|
|
|
|
def ri : t2PseudoInst<(outs rGPR:$Rd),
|
2012-05-21 17:57:17 +00:00
|
|
|
(ins rGPR:$Rn, t2_so_imm:$imm, pred:$p),
|
2011-10-18 19:18:52 +00:00
|
|
|
4, IIC_iALUi,
|
|
|
|
[(set rGPR:$Rd, CPSR, (opnode t2_so_imm:$imm,
|
2013-06-06 16:35:25 +00:00
|
|
|
rGPR:$Rn))]>,
|
|
|
|
Sched<[WriteALU, ReadALU]>;
|
2011-10-18 19:18:52 +00:00
|
|
|
// shifted register
|
|
|
|
def rs : t2PseudoInst<(outs rGPR:$Rd),
|
2012-05-21 17:57:17 +00:00
|
|
|
(ins rGPR:$Rn, t2_so_reg:$ShiftedRm, pred:$p),
|
2011-10-18 19:18:52 +00:00
|
|
|
4, IIC_iALUsi,
|
|
|
|
[(set rGPR:$Rd, CPSR, (opnode t2_so_reg:$ShiftedRm,
|
2013-06-06 16:35:25 +00:00
|
|
|
rGPR:$Rn))]>,
|
|
|
|
Sched<[WriteALUsi, ReadALU]>;
|
2009-06-23 17:48:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-23 19:39:13 +00:00
|
|
|
/// T2I_bin_ii12rs - Defines a set of (op reg, {so_imm|imm0_4095|r|so_reg})
|
|
|
|
/// patterns for a binary operation that produces a value.
|
2016-03-08 16:23:54 +00:00
|
|
|
multiclass T2I_bin_ii12rs<bits<3> op23_21, string opc, SDNode opnode,
|
2009-12-15 17:24:14 +00:00
|
|
|
bit Commutable = 0> {
|
2009-06-17 18:13:58 +00:00
|
|
|
// shifted imm
|
2010-08-30 19:49:58 +00:00
|
|
|
// The register-immediate version is re-materializable. This is useful
|
|
|
|
// in particular for taking the address of a local.
|
|
|
|
let isReMaterializable = 1 in {
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
def spImm : T2sTwoRegImm<
|
|
|
|
(outs GPRsp:$Rd), (ins GPRsp:$Rn, t2_so_imm:$imm), IIC_iALUi,
|
|
|
|
opc, ".w\t$Rd, $Rn, $imm",
|
|
|
|
[]>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
|
|
|
let Rn = 13;
|
|
|
|
let Rd = 13;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25-24} = 0b01;
|
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2AddSubSPImm";
|
|
|
|
}
|
|
|
|
|
2010-11-14 05:37:38 +00:00
|
|
|
def ri : T2sTwoRegImm<
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(outs rGPR:$Rd), (ins GPRnopc:$Rn, t2_so_imm:$imm), IIC_iALUi,
|
2011-10-03 20:51:59 +00:00
|
|
|
opc, ".w\t$Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
[(set rGPR:$Rd, (opnode GPRnopc:$Rn, t2_so_imm:$imm))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24} = 1;
|
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2010-08-30 19:49:58 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
// 12-bit imm
|
2010-12-08 23:04:16 +00:00
|
|
|
def ri12 : T2I<
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(outs rGPR:$Rd), (ins GPR:$Rn, imm0_4095:$imm), IIC_iALUi,
|
2010-11-14 05:37:38 +00:00
|
|
|
!strconcat(opc, "w"), "\t$Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
[(set rGPR:$Rd, (opnode GPR:$Rn, imm0_4095:$imm))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2010-12-08 23:04:16 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<12> imm;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
2010-12-08 23:04:16 +00:00
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{25-24} = 0b10;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
let Inst{20} = 0; // The S bit.
|
2010-12-08 23:04:16 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{15} = 0;
|
2010-12-08 23:04:16 +00:00
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{7-0} = imm{7-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
def spImm12 : T2I<
|
|
|
|
(outs GPRsp:$Rd), (ins GPRsp:$Rn, imm0_4095:$imm), IIC_iALUi,
|
|
|
|
!strconcat(opc, "w"), "\t$Rd, $Rn, $imm",
|
|
|
|
[]>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
|
|
|
bits<4> Rd = 13;
|
|
|
|
bits<4> Rn = 13;
|
|
|
|
bits<12> imm;
|
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{25-24} = 0b10;
|
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
let Inst{20} = 0; // The S bit.
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{7-0} = imm{7-0};
|
|
|
|
let DecoderMethod = "DecodeT2AddSubSPImm";
|
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2011-10-03 20:51:59 +00:00
|
|
|
def rr : T2sThreeReg<(outs GPRnopc:$Rd), (ins GPRnopc:$Rn, rGPR:$Rm),
|
|
|
|
IIC_iALUr, opc, ".w\t$Rd, $Rn, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set GPRnopc:$Rd, (opnode GPRnopc:$Rn, rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU, ReadALU, ReadALU]> {
|
2009-06-26 00:19:44 +00:00
|
|
|
let isCommutable = Commutable;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24} = 1;
|
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
2009-06-26 00:19:44 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rs : T2sTwoRegShiftedReg<
|
2011-10-03 20:51:59 +00:00
|
|
|
(outs GPRnopc:$Rd), (ins GPRnopc:$Rn, t2_so_reg:$ShiftedRm),
|
2010-11-14 05:37:38 +00:00
|
|
|
IIC_iALUsi, opc, ".w\t$Rd, $Rn, $ShiftedRm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set GPRnopc:$Rd, (opnode GPRnopc:$Rn, t2_so_reg:$ShiftedRm))]>,
|
|
|
|
Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
2010-01-08 17:41:33 +00:00
|
|
|
let Inst{24} = 1;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{23-21} = op23_21;
|
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2009-11-24 00:20:27 +00:00
|
|
|
/// T2I_adde_sube_irs - Defines a set of (op reg, {so_imm|r|so_reg}) patterns
|
2010-02-16 20:42:29 +00:00
|
|
|
/// for a binary operation that produces a value and use the carry
|
2009-11-24 00:20:27 +00:00
|
|
|
/// bit. It's not predicable.
|
2011-08-30 01:34:54 +00:00
|
|
|
let Defs = [CPSR], Uses = [CPSR] in {
|
2016-03-08 16:23:54 +00:00
|
|
|
multiclass T2I_adde_sube_irs<bits<4> opcod, string opc, SDNode opnode,
|
2010-02-16 21:23:02 +00:00
|
|
|
bit Commutable = 0> {
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted imm
|
2010-11-14 05:37:38 +00:00
|
|
|
def ri : T2sTwoRegImm<(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_imm:$imm),
|
2010-11-12 21:12:40 +00:00
|
|
|
IIC_iALUi, opc, "\t$Rd, $Rn, $imm",
|
2011-08-30 01:34:54 +00:00
|
|
|
[(set rGPR:$Rd, CPSR, (opnode rGPR:$Rn, t2_so_imm:$imm, CPSR))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Requires<[IsThumb2]>, Sched<[WriteALU, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rr : T2sThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iALUr,
|
2010-11-12 21:12:40 +00:00
|
|
|
opc, ".w\t$Rd, $Rn, $Rm",
|
2011-08-30 01:34:54 +00:00
|
|
|
[(set rGPR:$Rd, CPSR, (opnode rGPR:$Rn, rGPR:$Rm, CPSR))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Requires<[IsThumb2]>, Sched<[WriteALU, ReadALU, ReadALU]> {
|
2009-06-26 00:19:44 +00:00
|
|
|
let isCommutable = Commutable;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
2009-06-26 00:19:44 +00:00
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
// shifted register
|
2010-11-14 05:37:38 +00:00
|
|
|
def rs : T2sTwoRegShiftedReg<
|
2010-11-19 17:11:02 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, t2_so_reg:$ShiftedRm),
|
2010-11-12 21:12:40 +00:00
|
|
|
IIC_iALUsi, opc, ".w\t$Rd, $Rn, $ShiftedRm",
|
2011-08-30 01:34:54 +00:00
|
|
|
[(set rGPR:$Rd, CPSR, (opnode rGPR:$Rn, t2_so_reg:$ShiftedRm, CPSR))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Requires<[IsThumb2]>, Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
}
|
2010-02-16 20:42:29 +00:00
|
|
|
}
|
2011-04-23 03:55:32 +00:00
|
|
|
}
|
2010-02-16 20:42:29 +00:00
|
|
|
|
2009-06-23 19:39:13 +00:00
|
|
|
/// T2I_sh_ir - Defines a set of (op reg, {so_imm|r}) patterns for a shift /
|
|
|
|
// rotate operation that produces a value.
|
2016-03-08 16:23:54 +00:00
|
|
|
multiclass T2I_sh_ir<bits<2> opcod, string opc, Operand ty, SDNode opnode> {
|
2009-06-23 19:39:13 +00:00
|
|
|
// 5-bit imm
|
2010-11-15 19:58:36 +00:00
|
|
|
def ri : T2sTwoRegShiftImm<
|
2011-08-08 20:42:17 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rm, ty:$imm), IIC_iMOVsi,
|
2010-11-15 19:58:36 +00:00
|
|
|
opc, ".w\t$Rd, $Rm, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rm, (i32 ty:$imm)))]>,
|
|
|
|
Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-21} = 0b010010;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
[ARM] Turn some undefined encoding bits into 0s.
The family of 32-bit Thumb instruction encodings that include t2ORR,
t2AND and t2EOR are all listed in the ArmARM as having (0) in bit 15.
The Tablegen descriptions of those instructions listed them as ?. This
change tightens that up by making them into 0 + Unpredictable.
In the specific case of t2ORR, we tighten it up still further by
making the zero bit mandatory. This change comes from Arm v8.1-M, in
which encodings with that bit equal to 1 will now be used for
different instructions.
Reviewers: dmgreen, samparker, SjoerdMeijer, efriedma
Reviewed By: dmgreen, efriedma
Subscribers: efriedma, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60705
llvm-svn: 362470
2019-06-04 08:28:48 +00:00
|
|
|
let Inst{15} = 0b0;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{5-4} = opcod;
|
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2010-11-15 19:58:36 +00:00
|
|
|
def rr : T2sThreeReg<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iMOVsr,
|
|
|
|
opc, ".w\t$Rd, $Rn, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rn, rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0100;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-4} = 0b0000;
|
|
|
|
}
|
2011-09-02 21:28:54 +00:00
|
|
|
|
|
|
|
// Optional destination register
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", ".w $Rdn, $imm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rdn, rGPR:$Rdn, ty:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 21:28:54 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", ".w $Rdn, $Rm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rdn, rGPR:$Rdn, rGPR:$Rm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 21:28:54 +00:00
|
|
|
|
|
|
|
// Assembler aliases w/o the ".w" suffix.
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rd, $Rn, $imm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rd, rGPR:$Rn, ty:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 21:28:54 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rd, $Rn, $Rm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 21:28:54 +00:00
|
|
|
|
|
|
|
// and with the optional destination operand, too.
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rdn, $imm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") rGPR:$Rdn, rGPR:$Rdn, ty:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-02 21:28:54 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${s}${p}", " $Rdn, $Rm"),
|
2012-08-02 21:59:52 +00:00
|
|
|
(!cast<Instruction>(NAME#"rr") rGPR:$Rdn, rGPR:$Rdn, rGPR:$Rm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2009-06-23 19:39:13 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2009-12-15 17:24:14 +00:00
|
|
|
/// T2I_cmp_irs - Defines a set of (op r, {so_imm|r|so_reg}) cmp / test
|
2009-06-23 19:39:13 +00:00
|
|
|
/// patterns. Similar to T2I_bin_irs except the instruction does not produce
|
2009-06-23 17:48:47 +00:00
|
|
|
/// a explicit result, only implicitly set CPSR.
|
2019-05-08 10:59:08 +00:00
|
|
|
multiclass T2I_cmp_irs<bits<4> opcod, string opc, RegisterClass LHSGPR,
|
2010-09-29 00:49:25 +00:00
|
|
|
InstrItinClass iii, InstrItinClass iir, InstrItinClass iis,
|
2016-03-08 16:23:54 +00:00
|
|
|
SDPatternOperator opnode> {
|
2011-09-06 21:44:58 +00:00
|
|
|
let isCompare = 1, Defs = [CPSR] in {
|
2009-06-23 17:48:47 +00:00
|
|
|
// shifted imm
|
2010-11-15 19:58:36 +00:00
|
|
|
def ri : T2OneRegCmpImm<
|
2019-05-08 10:59:08 +00:00
|
|
|
(outs), (ins LHSGPR:$Rn, t2_so_imm:$imm), iii,
|
2010-11-15 19:58:36 +00:00
|
|
|
opc, ".w\t$Rn, $imm",
|
2019-05-08 10:59:08 +00:00
|
|
|
[(opnode LHSGPR:$Rn, t2_so_imm:$imm)]>, Sched<[WriteCMP]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
// register
|
2010-11-15 19:58:36 +00:00
|
|
|
def rr : T2TwoRegCmp<
|
2019-05-08 10:59:08 +00:00
|
|
|
(outs), (ins LHSGPR:$Rn, rGPR:$Rm), iir,
|
2011-08-23 17:37:32 +00:00
|
|
|
opc, ".w\t$Rn, $Rm",
|
2019-05-08 10:59:08 +00:00
|
|
|
[(opnode LHSGPR:$Rn, rGPR:$Rm)]>, Sched<[WriteCMP]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
// shifted register
|
2010-11-15 19:58:36 +00:00
|
|
|
def rs : T2OneRegCmpShiftedReg<
|
2019-05-08 10:59:08 +00:00
|
|
|
(outs), (ins LHSGPR:$Rn, t2_so_reg:$ShiftedRm), iis,
|
2010-11-15 19:58:36 +00:00
|
|
|
opc, ".w\t$Rn, $ShiftedRm",
|
2019-05-08 10:59:08 +00:00
|
|
|
[(opnode LHSGPR:$Rn, t2_so_reg:$ShiftedRm)]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteCMPsi]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
}
|
2011-09-06 21:44:58 +00:00
|
|
|
|
|
|
|
// Assembler aliases w/o the ".w" suffix.
|
|
|
|
// No alias here for 'rr' version as not all instantiations of this
|
|
|
|
// multiclass want one (CMP in particular, does not).
|
|
|
|
def : t2InstAlias<!strconcat(opc, "${p}", " $Rn, $imm"),
|
2019-05-08 10:59:08 +00:00
|
|
|
(!cast<Instruction>(NAME#"ri") LHSGPR:$Rn, t2_so_imm:$imm, pred:$p)>;
|
2011-09-06 21:44:58 +00:00
|
|
|
def : t2InstAlias<!strconcat(opc, "${p}", " $Rn, $shift"),
|
2019-05-08 10:59:08 +00:00
|
|
|
(!cast<Instruction>(NAME#"rs") LHSGPR:$Rn, t2_so_reg:$shift, pred:$p)>;
|
2009-06-17 18:13:58 +00:00
|
|
|
}
|
|
|
|
|
2009-06-30 02:15:48 +00:00
|
|
|
/// T2I_ld - Defines a set of (op r, {imm12|imm8|so_reg}) load patterns.
|
2010-09-30 01:08:25 +00:00
|
|
|
multiclass T2I_ld<bit signed, bits<2> opcod, string opc,
|
2011-08-11 20:40:40 +00:00
|
|
|
InstrItinClass iii, InstrItinClass iis, RegisterClass target,
|
|
|
|
PatFrag opnode> {
|
|
|
|
def i12 : T2Ii12<(outs target:$Rt), (ins t2addrmode_imm12:$addr), iii,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, ".w\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(set target:$Rt, (opnode t2addrmode_imm12:$addr))]>,
|
|
|
|
Sched<[WriteLd]> {
|
2011-09-07 20:58:57 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<17> addr;
|
|
|
|
let Inst{31-25} = 0b1111100;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{24} = signed;
|
|
|
|
let Inst{23} = 1;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 1; // load
|
2010-11-30 22:45:47 +00:00
|
|
|
let Inst{19-16} = addr{16-13}; // Rn
|
2011-09-07 20:58:57 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-11-30 22:45:47 +00:00
|
|
|
let Inst{11-0} = addr{11-0}; // imm
|
2013-06-18 08:03:06 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadImm12";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2011-09-07 20:58:57 +00:00
|
|
|
def i8 : T2Ii8 <(outs target:$Rt), (ins t2addrmode_negimm8:$addr), iii,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, "\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(set target:$Rt, (opnode t2addrmode_negimm8:$addr))]>,
|
|
|
|
Sched<[WriteLd]> {
|
2011-09-07 20:58:57 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<13> addr;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24} = signed;
|
|
|
|
let Inst{23} = 0;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 1; // load
|
2011-09-07 20:58:57 +00:00
|
|
|
let Inst{19-16} = addr{12-9}; // Rn
|
|
|
|
let Inst{15-12} = Rt;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{11} = 1;
|
|
|
|
// Offset: index==TRUE, wback==FALSE
|
|
|
|
let Inst{10} = 1; // The P bit.
|
2010-11-29 22:44:32 +00:00
|
|
|
let Inst{9} = addr{8}; // U
|
2011-09-07 20:58:57 +00:00
|
|
|
let Inst{8} = 0; // The W bit.
|
2010-11-29 22:44:32 +00:00
|
|
|
let Inst{7-0} = addr{7-0}; // imm
|
2013-06-18 08:03:06 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadImm8";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2011-08-11 20:40:40 +00:00
|
|
|
def s : T2Iso <(outs target:$Rt), (ins t2addrmode_so_reg:$addr), iis,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, ".w\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(set target:$Rt, (opnode t2addrmode_so_reg:$addr))]>,
|
|
|
|
Sched<[WriteLd]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24} = signed;
|
|
|
|
let Inst{23} = 0;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 1; // load
|
|
|
|
let Inst{11-6} = 0b000000;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<4> Rt;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<10> addr;
|
|
|
|
let Inst{19-16} = addr{9-6}; // Rn
|
|
|
|
let Inst{3-0} = addr{5-2}; // Rm
|
|
|
|
let Inst{5-4} = addr{1-0}; // imm
|
2011-08-09 20:55:18 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadShift";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2010-11-03 05:14:24 +00:00
|
|
|
|
2012-01-18 22:04:42 +00:00
|
|
|
// pci variant is very similar to i12, but supports negative offsets
|
|
|
|
// from the PC.
|
2011-08-11 20:40:40 +00:00
|
|
|
def pci : T2Ipc <(outs target:$Rt), (ins t2ldrlabel:$addr), iii,
|
2011-02-08 22:39:40 +00:00
|
|
|
opc, ".w\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(set target:$Rt, (opnode (ARMWrapper tconstpool:$addr)))]>,
|
|
|
|
Sched<[WriteLd]> {
|
2011-02-08 22:39:40 +00:00
|
|
|
let isReMaterializable = 1;
|
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24} = signed;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 1; // load
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
2013-06-24 09:11:38 +00:00
|
|
|
|
|
|
|
bits<4> Rt;
|
2011-02-08 22:39:40 +00:00
|
|
|
let Inst{15-12} = Rt{3-0};
|
2013-06-24 09:11:38 +00:00
|
|
|
|
|
|
|
bits<13> addr;
|
|
|
|
let Inst{23} = addr{12}; // add = (U == '1')
|
2011-02-08 22:39:40 +00:00
|
|
|
let Inst{11-0} = addr{11-0};
|
2013-06-18 08:03:06 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadLabel";
|
2011-02-08 22:39:40 +00:00
|
|
|
}
|
2009-06-30 02:15:48 +00:00
|
|
|
}
|
|
|
|
|
2009-06-30 22:11:34 +00:00
|
|
|
/// T2I_st - Defines a set of (op r, {imm12|imm8|so_reg}) store patterns.
|
2010-09-30 01:08:25 +00:00
|
|
|
multiclass T2I_st<bits<2> opcod, string opc,
|
2011-08-11 20:40:40 +00:00
|
|
|
InstrItinClass iii, InstrItinClass iis, RegisterClass target,
|
|
|
|
PatFrag opnode> {
|
|
|
|
def i12 : T2Ii12<(outs), (ins target:$Rt, t2addrmode_imm12:$addr), iii,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, ".w\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(opnode target:$Rt, t2addrmode_imm12:$addr)]>,
|
|
|
|
Sched<[WriteST]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0001;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 0; // !load
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<4> Rt;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-30 22:45:47 +00:00
|
|
|
bits<17> addr;
|
2011-04-12 18:48:00 +00:00
|
|
|
let addr{12} = 1; // add = TRUE
|
2010-11-30 22:45:47 +00:00
|
|
|
let Inst{19-16} = addr{16-13}; // Rn
|
|
|
|
let Inst{23} = addr{12}; // U
|
|
|
|
let Inst{11-0} = addr{11-0}; // imm
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2011-09-07 20:58:57 +00:00
|
|
|
def i8 : T2Ii8 <(outs), (ins target:$Rt, t2addrmode_negimm8:$addr), iii,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, "\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(opnode target:$Rt, t2addrmode_negimm8:$addr)]>,
|
|
|
|
Sched<[WriteST]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0000;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 0; // !load
|
|
|
|
let Inst{11} = 1;
|
|
|
|
// Offset: index==TRUE, wback==FALSE
|
|
|
|
let Inst{10} = 1; // The P bit.
|
|
|
|
let Inst{8} = 0; // The W bit.
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<4> Rt;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<13> addr;
|
|
|
|
let Inst{19-16} = addr{12-9}; // Rn
|
|
|
|
let Inst{9} = addr{8}; // U
|
|
|
|
let Inst{7-0} = addr{7-0}; // imm
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2011-08-11 20:40:40 +00:00
|
|
|
def s : T2Iso <(outs), (ins target:$Rt, t2addrmode_so_reg:$addr), iis,
|
2010-11-29 22:44:32 +00:00
|
|
|
opc, ".w\t$Rt, $addr",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(opnode target:$Rt, t2addrmode_so_reg:$addr)]>,
|
|
|
|
Sched<[WriteST]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0000;
|
|
|
|
let Inst{22-21} = opcod;
|
|
|
|
let Inst{20} = 0; // !load
|
|
|
|
let Inst{11-6} = 0b000000;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<4> Rt;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 22:44:32 +00:00
|
|
|
bits<10> addr;
|
|
|
|
let Inst{19-16} = addr{9-6}; // Rn
|
|
|
|
let Inst{3-0} = addr{5-2}; // Rm
|
|
|
|
let Inst{5-4} = addr{1-0}; // imm
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-30 22:11:34 +00:00
|
|
|
}
|
|
|
|
|
2010-09-30 01:08:25 +00:00
|
|
|
/// T2I_ext_rrot - A unary operation with two forms: one whose operand is a
|
2009-07-03 01:43:10 +00:00
|
|
|
/// register and one whose operand is a register rotated by 8/16/24.
|
2017-02-17 15:42:44 +00:00
|
|
|
class T2I_ext_rrot_base<bits<3> opcod, dag iops, dag oops,
|
|
|
|
string opc, string oprs,
|
|
|
|
list<dag> pattern>
|
|
|
|
: T2TwoReg<iops, oops, IIC_iEXTr, opc, oprs, pattern> {
|
2011-07-27 17:48:13 +00:00
|
|
|
bits<2> rot;
|
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0100;
|
|
|
|
let Inst{22-20} = opcod;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7} = 1;
|
2017-02-17 15:42:44 +00:00
|
|
|
let Inst{5-4} = rot; // rotate
|
2010-03-04 22:24:41 +00:00
|
|
|
}
|
|
|
|
|
2017-02-17 15:42:44 +00:00
|
|
|
class T2I_ext_rrot<bits<3> opcod, string opc>
|
|
|
|
: T2I_ext_rrot_base<opcod,
|
|
|
|
(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rm, rot_imm:$rot),
|
|
|
|
opc, ".w\t$Rd, $Rm$rot", []>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2]>,
|
|
|
|
Sched<[WriteALU, ReadALU]>;
|
2017-02-17 15:42:44 +00:00
|
|
|
|
|
|
|
// UXTB16, SXTB16 - Requires HasDSP, does not need the .w qualifier.
|
|
|
|
class T2I_ext_rrot_xtb16<bits<3> opcod, string opc>
|
|
|
|
: T2I_ext_rrot_base<opcod,
|
|
|
|
(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rm, rot_imm:$rot),
|
|
|
|
opc, "\t$Rd, $Rm$rot", []>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>,
|
|
|
|
Sched<[WriteALU, ReadALU]>;
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
|
2010-09-30 01:08:25 +00:00
|
|
|
/// T2I_exta_rrot - A binary operation with two forms: one whose operand is a
|
2009-07-03 01:43:10 +00:00
|
|
|
/// register and one whose operand is a register rotated by 8/16/24.
|
2017-02-17 15:42:44 +00:00
|
|
|
class T2I_exta_rrot<bits<3> opcod, string opc>
|
2011-07-27 17:48:13 +00:00
|
|
|
: T2ThreeReg<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm, rot_imm:$rot),
|
2015-01-19 16:36:02 +00:00
|
|
|
IIC_iEXTAsr, opc, "\t$Rd, $Rn, $Rm$rot", []>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2011-07-27 17:48:13 +00:00
|
|
|
bits<2> rot;
|
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0100;
|
|
|
|
let Inst{22-20} = opcod;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7} = 1;
|
|
|
|
let Inst{5-4} = rot;
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
}
|
|
|
|
|
2009-06-27 02:26:13 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Instructions
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-06-24 23:47:58 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Miscellaneous Instructions.
|
|
|
|
//
|
|
|
|
|
2010-11-15 21:30:39 +00:00
|
|
|
class T2PCOneRegImm<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string asm, list<dag> pattern>
|
|
|
|
: T2XI<oops, iops, itin, asm, pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<12> label;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-15 21:30:39 +00:00
|
|
|
let Inst{26} = label{11};
|
|
|
|
let Inst{14-12} = label{10-8};
|
|
|
|
let Inst{7-0} = label{7-0};
|
|
|
|
}
|
|
|
|
|
2009-06-24 23:47:58 +00:00
|
|
|
// LEApcrel - Load a pc-relative address into a register without offending the
|
|
|
|
// assembler.
|
2010-12-14 00:36:49 +00:00
|
|
|
def t2ADR : T2PCOneRegImm<(outs rGPR:$Rd),
|
|
|
|
(ins t2adrlabel:$addr, pred:$p),
|
2013-06-06 16:35:25 +00:00
|
|
|
IIC_iALUi, "adr{$p}.w\t$Rd, $addr", []>,
|
|
|
|
Sched<[WriteALU, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25-24} = 0b10;
|
|
|
|
// Inst{23:21} = '11' (add = FALSE) or '00' (add = TRUE)
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{20} = 0;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{15} = 0;
|
2010-12-14 20:46:39 +00:00
|
|
|
|
2010-12-14 00:36:49 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<13> addr;
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{23} = addr{12};
|
|
|
|
let Inst{21} = addr{12};
|
|
|
|
let Inst{26} = addr{11};
|
|
|
|
let Inst{14-12} = addr{10-8};
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2011-09-09 22:24:36 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2Adr";
|
2010-12-13 22:51:08 +00:00
|
|
|
}
|
2010-12-14 00:36:49 +00:00
|
|
|
|
2017-05-16 17:59:07 +00:00
|
|
|
let hasSideEffects = 0, isReMaterializable = 1 in
|
2010-12-15 18:48:45 +00:00
|
|
|
def t2LEApcrel : t2PseudoInst<(outs rGPR:$Rd), (ins i32imm:$label, pred:$p),
|
2013-06-06 16:35:25 +00:00
|
|
|
4, IIC_iALUi, []>, Sched<[WriteALU, ReadALU]>;
|
2012-08-24 21:44:11 +00:00
|
|
|
let hasSideEffects = 1 in
|
2013-09-05 11:59:43 +00:00
|
|
|
def t2LEApcrelJT : t2PseudoInst<(outs rGPR:$Rd),
|
2015-05-13 20:28:38 +00:00
|
|
|
(ins i32imm:$label, pred:$p),
|
2011-07-13 23:22:26 +00:00
|
|
|
4, IIC_iALUi,
|
2013-06-06 16:35:25 +00:00
|
|
|
[]>, Sched<[WriteALU, ReadALU]>;
|
2009-06-24 23:47:58 +00:00
|
|
|
|
2010-12-08 23:30:19 +00:00
|
|
|
|
2009-06-27 02:26:13 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Load / store Instructions.
|
|
|
|
//
|
|
|
|
|
2009-06-29 07:51:04 +00:00
|
|
|
// Load
|
2010-02-27 23:47:46 +00:00
|
|
|
let canFoldAsLoad = 1, isReMaterializable = 1 in
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2LDR : T2I_ld<0, 0b10, "ldr", IIC_iLoad_i, IIC_iLoad_si, GPR, load>;
|
2009-06-30 02:15:48 +00:00
|
|
|
|
|
|
|
// Loads with zero extension
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2LDRH : T2I_ld<0, 0b01, "ldrh", IIC_iLoad_bh_i, IIC_iLoad_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
GPRnopc, zextloadi16>;
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2LDRB : T2I_ld<0, 0b00, "ldrb", IIC_iLoad_bh_i, IIC_iLoad_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
GPRnopc, zextloadi8>;
|
2009-06-30 02:15:48 +00:00
|
|
|
|
|
|
|
// Loads with sign extension
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2LDRSH : T2I_ld<1, 0b01, "ldrsh", IIC_iLoad_bh_i, IIC_iLoad_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
GPRnopc, sextloadi16>;
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2LDRSB : T2I_ld<1, 0b00, "ldrsb", IIC_iLoad_bh_i, IIC_iLoad_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
GPRnopc, sextloadi8>;
|
2009-06-30 02:15:48 +00:00
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
let mayLoad = 1, hasSideEffects = 0, hasExtraDefRegAllocReq = 1 in {
|
2009-06-30 02:15:48 +00:00
|
|
|
// Load doubleword
|
2010-12-01 19:18:46 +00:00
|
|
|
def t2LDRDi8 : T2Ii8s4<1, 0, 1, (outs rGPR:$Rt, rGPR:$Rt2),
|
2009-09-27 09:46:04 +00:00
|
|
|
(ins t2addrmode_imm8s4:$addr),
|
2020-03-09 13:29:37 +00:00
|
|
|
IIC_iLoad_d_i, "ldrd", "\t$Rt, $Rt2, $addr", "",
|
|
|
|
[(set rGPR:$Rt, rGPR:$Rt2, (ARMldrd t2addrmode_imm8s4:$addr))]>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Sched<[WriteLd]>;
|
2014-11-26 00:46:26 +00:00
|
|
|
} // mayLoad = 1, hasSideEffects = 0, hasExtraDefRegAllocReq = 1
|
2009-06-30 02:15:48 +00:00
|
|
|
|
|
|
|
// zextload i1 -> zextload i8
|
|
|
|
def : T2Pat<(zextloadi1 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRBi12 t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(zextloadi1 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRBi8 t2addrmode_negimm8:$addr)>;
|
2009-06-30 02:15:48 +00:00
|
|
|
def : T2Pat<(zextloadi1 t2addrmode_so_reg:$addr),
|
|
|
|
(t2LDRBs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(zextloadi1 (ARMWrapper tconstpool:$addr)),
|
|
|
|
(t2LDRBpci tconstpool:$addr)>;
|
|
|
|
|
|
|
|
// extload -> zextload
|
|
|
|
// FIXME: Reduce the number of patterns by legalizing extload to zextload
|
|
|
|
// earlier?
|
|
|
|
def : T2Pat<(extloadi1 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRBi12 t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(extloadi1 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRBi8 t2addrmode_negimm8:$addr)>;
|
2009-06-30 02:15:48 +00:00
|
|
|
def : T2Pat<(extloadi1 t2addrmode_so_reg:$addr),
|
|
|
|
(t2LDRBs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(extloadi1 (ARMWrapper tconstpool:$addr)),
|
|
|
|
(t2LDRBpci tconstpool:$addr)>;
|
|
|
|
|
|
|
|
def : T2Pat<(extloadi8 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRBi12 t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(extloadi8 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRBi8 t2addrmode_negimm8:$addr)>;
|
2009-06-30 02:15:48 +00:00
|
|
|
def : T2Pat<(extloadi8 t2addrmode_so_reg:$addr),
|
|
|
|
(t2LDRBs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(extloadi8 (ARMWrapper tconstpool:$addr)),
|
|
|
|
(t2LDRBpci tconstpool:$addr)>;
|
|
|
|
|
|
|
|
def : T2Pat<(extloadi16 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRHi12 t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(extloadi16 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRHi8 t2addrmode_negimm8:$addr)>;
|
2009-06-30 02:15:48 +00:00
|
|
|
def : T2Pat<(extloadi16 t2addrmode_so_reg:$addr),
|
|
|
|
(t2LDRHs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(extloadi16 (ARMWrapper tconstpool:$addr)),
|
|
|
|
(t2LDRHpci tconstpool:$addr)>;
|
2009-06-29 07:51:04 +00:00
|
|
|
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
// FIXME: The destination register of the loads and stores can't be PC, but
|
|
|
|
// can be SP. We need another regclass (similar to rGPR) to represent
|
|
|
|
// that. Not a pressing issue since these are selected manually,
|
|
|
|
// not via pattern.
|
|
|
|
|
2009-07-02 07:28:31 +00:00
|
|
|
// Indexed loads
|
2010-11-30 00:14:31 +00:00
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
let mayLoad = 1, hasSideEffects = 0 in {
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDR_PRE : T2Ipreldst<0, 0b10, 1, 1, (outs GPR:$Rt, GPR:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iLoad_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldr", "\t$Rt, $addr!", "$addr.base = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2009-07-02 07:28:31 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDR_POST : T2Ipostldst<0, 0b10, 1, 0, (outs GPR:$Rt, GPR:$Rn_wb),
|
2011-09-08 01:01:32 +00:00
|
|
|
(ins addr_offset_none:$Rn, t2am_imm8_offset:$offset),
|
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iLoad_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldr", "\t$Rt, $Rn$offset", "$Rn = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2009-07-02 07:28:31 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRB_PRE : T2Ipreldst<0, 0b00, 1, 1, (outs GPR:$Rt, GPR:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iLoad_bh_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrb", "\t$Rt, $addr!", "$addr.base = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2013-07-22 09:06:12 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRB_POST : T2Ipostldst<0, 0b00, 1, 0, (outs GPR:$Rt, GPR:$Rn_wb),
|
2011-09-08 01:01:32 +00:00
|
|
|
(ins addr_offset_none:$Rn, t2am_imm8_offset:$offset),
|
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iLoad_bh_iu,
|
2019-05-15 12:41:58 +00:00
|
|
|
"ldrb", "\t$Rt, $Rn$offset", "$Rn = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2009-07-02 07:28:31 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRH_PRE : T2Ipreldst<0, 0b01, 1, 1, (outs GPR:$Rt, GPR:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iLoad_bh_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrh", "\t$Rt, $addr!", "$addr.base = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2013-07-22 09:06:12 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRH_POST : T2Ipostldst<0, 0b01, 1, 0, (outs GPR:$Rt, GPR:$Rn_wb),
|
2011-09-08 01:01:32 +00:00
|
|
|
(ins addr_offset_none:$Rn, t2am_imm8_offset:$offset),
|
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iLoad_bh_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrh", "\t$Rt, $Rn$offset", "$Rn = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2009-07-02 07:28:31 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRSB_PRE : T2Ipreldst<1, 0b00, 1, 1, (outs GPR:$Rt, GPR:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iLoad_bh_iu,
|
2011-09-08 00:39:19 +00:00
|
|
|
"ldrsb", "\t$Rt, $addr!", "$addr.base = $Rn_wb",
|
2017-02-22 07:22:57 +00:00
|
|
|
[]>, Sched<[WriteLd]>;
|
2013-07-22 09:06:12 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRSB_POST : T2Ipostldst<1, 0b00, 1, 0, (outs GPR:$Rt, GPR:$Rn_wb),
|
2011-09-08 01:01:32 +00:00
|
|
|
(ins addr_offset_none:$Rn, t2am_imm8_offset:$offset),
|
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iLoad_bh_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrsb", "\t$Rt, $Rn$offset", "$Rn = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2009-07-02 23:16:11 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRSH_PRE : T2Ipreldst<1, 0b01, 1, 1, (outs GPR:$Rt, GPR:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iLoad_bh_iu,
|
2011-09-08 00:39:19 +00:00
|
|
|
"ldrsh", "\t$Rt, $addr!", "$addr.base = $Rn_wb",
|
2017-02-22 07:22:57 +00:00
|
|
|
[]>, Sched<[WriteLd]>;
|
2013-07-22 09:06:12 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2LDRSH_POST : T2Ipostldst<1, 0b01, 1, 0, (outs GPR:$Rt, GPR:$Rn_wb),
|
2011-09-08 01:01:32 +00:00
|
|
|
(ins addr_offset_none:$Rn, t2am_imm8_offset:$offset),
|
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iLoad_bh_iu,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrsh", "\t$Rt, $Rn$offset", "$Rn = $Rn_wb", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2014-11-26 00:46:26 +00:00
|
|
|
} // mayLoad = 1, hasSideEffects = 0
|
2009-07-02 23:16:11 +00:00
|
|
|
|
2011-09-07 23:39:14 +00:00
|
|
|
// LDRT, LDRBT, LDRHT, LDRSBT, LDRSHT all have offset mode (PUW=0b110).
|
2010-03-03 18:45:36 +00:00
|
|
|
// Ref: A8.6.57 LDR (immediate, Thumb) Encoding T4
|
2010-09-30 01:08:25 +00:00
|
|
|
class T2IldT<bit signed, bits<2> type, string opc, InstrItinClass ii>
|
2011-09-07 23:39:14 +00:00
|
|
|
: T2Ii8<(outs rGPR:$Rt), (ins t2addrmode_posimm8:$addr), ii, opc,
|
2017-02-22 07:22:57 +00:00
|
|
|
"\t$Rt, $addr", []>, Sched<[WriteLd]> {
|
2011-09-07 23:39:14 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<13> addr;
|
2010-03-03 18:45:36 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24} = signed;
|
|
|
|
let Inst{23} = 0;
|
|
|
|
let Inst{22-21} = type;
|
|
|
|
let Inst{20} = 1; // load
|
2011-09-07 23:39:14 +00:00
|
|
|
let Inst{19-16} = addr{12-9};
|
|
|
|
let Inst{15-12} = Rt;
|
2010-03-03 18:45:36 +00:00
|
|
|
let Inst{11} = 1;
|
|
|
|
let Inst{10-8} = 0b110; // PUW.
|
2011-09-07 23:39:14 +00:00
|
|
|
let Inst{7-0} = addr{7-0};
|
2013-06-18 08:03:06 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadT";
|
2010-03-03 18:45:36 +00:00
|
|
|
}
|
|
|
|
|
2010-09-30 01:08:25 +00:00
|
|
|
def t2LDRT : T2IldT<0, 0b10, "ldrt", IIC_iLoad_i>;
|
|
|
|
def t2LDRBT : T2IldT<0, 0b00, "ldrbt", IIC_iLoad_bh_i>;
|
|
|
|
def t2LDRHT : T2IldT<0, 0b01, "ldrht", IIC_iLoad_bh_i>;
|
|
|
|
def t2LDRSBT : T2IldT<1, 0b00, "ldrsbt", IIC_iLoad_bh_i>;
|
|
|
|
def t2LDRSHT : T2IldT<1, 0b01, "ldrsht", IIC_iLoad_bh_i>;
|
2010-03-03 18:45:36 +00:00
|
|
|
|
2013-08-28 16:39:20 +00:00
|
|
|
class T2Ildacq<bits<4> bits23_20, bits<2> bit54, dag oops, dag iops,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: Thumb2I<oops, iops, AddrModeNone, 4, NoItinerary,
|
2016-01-15 10:23:46 +00:00
|
|
|
opc, asm, "", pattern>, Requires<[IsThumb, HasAcquireRelease]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> addr;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-24} = 0b000;
|
|
|
|
let Inst{23-20} = bits23_20;
|
|
|
|
let Inst{11-6} = 0b111110;
|
|
|
|
let Inst{5-4} = bit54;
|
|
|
|
let Inst{3-0} = 0b1111;
|
|
|
|
|
|
|
|
// Encode instruction operands
|
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
}
|
|
|
|
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2LDA : T2Ildacq<0b1101, 0b10, (outs rGPR:$Rt),
|
2017-02-22 07:22:57 +00:00
|
|
|
(ins addr_offset_none:$addr), "lda", "\t$Rt, $addr", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2LDAB : T2Ildacq<0b1101, 0b00, (outs rGPR:$Rt),
|
2017-02-22 07:22:57 +00:00
|
|
|
(ins addr_offset_none:$addr), "ldab", "\t$Rt, $addr", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2LDAH : T2Ildacq<0b1101, 0b01, (outs rGPR:$Rt),
|
2017-02-22 07:22:57 +00:00
|
|
|
(ins addr_offset_none:$addr), "ldah", "\t$Rt, $addr", []>,
|
|
|
|
Sched<[WriteLd]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
|
2009-06-30 22:11:34 +00:00
|
|
|
// Store
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2STR :T2I_st<0b10,"str", IIC_iStore_i, IIC_iStore_si, GPR, store>;
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2STRB:T2I_st<0b00,"strb", IIC_iStore_bh_i, IIC_iStore_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
rGPR, truncstorei8>;
|
2010-10-28 06:47:08 +00:00
|
|
|
defm t2STRH:T2I_st<0b01,"strh", IIC_iStore_bh_i, IIC_iStore_bh_si,
|
2016-03-08 16:23:54 +00:00
|
|
|
rGPR, truncstorei16>;
|
2009-06-30 22:11:34 +00:00
|
|
|
|
2009-06-30 22:50:01 +00:00
|
|
|
// Store doubleword
|
2014-11-26 00:46:26 +00:00
|
|
|
let mayStore = 1, hasSideEffects = 0, hasExtraSrcRegAllocReq = 1 in
|
2009-12-15 17:24:14 +00:00
|
|
|
def t2STRDi8 : T2Ii8s4<1, 0, 0, (outs),
|
2014-04-04 08:14:13 +00:00
|
|
|
(ins rGPR:$Rt, rGPR:$Rt2, t2addrmode_imm8s4:$addr),
|
2020-03-09 13:29:37 +00:00
|
|
|
IIC_iStore_d_r, "strd", "\t$Rt, $Rt2, $addr", "",
|
|
|
|
[(ARMstrd rGPR:$Rt, rGPR:$Rt2, t2addrmode_imm8s4:$addr)]>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Sched<[WriteST]>;
|
2009-06-30 22:50:01 +00:00
|
|
|
|
2009-07-03 00:06:39 +00:00
|
|
|
// Indexed stores
|
2011-10-16 06:38:10 +00:00
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
let mayStore = 1, hasSideEffects = 0 in {
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2STR_PRE : T2Ipreldst<0, 0b10, 0, 1, (outs GPRnopc:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins GPRnopc:$Rt, t2addrmode_imm8_pre:$addr),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iStore_iu,
|
2011-09-16 21:55:56 +00:00
|
|
|
"str", "\t$Rt, $addr!",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $Rn_wb,@earlyclobber $Rn_wb", []>,
|
|
|
|
Sched<[WriteST]>;
|
2013-07-22 09:06:12 +00:00
|
|
|
|
2011-09-16 21:55:56 +00:00
|
|
|
def t2STRH_PRE : T2Ipreldst<0, 0b01, 0, 1, (outs GPRnopc:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins rGPR:$Rt, t2addrmode_imm8_pre:$addr),
|
2011-09-16 21:55:56 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iStore_iu,
|
|
|
|
"strh", "\t$Rt, $addr!",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $Rn_wb,@earlyclobber $Rn_wb", []>,
|
|
|
|
Sched<[WriteST]>;
|
2011-09-16 21:55:56 +00:00
|
|
|
|
|
|
|
def t2STRB_PRE : T2Ipreldst<0, 0b00, 0, 1, (outs GPRnopc:$Rn_wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins rGPR:$Rt, t2addrmode_imm8_pre:$addr),
|
2011-09-16 21:55:56 +00:00
|
|
|
AddrModeT2_i8, IndexModePre, IIC_iStore_bh_iu,
|
|
|
|
"strb", "\t$Rt, $addr!",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $Rn_wb,@earlyclobber $Rn_wb", []>,
|
|
|
|
Sched<[WriteST]>;
|
2014-11-26 00:46:26 +00:00
|
|
|
} // mayStore = 1, hasSideEffects = 0
|
2009-07-03 00:06:39 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2STR_POST : T2Ipostldst<0, 0b10, 0, 0, (outs GPRnopc:$Rn_wb),
|
2011-12-13 21:10:25 +00:00
|
|
|
(ins GPRnopc:$Rt, addr_offset_none:$Rn,
|
2011-09-16 21:09:00 +00:00
|
|
|
t2am_imm8_offset:$offset),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iStore_iu,
|
2011-09-23 21:26:40 +00:00
|
|
|
"str", "\t$Rt, $Rn$offset",
|
2011-09-08 00:39:19 +00:00
|
|
|
"$Rn = $Rn_wb,@earlyclobber $Rn_wb",
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2011-12-13 21:10:25 +00:00
|
|
|
(post_store GPRnopc:$Rt, addr_offset_none:$Rn,
|
2017-02-22 07:22:57 +00:00
|
|
|
t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2009-07-03 00:06:39 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2STRH_POST : T2Ipostldst<0, 0b01, 0, 0, (outs GPRnopc:$Rn_wb),
|
2011-09-16 21:09:00 +00:00
|
|
|
(ins rGPR:$Rt, addr_offset_none:$Rn,
|
|
|
|
t2am_imm8_offset:$offset),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iStore_bh_iu,
|
2011-09-23 21:26:40 +00:00
|
|
|
"strh", "\t$Rt, $Rn$offset",
|
2011-09-08 00:39:19 +00:00
|
|
|
"$Rn = $Rn_wb,@earlyclobber $Rn_wb",
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2011-09-16 21:09:00 +00:00
|
|
|
(post_truncsti16 rGPR:$Rt, addr_offset_none:$Rn,
|
2017-02-22 07:22:57 +00:00
|
|
|
t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2009-07-03 00:06:39 +00:00
|
|
|
|
2011-09-08 00:39:19 +00:00
|
|
|
def t2STRB_POST : T2Ipostldst<0, 0b00, 0, 0, (outs GPRnopc:$Rn_wb),
|
2011-09-16 21:09:00 +00:00
|
|
|
(ins rGPR:$Rt, addr_offset_none:$Rn,
|
|
|
|
t2am_imm8_offset:$offset),
|
2010-09-30 01:08:25 +00:00
|
|
|
AddrModeT2_i8, IndexModePost, IIC_iStore_bh_iu,
|
2011-09-23 21:26:40 +00:00
|
|
|
"strb", "\t$Rt, $Rn$offset",
|
2011-09-08 00:39:19 +00:00
|
|
|
"$Rn = $Rn_wb,@earlyclobber $Rn_wb",
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2011-09-16 21:09:00 +00:00
|
|
|
(post_truncsti8 rGPR:$Rt, addr_offset_none:$Rn,
|
2017-02-22 07:22:57 +00:00
|
|
|
t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2009-07-03 00:06:39 +00:00
|
|
|
|
2011-09-16 21:55:56 +00:00
|
|
|
// Pseudo-instructions for pattern matching the pre-indexed stores. We can't
|
|
|
|
// put the patterns on the instruction definitions directly as ISel wants
|
|
|
|
// the address base and offset to be separate operands, not a single
|
|
|
|
// complex operand like we represent the instructions themselves. The
|
|
|
|
// pseudos map between the two.
|
|
|
|
let usesCustomInserter = 1,
|
|
|
|
Constraints = "$Rn = $Rn_wb,@earlyclobber $Rn_wb" in {
|
|
|
|
def t2STR_preidx: t2PseudoInst<(outs GPRnopc:$Rn_wb),
|
|
|
|
(ins rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset, pred:$p),
|
|
|
|
4, IIC_iStore_ru,
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2017-02-22 07:22:57 +00:00
|
|
|
(pre_store rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2011-09-16 21:55:56 +00:00
|
|
|
def t2STRB_preidx: t2PseudoInst<(outs GPRnopc:$Rn_wb),
|
|
|
|
(ins rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset, pred:$p),
|
|
|
|
4, IIC_iStore_ru,
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2017-02-22 07:22:57 +00:00
|
|
|
(pre_truncsti8 rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2011-09-16 21:55:56 +00:00
|
|
|
def t2STRH_preidx: t2PseudoInst<(outs GPRnopc:$Rn_wb),
|
|
|
|
(ins rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset, pred:$p),
|
|
|
|
4, IIC_iStore_ru,
|
|
|
|
[(set GPRnopc:$Rn_wb,
|
2017-02-22 07:22:57 +00:00
|
|
|
(pre_truncsti16 rGPR:$Rt, GPRnopc:$Rn, t2am_imm8_offset:$offset))]>,
|
|
|
|
Sched<[WriteST]>;
|
2011-09-16 21:55:56 +00:00
|
|
|
}
|
|
|
|
|
2010-03-03 18:45:36 +00:00
|
|
|
// STRT, STRBT, STRHT all have offset mode (PUW=0b110) and are for disassembly
|
|
|
|
// only.
|
|
|
|
// Ref: A8.6.193 STR (immediate, Thumb) Encoding T4
|
2010-09-30 01:08:25 +00:00
|
|
|
class T2IstT<bits<2> type, string opc, InstrItinClass ii>
|
2011-04-13 21:04:32 +00:00
|
|
|
: T2Ii8<(outs rGPR:$Rt), (ins t2addrmode_imm8:$addr), ii, opc,
|
2017-02-22 07:22:57 +00:00
|
|
|
"\t$Rt, $addr", []>, Sched<[WriteST]> {
|
2010-03-03 18:45:36 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24} = 0; // not signed
|
|
|
|
let Inst{23} = 0;
|
|
|
|
let Inst{22-21} = type;
|
|
|
|
let Inst{20} = 0; // store
|
|
|
|
let Inst{11} = 1;
|
|
|
|
let Inst{10-8} = 0b110; // PUW
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-30 18:38:28 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<13> addr;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2010-11-30 18:38:28 +00:00
|
|
|
let Inst{19-16} = addr{12-9};
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2010-03-03 18:45:36 +00:00
|
|
|
}
|
|
|
|
|
2010-09-30 01:08:25 +00:00
|
|
|
def t2STRT : T2IstT<0b10, "strt", IIC_iStore_i>;
|
|
|
|
def t2STRBT : T2IstT<0b00, "strbt", IIC_iStore_bh_i>;
|
|
|
|
def t2STRHT : T2IstT<0b01, "strht", IIC_iStore_bh_i>;
|
2009-07-01 00:01:13 +00:00
|
|
|
|
2010-03-11 01:13:36 +00:00
|
|
|
// ldrd / strd pre / post variants
|
|
|
|
|
2015-10-22 01:48:57 +00:00
|
|
|
let mayLoad = 1 in
|
2011-09-08 22:07:06 +00:00
|
|
|
def t2LDRD_PRE : T2Ii8s4<1, 1, 1, (outs rGPR:$Rt, rGPR:$Rt2, GPR:$wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins t2addrmode_imm8s4_pre:$addr), IIC_iLoad_d_ru,
|
2017-02-22 07:22:57 +00:00
|
|
|
"ldrd", "\t$Rt, $Rt2, $addr!", "$addr.base = $wb", []>,
|
|
|
|
Sched<[WriteLd]> {
|
2011-09-08 22:07:06 +00:00
|
|
|
let DecoderMethod = "DecodeT2LDRDPreInstruction";
|
|
|
|
}
|
2010-03-11 01:13:36 +00:00
|
|
|
|
2015-10-22 01:48:57 +00:00
|
|
|
let mayLoad = 1 in
|
2011-09-08 22:07:06 +00:00
|
|
|
def t2LDRD_POST : T2Ii8s4post<0, 1, 1, (outs rGPR:$Rt, rGPR:$Rt2, GPR:$wb),
|
|
|
|
(ins addr_offset_none:$addr, t2am_imm8s4_offset:$imm),
|
2011-09-13 20:46:26 +00:00
|
|
|
IIC_iLoad_d_ru, "ldrd", "\t$Rt, $Rt2, $addr$imm",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $wb", []>, Sched<[WriteLd]>;
|
2010-03-11 01:13:36 +00:00
|
|
|
|
2015-10-22 01:48:57 +00:00
|
|
|
let mayStore = 1 in
|
2011-09-08 22:07:06 +00:00
|
|
|
def t2STRD_PRE : T2Ii8s4<1, 1, 0, (outs GPR:$wb),
|
2013-06-18 08:12:51 +00:00
|
|
|
(ins rGPR:$Rt, rGPR:$Rt2, t2addrmode_imm8s4_pre:$addr),
|
2011-09-08 22:07:06 +00:00
|
|
|
IIC_iStore_d_ru, "strd", "\t$Rt, $Rt2, $addr!",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $wb", []>, Sched<[WriteST]> {
|
2011-09-08 22:07:06 +00:00
|
|
|
let DecoderMethod = "DecodeT2STRDPreInstruction";
|
|
|
|
}
|
2010-03-11 01:13:36 +00:00
|
|
|
|
2015-10-22 01:48:57 +00:00
|
|
|
let mayStore = 1 in
|
2011-09-08 22:07:06 +00:00
|
|
|
def t2STRD_POST : T2Ii8s4post<0, 1, 0, (outs GPR:$wb),
|
|
|
|
(ins rGPR:$Rt, rGPR:$Rt2, addr_offset_none:$addr,
|
|
|
|
t2am_imm8s4_offset:$imm),
|
2011-09-13 20:46:26 +00:00
|
|
|
IIC_iStore_d_ru, "strd", "\t$Rt, $Rt2, $addr$imm",
|
2017-02-22 07:22:57 +00:00
|
|
|
"$addr.base = $wb", []>, Sched<[WriteST]>;
|
2009-07-03 00:18:36 +00:00
|
|
|
|
2013-08-28 16:39:20 +00:00
|
|
|
class T2Istrrel<bits<2> bit54, dag oops, dag iops,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: Thumb2I<oops, iops, AddrModeNone, 4, NoItinerary, opc,
|
2017-02-22 07:22:57 +00:00
|
|
|
asm, "", pattern>, Requires<[IsThumb, HasAcquireRelease]>,
|
|
|
|
Sched<[WriteST]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> addr;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0001100;
|
|
|
|
let Inst{11-6} = 0b111110;
|
|
|
|
let Inst{5-4} = bit54;
|
|
|
|
let Inst{3-0} = 0b1111;
|
|
|
|
|
|
|
|
// Encode instruction operands
|
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
}
|
|
|
|
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2STL : T2Istrrel<0b10, (outs), (ins rGPR:$Rt, addr_offset_none:$addr),
|
2013-08-27 17:38:16 +00:00
|
|
|
"stl", "\t$Rt, $addr", []>;
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2STLB : T2Istrrel<0b00, (outs), (ins rGPR:$Rt, addr_offset_none:$addr),
|
2013-08-27 17:38:16 +00:00
|
|
|
"stlb", "\t$Rt, $addr", []>;
|
2013-08-28 16:39:20 +00:00
|
|
|
def t2STLH : T2Istrrel<0b01, (outs), (ins rGPR:$Rt, addr_offset_none:$addr),
|
2013-08-27 17:38:16 +00:00
|
|
|
"stlh", "\t$Rt, $addr", []>;
|
|
|
|
|
2010-03-04 17:40:44 +00:00
|
|
|
// T2Ipl (Preload Data/Instruction) signals the memory system of possible future
|
2011-10-26 22:22:01 +00:00
|
|
|
// data/instruction access.
|
2010-11-03 06:34:55 +00:00
|
|
|
// instr_write is inverted for Thumb mode: (prefetch 3) -> (preload 0),
|
|
|
|
// (prefetch 1) -> (preload 2), (prefetch 2) -> (preload 1).
|
2010-11-04 05:19:35 +00:00
|
|
|
multiclass T2Ipl<bits<1> write, bits<1> instr, string opc> {
|
2010-03-04 17:40:44 +00:00
|
|
|
|
2010-11-03 06:34:55 +00:00
|
|
|
def i12 : T2Ii12<(outs), (ins t2addrmode_imm12:$addr), IIC_Preload, opc,
|
2010-11-03 05:14:24 +00:00
|
|
|
"\t$addr",
|
2013-06-06 18:06:30 +00:00
|
|
|
[(ARMPreload t2addrmode_imm12:$addr, (i32 write), (i32 instr))]>,
|
|
|
|
Sched<[WritePreLd]> {
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{31-25} = 0b1111100;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{24} = instr;
|
2013-06-24 09:11:38 +00:00
|
|
|
let Inst{23} = 1;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{22} = 0;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{21} = write;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{20} = 1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-30 22:45:47 +00:00
|
|
|
bits<17> addr;
|
|
|
|
let Inst{19-16} = addr{16-13}; // Rn
|
2010-11-30 19:19:31 +00:00
|
|
|
let Inst{11-0} = addr{11-0}; // imm12
|
2013-06-24 09:11:38 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadImm12";
|
2010-03-04 17:40:44 +00:00
|
|
|
}
|
|
|
|
|
2011-09-07 20:58:57 +00:00
|
|
|
def i8 : T2Ii8<(outs), (ins t2addrmode_negimm8:$addr), IIC_Preload, opc,
|
2010-11-03 05:14:24 +00:00
|
|
|
"\t$addr",
|
2013-06-06 18:06:30 +00:00
|
|
|
[(ARMPreload t2addrmode_negimm8:$addr, (i32 write), (i32 instr))]>,
|
|
|
|
Sched<[WritePreLd]> {
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{31-25} = 0b1111100;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{24} = instr;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{23} = 0; // U = 0
|
|
|
|
let Inst{22} = 0;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{21} = write;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{20} = 1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{11-8} = 0b1100;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-30 19:19:31 +00:00
|
|
|
bits<13> addr;
|
|
|
|
let Inst{19-16} = addr{12-9}; // Rn
|
|
|
|
let Inst{7-0} = addr{7-0}; // imm8
|
2013-06-24 09:11:38 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadImm8";
|
2010-03-04 17:40:44 +00:00
|
|
|
}
|
|
|
|
|
2010-11-03 06:34:55 +00:00
|
|
|
def s : T2Iso<(outs), (ins t2addrmode_so_reg:$addr), IIC_Preload, opc,
|
2010-11-03 05:14:24 +00:00
|
|
|
"\t$addr",
|
2013-06-06 18:06:30 +00:00
|
|
|
[(ARMPreload t2addrmode_so_reg:$addr, (i32 write), (i32 instr))]>,
|
|
|
|
Sched<[WritePreLd]> {
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{31-25} = 0b1111100;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{24} = instr;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{23} = 0; // add = TRUE for T1
|
|
|
|
let Inst{22} = 0;
|
2010-11-04 05:19:35 +00:00
|
|
|
let Inst{21} = write;
|
2010-03-04 17:40:44 +00:00
|
|
|
let Inst{20} = 1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
2013-06-24 09:11:38 +00:00
|
|
|
let Inst{11-6} = 0b000000;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-30 19:19:31 +00:00
|
|
|
bits<10> addr;
|
|
|
|
let Inst{19-16} = addr{9-6}; // Rn
|
|
|
|
let Inst{3-0} = addr{5-2}; // Rm
|
|
|
|
let Inst{5-4} = addr{1-0}; // imm2
|
2011-08-09 20:55:18 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadShift";
|
2010-03-04 17:40:44 +00:00
|
|
|
}
|
2013-08-06 16:07:46 +00:00
|
|
|
}
|
2013-06-24 09:11:38 +00:00
|
|
|
|
2013-08-06 16:07:46 +00:00
|
|
|
defm t2PLD : T2Ipl<0, 0, "pld">, Requires<[IsThumb2]>;
|
|
|
|
defm t2PLDW : T2Ipl<1, 0, "pldw">, Requires<[IsThumb2,HasV7,HasMP]>;
|
|
|
|
defm t2PLI : T2Ipl<0, 1, "pli">, Requires<[IsThumb2,HasV7]>;
|
2013-06-24 09:11:38 +00:00
|
|
|
|
2020-02-28 10:55:53 +00:00
|
|
|
// PLD/PLDW/PLI aliases w/ the optional .w suffix
|
|
|
|
def : t2InstAlias<"pld${p}.w\t$addr",
|
|
|
|
(t2PLDi12 t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"pld${p}.w\t$addr",
|
|
|
|
(t2PLDi8 t2addrmode_negimm8:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"pld${p}.w\t$addr",
|
|
|
|
(t2PLDs t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
|
|
|
|
def : InstAlias<"pldw${p}.w\t$addr",
|
|
|
|
(t2PLDWi12 t2addrmode_imm12:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7,HasMP]>;
|
|
|
|
def : InstAlias<"pldw${p}.w\t$addr",
|
|
|
|
(t2PLDWi8 t2addrmode_negimm8:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7,HasMP]>;
|
|
|
|
def : InstAlias<"pldw${p}.w\t$addr",
|
|
|
|
(t2PLDWs t2addrmode_so_reg:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7,HasMP]>;
|
|
|
|
|
|
|
|
def : InstAlias<"pli${p}.w\t$addr",
|
|
|
|
(t2PLIi12 t2addrmode_imm12:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
def : InstAlias<"pli${p}.w\t$addr",
|
|
|
|
(t2PLIi8 t2addrmode_negimm8:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
def : InstAlias<"pli${p}.w\t$addr",
|
|
|
|
(t2PLIs t2addrmode_so_reg:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
|
2013-08-06 16:07:46 +00:00
|
|
|
// pci variant is very similar to i12, but supports negative offsets
|
|
|
|
// from the PC. Only PLD and PLI have pci variants (not PLDW)
|
|
|
|
class T2Iplpci<bits<1> inst, string opc> : T2Iso<(outs), (ins t2ldrlabel:$addr),
|
2014-04-03 23:43:18 +00:00
|
|
|
IIC_Preload, opc, "\t$addr",
|
2013-08-06 16:07:46 +00:00
|
|
|
[(ARMPreload (ARMWrapper tconstpool:$addr),
|
|
|
|
(i32 0), (i32 inst))]>, Sched<[WritePreLd]> {
|
|
|
|
let Inst{31-25} = 0b1111100;
|
|
|
|
let Inst{24} = inst;
|
|
|
|
let Inst{22-20} = 0b001;
|
|
|
|
let Inst{19-16} = 0b1111;
|
|
|
|
let Inst{15-12} = 0b1111;
|
2013-06-24 09:11:38 +00:00
|
|
|
|
2013-08-06 16:07:46 +00:00
|
|
|
bits<13> addr;
|
|
|
|
let Inst{23} = addr{12}; // add = (U == '1')
|
|
|
|
let Inst{11-0} = addr{11-0}; // imm12
|
|
|
|
|
|
|
|
let DecoderMethod = "DecodeT2LoadLabel";
|
2010-03-04 17:40:44 +00:00
|
|
|
}
|
|
|
|
|
2013-08-06 16:07:46 +00:00
|
|
|
def t2PLDpci : T2Iplpci<0, "pld">, Requires<[IsThumb2]>;
|
|
|
|
def t2PLIpci : T2Iplpci<1, "pli">, Requires<[IsThumb2,HasV7]>;
|
2010-03-04 17:40:44 +00:00
|
|
|
|
2020-02-28 10:55:53 +00:00
|
|
|
def : t2InstAlias<"pld${p}.w $addr",
|
|
|
|
(t2PLDpci t2ldrlabel:$addr, pred:$p)>;
|
|
|
|
def : InstAlias<"pli${p}.w $addr",
|
|
|
|
(t2PLIpci t2ldrlabel:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
|
|
|
|
// PLD/PLI with alternate literal form.
|
|
|
|
def : t2InstAlias<"pld${p} $addr",
|
|
|
|
(t2PLDpci t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def : InstAlias<"pli${p} $addr",
|
|
|
|
(t2PLIpci t2ldr_pcrel_imm12:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
def : t2InstAlias<"pld${p}.w $addr",
|
|
|
|
(t2PLDpci t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def : InstAlias<"pli${p}.w $addr",
|
|
|
|
(t2PLIpci t2ldr_pcrel_imm12:$addr, pred:$p), 0>,
|
|
|
|
Requires<[IsThumb2,HasV7]>;
|
|
|
|
|
2009-07-03 00:18:36 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Load / store multiple Instructions.
|
|
|
|
//
|
|
|
|
|
2011-09-12 21:28:46 +00:00
|
|
|
multiclass thumb2_ld_mult<string asm, InstrItinClass itin,
|
2010-11-13 09:09:38 +00:00
|
|
|
InstrItinClass itin_upd, bit L_bit> {
|
2010-11-16 01:16:36 +00:00
|
|
|
def IA :
|
2010-11-13 09:09:38 +00:00
|
|
|
T2XI<(outs), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
2011-09-07 16:22:42 +00:00
|
|
|
itin, !strconcat(asm, "${p}.w\t$Rn, $regs"), []> {
|
2010-11-13 09:09:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-13 09:09:38 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b01; // Increment After
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 0; // No writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
2011-10-24 17:16:24 +00:00
|
|
|
let Inst{15-0} = regs;
|
2010-11-13 09:09:38 +00:00
|
|
|
}
|
2010-11-16 01:16:36 +00:00
|
|
|
def IA_UPD :
|
2010-11-13 09:09:38 +00:00
|
|
|
T2XIt<(outs GPR:$wb), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
2011-09-07 16:22:42 +00:00
|
|
|
itin_upd, !strconcat(asm, "${p}.w\t$Rn!, $regs"), "$Rn = $wb", []> {
|
2010-11-13 09:09:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-13 09:09:38 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b01; // Increment After
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 1; // Writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
2011-10-24 17:16:24 +00:00
|
|
|
let Inst{15-0} = regs;
|
2010-11-13 09:09:38 +00:00
|
|
|
}
|
2010-11-16 01:16:36 +00:00
|
|
|
def DB :
|
2010-11-13 09:09:38 +00:00
|
|
|
T2XI<(outs), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
2011-09-07 18:39:47 +00:00
|
|
|
itin, !strconcat(asm, "db${p}\t$Rn, $regs"), []> {
|
2010-11-13 09:09:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b10; // Decrement Before
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 0; // No writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
2011-10-24 17:16:24 +00:00
|
|
|
let Inst{15-0} = regs;
|
2010-11-13 09:09:38 +00:00
|
|
|
}
|
2010-11-16 01:16:36 +00:00
|
|
|
def DB_UPD :
|
2010-11-13 09:09:38 +00:00
|
|
|
T2XIt<(outs GPR:$wb), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
2011-09-07 18:39:47 +00:00
|
|
|
itin_upd, !strconcat(asm, "db${p}\t$Rn!, $regs"), "$Rn = $wb", []> {
|
2010-11-13 09:09:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b10; // Decrement Before
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 1; // Writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
2011-10-24 17:16:24 +00:00
|
|
|
let Inst{15-0} = regs;
|
2010-11-13 09:09:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
let hasSideEffects = 0 in {
|
2010-11-13 10:57:02 +00:00
|
|
|
|
2018-12-03 10:32:42 +00:00
|
|
|
let mayLoad = 1, hasExtraDefRegAllocReq = 1, variadicOpsAreDefs = 1 in
|
2011-09-12 21:28:46 +00:00
|
|
|
defm t2LDM : thumb2_ld_mult<"ldm", IIC_iLoad_m, IIC_iLoad_mu, 1>;
|
|
|
|
|
|
|
|
multiclass thumb2_st_mult<string asm, InstrItinClass itin,
|
|
|
|
InstrItinClass itin_upd, bit L_bit> {
|
|
|
|
def IA :
|
|
|
|
T2XI<(outs), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
|
|
|
itin, !strconcat(asm, "${p}.w\t$Rn, $regs"), []> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b01; // Increment After
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 0; // No writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{14} = regs{14};
|
|
|
|
let Inst{13} = 0;
|
|
|
|
let Inst{12-0} = regs{12-0};
|
|
|
|
}
|
|
|
|
def IA_UPD :
|
|
|
|
T2XIt<(outs GPR:$wb), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
|
|
|
itin_upd, !strconcat(asm, "${p}.w\t$Rn!, $regs"), "$Rn = $wb", []> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b01; // Increment After
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 1; // Writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{14} = regs{14};
|
|
|
|
let Inst{13} = 0;
|
|
|
|
let Inst{12-0} = regs{12-0};
|
|
|
|
}
|
|
|
|
def DB :
|
|
|
|
T2XI<(outs), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
|
|
|
itin, !strconcat(asm, "db${p}\t$Rn, $regs"), []> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b10; // Decrement Before
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 0; // No writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{14} = regs{14};
|
|
|
|
let Inst{13} = 0;
|
|
|
|
let Inst{12-0} = regs{12-0};
|
|
|
|
}
|
|
|
|
def DB_UPD :
|
|
|
|
T2XIt<(outs GPR:$wb), (ins GPR:$Rn, pred:$p, reglist:$regs, variable_ops),
|
|
|
|
itin_upd, !strconcat(asm, "db${p}\t$Rn!, $regs"), "$Rn = $wb", []> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b00;
|
|
|
|
let Inst{24-23} = 0b10; // Decrement Before
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = 1; // Writeback
|
|
|
|
let Inst{20} = L_bit;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{14} = regs{14};
|
|
|
|
let Inst{13} = 0;
|
|
|
|
let Inst{12-0} = regs{12-0};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-13 10:57:02 +00:00
|
|
|
|
|
|
|
let mayStore = 1, hasExtraSrcRegAllocReq = 1 in
|
2011-09-12 21:28:46 +00:00
|
|
|
defm t2STM : thumb2_st_mult<"stm", IIC_iStore_m, IIC_iStore_mu, 0>;
|
2010-11-13 10:57:02 +00:00
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
} // hasSideEffects
|
2010-11-13 10:57:02 +00:00
|
|
|
|
2009-07-03 00:18:36 +00:00
|
|
|
|
2009-06-17 18:13:58 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Move Instructions.
|
|
|
|
//
|
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
let hasSideEffects = 0 in
|
2017-02-21 16:41:29 +00:00
|
|
|
def t2MOVr : T2sTwoReg<(outs GPRnopc:$Rd), (ins GPRnopc:$Rm), IIC_iMOVr,
|
2013-06-06 16:35:25 +00:00
|
|
|
"mov", ".w\t$Rd, $Rm", []>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
[ARM] Turn some undefined encoding bits into 0s.
The family of 32-bit Thumb instruction encodings that include t2ORR,
t2AND and t2EOR are all listed in the ArmARM as having (0) in bit 15.
The Tablegen descriptions of those instructions listed them as ?. This
change tightens that up by making them into 0 + Unpredictable.
In the specific case of t2ORR, we tighten it up still further by
making the zero bit mandatory. This change comes from Arm v8.1-M, in
which encodings with that bit equal to 1 will now be used for
different instructions.
Reviewers: dmgreen, samparker, SjoerdMeijer, efriedma
Reviewed By: dmgreen, efriedma
Subscribers: efriedma, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60705
llvm-svn: 362470
2019-06-04 08:28:48 +00:00
|
|
|
let Inst{15} = 0b0;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{14-12} = 0b000;
|
|
|
|
let Inst{7-4} = 0b0000;
|
|
|
|
}
|
2017-02-21 16:41:29 +00:00
|
|
|
def : t2InstAlias<"mov${p}.w $Rd, $Rm", (t2MOVr GPRnopc:$Rd, GPRnopc:$Rm,
|
2011-10-18 17:09:35 +00:00
|
|
|
pred:$p, zero_reg)>;
|
2017-02-21 16:41:29 +00:00
|
|
|
def : t2InstAlias<"movs${p}.w $Rd, $Rm", (t2MOVr GPRnopc:$Rd, GPRnopc:$Rm,
|
2011-09-10 00:15:36 +00:00
|
|
|
pred:$p, CPSR)>;
|
2017-02-21 16:41:29 +00:00
|
|
|
def : t2InstAlias<"movs${p} $Rd, $Rm", (t2MOVr GPRnopc:$Rd, GPRnopc:$Rm,
|
2011-09-10 00:15:36 +00:00
|
|
|
pred:$p, CPSR)>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2009-09-28 09:14:39 +00:00
|
|
|
// AddedComplexity to ensure isel tries t2MOVi before t2MOVi16.
|
2010-11-17 20:13:28 +00:00
|
|
|
let isReMaterializable = 1, isAsCheapAsAMove = 1, isMoveImm = 1,
|
|
|
|
AddedComplexity = 1 in
|
2010-11-16 00:29:56 +00:00
|
|
|
def t2MOVi : T2sOneRegImm<(outs rGPR:$Rd), (ins t2_so_imm:$imm), IIC_iMOVi,
|
|
|
|
"mov", ".w\t$Rd, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, t2_so_imm:$imm)]>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2009-06-26 16:10:07 +00:00
|
|
|
|
2011-09-10 00:15:36 +00:00
|
|
|
// cc_out is handled as part of the explicit mnemonic in the parser for 'mov'.
|
|
|
|
// Use aliases to get that to play nice here.
|
|
|
|
def : t2InstAlias<"movs${p}.w $Rd, $imm", (t2MOVi rGPR:$Rd, t2_so_imm:$imm,
|
|
|
|
pred:$p, CPSR)>;
|
|
|
|
def : t2InstAlias<"movs${p} $Rd, $imm", (t2MOVi rGPR:$Rd, t2_so_imm:$imm,
|
|
|
|
pred:$p, CPSR)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"mov${p}.w $Rd, $imm", (t2MOVi rGPR:$Rd, t2_so_imm:$imm,
|
|
|
|
pred:$p, zero_reg)>;
|
|
|
|
def : t2InstAlias<"mov${p} $Rd, $imm", (t2MOVi rGPR:$Rd, t2_so_imm:$imm,
|
|
|
|
pred:$p, zero_reg)>;
|
2011-06-27 23:54:06 +00:00
|
|
|
|
2010-11-17 20:13:28 +00:00
|
|
|
let isReMaterializable = 1, isAsCheapAsAMove = 1, isMoveImm = 1 in
|
2011-07-19 19:13:28 +00:00
|
|
|
def t2MOVi16 : T2I<(outs rGPR:$Rd), (ins imm0_65535_expr:$imm), IIC_iMOVi,
|
2010-11-16 00:29:56 +00:00
|
|
|
"movw", "\t$Rd, $imm",
|
2016-01-15 10:25:14 +00:00
|
|
|
[(set rGPR:$Rd, imm0_65535:$imm)]>, Sched<[WriteALU]>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{20} = 0; // The S bit.
|
|
|
|
let Inst{15} = 0;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-16 00:29:56 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<16> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-16 00:29:56 +00:00
|
|
|
let Inst{19-16} = imm{15-12};
|
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
2011-10-04 22:44:48 +00:00
|
|
|
let DecoderMethod = "DecodeT2MOVTWInstruction";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2016-01-15 10:25:14 +00:00
|
|
|
def : InstAlias<"mov${p} $Rd, $imm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2MOVi16 rGPR:$Rd, imm256_65535_expr:$imm, pred:$p), 0>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteALU]>;
|
2013-08-21 13:14:58 +00:00
|
|
|
|
2011-01-21 18:55:51 +00:00
|
|
|
def t2MOVi16_ga_pcrel : PseudoInst<(outs rGPR:$Rd),
|
2017-02-22 07:22:57 +00:00
|
|
|
(ins i32imm:$addr, pclabel:$id), IIC_iMOVi, []>,
|
|
|
|
Sched<[WriteALU]>;
|
2011-01-17 08:03:18 +00:00
|
|
|
|
|
|
|
let Constraints = "$src = $Rd" in {
|
2011-01-13 07:58:56 +00:00
|
|
|
def t2MOVTi16 : T2I<(outs rGPR:$Rd),
|
2011-07-19 19:13:28 +00:00
|
|
|
(ins rGPR:$src, imm0_65535_expr:$imm), IIC_iMOVi,
|
2010-11-16 00:29:56 +00:00
|
|
|
"movt", "\t$Rd, $imm",
|
|
|
|
[(set rGPR:$Rd,
|
2013-06-06 16:35:25 +00:00
|
|
|
(or (and rGPR:$src, 0xffff), lo16AllZero:$imm))]>,
|
2016-01-15 10:25:14 +00:00
|
|
|
Sched<[WriteALU]>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-21} = 0b0110;
|
|
|
|
let Inst{20} = 0; // The S bit.
|
|
|
|
let Inst{15} = 0;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-16 00:29:56 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<16> imm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-16 00:29:56 +00:00
|
|
|
let Inst{19-16} = imm{15-12};
|
|
|
|
let Inst{26} = imm{11};
|
|
|
|
let Inst{14-12} = imm{10-8};
|
|
|
|
let Inst{7-0} = imm{7-0};
|
2011-10-04 22:44:48 +00:00
|
|
|
let DecoderMethod = "DecodeT2MOVTWInstruction";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-17 18:13:58 +00:00
|
|
|
|
2011-01-21 18:55:51 +00:00
|
|
|
def t2MOVTi16_ga_pcrel : PseudoInst<(outs rGPR:$Rd),
|
2013-06-06 16:35:25 +00:00
|
|
|
(ins rGPR:$src, i32imm:$addr, pclabel:$id), IIC_iMOVi, []>,
|
2016-01-15 10:25:14 +00:00
|
|
|
Sched<[WriteALU]>, Requires<[IsThumb, HasV8MBaseline]>;
|
2011-01-17 08:03:18 +00:00
|
|
|
} // Constraints
|
|
|
|
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(or rGPR:$src, 0xffff0000), (t2MOVTi16 rGPR:$src, 0xffff)>;
|
2009-10-21 08:15:52 +00:00
|
|
|
|
2009-07-03 01:43:10 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Extend Instructions.
|
|
|
|
//
|
|
|
|
|
|
|
|
// Sign extenders
|
|
|
|
|
2017-02-17 15:42:44 +00:00
|
|
|
def t2SXTB : T2I_ext_rrot<0b100, "sxtb">;
|
|
|
|
def t2SXTH : T2I_ext_rrot<0b000, "sxth">;
|
|
|
|
def t2SXTB16 : T2I_ext_rrot_xtb16<0b010, "sxtb16">;
|
|
|
|
|
|
|
|
def t2SXTAB : T2I_exta_rrot<0b100, "sxtab">;
|
|
|
|
def t2SXTAH : T2I_exta_rrot<0b000, "sxtah">;
|
|
|
|
def t2SXTAB16 : T2I_exta_rrot<0b010, "sxtab16">;
|
|
|
|
|
|
|
|
def : T2Pat<(sext_inreg (rotr rGPR:$Rn, rot_imm:$rot), i8),
|
|
|
|
(t2SXTB rGPR:$Rn, rot_imm:$rot)>;
|
|
|
|
def : T2Pat<(sext_inreg (rotr rGPR:$Rn, rot_imm:$rot), i16),
|
|
|
|
(t2SXTH rGPR:$Rn, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn,
|
|
|
|
(sext_inreg (rotr rGPR:$Rm, rot_imm:$rot), i8)),
|
|
|
|
(t2SXTAB rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn,
|
|
|
|
(sext_inreg (rotr rGPR:$Rm, rot_imm:$rot), i16)),
|
|
|
|
(t2SXTAH rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_sxtb16 rGPR:$Rn),
|
|
|
|
(t2SXTB16 rGPR:$Rn, 0)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_sxtab16 rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SXTAB16 rGPR:$Rn, rGPR:$Rm, 0)>;
|
2018-08-22 12:58:36 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_sxtb16 (rotr rGPR:$Rn, rot_imm:$rot)),
|
|
|
|
(t2SXTB16 rGPR:$Rn, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_sxtab16 rGPR:$Rn, (rotr rGPR:$Rm, rot_imm:$rot)),
|
|
|
|
(t2SXTAB16 rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2009-07-03 01:43:10 +00:00
|
|
|
|
|
|
|
|
2014-07-23 13:59:07 +00:00
|
|
|
// A simple right-shift can also be used in most cases (the exception is the
|
|
|
|
// SXTH operations with a rotate of 24: there the non-contiguous bits are
|
|
|
|
// relevant).
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (sext_inreg
|
2016-08-10 09:34:34 +00:00
|
|
|
(srl rGPR:$Rm, rot_imm:$rot), i8)),
|
|
|
|
(t2SXTAB rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (sext_inreg
|
2016-08-10 09:34:34 +00:00
|
|
|
(srl rGPR:$Rm, imm8_or_16:$rot), i16)),
|
|
|
|
(t2SXTAH rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (sext_inreg
|
2016-08-10 09:34:34 +00:00
|
|
|
(rotr rGPR:$Rm, (i32 24)), i16)),
|
|
|
|
(t2SXTAH rGPR:$Rn, rGPR:$Rm, (i32 3))>;
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (sext_inreg
|
2016-08-10 09:34:34 +00:00
|
|
|
(or (srl rGPR:$Rm, (i32 24)),
|
|
|
|
(shl rGPR:$Rm, (i32 8))), i16)),
|
|
|
|
(t2SXTAH rGPR:$Rn, rGPR:$Rm, (i32 3))>;
|
2014-07-23 13:59:07 +00:00
|
|
|
|
2009-07-03 01:43:10 +00:00
|
|
|
// Zero extenders
|
|
|
|
|
|
|
|
let AddedComplexity = 16 in {
|
2017-02-17 15:42:44 +00:00
|
|
|
def t2UXTB : T2I_ext_rrot<0b101, "uxtb">;
|
|
|
|
def t2UXTH : T2I_ext_rrot<0b001, "uxth">;
|
|
|
|
def t2UXTB16 : T2I_ext_rrot_xtb16<0b011, "uxtb16">;
|
|
|
|
|
|
|
|
def : Thumb2DSPPat<(and (rotr rGPR:$Rm, rot_imm:$rot), 0x000000FF),
|
|
|
|
(t2UXTB rGPR:$Rm, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(and (rotr rGPR:$Rm, rot_imm:$rot), 0x0000FFFF),
|
|
|
|
(t2UXTH rGPR:$Rm, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(and (rotr rGPR:$Rm, rot_imm:$rot), 0x00FF00FF),
|
|
|
|
(t2UXTB16 rGPR:$Rm, rot_imm:$rot)>;
|
2009-07-03 01:43:10 +00:00
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_uxtb16 rGPR:$Rm),
|
|
|
|
(t2UXTB16 rGPR:$Rm, 0)>;
|
2018-08-22 12:58:36 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_uxtb16 (rotr rGPR:$Rn, rot_imm:$rot)),
|
|
|
|
(t2UXTB16 rGPR:$Rn, rot_imm:$rot)>;
|
2017-05-04 07:31:28 +00:00
|
|
|
|
2010-07-28 23:17:45 +00:00
|
|
|
// FIXME: This pattern incorrectly assumes the shl operator is a rotate.
|
|
|
|
// The transformation should probably be done as a combiner action
|
|
|
|
// instead so we can include a check for masking back in the upper
|
|
|
|
// eight bits of the source into the lower eight bits of the result.
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
//def : T2Pat<(and (shl rGPR:$Src, (i32 8)), 0xFF00FF),
|
2011-07-27 17:48:13 +00:00
|
|
|
// (t2UXTB16 rGPR:$Src, 3)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
// Requires<[HasDSP, IsThumb2]>;
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(and (srl rGPR:$Src, (i32 8)), 0xFF00FF),
|
2011-07-27 17:48:13 +00:00
|
|
|
(t2UXTB16 rGPR:$Src, 1)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2009-07-03 01:43:10 +00:00
|
|
|
|
2017-02-17 15:42:44 +00:00
|
|
|
def t2UXTAB : T2I_exta_rrot<0b101, "uxtab">;
|
|
|
|
def t2UXTAH : T2I_exta_rrot<0b001, "uxtah">;
|
|
|
|
def t2UXTAB16 : T2I_exta_rrot<0b011, "uxtab16">;
|
2014-07-23 13:59:07 +00:00
|
|
|
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (and (rotr rGPR:$Rm, rot_imm:$rot),
|
|
|
|
0x00FF)),
|
|
|
|
(t2UXTAB rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (and (rotr rGPR:$Rm, rot_imm:$rot),
|
|
|
|
0xFFFF)),
|
|
|
|
(t2UXTAH rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (and (srl rGPR:$Rm, rot_imm:$rot),
|
2016-08-10 09:34:34 +00:00
|
|
|
0xFF)),
|
|
|
|
(t2UXTAB rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2017-02-17 15:42:44 +00:00
|
|
|
def : Thumb2DSPPat<(add rGPR:$Rn, (and (srl rGPR:$Rm, imm8_or_16:$rot),
|
2016-08-10 09:34:34 +00:00
|
|
|
0xFFFF)),
|
|
|
|
(t2UXTAH rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_uxtab16 rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2UXTAB16 rGPR:$Rn, rGPR:$Rm, 0)>;
|
2018-08-22 12:58:36 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_uxtab16 rGPR:$Rn, (rotr rGPR:$Rm, rot_imm:$rot)),
|
|
|
|
(t2UXTAB16 rGPR:$Rn, rGPR:$Rm, rot_imm:$rot)>;
|
2009-07-03 01:43:10 +00:00
|
|
|
}
|
|
|
|
|
2014-07-23 13:59:07 +00:00
|
|
|
|
2009-06-17 18:13:58 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Arithmetic Instructions.
|
|
|
|
//
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2016-09-14 08:20:03 +00:00
|
|
|
let isAdd = 1 in
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2ADD : T2I_bin_ii12rs<0b000, "add", add, 1>;
|
|
|
|
defm t2SUB : T2I_bin_ii12rs<0b101, "sub", sub>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
|
|
|
// ADD and SUB with 's' bit set. No 12-bit immediate (T4) variants.
|
2011-09-21 02:20:46 +00:00
|
|
|
//
|
|
|
|
// Currently, t2ADDS/t2SUBS are pseudo opcodes that exist only in the
|
|
|
|
// selection DAG. They are "lowered" to real t2ADD/t2SUB opcodes by
|
|
|
|
// AdjustInstrPostInstrSelection where we determine whether or not to
|
|
|
|
// set the "s" bit based on CPSR liveness.
|
|
|
|
//
|
|
|
|
// FIXME: Eliminate t2ADDS/t2SUBS pseudo opcodes after adding tablegen
|
|
|
|
// support for an optional CPSR definition that corresponds to the DAG
|
|
|
|
// node's second value. We can then eliminate the implicit def of CPSR.
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2ADDS : T2I_bin_s_irs <IIC_iALUi, IIC_iALUr, IIC_iALUsi, ARMaddc, 1>;
|
|
|
|
defm t2SUBS : T2I_bin_s_irs <IIC_iALUi, IIC_iALUr, IIC_iALUsi, ARMsubc>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2018-12-03 11:16:21 +00:00
|
|
|
def : T2Pat<(ARMsubs GPRnopc:$Rn, t2_so_imm:$imm),
|
|
|
|
(t2SUBSri $Rn, t2_so_imm:$imm)>;
|
|
|
|
def : T2Pat<(ARMsubs GPRnopc:$Rn, rGPR:$Rm), (t2SUBSrr $Rn, $Rm)>;
|
|
|
|
def : T2Pat<(ARMsubs GPRnopc:$Rn, t2_so_reg:$ShiftedRm),
|
|
|
|
(t2SUBSrs $Rn, t2_so_reg:$ShiftedRm)>;
|
|
|
|
|
2011-09-20 18:22:31 +00:00
|
|
|
let hasPostISelHook = 1 in {
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2ADC : T2I_adde_sube_irs<0b1010, "adc", ARMadde, 1>;
|
|
|
|
defm t2SBC : T2I_adde_sube_irs<0b1011, "sbc", ARMsube>;
|
2011-09-20 18:22:31 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"adc${s}${p} $rd, $rn, $imm",
|
|
|
|
(t2SBCri rGPR:$rd, rGPR:$rn, t2_so_imm_not:$imm, pred:$p, s_cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"sbc${s}${p} $rd, $rn, $imm",
|
|
|
|
(t2ADCri rGPR:$rd, rGPR:$rn, t2_so_imm_not:$imm, pred:$p, s_cc_out:$s)>;
|
|
|
|
|
|
|
|
def : t2InstSubst<"add${s}${p}.w $rd, $rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$rd, GPRnopc:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"sub${s}${p}.w $rd, $rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri rGPR:$rd, GPRnopc:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
2018-07-04 16:11:15 +00:00
|
|
|
def : t2InstSubst<"subw${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri12 rGPR:$Rd, GPR:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
2018-07-04 16:11:15 +00:00
|
|
|
def : t2InstSubst<"sub${s}${p} $rd, $rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri rGPR:$rd, GPRnopc:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
2018-07-04 16:11:15 +00:00
|
|
|
def : t2InstSubst<"sub${p} $rd, $rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri12 rGPR:$rd, GPR:$rn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
// SP to SP alike
|
|
|
|
def : t2InstSubst<"add${s}${p}.w $rd, $rn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$rd, GPRsp:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"sub${s}${p}.w $rd, $rn, $imm",
|
|
|
|
(t2ADDspImm GPRsp:$rd, GPRsp:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"subw${p} $Rd, $Rn, $imm",
|
|
|
|
(t2ADDspImm12 GPRsp:$Rd, GPRsp:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
def : t2InstSubst<"sub${s}${p} $rd, $rn, $imm",
|
|
|
|
(t2ADDspImm GPRsp:$rd, GPRsp:$rn, t2_so_imm_neg:$imm, pred:$p, s_cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"sub${p} $rd, $rn, $imm",
|
|
|
|
(t2ADDspImm12 GPRsp:$rd, GPRsp:$rn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
|
2009-07-27 16:39:05 +00:00
|
|
|
// RSB
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2RSB : T2I_rbin_irs <0b1110, "rsb", sub>;
|
2011-09-06 18:52:20 +00:00
|
|
|
|
|
|
|
// FIXME: Eliminate them if we can write def : Pat patterns which defines
|
|
|
|
// CPSR and the implicit def of CPSR is not needed.
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2RSBS : T2I_rbin_s_is <ARMsubc>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
|
|
|
// (sub X, imm) gets canonicalized to (add X, -imm). Match this form.
|
2010-07-14 17:45:16 +00:00
|
|
|
// The assume-no-carry-in form uses the negation of the input since add/sub
|
|
|
|
// assume opposite meanings of the carry flag (i.e., carry == !borrow).
|
|
|
|
// See the definition of AddWithCarry() in the ARM ARM A2.2.1 for the gory
|
|
|
|
// details.
|
|
|
|
// The AddedComplexity preferences the first variant over the others since
|
|
|
|
// it can be shrunk to a 16-bit wide encoding, while the others cannot.
|
|
|
|
let AddedComplexity = 1 in
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
def : T2Pat<(add rGPR:$src, imm1_255_neg:$imm),
|
|
|
|
(t2SUBri rGPR:$src, imm1_255_neg:$imm)>;
|
|
|
|
def : T2Pat<(add rGPR:$src, t2_so_imm_neg:$imm),
|
|
|
|
(t2SUBri rGPR:$src, t2_so_imm_neg:$imm)>;
|
|
|
|
def : T2Pat<(add rGPR:$src, imm0_4095_neg:$imm),
|
|
|
|
(t2SUBri12 rGPR:$src, imm0_4095_neg:$imm)>;
|
2012-06-23 00:29:06 +00:00
|
|
|
def : T2Pat<(add GPR:$src, imm0_65535_neg:$imm),
|
|
|
|
(t2SUBrr GPR:$src, (t2MOVi16 (imm_neg_XFORM imm:$imm)))>;
|
|
|
|
|
2019-02-11 11:35:42 +00:00
|
|
|
// Do the same for v8m targets since they support movw with a 16-bit value.
|
|
|
|
def : T1Pat<(add tGPR:$src, imm0_65535_neg:$imm),
|
|
|
|
(tSUBrr tGPR:$src, (t2MOVi16 (imm_neg_XFORM imm:$imm)))>,
|
|
|
|
Requires<[HasV8MBaseline]>;
|
|
|
|
|
2010-07-14 17:45:16 +00:00
|
|
|
let AddedComplexity = 1 in
|
2012-11-14 19:39:15 +00:00
|
|
|
def : T2Pat<(ARMaddc rGPR:$src, imm1_255_neg:$imm),
|
|
|
|
(t2SUBSri rGPR:$src, imm1_255_neg:$imm)>;
|
2011-08-30 01:34:54 +00:00
|
|
|
def : T2Pat<(ARMaddc rGPR:$src, t2_so_imm_neg:$imm),
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
(t2SUBSri rGPR:$src, t2_so_imm_neg:$imm)>;
|
2012-06-23 00:29:06 +00:00
|
|
|
def : T2Pat<(ARMaddc rGPR:$src, imm0_65535_neg:$imm),
|
|
|
|
(t2SUBSrr rGPR:$src, (t2MOVi16 (imm_neg_XFORM imm:$imm)))>;
|
2010-07-14 17:45:16 +00:00
|
|
|
// The with-carry-in form matches bitwise not instead of the negation.
|
|
|
|
// Effectively, the inverse interpretation of the carry flag already accounts
|
|
|
|
// for part of the negation.
|
2009-08-04 01:41:15 +00:00
|
|
|
let AddedComplexity = 1 in
|
2011-08-30 01:34:54 +00:00
|
|
|
def : T2Pat<(ARMadde rGPR:$src, imm0_255_not:$imm, CPSR),
|
2011-04-23 03:55:32 +00:00
|
|
|
(t2SBCri rGPR:$src, imm0_255_not:$imm)>;
|
2011-08-30 01:34:54 +00:00
|
|
|
def : T2Pat<(ARMadde rGPR:$src, t2_so_imm_not:$imm, CPSR),
|
2011-04-23 03:55:32 +00:00
|
|
|
(t2SBCri rGPR:$src, t2_so_imm_not:$imm)>;
|
2012-06-23 00:29:06 +00:00
|
|
|
def : T2Pat<(ARMadde rGPR:$src, imm0_65535_neg:$imm, CPSR),
|
2012-10-24 19:53:01 +00:00
|
|
|
(t2SBCrr rGPR:$src, (t2MOVi16 (imm_not_XFORM imm:$imm)))>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-11-30 20:00:01 +00:00
|
|
|
def t2SEL : T2ThreeReg<(outs GPR:$Rd), (ins GPR:$Rn, GPR:$Rm),
|
2017-05-04 07:31:28 +00:00
|
|
|
NoItinerary, "sel", "\t$Rd, $Rn, $Rm",
|
|
|
|
[(set GPR:$Rd, (int_arm_sel GPR:$Rn, GPR:$Rm))]>,
|
2015-09-24 17:31:16 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]> {
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-24} = 0b010;
|
|
|
|
let Inst{23} = 0b1;
|
|
|
|
let Inst{22-20} = 0b010;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7} = 0b1;
|
|
|
|
let Inst{6-4} = 0b000;
|
|
|
|
}
|
|
|
|
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
// A6.3.13, A6.3.14, A6.3.15 Parallel addition and subtraction (signed/unsigned)
|
|
|
|
// And Miscellaneous operations -- for disassembly only
|
2010-07-29 17:56:55 +00:00
|
|
|
class T2I_pam<bits<3> op22_20, bits<4> op7_4, string opc,
|
2017-05-04 07:31:28 +00:00
|
|
|
list<dag> pat, dag iops, string asm>
|
2011-07-01 21:12:19 +00:00
|
|
|
: T2I<(outs rGPR:$Rd), iops, NoItinerary, opc, asm, pat>,
|
2015-09-24 17:31:16 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]> {
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0101;
|
|
|
|
let Inst{22-20} = op22_20;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-4} = op7_4;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-17 19:57:38 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{3-0} = Rm;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
}
|
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
class T2I_pam_intrinsics<bits<3> op22_20, bits<4> op7_4, string opc,
|
|
|
|
Intrinsic intrinsic>
|
|
|
|
: T2I_pam<op22_20, op7_4, opc,
|
|
|
|
[(set rGPR:$Rd, (intrinsic rGPR:$Rn, rGPR:$Rm))],
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm), "\t$Rd, $Rn, $Rm">;
|
|
|
|
|
|
|
|
class T2I_pam_intrinsics_rev<bits<3> op22_20, bits<4> op7_4, string opc>
|
|
|
|
: T2I_pam<op22_20, op7_4, opc, [],
|
|
|
|
(ins rGPR:$Rm, rGPR:$Rn), "\t$Rd, $Rm, $Rn">;
|
|
|
|
|
|
|
|
// Saturating add/subtract
|
|
|
|
def t2QADD16 : T2I_pam_intrinsics<0b001, 0b0001, "qadd16", int_arm_qadd16>;
|
|
|
|
def t2QADD8 : T2I_pam_intrinsics<0b000, 0b0001, "qadd8", int_arm_qadd8>;
|
|
|
|
def t2QASX : T2I_pam_intrinsics<0b010, 0b0001, "qasx", int_arm_qasx>;
|
|
|
|
def t2UQSUB8 : T2I_pam_intrinsics<0b100, 0b0101, "uqsub8", int_arm_uqsub8>;
|
|
|
|
def t2QSAX : T2I_pam_intrinsics<0b110, 0b0001, "qsax", int_arm_qsax>;
|
|
|
|
def t2QSUB16 : T2I_pam_intrinsics<0b101, 0b0001, "qsub16", int_arm_qsub16>;
|
|
|
|
def t2QSUB8 : T2I_pam_intrinsics<0b100, 0b0001, "qsub8", int_arm_qsub8>;
|
|
|
|
def t2UQADD16 : T2I_pam_intrinsics<0b001, 0b0101, "uqadd16", int_arm_uqadd16>;
|
|
|
|
def t2UQADD8 : T2I_pam_intrinsics<0b000, 0b0101, "uqadd8", int_arm_uqadd8>;
|
|
|
|
def t2UQASX : T2I_pam_intrinsics<0b010, 0b0101, "uqasx", int_arm_uqasx>;
|
|
|
|
def t2UQSAX : T2I_pam_intrinsics<0b110, 0b0101, "uqsax", int_arm_uqsax>;
|
|
|
|
def t2UQSUB16 : T2I_pam_intrinsics<0b101, 0b0101, "uqsub16", int_arm_uqsub16>;
|
|
|
|
def t2QADD : T2I_pam_intrinsics_rev<0b000, 0b1000, "qadd">;
|
|
|
|
def t2QSUB : T2I_pam_intrinsics_rev<0b000, 0b1010, "qsub">;
|
|
|
|
def t2QDADD : T2I_pam_intrinsics_rev<0b000, 0b1001, "qdadd">;
|
|
|
|
def t2QDSUB : T2I_pam_intrinsics_rev<0b000, 0b1011, "qdsub">;
|
|
|
|
|
|
|
|
def : Thumb2DSPPat<(int_arm_qadd rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QADD rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_qsub rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QSUB rGPR:$Rm, rGPR:$Rn)>;
|
2020-03-30 18:37:21 +01:00
|
|
|
def : Thumb2DSPPat<(int_arm_qadd rGPR:$Rm, (int_arm_qadd rGPR:$Rn, rGPR:$Rn)),
|
2017-05-04 07:31:28 +00:00
|
|
|
(t2QDADD rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_qsub rGPR:$Rm, (int_arm_qadd rGPR:$Rn, rGPR:$Rn)),
|
|
|
|
(t2QDSUB rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
|
2019-10-21 12:33:46 +00:00
|
|
|
def : Thumb2DSPPat<(saddsat rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QADD rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(ssubsat rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QSUB rGPR:$Rm, rGPR:$Rn)>;
|
2020-03-30 18:37:21 +01:00
|
|
|
def : Thumb2DSPPat<(saddsat rGPR:$Rm, (saddsat rGPR:$Rn, rGPR:$Rn)),
|
2019-10-21 14:06:49 +00:00
|
|
|
(t2QDADD rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(ssubsat rGPR:$Rm, (saddsat rGPR:$Rn, rGPR:$Rn)),
|
|
|
|
(t2QDSUB rGPR:$Rm, rGPR:$Rn)>;
|
2019-10-21 09:53:38 +00:00
|
|
|
def : Thumb2DSPPat<(ARMqadd8b rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QADD8 rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(ARMqsub8b rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QSUB8 rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(ARMqadd16b rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QADD16 rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
def : Thumb2DSPPat<(ARMqsub16b rGPR:$Rm, rGPR:$Rn),
|
|
|
|
(t2QSUB16 rGPR:$Rm, rGPR:$Rn)>;
|
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
// Signed/Unsigned add/subtract
|
|
|
|
|
|
|
|
def t2SASX : T2I_pam_intrinsics<0b010, 0b0000, "sasx", int_arm_sasx>;
|
|
|
|
def t2SADD16 : T2I_pam_intrinsics<0b001, 0b0000, "sadd16", int_arm_sadd16>;
|
|
|
|
def t2SADD8 : T2I_pam_intrinsics<0b000, 0b0000, "sadd8", int_arm_sadd8>;
|
|
|
|
def t2SSAX : T2I_pam_intrinsics<0b110, 0b0000, "ssax", int_arm_ssax>;
|
|
|
|
def t2SSUB16 : T2I_pam_intrinsics<0b101, 0b0000, "ssub16", int_arm_ssub16>;
|
|
|
|
def t2SSUB8 : T2I_pam_intrinsics<0b100, 0b0000, "ssub8", int_arm_ssub8>;
|
|
|
|
def t2UASX : T2I_pam_intrinsics<0b010, 0b0100, "uasx", int_arm_uasx>;
|
|
|
|
def t2UADD16 : T2I_pam_intrinsics<0b001, 0b0100, "uadd16", int_arm_uadd16>;
|
|
|
|
def t2UADD8 : T2I_pam_intrinsics<0b000, 0b0100, "uadd8", int_arm_uadd8>;
|
|
|
|
def t2USAX : T2I_pam_intrinsics<0b110, 0b0100, "usax", int_arm_usax>;
|
|
|
|
def t2USUB16 : T2I_pam_intrinsics<0b101, 0b0100, "usub16", int_arm_usub16>;
|
|
|
|
def t2USUB8 : T2I_pam_intrinsics<0b100, 0b0100, "usub8", int_arm_usub8>;
|
|
|
|
|
|
|
|
// Signed/Unsigned halving add/subtract
|
|
|
|
|
|
|
|
def t2SHASX : T2I_pam_intrinsics<0b010, 0b0010, "shasx", int_arm_shasx>;
|
|
|
|
def t2SHADD16 : T2I_pam_intrinsics<0b001, 0b0010, "shadd16", int_arm_shadd16>;
|
|
|
|
def t2SHADD8 : T2I_pam_intrinsics<0b000, 0b0010, "shadd8", int_arm_shadd8>;
|
|
|
|
def t2SHSAX : T2I_pam_intrinsics<0b110, 0b0010, "shsax", int_arm_shsax>;
|
|
|
|
def t2SHSUB16 : T2I_pam_intrinsics<0b101, 0b0010, "shsub16", int_arm_shsub16>;
|
|
|
|
def t2SHSUB8 : T2I_pam_intrinsics<0b100, 0b0010, "shsub8", int_arm_shsub8>;
|
|
|
|
def t2UHASX : T2I_pam_intrinsics<0b010, 0b0110, "uhasx", int_arm_uhasx>;
|
|
|
|
def t2UHADD16 : T2I_pam_intrinsics<0b001, 0b0110, "uhadd16", int_arm_uhadd16>;
|
|
|
|
def t2UHADD8 : T2I_pam_intrinsics<0b000, 0b0110, "uhadd8", int_arm_uhadd8>;
|
|
|
|
def t2UHSAX : T2I_pam_intrinsics<0b110, 0b0110, "uhsax", int_arm_uhsax>;
|
|
|
|
def t2UHSUB16 : T2I_pam_intrinsics<0b101, 0b0110, "uhsub16", int_arm_uhsub16>;
|
|
|
|
def t2UHSUB8 : T2I_pam_intrinsics<0b100, 0b0110, "uhsub8", int_arm_uhsub8>;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
|
2010-11-18 20:32:18 +00:00
|
|
|
// Helper class for disassembly only
|
|
|
|
// A6.3.16 & A6.3.17
|
|
|
|
// T2Imac - Thumb2 multiply [accumulate, and absolute difference] instructions.
|
|
|
|
class T2ThreeReg_mac<bit long, bits<3> op22_20, bits<4> op7_4, dag oops,
|
|
|
|
dag iops, InstrItinClass itin, string opc, string asm, list<dag> pattern>
|
|
|
|
: T2ThreeReg<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-24} = 0b011;
|
|
|
|
let Inst{23} = long;
|
|
|
|
let Inst{22-20} = op22_20;
|
|
|
|
let Inst{7-4} = op7_4;
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2FourReg_mac<bit long, bits<3> op22_20, bits<4> op7_4, dag oops,
|
|
|
|
dag iops, InstrItinClass itin, string opc, string asm, list<dag> pattern>
|
|
|
|
: T2FourReg<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-24} = 0b011;
|
|
|
|
let Inst{23} = long;
|
|
|
|
let Inst{22-20} = op22_20;
|
|
|
|
let Inst{7-4} = op7_4;
|
|
|
|
}
|
|
|
|
|
2011-09-20 00:26:34 +00:00
|
|
|
// Unsigned Sum of Absolute Differences [and Accumulate].
|
2010-11-18 20:32:18 +00:00
|
|
|
def t2USAD8 : T2ThreeReg_mac<0, 0b111, 0b0000, (outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm),
|
2017-05-04 07:31:28 +00:00
|
|
|
NoItinerary, "usad8", "\t$Rd, $Rn, $Rm",
|
|
|
|
[(set rGPR:$Rd, (int_arm_usad8 rGPR:$Rn, rGPR:$Rm))]>,
|
2015-09-24 17:31:16 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]> {
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
}
|
2010-11-18 20:32:18 +00:00
|
|
|
def t2USADA8 : T2FourReg_mac<0, 0b111, 0b0000, (outs rGPR:$Rd),
|
2010-11-19 17:11:02 +00:00
|
|
|
(ins rGPR:$Rn, rGPR:$Rm, rGPR:$Ra), NoItinerary,
|
2017-05-04 07:31:28 +00:00
|
|
|
"usada8", "\t$Rd, $Rn, $Rm, $Ra",
|
|
|
|
[(set rGPR:$Rd, (int_arm_usada8 rGPR:$Rn, rGPR:$Rm, rGPR:$Ra))]>,
|
2015-09-24 17:31:16 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
|
2011-09-20 00:26:34 +00:00
|
|
|
// Signed/Unsigned saturate.
|
2017-05-04 07:31:28 +00:00
|
|
|
let hasSideEffects = 1 in
|
2017-04-11 14:42:08 +00:00
|
|
|
class T2SatI<dag iops, string opc, string asm>
|
|
|
|
: T2I<(outs rGPR:$Rd), iops, NoItinerary, opc, asm, []> {
|
2010-11-17 19:57:38 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<5> sat_imm;
|
2017-04-11 14:42:08 +00:00
|
|
|
bits<6> sh;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{31-24} = 0b11110011;
|
|
|
|
let Inst{21} = sh{5};
|
|
|
|
let Inst{20} = 0;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{15} = 0;
|
2010-11-17 19:57:38 +00:00
|
|
|
let Inst{14-12} = sh{4-2};
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{7-6} = sh{1-0};
|
|
|
|
let Inst{5} = 0;
|
|
|
|
let Inst{4-0} = sat_imm;
|
2010-11-17 19:57:38 +00:00
|
|
|
}
|
|
|
|
|
2017-04-11 14:42:08 +00:00
|
|
|
def t2SSAT: T2SatI<(ins imm1_32:$sat_imm, rGPR:$Rn, t2_shift_imm:$sh),
|
|
|
|
"ssat", "\t$Rd, $sat_imm, $Rn$sh">,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2]>, Sched<[WriteALU]> {
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{23-22} = 0b00;
|
2011-09-19 20:00:02 +00:00
|
|
|
let Inst{5} = 0;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
}
|
|
|
|
|
2017-04-11 14:42:08 +00:00
|
|
|
def t2SSAT16: T2SatI<(ins imm1_16:$sat_imm, rGPR:$Rn),
|
|
|
|
"ssat16", "\t$Rd, $sat_imm, $Rn">,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>, Sched<[WriteALU]> {
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{23-22} = 0b00;
|
|
|
|
let sh = 0b100000;
|
|
|
|
let Inst{4} = 0;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
}
|
|
|
|
|
2017-04-11 14:42:08 +00:00
|
|
|
def t2USAT: T2SatI<(ins imm0_31:$sat_imm, rGPR:$Rn, t2_shift_imm:$sh),
|
|
|
|
"usat", "\t$Rd, $sat_imm, $Rn$sh">,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2]>, Sched<[WriteALU]> {
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{23-22} = 0b10;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
}
|
|
|
|
|
2017-04-11 14:42:08 +00:00
|
|
|
def t2USAT16: T2SatI<(ins imm0_15:$sat_imm, rGPR:$Rn),
|
|
|
|
"usat16", "\t$Rd, $sat_imm, $Rn">,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>, Sched<[WriteALU]> {
|
2017-04-11 14:42:08 +00:00
|
|
|
let Inst{23-22} = 0b10;
|
|
|
|
let sh = 0b100000;
|
|
|
|
let Inst{4} = 0;
|
Added the follwoing 32-bit Thumb instructions for disassembly only:
o Parallel addition and subtraction, signed/unsigned
o Miscellaneous operations: QADD, QDADD, QSUB, QDSUB
o Unsigned sum of absolute differences [and accumulate]: USAD8, USADA8
o Signed/Unsigned saturate: SSAT, SSAT16, USAT, USAT16
o Signed multiply accumulate long (halfwords): SMLAL<x><y>
o Signed multiply accumulate/subtract [long] (dual): SMLAD[x], SMLALD[X], SMLSD[X], SMLSLD[X]
o Signed dual multiply add/subtract [long]: SMUAD[X], SMUSD[X]
llvm-svn: 97276
2010-02-26 22:04:29 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2016-06-23 16:53:49 +00:00
|
|
|
def : T2Pat<(ARMssatnoshift GPRnopc:$Rn, imm0_31:$imm),
|
|
|
|
(t2SSAT imm0_31:$imm, GPRnopc:$Rn, 0)>;
|
[ARM] Lower unsigned saturation to USAT
Summary:
Implement lower of unsigned saturation on an interval [0, k] where k + 1 is a power of two using USAT instruction in a similar way to how [~k, k] is lowered using SSAT on ARM models that supports it.
Patch by Marten Svanfeldt
Reviewers: t.p.northover, pbarrio, eastig, SjoerdMeijer, javed.absar, fhahn
Reviewed By: fhahn
Subscribers: fhahn, aemerson, javed.absar, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D41348
llvm-svn: 321164
2017-12-20 11:13:57 +00:00
|
|
|
def : T2Pat<(ARMusatnoshift GPRnopc:$Rn, imm0_31:$imm),
|
|
|
|
(t2USAT imm0_31:$imm, GPRnopc:$Rn, 0)>;
|
2017-05-04 07:31:28 +00:00
|
|
|
def : T2Pat<(int_arm_ssat GPR:$a, imm1_32:$pos),
|
|
|
|
(t2SSAT imm1_32:$pos, GPR:$a, 0)>;
|
|
|
|
def : T2Pat<(int_arm_usat GPR:$a, imm0_31:$pos),
|
|
|
|
(t2USAT imm0_31:$pos, GPR:$a, 0)>;
|
|
|
|
def : T2Pat<(int_arm_ssat16 GPR:$a, imm1_16:$pos),
|
|
|
|
(t2SSAT16 imm1_16:$pos, GPR:$a)>;
|
|
|
|
def : T2Pat<(int_arm_usat16 GPR:$a, imm0_15:$pos),
|
|
|
|
(t2USAT16 imm0_15:$pos, GPR:$a)>;
|
2010-07-29 22:48:09 +00:00
|
|
|
|
2009-06-23 19:39:13 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Shift and rotate Instructions.
|
|
|
|
//
|
|
|
|
|
2017-02-27 14:40:51 +00:00
|
|
|
defm t2LSL : T2I_sh_ir<0b00, "lsl", imm1_31, shl>;
|
2016-03-08 16:23:54 +00:00
|
|
|
defm t2LSR : T2I_sh_ir<0b01, "lsr", imm_sr, srl>;
|
|
|
|
defm t2ASR : T2I_sh_ir<0b10, "asr", imm_sr, sra>;
|
|
|
|
defm t2ROR : T2I_sh_ir<0b11, "ror", imm0_31, rotr>;
|
2009-06-23 19:39:13 +00:00
|
|
|
|
2017-02-27 14:40:51 +00:00
|
|
|
// LSL #0 is actually MOV, and has slightly different permitted registers to
|
|
|
|
// LSL with non-zero shift
|
|
|
|
def : t2InstAlias<"lsl${s}${p} $Rd, $Rm, #0",
|
|
|
|
(t2MOVr GPRnopc:$Rd, GPRnopc:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"lsl${s}${p}.w $Rd, $Rm, #0",
|
|
|
|
(t2MOVr GPRnopc:$Rd, GPRnopc:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
|
2011-04-29 14:18:15 +00:00
|
|
|
// (rotr x, (and y, 0x...1f)) ==> (ROR x, y)
|
2012-07-02 17:22:47 +00:00
|
|
|
def : T2Pat<(rotr rGPR:$lhs, (and rGPR:$rhs, lo5AllOne)),
|
|
|
|
(t2RORrr rGPR:$lhs, rGPR:$rhs)>;
|
2011-04-29 14:18:15 +00:00
|
|
|
|
2009-09-01 18:32:09 +00:00
|
|
|
let Uses = [CPSR] in {
|
2010-11-17 19:57:38 +00:00
|
|
|
def t2RRX : T2sTwoReg<(outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iMOVsi,
|
|
|
|
"rrx", "\t$Rd, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (ARMrrx rGPR:$Rm))]>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
[ARM] Turn some undefined encoding bits into 0s.
The family of 32-bit Thumb instruction encodings that include t2ORR,
t2AND and t2EOR are all listed in the ArmARM as having (0) in bit 15.
The Tablegen descriptions of those instructions listed them as ?. This
change tightens that up by making them into 0 + Unpredictable.
In the specific case of t2ORR, we tighten it up still further by
making the zero bit mandatory. This change comes from Arm v8.1-M, in
which encodings with that bit equal to 1 will now be used for
different instructions.
Reviewers: dmgreen, samparker, SjoerdMeijer, efriedma
Reviewed By: dmgreen, efriedma
Subscribers: efriedma, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D60705
llvm-svn: 362470
2019-06-04 08:28:48 +00:00
|
|
|
let Inst{15} = 0b0;
|
|
|
|
let Unpredictable{15} = 0b1;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{14-12} = 0b000;
|
|
|
|
let Inst{7-4} = 0b0011;
|
|
|
|
}
|
2009-09-01 18:32:09 +00:00
|
|
|
}
|
2009-06-23 19:39:13 +00:00
|
|
|
|
2011-01-10 15:26:39 +00:00
|
|
|
let isCodeGenOnly = 1, Defs = [CPSR] in {
|
2010-11-15 19:58:36 +00:00
|
|
|
def t2MOVsrl_flag : T2TwoRegShiftImm<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iMOVsi,
|
|
|
|
"lsrs", ".w\t$Rd, $Rm, #1",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (ARMsrl_flag rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{5-4} = 0b01; // Shift type.
|
|
|
|
// Shift amount = Inst{14-12:7-6} = 1.
|
|
|
|
let Inst{14-12} = 0b000;
|
|
|
|
let Inst{7-6} = 0b01;
|
|
|
|
}
|
2010-11-15 19:58:36 +00:00
|
|
|
def t2MOVsra_flag : T2TwoRegShiftImm<
|
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iMOVsi,
|
|
|
|
"asrs", ".w\t$Rd, $Rm, #1",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (ARMsra_flag rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b0010;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{5-4} = 0b10; // Shift type.
|
|
|
|
// Shift amount = Inst{14-12:7-6} = 1.
|
|
|
|
let Inst{14-12} = 0b000;
|
|
|
|
let Inst{7-6} = 0b01;
|
|
|
|
}
|
2009-07-28 17:06:49 +00:00
|
|
|
}
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Bitwise Instructions.
|
|
|
|
//
|
|
|
|
|
2009-12-15 17:24:14 +00:00
|
|
|
defm t2AND : T2I_bin_w_irs<0b0000, "and",
|
2016-03-08 16:23:54 +00:00
|
|
|
IIC_iBITi, IIC_iBITr, IIC_iBITsi, and, 1>;
|
2009-12-15 17:24:14 +00:00
|
|
|
defm t2ORR : T2I_bin_w_irs<0b0010, "orr",
|
2016-03-08 16:23:54 +00:00
|
|
|
IIC_iBITi, IIC_iBITr, IIC_iBITsi, or, 1>;
|
2009-12-15 17:24:14 +00:00
|
|
|
defm t2EOR : T2I_bin_w_irs<0b0100, "eor",
|
2016-03-08 16:23:54 +00:00
|
|
|
IIC_iBITi, IIC_iBITr, IIC_iBITsi, xor, 1>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2009-12-15 17:24:14 +00:00
|
|
|
defm t2BIC : T2I_bin_w_irs<0b0001, "bic",
|
2010-09-29 00:27:46 +00:00
|
|
|
IIC_iBITi, IIC_iBITr, IIC_iBITsi,
|
2012-08-02 21:50:41 +00:00
|
|
|
BinOpFrag<(and node:$LHS, (not node:$RHS))>>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-11-17 22:16:31 +00:00
|
|
|
class T2BitFI<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2010-11-19 17:11:02 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern> {
|
2010-11-17 22:16:31 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<5> msb;
|
|
|
|
bits<5> lsb;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2010-11-17 22:16:31 +00:00
|
|
|
let Inst{4-0} = msb{4-0};
|
|
|
|
let Inst{14-12} = lsb{4-2};
|
|
|
|
let Inst{7-6} = lsb{1-0};
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2TwoRegBitFI<dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
|
|
|
: T2BitFI<oops, iops, itin, opc, asm, pattern> {
|
|
|
|
bits<4> Rn;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2010-11-17 22:16:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let Constraints = "$src = $Rd" in
|
|
|
|
def t2BFC : T2BitFI<(outs rGPR:$Rd), (ins rGPR:$src, bf_inv_mask_imm:$imm),
|
|
|
|
IIC_iUNAsi, "bfc", "\t$Rd, $imm",
|
2019-05-15 12:41:58 +00:00
|
|
|
[(set rGPR:$Rd, (and rGPR:$src, bf_inv_mask_imm:$imm))]>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
2011-04-15 22:52:15 +00:00
|
|
|
let Inst{26} = 0; // should be 0.
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-20} = 0b10110;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{15} = 0;
|
2011-04-15 22:52:15 +00:00
|
|
|
let Inst{5} = 0; // should be 0.
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-17 22:16:31 +00:00
|
|
|
bits<10> imm;
|
|
|
|
let msb{4-0} = imm{9-5};
|
|
|
|
let lsb{4-0} = imm{4-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-11-17 22:16:31 +00:00
|
|
|
def t2SBFX: T2TwoRegBitFI<
|
2011-07-27 21:09:25 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, imm0_31:$lsb, imm1_32:$msb),
|
2019-05-15 12:41:58 +00:00
|
|
|
IIC_iUNAsi, "sbfx", "\t$Rd, $Rn, $lsb, $msb", []>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-20} = 0b10100;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2009-10-13 18:59:48 +00:00
|
|
|
|
2010-11-17 22:16:31 +00:00
|
|
|
def t2UBFX: T2TwoRegBitFI<
|
2011-07-27 21:09:25 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, imm0_31:$lsb, imm1_32:$msb),
|
2019-05-15 12:41:58 +00:00
|
|
|
IIC_iUNAsi, "ubfx", "\t$Rd, $Rn, $lsb, $msb", []>, Sched<[WriteALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-20} = 0b11100;
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
2009-10-13 18:59:48 +00:00
|
|
|
|
2014-05-14 03:47:39 +00:00
|
|
|
// A8.8.247 UDF - Undefined (Encoding T2)
|
2014-05-22 04:46:46 +00:00
|
|
|
def t2UDF : T2XI<(outs), (ins imm0_65535:$imm16), IIC_Br, "udf.w\t$imm16",
|
|
|
|
[(int_arm_undefined imm0_65535:$imm16)]> {
|
2014-05-14 03:47:39 +00:00
|
|
|
bits<16> imm16;
|
|
|
|
let Inst{31-29} = 0b111;
|
|
|
|
let Inst{28-27} = 0b10;
|
|
|
|
let Inst{26-20} = 0b1111111;
|
|
|
|
let Inst{19-16} = imm16{15-12};
|
|
|
|
let Inst{15} = 0b1;
|
|
|
|
let Inst{14-12} = 0b010;
|
|
|
|
let Inst{11-0} = imm16{11-0};
|
|
|
|
}
|
|
|
|
|
2010-02-02 19:31:58 +00:00
|
|
|
// A8.6.18 BFI - Bitfield insert (Encoding T1)
|
2011-01-18 20:45:56 +00:00
|
|
|
let Constraints = "$src = $Rd" in {
|
|
|
|
def t2BFI : T2TwoRegBitFI<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$src, rGPR:$Rn, bf_inv_mask_imm:$imm),
|
|
|
|
IIC_iBITi, "bfi", "\t$Rd, $Rn, $imm",
|
|
|
|
[(set rGPR:$Rd, (ARMbfi rGPR:$src, rGPR:$Rn,
|
2019-05-15 12:41:58 +00:00
|
|
|
bf_inv_mask_imm:$imm))]>, Sched<[WriteALU]> {
|
2011-01-18 20:45:56 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
2011-04-15 00:35:08 +00:00
|
|
|
let Inst{26} = 0; // should be 0.
|
2011-01-18 20:45:56 +00:00
|
|
|
let Inst{25} = 1;
|
|
|
|
let Inst{24-20} = 0b10110;
|
|
|
|
let Inst{15} = 0;
|
2011-04-15 00:35:08 +00:00
|
|
|
let Inst{5} = 0; // should be 0.
|
2011-01-18 20:45:56 +00:00
|
|
|
|
|
|
|
bits<10> imm;
|
|
|
|
let msb{4-0} = imm{9-5};
|
|
|
|
let lsb{4-0} = imm{4-0};
|
|
|
|
}
|
2010-02-02 19:31:58 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-09-29 00:27:46 +00:00
|
|
|
defm t2ORN : T2I_bin_irs<0b0011, "orn",
|
|
|
|
IIC_iBITi, IIC_iBITr, IIC_iBITsi,
|
2012-08-02 21:50:41 +00:00
|
|
|
BinOpFrag<(or node:$LHS, (not node:$RHS))>, 0, "">;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2011-09-14 21:24:41 +00:00
|
|
|
/// T2I_un_irs - Defines a set of (op reg, {so_imm|r|so_reg}) patterns for a
|
|
|
|
/// unary operation that produces a value. These are predicable and can be
|
|
|
|
/// changed to modify CPSR.
|
|
|
|
multiclass T2I_un_irs<bits<4> opcod, string opc,
|
|
|
|
InstrItinClass iii, InstrItinClass iir, InstrItinClass iis,
|
2012-12-10 23:21:26 +00:00
|
|
|
PatFrag opnode,
|
|
|
|
bit Cheap = 0, bit ReMat = 0, bit MoveImm = 0> {
|
2011-09-14 21:24:41 +00:00
|
|
|
// shifted imm
|
|
|
|
def i : T2sOneRegImm<(outs rGPR:$Rd), (ins t2_so_imm:$imm), iii,
|
|
|
|
opc, "\t$Rd, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode t2_so_imm:$imm))]>, Sched<[WriteALU]> {
|
2011-09-14 21:24:41 +00:00
|
|
|
let isAsCheapAsAMove = Cheap;
|
|
|
|
let isReMaterializable = ReMat;
|
2012-12-10 23:21:26 +00:00
|
|
|
let isMoveImm = MoveImm;
|
2011-09-14 21:24:41 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{15} = 0;
|
|
|
|
}
|
|
|
|
// register
|
|
|
|
def r : T2sTwoReg<(outs rGPR:$Rd), (ins rGPR:$Rm), iir,
|
|
|
|
opc, ".w\t$Rd, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode rGPR:$Rm))]>, Sched<[WriteALU]> {
|
2011-09-14 21:24:41 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
|
|
|
}
|
|
|
|
// shifted register
|
|
|
|
def s : T2sOneRegShiftedReg<(outs rGPR:$Rd), (ins t2_so_reg:$ShiftedRm), iis,
|
|
|
|
opc, ".w\t$Rd, $ShiftedRm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (opnode t2_so_reg:$ShiftedRm))]>,
|
|
|
|
Sched<[WriteALU]> {
|
2011-09-14 21:24:41 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = opcod;
|
|
|
|
let Inst{19-16} = 0b1111; // Rn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-26 23:13:13 +00:00
|
|
|
// Prefer over of t2EORri ra, rb, -1 because mvn has 16-bit version
|
|
|
|
let AddedComplexity = 1 in
|
2010-09-29 00:49:25 +00:00
|
|
|
defm t2MVN : T2I_un_irs <0b0011, "mvn",
|
2010-09-29 22:42:35 +00:00
|
|
|
IIC_iMVNi, IIC_iMVNr, IIC_iMVNsi,
|
2016-03-08 16:23:54 +00:00
|
|
|
not, 1, 1, 1>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-07-20 16:07:04 +00:00
|
|
|
let AddedComplexity = 1 in
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(and rGPR:$src, t2_so_imm_not:$imm),
|
|
|
|
(t2BICri rGPR:$src, t2_so_imm_not:$imm)>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2012-06-18 14:51:32 +00:00
|
|
|
// top16Zero - answer true if the upper 16 bits of $src are 0, false otherwise
|
|
|
|
def top16Zero: PatLeaf<(i32 rGPR:$src), [{
|
2019-07-04 08:41:23 +00:00
|
|
|
return !SDValue(N,0)->getValueType(0).isVector() &&
|
|
|
|
CurDAG->MaskedValueIsZero(SDValue(N,0), APInt::getHighBitsSet(32, 16));
|
2012-06-18 14:51:32 +00:00
|
|
|
}]>;
|
|
|
|
|
|
|
|
// so_imm_notSext is needed instead of so_imm_not, as the value of imm
|
|
|
|
// will match the extended, not the original bitWidth for $src.
|
|
|
|
def : T2Pat<(and top16Zero:$src, t2_so_imm_notSext:$imm),
|
|
|
|
(t2BICri rGPR:$src, t2_so_imm_notSext:$imm)>;
|
|
|
|
|
|
|
|
|
2009-08-01 06:13:52 +00:00
|
|
|
// FIXME: Disable this pattern on Darwin to workaround an assembler bug.
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(or rGPR:$src, t2_so_imm_not:$imm),
|
|
|
|
(t2ORNri rGPR:$src, t2_so_imm_not:$imm)>,
|
2009-08-12 01:56:42 +00:00
|
|
|
Requires<[IsThumb2]>;
|
2009-07-06 22:23:46 +00:00
|
|
|
|
|
|
|
def : T2Pat<(t2_so_imm_not:$src),
|
|
|
|
(t2MVNi t2_so_imm_not:$src)>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2018-06-20 12:09:44 +00:00
|
|
|
// There are shorter Thumb encodings for ADD than ORR, so to increase
|
|
|
|
// Thumb2SizeReduction's chances later on we select a t2ADD for an or where
|
|
|
|
// possible.
|
|
|
|
def : T2Pat<(or AddLikeOrOp:$Rn, t2_so_imm:$imm),
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri rGPR:$Rn, t2_so_imm:$imm)>;
|
2018-06-20 12:09:44 +00:00
|
|
|
|
|
|
|
def : T2Pat<(or AddLikeOrOp:$Rn, imm0_4095:$Rm),
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri12 rGPR:$Rn, imm0_4095:$Rm)>;
|
2018-06-20 12:09:44 +00:00
|
|
|
|
|
|
|
def : T2Pat<(or AddLikeOrOp:$Rn, non_imm32:$Rm),
|
|
|
|
(t2ADDrr $Rn, $Rm)>;
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Multiply Instructions.
|
|
|
|
//
|
2009-06-26 00:19:44 +00:00
|
|
|
let isCommutable = 1 in
|
2010-11-18 01:08:42 +00:00
|
|
|
def t2MUL: T2ThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iMUL32,
|
|
|
|
"mul", "\t$Rd, $Rn, $Rm",
|
2017-02-22 07:22:57 +00:00
|
|
|
[(set rGPR:$Rd, (mul rGPR:$Rn, rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteMUL32, ReadMUL, ReadMUL]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
|
|
|
let Inst{22-20} = 0b000;
|
|
|
|
let Inst{15-12} = 0b1111; // Ra = 0b1111 (no accumulate)
|
|
|
|
let Inst{7-4} = 0b0000; // Multiply
|
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2016-07-19 14:44:05 +00:00
|
|
|
class T2FourRegMLA<bits<4> op7_4, string opc, list<dag> pattern>
|
|
|
|
: T2FourReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm, rGPR:$Ra), IIC_iMAC32,
|
|
|
|
opc, "\t$Rd, $Rn, $Rm, $Ra", pattern>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, UseMulOps]>,
|
|
|
|
Sched<[WriteMAC32, ReadMUL, ReadMUL, ReadMAC]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
|
|
|
let Inst{22-20} = 0b000;
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{7-4} = op7_4;
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2MLA : T2FourRegMLA<0b0000, "mla",
|
|
|
|
[(set rGPR:$Rd, (add (mul rGPR:$Rn, rGPR:$Rm),
|
|
|
|
rGPR:$Ra))]>;
|
|
|
|
def t2MLS: T2FourRegMLA<0b0001, "mls",
|
|
|
|
[(set rGPR:$Rd, (sub rGPR:$Ra, (mul rGPR:$Rn,
|
|
|
|
rGPR:$Rm)))]>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2009-07-07 01:17:28 +00:00
|
|
|
// Extra precision multiplies with low / high results
|
2014-11-26 00:46:26 +00:00
|
|
|
let hasSideEffects = 0 in {
|
2009-07-07 01:17:28 +00:00
|
|
|
let isCommutable = 1 in {
|
2017-03-14 18:43:37 +00:00
|
|
|
def t2SMULL : T2MulLong<0b000, 0b0000, "smull",
|
|
|
|
[(set rGPR:$RdLo, rGPR:$RdHi,
|
|
|
|
(smullohi rGPR:$Rn, rGPR:$Rm))]>;
|
|
|
|
def t2UMULL : T2MulLong<0b010, 0b0000, "umull",
|
|
|
|
[(set rGPR:$RdLo, rGPR:$RdHi,
|
|
|
|
(umullohi rGPR:$Rn, rGPR:$Rm))]>;
|
2009-12-15 17:24:14 +00:00
|
|
|
} // isCommutable
|
2009-07-07 01:17:28 +00:00
|
|
|
|
|
|
|
// Multiply + accumulate
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2SMLAL : T2MlaLong<0b100, 0b0000, "smlal">;
|
|
|
|
def t2UMLAL : T2MlaLong<0b110, 0b0000, "umlal">;
|
|
|
|
def t2UMAAL : T2MlaLong<0b110, 0b0110, "umaal">, Requires<[IsThumb2, HasDSP]>;
|
2014-11-26 00:46:26 +00:00
|
|
|
} // hasSideEffects
|
2009-07-07 01:17:28 +00:00
|
|
|
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
// Rounding variants of the below included for disassembly only
|
|
|
|
|
2009-07-07 01:17:28 +00:00
|
|
|
// Most significant word multiply
|
2016-07-19 14:44:05 +00:00
|
|
|
class T2SMMUL<bits<4> op7_4, string opc, list<dag> pattern>
|
|
|
|
: T2ThreeReg<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm), IIC_iMUL32,
|
|
|
|
opc, "\t$Rd, $Rn, $Rm", pattern>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>,
|
|
|
|
Sched<[WriteMUL32, ReadMUL, ReadMUL]> {
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
|
|
|
let Inst{22-20} = 0b101;
|
|
|
|
let Inst{15-12} = 0b1111; // Ra = 0b1111 (no accumulate)
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{7-4} = op7_4;
|
2016-07-01 17:26:42 +00:00
|
|
|
}
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2SMMUL : T2SMMUL<0b0000, "smmul", [(set rGPR:$Rd, (mulhs rGPR:$Rn,
|
|
|
|
rGPR:$Rm))]>;
|
2018-01-12 09:24:41 +00:00
|
|
|
def t2SMMULR :
|
|
|
|
T2SMMUL<0b0001, "smmulr",
|
|
|
|
[(set rGPR:$Rd, (ARMsmmlar rGPR:$Rn, rGPR:$Rm, (i32 0)))]>;
|
2016-07-01 17:26:42 +00:00
|
|
|
|
2016-07-19 14:44:05 +00:00
|
|
|
class T2FourRegSMMLA<bits<3> op22_20, bits<4> op7_4, string opc,
|
|
|
|
list<dag> pattern>
|
|
|
|
: T2FourReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm, rGPR:$Ra), IIC_iMAC32,
|
|
|
|
opc, "\t$Rd, $Rn, $Rm, $Ra", pattern>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP, UseMulOps]>,
|
|
|
|
Sched<[WriteMAC32, ReadMUL, ReadMUL, ReadMAC]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{22-20} = op22_20;
|
|
|
|
let Inst{7-4} = op7_4;
|
2016-07-01 17:26:42 +00:00
|
|
|
}
|
|
|
|
|
2016-07-25 09:20:20 +00:00
|
|
|
def t2SMMLA : T2FourRegSMMLA<0b101, 0b0000, "smmla",
|
2016-07-19 14:44:05 +00:00
|
|
|
[(set rGPR:$Rd, (add (mulhs rGPR:$Rm, rGPR:$Rn), rGPR:$Ra))]>;
|
2018-01-12 09:24:41 +00:00
|
|
|
def t2SMMLAR: T2FourRegSMMLA<0b101, 0b0001, "smmlar",
|
|
|
|
[(set rGPR:$Rd, (ARMsmmlar rGPR:$Rn, rGPR:$Rm, rGPR:$Ra))]>;
|
2016-07-25 09:20:20 +00:00
|
|
|
def t2SMMLS: T2FourRegSMMLA<0b110, 0b0000, "smmls", []>;
|
2018-01-12 09:24:41 +00:00
|
|
|
def t2SMMLSR: T2FourRegSMMLA<0b110, 0b0001, "smmlsr",
|
|
|
|
[(set rGPR:$Rd, (ARMsmmlsr rGPR:$Rn, rGPR:$Rm, rGPR:$Ra))]>;
|
2016-07-01 17:26:42 +00:00
|
|
|
|
2016-07-19 14:44:05 +00:00
|
|
|
class T2ThreeRegSMUL<bits<3> op22_20, bits<2> op5_4, string opc,
|
|
|
|
list<dag> pattern>
|
|
|
|
: T2ThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iMUL16, opc,
|
|
|
|
"\t$Rd, $Rn, $Rm", pattern>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>,
|
|
|
|
Sched<[WriteMUL16, ReadMUL, ReadMUL]> {
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{22-20} = op22_20;
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{15-12} = 0b1111; // Ra = 0b1111 (no accumulate)
|
|
|
|
let Inst{7-6} = 0b00;
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{5-4} = op5_4;
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2SMULBB : T2ThreeRegSMUL<0b001, 0b00, "smulbb",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (bb_mul rGPR:$Rn, rGPR:$Rm))]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2SMULBT : T2ThreeRegSMUL<0b001, 0b01, "smulbt",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (bt_mul rGPR:$Rn, rGPR:$Rm))]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2SMULTB : T2ThreeRegSMUL<0b001, 0b10, "smultb",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (tb_mul rGPR:$Rn, rGPR:$Rm))]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
def t2SMULTT : T2ThreeRegSMUL<0b001, 0b11, "smultt",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (tt_mul rGPR:$Rn, rGPR:$Rm))]>;
|
2017-03-14 09:13:22 +00:00
|
|
|
def t2SMULWB : T2ThreeRegSMUL<0b011, 0b00, "smulwb",
|
|
|
|
[(set rGPR:$Rd, (ARMsmulwb rGPR:$Rn, rGPR:$Rm))]>;
|
|
|
|
def t2SMULWT : T2ThreeRegSMUL<0b011, 0b01, "smulwt",
|
|
|
|
[(set rGPR:$Rd, (ARMsmulwt rGPR:$Rn, rGPR:$Rm))]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
2019-01-08 10:12:36 +00:00
|
|
|
def : Thumb2DSPPat<(mul sext_16_node:$Rn, (sext_bottom_16 rGPR:$Rm)),
|
|
|
|
(t2SMULBB rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(mul sext_16_node:$Rn, (sext_top_16 rGPR:$Rm)),
|
2016-08-02 12:44:27 +00:00
|
|
|
(t2SMULBT rGPR:$Rn, rGPR:$Rm)>;
|
2019-01-08 10:12:36 +00:00
|
|
|
def : Thumb2DSPPat<(mul (sext_top_16 rGPR:$Rn), sext_16_node:$Rm),
|
2016-08-02 12:44:27 +00:00
|
|
|
(t2SMULTB rGPR:$Rn, rGPR:$Rm)>;
|
2019-01-08 10:12:36 +00:00
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_smulbb rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULBB rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smulbt rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULBT rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smultb rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULTB rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smultt rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULTT rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smulwb rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULWB rGPR:$Rn, rGPR:$Rm)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smulwt rGPR:$Rn, rGPR:$Rm),
|
|
|
|
(t2SMULWT rGPR:$Rn, rGPR:$Rm)>;
|
2016-08-02 12:44:27 +00:00
|
|
|
|
|
|
|
class T2FourRegSMLA<bits<3> op22_20, bits<2> op5_4, string opc,
|
|
|
|
list<dag> pattern>
|
2016-07-19 14:44:05 +00:00
|
|
|
: T2FourReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm, rGPR:$Ra), IIC_iMUL16,
|
|
|
|
opc, "\t$Rd, $Rn, $Rm, $Ra", pattern>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP, UseMulOps]>,
|
|
|
|
Sched<[WriteMAC16, ReadMUL, ReadMUL, ReadMAC]> {
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-23} = 0b0110;
|
2016-07-25 09:29:24 +00:00
|
|
|
let Inst{22-20} = op22_20;
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{7-6} = 0b00;
|
2016-07-19 14:44:05 +00:00
|
|
|
let Inst{5-4} = op5_4;
|
|
|
|
}
|
|
|
|
|
2016-07-25 09:29:24 +00:00
|
|
|
def t2SMLABB : T2FourRegSMLA<0b001, 0b00, "smlabb",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (bb_mul rGPR:$Rn, rGPR:$Rm)))]>;
|
2016-07-25 09:29:24 +00:00
|
|
|
def t2SMLABT : T2FourRegSMLA<0b001, 0b01, "smlabt",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (bt_mul rGPR:$Rn, rGPR:$Rm)))]>;
|
2016-07-25 09:29:24 +00:00
|
|
|
def t2SMLATB : T2FourRegSMLA<0b001, 0b10, "smlatb",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (tb_mul rGPR:$Rn, rGPR:$Rm)))]>;
|
2016-07-25 09:29:24 +00:00
|
|
|
def t2SMLATT : T2FourRegSMLA<0b001, 0b11, "smlatt",
|
2019-01-08 10:12:36 +00:00
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (tt_mul rGPR:$Rn, rGPR:$Rm)))]>;
|
2017-03-14 09:13:22 +00:00
|
|
|
def t2SMLAWB : T2FourRegSMLA<0b011, 0b00, "smlawb",
|
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (ARMsmulwb rGPR:$Rn, rGPR:$Rm)))]>;
|
|
|
|
def t2SMLAWT : T2FourRegSMLA<0b011, 0b01, "smlawt",
|
|
|
|
[(set rGPR:$Rd, (add rGPR:$Ra, (ARMsmulwt rGPR:$Rn, rGPR:$Rm)))]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
2016-08-02 12:44:27 +00:00
|
|
|
def : Thumb2DSPMulPat<(add rGPR:$Ra, (mul sext_16_node:$Rn, sext_16_node:$Rm)),
|
|
|
|
(t2SMLABB rGPR:$Rn, rGPR:$Rm, rGPR:$Ra)>;
|
2019-01-08 10:12:36 +00:00
|
|
|
def : Thumb2DSPMulPat<(add rGPR:$Ra, (mul sext_16_node:$Rn,
|
|
|
|
(sext_bottom_16 rGPR:$Rm))),
|
|
|
|
(t2SMLABB rGPR:$Rn, rGPR:$Rm, rGPR:$Ra)>;
|
|
|
|
def : Thumb2DSPMulPat<(add rGPR:$Ra, (mul sext_16_node:$Rn,
|
|
|
|
(sext_top_16 rGPR:$Rm))),
|
2016-08-02 12:44:27 +00:00
|
|
|
(t2SMLABT rGPR:$Rn, rGPR:$Rm, rGPR:$Ra)>;
|
2019-01-08 10:12:36 +00:00
|
|
|
def : Thumb2DSPMulPat<(add rGPR:$Ra, (mul (sext_top_16 rGPR:$Rn),
|
|
|
|
sext_16_node:$Rm)),
|
2016-08-02 12:44:27 +00:00
|
|
|
(t2SMLATB rGPR:$Rn, rGPR:$Rm, rGPR:$Ra)>;
|
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(int_arm_smlabb GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLABB GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smlabt GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLABT GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smlatb GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLATB GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smlatt GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLATT GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smlawb GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLAWB GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
def : Thumb2DSPPat<(int_arm_smlawt GPR:$a, GPR:$b, GPR:$acc),
|
|
|
|
(t2SMLAWT GPR:$a, GPR:$b, GPR:$acc)>;
|
|
|
|
|
2016-07-01 17:26:42 +00:00
|
|
|
// Halfword multiple accumulate long: SMLAL<x><y>
|
2017-03-15 08:27:11 +00:00
|
|
|
def t2SMLALBB : T2MlaLong<0b100, 0b1000, "smlalbb">,
|
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
|
|
|
def t2SMLALBT : T2MlaLong<0b100, 0b1001, "smlalbt">,
|
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
|
|
|
def t2SMLALTB : T2MlaLong<0b100, 0b1010, "smlaltb">,
|
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
|
|
|
def t2SMLALTT : T2MlaLong<0b100, 0b1011, "smlaltt">,
|
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
|
|
|
|
|
|
|
def : Thumb2DSPPat<(ARMsmlalbb GPR:$Rn, GPR:$Rm, GPR:$RLo, GPR:$RHi),
|
|
|
|
(t2SMLALBB $Rn, $Rm, $RLo, $RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMsmlalbt GPR:$Rn, GPR:$Rm, GPR:$RLo, GPR:$RHi),
|
|
|
|
(t2SMLALBT $Rn, $Rm, $RLo, $RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMsmlaltb GPR:$Rn, GPR:$Rm, GPR:$RLo, GPR:$RHi),
|
|
|
|
(t2SMLALTB $Rn, $Rm, $RLo, $RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMsmlaltt GPR:$Rn, GPR:$Rm, GPR:$RLo, GPR:$RHi),
|
|
|
|
(t2SMLALTT $Rn, $Rm, $RLo, $RHi)>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
class T2DualHalfMul<bits<3> op22_20, bits<4> op7_4, string opc,
|
|
|
|
Intrinsic intrinsic>
|
2016-07-19 14:44:05 +00:00
|
|
|
: T2ThreeReg_mac<0, op22_20, op7_4,
|
|
|
|
(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm),
|
2017-05-04 07:31:28 +00:00
|
|
|
IIC_iMAC32, opc, "\t$Rd, $Rn, $Rm",
|
|
|
|
[(set rGPR:$Rd, (intrinsic rGPR:$Rn, rGPR:$Rm))]>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>,
|
|
|
|
Sched<[WriteMAC32, ReadMUL, ReadMUL, ReadMAC]> {
|
2016-07-01 17:26:42 +00:00
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
}
|
2016-07-19 14:44:05 +00:00
|
|
|
|
|
|
|
// Dual halfword multiple: SMUAD, SMUSD, SMLAD, SMLSD, SMLALD, SMLSLD
|
2017-05-04 07:31:28 +00:00
|
|
|
def t2SMUAD: T2DualHalfMul<0b010, 0b0000, "smuad", int_arm_smuad>;
|
|
|
|
def t2SMUADX: T2DualHalfMul<0b010, 0b0001, "smuadx", int_arm_smuadx>;
|
|
|
|
def t2SMUSD: T2DualHalfMul<0b100, 0b0000, "smusd", int_arm_smusd>;
|
|
|
|
def t2SMUSDX: T2DualHalfMul<0b100, 0b0001, "smusdx", int_arm_smusdx>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
class T2DualHalfMulAdd<bits<3> op22_20, bits<4> op7_4, string opc,
|
|
|
|
Intrinsic intrinsic>
|
2016-07-19 14:44:05 +00:00
|
|
|
: T2FourReg_mac<0, op22_20, op7_4,
|
|
|
|
(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rn, rGPR:$Rm, rGPR:$Ra),
|
2017-05-04 07:31:28 +00:00
|
|
|
IIC_iMAC32, opc, "\t$Rd, $Rn, $Rm, $Ra",
|
|
|
|
[(set rGPR:$Rd, (intrinsic rGPR:$Rn, rGPR:$Rm, rGPR:$Ra))]>,
|
2016-07-19 14:44:05 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>;
|
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
def t2SMLAD : T2DualHalfMulAdd<0b010, 0b0000, "smlad", int_arm_smlad>;
|
|
|
|
def t2SMLADX : T2DualHalfMulAdd<0b010, 0b0001, "smladx", int_arm_smladx>;
|
|
|
|
def t2SMLSD : T2DualHalfMulAdd<0b100, 0b0000, "smlsd", int_arm_smlsd>;
|
|
|
|
def t2SMLSDX : T2DualHalfMulAdd<0b100, 0b0001, "smlsdx", int_arm_smlsdx>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
|
|
|
class T2DualHalfMulAddLong<bits<3> op22_20, bits<4> op7_4, string opc>
|
|
|
|
: T2FourReg_mac<1, op22_20, op7_4,
|
|
|
|
(outs rGPR:$Ra, rGPR:$Rd),
|
2017-05-04 07:31:28 +00:00
|
|
|
(ins rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi),
|
2016-07-19 14:44:05 +00:00
|
|
|
IIC_iMAC64, opc, "\t$Ra, $Rd, $Rn, $Rm", []>,
|
2017-05-04 07:31:28 +00:00
|
|
|
RegConstraint<"$Ra = $RLo, $Rd = $RHi">,
|
2017-02-22 07:22:57 +00:00
|
|
|
Requires<[IsThumb2, HasDSP]>,
|
|
|
|
Sched<[WriteMAC64Lo, WriteMAC64Hi, ReadMUL, ReadMUL, ReadMAC, ReadMAC]>;
|
2016-07-19 14:44:05 +00:00
|
|
|
|
|
|
|
def t2SMLALD : T2DualHalfMulAddLong<0b100, 0b1100, "smlald">;
|
|
|
|
def t2SMLALDX : T2DualHalfMulAddLong<0b100, 0b1101, "smlaldx">;
|
|
|
|
def t2SMLSLD : T2DualHalfMulAddLong<0b101, 0b1100, "smlsld">;
|
|
|
|
def t2SMLSLDX : T2DualHalfMulAddLong<0b101, 0b1101, "smlsldx">;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2017-05-04 07:31:28 +00:00
|
|
|
def : Thumb2DSPPat<(ARMSmlald rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi),
|
|
|
|
(t2SMLALD rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMSmlaldx rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi),
|
|
|
|
(t2SMLALDX rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMSmlsld rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi),
|
|
|
|
(t2SMLSLD rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi)>;
|
|
|
|
def : Thumb2DSPPat<(ARMSmlsldx rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi),
|
|
|
|
(t2SMLSLDX rGPR:$Rn, rGPR:$Rm, rGPR:$RLo, rGPR:$RHi)>;
|
|
|
|
|
2011-06-21 19:00:54 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Division Instructions.
|
|
|
|
// Signed and unsigned division on v7-M
|
|
|
|
//
|
2012-09-29 21:43:49 +00:00
|
|
|
def t2SDIV : T2ThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iDIV,
|
2011-06-21 19:00:54 +00:00
|
|
|
"sdiv", "\t$Rd, $Rn, $Rm",
|
|
|
|
[(set rGPR:$Rd, (sdiv rGPR:$Rn, rGPR:$Rm))]>,
|
2017-04-20 09:38:25 +00:00
|
|
|
Requires<[HasDivideInThumb, IsThumb, HasV8MBaseline]>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Sched<[WriteDIV]> {
|
2011-06-21 19:00:54 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-21} = 0b011100;
|
|
|
|
let Inst{20} = 0b1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-4} = 0b1111;
|
|
|
|
}
|
|
|
|
|
2012-09-29 21:43:49 +00:00
|
|
|
def t2UDIV : T2ThreeReg<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), IIC_iDIV,
|
2011-06-21 19:00:54 +00:00
|
|
|
"udiv", "\t$Rd, $Rn, $Rm",
|
|
|
|
[(set rGPR:$Rd, (udiv rGPR:$Rn, rGPR:$Rm))]>,
|
2017-04-20 09:38:25 +00:00
|
|
|
Requires<[HasDivideInThumb, IsThumb, HasV8MBaseline]>,
|
2017-02-22 07:22:57 +00:00
|
|
|
Sched<[WriteDIV]> {
|
2011-06-21 19:00:54 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-21} = 0b011101;
|
|
|
|
let Inst{20} = 0b1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-4} = 0b1111;
|
|
|
|
}
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Misc. Arithmetic Instructions.
|
|
|
|
//
|
|
|
|
|
2010-02-16 21:23:02 +00:00
|
|
|
class T2I_misc<bits<2> op1, bits<2> op2, dag oops, dag iops,
|
|
|
|
InstrItinClass itin, string opc, string asm, list<dag> pattern>
|
2010-11-18 21:15:19 +00:00
|
|
|
: T2ThreeReg<oops, iops, itin, opc, asm, pattern> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-22} = 0b01010;
|
|
|
|
let Inst{21-20} = op1;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-6} = 0b10;
|
|
|
|
let Inst{5-4} = op2;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Rn{3-0} = Rm;
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2CLZ : T2I_misc<0b11, 0b00, (outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iUNAr,
|
2013-06-06 16:35:25 +00:00
|
|
|
"clz", "\t$Rd, $Rm", [(set rGPR:$Rd, (ctlz rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU]>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2RBIT : T2I_misc<0b01, 0b10, (outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iUNAr,
|
|
|
|
"rbit", "\t$Rd, $Rm",
|
2015-11-13 16:05:22 +00:00
|
|
|
[(set rGPR:$Rd, (bitreverse rGPR:$Rm))]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteALU]>;
|
2010-01-18 19:58:49 +00:00
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2REV : T2I_misc<0b01, 0b00, (outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iUNAr,
|
2013-06-06 16:35:25 +00:00
|
|
|
"rev", ".w\t$Rd, $Rm", [(set rGPR:$Rd, (bswap rGPR:$Rm))]>,
|
|
|
|
Sched<[WriteALU]>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2REV16 : T2I_misc<0b01, 0b01, (outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iUNAr,
|
|
|
|
"rev16", ".w\t$Rd, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (rotr (bswap rGPR:$Rm), (i32 16)))]>,
|
|
|
|
Sched<[WriteALU]>;
|
2011-06-17 20:47:21 +00:00
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2REVSH : T2I_misc<0b01, 0b11, (outs rGPR:$Rd), (ins rGPR:$Rm), IIC_iUNAr,
|
|
|
|
"revsh", ".w\t$Rd, $Rm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(set rGPR:$Rd, (sra (bswap rGPR:$Rm), (i32 16)))]>,
|
|
|
|
Sched<[WriteALU]>;
|
2011-03-18 21:52:42 +00:00
|
|
|
|
2011-06-15 17:17:48 +00:00
|
|
|
def : T2Pat<(or (sra (shl rGPR:$Rm, (i32 24)), (i32 16)),
|
2011-06-21 06:01:08 +00:00
|
|
|
(and (srl rGPR:$Rm, (i32 8)), 0xFF)),
|
2011-06-15 17:17:48 +00:00
|
|
|
(t2REVSH rGPR:$Rm)>;
|
|
|
|
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2PKHBT : T2ThreeReg<
|
2011-09-14 23:16:41 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm, pkh_lsl_amt:$sh),
|
|
|
|
IIC_iBITsi, "pkhbt", "\t$Rd, $Rn, $Rm$sh",
|
2010-11-18 21:15:19 +00:00
|
|
|
[(set rGPR:$Rd, (or (and rGPR:$Rn, 0xFFFF),
|
2011-07-20 20:49:03 +00:00
|
|
|
(and (shl rGPR:$Rm, pkh_lsl_amt:$sh),
|
2010-05-05 20:44:35 +00:00
|
|
|
0xFFFF0000)))]>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-20} = 0b01100;
|
|
|
|
let Inst{5} = 0; // BT form
|
|
|
|
let Inst{4} = 0;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2011-07-20 20:32:09 +00:00
|
|
|
bits<5> sh;
|
|
|
|
let Inst{14-12} = sh{4-2};
|
|
|
|
let Inst{7-6} = sh{1-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-07-07 05:35:52 +00:00
|
|
|
|
|
|
|
// Alternate cases for PKHBT where identities eliminate some nodes.
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(or (and rGPR:$src1, 0xFFFF), (and rGPR:$src2, 0xFFFF0000)),
|
|
|
|
(t2PKHBT rGPR:$src1, rGPR:$src2, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2010-08-17 17:23:19 +00:00
|
|
|
def : T2Pat<(or (and rGPR:$src1, 0xFFFF), (shl rGPR:$src2, imm16_31:$sh)),
|
2011-07-20 20:32:09 +00:00
|
|
|
(t2PKHBT rGPR:$src1, rGPR:$src2, imm16_31:$sh)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2009-07-07 05:35:52 +00:00
|
|
|
|
2010-08-16 22:26:55 +00:00
|
|
|
// Note: Shifts of 1-15 bits will be transformed to srl instead of sra and
|
|
|
|
// will match the pattern below.
|
2010-11-18 21:15:19 +00:00
|
|
|
def t2PKHTB : T2ThreeReg<
|
2011-09-14 23:16:41 +00:00
|
|
|
(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm, pkh_asr_amt:$sh),
|
|
|
|
IIC_iBITsi, "pkhtb", "\t$Rd, $Rn, $Rm$sh",
|
2010-11-18 21:15:19 +00:00
|
|
|
[(set rGPR:$Rd, (or (and rGPR:$Rn, 0xFFFF0000),
|
2011-07-20 20:49:03 +00:00
|
|
|
(and (sra rGPR:$Rm, pkh_asr_amt:$sh),
|
2010-08-17 17:23:19 +00:00
|
|
|
0xFFFF)))]>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>,
|
2013-06-06 16:35:25 +00:00
|
|
|
Sched<[WriteALUsi, ReadALU]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-20} = 0b01100;
|
|
|
|
let Inst{5} = 1; // TB form
|
|
|
|
let Inst{4} = 0;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2011-07-20 20:32:09 +00:00
|
|
|
bits<5> sh;
|
|
|
|
let Inst{14-12} = sh{4-2};
|
|
|
|
let Inst{7-6} = sh{1-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-07-07 05:35:52 +00:00
|
|
|
|
|
|
|
// Alternate cases for PKHTB where identities eliminate some nodes. Note that
|
|
|
|
// a shift amount of 0 is *not legal* here, it is PKHBT instead.
|
2013-07-09 22:59:22 +00:00
|
|
|
// We also can not replace a srl (17..31) by an arithmetic shift we would use in
|
|
|
|
// pkhtb src1, src2, asr (17..31).
|
|
|
|
def : T2Pat<(or (and rGPR:$src1, 0xFFFF0000), (srl rGPR:$src2, imm16:$sh)),
|
|
|
|
(t2PKHTB rGPR:$src1, rGPR:$src2, imm16:$sh)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2013-07-09 22:59:22 +00:00
|
|
|
def : T2Pat<(or (and rGPR:$src1, 0xFFFF0000), (sra rGPR:$src2, imm16_31:$sh)),
|
2011-07-20 20:32:09 +00:00
|
|
|
(t2PKHTB rGPR:$src1, rGPR:$src2, imm16_31:$sh)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
Many Thumb2 instructions can reference the full ARM register set (i.e.,
have 4 bits per register in the operand encoding), but have undefined
behavior when the operand value is 13 or 15 (SP and PC, respectively).
The trivial coalescer in linear scan sometimes will merge a copy from
SP into a subsequent instruction which uses the copy, and if that
instruction cannot legally reference SP, we get bad code such as:
mls r0,r9,r0,sp
instead of:
mov r2, sp
mls r0, r9, r0, r2
This patch adds a new register class for use by Thumb2 that excludes
the problematic registers (SP and PC) and is used instead of GPR
for those operands which cannot legally reference PC or SP. The
trivial coalescer explicitly requires that the register class
of the destination for the COPY instruction contain the source
register for the COPY to be considered for coalescing. This prevents
errant instructions like that above.
PR7499
llvm-svn: 109842
2010-07-30 02:41:01 +00:00
|
|
|
def : T2Pat<(or (and rGPR:$src1, 0xFFFF0000),
|
2010-08-17 17:23:19 +00:00
|
|
|
(and (srl rGPR:$src2, imm1_15:$sh), 0xFFFF)),
|
2011-07-20 20:32:09 +00:00
|
|
|
(t2PKHTB rGPR:$src1, rGPR:$src2, imm1_15:$sh)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2013-09-18 09:45:55 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// CRC32 Instructions
|
|
|
|
//
|
|
|
|
// Polynomials:
|
|
|
|
// + CRC32{B,H,W} 0x04C11DB7
|
|
|
|
// + CRC32C{B,H,W} 0x1EDC6F41
|
|
|
|
//
|
|
|
|
|
|
|
|
class T2I_crc32<bit C, bits<2> sz, string suffix, SDPatternOperator builtin>
|
|
|
|
: T2ThreeRegNoP<(outs rGPR:$Rd), (ins rGPR:$Rn, rGPR:$Rm), NoItinerary,
|
|
|
|
!strconcat("crc32", suffix, "\t$Rd, $Rn, $Rm"),
|
|
|
|
[(set rGPR:$Rd, (builtin rGPR:$Rn, rGPR:$Rm))]>,
|
2013-10-29 09:47:35 +00:00
|
|
|
Requires<[IsThumb2, HasV8, HasCRC]> {
|
2013-09-18 09:45:55 +00:00
|
|
|
let Inst{31-27} = 0b11111;
|
|
|
|
let Inst{26-21} = 0b010110;
|
|
|
|
let Inst{20} = C;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{7-6} = 0b10;
|
|
|
|
let Inst{5-4} = sz;
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2CRC32B : T2I_crc32<0, 0b00, "b", int_arm_crc32b>;
|
|
|
|
def t2CRC32CB : T2I_crc32<1, 0b00, "cb", int_arm_crc32cb>;
|
|
|
|
def t2CRC32H : T2I_crc32<0, 0b01, "h", int_arm_crc32h>;
|
|
|
|
def t2CRC32CH : T2I_crc32<1, 0b01, "ch", int_arm_crc32ch>;
|
|
|
|
def t2CRC32W : T2I_crc32<0, 0b10, "w", int_arm_crc32w>;
|
|
|
|
def t2CRC32CW : T2I_crc32<1, 0b10, "cw", int_arm_crc32cw>;
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Comparison Instructions...
|
|
|
|
//
|
2019-05-08 10:59:08 +00:00
|
|
|
defm t2CMP : T2I_cmp_irs<0b1101, "cmp", GPRnopc,
|
2016-03-08 16:23:54 +00:00
|
|
|
IIC_iCMPi, IIC_iCMPr, IIC_iCMPsi, ARMcmp>;
|
2010-12-07 20:41:06 +00:00
|
|
|
|
2011-09-06 21:44:58 +00:00
|
|
|
def : T2Pat<(ARMcmpZ GPRnopc:$lhs, t2_so_imm:$imm),
|
|
|
|
(t2CMPri GPRnopc:$lhs, t2_so_imm:$imm)>;
|
|
|
|
def : T2Pat<(ARMcmpZ GPRnopc:$lhs, rGPR:$rhs),
|
|
|
|
(t2CMPrr GPRnopc:$lhs, rGPR:$rhs)>;
|
|
|
|
def : T2Pat<(ARMcmpZ GPRnopc:$lhs, t2_so_reg:$rhs),
|
|
|
|
(t2CMPrs GPRnopc:$lhs, t2_so_reg:$rhs)>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2012-06-11 08:07:26 +00:00
|
|
|
let isCompare = 1, Defs = [CPSR] in {
|
|
|
|
// shifted imm
|
|
|
|
def t2CMNri : T2OneRegCmpImm<
|
|
|
|
(outs), (ins GPRnopc:$Rn, t2_so_imm:$imm), IIC_iCMPi,
|
|
|
|
"cmn", ".w\t$Rn, $imm",
|
2013-06-06 16:35:25 +00:00
|
|
|
[(ARMcmn GPRnopc:$Rn, (ineg t2_so_imm:$imm))]>,
|
|
|
|
Sched<[WriteCMP, ReadALU]> {
|
2012-06-11 08:07:26 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{25} = 0;
|
|
|
|
let Inst{24-21} = 0b1000;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{15} = 0;
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
}
|
|
|
|
// register
|
|
|
|
def t2CMNzrr : T2TwoRegCmp<
|
|
|
|
(outs), (ins GPRnopc:$Rn, rGPR:$Rm), IIC_iCMPr,
|
|
|
|
"cmn", ".w\t$Rn, $Rm",
|
|
|
|
[(BinOpFrag<(ARMcmpZ node:$LHS,(ineg node:$RHS))>
|
2013-06-06 16:35:25 +00:00
|
|
|
GPRnopc:$Rn, rGPR:$Rm)]>, Sched<[WriteCMP, ReadALU, ReadALU]> {
|
2012-06-11 08:07:26 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b1000;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{14-12} = 0b000; // imm3
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
let Inst{7-6} = 0b00; // imm2
|
|
|
|
let Inst{5-4} = 0b00; // type
|
|
|
|
}
|
|
|
|
// shifted register
|
|
|
|
def t2CMNzrs : T2OneRegCmpShiftedReg<
|
|
|
|
(outs), (ins GPRnopc:$Rn, t2_so_reg:$ShiftedRm), IIC_iCMPsi,
|
|
|
|
"cmn", ".w\t$Rn, $ShiftedRm",
|
|
|
|
[(BinOpFrag<(ARMcmpZ node:$LHS,(ineg node:$RHS))>
|
2013-06-06 16:35:25 +00:00
|
|
|
GPRnopc:$Rn, t2_so_reg:$ShiftedRm)]>,
|
|
|
|
Sched<[WriteCMPsi, ReadALU, ReadALU]> {
|
2012-06-11 08:07:26 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-25} = 0b01;
|
|
|
|
let Inst{24-21} = 0b1000;
|
|
|
|
let Inst{20} = 1; // The S bit.
|
|
|
|
let Inst{11-8} = 0b1111; // Rd
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assembler aliases w/o the ".w" suffix.
|
|
|
|
// No alias here for 'rr' version as not all instantiations of this multiclass
|
|
|
|
// want one (CMP in particular, does not).
|
2012-08-02 21:59:52 +00:00
|
|
|
def : t2InstAlias<"cmn${p} $Rn, $imm",
|
|
|
|
(t2CMNri GPRnopc:$Rn, t2_so_imm:$imm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"cmn${p} $Rn, $shift",
|
|
|
|
(t2CMNzrs GPRnopc:$Rn, t2_so_reg:$shift, pred:$p)>;
|
2010-08-26 15:50:25 +00:00
|
|
|
|
2012-06-11 08:07:26 +00:00
|
|
|
def : T2Pat<(ARMcmp GPR:$src, t2_so_imm_neg:$imm),
|
|
|
|
(t2CMNri GPR:$src, t2_so_imm_neg:$imm)>;
|
2010-08-26 15:50:25 +00:00
|
|
|
|
2012-06-11 08:07:26 +00:00
|
|
|
def : T2Pat<(ARMcmpZ GPRnopc:$src, t2_so_imm_neg:$imm),
|
|
|
|
(t2CMNri GPRnopc:$src, t2_so_imm_neg:$imm)>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2019-05-08 10:59:08 +00:00
|
|
|
defm t2TST : T2I_cmp_irs<0b0000, "tst", rGPR,
|
2010-09-29 00:49:25 +00:00
|
|
|
IIC_iTSTi, IIC_iTSTr, IIC_iTSTsi,
|
2012-08-02 21:59:52 +00:00
|
|
|
BinOpFrag<(ARMcmpZ (and_su node:$LHS, node:$RHS), 0)>>;
|
2019-05-08 10:59:08 +00:00
|
|
|
defm t2TEQ : T2I_cmp_irs<0b0100, "teq", rGPR,
|
2010-09-29 00:49:25 +00:00
|
|
|
IIC_iTSTi, IIC_iTSTr, IIC_iTSTsi,
|
2012-08-02 21:59:52 +00:00
|
|
|
BinOpFrag<(ARMcmpZ (xor_su node:$LHS, node:$RHS), 0)>>;
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2009-07-07 20:39:03 +00:00
|
|
|
// Conditional moves
|
2014-11-26 00:46:26 +00:00
|
|
|
let hasSideEffects = 0 in {
|
2012-04-04 18:23:42 +00:00
|
|
|
|
2012-08-16 23:14:20 +00:00
|
|
|
let isCommutable = 1, isSelect = 1 in
|
2011-07-01 17:14:11 +00:00
|
|
|
def t2MOVCCr : t2PseudoInst<(outs rGPR:$Rd),
|
2013-08-22 09:57:11 +00:00
|
|
|
(ins rGPR:$false, rGPR:$Rm, cmovpred:$p),
|
2011-07-13 23:22:26 +00:00
|
|
|
4, IIC_iCMOVr,
|
2013-08-22 09:57:11 +00:00
|
|
|
[(set rGPR:$Rd, (ARMcmov rGPR:$false, rGPR:$Rm,
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $Rd">, Sched<[WriteALU]>;
|
2009-07-07 20:39:03 +00:00
|
|
|
|
2010-11-17 20:13:28 +00:00
|
|
|
let isMoveImm = 1 in
|
2013-08-22 09:57:11 +00:00
|
|
|
def t2MOVCCi
|
|
|
|
: t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$false, t2_so_imm:$imm, cmovpred:$p),
|
2011-07-13 23:22:26 +00:00
|
|
|
4, IIC_iCMOVi,
|
2013-08-22 09:57:11 +00:00
|
|
|
[(set rGPR:$Rd, (ARMcmov rGPR:$false,t2_so_imm:$imm,
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $Rd">, Sched<[WriteALU]>;
|
2009-12-15 17:24:14 +00:00
|
|
|
|
2011-07-01 17:14:11 +00:00
|
|
|
let isCodeGenOnly = 1 in {
|
2010-11-17 20:13:28 +00:00
|
|
|
let isMoveImm = 1 in
|
2013-08-22 09:57:11 +00:00
|
|
|
def t2MOVCCi16
|
|
|
|
: t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$false, imm0_65535_expr:$imm, cmovpred:$p),
|
|
|
|
4, IIC_iCMOVi,
|
|
|
|
[(set rGPR:$Rd, (ARMcmov rGPR:$false, imm0_65535:$imm,
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $Rd">, Sched<[WriteALU]>;
|
2010-10-07 00:53:56 +00:00
|
|
|
|
2010-11-17 20:13:28 +00:00
|
|
|
let isMoveImm = 1 in
|
2013-08-22 09:57:11 +00:00
|
|
|
def t2MVNCCi
|
|
|
|
: t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$false, t2_so_imm:$imm, cmovpred:$p),
|
|
|
|
4, IIC_iCMOVi,
|
|
|
|
[(set rGPR:$Rd,
|
|
|
|
(ARMcmov rGPR:$false, t2_so_imm_not:$imm,
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $Rd">, Sched<[WriteALU]>;
|
|
|
|
|
|
|
|
class MOVCCShPseudo<SDPatternOperator opnode, Operand ty>
|
|
|
|
: t2PseudoInst<(outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$false, rGPR:$Rm, i32imm:$imm, cmovpred:$p),
|
|
|
|
4, IIC_iCMOVsi,
|
|
|
|
[(set rGPR:$Rd, (ARMcmov rGPR:$false,
|
|
|
|
(opnode rGPR:$Rm, (i32 ty:$imm)),
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $Rd">, Sched<[WriteALU]>;
|
|
|
|
|
|
|
|
def t2MOVCClsl : MOVCCShPseudo<shl, imm0_31>;
|
|
|
|
def t2MOVCClsr : MOVCCShPseudo<srl, imm_sr>;
|
|
|
|
def t2MOVCCasr : MOVCCShPseudo<sra, imm_sr>;
|
|
|
|
def t2MOVCCror : MOVCCShPseudo<rotr, imm0_31>;
|
2010-11-13 02:25:14 +00:00
|
|
|
|
2010-11-17 20:13:28 +00:00
|
|
|
let isMoveImm = 1 in
|
2013-08-22 09:57:11 +00:00
|
|
|
def t2MOVCCi32imm
|
|
|
|
: t2PseudoInst<(outs rGPR:$dst),
|
|
|
|
(ins rGPR:$false, i32imm:$src, cmovpred:$p),
|
|
|
|
8, IIC_iCMOVix2,
|
|
|
|
[(set rGPR:$dst, (ARMcmov rGPR:$false, imm:$src,
|
|
|
|
cmovpred:$p))]>,
|
|
|
|
RegConstraint<"$false = $dst">;
|
2012-03-20 21:28:05 +00:00
|
|
|
} // isCodeGenOnly = 1
|
Optimize a couple of common patterns involving conditional moves where the false
value is zero. Instead of a cmov + op, issue an conditional op instead. e.g.
cmp r9, r4
mov r4, #0
moveq r4, #1
orr lr, lr, r4
should be:
cmp r9, r4
orreq lr, lr, #1
That is, optimize (or x, (cmov 0, y, cond)) to (or.cond x, y). Similarly extend
this to xor as well as (and x, (cmov -1, y, cond)) => (and.cond x, y).
It's possible to extend this to ADD and SUB but I don't think they are common.
rdar://8659097
llvm-svn: 151224
2012-02-23 01:19:06 +00:00
|
|
|
|
2014-11-26 00:46:26 +00:00
|
|
|
} // hasSideEffects
|
2009-08-01 01:43:45 +00:00
|
|
|
|
2009-12-14 18:56:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Atomic operations intrinsics
|
|
|
|
//
|
|
|
|
|
|
|
|
// memory barriers protect the atomic sequences
|
|
|
|
let hasSideEffects = 1 in {
|
2013-06-26 16:52:32 +00:00
|
|
|
def t2DMB : T2I<(outs), (ins memb_opt:$opt), NoItinerary,
|
2013-09-05 15:35:24 +00:00
|
|
|
"dmb", "\t$opt", [(int_arm_dmb (i32 imm0_15:$opt))]>,
|
2014-07-03 16:00:41 +00:00
|
|
|
Requires<[IsThumb, HasDB]> {
|
2010-10-30 00:54:37 +00:00
|
|
|
bits<4> opt;
|
|
|
|
let Inst{31-4} = 0xf3bf8f5;
|
|
|
|
let Inst{3-0} = opt;
|
2009-12-14 18:56:47 +00:00
|
|
|
}
|
|
|
|
|
2013-06-26 16:52:32 +00:00
|
|
|
def t2DSB : T2I<(outs), (ins memb_opt:$opt), NoItinerary,
|
2013-09-05 15:35:24 +00:00
|
|
|
"dsb", "\t$opt", [(int_arm_dsb (i32 imm0_15:$opt))]>,
|
2014-07-03 16:00:41 +00:00
|
|
|
Requires<[IsThumb, HasDB]> {
|
2010-10-30 00:54:37 +00:00
|
|
|
bits<4> opt;
|
|
|
|
let Inst{31-4} = 0xf3bf8f4;
|
|
|
|
let Inst{3-0} = opt;
|
2010-03-03 00:16:28 +00:00
|
|
|
}
|
|
|
|
|
2013-06-26 16:52:32 +00:00
|
|
|
def t2ISB : T2I<(outs), (ins instsyncb_opt:$opt), NoItinerary,
|
2014-07-03 16:00:41 +00:00
|
|
|
"isb", "\t$opt", [(int_arm_isb (i32 imm0_15:$opt))]>,
|
|
|
|
Requires<[IsThumb, HasDB]> {
|
2011-09-06 22:53:27 +00:00
|
|
|
bits<4> opt;
|
2010-10-30 00:54:37 +00:00
|
|
|
let Inst{31-4} = 0xf3bf8f6;
|
2011-09-06 22:53:27 +00:00
|
|
|
let Inst{3-0} = opt;
|
2010-03-03 00:16:28 +00:00
|
|
|
}
|
2018-07-06 08:03:12 +00:00
|
|
|
|
|
|
|
let hasNoSchedulingInfo = 1 in
|
|
|
|
def t2TSB : T2I<(outs), (ins tsb_opt:$opt), NoItinerary,
|
|
|
|
"tsb", "\t$opt", []>, Requires<[IsThumb, HasV8_4a]> {
|
|
|
|
let Inst{31-0} = 0xf3af8012;
|
|
|
|
}
|
2014-07-03 16:00:41 +00:00
|
|
|
}
|
2010-03-03 00:16:28 +00:00
|
|
|
|
2018-09-27 13:41:14 +00:00
|
|
|
// Armv8.5-A speculation barrier
|
|
|
|
def t2SB : Thumb2XI<(outs), (ins), AddrModeNone, 4, NoItinerary, "sb", "", []>,
|
2019-01-03 12:09:12 +00:00
|
|
|
Requires<[IsThumb2, HasSB]>, Sched<[]> {
|
2018-09-27 13:41:14 +00:00
|
|
|
let Inst{31-0} = 0xf3bf8f70;
|
|
|
|
let Unpredictable = 0x000f2f0f;
|
|
|
|
let hasSideEffects = 1;
|
|
|
|
}
|
|
|
|
|
2013-08-27 17:38:16 +00:00
|
|
|
class T2I_ldrex<bits<4> opcod, dag oops, dag iops, AddrMode am, int sz,
|
2009-12-15 17:24:14 +00:00
|
|
|
InstrItinClass itin, string opc, string asm, string cstr,
|
|
|
|
list<dag> pattern, bits<4> rt2 = 0b1111>
|
|
|
|
: Thumb2I<oops, iops, am, sz, itin, opc, asm, cstr, pattern> {
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0001101;
|
|
|
|
let Inst{11-8} = rt2;
|
2013-08-27 17:38:16 +00:00
|
|
|
let Inst{7-4} = opcod;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{3-0} = 0b1111;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2011-03-24 21:04:58 +00:00
|
|
|
bits<4> addr;
|
2010-11-19 00:28:38 +00:00
|
|
|
bits<4> Rt;
|
2011-03-24 21:04:58 +00:00
|
|
|
let Inst{19-16} = addr;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2013-08-27 17:38:16 +00:00
|
|
|
class T2I_strex<bits<4> opcod, dag oops, dag iops, AddrMode am, int sz,
|
2009-12-15 17:24:14 +00:00
|
|
|
InstrItinClass itin, string opc, string asm, string cstr,
|
|
|
|
list<dag> pattern, bits<4> rt2 = 0b1111>
|
|
|
|
: Thumb2I<oops, iops, am, sz, itin, opc, asm, cstr, pattern> {
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0001100;
|
|
|
|
let Inst{11-8} = rt2;
|
2013-08-27 17:38:16 +00:00
|
|
|
let Inst{7-4} = opcod;
|
2010-11-19 17:11:02 +00:00
|
|
|
|
2010-11-19 00:28:38 +00:00
|
|
|
bits<4> Rd;
|
2011-03-24 21:04:58 +00:00
|
|
|
bits<4> addr;
|
2010-11-19 00:28:38 +00:00
|
|
|
bits<4> Rt;
|
2011-03-24 21:04:58 +00:00
|
|
|
let Inst{3-0} = Rd;
|
|
|
|
let Inst{19-16} = addr;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
|
|
|
|
2009-12-14 18:56:47 +00:00
|
|
|
let mayLoad = 1 in {
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDREXB : T2I_ldrex<0b0100, (outs rGPR:$Rt), (ins addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2013-07-16 09:46:55 +00:00
|
|
|
"ldrexb", "\t$Rt, $addr", "",
|
2016-01-15 10:26:51 +00:00
|
|
|
[(set rGPR:$Rt, (ldrex_1 addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteLd]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDREXH : T2I_ldrex<0b0101, (outs rGPR:$Rt), (ins addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2013-07-16 09:46:55 +00:00
|
|
|
"ldrexh", "\t$Rt, $addr", "",
|
2016-01-15 10:26:51 +00:00
|
|
|
[(set rGPR:$Rt, (ldrex_2 addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteLd]>;
|
2011-09-09 18:37:27 +00:00
|
|
|
def t2LDREX : Thumb2I<(outs rGPR:$Rt), (ins t2addrmode_imm0_1020s4:$addr),
|
2018-09-07 09:21:25 +00:00
|
|
|
AddrModeT2_ldrex, 4, NoItinerary,
|
2013-07-16 09:46:55 +00:00
|
|
|
"ldrex", "\t$Rt, $addr", "",
|
2016-01-15 10:26:51 +00:00
|
|
|
[(set rGPR:$Rt, (ldrex_4 t2addrmode_imm0_1020s4:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteLd]> {
|
2011-09-09 18:37:27 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<12> addr;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0000101;
|
2011-09-09 18:37:27 +00:00
|
|
|
let Inst{19-16} = addr{11-8};
|
2010-12-10 21:52:38 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2011-09-09 18:37:27 +00:00
|
|
|
let Inst{11-8} = 0b1111;
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2011-05-28 04:07:29 +00:00
|
|
|
let hasExtraDefRegAllocReq = 1 in
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDREXD : T2I_ldrex<0b0111, (outs rGPR:$Rt, rGPR:$Rt2),
|
2011-09-09 18:37:27 +00:00
|
|
|
(ins addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2011-03-24 21:04:58 +00:00
|
|
|
"ldrexd", "\t$Rt, $Rt2, $addr", "",
|
2014-09-29 10:57:29 +00:00
|
|
|
[], {?, ?, ?, ?}>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2, IsNotMClass]>, Sched<[WriteLd]> {
|
2010-11-19 00:28:38 +00:00
|
|
|
bits<4> Rt2;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rt2;
|
2010-11-19 00:28:38 +00:00
|
|
|
}
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDAEXB : T2I_ldrex<0b1100, (outs rGPR:$Rt), (ins addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"ldaexb", "\t$Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rt, (ldaex_1 addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>, Sched<[WriteLd]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDAEXH : T2I_ldrex<0b1101, (outs rGPR:$Rt), (ins addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"ldaexh", "\t$Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rt, (ldaex_2 addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>, Sched<[WriteLd]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2LDAEX : Thumb2I<(outs rGPR:$Rt), (ins addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"ldaex", "\t$Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rt, (ldaex_4 addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>, Sched<[WriteLd]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> addr;
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0001101;
|
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
let Inst{11-8} = 0b1111;
|
|
|
|
let Inst{7-0} = 0b11101111;
|
|
|
|
}
|
|
|
|
let hasExtraDefRegAllocReq = 1 in
|
|
|
|
def t2LDAEXD : T2I_ldrex<0b1111, (outs rGPR:$Rt, rGPR:$Rt2),
|
|
|
|
(ins addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"ldaexd", "\t$Rt, $Rt2, $addr", "",
|
2016-01-15 10:23:46 +00:00
|
|
|
[], {?, ?, ?, ?}>, Requires<[IsThumb,
|
2019-05-15 12:41:58 +00:00
|
|
|
HasAcquireRelease, HasV7Clrex, IsNotMClass]>, Sched<[WriteLd]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rt2;
|
|
|
|
let Inst{11-8} = Rt2;
|
|
|
|
|
|
|
|
let Inst{7} = 1;
|
|
|
|
}
|
2009-12-14 18:56:47 +00:00
|
|
|
}
|
|
|
|
|
2010-11-19 00:28:38 +00:00
|
|
|
let mayStore = 1, Constraints = "@earlyclobber $Rd" in {
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2STREXB : T2I_strex<0b0100, (outs rGPR:$Rd),
|
2011-09-09 18:37:27 +00:00
|
|
|
(ins rGPR:$Rt, addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2013-07-16 09:46:55 +00:00
|
|
|
"strexb", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
2016-01-15 10:26:51 +00:00
|
|
|
(strex_1 rGPR:$Rt, addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteST]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2STREXH : T2I_strex<0b0101, (outs rGPR:$Rd),
|
2011-09-09 18:37:27 +00:00
|
|
|
(ins rGPR:$Rt, addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2013-07-16 09:46:55 +00:00
|
|
|
"strexh", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
2016-01-15 10:26:51 +00:00
|
|
|
(strex_2 rGPR:$Rt, addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteST]>;
|
2013-07-16 09:46:55 +00:00
|
|
|
|
2011-09-09 18:37:27 +00:00
|
|
|
def t2STREX : Thumb2I<(outs rGPR:$Rd), (ins rGPR:$Rt,
|
|
|
|
t2addrmode_imm0_1020s4:$addr),
|
2018-09-07 09:21:25 +00:00
|
|
|
AddrModeT2_ldrex, 4, NoItinerary,
|
2011-03-24 21:04:58 +00:00
|
|
|
"strex", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
2016-01-15 10:26:51 +00:00
|
|
|
(strex_4 rGPR:$Rt, t2addrmode_imm0_1020s4:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteST]> {
|
2010-12-10 21:52:38 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rt;
|
2011-09-09 18:37:27 +00:00
|
|
|
bits<12> addr;
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0000100;
|
|
|
|
let Inst{19-16} = addr{11-8};
|
2010-12-10 21:52:38 +00:00
|
|
|
let Inst{15-12} = Rt;
|
2011-09-09 18:37:27 +00:00
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2012-01-23 22:57:52 +00:00
|
|
|
let hasExtraSrcRegAllocReq = 1 in
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2STREXD : T2I_strex<0b0111, (outs rGPR:$Rd),
|
2011-09-09 18:37:27 +00:00
|
|
|
(ins rGPR:$Rt, rGPR:$Rt2, addr_offset_none:$addr),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 4, NoItinerary,
|
2011-03-24 21:04:58 +00:00
|
|
|
"strexd", "\t$Rd, $Rt, $Rt2, $addr", "", [],
|
2014-09-29 10:57:29 +00:00
|
|
|
{?, ?, ?, ?}>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb2, IsNotMClass]>, Sched<[WriteST]> {
|
2010-11-19 00:28:38 +00:00
|
|
|
bits<4> Rt2;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rt2;
|
2010-11-19 00:28:38 +00:00
|
|
|
}
|
2013-08-27 17:38:16 +00:00
|
|
|
def t2STLEXB : T2I_strex<0b1100, (outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rt, addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"stlexb", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
|
|
|
(stlex_1 rGPR:$Rt, addr_offset_none:$addr))]>,
|
2016-01-15 10:23:46 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease,
|
2019-05-15 12:41:58 +00:00
|
|
|
HasV7Clrex]>, Sched<[WriteST]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
|
|
|
|
def t2STLEXH : T2I_strex<0b1101, (outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rt, addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"stlexh", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
|
|
|
(stlex_2 rGPR:$Rt, addr_offset_none:$addr))]>,
|
2016-01-15 10:23:46 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease,
|
2019-05-15 12:41:58 +00:00
|
|
|
HasV7Clrex]>, Sched<[WriteST]>;
|
2013-08-27 17:38:16 +00:00
|
|
|
|
|
|
|
def t2STLEX : Thumb2I<(outs rGPR:$Rd), (ins rGPR:$Rt,
|
|
|
|
addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"stlex", "\t$Rd, $Rt, $addr", "",
|
2014-03-26 14:39:31 +00:00
|
|
|
[(set rGPR:$Rd,
|
|
|
|
(stlex_4 rGPR:$Rt, addr_offset_none:$addr))]>,
|
2019-05-15 12:41:58 +00:00
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>,
|
|
|
|
Sched<[WriteST]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> addr;
|
|
|
|
let Inst{31-27} = 0b11101;
|
|
|
|
let Inst{26-20} = 0b0001100;
|
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
let Inst{11-4} = 0b11111110;
|
|
|
|
let Inst{3-0} = Rd;
|
|
|
|
}
|
|
|
|
let hasExtraSrcRegAllocReq = 1 in
|
|
|
|
def t2STLEXD : T2I_strex<0b1111, (outs rGPR:$Rd),
|
|
|
|
(ins rGPR:$Rt, rGPR:$Rt2, addr_offset_none:$addr),
|
|
|
|
AddrModeNone, 4, NoItinerary,
|
|
|
|
"stlexd", "\t$Rd, $Rt, $Rt2, $addr", "", [],
|
2016-01-15 10:23:46 +00:00
|
|
|
{?, ?, ?, ?}>, Requires<[IsThumb, HasAcquireRelease,
|
2019-05-15 12:41:58 +00:00
|
|
|
HasV7Clrex, IsNotMClass]>, Sched<[WriteST]> {
|
2013-08-27 17:38:16 +00:00
|
|
|
bits<4> Rt2;
|
|
|
|
let Inst{11-8} = Rt2;
|
|
|
|
}
|
2012-01-23 22:57:52 +00:00
|
|
|
}
|
2009-12-14 18:56:47 +00:00
|
|
|
|
2013-07-16 09:46:55 +00:00
|
|
|
def t2CLREX : T2I<(outs), (ins), NoItinerary, "clrex", "", [(int_arm_clrex)]>,
|
2016-01-15 10:23:46 +00:00
|
|
|
Requires<[IsThumb, HasV7Clrex]> {
|
2011-01-20 19:18:32 +00:00
|
|
|
let Inst{31-16} = 0xf3bf;
|
2010-03-02 22:11:06 +00:00
|
|
|
let Inst{15-14} = 0b10;
|
2011-01-20 19:18:32 +00:00
|
|
|
let Inst{13} = 0;
|
2010-03-02 22:11:06 +00:00
|
|
|
let Inst{12} = 0;
|
2011-01-20 19:18:32 +00:00
|
|
|
let Inst{11-8} = 0b1111;
|
2010-03-02 22:11:06 +00:00
|
|
|
let Inst{7-4} = 0b0010;
|
2011-01-20 19:18:32 +00:00
|
|
|
let Inst{3-0} = 0b1111;
|
2010-03-02 22:11:06 +00:00
|
|
|
}
|
|
|
|
|
2013-07-16 09:46:55 +00:00
|
|
|
def : T2Pat<(and (ldrex_1 addr_offset_none:$addr), 0xff),
|
2016-01-15 10:26:51 +00:00
|
|
|
(t2LDREXB addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]>;
|
2013-07-16 09:46:55 +00:00
|
|
|
def : T2Pat<(and (ldrex_2 addr_offset_none:$addr), 0xffff),
|
2016-01-15 10:26:51 +00:00
|
|
|
(t2LDREXH addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]>;
|
2013-07-16 09:46:55 +00:00
|
|
|
def : T2Pat<(strex_1 (and GPR:$Rt, 0xff), addr_offset_none:$addr),
|
2016-01-15 10:26:51 +00:00
|
|
|
(t2STREXB GPR:$Rt, addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]>;
|
2013-07-16 09:46:55 +00:00
|
|
|
def : T2Pat<(strex_2 (and GPR:$Rt, 0xffff), addr_offset_none:$addr),
|
2016-01-15 10:26:51 +00:00
|
|
|
(t2STREXH GPR:$Rt, addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]>;
|
2013-07-16 09:46:55 +00:00
|
|
|
|
2014-03-26 14:39:31 +00:00
|
|
|
def : T2Pat<(and (ldaex_1 addr_offset_none:$addr), 0xff),
|
2016-01-15 10:23:46 +00:00
|
|
|
(t2LDAEXB addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>;
|
2014-03-26 14:39:31 +00:00
|
|
|
def : T2Pat<(and (ldaex_2 addr_offset_none:$addr), 0xffff),
|
2016-01-15 10:23:46 +00:00
|
|
|
(t2LDAEXH addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>;
|
2014-03-26 14:39:31 +00:00
|
|
|
def : T2Pat<(stlex_1 (and GPR:$Rt, 0xff), addr_offset_none:$addr),
|
2016-01-15 10:23:46 +00:00
|
|
|
(t2STLEXB GPR:$Rt, addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>;
|
2014-03-26 14:39:31 +00:00
|
|
|
def : T2Pat<(stlex_2 (and GPR:$Rt, 0xffff), addr_offset_none:$addr),
|
2016-01-15 10:23:46 +00:00
|
|
|
(t2STLEXH GPR:$Rt, addr_offset_none:$addr)>,
|
|
|
|
Requires<[IsThumb, HasAcquireRelease, HasV7Clrex]>;
|
2014-03-26 14:39:31 +00:00
|
|
|
|
2009-08-11 19:42:21 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// SJLJ Exception handling intrinsics
|
2009-08-13 15:11:43 +00:00
|
|
|
// eh_sjlj_setjmp() is an instruction sequence to store the return
|
2009-08-11 19:42:21 +00:00
|
|
|
// address and save #0 in R0 for the non-longjmp case.
|
|
|
|
// Since by its nature we may be coming from some other function to get
|
|
|
|
// here, and we're using the stack frame for the containing function to
|
|
|
|
// save/restore registers, we can't keep anything live in regs across
|
|
|
|
// the eh_sjlj_setjmp(), else it will almost certainly have been tromped upon
|
2011-04-15 05:18:47 +00:00
|
|
|
// when we get here from a longjmp(). We force everything out of registers
|
2009-08-11 19:42:21 +00:00
|
|
|
// except for our own input by listing the relevant registers in Defs. By
|
|
|
|
// doing so, we also cause the prologue/epilogue code to actively preserve
|
2020-04-23 14:26:07 +09:00
|
|
|
// all of the callee-saved registers, which is exactly what we want.
|
2010-05-27 23:49:24 +00:00
|
|
|
// $val is a scratch register for our use.
|
2010-02-08 23:22:00 +00:00
|
|
|
let Defs =
|
2011-06-07 00:08:49 +00:00
|
|
|
[ R0, R1, R2, R3, R4, R5, R6, R7, R8, R9, R10, R11, R12, LR, CPSR,
|
2012-01-13 22:55:42 +00:00
|
|
|
Q0, Q1, Q2, Q3, Q8, Q9, Q10, Q11, Q12, Q13, Q14, Q15],
|
2011-10-17 22:26:23 +00:00
|
|
|
hasSideEffects = 1, isBarrier = 1, isCodeGenOnly = 1,
|
|
|
|
usesCustomInserter = 1 in {
|
2010-08-26 17:02:47 +00:00
|
|
|
def t2Int_eh_sjlj_setjmp : Thumb2XI<(outs), (ins tGPR:$src, tGPR:$val),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 0, NoItinerary, "", "",
|
2010-08-26 17:02:47 +00:00
|
|
|
[(set R0, (ARMeh_sjlj_setjmp tGPR:$src, tGPR:$val))]>,
|
2010-04-09 20:41:18 +00:00
|
|
|
Requires<[IsThumb2, HasVFP2]>;
|
2009-08-11 19:42:21 +00:00
|
|
|
}
|
|
|
|
|
2010-04-09 20:41:18 +00:00
|
|
|
let Defs =
|
2011-06-07 00:08:49 +00:00
|
|
|
[ R0, R1, R2, R3, R4, R5, R6, R7, R8, R9, R10, R11, R12, LR, CPSR ],
|
2011-10-17 22:26:23 +00:00
|
|
|
hasSideEffects = 1, isBarrier = 1, isCodeGenOnly = 1,
|
|
|
|
usesCustomInserter = 1 in {
|
2010-08-26 17:02:47 +00:00
|
|
|
def t2Int_eh_sjlj_setjmp_nofp : Thumb2XI<(outs), (ins tGPR:$src, tGPR:$val),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 0, NoItinerary, "", "",
|
2010-08-26 17:02:47 +00:00
|
|
|
[(set R0, (ARMeh_sjlj_setjmp tGPR:$src, tGPR:$val))]>,
|
2010-04-09 20:41:18 +00:00
|
|
|
Requires<[IsThumb2, NoVFP]>;
|
|
|
|
}
|
2009-08-11 19:42:21 +00:00
|
|
|
|
|
|
|
|
2009-06-30 18:04:13 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Control-Flow Instructions
|
|
|
|
//
|
|
|
|
|
2009-07-09 22:58:39 +00:00
|
|
|
// FIXME: remove when we have a way to marking a MI with these properties.
|
|
|
|
// FIXME: Should pc be an implicit operand like PICADD, etc?
|
2009-10-01 08:22:27 +00:00
|
|
|
let isReturn = 1, isTerminator = 1, isBarrier = 1, mayLoad = 1,
|
2010-10-31 19:10:56 +00:00
|
|
|
hasExtraDefRegAllocReq = 1, isCodeGenOnly = 1 in
|
2011-07-08 17:40:42 +00:00
|
|
|
def t2LDMIA_RET: t2PseudoExpand<(outs GPR:$wb), (ins GPR:$Rn, pred:$p,
|
2011-06-30 18:25:42 +00:00
|
|
|
reglist:$regs, variable_ops),
|
2011-07-13 23:22:26 +00:00
|
|
|
4, IIC_iLoad_mBr, [],
|
2011-07-08 17:40:42 +00:00
|
|
|
(t2LDMIA_UPD GPR:$wb, GPR:$Rn, pred:$p, reglist:$regs)>,
|
2011-06-30 18:25:42 +00:00
|
|
|
RegConstraint<"$Rn = $wb">;
|
2009-07-09 22:58:39 +00:00
|
|
|
|
2009-06-30 18:04:13 +00:00
|
|
|
let isBranch = 1, isTerminator = 1, isBarrier = 1 in {
|
|
|
|
let isPredicable = 1 in
|
2016-07-11 22:29:37 +00:00
|
|
|
def t2B : T2I<(outs), (ins thumb_br_target:$target), IIC_Br,
|
2011-09-09 21:48:23 +00:00
|
|
|
"b", ".w\t$target",
|
2016-01-15 10:26:17 +00:00
|
|
|
[(br bb:$target)]>, Sched<[WriteBr]>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{15-14} = 0b10;
|
|
|
|
let Inst{12} = 1;
|
2010-11-29 18:54:38 +00:00
|
|
|
|
2012-10-29 23:27:20 +00:00
|
|
|
bits<24> target;
|
2013-08-09 10:38:32 +00:00
|
|
|
let Inst{26} = target{23};
|
|
|
|
let Inst{13} = target{22};
|
|
|
|
let Inst{11} = target{21};
|
2012-10-29 23:27:20 +00:00
|
|
|
let Inst{25-16} = target{20-11};
|
2010-11-29 18:54:38 +00:00
|
|
|
let Inst{10-0} = target{10-0};
|
2012-04-12 23:13:34 +00:00
|
|
|
let DecoderMethod = "DecodeT2BInstruction";
|
2014-04-03 23:43:18 +00:00
|
|
|
let AsmMatchConverter = "cvtThumbBranches";
|
|
|
|
}
|
2009-06-30 18:04:13 +00:00
|
|
|
|
2017-05-30 18:52:33 +00:00
|
|
|
let Size = 4, isNotDuplicable = 1, isBranch = 1, isTerminator = 1,
|
|
|
|
isBarrier = 1, isIndirectBranch = 1 in {
|
2016-12-15 07:59:08 +00:00
|
|
|
|
|
|
|
// available in both v8-M.Baseline and Thumb2 targets
|
|
|
|
def t2BR_JT : t2basePseudoInst<(outs),
|
2015-05-13 20:28:38 +00:00
|
|
|
(ins GPR:$target, GPR:$index, i32imm:$jt),
|
2011-07-13 23:22:26 +00:00
|
|
|
0, IIC_Br,
|
2015-05-13 20:28:38 +00:00
|
|
|
[(ARMbr2jt GPR:$target, GPR:$index, tjumptable:$jt)]>,
|
2013-06-06 18:42:09 +00:00
|
|
|
Sched<[WriteBr]>;
|
2009-07-29 02:18:14 +00:00
|
|
|
|
2015-05-31 19:22:07 +00:00
|
|
|
// FIXME: Add a case that can be predicated.
|
2010-12-15 19:03:16 +00:00
|
|
|
def t2TBB_JT : t2PseudoInst<(outs),
|
2015-05-31 19:22:07 +00:00
|
|
|
(ins GPR:$base, GPR:$index, i32imm:$jt, i32imm:$pclbl), 0, IIC_Br, []>,
|
2013-06-06 18:42:09 +00:00
|
|
|
Sched<[WriteBr]>;
|
2009-07-29 02:18:14 +00:00
|
|
|
|
2010-12-15 19:03:16 +00:00
|
|
|
def t2TBH_JT : t2PseudoInst<(outs),
|
2015-05-31 19:22:07 +00:00
|
|
|
(ins GPR:$base, GPR:$index, i32imm:$jt, i32imm:$pclbl), 0, IIC_Br, []>,
|
2013-06-06 18:42:09 +00:00
|
|
|
Sched<[WriteBr]>;
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
|
2011-09-19 22:21:13 +00:00
|
|
|
def t2TBB : T2I<(outs), (ins addrmode_tbb:$addr), IIC_Br,
|
2013-06-06 18:42:09 +00:00
|
|
|
"tbb", "\t$addr", []>, Sched<[WriteBrTbl]> {
|
2010-11-29 22:37:40 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-12-17 18:42:56 +00:00
|
|
|
let Inst{31-20} = 0b111010001101;
|
2010-11-29 22:37:40 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-5} = 0b11110000000;
|
|
|
|
let Inst{4} = 0; // B form
|
|
|
|
let Inst{3-0} = Rm;
|
2011-09-19 22:21:13 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeThumbTableBranch";
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
}
|
|
|
|
|
2011-09-19 22:21:13 +00:00
|
|
|
def t2TBH : T2I<(outs), (ins addrmode_tbh:$addr), IIC_Br,
|
2013-06-06 18:42:09 +00:00
|
|
|
"tbh", "\t$addr", []>, Sched<[WriteBrTbl]> {
|
2010-11-29 22:37:40 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rm;
|
2010-12-17 18:42:56 +00:00
|
|
|
let Inst{31-20} = 0b111010001101;
|
2010-11-29 22:37:40 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-5} = 0b11110000000;
|
|
|
|
let Inst{4} = 1; // H form
|
|
|
|
let Inst{3-0} = Rm;
|
2011-09-19 22:21:13 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeThumbTableBranch";
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
}
|
2009-07-29 02:18:14 +00:00
|
|
|
} // isNotDuplicable, isIndirectBranch
|
|
|
|
|
2009-06-30 19:50:22 +00:00
|
|
|
} // isBranch, isTerminator, isBarrier
|
2009-06-30 18:04:13 +00:00
|
|
|
|
|
|
|
// FIXME: should be able to write a pattern for ARMBrcond, but can't use
|
2011-09-09 21:48:23 +00:00
|
|
|
// a two-value operand where a dag node expects ", "two operands. :(
|
2009-06-30 18:04:13 +00:00
|
|
|
let isBranch = 1, isTerminator = 1 in
|
2009-08-06 16:52:47 +00:00
|
|
|
def t2Bcc : T2I<(outs), (ins brtarget:$target), IIC_Br,
|
2009-10-27 00:08:59 +00:00
|
|
|
"b", ".w\t$target",
|
2013-06-06 18:42:09 +00:00
|
|
|
[/*(ARMbrcond bb:$target, imm:$cc)*/]>, Sched<[WriteBr]> {
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{15-14} = 0b10;
|
|
|
|
let Inst{12} = 0;
|
2010-12-14 20:46:39 +00:00
|
|
|
|
2010-12-09 00:27:41 +00:00
|
|
|
bits<4> p;
|
|
|
|
let Inst{25-22} = p;
|
|
|
|
|
|
|
|
bits<21> target;
|
|
|
|
let Inst{26} = target{20};
|
|
|
|
let Inst{11} = target{19};
|
|
|
|
let Inst{13} = target{18};
|
|
|
|
let Inst{21-16} = target{17-12};
|
|
|
|
let Inst{10-0} = target{11-1};
|
2011-08-09 20:55:18 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeThumb2BCCInstruction";
|
2013-08-09 10:38:32 +00:00
|
|
|
let AsmMatchConverter = "cvtThumbBranches";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-06-23 17:48:47 +00:00
|
|
|
|
2014-01-06 14:28:05 +00:00
|
|
|
// Tail calls. The MachO version of thumb tail calls uses a t2 branch, so
|
2011-07-08 20:32:21 +00:00
|
|
|
// it goes here.
|
|
|
|
let isCall = 1, isTerminator = 1, isReturn = 1, isBarrier = 1 in {
|
2011-12-20 18:26:50 +00:00
|
|
|
// IOS version.
|
2012-02-24 01:19:29 +00:00
|
|
|
let Uses = [SP] in
|
2011-09-09 21:48:23 +00:00
|
|
|
def tTAILJMPd: tPseudoExpand<(outs),
|
2016-07-11 22:29:37 +00:00
|
|
|
(ins thumb_br_target:$dst, pred:$p),
|
2011-07-13 23:22:26 +00:00
|
|
|
4, IIC_Br, [],
|
2016-07-11 22:29:37 +00:00
|
|
|
(t2B thumb_br_target:$dst, pred:$p)>,
|
2014-01-06 14:28:05 +00:00
|
|
|
Requires<[IsThumb2, IsMachO]>, Sched<[WriteBr]>;
|
2011-07-08 20:32:21 +00:00
|
|
|
}
|
2009-07-10 01:54:42 +00:00
|
|
|
|
|
|
|
// IT block
|
2020-02-27 15:36:53 +00:00
|
|
|
let Defs = [ITSTATE] in
|
2009-07-10 01:54:42 +00:00
|
|
|
def t2IT : Thumb2XI<(outs), (ins it_pred:$cc, it_mask:$mask),
|
2011-07-13 23:22:26 +00:00
|
|
|
AddrModeNone, 2, IIC_iALUx,
|
2013-10-03 09:31:51 +00:00
|
|
|
"it$mask\t$cc", "", []>,
|
|
|
|
ComplexDeprecationPredicate<"IT"> {
|
2009-12-15 17:24:14 +00:00
|
|
|
// 16-bit instruction.
|
2009-12-16 02:32:54 +00:00
|
|
|
let Inst{31-16} = 0x0000;
|
2009-12-15 17:24:14 +00:00
|
|
|
let Inst{15-8} = 0b10111111;
|
2010-11-29 18:54:38 +00:00
|
|
|
|
|
|
|
bits<4> cc;
|
|
|
|
bits<4> mask;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{7-4} = cc;
|
|
|
|
let Inst{3-0} = mask;
|
2011-08-30 22:58:27 +00:00
|
|
|
|
|
|
|
let DecoderMethod = "DecodeIT";
|
2009-12-15 17:24:14 +00:00
|
|
|
}
|
2009-07-10 01:54:42 +00:00
|
|
|
|
2010-02-25 19:05:29 +00:00
|
|
|
// Branch and Exchange Jazelle -- for disassembly only
|
|
|
|
// Rm = Inst{19-16}
|
[ARM] Implement -mimplicit-it assembler option
This option, compatible with gas's -mimplicit-it, controls the
generation/checking of implicit IT blocks in ARM/Thumb assembly.
This option allows two behaviours that were not possible before:
- When in ARM mode, emit a warning when assembling a conditional
instruction that is not in an IT block. This is enabled with
-mimplicit-it=never and -mimplicit-it=thumb.
- When in Thumb mode, automatically generate IT instructions when an
instruction with a condition code appears outside of an IT block. This
is enabled with -mimplicit-it=thumb and -mimplicit-it=always.
The default option is -mimplicit-it=arm, which matches the existing
behaviour (allow conditional ARM instructions outside IT blocks without
warning, and error if a conditional Thumb instruction is outside an IT
block).
The general strategy for generating IT blocks in Thumb mode is to keep a
small list of instructions which should be in the IT block, and only
emit them when we encounter something in the input which means we cannot
continue the block. This could be caused by:
- A non-predicable instruction
- An instruction with a condition not compatible with the IT block
- The IT block already contains 4 instructions
- A branch-like instruction (including ALU instructions with the PC as
the destination), which cannot appear in the middle of an IT block
- A label (branching into an IT block is not legal)
- A change of section, architecture, ISA, etc
- The end of the assembly file.
Some of these, such as change of section and end of file, are parsed
outside of the ARM asm parser, so I've added a new virtual function to
AsmParser to ensure any previously-parsed instructions have been
emitted. The ARM implementation of this flushes the currently pending IT
block.
We now have to try instruction matching up to 3 times, because we cannot
know if the current IT block is valid before matching, and instruction
matching changes depending on the IT block state (due to the 16-bit ALU
instructions, which set the flags iff not in an IT block). In the common
case of not having an open implicit IT block and the instruction being
matched not needing one, we still only have to run the matcher once.
I've removed the ITState.FirstCond variable, because it does not store
any information that isn't already represented by CurPosition. I've also
updated the comment on CurPosition to accurately describe it's meaning
(which this patch doesn't change).
Differential Revision: https://reviews.llvm.org/D22760
llvm-svn: 276747
2016-07-26 14:19:47 +00:00
|
|
|
let isBranch = 1, isTerminator = 1, isBarrier = 1, isIndirectBranch = 1 in
|
2015-04-15 17:28:23 +00:00
|
|
|
def t2BXJ : T2I<(outs), (ins GPRnopc:$func), NoItinerary, "bxj", "\t$func", []>,
|
|
|
|
Sched<[WriteBr]>, Requires<[IsThumb2, IsNotMClass]> {
|
2011-09-02 23:43:09 +00:00
|
|
|
bits<4> func;
|
2010-02-25 19:05:29 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{26} = 0;
|
|
|
|
let Inst{25-20} = 0b111100;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = func;
|
2011-09-02 23:43:09 +00:00
|
|
|
let Inst{15-0} = 0b1000111100000000;
|
2010-02-25 19:05:29 +00:00
|
|
|
}
|
|
|
|
|
2011-08-18 17:51:36 +00:00
|
|
|
// Compare and branch on zero / non-zero
|
|
|
|
let isBranch = 1, isTerminator = 1 in {
|
2016-07-11 22:29:37 +00:00
|
|
|
def tCBZ : T1I<(outs), (ins tGPR:$Rn, thumb_cb_target:$target), IIC_Br,
|
2011-08-18 17:51:36 +00:00
|
|
|
"cbz\t$Rn, $target", []>,
|
|
|
|
T1Misc<{0,0,?,1,?,?,?}>,
|
2016-01-15 10:26:17 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteBr]> {
|
2011-08-18 17:51:36 +00:00
|
|
|
// A8.6.27
|
|
|
|
bits<6> target;
|
|
|
|
bits<3> Rn;
|
|
|
|
let Inst{9} = target{5};
|
|
|
|
let Inst{7-3} = target{4-0};
|
|
|
|
let Inst{2-0} = Rn;
|
|
|
|
}
|
|
|
|
|
2016-07-11 22:29:37 +00:00
|
|
|
def tCBNZ : T1I<(outs), (ins tGPR:$Rn, thumb_cb_target:$target), IIC_Br,
|
2011-08-18 17:51:36 +00:00
|
|
|
"cbnz\t$Rn, $target", []>,
|
|
|
|
T1Misc<{1,0,?,1,?,?,?}>,
|
2016-01-15 10:26:17 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline]>, Sched<[WriteBr]> {
|
2011-08-18 17:51:36 +00:00
|
|
|
// A8.6.27
|
|
|
|
bits<6> target;
|
|
|
|
bits<3> Rn;
|
|
|
|
let Inst{9} = target{5};
|
|
|
|
let Inst{7-3} = target{4-0};
|
|
|
|
let Inst{2-0} = Rn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-19 23:38:34 +00:00
|
|
|
// Change Processor State is a system instruction.
|
2011-02-14 13:09:44 +00:00
|
|
|
// FIXME: Since the asm parser has currently no clean way to handle optional
|
|
|
|
// operands, create 3 versions of the same instruction. Once there's a clean
|
|
|
|
// framework to represent optional operands, change this behavior.
|
|
|
|
class t2CPS<dag iops, string asm_op> : T2XI<(outs), iops, NoItinerary,
|
2014-09-24 14:20:01 +00:00
|
|
|
!strconcat("cps", asm_op), []>,
|
|
|
|
Requires<[IsThumb2, IsNotMClass]> {
|
2011-02-14 13:09:44 +00:00
|
|
|
bits<2> imod;
|
|
|
|
bits<3> iflags;
|
|
|
|
bits<5> mode;
|
|
|
|
bit M;
|
|
|
|
|
2013-04-26 17:54:54 +00:00
|
|
|
let Inst{31-11} = 0b111100111010111110000;
|
2011-02-14 13:09:44 +00:00
|
|
|
let Inst{10-9} = imod;
|
|
|
|
let Inst{8} = M;
|
|
|
|
let Inst{7-5} = iflags;
|
|
|
|
let Inst{4-0} = mode;
|
2011-08-23 17:45:18 +00:00
|
|
|
let DecoderMethod = "DecodeT2CPSInstruction";
|
2011-02-14 13:09:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let M = 1 in
|
|
|
|
def t2CPS3p : t2CPS<(ins imod_op:$imod, iflags_op:$iflags, i32imm:$mode),
|
2013-08-09 13:52:32 +00:00
|
|
|
"$imod\t$iflags, $mode">;
|
2011-02-14 13:09:44 +00:00
|
|
|
let mode = 0, M = 0 in
|
|
|
|
def t2CPS2p : t2CPS<(ins imod_op:$imod, iflags_op:$iflags),
|
|
|
|
"$imod.w\t$iflags">;
|
|
|
|
let imod = 0, iflags = 0, M = 1 in
|
2011-09-19 23:58:31 +00:00
|
|
|
def t2CPS1p : t2CPS<(ins imm0_31:$mode), "\t$mode">;
|
Added 32-bit Thumb instructions: CPS, SDIV, UDIV, SXTB16, SXTAB16, UXTAB16, SEL,
SMMULR, SMMLAR, SMMLSR, TBB, TBH, and 16-bit Thumb instruction CPS for
disassembly only.
llvm-svn: 97573
2010-03-02 18:14:57 +00:00
|
|
|
|
2013-08-09 13:52:32 +00:00
|
|
|
def : t2InstAlias<"cps$imod.w $iflags, $mode",
|
|
|
|
(t2CPS3p imod_op:$imod, iflags_op:$iflags, i32imm:$mode), 0>;
|
|
|
|
def : t2InstAlias<"cps.w $mode", (t2CPS1p imm0_31:$mode), 0>;
|
|
|
|
|
2010-03-03 02:09:43 +00:00
|
|
|
// A6.3.4 Branches and miscellaneous control
|
|
|
|
// Table A6-14 Change Processor State, and hint instructions
|
2014-04-25 17:24:24 +00:00
|
|
|
def t2HINT : T2I<(outs), (ins imm0_239:$imm), NoItinerary, "hint", ".w\t$imm",
|
|
|
|
[(int_arm_hint imm0_239:$imm)]> {
|
2013-10-23 10:14:40 +00:00
|
|
|
bits<8> imm;
|
2013-04-26 17:54:54 +00:00
|
|
|
let Inst{31-3} = 0b11110011101011111000000000000;
|
2013-10-23 10:14:40 +00:00
|
|
|
let Inst{7-0} = imm;
|
2012-06-18 19:45:50 +00:00
|
|
|
}
|
|
|
|
|
2016-06-03 13:19:43 +00:00
|
|
|
def : t2InstAlias<"hint$p $imm", (t2HINT imm0_239:$imm, pred:$p), 0>;
|
|
|
|
def : t2InstAlias<"nop$p.w", (t2HINT 0, pred:$p), 1>;
|
|
|
|
def : t2InstAlias<"yield$p.w", (t2HINT 1, pred:$p), 1>;
|
|
|
|
def : t2InstAlias<"wfe$p.w", (t2HINT 2, pred:$p), 1>;
|
|
|
|
def : t2InstAlias<"wfi$p.w", (t2HINT 3, pred:$p), 1>;
|
|
|
|
def : t2InstAlias<"sev$p.w", (t2HINT 4, pred:$p), 1>;
|
|
|
|
def : t2InstAlias<"sevl$p.w", (t2HINT 5, pred:$p), 1> {
|
2013-10-01 12:39:11 +00:00
|
|
|
let Predicates = [IsThumb2, HasV8];
|
|
|
|
}
|
2016-06-03 14:03:27 +00:00
|
|
|
def : t2InstAlias<"esb$p.w", (t2HINT 16, pred:$p), 1> {
|
|
|
|
let Predicates = [IsThumb2, HasRAS];
|
|
|
|
}
|
|
|
|
def : t2InstAlias<"esb$p", (t2HINT 16, pred:$p), 0> {
|
|
|
|
let Predicates = [IsThumb2, HasRAS];
|
|
|
|
}
|
2018-02-06 09:24:47 +00:00
|
|
|
def : t2InstAlias<"csdb$p.w", (t2HINT 20, pred:$p), 0>;
|
|
|
|
def : t2InstAlias<"csdb$p", (t2HINT 20, pred:$p), 1>;
|
2010-03-03 02:09:43 +00:00
|
|
|
|
2014-08-26 12:47:26 +00:00
|
|
|
def t2DBG : T2I<(outs), (ins imm0_15:$opt), NoItinerary, "dbg", "\t$opt",
|
|
|
|
[(int_arm_dbg imm0_15:$opt)]> {
|
2010-11-30 20:00:01 +00:00
|
|
|
bits<4> opt;
|
2011-09-06 22:06:40 +00:00
|
|
|
let Inst{31-20} = 0b111100111010;
|
|
|
|
let Inst{19-16} = 0b1111;
|
|
|
|
let Inst{15-8} = 0b10000000;
|
|
|
|
let Inst{7-4} = 0b1111;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{3-0} = opt;
|
2010-03-03 02:09:43 +00:00
|
|
|
}
|
|
|
|
|
2011-09-19 23:38:34 +00:00
|
|
|
// Secure Monitor Call is a system instruction.
|
2010-02-25 20:25:24 +00:00
|
|
|
// Option = Inst{19-16}
|
[ARM] Implement -mimplicit-it assembler option
This option, compatible with gas's -mimplicit-it, controls the
generation/checking of implicit IT blocks in ARM/Thumb assembly.
This option allows two behaviours that were not possible before:
- When in ARM mode, emit a warning when assembling a conditional
instruction that is not in an IT block. This is enabled with
-mimplicit-it=never and -mimplicit-it=thumb.
- When in Thumb mode, automatically generate IT instructions when an
instruction with a condition code appears outside of an IT block. This
is enabled with -mimplicit-it=thumb and -mimplicit-it=always.
The default option is -mimplicit-it=arm, which matches the existing
behaviour (allow conditional ARM instructions outside IT blocks without
warning, and error if a conditional Thumb instruction is outside an IT
block).
The general strategy for generating IT blocks in Thumb mode is to keep a
small list of instructions which should be in the IT block, and only
emit them when we encounter something in the input which means we cannot
continue the block. This could be caused by:
- A non-predicable instruction
- An instruction with a condition not compatible with the IT block
- The IT block already contains 4 instructions
- A branch-like instruction (including ALU instructions with the PC as
the destination), which cannot appear in the middle of an IT block
- A label (branching into an IT block is not legal)
- A change of section, architecture, ISA, etc
- The end of the assembly file.
Some of these, such as change of section and end of file, are parsed
outside of the ARM asm parser, so I've added a new virtual function to
AsmParser to ensure any previously-parsed instructions have been
emitted. The ARM implementation of this flushes the currently pending IT
block.
We now have to try instruction matching up to 3 times, because we cannot
know if the current IT block is valid before matching, and instruction
matching changes depending on the IT block state (due to the 16-bit ALU
instructions, which set the flags iff not in an IT block). In the common
case of not having an open implicit IT block and the instruction being
matched not needing one, we still only have to run the matcher once.
I've removed the ITState.FirstCond variable, because it does not store
any information that isn't already represented by CurPosition. I've also
updated the comment on CurPosition to accurately describe it's meaning
(which this patch doesn't change).
Differential Revision: https://reviews.llvm.org/D22760
llvm-svn: 276747
2016-07-26 14:19:47 +00:00
|
|
|
let isCall = 1, Uses = [SP] in
|
2014-04-03 23:43:18 +00:00
|
|
|
def t2SMC : T2I<(outs), (ins imm0_15:$opt), NoItinerary, "smc", "\t$opt",
|
2013-04-10 12:08:35 +00:00
|
|
|
[]>, Requires<[IsThumb2, HasTrustZone]> {
|
2010-02-25 20:25:24 +00:00
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{26-20} = 0b1111111;
|
|
|
|
let Inst{15-12} = 0b1000;
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 19:22:08 +00:00
|
|
|
bits<4> opt;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = opt;
|
2010-11-29 19:22:08 +00:00
|
|
|
}
|
|
|
|
|
2013-09-05 14:14:19 +00:00
|
|
|
class T2DCPS<bits<2> opt, string opc>
|
|
|
|
: T2I<(outs), (ins), NoItinerary, opc, "", []>, Requires<[IsThumb2, HasV8]> {
|
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{26-20} = 0b1111000;
|
|
|
|
let Inst{19-16} = 0b1111;
|
|
|
|
let Inst{15-12} = 0b1000;
|
|
|
|
let Inst{11-2} = 0b0000000000;
|
|
|
|
let Inst{1-0} = opt;
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2DCPS1 : T2DCPS<0b01, "dcps1">;
|
|
|
|
def t2DCPS2 : T2DCPS<0b10, "dcps2">;
|
|
|
|
def t2DCPS3 : T2DCPS<0b11, "dcps3">;
|
|
|
|
|
2011-09-16 18:25:22 +00:00
|
|
|
class T2SRS<bits<2> Op, bit W, dag oops, dag iops, InstrItinClass itin,
|
|
|
|
string opc, string asm, list<dag> pattern>
|
2014-10-20 15:37:35 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern>,
|
|
|
|
Requires<[IsThumb2,IsNotMClass]> {
|
2010-11-29 19:22:08 +00:00
|
|
|
bits<5> mode;
|
2011-09-16 18:25:22 +00:00
|
|
|
let Inst{31-25} = 0b1110100;
|
|
|
|
let Inst{24-23} = Op;
|
|
|
|
let Inst{22} = 0;
|
|
|
|
let Inst{21} = W;
|
|
|
|
let Inst{20-16} = 0b01101;
|
|
|
|
let Inst{15-5} = 0b11000000000;
|
2010-11-29 19:22:08 +00:00
|
|
|
let Inst{4-0} = mode{4-0};
|
2010-02-25 20:25:24 +00:00
|
|
|
}
|
|
|
|
|
2011-09-16 18:25:22 +00:00
|
|
|
// Store Return State is a system instruction.
|
|
|
|
def t2SRSDB_UPD : T2SRS<0b00, 1, (outs), (ins imm0_31:$mode), NoItinerary,
|
|
|
|
"srsdb", "\tsp!, $mode", []>;
|
|
|
|
def t2SRSDB : T2SRS<0b00, 0, (outs), (ins imm0_31:$mode), NoItinerary,
|
|
|
|
"srsdb","\tsp, $mode", []>;
|
|
|
|
def t2SRSIA_UPD : T2SRS<0b11, 1, (outs), (ins imm0_31:$mode), NoItinerary,
|
|
|
|
"srsia","\tsp!, $mode", []>;
|
|
|
|
def t2SRSIA : T2SRS<0b11, 0, (outs), (ins imm0_31:$mode), NoItinerary,
|
|
|
|
"srsia","\tsp, $mode", []>;
|
2010-11-29 19:22:08 +00:00
|
|
|
|
2013-02-23 00:52:09 +00:00
|
|
|
|
|
|
|
def : t2InstAlias<"srsdb${p} $mode", (t2SRSDB imm0_31:$mode, pred:$p)>;
|
|
|
|
def : t2InstAlias<"srsdb${p} $mode!", (t2SRSDB_UPD imm0_31:$mode, pred:$p)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"srsia${p} $mode", (t2SRSIA imm0_31:$mode, pred:$p)>;
|
|
|
|
def : t2InstAlias<"srsia${p} $mode!", (t2SRSIA_UPD imm0_31:$mode, pred:$p)>;
|
|
|
|
|
2011-09-16 18:25:22 +00:00
|
|
|
// Return From Exception is a system instruction.
|
[ARM] Implement -mimplicit-it assembler option
This option, compatible with gas's -mimplicit-it, controls the
generation/checking of implicit IT blocks in ARM/Thumb assembly.
This option allows two behaviours that were not possible before:
- When in ARM mode, emit a warning when assembling a conditional
instruction that is not in an IT block. This is enabled with
-mimplicit-it=never and -mimplicit-it=thumb.
- When in Thumb mode, automatically generate IT instructions when an
instruction with a condition code appears outside of an IT block. This
is enabled with -mimplicit-it=thumb and -mimplicit-it=always.
The default option is -mimplicit-it=arm, which matches the existing
behaviour (allow conditional ARM instructions outside IT blocks without
warning, and error if a conditional Thumb instruction is outside an IT
block).
The general strategy for generating IT blocks in Thumb mode is to keep a
small list of instructions which should be in the IT block, and only
emit them when we encounter something in the input which means we cannot
continue the block. This could be caused by:
- A non-predicable instruction
- An instruction with a condition not compatible with the IT block
- The IT block already contains 4 instructions
- A branch-like instruction (including ALU instructions with the PC as
the destination), which cannot appear in the middle of an IT block
- A label (branching into an IT block is not legal)
- A change of section, architecture, ISA, etc
- The end of the assembly file.
Some of these, such as change of section and end of file, are parsed
outside of the ARM asm parser, so I've added a new virtual function to
AsmParser to ensure any previously-parsed instructions have been
emitted. The ARM implementation of this flushes the currently pending IT
block.
We now have to try instruction matching up to 3 times, because we cannot
know if the current IT block is valid before matching, and instruction
matching changes depending on the IT block state (due to the 16-bit ALU
instructions, which set the flags iff not in an IT block). In the common
case of not having an open implicit IT block and the instruction being
matched not needing one, we still only have to run the matcher once.
I've removed the ITState.FirstCond variable, because it does not store
any information that isn't already represented by CurPosition. I've also
updated the comment on CurPosition to accurately describe it's meaning
(which this patch doesn't change).
Differential Revision: https://reviews.llvm.org/D22760
llvm-svn: 276747
2016-07-26 14:19:47 +00:00
|
|
|
let isReturn = 1, isBarrier = 1, isTerminator = 1, Defs = [PC] in
|
2010-11-29 20:38:48 +00:00
|
|
|
class T2RFE<bits<12> op31_20, dag oops, dag iops, InstrItinClass itin,
|
2010-11-29 19:22:08 +00:00
|
|
|
string opc, string asm, list<dag> pattern>
|
2014-10-20 15:37:35 +00:00
|
|
|
: T2I<oops, iops, itin, opc, asm, pattern>,
|
|
|
|
Requires<[IsThumb2,IsNotMClass]> {
|
2010-11-29 20:38:48 +00:00
|
|
|
let Inst{31-20} = op31_20{11-0};
|
2010-12-02 23:05:38 +00:00
|
|
|
|
2010-11-29 19:22:08 +00:00
|
|
|
bits<4> Rn;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{19-16} = Rn;
|
2011-04-12 21:41:51 +00:00
|
|
|
let Inst{15-0} = 0xc000;
|
2010-11-29 19:22:08 +00:00
|
|
|
}
|
|
|
|
|
2010-11-29 20:38:48 +00:00
|
|
|
def t2RFEDBW : T2RFE<0b111010000011,
|
2011-04-12 21:41:51 +00:00
|
|
|
(outs), (ins GPR:$Rn), NoItinerary, "rfedb", "\t$Rn!",
|
2010-11-29 20:38:48 +00:00
|
|
|
[/* For disassembly only; pattern left blank */]>;
|
|
|
|
def t2RFEDB : T2RFE<0b111010000001,
|
2011-04-12 21:41:51 +00:00
|
|
|
(outs), (ins GPR:$Rn), NoItinerary, "rfedb", "\t$Rn",
|
2010-11-29 20:38:48 +00:00
|
|
|
[/* For disassembly only; pattern left blank */]>;
|
|
|
|
def t2RFEIAW : T2RFE<0b111010011011,
|
2011-04-12 21:41:51 +00:00
|
|
|
(outs), (ins GPR:$Rn), NoItinerary, "rfeia", "\t$Rn!",
|
2010-11-29 20:38:48 +00:00
|
|
|
[/* For disassembly only; pattern left blank */]>;
|
|
|
|
def t2RFEIA : T2RFE<0b111010011001,
|
2011-04-12 21:41:51 +00:00
|
|
|
(outs), (ins GPR:$Rn), NoItinerary, "rfeia", "\t$Rn",
|
2010-11-29 20:38:48 +00:00
|
|
|
[/* For disassembly only; pattern left blank */]>;
|
2010-02-25 20:25:24 +00:00
|
|
|
|
2013-07-31 21:05:30 +00:00
|
|
|
// B9.3.19 SUBS PC, LR, #imm (Thumb2) system instruction.
|
2013-10-01 14:33:28 +00:00
|
|
|
// Exception return instruction is "subs pc, lr, #imm".
|
|
|
|
let isReturn = 1, isBarrier = 1, isTerminator = 1, Defs = [PC] in
|
2013-07-31 21:05:30 +00:00
|
|
|
def t2SUBS_PC_LR : T2I <(outs), (ins imm0_255:$imm), NoItinerary,
|
2013-10-01 14:33:28 +00:00
|
|
|
"subs", "\tpc, lr, $imm",
|
|
|
|
[(ARMintretflag imm0_255:$imm)]>,
|
2014-10-20 15:37:35 +00:00
|
|
|
Requires<[IsThumb2,IsNotMClass]> {
|
2013-07-31 21:05:30 +00:00
|
|
|
let Inst{31-8} = 0b111100111101111010001111;
|
|
|
|
|
|
|
|
bits<8> imm;
|
|
|
|
let Inst{7-0} = imm;
|
|
|
|
}
|
|
|
|
|
2014-12-01 08:39:19 +00:00
|
|
|
// Hypervisor Call is a system instruction.
|
|
|
|
let isCall = 1 in {
|
|
|
|
def t2HVC : T2XI <(outs), (ins imm0_65535:$imm16), IIC_Br, "hvc.w\t$imm16", []>,
|
|
|
|
Requires<[IsThumb2, HasVirtualization]>, Sched<[WriteBr]> {
|
|
|
|
bits<16> imm16;
|
|
|
|
let Inst{31-20} = 0b111101111110;
|
|
|
|
let Inst{19-16} = imm16{15-12};
|
|
|
|
let Inst{15-12} = 0b1000;
|
|
|
|
let Inst{11-0} = imm16{11-0};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alias for HVC without the ".w" optional width specifier
|
|
|
|
def : t2InstAlias<"hvc\t$imm16", (t2HVC imm0_65535:$imm16)>;
|
|
|
|
|
|
|
|
// ERET - Return from exception in Hypervisor mode.
|
|
|
|
// B9.3.3, B9.3.20: ERET is an alias for "SUBS PC, LR, #0" in an implementation that
|
|
|
|
// includes virtualization extensions.
|
2016-06-03 13:19:43 +00:00
|
|
|
def t2ERET : InstAlias<"eret${p}", (t2SUBS_PC_LR 0, pred:$p), 1>,
|
2014-12-01 08:39:19 +00:00
|
|
|
Requires<[IsThumb2, HasVirtualization]>;
|
|
|
|
|
2009-06-23 17:48:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Non-Instruction Patterns
|
|
|
|
//
|
|
|
|
|
2009-09-28 09:14:39 +00:00
|
|
|
// 32-bit immediate using movw + movt.
|
2010-09-24 22:03:46 +00:00
|
|
|
// This is a single pseudo instruction to make it re-materializable.
|
|
|
|
// FIXME: Remove this when we can do generalized remat.
|
2011-01-19 02:16:49 +00:00
|
|
|
let isReMaterializable = 1, isMoveImm = 1 in
|
2010-10-06 22:01:26 +00:00
|
|
|
def t2MOVi32imm : PseudoInst<(outs rGPR:$dst), (ins i32imm:$src), IIC_iMOVix2,
|
2010-11-18 01:38:26 +00:00
|
|
|
[(set rGPR:$dst, (i32 imm:$src))]>,
|
2014-01-23 13:43:47 +00:00
|
|
|
Requires<[IsThumb, UseMovt]>;
|
2009-11-06 23:52:48 +00:00
|
|
|
|
2011-01-21 18:55:51 +00:00
|
|
|
// Pseudo instruction that combines movw + movt + add pc (if pic).
|
2011-01-20 08:34:58 +00:00
|
|
|
// It also makes it possible to rematerialize the instructions.
|
|
|
|
// FIXME: Remove this when we can do generalized remat and when machine licm
|
|
|
|
// can properly the instructions.
|
2011-01-21 18:55:51 +00:00
|
|
|
let isReMaterializable = 1 in {
|
|
|
|
def t2MOV_ga_pcrel : PseudoInst<(outs rGPR:$dst), (ins i32imm:$addr),
|
|
|
|
IIC_iMOVix2addpc,
|
2011-01-20 08:34:58 +00:00
|
|
|
[(set rGPR:$dst, (ARMWrapperPIC tglobaladdr:$addr))]>,
|
2017-11-13 20:45:38 +00:00
|
|
|
Requires<[IsThumb, HasV8MBaseline, UseMovtInPic]>;
|
2011-01-17 08:03:18 +00:00
|
|
|
|
2011-01-21 18:55:51 +00:00
|
|
|
}
|
|
|
|
|
2016-01-07 09:03:03 +00:00
|
|
|
def : T2Pat<(ARMWrapperPIC tglobaltlsaddr :$dst),
|
|
|
|
(t2MOV_ga_pcrel tglobaltlsaddr:$dst)>,
|
2017-11-13 20:45:38 +00:00
|
|
|
Requires<[IsThumb2, UseMovtInPic]>;
|
2016-01-07 09:03:03 +00:00
|
|
|
def : T2Pat<(ARMWrapper tglobaltlsaddr:$dst),
|
|
|
|
(t2MOVi32imm tglobaltlsaddr:$dst)>,
|
|
|
|
Requires<[IsThumb2, UseMovt]>;
|
|
|
|
|
2009-11-24 00:44:37 +00:00
|
|
|
// ConstantPool, GlobalAddress, and JumpTable
|
2016-06-07 03:14:57 +00:00
|
|
|
def : T2Pat<(ARMWrapper tconstpool :$dst), (t2LEApcrel tconstpool :$dst)>;
|
|
|
|
def : T2Pat<(ARMWrapper texternalsym :$dst), (t2MOVi32imm texternalsym :$dst)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline, UseMovt]>;
|
|
|
|
def : T2Pat<(ARMWrapper tglobaladdr :$dst), (t2MOVi32imm tglobaladdr :$dst)>,
|
|
|
|
Requires<[IsThumb, HasV8MBaseline, UseMovt]>;
|
|
|
|
|
|
|
|
def : T2Pat<(ARMWrapperJT tjumptable:$dst), (t2LEApcrelJT tjumptable:$dst)>;
|
2009-11-24 00:44:37 +00:00
|
|
|
|
2009-11-06 23:52:48 +00:00
|
|
|
// Pseudo instruction that combines ldr from constpool and add pc. This should
|
|
|
|
// be expanded into two instructions late to allow if-conversion and
|
|
|
|
// scheduling.
|
2010-02-27 23:47:46 +00:00
|
|
|
let canFoldAsLoad = 1, isReMaterializable = 1 in
|
2011-01-20 08:34:58 +00:00
|
|
|
def t2LDRpci_pic : PseudoInst<(outs rGPR:$dst), (ins i32imm:$addr, pclabel:$cp),
|
2010-11-18 01:38:26 +00:00
|
|
|
IIC_iLoadiALU,
|
2011-01-20 08:34:58 +00:00
|
|
|
[(set rGPR:$dst, (ARMpic_add (load (ARMWrapper tconstpool:$addr)),
|
2009-11-06 23:52:48 +00:00
|
|
|
imm:$cp))]>,
|
|
|
|
Requires<[IsThumb2]>;
|
2011-10-10 22:59:55 +00:00
|
|
|
|
2020-04-23 14:26:07 +09:00
|
|
|
// Pseudo instruction that combines movs + predicated rsbmi
|
2011-10-10 22:59:55 +00:00
|
|
|
// to implement integer ABS
|
2019-09-29 08:38:48 +00:00
|
|
|
let usesCustomInserter = 1, Defs = [CPSR], hasNoSchedulingInfo = 1 in {
|
2011-10-10 22:59:55 +00:00
|
|
|
def t2ABS : PseudoInst<(outs rGPR:$dst), (ins rGPR:$src),
|
|
|
|
NoItinerary, []>, Requires<[IsThumb2]>;
|
|
|
|
}
|
|
|
|
|
2011-09-07 21:10:42 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Coprocessor load/store -- for disassembly only
|
|
|
|
//
|
2016-05-31 12:39:30 +00:00
|
|
|
class T2CI<bits<4> op31_28, dag oops, dag iops, string opc, string asm, list<dag> pattern>
|
|
|
|
: T2I<oops, iops, NoItinerary, opc, asm, pattern> {
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{31-28} = op31_28;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{27-25} = 0b110;
|
|
|
|
}
|
|
|
|
|
2016-05-31 12:39:30 +00:00
|
|
|
multiclass t2LdStCop<bits<4> op31_28, bit load, bit Dbit, string asm, list<dag> pattern> {
|
2011-10-12 20:54:17 +00:00
|
|
|
def _OFFSET : T2CI<op31_28,
|
|
|
|
(outs), (ins p_imm:$cop, c_imm:$CRd, addrmode5:$addr),
|
2016-05-31 12:39:30 +00:00
|
|
|
asm, "\t$cop, $CRd, $addr", pattern> {
|
2011-10-12 20:54:17 +00:00
|
|
|
bits<13> addr;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<4> CRd;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{24} = 1; // P = 1
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{23} = addr{8};
|
|
|
|
let Inst{22} = Dbit;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{21} = 0; // W = 0
|
|
|
|
let Inst{20} = load;
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{19-16} = addr{12-9};
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2011-09-07 21:10:42 +00:00
|
|
|
let DecoderMethod = "DecodeCopMemInstruction";
|
|
|
|
}
|
2011-10-12 20:54:17 +00:00
|
|
|
def _PRE : T2CI<op31_28,
|
2013-06-14 11:21:35 +00:00
|
|
|
(outs), (ins p_imm:$cop, c_imm:$CRd, addrmode5_pre:$addr),
|
2016-05-31 12:39:30 +00:00
|
|
|
asm, "\t$cop, $CRd, $addr!", []> {
|
2011-10-12 20:54:17 +00:00
|
|
|
bits<13> addr;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<4> CRd;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{24} = 1; // P = 1
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{23} = addr{8};
|
|
|
|
let Inst{22} = Dbit;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{21} = 1; // W = 1
|
|
|
|
let Inst{20} = load;
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{19-16} = addr{12-9};
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{7-0} = addr{7-0};
|
2011-09-07 21:10:42 +00:00
|
|
|
let DecoderMethod = "DecodeCopMemInstruction";
|
|
|
|
}
|
2011-10-12 20:54:17 +00:00
|
|
|
def _POST: T2CI<op31_28,
|
|
|
|
(outs), (ins p_imm:$cop, c_imm:$CRd, addr_offset_none:$addr,
|
|
|
|
postidx_imm8s4:$offset),
|
2016-05-31 12:39:30 +00:00
|
|
|
asm, "\t$cop, $CRd, $addr, $offset", []> {
|
2011-10-12 20:54:17 +00:00
|
|
|
bits<9> offset;
|
|
|
|
bits<4> addr;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<4> CRd;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{24} = 0; // P = 0
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{23} = offset{8};
|
|
|
|
let Inst{22} = Dbit;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{21} = 1; // W = 1
|
|
|
|
let Inst{20} = load;
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{7-0} = offset{7-0};
|
2011-09-07 21:10:42 +00:00
|
|
|
let DecoderMethod = "DecodeCopMemInstruction";
|
|
|
|
}
|
2011-10-12 20:54:17 +00:00
|
|
|
def _OPTION : T2CI<op31_28, (outs),
|
|
|
|
(ins p_imm:$cop, c_imm:$CRd, addr_offset_none:$addr,
|
|
|
|
coproc_option_imm:$option),
|
2016-05-31 12:39:30 +00:00
|
|
|
asm, "\t$cop, $CRd, $addr, $option", []> {
|
2011-10-12 20:54:17 +00:00
|
|
|
bits<8> option;
|
|
|
|
bits<4> addr;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<4> CRd;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{24} = 0; // P = 0
|
|
|
|
let Inst{23} = 1; // U = 1
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{22} = Dbit;
|
2011-09-07 21:10:42 +00:00
|
|
|
let Inst{21} = 0; // W = 0
|
|
|
|
let Inst{20} = load;
|
2011-10-12 20:54:17 +00:00
|
|
|
let Inst{19-16} = addr;
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{7-0} = option;
|
2011-09-07 21:10:42 +00:00
|
|
|
let DecoderMethod = "DecodeCopMemInstruction";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-11 09:52:30 +00:00
|
|
|
let DecoderNamespace = "Thumb2CoProc" in {
|
2019-09-19 16:26:14 +00:00
|
|
|
defm t2LDC : t2LdStCop<0b1110, 1, 0, "ldc", [(int_arm_ldc timm:$cop, timm:$CRd, addrmode5:$addr)]>;
|
|
|
|
defm t2LDCL : t2LdStCop<0b1110, 1, 1, "ldcl", [(int_arm_ldcl timm:$cop, timm:$CRd, addrmode5:$addr)]>;
|
|
|
|
defm t2LDC2 : t2LdStCop<0b1111, 1, 0, "ldc2", [(int_arm_ldc2 timm:$cop, timm:$CRd, addrmode5:$addr)]>, Requires<[PreV8,IsThumb2]>;
|
|
|
|
defm t2LDC2L : t2LdStCop<0b1111, 1, 1, "ldc2l", [(int_arm_ldc2l timm:$cop, timm:$CRd, addrmode5:$addr)]>, Requires<[PreV8,IsThumb2]>;
|
2016-05-31 12:39:30 +00:00
|
|
|
|
2019-09-19 16:26:14 +00:00
|
|
|
defm t2STC : t2LdStCop<0b1110, 0, 0, "stc", [(int_arm_stc timm:$cop, timm:$CRd, addrmode5:$addr)]>;
|
|
|
|
defm t2STCL : t2LdStCop<0b1110, 0, 1, "stcl", [(int_arm_stcl timm:$cop, timm:$CRd, addrmode5:$addr)]>;
|
|
|
|
defm t2STC2 : t2LdStCop<0b1111, 0, 0, "stc2", [(int_arm_stc2 timm:$cop, timm:$CRd, addrmode5:$addr)]>, Requires<[PreV8,IsThumb2]>;
|
|
|
|
defm t2STC2L : t2LdStCop<0b1111, 0, 1, "stc2l", [(int_arm_stc2l timm:$cop, timm:$CRd, addrmode5:$addr)]>, Requires<[PreV8,IsThumb2]>;
|
2017-08-11 09:52:30 +00:00
|
|
|
}
|
2011-09-07 21:10:42 +00:00
|
|
|
|
2010-02-25 18:46:43 +00:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Move between special register and ARM core register -- for disassembly only
|
|
|
|
//
|
2011-09-14 20:03:46 +00:00
|
|
|
// Move to ARM core register from Special Register
|
2011-09-28 14:21:38 +00:00
|
|
|
|
|
|
|
// A/R class MRS.
|
|
|
|
//
|
|
|
|
// A/R class can only move from CPSR or SPSR.
|
2012-04-23 22:04:10 +00:00
|
|
|
def t2MRS_AR : T2I<(outs GPR:$Rd), (ins), NoItinerary, "mrs", "\t$Rd, apsr",
|
2013-09-23 14:26:15 +00:00
|
|
|
[]>, Requires<[IsThumb2,IsNotMClass]> {
|
2011-09-14 20:03:46 +00:00
|
|
|
bits<4> Rd;
|
|
|
|
let Inst{31-12} = 0b11110011111011111000;
|
|
|
|
let Inst{11-8} = Rd;
|
2014-08-07 05:46:54 +00:00
|
|
|
let Inst{7-0} = 0b00000000;
|
2010-11-29 19:29:15 +00:00
|
|
|
}
|
|
|
|
|
2011-09-28 14:21:38 +00:00
|
|
|
def : t2InstAlias<"mrs${p} $Rd, cpsr", (t2MRS_AR GPR:$Rd, pred:$p)>;
|
2011-09-14 20:03:46 +00:00
|
|
|
|
2012-04-23 22:04:10 +00:00
|
|
|
def t2MRSsys_AR: T2I<(outs GPR:$Rd), (ins), NoItinerary, "mrs", "\t$Rd, spsr",
|
2013-09-23 14:26:15 +00:00
|
|
|
[]>, Requires<[IsThumb2,IsNotMClass]> {
|
2010-11-29 20:38:48 +00:00
|
|
|
bits<4> Rd;
|
2011-09-14 20:03:46 +00:00
|
|
|
let Inst{31-12} = 0b11110011111111111000;
|
2010-12-08 22:10:43 +00:00
|
|
|
let Inst{11-8} = Rd;
|
2014-08-07 05:46:54 +00:00
|
|
|
let Inst{7-0} = 0b00000000;
|
2010-02-25 18:46:43 +00:00
|
|
|
}
|
|
|
|
|
2014-08-15 10:47:12 +00:00
|
|
|
def t2MRSbanked : T2I<(outs rGPR:$Rd), (ins banked_reg:$banked),
|
|
|
|
NoItinerary, "mrs", "\t$Rd, $banked", []>,
|
|
|
|
Requires<[IsThumb, HasVirtualization]> {
|
|
|
|
bits<6> banked;
|
|
|
|
bits<4> Rd;
|
|
|
|
|
|
|
|
let Inst{31-21} = 0b11110011111;
|
|
|
|
let Inst{20} = banked{5}; // R bit
|
|
|
|
let Inst{19-16} = banked{3-0};
|
|
|
|
let Inst{15-12} = 0b1000;
|
|
|
|
let Inst{11-8} = Rd;
|
|
|
|
let Inst{7-5} = 0b001;
|
|
|
|
let Inst{4} = banked{4};
|
|
|
|
let Inst{3-0} = 0b0000;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-28 14:21:38 +00:00
|
|
|
// M class MRS.
|
|
|
|
//
|
|
|
|
// This MRS has a mask field in bits 7-0 and can take more values than
|
|
|
|
// the A/R class (a full msr_mask).
|
2014-09-01 11:25:07 +00:00
|
|
|
def t2MRS_M : T2I<(outs rGPR:$Rd), (ins msr_mask:$SYSm), NoItinerary,
|
|
|
|
"mrs", "\t$Rd, $SYSm", []>,
|
2012-04-27 01:27:19 +00:00
|
|
|
Requires<[IsThumb,IsMClass]> {
|
2011-09-28 14:21:38 +00:00
|
|
|
bits<4> Rd;
|
2014-09-01 11:25:07 +00:00
|
|
|
bits<8> SYSm;
|
2011-09-28 14:21:38 +00:00
|
|
|
let Inst{31-12} = 0b11110011111011111000;
|
|
|
|
let Inst{11-8} = Rd;
|
2014-09-01 11:25:07 +00:00
|
|
|
let Inst{7-0} = SYSm;
|
|
|
|
|
|
|
|
let Unpredictable{20-16} = 0b11111;
|
|
|
|
let Unpredictable{13} = 0b1;
|
2011-09-28 14:21:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-18 19:45:59 +00:00
|
|
|
// Move from ARM core register to Special Register
|
|
|
|
//
|
2011-09-28 14:21:38 +00:00
|
|
|
// A/R class MSR.
|
|
|
|
//
|
2011-02-18 19:45:59 +00:00
|
|
|
// No need to have both system and application versions, the encodings are the
|
|
|
|
// same and the assembly parser has no way to distinguish between them. The mask
|
|
|
|
// operand contains the special register (R Bit) in bit 4 and bits 3-0 contains
|
|
|
|
// the mask with the fields to be accessed in the special register.
|
2016-06-08 15:26:34 +00:00
|
|
|
let Defs = [CPSR] in
|
2011-09-28 14:21:38 +00:00
|
|
|
def t2MSR_AR : T2I<(outs), (ins msr_mask:$mask, rGPR:$Rn),
|
|
|
|
NoItinerary, "msr", "\t$mask, $Rn", []>,
|
2013-09-23 14:26:15 +00:00
|
|
|
Requires<[IsThumb2,IsNotMClass]> {
|
2011-02-18 19:45:59 +00:00
|
|
|
bits<5> mask;
|
2010-11-29 19:29:15 +00:00
|
|
|
bits<4> Rn;
|
2011-09-14 20:03:46 +00:00
|
|
|
let Inst{31-21} = 0b11110011100;
|
2011-02-18 19:45:59 +00:00
|
|
|
let Inst{20} = mask{4}; // R Bit
|
2011-09-14 20:03:46 +00:00
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = 0b1000;
|
2011-02-18 19:45:59 +00:00
|
|
|
let Inst{11-8} = mask{3-0};
|
2011-09-14 20:03:46 +00:00
|
|
|
let Inst{7-0} = 0;
|
2010-11-29 19:29:15 +00:00
|
|
|
}
|
|
|
|
|
2014-08-15 10:47:12 +00:00
|
|
|
// However, the MSR (banked register) system instruction (ARMv7VE) *does* have a
|
|
|
|
// separate encoding (distinguished by bit 5.
|
|
|
|
def t2MSRbanked : T2I<(outs), (ins banked_reg:$banked, rGPR:$Rn),
|
|
|
|
NoItinerary, "msr", "\t$banked, $Rn", []>,
|
|
|
|
Requires<[IsThumb, HasVirtualization]> {
|
|
|
|
bits<6> banked;
|
|
|
|
bits<4> Rn;
|
|
|
|
|
|
|
|
let Inst{31-21} = 0b11110011100;
|
|
|
|
let Inst{20} = banked{5}; // R bit
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = 0b1000;
|
|
|
|
let Inst{11-8} = banked{3-0};
|
|
|
|
let Inst{7-5} = 0b001;
|
|
|
|
let Inst{4} = banked{4};
|
|
|
|
let Inst{3-0} = 0b0000;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-28 14:21:38 +00:00
|
|
|
// M class MSR.
|
|
|
|
//
|
|
|
|
// Move from ARM core register to Special Register
|
2016-06-08 15:26:34 +00:00
|
|
|
let Defs = [CPSR] in
|
2011-09-28 14:21:38 +00:00
|
|
|
def t2MSR_M : T2I<(outs), (ins msr_mask:$SYSm, rGPR:$Rn),
|
|
|
|
NoItinerary, "msr", "\t$SYSm, $Rn", []>,
|
2012-04-27 01:27:19 +00:00
|
|
|
Requires<[IsThumb,IsMClass]> {
|
2012-05-17 22:18:01 +00:00
|
|
|
bits<12> SYSm;
|
2011-09-28 14:21:38 +00:00
|
|
|
bits<4> Rn;
|
|
|
|
let Inst{31-21} = 0b11110011100;
|
|
|
|
let Inst{20} = 0b0;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = 0b1000;
|
2014-09-01 11:25:07 +00:00
|
|
|
let Inst{11-10} = SYSm{11-10};
|
|
|
|
let Inst{9-8} = 0b00;
|
|
|
|
let Inst{7-0} = SYSm{7-0};
|
|
|
|
|
|
|
|
let Unpredictable{20} = 0b1;
|
|
|
|
let Unpredictable{13} = 0b1;
|
|
|
|
let Unpredictable{9-8} = 0b11;
|
2011-09-28 14:21:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-20 16:58:48 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2011-07-13 21:14:23 +00:00
|
|
|
// Move between coprocessor and ARM core register
|
2011-01-20 16:58:48 +00:00
|
|
|
//
|
|
|
|
|
2011-07-13 21:35:10 +00:00
|
|
|
class t2MovRCopro<bits<4> Op, string opc, bit direction, dag oops, dag iops,
|
|
|
|
list<dag> pattern>
|
2013-06-26 16:52:40 +00:00
|
|
|
: T2Cop<Op, oops, iops, opc, "\t$cop, $opc1, $Rt, $CRn, $CRm, $opc2",
|
2011-07-13 21:14:23 +00:00
|
|
|
pattern> {
|
|
|
|
let Inst{27-24} = 0b1110;
|
|
|
|
let Inst{20} = direction;
|
|
|
|
let Inst{4} = 1;
|
|
|
|
|
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<3> opc1;
|
|
|
|
bits<3> opc2;
|
|
|
|
bits<4> CRm;
|
|
|
|
bits<4> CRn;
|
|
|
|
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{23-21} = opc1;
|
|
|
|
let Inst{7-5} = opc2;
|
|
|
|
let Inst{3-0} = CRm;
|
|
|
|
let Inst{19-16} = CRn;
|
2017-08-11 09:52:30 +00:00
|
|
|
|
|
|
|
let DecoderNamespace = "Thumb2CoProc";
|
2011-07-13 21:14:23 +00:00
|
|
|
}
|
|
|
|
|
2015-06-01 12:02:47 +00:00
|
|
|
class t2MovRRCopro<bits<4> Op, string opc, bit direction, dag oops, dag iops,
|
2011-07-13 21:35:10 +00:00
|
|
|
list<dag> pattern = []>
|
2015-06-01 12:02:47 +00:00
|
|
|
: T2Cop<Op, oops, iops, opc, "\t$cop, $opc1, $Rt, $Rt2, $CRm", pattern> {
|
2011-07-13 21:14:23 +00:00
|
|
|
let Inst{27-24} = 0b1100;
|
|
|
|
let Inst{23-21} = 0b010;
|
|
|
|
let Inst{20} = direction;
|
|
|
|
|
|
|
|
bits<4> Rt;
|
|
|
|
bits<4> Rt2;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<4> opc1;
|
|
|
|
bits<4> CRm;
|
|
|
|
|
|
|
|
let Inst{15-12} = Rt;
|
|
|
|
let Inst{19-16} = Rt2;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{7-4} = opc1;
|
|
|
|
let Inst{3-0} = CRm;
|
2017-08-11 09:52:30 +00:00
|
|
|
|
|
|
|
let DecoderNamespace = "Thumb2CoProc";
|
2011-07-13 21:14:23 +00:00
|
|
|
}
|
|
|
|
|
2011-07-13 21:35:10 +00:00
|
|
|
/* from ARM core register to coprocessor */
|
|
|
|
def t2MCR : t2MovRCopro<0b1110, "mcr", 0,
|
|
|
|
(outs),
|
2011-07-14 21:19:17 +00:00
|
|
|
(ins p_imm:$cop, imm0_7:$opc1, GPR:$Rt, c_imm:$CRn,
|
|
|
|
c_imm:$CRm, imm0_7:$opc2),
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_mcr timm:$cop, timm:$opc1, GPR:$Rt, timm:$CRn,
|
|
|
|
timm:$CRm, timm:$opc2)]>,
|
2013-09-17 09:54:57 +00:00
|
|
|
ComplexDeprecationPredicate<"MCR">;
|
2013-06-26 16:52:40 +00:00
|
|
|
def : t2InstAlias<"mcr${p} $cop, $opc1, $Rt, $CRn, $CRm",
|
2012-03-16 00:45:58 +00:00
|
|
|
(t2MCR p_imm:$cop, imm0_7:$opc1, GPR:$Rt, c_imm:$CRn,
|
2013-06-26 16:52:40 +00:00
|
|
|
c_imm:$CRm, 0, pred:$p)>;
|
2011-07-13 21:35:10 +00:00
|
|
|
def t2MCR2 : t2MovRCopro<0b1111, "mcr2", 0,
|
2011-07-14 21:19:17 +00:00
|
|
|
(outs), (ins p_imm:$cop, imm0_7:$opc1, GPR:$Rt, c_imm:$CRn,
|
|
|
|
c_imm:$CRm, imm0_7:$opc2),
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_mcr2 timm:$cop, timm:$opc1, GPR:$Rt, timm:$CRn,
|
|
|
|
timm:$CRm, timm:$opc2)]> {
|
2013-11-08 16:25:50 +00:00
|
|
|
let Predicates = [IsThumb2, PreV8];
|
|
|
|
}
|
2013-06-26 16:52:40 +00:00
|
|
|
def : t2InstAlias<"mcr2${p} $cop, $opc1, $Rt, $CRn, $CRm",
|
2012-03-16 00:45:58 +00:00
|
|
|
(t2MCR2 p_imm:$cop, imm0_7:$opc1, GPR:$Rt, c_imm:$CRn,
|
2013-06-26 16:52:40 +00:00
|
|
|
c_imm:$CRm, 0, pred:$p)>;
|
2011-07-13 21:35:10 +00:00
|
|
|
|
|
|
|
/* from coprocessor to ARM core register */
|
|
|
|
def t2MRC : t2MovRCopro<0b1110, "mrc", 1,
|
2013-08-06 15:52:36 +00:00
|
|
|
(outs GPRwithAPSR:$Rt), (ins p_imm:$cop, imm0_7:$opc1, c_imm:$CRn,
|
2011-07-19 20:35:35 +00:00
|
|
|
c_imm:$CRm, imm0_7:$opc2), []>;
|
2013-06-26 16:52:40 +00:00
|
|
|
def : t2InstAlias<"mrc${p} $cop, $opc1, $Rt, $CRn, $CRm",
|
2013-08-06 15:52:36 +00:00
|
|
|
(t2MRC GPRwithAPSR:$Rt, p_imm:$cop, imm0_7:$opc1, c_imm:$CRn,
|
2013-06-26 16:52:40 +00:00
|
|
|
c_imm:$CRm, 0, pred:$p)>;
|
2011-07-13 21:35:10 +00:00
|
|
|
|
|
|
|
def t2MRC2 : t2MovRCopro<0b1111, "mrc2", 1,
|
2013-08-06 15:52:36 +00:00
|
|
|
(outs GPRwithAPSR:$Rt), (ins p_imm:$cop, imm0_7:$opc1, c_imm:$CRn,
|
2013-11-08 16:25:50 +00:00
|
|
|
c_imm:$CRm, imm0_7:$opc2), []> {
|
|
|
|
let Predicates = [IsThumb2, PreV8];
|
|
|
|
}
|
2013-06-26 16:52:40 +00:00
|
|
|
def : t2InstAlias<"mrc2${p} $cop, $opc1, $Rt, $CRn, $CRm",
|
2013-08-06 15:52:36 +00:00
|
|
|
(t2MRC2 GPRwithAPSR:$Rt, p_imm:$cop, imm0_7:$opc1, c_imm:$CRn,
|
2013-06-26 16:52:40 +00:00
|
|
|
c_imm:$CRm, 0, pred:$p)>;
|
2011-01-20 16:58:48 +00:00
|
|
|
|
2019-09-19 16:26:14 +00:00
|
|
|
def : T2v6Pat<(int_arm_mrc timm:$cop, timm:$opc1, timm:$CRn, timm:$CRm, timm:$opc2),
|
|
|
|
(t2MRC p_imm:$cop, imm0_7:$opc1, c_imm:$CRn, c_imm:$CRm, imm0_7:$opc2)>;
|
2011-01-20 16:58:48 +00:00
|
|
|
|
2019-09-19 16:26:14 +00:00
|
|
|
def : T2v6Pat<(int_arm_mrc2 timm:$cop, timm:$opc1, timm:$CRn, timm:$CRm, timm:$opc2),
|
|
|
|
(t2MRC2 p_imm:$cop, imm0_7:$opc1, c_imm:$CRn, c_imm:$CRm, imm0_7:$opc2)>;
|
2011-01-20 16:58:48 +00:00
|
|
|
|
|
|
|
|
2011-07-13 21:35:10 +00:00
|
|
|
/* from ARM core register to coprocessor */
|
2015-06-01 12:02:47 +00:00
|
|
|
def t2MCRR : t2MovRRCopro<0b1110, "mcrr", 0, (outs),
|
|
|
|
(ins p_imm:$cop, imm0_15:$opc1, GPR:$Rt, GPR:$Rt2,
|
|
|
|
c_imm:$CRm),
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_mcrr timm:$cop, timm:$opc1, GPR:$Rt, GPR:$Rt2,
|
|
|
|
timm:$CRm)]>;
|
2015-06-01 12:02:47 +00:00
|
|
|
def t2MCRR2 : t2MovRRCopro<0b1111, "mcrr2", 0, (outs),
|
|
|
|
(ins p_imm:$cop, imm0_15:$opc1, GPR:$Rt, GPR:$Rt2,
|
|
|
|
c_imm:$CRm),
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_mcrr2 timm:$cop, timm:$opc1, GPR:$Rt,
|
|
|
|
GPR:$Rt2, timm:$CRm)]> {
|
2013-11-08 16:25:50 +00:00
|
|
|
let Predicates = [IsThumb2, PreV8];
|
|
|
|
}
|
|
|
|
|
2011-07-13 21:35:10 +00:00
|
|
|
/* from coprocessor to ARM core register */
|
2015-06-01 12:02:47 +00:00
|
|
|
def t2MRRC : t2MovRRCopro<0b1110, "mrrc", 1, (outs GPR:$Rt, GPR:$Rt2),
|
|
|
|
(ins p_imm:$cop, imm0_15:$opc1, c_imm:$CRm)>;
|
2011-07-13 21:35:10 +00:00
|
|
|
|
2015-06-01 12:02:47 +00:00
|
|
|
def t2MRRC2 : t2MovRRCopro<0b1111, "mrrc2", 1, (outs GPR:$Rt, GPR:$Rt2),
|
|
|
|
(ins p_imm:$cop, imm0_15:$opc1, c_imm:$CRm)> {
|
2013-11-08 16:25:50 +00:00
|
|
|
let Predicates = [IsThumb2, PreV8];
|
|
|
|
}
|
2011-01-20 16:58:48 +00:00
|
|
|
|
2011-01-20 18:32:09 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2011-07-13 21:14:23 +00:00
|
|
|
// Other Coprocessor Instructions.
|
2011-01-20 18:32:09 +00:00
|
|
|
//
|
|
|
|
|
2013-08-08 10:20:41 +00:00
|
|
|
def t2CDP : T2Cop<0b1110, (outs), (ins p_imm:$cop, imm0_15:$opc1,
|
2011-07-13 22:01:08 +00:00
|
|
|
c_imm:$CRd, c_imm:$CRn, c_imm:$CRm, imm0_7:$opc2),
|
2013-06-26 16:52:40 +00:00
|
|
|
"cdp", "\t$cop, $opc1, $CRd, $CRn, $CRm, $opc2",
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_cdp timm:$cop, timm:$opc1, timm:$CRd, timm:$CRn,
|
|
|
|
timm:$CRm, timm:$opc2)]> {
|
2011-07-13 21:14:23 +00:00
|
|
|
let Inst{27-24} = 0b1110;
|
|
|
|
|
|
|
|
bits<4> opc1;
|
|
|
|
bits<4> CRn;
|
|
|
|
bits<4> CRd;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<3> opc2;
|
|
|
|
bits<4> CRm;
|
|
|
|
|
|
|
|
let Inst{3-0} = CRm;
|
|
|
|
let Inst{4} = 0;
|
|
|
|
let Inst{7-5} = opc2;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{19-16} = CRn;
|
|
|
|
let Inst{23-20} = opc1;
|
2013-11-08 16:25:50 +00:00
|
|
|
|
|
|
|
let Predicates = [IsThumb2, PreV8];
|
2017-08-11 09:52:30 +00:00
|
|
|
let DecoderNamespace = "Thumb2CoProc";
|
2011-07-13 21:14:23 +00:00
|
|
|
}
|
|
|
|
|
2011-07-13 22:06:11 +00:00
|
|
|
def t2CDP2 : T2Cop<0b1111, (outs), (ins p_imm:$cop, imm0_15:$opc1,
|
2011-07-13 22:01:08 +00:00
|
|
|
c_imm:$CRd, c_imm:$CRn, c_imm:$CRm, imm0_7:$opc2),
|
2013-06-26 16:52:40 +00:00
|
|
|
"cdp2", "\t$cop, $opc1, $CRd, $CRn, $CRm, $opc2",
|
2019-09-19 16:26:14 +00:00
|
|
|
[(int_arm_cdp2 timm:$cop, timm:$opc1, timm:$CRd, timm:$CRn,
|
|
|
|
timm:$CRm, timm:$opc2)]> {
|
2011-01-20 18:32:09 +00:00
|
|
|
let Inst{27-24} = 0b1110;
|
|
|
|
|
|
|
|
bits<4> opc1;
|
|
|
|
bits<4> CRn;
|
|
|
|
bits<4> CRd;
|
|
|
|
bits<4> cop;
|
|
|
|
bits<3> opc2;
|
|
|
|
bits<4> CRm;
|
|
|
|
|
|
|
|
let Inst{3-0} = CRm;
|
|
|
|
let Inst{4} = 0;
|
|
|
|
let Inst{7-5} = opc2;
|
|
|
|
let Inst{11-8} = cop;
|
|
|
|
let Inst{15-12} = CRd;
|
|
|
|
let Inst{19-16} = CRn;
|
|
|
|
let Inst{23-20} = opc1;
|
2013-11-08 16:25:50 +00:00
|
|
|
|
|
|
|
let Predicates = [IsThumb2, PreV8];
|
2017-08-11 09:52:30 +00:00
|
|
|
let DecoderNamespace = "Thumb2CoProc";
|
2011-01-20 18:32:09 +00:00
|
|
|
}
|
2011-07-27 16:47:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2015-04-16 11:34:25 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ARMv8.1 Privilege Access Never extension
|
|
|
|
//
|
|
|
|
// SETPAN #imm1
|
|
|
|
|
|
|
|
def t2SETPAN : T1I<(outs), (ins imm0_1:$imm), NoItinerary, "setpan\t$imm", []>,
|
|
|
|
T1Misc<0b0110000>, Requires<[IsThumb2, HasV8, HasV8_1a]> {
|
|
|
|
bits<1> imm;
|
|
|
|
|
|
|
|
let Inst{4} = 0b1;
|
|
|
|
let Inst{3} = imm;
|
|
|
|
let Inst{2-0} = 0b000;
|
|
|
|
|
|
|
|
let Unpredictable{4} = 0b1;
|
|
|
|
let Unpredictable{2-0} = 0b111;
|
|
|
|
}
|
|
|
|
|
2016-01-25 11:24:47 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ARMv8-M Security Extensions instructions
|
|
|
|
//
|
|
|
|
|
|
|
|
let hasSideEffects = 1 in
|
|
|
|
def t2SG : T2I<(outs), (ins), NoItinerary, "sg", "", []>,
|
|
|
|
Requires<[Has8MSecExt]> {
|
|
|
|
let Inst = 0xe97fe97f;
|
|
|
|
}
|
|
|
|
|
|
|
|
class T2TT<bits<2> at, string asm, list<dag> pattern>
|
|
|
|
: T2I<(outs rGPR:$Rt), (ins GPRnopc:$Rn), NoItinerary, asm, "\t$Rt, $Rn",
|
|
|
|
pattern> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> Rt;
|
|
|
|
|
|
|
|
let Inst{31-20} = 0b111010000100;
|
|
|
|
let Inst{19-16} = Rn;
|
|
|
|
let Inst{15-12} = 0b1111;
|
|
|
|
let Inst{11-8} = Rt;
|
|
|
|
let Inst{7-6} = at;
|
|
|
|
let Inst{5-0} = 0b000000;
|
|
|
|
|
|
|
|
let Unpredictable{5-0} = 0b111111;
|
|
|
|
}
|
|
|
|
|
2019-11-18 18:03:41 +00:00
|
|
|
def t2TT : T2TT<0b00, "tt",
|
|
|
|
[(set rGPR:$Rt, (int_arm_cmse_tt GPRnopc:$Rn))]>,
|
|
|
|
Requires<[IsThumb, Has8MSecExt]>;
|
|
|
|
def t2TTT : T2TT<0b01, "ttt",
|
|
|
|
[(set rGPR:$Rt, (int_arm_cmse_ttt GPRnopc:$Rn))]>,
|
|
|
|
Requires<[IsThumb, Has8MSecExt]>;
|
|
|
|
def t2TTA : T2TT<0b10, "tta",
|
|
|
|
[(set rGPR:$Rt, (int_arm_cmse_tta GPRnopc:$Rn))]>,
|
|
|
|
Requires<[IsThumb, Has8MSecExt]>;
|
|
|
|
def t2TTAT : T2TT<0b11, "ttat",
|
|
|
|
[(set rGPR:$Rt, (int_arm_cmse_ttat GPRnopc:$Rn))]>,
|
|
|
|
Requires<[IsThumb, Has8MSecExt]>;
|
2016-01-25 11:24:47 +00:00
|
|
|
|
2011-07-27 16:47:19 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Non-Instruction Patterns
|
|
|
|
//
|
|
|
|
|
|
|
|
// SXT/UXT with no rotate
|
2011-07-27 17:48:13 +00:00
|
|
|
let AddedComplexity = 16 in {
|
|
|
|
def : T2Pat<(and rGPR:$Rm, 0x000000FF), (t2UXTB rGPR:$Rm, 0)>,
|
2011-08-08 19:49:37 +00:00
|
|
|
Requires<[IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(and rGPR:$Rm, 0x0000FFFF), (t2UXTH rGPR:$Rm, 0)>,
|
2011-08-08 19:49:37 +00:00
|
|
|
Requires<[IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(and rGPR:$Rm, 0x00FF00FF), (t2UXTB16 rGPR:$Rm, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(add rGPR:$Rn, (and rGPR:$Rm, 0x00FF)),
|
|
|
|
(t2UXTAB rGPR:$Rn, rGPR:$Rm, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(add rGPR:$Rn, (and rGPR:$Rm, 0xFFFF)),
|
|
|
|
(t2UXTAH rGPR:$Rn, rGPR:$Rm, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def : T2Pat<(sext_inreg rGPR:$Src, i8), (t2SXTB rGPR:$Src, 0)>,
|
2011-08-08 19:49:37 +00:00
|
|
|
Requires<[IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(sext_inreg rGPR:$Src, i16), (t2SXTH rGPR:$Src, 0)>,
|
2011-08-08 19:49:37 +00:00
|
|
|
Requires<[IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(add rGPR:$Rn, (sext_inreg rGPR:$Rm, i8)),
|
|
|
|
(t2SXTAB rGPR:$Rn, rGPR:$Rm, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-07-27 17:48:13 +00:00
|
|
|
def : T2Pat<(add rGPR:$Rn, (sext_inreg rGPR:$Rm, i16)),
|
|
|
|
(t2SXTAH rGPR:$Rn, rGPR:$Rm, 0)>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-08-26 02:59:24 +00:00
|
|
|
|
|
|
|
// Atomic load/store patterns
|
2012-08-28 03:11:27 +00:00
|
|
|
def : T2Pat<(atomic_load_8 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRBi12 t2addrmode_imm12:$addr)>;
|
|
|
|
def : T2Pat<(atomic_load_8 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRBi8 t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_load_8 t2addrmode_so_reg:$addr),
|
2012-08-28 03:11:27 +00:00
|
|
|
(t2LDRBs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(atomic_load_16 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRHi12 t2addrmode_imm12:$addr)>;
|
|
|
|
def : T2Pat<(atomic_load_16 t2addrmode_negimm8:$addr),
|
|
|
|
(t2LDRHi8 t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_load_16 t2addrmode_so_reg:$addr),
|
2012-08-28 03:11:27 +00:00
|
|
|
(t2LDRHs t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(atomic_load_32 t2addrmode_imm12:$addr),
|
|
|
|
(t2LDRi12 t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(atomic_load_32 t2addrmode_negimm8:$addr),
|
2012-08-28 03:11:27 +00:00
|
|
|
(t2LDRi8 t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_load_32 t2addrmode_so_reg:$addr),
|
2012-08-28 03:11:27 +00:00
|
|
|
(t2LDRs t2addrmode_so_reg:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_store_8 t2addrmode_imm12:$addr, GPR:$val),
|
|
|
|
(t2STRBi12 GPR:$val, t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(atomic_store_8 t2addrmode_negimm8:$addr, GPR:$val),
|
|
|
|
(t2STRBi8 GPR:$val, t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_store_8 t2addrmode_so_reg:$addr, GPR:$val),
|
|
|
|
(t2STRBs GPR:$val, t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(atomic_store_16 t2addrmode_imm12:$addr, GPR:$val),
|
|
|
|
(t2STRHi12 GPR:$val, t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(atomic_store_16 t2addrmode_negimm8:$addr, GPR:$val),
|
|
|
|
(t2STRHi8 GPR:$val, t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_store_16 t2addrmode_so_reg:$addr, GPR:$val),
|
|
|
|
(t2STRHs GPR:$val, t2addrmode_so_reg:$addr)>;
|
|
|
|
def : T2Pat<(atomic_store_32 t2addrmode_imm12:$addr, GPR:$val),
|
|
|
|
(t2STRi12 GPR:$val, t2addrmode_imm12:$addr)>;
|
2011-09-07 20:58:57 +00:00
|
|
|
def : T2Pat<(atomic_store_32 t2addrmode_negimm8:$addr, GPR:$val),
|
|
|
|
(t2STRi8 GPR:$val, t2addrmode_negimm8:$addr)>;
|
2011-08-26 02:59:24 +00:00
|
|
|
def : T2Pat<(atomic_store_32 t2addrmode_so_reg:$addr, GPR:$val),
|
|
|
|
(t2STRs GPR:$val, t2addrmode_so_reg:$addr)>;
|
2011-08-31 18:23:08 +00:00
|
|
|
|
2018-12-17 15:05:32 +00:00
|
|
|
let AddedComplexity = 8, Predicates = [IsThumb, HasAcquireRelease, HasV7Clrex] in {
|
|
|
|
def : Pat<(atomic_load_acquire_8 addr_offset_none:$addr), (t2LDAB addr_offset_none:$addr)>;
|
|
|
|
def : Pat<(atomic_load_acquire_16 addr_offset_none:$addr), (t2LDAH addr_offset_none:$addr)>;
|
|
|
|
def : Pat<(atomic_load_acquire_32 addr_offset_none:$addr), (t2LDA addr_offset_none:$addr)>;
|
|
|
|
def : Pat<(atomic_store_release_8 addr_offset_none:$addr, GPR:$val), (t2STLB GPR:$val, addr_offset_none:$addr)>;
|
|
|
|
def : Pat<(atomic_store_release_16 addr_offset_none:$addr, GPR:$val), (t2STLH GPR:$val, addr_offset_none:$addr)>;
|
|
|
|
def : Pat<(atomic_store_release_32 addr_offset_none:$addr, GPR:$val), (t2STL GPR:$val, addr_offset_none:$addr)>;
|
2013-09-26 12:22:36 +00:00
|
|
|
}
|
|
|
|
|
2011-08-31 18:23:08 +00:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Assembler aliases
|
|
|
|
//
|
|
|
|
|
|
|
|
// Aliases for ADC without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"adc${s}${p} $Rd, $Rn, $Rm",
|
|
|
|
(t2ADCrr rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"adc${s}${p} $Rd, $Rn, $ShiftedRm",
|
|
|
|
(t2ADCrs rGPR:$Rd, rGPR:$Rn, t2_so_reg:$ShiftedRm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
|
|
|
|
// Aliases for SBC without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"sbc${s}${p} $Rd, $Rn, $Rm",
|
|
|
|
(t2SBCrr rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"sbc${s}${p} $Rd, $Rn, $ShiftedRm",
|
|
|
|
(t2SBCrs rGPR:$Rd, rGPR:$Rn, t2_so_reg:$ShiftedRm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
|
2011-09-02 18:14:46 +00:00
|
|
|
// Aliases for ADD without the ".w" optional width specifier.
|
2011-09-01 00:28:52 +00:00
|
|
|
def : t2InstAlias<"add${s}${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri rGPR:$Rd, GPRnopc:$Rn, t2_so_imm:$imm, pred:$p,
|
2013-08-19 15:02:25 +00:00
|
|
|
cc_out:$s)>;
|
2011-09-01 00:28:52 +00:00
|
|
|
def : t2InstAlias<"add${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri12 rGPR:$Rd, GPR:$Rn, imm0_4095:$imm, pred:$p)>;
|
2011-09-02 18:14:46 +00:00
|
|
|
def : t2InstAlias<"add${s}${p} $Rd, $Rn, $Rm",
|
2011-10-03 20:51:59 +00:00
|
|
|
(t2ADDrr GPRnopc:$Rd, GPRnopc:$Rn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
2011-09-02 18:14:46 +00:00
|
|
|
def : t2InstAlias<"add${s}${p} $Rd, $Rn, $ShiftedRm",
|
2011-10-03 20:51:59 +00:00
|
|
|
(t2ADDrs GPRnopc:$Rd, GPRnopc:$Rn, t2_so_reg:$ShiftedRm,
|
2011-09-16 22:58:42 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
// ... and with the destination and source register combined.
|
|
|
|
def : t2InstAlias<"add${s}${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
def : t2InstAlias<"add${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2ADDri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"addw${p} $Rdn, $imm",
|
|
|
|
(t2ADDri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095:$imm, pred:$p)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
def : t2InstAlias<"add${s}${p} $Rdn, $Rm",
|
|
|
|
(t2ADDrr GPRnopc:$Rdn, GPRnopc:$Rdn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"add${s}${p} $Rdn, $ShiftedRm",
|
|
|
|
(t2ADDrs GPRnopc:$Rdn, GPRnopc:$Rdn, t2_so_reg:$ShiftedRm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
2011-09-16 22:58:42 +00:00
|
|
|
|
2012-04-05 20:57:13 +00:00
|
|
|
// add w/ negative immediates is just a sub.
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rd, GPRnopc:$Rn, t2_so_imm_neg:$imm, pred:$p,
|
2012-04-05 20:57:13 +00:00
|
|
|
cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rd, GPR:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_neg:$imm, pred:$p,
|
2012-04-05 20:57:13 +00:00
|
|
|
cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095_neg:$imm, pred:$p)>;
|
2012-04-05 20:57:13 +00:00
|
|
|
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p}.w $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rd, GPRnopc:$Rn, t2_so_imm_neg:$imm, pred:$p,
|
2012-05-01 21:17:34 +00:00
|
|
|
cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"addw${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rd, rGPR:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p}.w $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_neg:$imm, pred:$p,
|
2012-05-01 21:17:34 +00:00
|
|
|
cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"addw${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095_neg:$imm, pred:$p)>;
|
2012-05-01 21:17:34 +00:00
|
|
|
|
2012-04-05 20:57:13 +00:00
|
|
|
|
2011-09-16 22:58:42 +00:00
|
|
|
// Aliases for SUB without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"sub${s}${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rd, GPRnopc:$Rn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
2011-09-16 22:58:42 +00:00
|
|
|
def : t2InstAlias<"sub${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rd, GPR:$Rn, imm0_4095:$imm, pred:$p)>;
|
2011-09-16 22:58:42 +00:00
|
|
|
def : t2InstAlias<"sub${s}${p} $Rd, $Rn, $Rm",
|
2011-10-03 20:51:59 +00:00
|
|
|
(t2SUBrr GPRnopc:$Rd, GPRnopc:$Rn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
2011-09-16 22:58:42 +00:00
|
|
|
def : t2InstAlias<"sub${s}${p} $Rd, $Rn, $ShiftedRm",
|
2011-10-03 20:51:59 +00:00
|
|
|
(t2SUBrs GPRnopc:$Rd, GPRnopc:$Rn, t2_so_reg:$ShiftedRm,
|
2011-09-02 18:14:46 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
// ... and with the destination and source register combined.
|
|
|
|
def : t2InstAlias<"sub${s}${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
def : t2InstAlias<"sub${p} $Rdn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"subw${p} $Rdn, $imm",
|
|
|
|
(t2SUBri12 rGPR:$Rdn, rGPR:$Rdn, imm0_4095:$imm, pred:$p)>;
|
2012-04-10 17:31:55 +00:00
|
|
|
def : t2InstAlias<"sub${s}${p}.w $Rdn, $Rm",
|
|
|
|
(t2SUBrr GPRnopc:$Rdn, GPRnopc:$Rdn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
2011-10-28 16:57:07 +00:00
|
|
|
def : t2InstAlias<"sub${s}${p} $Rdn, $Rm",
|
|
|
|
(t2SUBrr GPRnopc:$Rdn, GPRnopc:$Rdn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"sub${s}${p} $Rdn, $ShiftedRm",
|
|
|
|
(t2SUBrs GPRnopc:$Rdn, GPRnopc:$Rdn, t2_so_reg:$ShiftedRm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
// SP to SP alike aliases
|
|
|
|
// Aliases for ADD without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"add${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2ADDspImm GPRsp:$Rd, GPRsp:$Rn, t2_so_imm:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"add${p} $Rd, $Rn, $imm",
|
|
|
|
(t2ADDspImm12 GPRsp:$Rd, GPRsp:$Rn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
// ... and with the destination and source register combined.
|
|
|
|
def : t2InstAlias<"add${s}${p} $Rdn, $imm",
|
|
|
|
(t2ADDspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"add${s}${p}.w $Rdn, $imm",
|
|
|
|
(t2ADDspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"add${p} $Rdn, $imm",
|
|
|
|
(t2ADDspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"addw${p} $Rdn, $imm",
|
|
|
|
(t2ADDspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
// add w/ negative immediates is just a sub.
|
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rd, GPRsp:$Rn, t2_so_imm_neg:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"add${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rd, GPRsp:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
def : t2InstSubst<"add${s}${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm_neg:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"add${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
def : t2InstSubst<"add${s}${p}.w $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rd, GPRsp:$Rn, t2_so_imm_neg:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"addw${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rd, GPRsp:$Rn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
def : t2InstSubst<"add${s}${p}.w $Rdn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm_neg:$imm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"addw${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095_neg:$imm, pred:$p)>;
|
|
|
|
|
|
|
|
|
|
|
|
// Aliases for SUB without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"sub${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rd, GPRsp:$Rn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"sub${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rd, GPRsp:$Rn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
// ... and with the destination and source register combined.
|
|
|
|
def : t2InstAlias<"sub${s}${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"sub${s}${p}.w $Rdn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rdn, GPRsp:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"sub${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"subw${p} $Rdn, $imm",
|
|
|
|
(t2SUBspImm12 GPRsp:$Rdn, GPRsp:$Rdn, imm0_4095:$imm, pred:$p)>;
|
|
|
|
|
2011-09-06 21:44:58 +00:00
|
|
|
// Alias for compares without the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"cmn${p} $Rn, $Rm",
|
|
|
|
(t2CMNzrr GPRnopc:$Rn, rGPR:$Rm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"teq${p} $Rn, $Rm",
|
2019-05-08 10:59:08 +00:00
|
|
|
(t2TEQrr rGPR:$Rn, rGPR:$Rm, pred:$p)>;
|
2011-09-06 21:44:58 +00:00
|
|
|
def : t2InstAlias<"tst${p} $Rn, $Rm",
|
2019-05-08 10:59:08 +00:00
|
|
|
(t2TSTrr rGPR:$Rn, rGPR:$Rm, pred:$p)>;
|
2011-09-06 21:44:58 +00:00
|
|
|
|
2011-09-06 22:14:58 +00:00
|
|
|
// Memory barriers
|
2020-02-28 11:03:14 +00:00
|
|
|
def : InstAlias<"dmb${p}.w\t$opt", (t2DMB memb_opt:$opt, pred:$p), 0>, Requires<[HasDB]>;
|
2016-06-03 13:19:43 +00:00
|
|
|
def : InstAlias<"dmb${p}", (t2DMB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
2020-02-28 11:03:14 +00:00
|
|
|
def : InstAlias<"dmb${p}.w", (t2DMB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
|
|
|
def : InstAlias<"dsb${p}.w\t$opt", (t2DSB memb_opt:$opt, pred:$p), 0>, Requires<[HasDB]>;
|
2016-06-03 13:19:43 +00:00
|
|
|
def : InstAlias<"dsb${p}", (t2DSB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
2020-02-28 11:03:14 +00:00
|
|
|
def : InstAlias<"dsb${p}.w", (t2DSB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
|
|
|
def : InstAlias<"isb${p}.w\t$opt", (t2ISB memb_opt:$opt, pred:$p), 0>, Requires<[HasDB]>;
|
2016-06-03 13:19:43 +00:00
|
|
|
def : InstAlias<"isb${p}", (t2ISB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
2020-02-28 11:03:14 +00:00
|
|
|
def : InstAlias<"isb${p}.w", (t2ISB 0xf, pred:$p), 0>, Requires<[HasDB]>;
|
2018-09-28 08:27:56 +00:00
|
|
|
|
|
|
|
// Non-predicable aliases of a predicable DSB: the predicate is (14, 0) where
|
|
|
|
// 14 = AL (always execute) and 0 = "instruction doesn't read the CPSR".
|
|
|
|
def : InstAlias<"ssbb", (t2DSB 0x0, 14, 0), 1>, Requires<[HasDB, IsThumb2]>;
|
|
|
|
def : InstAlias<"pssbb", (t2DSB 0x4, 14, 0), 1>, Requires<[HasDB, IsThumb2]>;
|
|
|
|
|
2017-12-21 11:17:49 +00:00
|
|
|
// Armv8-R 'Data Full Barrier'
|
|
|
|
def : InstAlias<"dfb${p}", (t2DSB 0xc, pred:$p), 1>, Requires<[HasDFB]>;
|
2011-09-07 20:58:57 +00:00
|
|
|
|
2011-09-09 19:42:40 +00:00
|
|
|
// Alias for LDR, LDRB, LDRH, LDRSB, and LDRSH without the ".w" optional
|
|
|
|
// width specifier.
|
2011-09-07 21:41:25 +00:00
|
|
|
def : t2InstAlias<"ldr${p} $Rt, $addr",
|
|
|
|
(t2LDRi12 GPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrb${p} $Rt, $addr",
|
|
|
|
(t2LDRBi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrh${p} $Rt, $addr",
|
|
|
|
(t2LDRHi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
2011-09-09 19:42:40 +00:00
|
|
|
def : t2InstAlias<"ldrsb${p} $Rt, $addr",
|
|
|
|
(t2LDRSBi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsh${p} $Rt, $addr",
|
|
|
|
(t2LDRSHi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
|
2011-09-07 23:10:15 +00:00
|
|
|
def : t2InstAlias<"ldr${p} $Rt, $addr",
|
|
|
|
(t2LDRs GPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrb${p} $Rt, $addr",
|
|
|
|
(t2LDRBs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrh${p} $Rt, $addr",
|
|
|
|
(t2LDRHs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
2011-09-09 19:42:40 +00:00
|
|
|
def : t2InstAlias<"ldrsb${p} $Rt, $addr",
|
|
|
|
(t2LDRSBs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsh${p} $Rt, $addr",
|
|
|
|
(t2LDRSHs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
2011-09-14 21:24:41 +00:00
|
|
|
|
2011-10-26 22:22:01 +00:00
|
|
|
def : t2InstAlias<"ldr${p} $Rt, $addr",
|
2016-11-10 13:20:41 +00:00
|
|
|
(t2LDRpci GPR:$Rt, t2ldrlabel:$addr, pred:$p)>;
|
2011-10-26 22:22:01 +00:00
|
|
|
def : t2InstAlias<"ldrb${p} $Rt, $addr",
|
|
|
|
(t2LDRBpci rGPR:$Rt, t2ldrlabel:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrh${p} $Rt, $addr",
|
|
|
|
(t2LDRHpci rGPR:$Rt, t2ldrlabel:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsb${p} $Rt, $addr",
|
|
|
|
(t2LDRSBpci rGPR:$Rt, t2ldrlabel:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsh${p} $Rt, $addr",
|
|
|
|
(t2LDRSHpci rGPR:$Rt, t2ldrlabel:$addr, pred:$p)>;
|
|
|
|
|
2011-10-27 17:16:55 +00:00
|
|
|
// Alias for MVN with(out) the ".w" optional width specifier.
|
|
|
|
def : t2InstAlias<"mvn${s}${p}.w $Rd, $imm",
|
|
|
|
(t2MVNi rGPR:$Rd, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
2011-09-14 21:24:41 +00:00
|
|
|
def : t2InstAlias<"mvn${s}${p} $Rd, $Rm",
|
|
|
|
(t2MVNr rGPR:$Rd, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"mvn${s}${p} $Rd, $ShiftedRm",
|
|
|
|
(t2MVNs rGPR:$Rd, t2_so_reg:$ShiftedRm, pred:$p, cc_out:$s)>;
|
2011-09-14 23:16:41 +00:00
|
|
|
|
2016-01-18 11:56:35 +00:00
|
|
|
// PKHBT/PKHTB with default shift amount. PKHTB is equivalent to PKHBT with the
|
|
|
|
// input operands swapped when the shift amount is zero (i.e., unspecified).
|
2011-09-14 23:16:41 +00:00
|
|
|
def : InstAlias<"pkhbt${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2PKHBT rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-09-14 23:16:41 +00:00
|
|
|
def : InstAlias<"pkhtb${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2PKHBT rGPR:$Rd, rGPR:$Rm, rGPR:$Rn, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-09-14 23:16:41 +00:00
|
|
|
|
2011-09-15 15:55:04 +00:00
|
|
|
// PUSH/POP aliases for STM/LDM
|
|
|
|
def : t2InstAlias<"push${p}.w $regs", (t2STMDB_UPD SP, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"push${p} $regs", (t2STMDB_UPD SP, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"pop${p}.w $regs", (t2LDMIA_UPD SP, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"pop${p} $regs", (t2LDMIA_UPD SP, pred:$p, reglist:$regs)>;
|
|
|
|
|
2011-12-07 18:32:28 +00:00
|
|
|
// STMIA/STMIA_UPD aliases w/o the optional .w suffix
|
|
|
|
def : t2InstAlias<"stm${p} $Rn, $regs",
|
|
|
|
(t2STMIA GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"stm${p} $Rn!, $regs",
|
|
|
|
(t2STMIA_UPD GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
|
|
|
|
// LDMIA/LDMIA_UPD aliases w/o the optional .w suffix
|
|
|
|
def : t2InstAlias<"ldm${p} $Rn, $regs",
|
|
|
|
(t2LDMIA GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"ldm${p} $Rn!, $regs",
|
|
|
|
(t2LDMIA_UPD GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
|
2011-11-09 23:44:23 +00:00
|
|
|
// STMDB/STMDB_UPD aliases w/ the optional .w suffix
|
|
|
|
def : t2InstAlias<"stmdb${p}.w $Rn, $regs",
|
|
|
|
(t2STMDB GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"stmdb${p}.w $Rn!, $regs",
|
|
|
|
(t2STMDB_UPD GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
|
2011-10-27 17:33:59 +00:00
|
|
|
// LDMDB/LDMDB_UPD aliases w/ the optional .w suffix
|
|
|
|
def : t2InstAlias<"ldmdb${p}.w $Rn, $regs",
|
|
|
|
(t2LDMDB GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
def : t2InstAlias<"ldmdb${p}.w $Rn!, $regs",
|
|
|
|
(t2LDMDB_UPD GPR:$Rn, pred:$p, reglist:$regs)>;
|
|
|
|
|
2011-09-15 19:46:13 +00:00
|
|
|
// Alias for REV/REV16/REVSH without the ".w" optional width specifier.
|
2011-09-15 18:13:30 +00:00
|
|
|
def : t2InstAlias<"rev${p} $Rd, $Rm", (t2REV rGPR:$Rd, rGPR:$Rm, pred:$p)>;
|
2011-09-15 19:46:13 +00:00
|
|
|
def : t2InstAlias<"rev16${p} $Rd, $Rm", (t2REV16 rGPR:$Rd, rGPR:$Rm, pred:$p)>;
|
|
|
|
def : t2InstAlias<"revsh${p} $Rd, $Rm", (t2REVSH rGPR:$Rd, rGPR:$Rm, pred:$p)>;
|
2011-09-15 20:54:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Alias for RSB without the ".w" optional width specifier, and with optional
|
|
|
|
// implied destination register.
|
|
|
|
def : t2InstAlias<"rsb${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2RSBri rGPR:$Rd, rGPR:$Rn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"rsb${s}${p} $Rdn, $imm",
|
|
|
|
(t2RSBri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm:$imm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"rsb${s}${p} $Rdn, $Rm",
|
|
|
|
(t2RSBrr rGPR:$Rdn, rGPR:$Rdn, rGPR:$Rm, pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstAlias<"rsb${s}${p} $Rdn, $ShiftedRm",
|
|
|
|
(t2RSBrs rGPR:$Rdn, rGPR:$Rdn, t2_so_reg:$ShiftedRm, pred:$p,
|
|
|
|
cc_out:$s)>;
|
2011-09-16 18:32:30 +00:00
|
|
|
|
|
|
|
// SSAT/USAT optional shift operand.
|
|
|
|
def : t2InstAlias<"ssat${p} $Rd, $sat_imm, $Rn",
|
|
|
|
(t2SSAT rGPR:$Rd, imm1_32:$sat_imm, rGPR:$Rn, 0, pred:$p)>;
|
|
|
|
def : t2InstAlias<"usat${p} $Rd, $sat_imm, $Rn",
|
|
|
|
(t2USAT rGPR:$Rd, imm0_31:$sat_imm, rGPR:$Rn, 0, pred:$p)>;
|
|
|
|
|
2011-09-16 20:50:13 +00:00
|
|
|
// STM w/o the .w suffix.
|
|
|
|
def : t2InstAlias<"stm${p} $Rn, $regs",
|
|
|
|
(t2STMIA GPR:$Rn, pred:$p, reglist:$regs)>;
|
2011-09-16 21:06:12 +00:00
|
|
|
|
|
|
|
// Alias for STR, STRB, and STRH without the ".w" optional
|
|
|
|
// width specifier.
|
|
|
|
def : t2InstAlias<"str${p} $Rt, $addr",
|
|
|
|
(t2STRi12 GPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"strb${p} $Rt, $addr",
|
|
|
|
(t2STRBi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"strh${p} $Rt, $addr",
|
|
|
|
(t2STRHi12 rGPR:$Rt, t2addrmode_imm12:$addr, pred:$p)>;
|
|
|
|
|
|
|
|
def : t2InstAlias<"str${p} $Rt, $addr",
|
|
|
|
(t2STRs GPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"strb${p} $Rt, $addr",
|
|
|
|
(t2STRBs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"strh${p} $Rt, $addr",
|
|
|
|
(t2STRHs rGPR:$Rt, t2addrmode_so_reg:$addr, pred:$p)>;
|
2011-09-19 17:56:37 +00:00
|
|
|
|
|
|
|
// Extend instruction optional rotate operand.
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"sxtab${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2SXTAB rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"sxtah${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2SXTAH rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"sxtab16${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2SXTAB16 rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"sxtb16${p} $Rd, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2SXTB16 rGPR:$Rd, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-09-27 22:18:54 +00:00
|
|
|
|
2011-09-19 20:29:33 +00:00
|
|
|
def : t2InstAlias<"sxtb${p} $Rd, $Rm",
|
|
|
|
(t2SXTB rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
|
|
|
def : t2InstAlias<"sxth${p} $Rd, $Rm",
|
|
|
|
(t2SXTH rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
2011-09-27 22:18:54 +00:00
|
|
|
def : t2InstAlias<"sxtb${p}.w $Rd, $Rm",
|
|
|
|
(t2SXTB rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
|
|
|
def : t2InstAlias<"sxth${p}.w $Rd, $Rm",
|
|
|
|
(t2SXTH rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
2011-09-19 20:29:33 +00:00
|
|
|
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"uxtab${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2UXTAB rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"uxtah${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2UXTAH rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"uxtab16${p} $Rd, $Rn, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2UXTAB16 rGPR:$Rd, rGPR:$Rn, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"uxtb16${p} $Rd, $Rm",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2UXTB16 rGPR:$Rd, rGPR:$Rm, 0, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2015-01-19 16:36:02 +00:00
|
|
|
|
2011-09-20 00:46:54 +00:00
|
|
|
def : t2InstAlias<"uxtb${p} $Rd, $Rm",
|
|
|
|
(t2UXTB rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
|
|
|
def : t2InstAlias<"uxth${p} $Rd, $Rm",
|
|
|
|
(t2UXTH rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
2011-09-27 22:18:54 +00:00
|
|
|
def : t2InstAlias<"uxtb${p}.w $Rd, $Rm",
|
|
|
|
(t2UXTB rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
|
|
|
def : t2InstAlias<"uxth${p}.w $Rd, $Rm",
|
|
|
|
(t2UXTH rGPR:$Rd, rGPR:$Rm, 0, pred:$p)>;
|
|
|
|
|
2011-09-19 20:29:33 +00:00
|
|
|
// Extend instruction w/o the ".w" optional width specifier.
|
2011-09-20 00:46:54 +00:00
|
|
|
def : t2InstAlias<"uxtb${p} $Rd, $Rm$rot",
|
|
|
|
(t2UXTB rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p)>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"uxtb16${p} $Rd, $Rm$rot",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2UXTB16 rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-09-20 00:46:54 +00:00
|
|
|
def : t2InstAlias<"uxth${p} $Rd, $Rm$rot",
|
|
|
|
(t2UXTH rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p)>;
|
|
|
|
|
2011-09-19 20:29:33 +00:00
|
|
|
def : t2InstAlias<"sxtb${p} $Rd, $Rm$rot",
|
|
|
|
(t2SXTB rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p)>;
|
2015-01-19 16:36:02 +00:00
|
|
|
def : InstAlias<"sxtb16${p} $Rd, $Rm$rot",
|
2016-06-03 13:19:43 +00:00
|
|
|
(t2SXTB16 rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p), 0>,
|
2017-02-17 15:42:44 +00:00
|
|
|
Requires<[HasDSP, IsThumb2]>;
|
2011-09-19 20:29:33 +00:00
|
|
|
def : t2InstAlias<"sxth${p} $Rd, $Rm$rot",
|
|
|
|
(t2SXTH rGPR:$Rd, rGPR:$Rm, rot_imm:$rot, pred:$p)>;
|
2011-10-28 22:36:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
// "mov Rd, t2_so_imm_not" can be handled via "mvn" in assembly, just like
|
|
|
|
// for isel.
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"mov${p} $Rd, $imm",
|
2011-10-28 22:36:30 +00:00
|
|
|
(t2MVNi rGPR:$Rd, t2_so_imm_not:$imm, pred:$p, zero_reg)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"mvn${s}${p} $Rd, $imm",
|
|
|
|
(t2MOVi rGPR:$Rd, t2_so_imm_not:$imm, pred:$p, s_cc_out:$s)>;
|
2011-12-09 22:02:17 +00:00
|
|
|
// Same for AND <--> BIC
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"bic${s}${p} $Rd, $Rn, $imm",
|
2013-08-16 11:55:44 +00:00
|
|
|
(t2ANDri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
2011-12-09 22:02:17 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"bic${s}${p} $Rdn, $imm",
|
2013-08-16 11:55:44 +00:00
|
|
|
(t2ANDri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
2011-12-09 22:02:17 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
2018-07-04 16:11:15 +00:00
|
|
|
def : t2InstSubst<"bic${s}${p}.w $Rd, $Rn, $imm",
|
|
|
|
(t2ANDri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"bic${s}${p}.w $Rdn, $imm",
|
|
|
|
(t2ANDri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"and${s}${p} $Rd, $Rn, $imm",
|
2013-08-16 11:55:44 +00:00
|
|
|
(t2BICri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
2011-12-09 22:02:17 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"and${s}${p} $Rdn, $imm",
|
2013-08-16 11:55:44 +00:00
|
|
|
(t2BICri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
2018-07-04 16:11:15 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"and${s}${p}.w $Rd, $Rn, $imm",
|
|
|
|
(t2BICri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"and${s}${p}.w $Rdn, $imm",
|
|
|
|
(t2BICri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
2011-12-09 22:02:17 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
2017-05-05 11:31:25 +00:00
|
|
|
// And ORR <--> ORN
|
|
|
|
def : t2InstSubst<"orn${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2ORRri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"orn${s}${p} $Rdn, $imm",
|
|
|
|
(t2ORRri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"orr${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2ORNri rGPR:$Rd, rGPR:$Rn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"orr${s}${p} $Rdn, $imm",
|
|
|
|
(t2ORNri rGPR:$Rdn, rGPR:$Rdn, t2_so_imm_not:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
2011-12-14 17:30:24 +00:00
|
|
|
// Likewise, "add Rd, t2_so_imm_neg" -> sub
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $Rn, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBri rGPR:$Rd, GPRnopc:$Rn, t2_so_imm_neg:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $Rn, $imm",
|
|
|
|
(t2SUBspImm GPRsp:$Rd, GPRsp:$Rn, t2_so_imm_neg:$imm,
|
|
|
|
pred:$p, cc_out:$s)>;
|
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $imm",
|
|
|
|
(t2SUBri rGPR:$Rd, rGPR:$Rd, t2_so_imm_neg:$imm,
|
2011-12-08 00:31:07 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"add${s}${p} $Rd, $imm",
|
[ARM][Thumb2] Fix ADD/SUB invalid writes to SP
Summary:
This patch fixes pr23772 [ARM] r226200 can emit illegal thumb2 instruction: "sub sp, r12, #80".
The violation was that SUB and ADD (reg, immediate) instructions can only write to SP if the source register is also SP. So the above instructions was unpredictable.
To enforce that the instruction t2(ADD|SUB)ri does not write to SP we now enforce the destination register to be rGPR (That exclude PC and SP).
Different than the ARM specification, that defines one instruction that can read from SP, and one that can't, here we inserted one that can't write to SP, and other that can only write to SP as to reuse most of the hard-coded size optimizations.
When performing this change, it uncovered that emitting Thumb2 Reg plus Immediate could not emit all variants of ADD SP, SP #imm instructions before so it was refactored to be able to. (see test/CodeGen/Thumb2/mve-stacksplot.mir where we use a subw sp, sp, Imm12 variant )
It also uncovered a disassembly issue of adr.w instructions, that were only written as SUBW instructions (see llvm/test/MC/Disassembler/ARM/thumb2.txt).
Reviewers: eli.friedman, dmgreen, carwil, olista01, efriedma, andreadb
Reviewed By: efriedma
Subscribers: gbedwell, john.brawn, efriedma, ostannard, kristof.beyls, hiraditya, dmgreen, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D70680
2020-01-13 11:36:02 +00:00
|
|
|
(t2SUBspImm GPRsp:$Rd, GPRsp:$Rd, t2_so_imm_neg:$imm,
|
2011-12-08 00:31:07 +00:00
|
|
|
pred:$p, cc_out:$s)>;
|
2011-12-14 17:30:24 +00:00
|
|
|
// Same for CMP <--> CMN via t2_so_imm_neg
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"cmp${p} $Rd, $imm",
|
2012-06-11 08:07:26 +00:00
|
|
|
(t2CMNri rGPR:$Rd, t2_so_imm_neg:$imm, pred:$p)>;
|
[ARM] [Assembler] Support negative immediates for A32, T32 and T16
Summary:
To support negative immediates for certain arithmetic instructions, the
instruction is converted to the inverse instruction with a negated (or inverted)
immediate. For example, "ADD r0, r1, #FFFFFFFF" cannot be encoded as an ADD
instruction. However, "SUB r0, r1, #1" is equivalent.
These conversions are different from instruction aliases. An alias maps
several assembler instructions onto one encoding. A conversion, however, maps
an *invalid* instruction--e.g. with an immediate that cannot be represented in
the encoding--to a different (but equivalent) instruction.
Several instructions with negative immediates were being converted already, but
this was not systematically tested, nor did it cover all instructions.
This patch implements all possible substitutions for ARM, Thumb1 and
Thumb2 assembler and adds tests. It also adds a feature flag
(-mattr=+no-neg-immediates) to turn these substitutions off. This is
helpful for users who want their code to assemble to exactly what they
wrote.
Reviewers: t.p.northover, rovka, samparker, javed.absar, peter.smith, rengolin
Reviewed By: javed.absar
Subscribers: aadg, aemerson, llvm-commits
Differential Revision: https://reviews.llvm.org/D30571
llvm-svn: 298380
2017-03-21 14:59:17 +00:00
|
|
|
def : t2InstSubst<"cmn${p} $Rd, $imm",
|
2011-12-14 17:30:24 +00:00
|
|
|
(t2CMPri rGPR:$Rd, t2_so_imm_neg:$imm, pred:$p)>;
|
2011-11-15 19:55:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Wide 'mul' encoding can be specified with only two operands.
|
|
|
|
def : t2InstAlias<"mul${p} $Rn, $Rm",
|
2011-12-06 05:03:45 +00:00
|
|
|
(t2MUL rGPR:$Rn, rGPR:$Rm, rGPR:$Rn, pred:$p)>;
|
2011-12-13 20:23:22 +00:00
|
|
|
|
|
|
|
// "neg" is and alias for "rsb rd, rn, #0"
|
|
|
|
def : t2InstAlias<"neg${s}${p} $Rd, $Rm",
|
|
|
|
(t2RSBri rGPR:$Rd, rGPR:$Rm, 0, pred:$p, cc_out:$s)>;
|
2011-12-13 22:45:11 +00:00
|
|
|
|
|
|
|
// MOV so_reg assembler pseudos. InstAlias isn't expressive enough for
|
|
|
|
// these, unfortunately.
|
2017-02-27 14:40:51 +00:00
|
|
|
// FIXME: LSL #0 in the shift should allow SP to be used as either the
|
|
|
|
// source or destination (but not both).
|
2011-12-13 22:45:11 +00:00
|
|
|
def t2MOVsi: t2AsmPseudo<"mov${p} $Rd, $shift",
|
|
|
|
(ins rGPR:$Rd, t2_so_reg:$shift, pred:$p)>;
|
|
|
|
def t2MOVSsi: t2AsmPseudo<"movs${p} $Rd, $shift",
|
|
|
|
(ins rGPR:$Rd, t2_so_reg:$shift, pred:$p)>;
|
2011-12-15 23:52:17 +00:00
|
|
|
|
2011-12-21 20:54:00 +00:00
|
|
|
def t2MOVsr: t2AsmPseudo<"mov${p} $Rd, $shift",
|
|
|
|
(ins rGPR:$Rd, so_reg_reg:$shift, pred:$p)>;
|
|
|
|
def t2MOVSsr: t2AsmPseudo<"movs${p} $Rd, $shift",
|
|
|
|
(ins rGPR:$Rd, so_reg_reg:$shift, pred:$p)>;
|
|
|
|
|
2017-06-22 10:30:53 +00:00
|
|
|
// Aliases for the above with the .w qualifier
|
|
|
|
def : t2InstAlias<"mov${p}.w $Rd, $shift",
|
|
|
|
(t2MOVsi rGPR:$Rd, t2_so_reg:$shift, pred:$p)>;
|
|
|
|
def : t2InstAlias<"movs${p}.w $Rd, $shift",
|
|
|
|
(t2MOVSsi rGPR:$Rd, t2_so_reg:$shift, pred:$p)>;
|
|
|
|
def : t2InstAlias<"mov${p}.w $Rd, $shift",
|
|
|
|
(t2MOVsr rGPR:$Rd, so_reg_reg:$shift, pred:$p)>;
|
|
|
|
def : t2InstAlias<"movs${p}.w $Rd, $shift",
|
|
|
|
(t2MOVSsr rGPR:$Rd, so_reg_reg:$shift, pred:$p)>;
|
|
|
|
|
2011-12-15 23:52:17 +00:00
|
|
|
// ADR w/o the .w suffix
|
|
|
|
def : t2InstAlias<"adr${p} $Rd, $addr",
|
|
|
|
(t2ADR rGPR:$Rd, t2adrlabel:$addr, pred:$p)>;
|
2012-01-18 22:46:46 +00:00
|
|
|
|
|
|
|
// LDR(literal) w/ alternate [pc, #imm] syntax.
|
|
|
|
def t2LDRpcrel : t2AsmPseudo<"ldr${p} $Rt, $addr",
|
2013-08-15 15:43:06 +00:00
|
|
|
(ins GPR:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
2012-01-18 22:46:46 +00:00
|
|
|
def t2LDRBpcrel : t2AsmPseudo<"ldrb${p} $Rt, $addr",
|
|
|
|
(ins GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def t2LDRHpcrel : t2AsmPseudo<"ldrh${p} $Rt, $addr",
|
|
|
|
(ins GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def t2LDRSBpcrel : t2AsmPseudo<"ldrsb${p} $Rt, $addr",
|
|
|
|
(ins GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def t2LDRSHpcrel : t2AsmPseudo<"ldrsh${p} $Rt, $addr",
|
|
|
|
(ins GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
// Version w/ the .w suffix.
|
|
|
|
def : t2InstAlias<"ldr${p}.w $Rt, $addr",
|
2013-08-15 15:43:06 +00:00
|
|
|
(t2LDRpcrel GPR:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p), 0>;
|
2012-01-18 22:46:46 +00:00
|
|
|
def : t2InstAlias<"ldrb${p}.w $Rt, $addr",
|
|
|
|
(t2LDRBpcrel GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrh${p}.w $Rt, $addr",
|
|
|
|
(t2LDRHpcrel GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsb${p}.w $Rt, $addr",
|
|
|
|
(t2LDRSBpcrel GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
|
|
|
def : t2InstAlias<"ldrsh${p}.w $Rt, $addr",
|
|
|
|
(t2LDRSHpcrel GPRnopc:$Rt, t2ldr_pcrel_imm12:$addr, pred:$p)>;
|
2012-01-21 00:07:56 +00:00
|
|
|
|
|
|
|
def : t2InstAlias<"add${p} $Rd, pc, $imm",
|
|
|
|
(t2ADR rGPR:$Rd, imm0_4095:$imm, pred:$p)>;
|
2013-07-16 21:52:34 +00:00
|
|
|
|
2016-05-12 21:22:31 +00:00
|
|
|
// Pseudo instruction ldr Rt, =immediate
|
|
|
|
def t2LDRConstPool
|
|
|
|
: t2AsmPseudo<"ldr${p} $Rt, $immediate",
|
2017-07-11 09:47:12 +00:00
|
|
|
(ins GPR:$Rt, const_pool_asm_imm:$immediate, pred:$p)>;
|
2016-09-13 11:15:51 +00:00
|
|
|
// Version w/ the .w suffix.
|
|
|
|
def : t2InstAlias<"ldr${p}.w $Rt, $immediate",
|
|
|
|
(t2LDRConstPool GPRnopc:$Rt,
|
|
|
|
const_pool_asm_imm:$immediate, pred:$p)>;
|
2016-05-12 21:22:31 +00:00
|
|
|
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ARMv8.1m instructions
|
|
|
|
//
|
|
|
|
|
|
|
|
class V8_1MI<dag oops, dag iops, AddrMode am, InstrItinClass itin, string asm,
|
|
|
|
string ops, string cstr, list<dag> pattern>
|
|
|
|
: Thumb2XI<oops, iops, am, 4, itin, !strconcat(asm, "\t", ops), cstr,
|
|
|
|
pattern>,
|
|
|
|
Requires<[HasV8_1MMainline]>;
|
|
|
|
|
|
|
|
def t2CLRM : V8_1MI<(outs),
|
|
|
|
(ins pred:$p, reglist_with_apsr:$regs, variable_ops),
|
2020-02-05 13:54:18 +00:00
|
|
|
AddrModeNone, NoItinerary, "clrm${p}", "$regs", "", []> {
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
bits<16> regs;
|
|
|
|
|
|
|
|
let Inst{31-16} = 0b1110100010011111;
|
|
|
|
let Inst{15-14} = regs{15-14};
|
|
|
|
let Inst{13} = 0b0;
|
|
|
|
let Inst{12-0} = regs{12-0};
|
|
|
|
}
|
|
|
|
|
|
|
|
class t2BF<dag iops, string asm, string ops>
|
|
|
|
: V8_1MI<(outs ), iops, AddrModeNone, NoItinerary, asm, ops, "", []> {
|
|
|
|
|
|
|
|
let Inst{31-27} = 0b11110;
|
|
|
|
let Inst{15-14} = 0b11;
|
|
|
|
let Inst{12} = 0b0;
|
|
|
|
let Inst{0} = 0b1;
|
|
|
|
|
|
|
|
let Predicates = [IsThumb2, HasV8_1MMainline, HasLOB];
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2BF_LabelPseudo
|
|
|
|
: t2PseudoInst<(outs ), (ins pclabel:$cp), 0, NoItinerary, []> {
|
|
|
|
let isTerminator = 1;
|
|
|
|
let Predicates = [IsThumb2, HasV8_1MMainline, HasLOB];
|
2019-09-29 08:38:48 +00:00
|
|
|
let hasNoSchedulingInfo = 1;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def t2BFi : t2BF<(ins bflabel_u4:$b_label, bflabel_s16:$label, pred:$p),
|
|
|
|
!strconcat("bf", "${p}"), "$b_label, $label"> {
|
|
|
|
bits<4> b_label;
|
|
|
|
bits<16> label;
|
|
|
|
|
|
|
|
let Inst{26-23} = b_label{3-0};
|
|
|
|
let Inst{22-21} = 0b10;
|
|
|
|
let Inst{20-16} = label{15-11};
|
|
|
|
let Inst{13} = 0b1;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2BFic : t2BF<(ins bflabel_u4:$b_label, bflabel_s12:$label,
|
|
|
|
bfafter_target:$ba_label, pred_noal:$bcond), "bfcsel",
|
|
|
|
"$b_label, $label, $ba_label, $bcond"> {
|
|
|
|
bits<4> bcond;
|
|
|
|
bits<12> label;
|
|
|
|
bits<1> ba_label;
|
|
|
|
bits<4> b_label;
|
|
|
|
|
|
|
|
let Inst{26-23} = b_label{3-0};
|
|
|
|
let Inst{22} = 0b0;
|
|
|
|
let Inst{21-18} = bcond{3-0};
|
|
|
|
let Inst{17} = ba_label{0};
|
|
|
|
let Inst{16} = label{11};
|
|
|
|
let Inst{13} = 0b1;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2BFr : t2BF<(ins bflabel_u4:$b_label, rGPR:$Rn, pred:$p),
|
|
|
|
!strconcat("bfx", "${p}"), "$b_label, $Rn"> {
|
|
|
|
bits<4> b_label;
|
|
|
|
bits<4> Rn;
|
|
|
|
|
|
|
|
let Inst{26-23} = b_label{3-0};
|
|
|
|
let Inst{22-20} = 0b110;
|
|
|
|
let Inst{19-16} = Rn{3-0};
|
|
|
|
let Inst{13-1} = 0b1000000000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2BFLi : t2BF<(ins bflabel_u4:$b_label, bflabel_s18:$label, pred:$p),
|
|
|
|
!strconcat("bfl", "${p}"), "$b_label, $label"> {
|
|
|
|
bits<4> b_label;
|
|
|
|
bits<18> label;
|
|
|
|
|
|
|
|
let Inst{26-23} = b_label{3-0};
|
|
|
|
let Inst{22-16} = label{17-11};
|
|
|
|
let Inst{13} = 0b0;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2BFLr : t2BF<(ins bflabel_u4:$b_label, rGPR:$Rn, pred:$p),
|
|
|
|
!strconcat("bflx", "${p}"), "$b_label, $Rn"> {
|
|
|
|
bits<4> b_label;
|
|
|
|
bits<4> Rn;
|
|
|
|
|
|
|
|
let Inst{26-23} = b_label{3-0};
|
|
|
|
let Inst{22-20} = 0b111;
|
|
|
|
let Inst{19-16} = Rn{3-0};
|
|
|
|
let Inst{13-1} = 0b1000000000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
class t2LOL<dag oops, dag iops, string asm, string ops>
|
|
|
|
: V8_1MI<oops, iops, AddrModeNone, NoItinerary, asm, ops, "", [] > {
|
|
|
|
let Inst{31-23} = 0b111100000;
|
|
|
|
let Inst{15-14} = 0b11;
|
|
|
|
let Inst{0} = 0b1;
|
|
|
|
let DecoderMethod = "DecodeLOLoop";
|
|
|
|
let Predicates = [IsThumb2, HasV8_1MMainline, HasLOB];
|
|
|
|
}
|
|
|
|
|
2019-06-25 10:45:51 +00:00
|
|
|
let isNotDuplicable = 1 in {
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
def t2WLS : t2LOL<(outs GPRlr:$LR),
|
|
|
|
(ins rGPR:$Rn, wlslabel_u11:$label),
|
|
|
|
"wls", "$LR, $Rn, $label"> {
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<11> label;
|
|
|
|
let Inst{22-20} = 0b100;
|
|
|
|
let Inst{19-16} = Rn{3-0};
|
|
|
|
let Inst{13-12} = 0b00;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
|
|
|
let usesCustomInserter = 1;
|
2019-12-20 09:32:36 +00:00
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def t2DLS : t2LOL<(outs GPRlr:$LR), (ins rGPR:$Rn),
|
|
|
|
"dls", "$LR, $Rn"> {
|
|
|
|
bits<4> Rn;
|
|
|
|
let Inst{22-20} = 0b100;
|
|
|
|
let Inst{19-16} = Rn{3-0};
|
|
|
|
let Inst{13-1} = 0b1000000000000;
|
|
|
|
let usesCustomInserter = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2LEUpdate : t2LOL<(outs GPRlr:$LRout),
|
|
|
|
(ins GPRlr:$LRin, lelabel_u11:$label),
|
|
|
|
"le", "$LRin, $label"> {
|
|
|
|
bits<11> label;
|
|
|
|
let Inst{22-16} = 0b0001111;
|
|
|
|
let Inst{13-12} = 0b00;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
|
|
|
let usesCustomInserter = 1;
|
2019-12-20 09:32:36 +00:00
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def t2LE : t2LOL<(outs ), (ins lelabel_u11:$label), "le", "$label"> {
|
|
|
|
bits<11> label;
|
|
|
|
let Inst{22-16} = 0b0101111;
|
|
|
|
let Inst{13-12} = 0b00;
|
|
|
|
let Inst{11} = label{0};
|
|
|
|
let Inst{10-1} = label{10-1};
|
2019-12-20 09:32:36 +00:00
|
|
|
let isBranch = 1;
|
|
|
|
let isTerminator = 1;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|
|
|
|
|
2019-06-25 10:45:51 +00:00
|
|
|
def t2DoLoopStart :
|
|
|
|
t2PseudoInst<(outs), (ins rGPR:$elts), 4, IIC_Br,
|
|
|
|
[(int_set_loop_iterations rGPR:$elts)]>, Sched<[WriteBr]>;
|
|
|
|
|
2020-01-17 13:08:24 +00:00
|
|
|
let hasSideEffects = 0 in
|
2019-06-25 10:45:51 +00:00
|
|
|
def t2LoopDec :
|
|
|
|
t2PseudoInst<(outs GPRlr:$Rm), (ins GPRlr:$Rn, imm0_7:$size),
|
|
|
|
4, IIC_Br, []>, Sched<[WriteBr]>;
|
|
|
|
|
2019-07-26 08:15:01 +00:00
|
|
|
let isBranch = 1, isTerminator = 1, hasSideEffects = 1, Defs = [CPSR] in {
|
|
|
|
// Set WhileLoopStart and LoopEnd to occupy 8 bytes because they may
|
|
|
|
// get converted into t2CMP and t2Bcc.
|
2019-07-01 08:21:28 +00:00
|
|
|
def t2WhileLoopStart :
|
|
|
|
t2PseudoInst<(outs),
|
|
|
|
(ins rGPR:$elts, brtarget:$target),
|
2019-07-26 08:15:01 +00:00
|
|
|
8, IIC_Br, []>,
|
2019-07-01 08:21:28 +00:00
|
|
|
Sched<[WriteBr]>;
|
|
|
|
|
2019-06-25 10:45:51 +00:00
|
|
|
def t2LoopEnd :
|
|
|
|
t2PseudoInst<(outs), (ins GPRlr:$elts, brtarget:$target),
|
|
|
|
8, IIC_Br, []>, Sched<[WriteBr]>;
|
|
|
|
|
2019-07-01 08:21:28 +00:00
|
|
|
} // end isBranch, isTerminator, hasSideEffects
|
|
|
|
|
2019-06-25 10:45:51 +00:00
|
|
|
} // end isNotDuplicable
|
|
|
|
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
class CS<string iname, bits<4> opcode, list<dag> pattern=[]>
|
2019-07-31 14:22:45 +00:00
|
|
|
: V8_1MI<(outs rGPR:$Rd), (ins GPRwithZRnosp:$Rn, GPRwithZRnosp:$Rm, pred_noal:$fcond),
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
AddrModeNone, NoItinerary, iname, "$Rd, $Rn, $Rm, $fcond", "", pattern> {
|
|
|
|
bits<4> Rd;
|
|
|
|
bits<4> Rm;
|
|
|
|
bits<4> Rn;
|
|
|
|
bits<4> fcond;
|
|
|
|
|
|
|
|
let Inst{31-20} = 0b111010100101;
|
|
|
|
let Inst{19-16} = Rn{3-0};
|
|
|
|
let Inst{15-12} = opcode;
|
|
|
|
let Inst{11-8} = Rd{3-0};
|
|
|
|
let Inst{7-4} = fcond{3-0};
|
|
|
|
let Inst{3-0} = Rm{3-0};
|
|
|
|
|
|
|
|
let Uses = [CPSR];
|
|
|
|
}
|
|
|
|
|
|
|
|
def t2CSEL : CS<"csel", 0b1000>;
|
|
|
|
def t2CSINC : CS<"csinc", 0b1001>;
|
|
|
|
def t2CSINV : CS<"csinv", 0b1010>;
|
|
|
|
def t2CSNEG : CS<"csneg", 0b1011>;
|
|
|
|
|
2019-09-03 10:53:07 +00:00
|
|
|
let Predicates = [HasV8_1MMainline] in {
|
|
|
|
def : T2Pat<(ARMcsinc GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm),
|
|
|
|
(t2CSINC GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm)>;
|
|
|
|
def : T2Pat<(ARMcsinv GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm),
|
|
|
|
(t2CSINV GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm)>;
|
|
|
|
def : T2Pat<(ARMcsneg GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm),
|
|
|
|
(t2CSNEG GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm)>;
|
|
|
|
|
|
|
|
multiclass ModifiedV8_1CSEL<Instruction Insn, dag modvalue> {
|
|
|
|
def : T2Pat<(ARMcmov modvalue, GPRwithZR:$tval, cmovpred:$imm),
|
|
|
|
(Insn GPRwithZR:$tval, GPRwithZR:$fval, imm0_31:$imm)>;
|
|
|
|
def : T2Pat<(ARMcmov GPRwithZR:$tval, modvalue, cmovpred:$imm),
|
|
|
|
(Insn GPRwithZR:$tval, GPRwithZR:$fval,
|
|
|
|
(i32 (inv_cond_XFORM imm:$imm)))>;
|
|
|
|
}
|
|
|
|
defm : ModifiedV8_1CSEL<t2CSINC, (add rGPR:$fval, 1)>;
|
|
|
|
defm : ModifiedV8_1CSEL<t2CSINV, (xor rGPR:$fval, -1)>;
|
|
|
|
defm : ModifiedV8_1CSEL<t2CSNEG, (sub 0, rGPR:$fval)>;
|
|
|
|
}
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
|
|
|
|
// CS aliases.
|
|
|
|
let Predicates = [HasV8_1MMainline] in {
|
|
|
|
def : InstAlias<"csetm\t$Rd, $fcond",
|
|
|
|
(t2CSINV rGPR:$Rd, ZR, ZR, pred_noal_inv:$fcond)>;
|
|
|
|
|
|
|
|
def : InstAlias<"cset\t$Rd, $fcond",
|
|
|
|
(t2CSINC rGPR:$Rd, ZR, ZR, pred_noal_inv:$fcond)>;
|
|
|
|
|
|
|
|
def : InstAlias<"cinc\t$Rd, $Rn, $fcond",
|
[ARM] Tighten restrictions on use of SP in v8.1-M CSEL.
In the `CSEL Rd,Rm,Rn` instruction family (also including CSINC, CSINV
and CSNEG), the architecture lists it as CONSTRAINED UNPREDICTABLE
(i.e. SoftFail) to use SP in the Rd or Rm slot, but outright illegal
to use it in the Rn slot, not least because some encodings of that
form are used by MVE instructions such as UQRSHLL.
MC was treating all three slots the same, as SoftFail. So the only
reason UQRSHLL was disassembled correctly at all was because the MVE
decode table is separate from the Thumb2 one and takes priority; if
you turned off MVE, then encodings such as `[0x5f,0xea,0x0d,0x83]`
would disassemble as spurious CSELs.
Fixed by inventing another version of the `GPRwithZR` register class,
which disallows SP completely instead of just SoftFailing it.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63862
llvm-svn: 364531
2019-06-27 12:40:40 +00:00
|
|
|
(t2CSINC rGPR:$Rd, GPRwithZRnosp:$Rn, GPRwithZRnosp:$Rn, pred_noal_inv:$fcond)>;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
|
|
|
|
def : InstAlias<"cinv\t$Rd, $Rn, $fcond",
|
[ARM] Tighten restrictions on use of SP in v8.1-M CSEL.
In the `CSEL Rd,Rm,Rn` instruction family (also including CSINC, CSINV
and CSNEG), the architecture lists it as CONSTRAINED UNPREDICTABLE
(i.e. SoftFail) to use SP in the Rd or Rm slot, but outright illegal
to use it in the Rn slot, not least because some encodings of that
form are used by MVE instructions such as UQRSHLL.
MC was treating all three slots the same, as SoftFail. So the only
reason UQRSHLL was disassembled correctly at all was because the MVE
decode table is separate from the Thumb2 one and takes priority; if
you turned off MVE, then encodings such as `[0x5f,0xea,0x0d,0x83]`
would disassemble as spurious CSELs.
Fixed by inventing another version of the `GPRwithZR` register class,
which disallows SP completely instead of just SoftFailing it.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63862
llvm-svn: 364531
2019-06-27 12:40:40 +00:00
|
|
|
(t2CSINV rGPR:$Rd, GPRwithZRnosp:$Rn, GPRwithZRnosp:$Rn, pred_noal_inv:$fcond)>;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
|
|
|
|
def : InstAlias<"cneg\t$Rd, $Rn, $fcond",
|
[ARM] Tighten restrictions on use of SP in v8.1-M CSEL.
In the `CSEL Rd,Rm,Rn` instruction family (also including CSINC, CSINV
and CSNEG), the architecture lists it as CONSTRAINED UNPREDICTABLE
(i.e. SoftFail) to use SP in the Rd or Rm slot, but outright illegal
to use it in the Rn slot, not least because some encodings of that
form are used by MVE instructions such as UQRSHLL.
MC was treating all three slots the same, as SoftFail. So the only
reason UQRSHLL was disassembled correctly at all was because the MVE
decode table is separate from the Thumb2 one and takes priority; if
you turned off MVE, then encodings such as `[0x5f,0xea,0x0d,0x83]`
would disassemble as spurious CSELs.
Fixed by inventing another version of the `GPRwithZR` register class,
which disallows SP completely instead of just SoftFailing it.
Reviewers: DavidSpickett, ostannard
Subscribers: javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63862
llvm-svn: 364531
2019-06-27 12:40:40 +00:00
|
|
|
(t2CSNEG rGPR:$Rd, GPRwithZRnosp:$Rn, GPRwithZRnosp:$Rn, pred_noal_inv:$fcond)>;
|
[ARM] Add the non-MVE instructions in Arm v8.1-M.
This adds support for the new family of conditional selection /
increment / negation instructions; the low-overhead branch
instructions (e.g. BF, WLS, DLS); the CLRM instruction to zero a whole
list of registers at once; the new VMRS/VMSR and VLDR/VSTR
instructions to get data in and out of 8.1-M system registers,
particularly including the new VPR register used by MVE vector
predication.
To support this, we also add a register name 'zr' (used by the CSEL
family to force one of the inputs to the constant 0), and operand
types for lists of registers that are also allowed to include APSR or
VPR (used by CLRM). The VLDR/VSTR instructions also need a new
addressing mode.
The low-overhead branch instructions exist in their own separate
architecture extension, which we treat as enabled by default, but you
can say -mattr=-lob or equivalent to turn it off.
Reviewers: dmgreen, samparker, SjoerdMeijer, t.p.northover
Reviewed By: samparker
Subscribers: miyuki, javed.absar, kristof.beyls, hiraditya, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D62667
llvm-svn: 363039
2019-06-11 09:29:18 +00:00
|
|
|
}
|