diff --git a/llvm/include/llvm/CodeGen/MachinePassRegistry.def b/llvm/include/llvm/CodeGen/MachinePassRegistry.def --- a/llvm/include/llvm/CodeGen/MachinePassRegistry.def +++ b/llvm/include/llvm/CodeGen/MachinePassRegistry.def @@ -199,4 +199,5 @@ DUMMY_MACHINE_FUNCTION_PASS("machineverifier", MachineVerifierPass, ()) DUMMY_MACHINE_FUNCTION_PASS("machine-cycles", MachineCycleInfoWrapperPass, ()) DUMMY_MACHINE_FUNCTION_PASS("print-machine-cycles", MachineCycleInfoPrinterPass, ()) +DUMMY_MACHINE_FUNCTION_PASS("spill2reg", Spill2RegPass, ()) #undef DUMMY_MACHINE_FUNCTION_PASS diff --git a/llvm/include/llvm/CodeGen/Passes.h b/llvm/include/llvm/CodeGen/Passes.h --- a/llvm/include/llvm/CodeGen/Passes.h +++ b/llvm/include/llvm/CodeGen/Passes.h @@ -554,6 +554,9 @@ /// When learning an eviction policy, extract score(reward) information, /// otherwise this does nothing FunctionPass *createRegAllocScoringPass(); + + /// This pass replaces spills to stack with spills to registers. + extern char &Spill2RegID; } // End llvm namespace #endif diff --git a/llvm/include/llvm/InitializePasses.h b/llvm/include/llvm/InitializePasses.h --- a/llvm/include/llvm/InitializePasses.h +++ b/llvm/include/llvm/InitializePasses.h @@ -463,6 +463,7 @@ void initializeWriteBitcodePassPass(PassRegistry&); void initializeWriteThinLTOBitcodePass(PassRegistry&); void initializeXRayInstrumentationPass(PassRegistry&); +void initializeSpill2RegPass(PassRegistry &); } // end namespace llvm diff --git a/llvm/lib/CodeGen/CMakeLists.txt b/llvm/lib/CodeGen/CMakeLists.txt --- a/llvm/lib/CodeGen/CMakeLists.txt +++ b/llvm/lib/CodeGen/CMakeLists.txt @@ -198,6 +198,7 @@ SjLjEHPrepare.cpp SlotIndexes.cpp SpillPlacement.cpp + Spill2Reg.cpp SplitKit.cpp StackColoring.cpp StackMapLivenessAnalysis.cpp diff --git a/llvm/lib/CodeGen/CodeGen.cpp b/llvm/lib/CodeGen/CodeGen.cpp --- a/llvm/lib/CodeGen/CodeGen.cpp +++ b/llvm/lib/CodeGen/CodeGen.cpp @@ -125,6 +125,7 @@ initializeWasmEHPreparePass(Registry); initializeWinEHPreparePass(Registry); initializeXRayInstrumentationPass(Registry); + initializeSpill2RegPass(Registry); } void LLVMInitializeCodeGen(LLVMPassRegistryRef R) { diff --git a/llvm/lib/CodeGen/Spill2Reg.cpp b/llvm/lib/CodeGen/Spill2Reg.cpp new file mode 100644 --- /dev/null +++ b/llvm/lib/CodeGen/Spill2Reg.cpp @@ -0,0 +1,66 @@ +//===- Spill2Reg.cpp - Spill To Register Optimization ---------------------===// +// +// 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 Spill2Reg, an optimization which selectively +/// replaces spills/reloads to/from the stack with register copies to/from the +/// vector register file. This works even on targets where load/stores have +/// similar latency to register copies because it can free up memory units which +/// helps avoid back-end stalls. +/// +//===----------------------------------------------------------------------===// + +#include "llvm/CodeGen/MachineFunctionPass.h" +#include "llvm/CodeGen/Passes.h" +#include "llvm/InitializePasses.h" +#include "llvm/Support/CommandLine.h" + +using namespace llvm; + +static cl::opt<bool> EnableSpill2Reg("enable-spill2reg", cl::Hidden, + cl::init(true), + cl::desc("Enable Spill2Reg pass")); + +namespace { + +class Spill2Reg : public MachineFunctionPass { +public: + static char ID; + Spill2Reg() : MachineFunctionPass(ID) { + initializeSpill2RegPass(*PassRegistry::getPassRegistry()); + } + void getAnalysisUsage(AnalysisUsage &AU) const override; + void releaseMemory() override; + bool runOnMachineFunction(MachineFunction &) override; + void print(raw_ostream &O, const Module * = nullptr) const override; +}; + +} // namespace + +void Spill2Reg::getAnalysisUsage(AnalysisUsage &AU) const { + AU.setPreservesCFG(); + MachineFunctionPass::getAnalysisUsage(AU); +} + +void Spill2Reg::releaseMemory() {} + +bool Spill2Reg::runOnMachineFunction(MachineFunction &MFn) { + if (!EnableSpill2Reg) + return false; + + llvm_unreachable("Unimplemented"); +} + +void Spill2Reg::print(raw_ostream &O, const Module *m) const { ; } + +char Spill2Reg::ID = 0; + +char &llvm::Spill2RegID = Spill2Reg::ID; + +INITIALIZE_PASS_BEGIN(Spill2Reg, "spill2reg", "Spill2Reg", false, false) +INITIALIZE_PASS_END(Spill2Reg, "spill2reg", "Spill2Reg", false, false) diff --git a/llvm/lib/CodeGen/TargetPassConfig.cpp b/llvm/lib/CodeGen/TargetPassConfig.cpp --- a/llvm/lib/CodeGen/TargetPassConfig.cpp +++ b/llvm/lib/CodeGen/TargetPassConfig.cpp @@ -1399,6 +1399,9 @@ // Finally rewrite virtual registers. addPass(&VirtRegRewriterID); + // Replace spills to stack with spills to registers. + addPass(&Spill2RegID); + // Regalloc scoring for ML-driven eviction - noop except when learning a new // eviction policy. addPass(createRegAllocScoringPass());