Index: llvm/lib/Target/Mips/CMakeLists.txt =================================================================== --- llvm/lib/Target/Mips/CMakeLists.txt +++ llvm/lib/Target/Mips/CMakeLists.txt @@ -1,6 +1,9 @@ add_llvm_component_group(Mips HAS_JIT) set(LLVM_TARGET_DEFINITIONS Mips.td) +tablegen(LLVM MipsGenGlobalISel.inc -gen-global-isel) +tablegen(LLVM MipsGenPostLegalizeGICombiner.inc -gen-global-isel-combiner + -combiners="MipsPostLegalizerCombinerHelper") tablegen(LLVM MipsGenAsmMatcher.inc -gen-asm-matcher) tablegen(LLVM MipsGenAsmWriter.inc -gen-asm-writer) @@ -8,7 +11,6 @@ tablegen(LLVM MipsGenDAGISel.inc -gen-dag-isel) tablegen(LLVM MipsGenDisassemblerTables.inc -gen-disassembler) tablegen(LLVM MipsGenFastISel.inc -gen-fast-isel) -tablegen(LLVM MipsGenGlobalISel.inc -gen-global-isel) tablegen(LLVM MipsGenInstrInfo.inc -gen-instr-info) tablegen(LLVM MipsGenMCCodeEmitter.inc -gen-emitter) tablegen(LLVM MipsGenMCPseudoLowering.inc -gen-pseudo-lowering) @@ -48,6 +50,7 @@ MipsOptimizePICCall.cpp MipsOs16.cpp MipsPreLegalizerCombiner.cpp + MipsPostLegalizerCombiner.cpp MipsRegisterBankInfo.cpp MipsRegisterInfo.cpp MipsSEFrameLowering.cpp Index: llvm/lib/Target/Mips/Mips.h =================================================================== --- llvm/lib/Target/Mips/Mips.h +++ llvm/lib/Target/Mips/Mips.h @@ -38,6 +38,7 @@ FunctionPass *createMicroMipsSizeReducePass(); FunctionPass *createMipsExpandPseudoPass(); FunctionPass *createMipsPreLegalizeCombiner(); + FunctionPass *createMipsPostLegalizeCombiner(bool IsOptNone); FunctionPass *createMipsMulMulBugPass(); InstructionSelector *createMipsInstructionSelector(const MipsTargetMachine &, @@ -48,6 +49,7 @@ void initializeMipsBranchExpansionPass(PassRegistry &); void initializeMicroMipsSizeReducePass(PassRegistry &); void initializeMipsPreLegalizerCombinerPass(PassRegistry&); + void initializeMipsPostLegalizerCombinerPass(PassRegistry &); void initializeMipsMulMulBugFixPass(PassRegistry&); } // end namespace llvm; Index: llvm/lib/Target/Mips/Mips.td =================================================================== --- llvm/lib/Target/Mips/Mips.td +++ llvm/lib/Target/Mips/Mips.td @@ -217,6 +217,7 @@ include "MipsInstrInfo.td" include "MipsCallingConv.td" include "MipsRegisterBanks.td" +include "MipsCombine.td" // Avoid forward declaration issues. include "MipsScheduleP5600.td" Index: llvm/lib/Target/Mips/MipsCombine.td =================================================================== --- /dev/null +++ llvm/lib/Target/Mips/MipsCombine.td @@ -0,0 +1,15 @@ +//=- MipsCombine.td - Define Mips Combine Rules --------------*- 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 +// +//===----------------------------------------------------------------------===// + +include "llvm/Target/GlobalISel/Combine.td" + +def MipsPostLegalizerCombinerHelper: GICombinerHelper< + "MipsGenPostLegalizerCombinerHelper", []> { + let DisableRuleOption = "mipspostlegalizercombiner-disable-rule"; +} + Index: llvm/lib/Target/Mips/MipsPostLegalizerCombiner.cpp =================================================================== --- /dev/null +++ llvm/lib/Target/Mips/MipsPostLegalizerCombiner.cpp @@ -0,0 +1,148 @@ +//=== lib/CodeGen/GlobalISel/MipsPostLegalizerCombiner.cpp ----------------===// +// +// 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 pass does combining of machine instructions at the generic MI level, +// after the legalizer. +// +//===----------------------------------------------------------------------===// + +#include "MCTargetDesc/MipsMCTargetDesc.h" +#include "Mips.h" +#include "MipsLegalizerInfo.h" +#include "MipsSubtarget.h" +#include "llvm/CodeGen/GlobalISel/Combiner.h" +#include "llvm/CodeGen/GlobalISel/CombinerHelper.h" +#include "llvm/CodeGen/GlobalISel/CombinerInfo.h" +#include "llvm/CodeGen/GlobalISel/GISelKnownBits.h" +#include "llvm/CodeGen/GlobalISel/MIPatternMatch.h" +#include "llvm/CodeGen/MachineDominators.h" +#include "llvm/CodeGen/TargetPassConfig.h" +#include "llvm/Target/TargetMachine.h" + +#define DEBUG_TYPE "mips-postlegalizer-combiner" + +using namespace llvm; +using namespace MIPatternMatch; + +#define MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_DEPS +#include "MipsGenPostLegalizeGICombiner.inc" +#undef MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_DEPS + +namespace { +#define MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_H +#include "MipsGenPostLegalizeGICombiner.inc" +#undef MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_H + +class MipsPostLegalizerCombinerInfo final : public CombinerInfo { + GISelKnownBits *KB; + +public: + MipsGenPostLegalizerCombinerHelperRuleConfig GeneratedRuleCfg; + + MipsPostLegalizerCombinerInfo(bool EnableOpt, bool OptSize, bool MinSize, + GISelKnownBits *KB, const MipsLegalizerInfo *LI) + : CombinerInfo(/*AllowIllegalOps*/ false, /*ShouldLegalizeIllegal*/ true, + /*LegalizerInfo*/ LI, EnableOpt, OptSize, MinSize), + KB(KB) { + if (!GeneratedRuleCfg.parseCommandLineOption()) + report_fatal_error("Invalid rule identifier"); + } + + bool combine(GISelChangeObserver &Observer, MachineInstr &MI, + MachineIRBuilder &B) const override; +}; + +bool MipsPostLegalizerCombinerInfo::combine(GISelChangeObserver &Observer, + MachineInstr &MI, + MachineIRBuilder &B) const { + + CombinerHelper Helper(Observer, B, KB, + /*DominatorTree*/ nullptr, LInfo); + MipsGenPostLegalizerCombinerHelper Generated(GeneratedRuleCfg, Helper); + return Generated.tryCombineAll(Observer, MI, B, Helper); +} + +#define MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_CPP +#include "MipsGenPostLegalizeGICombiner.inc" +#undef MIPSPOSTLEGALIZERCOMBINERHELPER_GENCOMBINERHELPER_CPP + +// Pass boilerplate +// ================ + +class MipsPostLegalizerCombiner : public MachineFunctionPass { +public: + static char ID; + + MipsPostLegalizerCombiner(bool IsOptNone = false); + + StringRef getPassName() const override { + return "MipsPostLegalizerCombiner"; + } + + bool runOnMachineFunction(MachineFunction &MF) override; + + void getAnalysisUsage(AnalysisUsage &AU) const override; + +private: + bool IsOptNone; +}; +} // end anonymous namespace + +void MipsPostLegalizerCombiner::getAnalysisUsage(AnalysisUsage &AU) const { + AU.addRequired(); + AU.setPreservesCFG(); + getSelectionDAGFallbackAnalysisUsage(AU); + AU.addRequired(); + AU.addPreserved(); + if (!IsOptNone) { + AU.addRequired(); + AU.addPreserved(); + } + MachineFunctionPass::getAnalysisUsage(AU); +} + +MipsPostLegalizerCombiner::MipsPostLegalizerCombiner(bool IsOptNone) + : MachineFunctionPass(ID), IsOptNone(IsOptNone) { + initializeMipsPostLegalizerCombinerPass(*PassRegistry::getPassRegistry()); +} + +bool MipsPostLegalizerCombiner::runOnMachineFunction(MachineFunction &MF) { + if (MF.getProperties().hasProperty( + MachineFunctionProperties::Property::FailedISel)) + return false; + auto *TPC = &getAnalysis(); + const Function &F = MF.getFunction(); + bool EnableOpt = + MF.getTarget().getOptLevel() != CodeGenOpt::None && !skipFunction(F); + + const MipsSubtarget &ST = MF.getSubtarget(); + const MipsLegalizerInfo *LI = + static_cast(ST.getLegalizerInfo()); + + GISelKnownBits *KB = &getAnalysis().get(MF); + MipsPostLegalizerCombinerInfo PCInfo(EnableOpt, F.hasOptSize(), + F.hasMinSize(), KB, LI); + Combiner C(PCInfo, TPC); + return C.combineMachineInstrs(MF, /*CSEInfo*/ nullptr); +} + +char MipsPostLegalizerCombiner::ID = 0; +INITIALIZE_PASS_BEGIN(MipsPostLegalizerCombiner, DEBUG_TYPE, + "Combine Mips machine instrs after legalization", false, + false) +INITIALIZE_PASS_DEPENDENCY(TargetPassConfig) +INITIALIZE_PASS_DEPENDENCY(GISelKnownBitsAnalysis) +INITIALIZE_PASS_END(MipsPostLegalizerCombiner, DEBUG_TYPE, + "Combine Mips machine instrs after legalization", false, + false) + +namespace llvm { +FunctionPass *createMipsPostLegalizeCombiner(bool IsOptNone) { + return new MipsPostLegalizerCombiner(IsOptNone); +} +} // end namespace llvm Index: llvm/lib/Target/Mips/MipsTargetMachine.cpp =================================================================== --- llvm/lib/Target/Mips/MipsTargetMachine.cpp +++ llvm/lib/Target/Mips/MipsTargetMachine.cpp @@ -63,6 +63,7 @@ initializeMipsBranchExpansionPass(*PR); initializeMicroMipsSizeReducePass(*PR); initializeMipsPreLegalizerCombinerPass(*PR); + initializeMipsPostLegalizerCombinerPass(*PR); initializeMipsMulMulBugFixPass(*PR); } @@ -239,6 +240,7 @@ bool addIRTranslator() override; void addPreLegalizeMachineIR() override; bool addLegalizeMachineIR() override; + void addPreRegBankSelect() override; bool addRegBankSelect() override; bool addGlobalInstructionSelect() override; @@ -334,6 +336,11 @@ return false; } +void MipsPassConfig::addPreRegBankSelect() { + bool IsOptNone = getOptLevel() == CodeGenOpt::None; + addPass(createMipsPostLegalizeCombiner(IsOptNone)); +} + bool MipsPassConfig::addRegBankSelect() { addPass(new RegBankSelect()); return false;