188 lines
6.5 KiB
TableGen
188 lines
6.5 KiB
TableGen
//===-- RISCVInstrInfoXMips.td -----------------------------*- tablegen -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file describes the vendor extensions defined by MIPS.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Operand definitions.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// A 7-bit unsigned immediate where the least significant three bits are zero.
|
|
def uimm7_lsb000 : RISCVOp,
|
|
ImmLeaf<XLenVT, [{return isShiftedUInt<4, 3>(Imm);}]> {
|
|
let ParserMatchClass = UImmAsmOperand<7, "Lsb000">;
|
|
let EncoderMethod = "getImmOpValue";
|
|
let DecoderMethod = "decodeUImmOperand<7>";
|
|
let OperandType = "OPERAND_UIMM7_LSB000";
|
|
let MCOperandPredicate = [{
|
|
int64_t Imm;
|
|
if (!MCOp.evaluateAsConstantImm(Imm))
|
|
return false;
|
|
return isShiftedUInt<4, 3>(Imm);
|
|
}];
|
|
}
|
|
|
|
// A 9-bit unsigned offset
|
|
def uimm9 : RISCVUImmOp<9>;
|
|
|
|
// Custom prefetch ADDR selector
|
|
def AddrRegImm9 : ComplexPattern<iPTR, 2, "SelectAddrRegImm9">;
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// MIPS custom instruction formats
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Load double pair format.
|
|
class LDPFormat<dag outs, dag ins, string opcodestr, string argstr>
|
|
: RVInst<outs, ins, opcodestr, argstr, [], InstFormatI> {
|
|
bits<7> imm7;
|
|
bits<5> rs1;
|
|
bits<5> rd1;
|
|
bits<5> rd2;
|
|
|
|
let Inst{31-27} = rd2;
|
|
let Inst{26-23} = imm7{6-3};
|
|
let Inst{22-20} = 0b000;
|
|
let Inst{19-15} = rs1;
|
|
let Inst{14-12} = 0b100;
|
|
let Inst{11-7} = rd1;
|
|
let Inst{6-0} = OPC_CUSTOM_0.Value;
|
|
}
|
|
|
|
// Load word pair format.
|
|
class LWPFormat<dag outs, dag ins, string opcodestr, string argstr>
|
|
: RVInst<outs, ins, opcodestr, argstr, [], InstFormatI> {
|
|
bits<7> imm7;
|
|
bits<5> rs1;
|
|
bits<5> rd1;
|
|
bits<5> rd2;
|
|
|
|
let Inst{31-27} = rd2;
|
|
let Inst{26-22} = imm7{6-2};
|
|
let Inst{21-20} = 0b01;
|
|
let Inst{19-15} = rs1;
|
|
let Inst{14-12} = 0b100;
|
|
let Inst{11-7} = rd1;
|
|
let Inst{6-0} = OPC_CUSTOM_0.Value;
|
|
}
|
|
|
|
// Store double pair format.
|
|
class SDPFormat<dag outs, dag ins, string opcodestr, string argstr>
|
|
: RVInst<outs, ins, opcodestr, argstr, [], InstFormatI> {
|
|
bits<7> imm7;
|
|
bits<5> rs3;
|
|
bits<5> rs2;
|
|
bits<5> rs1;
|
|
|
|
let Inst{31-27} = rs3;
|
|
let Inst{26-25} = imm7{6-5};
|
|
let Inst{24-20} = rs2;
|
|
let Inst{19-15} = rs1;
|
|
let Inst{14-12} = 0b101;
|
|
let Inst{11-10} = imm7{4-3};
|
|
let Inst{9-7} = 0b000;
|
|
let Inst{6-0} = OPC_CUSTOM_0.Value;
|
|
}
|
|
|
|
// Store word pair format.
|
|
class SWPFormat<dag outs, dag ins, string opcodestr, string argstr>
|
|
: RVInst<outs, ins, opcodestr, argstr, [], InstFormatI> {
|
|
bits<7> imm7;
|
|
bits<5> rs3;
|
|
bits<5> rs2;
|
|
bits<5> rs1;
|
|
|
|
let Inst{31-27} = rs3;
|
|
let Inst{26-25} = imm7{6-5};
|
|
let Inst{24-20} = rs2;
|
|
let Inst{19-15} = rs1;
|
|
let Inst{14-12} = 0b101;
|
|
let Inst{11-9} = imm7{4-2};
|
|
let Inst{8-7} = 0b01;
|
|
let Inst{6-0} = OPC_CUSTOM_0.Value;
|
|
}
|
|
|
|
// Prefetch format.
|
|
let hasSideEffects = 0, mayLoad = 1, mayStore = 1 in
|
|
class Mips_prefetch_ri<dag outs, dag ins, string opcodestr, string argstr>
|
|
: RVInst<outs, ins, opcodestr, argstr, [], InstFormatI> {
|
|
bits<9> imm9;
|
|
bits<5> rs1;
|
|
bits<5> hint;
|
|
|
|
let Inst{31-29} = 0b000;
|
|
let Inst{28-20} = imm9;
|
|
let Inst{19-15} = rs1;
|
|
let Inst{14-12} = 0b000;
|
|
let Inst{11-7} = hint;
|
|
let Inst{6-0} = OPC_CUSTOM_0.Value;
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// MIPS extensions
|
|
//===----------------------------------------------------------------------===//
|
|
let Predicates = [HasVendorXMIPSCBOP] ,DecoderNamespace = "Xmipscbop" in {
|
|
def MIPS_PREF : Mips_prefetch_ri<(outs), (ins GPR:$rs1, uimm9:$imm9, uimm5:$hint),
|
|
"mips.pref", "$hint, ${imm9}(${rs1})">,
|
|
Sched<[]>;
|
|
}
|
|
|
|
let Predicates = [HasVendorXMIPSCBOP] in {
|
|
// Prefetch Data Write.
|
|
def : Pat<(prefetch (AddrRegImm9 (XLenVT GPR:$rs1), uimm9:$imm9),
|
|
(i32 1), timm, (i32 1)),
|
|
(MIPS_PREF GPR:$rs1, uimm9:$imm9, 9)>;
|
|
// Prefetch Data Read.
|
|
def : Pat<(prefetch (AddrRegImm9 (XLenVT GPR:$rs1), uimm9:$imm9),
|
|
(i32 0), timm, (i32 1)),
|
|
(MIPS_PREF GPR:$rs1, uimm9:$imm9, 8)>;
|
|
}
|
|
|
|
let Predicates = [HasVendorXMIPSCMov], hasSideEffects = 0, mayLoad = 0, mayStore = 0,
|
|
DecoderNamespace = "Xmipscmov" in {
|
|
def MIPS_CCMOV : RVInstR4<0b11, 0b011, OPC_CUSTOM_0, (outs GPR:$rd),
|
|
(ins GPR:$rs1, GPR:$rs2, GPR:$rs3),
|
|
"mips.ccmov", "$rd, $rs2, $rs1, $rs3">,
|
|
Sched<[]>;
|
|
}
|
|
|
|
let Predicates = [UseCCMovInsn] in {
|
|
def : Pat<(select (riscv_setne (XLenVT GPR:$rs2)),
|
|
(XLenVT GPR:$rs1), (XLenVT GPR:$rs3)),
|
|
(MIPS_CCMOV GPR:$rs1, GPR:$rs2, GPR:$rs3)>;
|
|
def : Pat<(select (riscv_seteq (XLenVT GPR:$rs2)),
|
|
(XLenVT GPR:$rs3), (XLenVT GPR:$rs1)),
|
|
(MIPS_CCMOV GPR:$rs1, GPR:$rs2, GPR:$rs3)>;
|
|
|
|
def : Pat<(select (XLenVT GPR:$rs2), (XLenVT GPR:$rs1), (XLenVT GPR:$rs3)),
|
|
(MIPS_CCMOV GPR:$rs1, GPR:$rs2, GPR:$rs3)>;
|
|
}
|
|
|
|
let Predicates = [HasVendorXMIPSLSP], hasSideEffects = 0,
|
|
DecoderNamespace = "Xmipslsp" in {
|
|
let mayLoad = 1, mayStore = 0 in {
|
|
def MIPS_LWP : LWPFormat<(outs GPR:$rd1, GPR:$rd2), (ins GPR:$rs1, uimm7_lsb00:$imm7),
|
|
"mips.lwp", "$rd1, $rd2, ${imm7}(${rs1})">,
|
|
Sched<[WriteLDW, WriteLDW, ReadMemBase]>;
|
|
def MIPS_LDP : LDPFormat<(outs GPR:$rd1, GPR:$rd2), (ins GPR:$rs1, uimm7_lsb000:$imm7),
|
|
"mips.ldp", "$rd1, $rd2, ${imm7}(${rs1})">,
|
|
Sched<[WriteLDD, WriteLDD, ReadMemBase]>;
|
|
} // mayLoad = 1, mayStore = 0
|
|
|
|
let mayLoad = 0, mayStore = 1 in {
|
|
def MIPS_SWP : SWPFormat<(outs), (ins GPR:$rs2, GPR:$rs3, GPR:$rs1, uimm7_lsb00:$imm7),
|
|
"mips.swp", "$rs2, $rs3, ${imm7}(${rs1})">,
|
|
Sched<[WriteSTW, ReadStoreData, ReadStoreData, ReadMemBase]>;
|
|
def MIPS_SDP : SDPFormat<(outs), (ins GPR:$rs2, GPR:$rs3, GPR:$rs1, uimm7_lsb000:$imm7),
|
|
"mips.sdp", "$rs2, $rs3, ${imm7}(${rs1})">,
|
|
Sched<[WriteSTD, ReadStoreData, ReadStoreData, ReadMemBase]>;
|
|
} // mayLoad = 0, mayStore = 1
|
|
} // Predicates = [HasVendorXMIPSLSP], hasSideEffects = 0, DecoderNamespace = "Xmipslsp"
|