| //===-- 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 : RISCVUImmLsbZeroOp<7, 3>; |
| |
| // 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 Custom Barrier Insns Format. |
| let hasSideEffects = 1, mayLoad = 0, mayStore = 0 in |
| class MIPSExtInst_ri<bits<6> shimm5, string opcodestr> |
| : RVInstIShift<0b00000, 0b001, OPC_OP_IMM, (outs), (ins), opcodestr, ""> { |
| let shamt = shimm5; |
| let rd = 0; |
| let rs1 = 0; |
| } |
| |
| //===----------------------------------------------------------------------===// |
| // MIPS extensions |
| //===----------------------------------------------------------------------===// |
| let Predicates = [HasVendorXMIPSEXECTL], DecoderNamespace = "XMIPS" in { |
| def MIPS_EHB : MIPSExtInst_ri<0b000011, "mips.ehb">; |
| def MIPS_IHB : MIPSExtInst_ri<0b000001, "mips.ihb">; |
| def MIPS_PAUSE : MIPSExtInst_ri<0b000101, "mips.pause">; |
| } |
| |
| let Predicates = [HasVendorXMIPSEXECTL] in { |
| // Intrinsics |
| def : Pat<(int_riscv_mips_pause), (MIPS_PAUSE)>; |
| def : Pat<(int_riscv_mips_ihb), (MIPS_IHB)>; |
| def : Pat<(int_riscv_mips_ehb), (MIPS_EHB)>; |
| } |
| |
| let Predicates = [HasVendorXMIPSCBOP], DecoderNamespace = "XMIPS" 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 = "XMIPS" 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 = [UseMIPSCCMovInsn] 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 = "XMIPS" 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 = "XMIPS" |