mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-04 05:17:07 +00:00 
			
		
		
		
	git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@126002 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			130 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			130 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//===--- LiveRangeEdit.cpp - Basic tools for editing a register live range --===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// The LiveRangeEdit class represents changes done to a virtual register when it
 | 
						|
// is spilled or split.
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "LiveRangeEdit.h"
 | 
						|
#include "VirtRegMap.h"
 | 
						|
#include "llvm/CodeGen/LiveIntervalAnalysis.h"
 | 
						|
#include "llvm/CodeGen/MachineRegisterInfo.h"
 | 
						|
#include "llvm/Target/TargetInstrInfo.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
 | 
						|
LiveInterval &LiveRangeEdit::create(MachineRegisterInfo &mri,
 | 
						|
                                    LiveIntervals &lis,
 | 
						|
                                    VirtRegMap &vrm) {
 | 
						|
  const TargetRegisterClass *RC = mri.getRegClass(getReg());
 | 
						|
  unsigned VReg = mri.createVirtualRegister(RC);
 | 
						|
  vrm.grow();
 | 
						|
  vrm.setIsSplitFromReg(VReg, vrm.getOriginal(getReg()));
 | 
						|
  LiveInterval &li = lis.getOrCreateInterval(VReg);
 | 
						|
  newRegs_.push_back(&li);
 | 
						|
  return li;
 | 
						|
}
 | 
						|
 | 
						|
void LiveRangeEdit::scanRemattable(LiveIntervals &lis,
 | 
						|
                                   const TargetInstrInfo &tii,
 | 
						|
                                   AliasAnalysis *aa) {
 | 
						|
  for (LiveInterval::vni_iterator I = parent_.vni_begin(),
 | 
						|
       E = parent_.vni_end(); I != E; ++I) {
 | 
						|
    VNInfo *VNI = *I;
 | 
						|
    if (VNI->isUnused())
 | 
						|
      continue;
 | 
						|
    MachineInstr *DefMI = lis.getInstructionFromIndex(VNI->def);
 | 
						|
    if (!DefMI)
 | 
						|
      continue;
 | 
						|
    if (tii.isTriviallyReMaterializable(DefMI, aa))
 | 
						|
      remattable_.insert(VNI);
 | 
						|
  }
 | 
						|
  scannedRemattable_ = true;
 | 
						|
}
 | 
						|
 | 
						|
bool LiveRangeEdit::anyRematerializable(LiveIntervals &lis,
 | 
						|
                                        const TargetInstrInfo &tii,
 | 
						|
                                        AliasAnalysis *aa) {
 | 
						|
  if (!scannedRemattable_)
 | 
						|
    scanRemattable(lis, tii, aa);
 | 
						|
  return !remattable_.empty();
 | 
						|
}
 | 
						|
 | 
						|
/// allUsesAvailableAt - Return true if all registers used by OrigMI at
 | 
						|
/// OrigIdx are also available with the same value at UseIdx.
 | 
						|
bool LiveRangeEdit::allUsesAvailableAt(const MachineInstr *OrigMI,
 | 
						|
                                       SlotIndex OrigIdx,
 | 
						|
                                       SlotIndex UseIdx,
 | 
						|
                                       LiveIntervals &lis) {
 | 
						|
  OrigIdx = OrigIdx.getUseIndex();
 | 
						|
  UseIdx = UseIdx.getUseIndex();
 | 
						|
  for (unsigned i = 0, e = OrigMI->getNumOperands(); i != e; ++i) {
 | 
						|
    const MachineOperand &MO = OrigMI->getOperand(i);
 | 
						|
    if (!MO.isReg() || !MO.getReg() || MO.getReg() == getReg())
 | 
						|
      continue;
 | 
						|
    // Reserved registers are OK.
 | 
						|
    if (MO.isUndef() || !lis.hasInterval(MO.getReg()))
 | 
						|
      continue;
 | 
						|
    // We don't want to move any defs.
 | 
						|
    if (MO.isDef())
 | 
						|
      return false;
 | 
						|
    // We cannot depend on virtual registers in uselessRegs_.
 | 
						|
    for (unsigned ui = 0, ue = uselessRegs_.size(); ui != ue; ++ui)
 | 
						|
      if (uselessRegs_[ui]->reg == MO.getReg())
 | 
						|
        return false;
 | 
						|
 | 
						|
    LiveInterval &li = lis.getInterval(MO.getReg());
 | 
						|
    const VNInfo *OVNI = li.getVNInfoAt(OrigIdx);
 | 
						|
    if (!OVNI)
 | 
						|
      continue;
 | 
						|
    if (OVNI != li.getVNInfoAt(UseIdx))
 | 
						|
      return false;
 | 
						|
  }
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
bool LiveRangeEdit::canRematerializeAt(Remat &RM,
 | 
						|
                                       SlotIndex UseIdx,
 | 
						|
                                       bool cheapAsAMove,
 | 
						|
                                       LiveIntervals &lis) {
 | 
						|
  assert(scannedRemattable_ && "Call anyRematerializable first");
 | 
						|
 | 
						|
  // Use scanRemattable info.
 | 
						|
  if (!remattable_.count(RM.ParentVNI))
 | 
						|
    return false;
 | 
						|
 | 
						|
  // No defining instruction.
 | 
						|
  RM.OrigMI = lis.getInstructionFromIndex(RM.ParentVNI->def);
 | 
						|
  assert(RM.OrigMI && "Defining instruction for remattable value disappeared");
 | 
						|
 | 
						|
  // If only cheap remats were requested, bail out early.
 | 
						|
  if (cheapAsAMove && !RM.OrigMI->getDesc().isAsCheapAsAMove())
 | 
						|
    return false;
 | 
						|
 | 
						|
  // Verify that all used registers are available with the same values.
 | 
						|
  if (!allUsesAvailableAt(RM.OrigMI, RM.ParentVNI->def, UseIdx, lis))
 | 
						|
    return false;
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
SlotIndex LiveRangeEdit::rematerializeAt(MachineBasicBlock &MBB,
 | 
						|
                                         MachineBasicBlock::iterator MI,
 | 
						|
                                         unsigned DestReg,
 | 
						|
                                         const Remat &RM,
 | 
						|
                                         LiveIntervals &lis,
 | 
						|
                                         const TargetInstrInfo &tii,
 | 
						|
                                         const TargetRegisterInfo &tri) {
 | 
						|
  assert(RM.OrigMI && "Invalid remat");
 | 
						|
  tii.reMaterialize(MBB, MI, DestReg, 0, RM.OrigMI, tri);
 | 
						|
  rematted_.insert(RM.ParentVNI);
 | 
						|
  return lis.InsertMachineInstrInMaps(--MI).getDefIndex();
 | 
						|
}
 | 
						|
 |