//===-- M68kInstrArithmetic.td - Integer Arith Instrs ------*- 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 // //===----------------------------------------------------------------------===// /// /// \file /// This file describes the integer arithmetic instructions in the M68k /// architecture. Here is the current status of the file: /// /// Machine: /// /// ADD [~] ADDA [~] ADDI [~] ADDQ [ ] ADDX [~] /// CLR [ ] CMP [~] CMPA [~] CMPI [~] CMPM [ ] /// CMP2 [ ] DIVS/DIVU [~] DIVSL/DIVUL [ ] EXT [~] EXTB [ ] /// MULS/MULU [~] NEG [~] NEGX [~] SUB [~] SUBA [~] /// SUBI [~] SUBQ [ ] SUBX [~] /// /// Map: /// /// [ ] - was not touched at all /// [!] - requires extarnal stuff implemented /// [~] - functional implementation /// [X] - complete implementation /// //===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===// // OPMODE Encoding //===----------------------------------------------------------------------===// class MxOpModeEncoding encoding> { bits<3> Value = encoding; } // op EA, Dn def MxOpMode8_d_EA : MxOpModeEncoding<0b000>; def MxOpMode16_d_EA : MxOpModeEncoding<0b001>; def MxOpMode32_d_EA : MxOpModeEncoding<0b010>; // op Dn, EA def MxOpMode8_EA_d : MxOpModeEncoding<0b100>; def MxOpMode16_EA_d : MxOpModeEncoding<0b101>; def MxOpMode32_EA_d : MxOpModeEncoding<0b110>; // op EA, An def MxOpMode16_a_EA : MxOpModeEncoding<0b011>; def MxOpMode32_a_EA : MxOpModeEncoding<0b111>; //===----------------------------------------------------------------------===// // Encoding //===----------------------------------------------------------------------===// let Defs = [CCR] in { let Constraints = "$src = $dst" in { /// Encoding for Normal forms /// ---------------------------------------------------- /// F E D C | B A 9 | 8 7 6 | 5 4 3 | 2 1 0 /// ---------------------------------------------------- /// | | | EFFECTIVE ADDRESS /// x x x x | REG | OP MODE | MODE | REG /// ---------------------------------------------------- // $reg, $ccr <- $reg op $reg class MxBiArOp_R_RR_xEA CMD> : MxInst<(outs DST_TYPE.ROp:$dst), (ins DST_TYPE.ROp:$src, SRC_TYPE.ROp:$opd), MN#"."#DST_TYPE.Prefix#"\t$opd, $dst", [(set DST_TYPE.VT:$dst, CCR, (NODE DST_TYPE.VT:$src, SRC_TYPE.VT:$opd))]> { let Inst = (descend CMD, (operand "$dst", 3), !cast("MxOpMode"#DST_TYPE.Size#"_"#DST_TYPE.RLet#"_EA").Value, !cond( !eq(SRC_TYPE.RLet, "r") : (descend 0b00, (operand "$opd", 4)), !eq(SRC_TYPE.RLet, "d") : (descend 0b000, (operand "$opd", 3)) ) ); } /// This Op is similar to the one above except it uses reversed opmode, some /// commands(e.g. eor) do not support dEA or rEA modes and require EAd for /// register only operations. /// NOTE when using dd commands it is irrelevant which opmode to use(as it seems) /// but some opcodes support address register and some do not which creates this /// mess. class MxBiArOp_R_RR_EAd CMD> : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src, TYPE.ROp:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", [(set TYPE.VT:$dst, CCR, (NODE TYPE.VT:$src, TYPE.VT:$opd))]> { let Inst = (descend CMD, (operand "$opd", 3), !cast("MxOpMode"#TYPE.Size#"_EA_"#TYPE.RLet).Value, /*Destination can only be a data register*/ /*MODE*/0b000, /*REGISTER*/(operand "$dst", 3)); } let mayLoad = 1 in class MxBiArOp_R_RM CMD, MxEncMemOp SRC_ENC> : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src, OPD:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", [(set TYPE.VT:$dst, CCR, (NODE TYPE.VT:$src, (TYPE.Load PAT:$opd)))]> { let Inst = (ascend (descend CMD, (operand "$dst", 3), !cast("MxOpMode"#TYPE.Size#"_"#TYPE.RLet#"_EA").Value, SRC_ENC.EA), SRC_ENC.Supplement ); } /// Encoding for Immediate forms /// --------------------------------------------------- /// F E D C B A 9 8 | 7 6 | 5 4 3 | 2 1 0 /// --------------------------------------------------- /// | | EFFECTIVE ADDRESS /// x x x x x x x x | SIZE | MODE | REG /// --------------------------------------------------- /// 16-BIT WORD DATA | 8-BIT BYTE DATA /// --------------------------------------------------- /// 32-BIT LONG DATA /// --------------------------------------------------- /// NOTE It is used to store an immediate to memory, imm-to-reg are handled with /// normal version // $reg <- $reg op $imm class MxBiArOp_R_RI_xEA CMD> : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src, TYPE.IOp:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", [(set TYPE.VT:$dst, CCR, (NODE TYPE.VT:$src, TYPE.IPat:$opd))]> { let Inst = (ascend (descend CMD, (operand "$dst", 3), !cast("MxOpMode"#TYPE.Size#"_"#TYPE.RLet#"_EA").Value, MxEncAddrMode_i<"opd", TYPE.Size>.EA), MxEncAddrMode_i<"opd", TYPE.Size>.Supplement ); } // Again, there are two ways to write an immediate to Dn register either dEA // opmode or using *I encoding, and again some instructions also support address // registers some do not. class MxBiArOp_R_RI CMD> : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src, TYPE.IOp:$opd), MN#"i."#TYPE.Prefix#"\t$opd, $dst", [(set TYPE.VT:$dst, CCR, (NODE TYPE.VT:$src, TYPE.IPat:$opd))]> { let Inst = (ascend (descend 0b0000, CMD, !cast("MxEncSize"#TYPE.Size).Value, // The destination cannot be address register, so it's always // the MODE for data register direct mode. /*MODE*/0b000, /*REGISTER*/(operand "$dst", 3)), // Source (i.e. immediate value) encoding MxEncAddrMode_i<"opd", TYPE.Size>.Supplement ); } } // Constraints let mayLoad = 1, mayStore = 1 in { // FIXME MxBiArOp_FMR/FMI cannot consume CCR from MxAdd/MxSub which leads for // MxAdd to survive the match and subsequent mismatch. class MxBiArOp_MR CMD, MxEncMemOp DST_ENC> : MxInst<(outs), (ins MEMOpd:$dst, TYPE.ROp:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", []> { let Inst = (ascend (descend CMD, (operand "$opd", 3), !cast("MxOpMode"#TYPE.Size#"_EA_"#TYPE.RLet).Value, DST_ENC.EA), DST_ENC.Supplement ); } class MxBiArOp_MI CMD, MxEncMemOp DST_ENC> : MxInst<(outs), (ins MEMOpd:$dst, TYPE.IOp:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", []> { let Inst = (ascend (descend 0b0000, CMD, !cast("MxEncSize"#TYPE.Size).Value, DST_ENC.EA), // Source (i.e. immediate value) encoding MxEncAddrMode_i<"opd", TYPE.Size>.Supplement, // Destination encoding DST_ENC.Supplement ); } } // mayLoad, mayStore } // Defs = [CCR] multiclass MxBiArOp_DF CMD, bits<4> CMDI> { foreach SZ = [8, 16, 32] in { // op $mem, $reg def NAME#SZ#"dk" : MxBiArOp_R_RM("MxType"#SZ#"d"), !cast("MxType"#SZ).KOp, !cast("MxType"#SZ).KPat, CMD, MxEncAddrMode_k<"opd">>; def NAME#SZ#"dq" : MxBiArOp_R_RM("MxType"#SZ#"d"), !cast("MxType"#SZ).QOp, !cast("MxType"#SZ).QPat, CMD, MxEncAddrMode_q<"opd">>; def NAME#SZ#"dp" : MxBiArOp_R_RM("MxType"#SZ#"d"), !cast("MxType"#SZ).POp, !cast("MxType"#SZ).PPat, CMD, MxEncAddrMode_p<"opd">>; def NAME#SZ#"df" : MxBiArOp_R_RM("MxType"#SZ#"d"), !cast("MxType"#SZ).FOp, !cast("MxType"#SZ).FPat, CMD, MxEncAddrMode_f<"opd">>; def NAME#SZ#"dj" : MxBiArOp_R_RM("MxType"#SZ#"d"), !cast("MxType"#SZ).JOp, !cast("MxType"#SZ).JPat, CMD, MxEncAddrMode_j<"opd">>; // op $imm, $reg def NAME#SZ#"di" : MxBiArOp_R_RI_xEA("MxType"#SZ#"d"), CMD>; // op $reg, $mem def NAME#SZ#"pd" : MxBiArOp_MR("MxType"#SZ#"d"), !cast("MxType"#SZ).POp, CMD, MxEncAddrMode_p<"dst">>; def NAME#SZ#"fd" : MxBiArOp_MR("MxType"#SZ#"d"), !cast("MxType"#SZ).FOp, CMD, MxEncAddrMode_f<"dst">>; def NAME#SZ#"jd" : MxBiArOp_MR("MxType"#SZ#"d"), !cast("MxType"#SZ).JOp, CMD, MxEncAddrMode_j<"dst">>; // op $imm, $mem def NAME#SZ#"pi" : MxBiArOp_MI("MxType"#SZ), !cast("MxType"#SZ).POp, CMDI, MxEncAddrMode_p<"dst">>; def NAME#SZ#"fi" : MxBiArOp_MI("MxType"#SZ), !cast("MxType"#SZ).FOp, CMDI, MxEncAddrMode_f<"dst">>; def NAME#SZ#"ji" : MxBiArOp_MI("MxType"#SZ), !cast("MxType"#SZ).JOp, CMDI, MxEncAddrMode_j<"dst">>; // op $reg, $reg let isCommutable = isComm in def NAME#SZ#"dd" : MxBiArOp_R_RR_xEA("MxType"#SZ#"d"), !cast("MxType"#SZ#"d"), CMD>; } // foreach SZ foreach SZ = [16, 32] in def NAME#SZ#"dr" : MxBiArOp_R_RR_xEA("MxType"#SZ#"d"), !cast("MxType"#SZ#"r"), CMD>; } // MxBiArOp_DF // These special snowflakes allowed to match address registers but since *A // operations do not produce CCR we should not match them against Mx nodes that // produce it. let Pattern = [(null_frag)] in multiclass MxBiArOp_AF CMD> { def NAME#"32ak" : MxBiArOp_R_RM>; def NAME#"32aq" : MxBiArOp_R_RM>; def NAME#"32af" : MxBiArOp_R_RM>; def NAME#"32ap" : MxBiArOp_R_RM>; def NAME#"32aj" : MxBiArOp_R_RM>; def NAME#"32ab" : MxBiArOp_R_RM>; def NAME#"32ai" : MxBiArOp_R_RI_xEA; def NAME#"32ar" : MxBiArOp_R_RR_xEA; } // MxBiArOp_AF // NOTE These naturally produce CCR //===----------------------------------------------------------------------===// // Add/Sub //===----------------------------------------------------------------------===// defm ADD : MxBiArOp_DF<"add", MxAdd, 1, 0xD, 0x6>; defm ADD : MxBiArOp_AF<"adda", MxAdd, 0xD>; defm SUB : MxBiArOp_DF<"sub", MxSub, 0, 0x9, 0x4>; defm SUB : MxBiArOp_AF<"suba", MxSub, 0x9>; // This pattern is used to enable the instruction selector to select ADD32ab // for global values that are allocated in thread-local storage, i.e.: // t8: i32 = ISD::ADD GLOBAL_OFFSET_TABLE, TargetGlobalTLSAddress:i32 // ====> // t8: i32,i8 = ADD32ab GLOBAL_OFFSET_TABLE, TargetGlobalTLSAddress:i32 def : Pat<(add MxARD32:$src, tglobaltlsaddr:$opd), (ADD32ab MxARD32:$src, MxAL32:$opd)>; let Uses = [CCR], Defs = [CCR] in { let Constraints = "$src = $dst" in { /// Encoding for Extended forms /// ------------------------------------------------------ /// F E D C | B A 9 | 8 | 7 6 | 5 4 | 3 | 2 1 0 /// ------------------------------------------------------ /// x x x x | REG Rx | 1 | SIZE | 0 0 | M | REG Ry /// ------------------------------------------------------ /// Rx - destination /// Ry - source /// M - address mode switch // $reg, ccr <- $reg op $reg op ccr class MxBiArOp_R_RRX CMD> : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src, TYPE.ROp:$opd), MN#"."#TYPE.Prefix#"\t$opd, $dst", [(set TYPE.VT:$dst, CCR, (NODE TYPE.VT:$src, TYPE.VT:$opd, CCR))]> { let Inst = (descend CMD, // Destination register (operand "$dst", 3), 0b1, // SIZE !cond(!eq(TYPE.Size, 8): 0b00, !eq(TYPE.Size, 16): 0b01, !eq(TYPE.Size, 32): 0b10), 0b00, /*R/M*/0b0, // Source register (operand "$opd", 3) ); } } // Constraints } // Uses, Defs multiclass MxBiArOp_RFF CMD> { let isCommutable = isComm in { foreach SZ = [8, 16, 32] in def NAME#SZ#"dd" : MxBiArOp_R_RRX("MxType"#SZ#"d"), CMD>; } // isComm } // MxBiArOp_RFF // NOTE These consume and produce CCR defm ADDX : MxBiArOp_RFF<"addx", MxAddX, 1, 0xD>; defm SUBX : MxBiArOp_RFF<"subx", MxSubX, 0, 0x9>; //===----------------------------------------------------------------------===// // And/Xor/Or //===----------------------------------------------------------------------===// defm AND : MxBiArOp_DF<"and", MxAnd, 1, 0xC, 0x2>; defm OR : MxBiArOp_DF<"or", MxOr, 1, 0x8, 0x0>; multiclass MxBiArOp_DF_EAd CMD, bits<4> CMDI> { foreach SZ = [8, 16, 32] in { let isCommutable = 1 in def NAME#SZ#"dd" : MxBiArOp_R_RR_EAd("MxType"#SZ#"d"), CMD>; def NAME#SZ#"di" : MxBiArOp_R_RI("MxType"#SZ#"d"), CMDI>; } // foreach SZ } // MxBiArOp_DF_EAd defm XOR : MxBiArOp_DF_EAd<"eor", MxXor, 0xB, 0xA>; //===----------------------------------------------------------------------===// // CMP //===----------------------------------------------------------------------===// let Defs = [CCR] in { class MxCmp_RR : MxInst<(outs), (ins LHS_TYPE.ROp:$lhs, RHS_TYPE.ROp:$rhs), "cmp."#RHS_TYPE.Prefix#"\t$lhs, $rhs", [(set CCR, (MxCmp LHS_TYPE.VT:$lhs, RHS_TYPE.VT:$rhs))]> { let Inst = (descend 0b1011, // REGISTER (operand "$rhs", 3), // OPMODE !cast("MxOpMode"#RHS_TYPE.Size#"_"#RHS_TYPE.RLet#"_EA").Value, // MODE without last bit 0b00, // REGISTER prefixed by D/A bit (operand "$lhs", 4) ); } class MxCmp_RI : MxInst<(outs), (ins TYPE.IOp:$imm, TYPE.ROp:$reg), "cmpi."#TYPE.Prefix#"\t$imm, $reg", [(set CCR, (MxCmp TYPE.IPat:$imm, TYPE.VT:$reg))]> { let Inst = (ascend (descend 0b00001100, !cast("MxEncSize"#TYPE.Size).Value, // The destination cannot be address register, so it's always // the MODE for data register direct mode. /*MODE*/0b000, /*REGISTER*/(operand "$reg", 3)), // Source (i.e. immediate value) encoding MxEncAddrMode_i<"imm", TYPE.Size>.Supplement ); } let mayLoad = 1 in { class MxCmp_MI : MxInst<(outs), (ins TYPE.IOp:$imm, MEMOpd:$mem), "cmpi."#TYPE.Prefix#"\t$imm, $mem", [(set CCR, (MxCmp TYPE.IPat:$imm, (load MEMPat:$mem)))]> { let Inst = (ascend (descend 0b00001100, !cast("MxEncSize"#TYPE.Size).Value, MEM_ENC.EA), // Source (i.e. immediate value) encoding MxEncAddrMode_i<"imm", TYPE.Size>.Supplement, // Destination (i.e. memory operand) encoding MEM_ENC.Supplement ); } // FIXME: What about abs.W? class MxCmp_BI : MxInst<(outs), (ins TYPE.IOp:$imm, MxAL32:$abs), "cmpi."#TYPE.Prefix#"\t$imm, $abs", [(set CCR, (MxCmp TYPE.IPat:$imm, (load (i32 (MxWrapper tglobaladdr:$abs)))))]> { defvar AbsEncoding = MxEncAddrMode_abs<"abs", true>; let Inst = (ascend (descend 0b00001100, !cast("MxEncSize"#TYPE.Size).Value, AbsEncoding.EA), // Source (i.e. immediate value) encoding MxEncAddrMode_i<"imm", TYPE.Size>.Supplement, // Destination (i.e. memory operand) encoding AbsEncoding.Supplement ); } class MxCmp_RM : MxInst<(outs), (ins TYPE.ROp:$reg, MEMOpd:$mem), "cmp."#TYPE.Prefix#"\t$mem, $reg", [(set CCR, (MxCmp (load MEMPat:$mem), TYPE.ROp:$reg))]> { let Inst = (ascend (descend 0b1011, // REGISTER (operand "$reg", 3), // OPMODE !cast("MxOpMode"#TYPE.Size#"_d_EA").Value, MEM_ENC.EA), MEM_ENC.Supplement ); } } // let mayLoad = 1 } // let Defs = [CCR] multiclass MMxCmp_RM { def NAME#TYPE.KOp.Letter : MxCmp_RM>; def NAME#TYPE.QOp.Letter : MxCmp_RM>; def NAME#TYPE.POp.Letter : MxCmp_RM>; def NAME#TYPE.FOp.Letter : MxCmp_RM>; def NAME#TYPE.JOp.Letter : MxCmp_RM>; } multiclass MMxCmp_MI { def NAME#TYPE.KOp.Letter#"i" : MxCmp_MI>; def NAME#TYPE.QOp.Letter#"i" : MxCmp_MI>; def NAME#TYPE.POp.Letter#"i" : MxCmp_MI>; def NAME#TYPE.FOp.Letter#"i" : MxCmp_MI>; def NAME#TYPE.JOp.Letter#"i" : MxCmp_MI>; } foreach S = [8, 16, 32] in { def CMP#S#di : MxCmp_RI("MxType"#S#"d")>; def CMP#S#bi : MxCmp_BI("MxType"#S#"d")>; } // foreach def CMP8dd : MxCmp_RR; foreach S = [16, 32] in { def CMP#S#dr : MxCmp_RR("MxType"#S#"r"), !cast("MxType"#S#"d")>; } // cmp mem, Dn defm CMP8d : MMxCmp_RM; defm CMP16d : MMxCmp_RM; defm CMP32d : MMxCmp_RM; // cmp #imm, mem defm CMP8 : MMxCmp_MI; defm CMP16 : MMxCmp_MI; defm CMP32 : MMxCmp_MI; //===----------------------------------------------------------------------===// // EXT //===----------------------------------------------------------------------===// /// --------------------------------------------------- /// F E D C B A 9 | 8 7 6 | 5 4 3 | 2 1 0 /// --------------------------------------------------- /// 0 1 0 0 1 0 0 | OPMODE | 0 0 0 | REG /// --------------------------------------------------- let Defs = [CCR] in let Constraints = "$src = $dst" in class MxExt : MxInst<(outs TO.ROp:$dst), (ins TO.ROp:$src), "ext."#TO.Prefix#"\t$src", []> { let Inst = (descend 0b0100100, // OPMODE !cond( // byte -> word !and(!eq(FROM.Size, 8), !eq(TO.Size, 16)): 0b010, // word -> long !and(!eq(FROM.Size, 16), !eq(TO.Size, 32)): 0b011, // byte -> long !and(!eq(FROM.Size, 8), !eq(TO.Size, 32)): 0b111 ), 0b000, // REGISTER (operand "$src", 3) ); } def EXT16 : MxExt; def EXT32 : MxExt; def : Pat<(sext_inreg i16:$src, i8), (EXT16 $src)>; def : Pat<(sext_inreg i32:$src, i16), (EXT32 $src)>; def : Pat<(sext_inreg i32:$src, i8), (EXT32 (MOVXd32d16 (EXT16 (EXTRACT_SUBREG $src, MxSubRegIndex16Lo))))>; //===----------------------------------------------------------------------===// // DIV/MUL //===----------------------------------------------------------------------===// /// Word operation: /// ---------------------------------------------------- /// F E D C | B A 9 | 8 7 6 | 5 4 3 | 2 1 0 /// ---------------------------------------------------- /// | | | EFFECTIVE ADDRESS /// x x x x | REG | OP MODE | MODE | REG /// ---------------------------------------------------- let Defs = [CCR] in { let Constraints = "$src = $dst" in { // $dreg <- $dreg op $dreg class MxDiMuOp_DD CMD, bit SIGNED = false, MxOperand DST, MxOperand OPD> : MxInst<(outs DST:$dst), (ins DST:$src, OPD:$opd), MN#"\t$opd, $dst", []> { let Inst = (descend CMD, // REGISTER (operand "$dst", 3), !if(SIGNED, 0b111, 0b011), /*MODE*/0b000, /*REGISTER*/(operand "$opd", 3) ); } // $dreg <- $dreg op $dreg class MxDiMuOp_DD_Long CMD, bit SIGNED = false> : MxInst<(outs MxDRD32:$dst), (ins MxDRD32:$src, MxDRD32:$opd), MN#"\t$opd, $dst", []> { let Inst = (ascend (descend CMD, /*MODE*/0b000, /*REGISTER*/(operand "$opd", 3)), (descend 0b0, // REGISTER (operand "$dst", 3), !if(SIGNED, 0b1, 0b0), /*SIZE*/0b0, 0b0000000, // Dr REGISTER 0b000) ); } // $reg <- $reg op $imm class MxDiMuOp_DI CMD, bit SIGNED = false, MxOperand DST, MxOperand OPD> : MxInst<(outs DST:$dst), (ins DST:$src, OPD:$opd), MN#"\t$opd, $dst", []> { // FIXME: Support immediates with different widths. defvar ImmEnc = MxEncAddrMode_i<"opd", 16>; let Inst = (ascend (descend CMD, // REGISTER (operand "$dst", 3), !if(SIGNED, 0b111, 0b011), ImmEnc.EA), ImmEnc.Supplement ); } } // let Constraints } // Defs = [CCR] multiclass MxDiMuOp CMD, bit isComm = 0> { let isCommutable = isComm in { def "S"#NAME#"d32d16" : MxDiMuOp_DD; def "U"#NAME#"d32d16" : MxDiMuOp_DD; } def "S"#NAME#"d32i16" : MxDiMuOp_DI; def "U"#NAME#"d32i16" : MxDiMuOp_DI; } defm DIV : MxDiMuOp<"div", 0x8>; def SDIVd32d32 : MxDiMuOp_DD_Long<"divs.l", 0x131, /*SIGNED*/true>; def UDIVd32d32 : MxDiMuOp_DD_Long<"divu.l", 0x131, /*SIGNED*/false>; // This is used to cast immediates to 16-bits for operations which don't // support smaller immediate sizes. def as_i16imm : SDNodeXFormgetTargetConstant(N->getSExtValue(), SDLoc(N), MVT::i16); }]>; // RR i8 def : Pat<(sdiv i8:$dst, i8:$opd), (EXTRACT_SUBREG (SDIVd32d16 (MOVSXd32d8 $dst), (MOVSXd16d8 $opd)), MxSubRegIndex8Lo)>; def : Pat<(udiv i8:$dst, i8:$opd), (EXTRACT_SUBREG (UDIVd32d16 (MOVZXd32d8 $dst), (MOVZXd16d8 $opd)), MxSubRegIndex8Lo)>; def : Pat<(srem i8:$dst, i8:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SDIVd32d16 (MOVSXd32d8 $dst), (MOVSXd16d8 $opd)), 8), 8), MxSubRegIndex8Lo)>; def : Pat<(urem i8:$dst, i8:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UDIVd32d16 (MOVZXd32d8 $dst), (MOVZXd16d8 $opd)), 8), 8), MxSubRegIndex8Lo)>; // RR i16 def : Pat<(sdiv i16:$dst, i16:$opd), (EXTRACT_SUBREG (SDIVd32d16 (MOVSXd32d16 $dst), $opd), MxSubRegIndex16Lo)>; def : Pat<(udiv i16:$dst, i16:$opd), (EXTRACT_SUBREG (UDIVd32d16 (MOVZXd32d16 $dst), $opd), MxSubRegIndex16Lo)>; def : Pat<(srem i16:$dst, i16:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SDIVd32d16 (MOVSXd32d16 $dst), $opd), 8), 8), MxSubRegIndex16Lo)>; def : Pat<(urem i16:$dst, i16:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UDIVd32d16 (MOVZXd32d16 $dst), $opd), 8), 8), MxSubRegIndex16Lo)>; // RR i32 def : Pat<(sdiv i32:$dst, i32:$opd), (SDIVd32d32 $dst, $opd)>; def : Pat<(udiv i32:$dst, i32:$opd), (UDIVd32d32 $dst, $opd)>; // RI i8 def : Pat<(sdiv i8:$dst, MximmSExt8:$opd), (EXTRACT_SUBREG (SDIVd32i16 (MOVSXd32d8 $dst), (as_i16imm $opd)), MxSubRegIndex8Lo)>; def : Pat<(udiv i8:$dst, MximmSExt8:$opd), (EXTRACT_SUBREG (UDIVd32i16 (MOVZXd32d8 $dst), (as_i16imm $opd)), MxSubRegIndex8Lo)>; def : Pat<(srem i8:$dst, MximmSExt8:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SDIVd32i16 (MOVSXd32d8 $dst), (as_i16imm $opd)), 8), 8), MxSubRegIndex8Lo)>; def : Pat<(urem i8:$dst, MximmSExt8:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UDIVd32i16 (MOVZXd32d8 $dst), (as_i16imm $opd)), 8), 8), MxSubRegIndex8Lo)>; // RI i16 def : Pat<(sdiv i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (SDIVd32i16 (MOVSXd32d16 $dst), imm:$opd), MxSubRegIndex16Lo)>; def : Pat<(udiv i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (UDIVd32i16 (MOVZXd32d16 $dst), imm:$opd), MxSubRegIndex16Lo)>; def : Pat<(srem i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SDIVd32i16 (MOVSXd32d16 $dst), imm:$opd), 8), 8), MxSubRegIndex16Lo)>; def : Pat<(urem i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UDIVd32i16 (MOVZXd32d16 $dst), imm:$opd), 8), 8), MxSubRegIndex16Lo)>; defm MUL : MxDiMuOp<"mul", 0xC, 1>; def SMULd32d32 : MxDiMuOp_DD_Long<"muls.l", 0x130, /*SIGNED*/true>; def UMULd32d32 : MxDiMuOp_DD_Long<"mulu.l", 0x130, /*SIGNED*/false>; // RR def : Pat<(mul i16:$dst, i16:$opd), (EXTRACT_SUBREG (SMULd32d16 (MOVXd32d16 $dst), $opd), MxSubRegIndex16Lo)>; def : Pat<(mulhs i16:$dst, i16:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SMULd32d16 (MOVXd32d16 $dst), $opd), 8), 8), MxSubRegIndex16Lo)>; def : Pat<(mulhu i16:$dst, i16:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UMULd32d16 (MOVXd32d16 $dst), $opd), 8), 8), MxSubRegIndex16Lo)>; def : Pat<(mul i32:$dst, i32:$opd), (SMULd32d32 $dst, $opd)>; // RI def : Pat<(mul i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (SMULd32i16 (MOVXd32d16 $dst), imm:$opd), MxSubRegIndex16Lo)>; def : Pat<(mulhs i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (ASR32di (ASR32di (SMULd32i16 (MOVXd32d16 $dst), imm:$opd), 8), 8), MxSubRegIndex16Lo)>; def : Pat<(mulhu i16:$dst, MximmSExt16:$opd), (EXTRACT_SUBREG (LSR32di (LSR32di (UMULd32i16 (MOVXd32d16 $dst), imm:$opd), 8), 8), MxSubRegIndex16Lo)>; //===----------------------------------------------------------------------===// // NEG/NEGX //===----------------------------------------------------------------------===// /// ------------+------------+------+---------+--------- /// F E D C | B A 9 8 | 7 6 | 5 4 3 | 2 1 0 /// ------------+------------+------+------------------- /// | | | EFFECTIVE ADDRESS /// 0 1 0 0 | x x x x | SIZE | MODE | REG /// ------------+------------+------+---------+--------- let Defs = [CCR] in { let Constraints = "$src = $dst" in { class MxNeg_D : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src), "neg."#TYPE.Prefix#"\t$dst", [(set TYPE.VT:$dst, (ineg TYPE.VT:$src))]> { let Inst = (descend 0b01000100, /*SIZE*/!cast("MxEncSize"#TYPE.Size).Value, //MODE without last bit 0b00, //REGISTER prefixed by D/A bit (operand "$dst", 4) ); } let Uses = [CCR] in { class MxNegX_D : MxInst<(outs TYPE.ROp:$dst), (ins TYPE.ROp:$src), "negx."#TYPE.Prefix#"\t$dst", [(set TYPE.VT:$dst, (MxSubX 0, TYPE.VT:$src, CCR))]> { let Inst = (descend 0b01000000, /*SIZE*/!cast("MxEncSize"#TYPE.Size).Value, //MODE without last bit 0b00, //REGISTER prefixed by D/A bit (operand "$dst", 4) ); } } } // let Constraints } // let Defs = [CCR] foreach S = [8, 16, 32] in { def NEG#S#d : MxNeg_D("MxType"#S#"d")>; def NEGX#S#d : MxNegX_D("MxType"#S#"d")>; } def : Pat<(MxSub 0, i8 :$src), (NEG8d MxDRD8 :$src)>; def : Pat<(MxSub 0, i16:$src), (NEG16d MxDRD16:$src)>; def : Pat<(MxSub 0, i32:$src), (NEG32d MxDRD32:$src)>; // SExt of i1 values. // Although we specify `ZeroOrOneBooleanContent` for boolean content, // we're still adding an AND here as we don't know the origin of the i1 value. def : Pat<(sext_inreg i8:$src, i1), (NEG8d (AND8di MxDRD8:$src, 1))>; def : Pat<(sext_inreg i16:$src, i1), (NEG16d (AND16di MxDRD16:$src, 1))>; def : Pat<(sext_inreg i32:$src, i1), (NEG32d (AND32di MxDRD32:$src, 1))>; //===----------------------------------------------------------------------===// // no-CCR Patterns //===----------------------------------------------------------------------===// /// Basically the reason for this stuff is that add and addc share the same /// operand types constraints for whatever reasons and I had to define a common /// MxAdd and MxSub instructions that produce CCR and then pattern-map add and addc /// to it. /// NOTE On the other hand I see no reason why I cannot just drop explicit CCR /// result. Anyway works for now, hopefully I will better understand how this stuff /// is designed later foreach N = ["add", "addc"] in { // add reg, reg def : Pat<(!cast(N) i8 :$src, i8 :$opd), (ADD8dd MxDRD8 :$src, MxDRD8 :$opd)>; def : Pat<(!cast(N) i16:$src, i16:$opd), (ADD16dr MxXRD16:$src, MxDRD16:$opd)>; def : Pat<(!cast(N) i32:$src, i32:$opd), (ADD32dr MxXRD32:$src, MxDRD32:$opd)>; // add (An), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.JPat:$opd)), (ADD8dj MxDRD8:$src, MxType8.JOp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.JPat:$opd)), (ADD16dj MxDRD16:$src, MxType16.JOp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.JPat:$opd)), (ADD32dj MxDRD32:$src, MxType32.JOp:$opd)>; // add (i,An), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.PPat:$opd)), (ADD8dp MxDRD8:$src, MxType8.POp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.PPat:$opd)), (ADD16dp MxDRD16:$src, MxType16.POp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.PPat:$opd)), (ADD32dp MxDRD32:$src, MxType32.POp:$opd)>; // add (i,An,Xn), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.FPat:$opd)), (ADD8df MxDRD8:$src, MxType8.FOp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.FPat:$opd)), (ADD16df MxDRD16:$src, MxType16.FOp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.FPat:$opd)), (ADD32df MxDRD32:$src, MxType32.FOp:$opd)>; // add reg, imm def : Pat<(!cast(N) i8: $src, MximmSExt8:$opd), (ADD8di MxDRD8 :$src, imm:$opd)>; def : Pat<(!cast(N) i16:$src, MximmSExt16:$opd), (ADD16di MxDRD16:$src, imm:$opd)>; // LEAp is more complex and thus will be selected over normal ADD32ri but it cannot // be used with data registers, here by adding complexity to a simple ADD32ri insts // we make sure it will be selected over LEAp let AddedComplexity = 15 in { def : Pat<(!cast(N) i32:$src, MximmSExt32:$opd), (ADD32di MxDRD32:$src, imm:$opd)>; } // AddedComplexity = 15 // add imm, (An) def : Pat<(store (!cast(N) (load MxType8.JPat:$dst), MxType8.IPat:$opd), MxType8.JPat:$dst), (ADD8ji MxType8.JOp:$dst, imm:$opd)>; def : Pat<(store (!cast(N) (load MxType16.JPat:$dst), MxType16.IPat:$opd), MxType16.JPat:$dst), (ADD16ji MxType16.JOp:$dst, imm:$opd)>; def : Pat<(store (!cast(N) (load MxType32.JPat:$dst), MxType32.IPat:$opd), MxType32.JPat:$dst), (ADD32ji MxType32.JOp:$dst, imm:$opd)>; } // foreach add, addc def : Pat<(adde i8 :$src, i8 :$opd), (ADDX8dd MxDRD8 :$src, MxDRD8 :$opd)>; def : Pat<(adde i16:$src, i16:$opd), (ADDX16dd MxDRD16:$src, MxDRD16:$opd)>; def : Pat<(adde i32:$src, i32:$opd), (ADDX32dd MxDRD32:$src, MxDRD32:$opd)>; foreach N = ["sub", "subc"] in { // sub reg, reg def : Pat<(!cast(N) i8 :$src, i8 :$opd), (SUB8dd MxDRD8 :$src, MxDRD8 :$opd)>; def : Pat<(!cast(N) i16:$src, i16:$opd), (SUB16dd MxDRD16:$src, MxDRD16:$opd)>; def : Pat<(!cast(N) i32:$src, i32:$opd), (SUB32dd MxDRD32:$src, MxDRD32:$opd)>; // sub (An), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.JPat:$opd)), (SUB8dj MxDRD8:$src, MxType8.JOp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.JPat:$opd)), (SUB16dj MxDRD16:$src, MxType16.JOp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.JPat:$opd)), (SUB32dj MxDRD32:$src, MxType32.JOp:$opd)>; // sub (i,An), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.PPat:$opd)), (SUB8dp MxDRD8:$src, MxType8.POp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.PPat:$opd)), (SUB16dp MxDRD16:$src, MxType16.POp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.PPat:$opd)), (SUB32dp MxDRD32:$src, MxType32.POp:$opd)>; // sub (i,An,Xn), reg def : Pat<(!cast(N) MxType8.VT:$src, (Mxloadi8 MxType8.FPat:$opd)), (SUB8df MxDRD8:$src, MxType8.FOp:$opd)>; def : Pat<(!cast(N) MxType16.VT:$src, (Mxloadi16 MxType16.FPat:$opd)), (SUB16df MxDRD16:$src, MxType16.FOp:$opd)>; def : Pat<(!cast(N) MxType32.VT:$src, (Mxloadi32 MxType32.FPat:$opd)), (SUB32df MxDRD32:$src, MxType32.FOp:$opd)>; // sub reg, imm def : Pat<(!cast(N) i8 :$src, MximmSExt8 :$opd), (SUB8di MxDRD8 :$src, imm:$opd)>; def : Pat<(!cast(N) i16:$src, MximmSExt16:$opd), (SUB16di MxDRD16:$src, imm:$opd)>; def : Pat<(!cast(N) i32:$src, MximmSExt32:$opd), (SUB32di MxDRD32:$src, imm:$opd)>; // sub imm, (An) def : Pat<(store (!cast(N) (load MxType8.JPat:$dst), MxType8.IPat:$opd), MxType8.JPat:$dst), (SUB8ji MxType8.JOp:$dst, imm:$opd)>; def : Pat<(store (!cast(N) (load MxType16.JPat:$dst), MxType16.IPat:$opd), MxType16.JPat:$dst), (SUB16ji MxType16.JOp:$dst, imm:$opd)>; def : Pat<(store (!cast(N) (load MxType32.JPat:$dst), MxType32.IPat:$opd), MxType32.JPat:$dst), (SUB32ji MxType32.JOp:$dst, imm:$opd)>; } // foreach sub, subx def : Pat<(sube i8 :$src, i8 :$opd), (SUBX8dd MxDRD8 :$src, MxDRD8 :$opd)>; def : Pat<(sube i16:$src, i16:$opd), (SUBX16dd MxDRD16:$src, MxDRD16:$opd)>; def : Pat<(sube i32:$src, i32:$opd), (SUBX32dd MxDRD32:$src, MxDRD32:$opd)>; multiclass BitwisePat { // op reg, reg def : Pat<(OP i8 :$src, i8 :$opd), (!cast(INST#"8dd") MxDRD8 :$src, MxDRD8 :$opd)>; def : Pat<(OP i16:$src, i16:$opd), (!cast(INST#"16dd") MxDRD16:$src, MxDRD16:$opd)>; def : Pat<(OP i32:$src, i32:$opd), (!cast(INST#"32dd") MxDRD32:$src, MxDRD32:$opd)>; // op reg, imm def : Pat<(OP i8: $src, MximmSExt8 :$opd), (!cast(INST#"8di") MxDRD8 :$src, imm:$opd)>; def : Pat<(OP i16:$src, MximmSExt16:$opd), (!cast(INST#"16di") MxDRD16:$src, imm:$opd)>; def : Pat<(OP i32:$src, MximmSExt32:$opd), (!cast(INST#"32di") MxDRD32:$src, imm:$opd)>; } defm : BitwisePat<"AND", and>; defm : BitwisePat<"OR", or>; defm : BitwisePat<"XOR", xor>; //===----------------------------------------------------------------------===// // Floating point arithmetic instruction //===----------------------------------------------------------------------===// let Defs = [FPS] in class MxFArithBase_FF patterns> : MxInst { let Uses = !if(!eq(rounding, ""), [FPC], []); let Predicates = !if(!eq(rounding, ""), [AtLeastM68881], [AtLeastM68040]); } class MxFPOpModeSelector single, bits<7> double, bits<7> extended> { bits<7> Mode = !cond(!eq(rounding, "s"): single, !eq(rounding, "d"): double, !eq(rounding, ""): extended); } //===----------------------------------------------------------------------===// // Unary floating point instruction //===----------------------------------------------------------------------===// class MxFUnary_FF opmode> : MxFArithBase_FF<(outs Opnd.Op:$dst), (ins Opnd.Op:$src), "f"#rounding#mnemonic#".x\t$src, $dst", rounding, [(null_frag)]> { let Inst = (ascend (descend 0b1111, /*COPROCESSOR ID*/0b001, 0b000, /*MODE+REGISTER*/0b000000), (descend 0b0, /* R/M */ 0b0, 0b0, /*SOURCE SPECIFIER*/ (operand "$src", 3), /*DESTINATION*/ (operand "$dst", 3), /*OPMODE*/ opmode) ); } multiclass MxFUnaryOp single, bits<7> double, bits<7> extended> { foreach rounding = ["", "s", "d"] in { defvar opmode = MxFPOpModeSelector.Mode; def F # !toupper(rounding) # !substr(NAME, 1) # "80fp_fp" : MxFUnary_FF; let isCodeGenOnly = 1 in foreach size = [32, 64] in def F # !toupper(rounding) # !substr(NAME, 1) # size # "fp_fp" : MxFUnary_FF("MxOp"#size#"AddrMode_fpr"), rounding, mnemonic, opmode>; } } defm FABS : MxFUnaryOp<"abs", 0b1011000, 0b1011100, 0b0011000>; defm FNEG : MxFUnaryOp<"neg", 0b1011010, 0b1011110, 0b0011010>; //===----------------------------------------------------------------------===// // Binary floating point instruction //===----------------------------------------------------------------------===// let Constraints = "$src = $dst" in class MxFBinary_FF opmode> : MxFArithBase_FF<(outs Opnd.Op:$dst), (ins Opnd.Op:$src, Opnd.Op:$opd), "f"#rounding#mnemonic#".x\t$opd, $dst", rounding, [(null_frag)]> { let Inst = (ascend (descend 0b1111, /*COPROCESSOR ID*/0b001, 0b000, /*MODE+REGISTER*/0b000000), (descend 0b0, /* R/M */ 0b0, 0b0, /*SOURCE SPECIFIER*/ (operand "$opd", 3), /*DESTINATION*/ (operand "$dst", 3), /*OPMODE*/ opmode) ); } multiclass MxFBinaryOp single, bits<7> double, bits<7> extended> { foreach rounding = ["", "s", "d"] in { defvar opmode = MxFPOpModeSelector.Mode; def F # !toupper(rounding) # !substr(NAME, 1) # "80fp_fp" : MxFBinary_FF; let isCodeGenOnly = 1 in foreach size = [32, 64] in def F # !toupper(rounding) # !substr(NAME, 1) # size # "fp_fp" : MxFBinary_FF("MxOp"#size#"AddrMode_fpr"), rounding, mnemonic, opmode>; } } defm FADD : MxFBinaryOp<"add", 0b1100010, 0b1100110, 0b0100010>; defm FSUB : MxFBinaryOp<"sub", 0b1101000, 0b1101100, 0b0101000>; defm FMUL : MxFBinaryOp<"mul", 0b1100011, 0b1100111, 0b0100011>; defm FDIV : MxFBinaryOp<"div", 0b1100000, 0b1100100, 0b0100000>;