You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			203 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			203 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===-- WebAssemblyStoreResults.cpp - Optimize using store result values --===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | ///
 | ||
|  | /// \file
 | ||
|  | /// \brief This file implements an optimization pass using store result values.
 | ||
|  | ///
 | ||
|  | /// WebAssembly's store instructions return the stored value. This is to enable
 | ||
|  | /// an optimization wherein uses of the stored value can be replaced by uses of
 | ||
|  | /// the store's result value, making the stored value register more likely to
 | ||
|  | /// be single-use, thus more likely to be useful to register stackifying, and
 | ||
|  | /// potentially also exposing the store to register stackifying. These both can
 | ||
|  | /// reduce get_local/set_local traffic.
 | ||
|  | ///
 | ||
|  | /// This pass also performs this optimization for memcpy, memmove, and memset
 | ||
|  | /// calls, since the LLVM intrinsics for these return void so they can't use the
 | ||
|  | /// returned attribute and consequently aren't handled by the OptimizeReturned
 | ||
|  | /// pass.
 | ||
|  | ///
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
 | ||
|  | #include "WebAssembly.h"
 | ||
|  | #include "WebAssemblyMachineFunctionInfo.h"
 | ||
|  | #include "WebAssemblySubtarget.h"
 | ||
|  | #include "llvm/Analysis/TargetLibraryInfo.h"
 | ||
|  | #include "llvm/CodeGen/LiveIntervals.h"
 | ||
|  | #include "llvm/CodeGen/MachineBlockFrequencyInfo.h"
 | ||
|  | #include "llvm/CodeGen/MachineDominators.h"
 | ||
|  | #include "llvm/CodeGen/MachineRegisterInfo.h"
 | ||
|  | #include "llvm/CodeGen/Passes.h"
 | ||
|  | #include "llvm/Support/Debug.h"
 | ||
|  | #include "llvm/Support/raw_ostream.h"
 | ||
|  | using namespace llvm; | ||
|  | 
 | ||
|  | #define DEBUG_TYPE "wasm-store-results"
 | ||
|  | 
 | ||
|  | namespace { | ||
|  | class WebAssemblyStoreResults final : public MachineFunctionPass { | ||
|  | public: | ||
|  |   static char ID; // Pass identification, replacement for typeid
 | ||
|  |   WebAssemblyStoreResults() : MachineFunctionPass(ID) {} | ||
|  | 
 | ||
|  |   StringRef getPassName() const override { return "WebAssembly Store Results"; } | ||
|  | 
 | ||
|  |   void getAnalysisUsage(AnalysisUsage &AU) const override { | ||
|  |     AU.setPreservesCFG(); | ||
|  |     AU.addRequired<MachineBlockFrequencyInfo>(); | ||
|  |     AU.addPreserved<MachineBlockFrequencyInfo>(); | ||
|  |     AU.addRequired<MachineDominatorTree>(); | ||
|  |     AU.addPreserved<MachineDominatorTree>(); | ||
|  |     AU.addRequired<LiveIntervals>(); | ||
|  |     AU.addPreserved<SlotIndexes>(); | ||
|  |     AU.addPreserved<LiveIntervals>(); | ||
|  |     AU.addRequired<TargetLibraryInfoWrapperPass>(); | ||
|  |     MachineFunctionPass::getAnalysisUsage(AU); | ||
|  |   } | ||
|  | 
 | ||
|  |   bool runOnMachineFunction(MachineFunction &MF) override; | ||
|  | 
 | ||
|  | private: | ||
|  | }; | ||
|  | } // end anonymous namespace
 | ||
|  | 
 | ||
|  | char WebAssemblyStoreResults::ID = 0; | ||
|  | FunctionPass *llvm::createWebAssemblyStoreResults() { | ||
|  |   return new WebAssemblyStoreResults(); | ||
|  | } | ||
|  | 
 | ||
|  | // Replace uses of FromReg with ToReg if they are dominated by MI.
 | ||
|  | static bool ReplaceDominatedUses(MachineBasicBlock &MBB, MachineInstr &MI, | ||
|  |                                  unsigned FromReg, unsigned ToReg, | ||
|  |                                  const MachineRegisterInfo &MRI, | ||
|  |                                  MachineDominatorTree &MDT, | ||
|  |                                  LiveIntervals &LIS) { | ||
|  |   bool Changed = false; | ||
|  | 
 | ||
|  |   LiveInterval *FromLI = &LIS.getInterval(FromReg); | ||
|  |   LiveInterval *ToLI = &LIS.getInterval(ToReg); | ||
|  | 
 | ||
|  |   SlotIndex FromIdx = LIS.getInstructionIndex(MI).getRegSlot(); | ||
|  |   VNInfo *FromVNI = FromLI->getVNInfoAt(FromIdx); | ||
|  | 
 | ||
|  |   SmallVector<SlotIndex, 4> Indices; | ||
|  | 
 | ||
|  |   for (auto I = MRI.use_nodbg_begin(FromReg), E = MRI.use_nodbg_end(); I != E;) { | ||
|  |     MachineOperand &O = *I++; | ||
|  |     MachineInstr *Where = O.getParent(); | ||
|  | 
 | ||
|  |     // Check that MI dominates the instruction in the normal way.
 | ||
|  |     if (&MI == Where || !MDT.dominates(&MI, Where)) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // If this use gets a different value, skip it.
 | ||
|  |     SlotIndex WhereIdx = LIS.getInstructionIndex(*Where); | ||
|  |     VNInfo *WhereVNI = FromLI->getVNInfoAt(WhereIdx); | ||
|  |     if (WhereVNI && WhereVNI != FromVNI) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Make sure ToReg isn't clobbered before it gets there.
 | ||
|  |     VNInfo *ToVNI = ToLI->getVNInfoAt(WhereIdx); | ||
|  |     if (ToVNI && ToVNI != FromVNI) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     Changed = true; | ||
|  |     DEBUG(dbgs() << "Setting operand " << O << " in " << *Where << " from " | ||
|  |                  << MI << "\n"); | ||
|  |     O.setReg(ToReg); | ||
|  | 
 | ||
|  |     // If the store's def was previously dead, it is no longer.
 | ||
|  |     if (!O.isUndef()) { | ||
|  |       MI.getOperand(0).setIsDead(false); | ||
|  | 
 | ||
|  |       Indices.push_back(WhereIdx.getRegSlot()); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (Changed) { | ||
|  |     // Extend ToReg's liveness.
 | ||
|  |     LIS.extendToIndices(*ToLI, Indices); | ||
|  | 
 | ||
|  |     // Shrink FromReg's liveness.
 | ||
|  |     LIS.shrinkToUses(FromLI); | ||
|  | 
 | ||
|  |     // If we replaced all dominated uses, FromReg is now killed at MI.
 | ||
|  |     if (!FromLI->liveAt(FromIdx.getDeadSlot())) | ||
|  |       MI.addRegisterKilled(FromReg, | ||
|  |                            MBB.getParent()->getSubtarget<WebAssemblySubtarget>() | ||
|  |                                  .getRegisterInfo()); | ||
|  |   } | ||
|  | 
 | ||
|  |   return Changed; | ||
|  | } | ||
|  | 
 | ||
|  | static bool optimizeCall(MachineBasicBlock &MBB, MachineInstr &MI, | ||
|  |                          const MachineRegisterInfo &MRI, | ||
|  |                          MachineDominatorTree &MDT, | ||
|  |                          LiveIntervals &LIS, | ||
|  |                          const WebAssemblyTargetLowering &TLI, | ||
|  |                          const TargetLibraryInfo &LibInfo) { | ||
|  |   MachineOperand &Op1 = MI.getOperand(1); | ||
|  |   if (!Op1.isSymbol()) | ||
|  |     return false; | ||
|  | 
 | ||
|  |   StringRef Name(Op1.getSymbolName()); | ||
|  |   bool callReturnsInput = Name == TLI.getLibcallName(RTLIB::MEMCPY) || | ||
|  |                           Name == TLI.getLibcallName(RTLIB::MEMMOVE) || | ||
|  |                           Name == TLI.getLibcallName(RTLIB::MEMSET); | ||
|  |   if (!callReturnsInput) | ||
|  |     return false; | ||
|  | 
 | ||
|  |   LibFunc Func; | ||
|  |   if (!LibInfo.getLibFunc(Name, Func)) | ||
|  |     return false; | ||
|  | 
 | ||
|  |   unsigned FromReg = MI.getOperand(2).getReg(); | ||
|  |   unsigned ToReg = MI.getOperand(0).getReg(); | ||
|  |   if (MRI.getRegClass(FromReg) != MRI.getRegClass(ToReg)) | ||
|  |     report_fatal_error("Store results: call to builtin function with wrong " | ||
|  |                        "signature, from/to mismatch"); | ||
|  |   return ReplaceDominatedUses(MBB, MI, FromReg, ToReg, MRI, MDT, LIS); | ||
|  | } | ||
|  | 
 | ||
|  | bool WebAssemblyStoreResults::runOnMachineFunction(MachineFunction &MF) { | ||
|  |   DEBUG({ | ||
|  |     dbgs() << "********** Store Results **********\n" | ||
|  |            << "********** Function: " << MF.getName() << '\n'; | ||
|  |   }); | ||
|  | 
 | ||
|  |   MachineRegisterInfo &MRI = MF.getRegInfo(); | ||
|  |   MachineDominatorTree &MDT = getAnalysis<MachineDominatorTree>(); | ||
|  |   const WebAssemblyTargetLowering &TLI = | ||
|  |       *MF.getSubtarget<WebAssemblySubtarget>().getTargetLowering(); | ||
|  |   const auto &LibInfo = getAnalysis<TargetLibraryInfoWrapperPass>().getTLI(); | ||
|  |   LiveIntervals &LIS = getAnalysis<LiveIntervals>(); | ||
|  |   bool Changed = false; | ||
|  | 
 | ||
|  |   // We don't preserve SSA form.
 | ||
|  |   MRI.leaveSSA(); | ||
|  | 
 | ||
|  |   assert(MRI.tracksLiveness() && "StoreResults expects liveness tracking"); | ||
|  | 
 | ||
|  |   for (auto &MBB : MF) { | ||
|  |     DEBUG(dbgs() << "Basic Block: " << MBB.getName() << '\n'); | ||
|  |     for (auto &MI : MBB) | ||
|  |       switch (MI.getOpcode()) { | ||
|  |       default: | ||
|  |         break; | ||
|  |       case WebAssembly::CALL_I32: | ||
|  |       case WebAssembly::CALL_I64: | ||
|  |         Changed |= optimizeCall(MBB, MI, MRI, MDT, LIS, TLI, LibInfo); | ||
|  |         break; | ||
|  |       } | ||
|  |   } | ||
|  | 
 | ||
|  |   return Changed; | ||
|  | } |