| //===- MipsLegalizerInfo.cpp ------------------------------------*- C++ -*-===// |
| // |
| // 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 implements the targeting of the Machinelegalizer class for Mips. |
| /// \todo This should be generated by TableGen. |
| //===----------------------------------------------------------------------===// |
| |
| #include "MipsLegalizerInfo.h" |
| #include "MipsTargetMachine.h" |
| #include "llvm/CodeGen/GlobalISel/LegalizerHelper.h" |
| #include "llvm/IR/IntrinsicsMips.h" |
| |
| using namespace llvm; |
| |
| struct TypesAndMemOps { |
| LLT ValTy; |
| LLT PtrTy; |
| unsigned MemSize; |
| bool MustBeNaturallyAligned; |
| }; |
| |
| static bool |
| CheckTy0Ty1MemSizeAlign(const LegalityQuery &Query, |
| std::initializer_list<TypesAndMemOps> SupportedValues) { |
| for (auto &Val : SupportedValues) { |
| if (Val.ValTy != Query.Types[0]) |
| continue; |
| if (Val.PtrTy != Query.Types[1]) |
| continue; |
| if (Val.MemSize != Query.MMODescrs[0].SizeInBits) |
| continue; |
| if (Val.MustBeNaturallyAligned && |
| Query.MMODescrs[0].SizeInBits % Query.MMODescrs[0].AlignInBits != 0) |
| continue; |
| return true; |
| } |
| return false; |
| } |
| |
| static bool CheckTyN(unsigned N, const LegalityQuery &Query, |
| std::initializer_list<LLT> SupportedValues) { |
| for (auto &Val : SupportedValues) |
| if (Val == Query.Types[N]) |
| return true; |
| return false; |
| } |
| |
| MipsLegalizerInfo::MipsLegalizerInfo(const MipsSubtarget &ST) { |
| using namespace TargetOpcode; |
| |
| const LLT s1 = LLT::scalar(1); |
| const LLT s32 = LLT::scalar(32); |
| const LLT s64 = LLT::scalar(64); |
| const LLT v16s8 = LLT::vector(16, 8); |
| const LLT v8s16 = LLT::vector(8, 16); |
| const LLT v4s32 = LLT::vector(4, 32); |
| const LLT v2s64 = LLT::vector(2, 64); |
| const LLT p0 = LLT::pointer(0, 32); |
| |
| getActionDefinitionsBuilder({G_ADD, G_SUB, G_MUL}) |
| .legalIf([=, &ST](const LegalityQuery &Query) { |
| if (CheckTyN(0, Query, {s32})) |
| return true; |
| if (ST.hasMSA() && CheckTyN(0, Query, {v16s8, v8s16, v4s32, v2s64})) |
| return true; |
| return false; |
| }) |
| .clampScalar(0, s32, s32); |
| |
| getActionDefinitionsBuilder({G_UADDO, G_UADDE, G_USUBO, G_USUBE, G_UMULO}) |
| .lowerFor({{s32, s1}}); |
| |
| getActionDefinitionsBuilder(G_UMULH) |
| .legalFor({s32}) |
| .maxScalar(0, s32); |
| |
| getActionDefinitionsBuilder({G_LOAD, G_STORE}) |
| .legalIf([=, &ST](const LegalityQuery &Query) { |
| if (CheckTy0Ty1MemSizeAlign(Query, {{s32, p0, 8, ST.hasMips32r6()}, |
| {s32, p0, 16, ST.hasMips32r6()}, |
| {s32, p0, 32, ST.hasMips32r6()}, |
| {p0, p0, 32, ST.hasMips32r6()}, |
| {s64, p0, 64, ST.hasMips32r6()}})) |
| return true; |
| if (ST.hasMSA() && |
| CheckTy0Ty1MemSizeAlign(Query, {{v16s8, p0, 128, false}, |
| {v8s16, p0, 128, false}, |
| {v4s32, p0, 128, false}, |
| {v2s64, p0, 128, false}})) |
| return true; |
| return false; |
| }) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_IMPLICIT_DEF) |
| .legalFor({s32, s64}); |
| |
| getActionDefinitionsBuilder(G_UNMERGE_VALUES) |
| .legalFor({{s32, s64}}); |
| |
| getActionDefinitionsBuilder(G_MERGE_VALUES) |
| .legalFor({{s64, s32}}); |
| |
| getActionDefinitionsBuilder({G_ZEXTLOAD, G_SEXTLOAD}) |
| .legalForTypesWithMemDesc({{s32, p0, 8, 8}, |
| {s32, p0, 16, 8}}) |
| .clampScalar(0, s32, s32); |
| |
| getActionDefinitionsBuilder({G_ZEXT, G_SEXT}) |
| .legalIf([](const LegalityQuery &Query) { return false; }) |
| .maxScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_TRUNC) |
| .legalIf([](const LegalityQuery &Query) { return false; }) |
| .maxScalar(1, s32); |
| |
| getActionDefinitionsBuilder(G_SELECT) |
| .legalForCartesianProduct({p0, s32, s64}, {s32}) |
| .minScalar(0, s32) |
| .minScalar(1, s32); |
| |
| getActionDefinitionsBuilder(G_BRCOND) |
| .legalFor({s32}) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_BRJT) |
| .legalFor({{p0, s32}}); |
| |
| getActionDefinitionsBuilder(G_BRINDIRECT) |
| .legalFor({p0}); |
| |
| getActionDefinitionsBuilder(G_PHI) |
| .legalFor({p0, s32, s64}) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder({G_AND, G_OR, G_XOR}) |
| .legalFor({s32}) |
| .clampScalar(0, s32, s32); |
| |
| getActionDefinitionsBuilder({G_SDIV, G_SREM, G_UDIV, G_UREM}) |
| .legalIf([=, &ST](const LegalityQuery &Query) { |
| if (CheckTyN(0, Query, {s32})) |
| return true; |
| if (ST.hasMSA() && CheckTyN(0, Query, {v16s8, v8s16, v4s32, v2s64})) |
| return true; |
| return false; |
| }) |
| .minScalar(0, s32) |
| .libcallFor({s64}); |
| |
| getActionDefinitionsBuilder({G_SHL, G_ASHR, G_LSHR}) |
| .legalFor({{s32, s32}}) |
| .clampScalar(1, s32, s32) |
| .clampScalar(0, s32, s32); |
| |
| getActionDefinitionsBuilder(G_ICMP) |
| .legalForCartesianProduct({s32}, {s32, p0}) |
| .clampScalar(1, s32, s32) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_CONSTANT) |
| .legalFor({s32}) |
| .clampScalar(0, s32, s32); |
| |
| getActionDefinitionsBuilder({G_PTR_ADD, G_INTTOPTR}) |
| .legalFor({{p0, s32}}); |
| |
| getActionDefinitionsBuilder(G_PTRTOINT) |
| .legalFor({{s32, p0}}); |
| |
| getActionDefinitionsBuilder(G_FRAME_INDEX) |
| .legalFor({p0}); |
| |
| getActionDefinitionsBuilder({G_GLOBAL_VALUE, G_JUMP_TABLE}) |
| .legalFor({p0}); |
| |
| getActionDefinitionsBuilder(G_DYN_STACKALLOC) |
| .lowerFor({{p0, s32}}); |
| |
| getActionDefinitionsBuilder(G_VASTART) |
| .legalFor({p0}); |
| |
| getActionDefinitionsBuilder(G_BSWAP) |
| .legalIf([=, &ST](const LegalityQuery &Query) { |
| if (ST.hasMips32r2() && CheckTyN(0, Query, {s32})) |
| return true; |
| return false; |
| }) |
| .lowerIf([=, &ST](const LegalityQuery &Query) { |
| if (!ST.hasMips32r2() && CheckTyN(0, Query, {s32})) |
| return true; |
| return false; |
| }) |
| .maxScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_BITREVERSE) |
| .lowerFor({s32}) |
| .maxScalar(0, s32); |
| |
| // FP instructions |
| getActionDefinitionsBuilder(G_FCONSTANT) |
| .legalFor({s32, s64}); |
| |
| getActionDefinitionsBuilder({G_FADD, G_FSUB, G_FMUL, G_FDIV, G_FABS, G_FSQRT}) |
| .legalIf([=, &ST](const LegalityQuery &Query) { |
| if (CheckTyN(0, Query, {s32, s64})) |
| return true; |
| if (ST.hasMSA() && CheckTyN(0, Query, {v16s8, v8s16, v4s32, v2s64})) |
| return true; |
| return false; |
| }); |
| |
| getActionDefinitionsBuilder(G_FCMP) |
| .legalFor({{s32, s32}, {s32, s64}}) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder({G_FCEIL, G_FFLOOR}) |
| .libcallFor({s32, s64}); |
| |
| getActionDefinitionsBuilder(G_FPEXT) |
| .legalFor({{s64, s32}}); |
| |
| getActionDefinitionsBuilder(G_FPTRUNC) |
| .legalFor({{s32, s64}}); |
| |
| // FP to int conversion instructions |
| getActionDefinitionsBuilder(G_FPTOSI) |
| .legalForCartesianProduct({s32}, {s64, s32}) |
| .libcallForCartesianProduct({s64}, {s64, s32}) |
| .minScalar(0, s32); |
| |
| getActionDefinitionsBuilder(G_FPTOUI) |
| .libcallForCartesianProduct({s64}, {s64, s32}) |
| .lowerForCartesianProduct({s32}, {s64, s32}) |
| .minScalar(0, s32); |
| |
| // Int to FP conversion instructions |
| getActionDefinitionsBuilder(G_SITOFP) |
| .legalForCartesianProduct({s64, s32}, {s32}) |
| .libcallForCartesianProduct({s64, s32}, {s64}) |
| .minScalar(1, s32); |
| |
| getActionDefinitionsBuilder(G_UITOFP) |
| .libcallForCartesianProduct({s64, s32}, {s64}) |
| .customForCartesianProduct({s64, s32}, {s32}) |
| .minScalar(1, s32); |
| |
| getActionDefinitionsBuilder(G_SEXT_INREG).lower(); |
| |
| computeTables(); |
| verify(*ST.getInstrInfo()); |
| } |
| |
| bool MipsLegalizerInfo::legalizeCustom(MachineInstr &MI, |
| MachineRegisterInfo &MRI, |
| MachineIRBuilder &MIRBuilder, |
| GISelChangeObserver &Observer) const { |
| |
| using namespace TargetOpcode; |
| |
| MIRBuilder.setInstr(MI); |
| const MipsSubtarget &STI = |
| static_cast<const MipsSubtarget &>(MIRBuilder.getMF().getSubtarget()); |
| const LLT s32 = LLT::scalar(32); |
| const LLT s64 = LLT::scalar(64); |
| |
| switch (MI.getOpcode()) { |
| case G_UITOFP: { |
| Register Dst = MI.getOperand(0).getReg(); |
| Register Src = MI.getOperand(1).getReg(); |
| LLT DstTy = MRI.getType(Dst); |
| LLT SrcTy = MRI.getType(Src); |
| |
| if (SrcTy != s32) |
| return false; |
| if (DstTy != s32 && DstTy != s64) |
| return false; |
| |
| // Let 0xABCDEFGH be given unsigned in MI.getOperand(1). First let's convert |
| // unsigned to double. Mantissa has 52 bits so we use following trick: |
| // First make floating point bit mask 0x43300000ABCDEFGH. |
| // Mask represents 2^52 * 0x1.00000ABCDEFGH i.e. 0x100000ABCDEFGH.0 . |
| // Next, subtract 2^52 * 0x1.0000000000000 i.e. 0x10000000000000.0 from it. |
| // Done. Trunc double to float if needed. |
| |
| MachineInstrBuilder Bitcast = MIRBuilder.buildInstr( |
| STI.isFP64bit() ? Mips::BuildPairF64_64 : Mips::BuildPairF64, {s64}, |
| {Src, MIRBuilder.buildConstant(s32, UINT32_C(0x43300000))}); |
| Bitcast.constrainAllUses(MIRBuilder.getTII(), *STI.getRegisterInfo(), |
| *STI.getRegBankInfo()); |
| |
| MachineInstrBuilder TwoP52FP = MIRBuilder.buildFConstant( |
| s64, BitsToDouble(UINT64_C(0x4330000000000000))); |
| |
| if (DstTy == s64) |
| MIRBuilder.buildFSub(Dst, Bitcast, TwoP52FP); |
| else { |
| MachineInstrBuilder ResF64 = MIRBuilder.buildFSub(s64, Bitcast, TwoP52FP); |
| MIRBuilder.buildFPTrunc(Dst, ResF64); |
| } |
| |
| MI.eraseFromParent(); |
| break; |
| } |
| default: |
| return false; |
| } |
| |
| return true; |
| } |
| |
| static bool SelectMSA3OpIntrinsic(MachineInstr &MI, unsigned Opcode, |
| MachineIRBuilder &MIRBuilder, |
| const MipsSubtarget &ST) { |
| assert(ST.hasMSA() && "MSA intrinsic not supported on target without MSA."); |
| if (!MIRBuilder.buildInstr(Opcode) |
| .add(MI.getOperand(0)) |
| .add(MI.getOperand(2)) |
| .add(MI.getOperand(3)) |
| .constrainAllUses(MIRBuilder.getTII(), *ST.getRegisterInfo(), |
| *ST.getRegBankInfo())) |
| return false; |
| MI.eraseFromParent(); |
| return true; |
| } |
| |
| static bool MSA3OpIntrinsicToGeneric(MachineInstr &MI, unsigned Opcode, |
| MachineIRBuilder &MIRBuilder, |
| const MipsSubtarget &ST) { |
| assert(ST.hasMSA() && "MSA intrinsic not supported on target without MSA."); |
| MIRBuilder.buildInstr(Opcode) |
| .add(MI.getOperand(0)) |
| .add(MI.getOperand(2)) |
| .add(MI.getOperand(3)); |
| MI.eraseFromParent(); |
| return true; |
| } |
| |
| static bool MSA2OpIntrinsicToGeneric(MachineInstr &MI, unsigned Opcode, |
| MachineIRBuilder &MIRBuilder, |
| const MipsSubtarget &ST) { |
| assert(ST.hasMSA() && "MSA intrinsic not supported on target without MSA."); |
| MIRBuilder.buildInstr(Opcode) |
| .add(MI.getOperand(0)) |
| .add(MI.getOperand(2)); |
| MI.eraseFromParent(); |
| return true; |
| } |
| |
| bool MipsLegalizerInfo::legalizeIntrinsic(MachineInstr &MI, |
| MachineRegisterInfo &MRI, |
| MachineIRBuilder &MIRBuilder) const { |
| const MipsSubtarget &ST = |
| static_cast<const MipsSubtarget &>(MI.getMF()->getSubtarget()); |
| const MipsInstrInfo &TII = *ST.getInstrInfo(); |
| const MipsRegisterInfo &TRI = *ST.getRegisterInfo(); |
| const RegisterBankInfo &RBI = *ST.getRegBankInfo(); |
| MIRBuilder.setInstr(MI); |
| |
| switch (MI.getIntrinsicID()) { |
| case Intrinsic::memcpy: |
| case Intrinsic::memset: |
| case Intrinsic::memmove: |
| if (createMemLibcall(MIRBuilder, MRI, MI) == |
| LegalizerHelper::UnableToLegalize) |
| return false; |
| MI.eraseFromParent(); |
| return true; |
| case Intrinsic::trap: { |
| MachineInstr *Trap = MIRBuilder.buildInstr(Mips::TRAP); |
| MI.eraseFromParent(); |
| return constrainSelectedInstRegOperands(*Trap, TII, TRI, RBI); |
| } |
| case Intrinsic::vacopy: { |
| Register Tmp = MRI.createGenericVirtualRegister(LLT::pointer(0, 32)); |
| MachinePointerInfo MPO; |
| MIRBuilder.buildLoad(Tmp, MI.getOperand(2), |
| *MI.getMF()->getMachineMemOperand( |
| MPO, MachineMemOperand::MOLoad, 4, 4)); |
| MIRBuilder.buildStore(Tmp, MI.getOperand(1), |
| *MI.getMF()->getMachineMemOperand( |
| MPO, MachineMemOperand::MOStore, 4, 4)); |
| MI.eraseFromParent(); |
| return true; |
| } |
| case Intrinsic::mips_addv_b: |
| case Intrinsic::mips_addv_h: |
| case Intrinsic::mips_addv_w: |
| case Intrinsic::mips_addv_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_ADD, MIRBuilder, ST); |
| case Intrinsic::mips_addvi_b: |
| return SelectMSA3OpIntrinsic(MI, Mips::ADDVI_B, MIRBuilder, ST); |
| case Intrinsic::mips_addvi_h: |
| return SelectMSA3OpIntrinsic(MI, Mips::ADDVI_H, MIRBuilder, ST); |
| case Intrinsic::mips_addvi_w: |
| return SelectMSA3OpIntrinsic(MI, Mips::ADDVI_W, MIRBuilder, ST); |
| case Intrinsic::mips_addvi_d: |
| return SelectMSA3OpIntrinsic(MI, Mips::ADDVI_D, MIRBuilder, ST); |
| case Intrinsic::mips_subv_b: |
| case Intrinsic::mips_subv_h: |
| case Intrinsic::mips_subv_w: |
| case Intrinsic::mips_subv_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_SUB, MIRBuilder, ST); |
| case Intrinsic::mips_subvi_b: |
| return SelectMSA3OpIntrinsic(MI, Mips::SUBVI_B, MIRBuilder, ST); |
| case Intrinsic::mips_subvi_h: |
| return SelectMSA3OpIntrinsic(MI, Mips::SUBVI_H, MIRBuilder, ST); |
| case Intrinsic::mips_subvi_w: |
| return SelectMSA3OpIntrinsic(MI, Mips::SUBVI_W, MIRBuilder, ST); |
| case Intrinsic::mips_subvi_d: |
| return SelectMSA3OpIntrinsic(MI, Mips::SUBVI_D, MIRBuilder, ST); |
| case Intrinsic::mips_mulv_b: |
| case Intrinsic::mips_mulv_h: |
| case Intrinsic::mips_mulv_w: |
| case Intrinsic::mips_mulv_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_MUL, MIRBuilder, ST); |
| case Intrinsic::mips_div_s_b: |
| case Intrinsic::mips_div_s_h: |
| case Intrinsic::mips_div_s_w: |
| case Intrinsic::mips_div_s_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_SDIV, MIRBuilder, ST); |
| case Intrinsic::mips_mod_s_b: |
| case Intrinsic::mips_mod_s_h: |
| case Intrinsic::mips_mod_s_w: |
| case Intrinsic::mips_mod_s_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_SREM, MIRBuilder, ST); |
| case Intrinsic::mips_div_u_b: |
| case Intrinsic::mips_div_u_h: |
| case Intrinsic::mips_div_u_w: |
| case Intrinsic::mips_div_u_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_UDIV, MIRBuilder, ST); |
| case Intrinsic::mips_mod_u_b: |
| case Intrinsic::mips_mod_u_h: |
| case Intrinsic::mips_mod_u_w: |
| case Intrinsic::mips_mod_u_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_UREM, MIRBuilder, ST); |
| case Intrinsic::mips_fadd_w: |
| case Intrinsic::mips_fadd_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_FADD, MIRBuilder, ST); |
| case Intrinsic::mips_fsub_w: |
| case Intrinsic::mips_fsub_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_FSUB, MIRBuilder, ST); |
| case Intrinsic::mips_fmul_w: |
| case Intrinsic::mips_fmul_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_FMUL, MIRBuilder, ST); |
| case Intrinsic::mips_fdiv_w: |
| case Intrinsic::mips_fdiv_d: |
| return MSA3OpIntrinsicToGeneric(MI, TargetOpcode::G_FDIV, MIRBuilder, ST); |
| case Intrinsic::mips_fmax_a_w: |
| return SelectMSA3OpIntrinsic(MI, Mips::FMAX_A_W, MIRBuilder, ST); |
| case Intrinsic::mips_fmax_a_d: |
| return SelectMSA3OpIntrinsic(MI, Mips::FMAX_A_D, MIRBuilder, ST); |
| case Intrinsic::mips_fsqrt_w: |
| return MSA2OpIntrinsicToGeneric(MI, TargetOpcode::G_FSQRT, MIRBuilder, ST); |
| case Intrinsic::mips_fsqrt_d: |
| return MSA2OpIntrinsicToGeneric(MI, TargetOpcode::G_FSQRT, MIRBuilder, ST); |
| default: |
| break; |
| } |
| return true; |
| } |