2016-06-01 06:38:06 +08:00
|
|
|
//===-- RenameIndependentSubregs.cpp - Live Interval Analysis -------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2016-06-01 06:38:06 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
/// Rename independent subregisters looks for virtual registers with
|
|
|
|
/// independently used subregisters and renames them to new virtual registers.
|
|
|
|
/// Example: In the following:
|
2017-11-30 20:12:19 +08:00
|
|
|
/// %0:sub0<read-undef> = ...
|
|
|
|
/// %0:sub1 = ...
|
|
|
|
/// use %0:sub0
|
|
|
|
/// %0:sub0 = ...
|
|
|
|
/// use %0:sub0
|
|
|
|
/// use %0:sub1
|
2016-06-01 06:38:06 +08:00
|
|
|
/// sub0 and sub1 are never used together, and we have two independent sub0
|
|
|
|
/// definitions. This pass will rename to:
|
2017-11-30 20:12:19 +08:00
|
|
|
/// %0:sub0<read-undef> = ...
|
|
|
|
/// %1:sub1<read-undef> = ...
|
|
|
|
/// use %1:sub1
|
|
|
|
/// %2:sub1<read-undef> = ...
|
|
|
|
/// use %2:sub1
|
|
|
|
/// use %0:sub0
|
2016-06-01 06:38:06 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "LiveRangeUtils.h"
|
|
|
|
#include "PHIEliminationUtils.h"
|
|
|
|
#include "llvm/CodeGen/LiveInterval.h"
|
2017-12-13 10:51:04 +08:00
|
|
|
#include "llvm/CodeGen/LiveIntervals.h"
|
2016-06-01 06:38:06 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
2016-06-01 06:38:06 +08:00
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
|
|
|
#include "llvm/CodeGen/Passes.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "llvm/CodeGen/TargetInstrInfo.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-14 05:15:01 +08:00
|
|
|
#include "llvm/InitializePasses.h"
|
2016-06-01 06:38:06 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "rename-independent-subregs"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class RenameIndependentSubregs : public MachineFunctionPass {
|
|
|
|
public:
|
|
|
|
static char ID;
|
|
|
|
RenameIndependentSubregs() : MachineFunctionPass(ID) {}
|
|
|
|
|
2016-10-01 10:56:57 +08:00
|
|
|
StringRef getPassName() const override {
|
2016-06-01 06:38:06 +08:00
|
|
|
return "Rename Disconnected Subregister Components";
|
|
|
|
}
|
|
|
|
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
|
|
|
AU.setPreservesCFG();
|
|
|
|
AU.addRequired<LiveIntervals>();
|
|
|
|
AU.addPreserved<LiveIntervals>();
|
|
|
|
AU.addRequired<SlotIndexes>();
|
|
|
|
AU.addPreserved<SlotIndexes>();
|
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool runOnMachineFunction(MachineFunction &MF) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct SubRangeInfo {
|
|
|
|
ConnectedVNInfoEqClasses ConEQ;
|
|
|
|
LiveInterval::SubRange *SR;
|
|
|
|
unsigned Index;
|
|
|
|
|
|
|
|
SubRangeInfo(LiveIntervals &LIS, LiveInterval::SubRange &SR,
|
|
|
|
unsigned Index)
|
|
|
|
: ConEQ(LIS), SR(&SR), Index(Index) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Split unrelated subregister components and rename them to new vregs.
|
|
|
|
bool renameComponents(LiveInterval &LI) const;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Build a vector of SubRange infos and a union find set of
|
2016-06-01 06:38:06 +08:00
|
|
|
/// equivalence classes.
|
|
|
|
/// Returns true if more than 1 equivalence class was found.
|
|
|
|
bool findComponents(IntEqClasses &Classes,
|
|
|
|
SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
LiveInterval &LI) const;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Distribute the LiveInterval segments into the new LiveIntervals
|
2016-06-01 06:38:06 +08:00
|
|
|
/// belonging to their class.
|
|
|
|
void distribute(const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const;
|
|
|
|
|
2018-05-01 23:54:18 +08:00
|
|
|
/// Constructs main liverange and add missing undef+dead flags.
|
2016-06-01 06:38:06 +08:00
|
|
|
void computeMainRangesFixFlags(const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const;
|
|
|
|
|
|
|
|
/// Rewrite Machine Operands to use the new vreg belonging to their class.
|
|
|
|
void rewriteOperands(const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const;
|
|
|
|
|
|
|
|
|
|
|
|
LiveIntervals *LIS;
|
|
|
|
MachineRegisterInfo *MRI;
|
|
|
|
const TargetInstrInfo *TII;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
char RenameIndependentSubregs::ID;
|
|
|
|
|
|
|
|
char &llvm::RenameIndependentSubregsID = RenameIndependentSubregs::ID;
|
|
|
|
|
2017-05-26 05:26:32 +08:00
|
|
|
INITIALIZE_PASS_BEGIN(RenameIndependentSubregs, DEBUG_TYPE,
|
2016-06-01 06:38:06 +08:00
|
|
|
"Rename Independent Subregisters", false, false)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(SlotIndexes)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(LiveIntervals)
|
2017-05-26 05:26:32 +08:00
|
|
|
INITIALIZE_PASS_END(RenameIndependentSubregs, DEBUG_TYPE,
|
2016-06-01 06:38:06 +08:00
|
|
|
"Rename Independent Subregisters", false, false)
|
|
|
|
|
|
|
|
bool RenameIndependentSubregs::renameComponents(LiveInterval &LI) const {
|
|
|
|
// Shortcut: We cannot have split components with a single definition.
|
|
|
|
if (LI.valnos.size() < 2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
SmallVector<SubRangeInfo, 4> SubRangeInfos;
|
|
|
|
IntEqClasses Classes;
|
|
|
|
if (!findComponents(Classes, SubRangeInfos, LI))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Create a new VReg for each class.
|
2020-09-16 05:54:38 +08:00
|
|
|
unsigned Reg = LI.reg();
|
2016-06-01 06:38:06 +08:00
|
|
|
const TargetRegisterClass *RegClass = MRI->getRegClass(Reg);
|
|
|
|
SmallVector<LiveInterval*, 4> Intervals;
|
|
|
|
Intervals.push_back(&LI);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << printReg(Reg) << ": Found " << Classes.getNumClasses()
|
|
|
|
<< " equivalence classes.\n");
|
|
|
|
LLVM_DEBUG(dbgs() << printReg(Reg) << ": Splitting into newly created:");
|
2016-06-01 06:38:06 +08:00
|
|
|
for (unsigned I = 1, NumClasses = Classes.getNumClasses(); I < NumClasses;
|
|
|
|
++I) {
|
Apply llvm-prefer-register-over-unsigned from clang-tidy to LLVM
Summary:
This clang-tidy check is looking for unsigned integer variables whose initializer
starts with an implicit cast from llvm::Register and changes the type of the
variable to llvm::Register (dropping the llvm:: where possible).
Partial reverts in:
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
X86FixupLEAs.cpp - Some functions return unsigned and arguably should be MCRegister
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
HexagonBitSimplify.cpp - Function takes BitTracker::RegisterRef which appears to be unsigned&
MachineVerifier.cpp - Ambiguous operator==() given MCRegister and const Register
PPCFastISel.cpp - No Register::operator-=()
PeepholeOptimizer.cpp - TargetInstrInfo::optimizeLoadInstr() takes an unsigned&
MachineTraceMetrics.cpp - MachineTraceMetrics lacks a suitable constructor
Manual fixups in:
ARMFastISel.cpp - ARMEmitLoad() now takes a Register& instead of unsigned&
HexagonSplitDouble.cpp - Ternary operator was ambiguous between unsigned/Register
HexagonConstExtenders.cpp - Has a local class named Register, used llvm::Register instead of Register.
PPCFastISel.cpp - PPCEmitLoad() now takes a Register& instead of unsigned&
Depends on D65919
Reviewers: arsenm, bogner, craig.topper, RKSimon
Reviewed By: arsenm
Subscribers: RKSimon, craig.topper, lenary, aemerson, wuzish, jholewinski, MatzeB, qcolombet, dschuff, jyknight, dylanmckay, sdardis, nemanjai, jvesely, wdng, nhaehnle, sbc100, jgravelle-google, kristof.beyls, hiraditya, aheejin, kbarton, fedor.sergeev, javed.absar, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, MaskRay, zzheng, edward-jones, atanasyan, rogfer01, MartinMosbeck, brucehoult, the_o, tpr, PkmX, jocewei, jsji, Petar.Avramovic, asbirlea, Jim, s.egerton, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D65962
llvm-svn: 369041
2019-08-16 03:22:08 +08:00
|
|
|
Register NewVReg = MRI->createVirtualRegister(RegClass);
|
2016-06-01 06:38:06 +08:00
|
|
|
LiveInterval &NewLI = LIS->createEmptyInterval(NewVReg);
|
|
|
|
Intervals.push_back(&NewLI);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << ' ' << printReg(NewVReg));
|
2016-06-01 06:38:06 +08:00
|
|
|
}
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << '\n');
|
2016-06-01 06:38:06 +08:00
|
|
|
|
|
|
|
rewriteOperands(Classes, SubRangeInfos, Intervals);
|
|
|
|
distribute(Classes, SubRangeInfos, Intervals);
|
|
|
|
computeMainRangesFixFlags(Classes, SubRangeInfos, Intervals);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RenameIndependentSubregs::findComponents(IntEqClasses &Classes,
|
|
|
|
SmallVectorImpl<RenameIndependentSubregs::SubRangeInfo> &SubRangeInfos,
|
|
|
|
LiveInterval &LI) const {
|
|
|
|
// First step: Create connected components for the VNInfos inside the
|
|
|
|
// subranges and count the global number of such components.
|
|
|
|
unsigned NumComponents = 0;
|
|
|
|
for (LiveInterval::SubRange &SR : LI.subranges()) {
|
|
|
|
SubRangeInfos.push_back(SubRangeInfo(*LIS, SR, NumComponents));
|
|
|
|
ConnectedVNInfoEqClasses &ConEQ = SubRangeInfos.back().ConEQ;
|
|
|
|
|
|
|
|
unsigned NumSubComponents = ConEQ.Classify(SR);
|
|
|
|
NumComponents += NumSubComponents;
|
|
|
|
}
|
|
|
|
// Shortcut: With only 1 subrange, the normal separate component tests are
|
|
|
|
// enough and we do not need to perform the union-find on the subregister
|
|
|
|
// segments.
|
|
|
|
if (SubRangeInfos.size() < 2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Next step: Build union-find structure over all subranges and merge classes
|
|
|
|
// across subranges when they are affected by the same MachineOperand.
|
|
|
|
const TargetRegisterInfo &TRI = *MRI->getTargetRegisterInfo();
|
|
|
|
Classes.grow(NumComponents);
|
2020-09-16 05:54:38 +08:00
|
|
|
unsigned Reg = LI.reg();
|
2016-06-01 06:38:06 +08:00
|
|
|
for (const MachineOperand &MO : MRI->reg_nodbg_operands(Reg)) {
|
|
|
|
if (!MO.isDef() && !MO.readsReg())
|
|
|
|
continue;
|
|
|
|
unsigned SubRegIdx = MO.getSubReg();
|
|
|
|
LaneBitmask LaneMask = TRI.getSubRegIndexLaneMask(SubRegIdx);
|
|
|
|
unsigned MergedID = ~0u;
|
|
|
|
for (RenameIndependentSubregs::SubRangeInfo &SRInfo : SubRangeInfos) {
|
|
|
|
const LiveInterval::SubRange &SR = *SRInfo.SR;
|
2016-12-15 22:36:06 +08:00
|
|
|
if ((SR.LaneMask & LaneMask).none())
|
2016-06-01 06:38:06 +08:00
|
|
|
continue;
|
|
|
|
SlotIndex Pos = LIS->getInstructionIndex(*MO.getParent());
|
|
|
|
Pos = MO.isDef() ? Pos.getRegSlot(MO.isEarlyClobber())
|
|
|
|
: Pos.getBaseIndex();
|
|
|
|
const VNInfo *VNI = SR.getVNInfoAt(Pos);
|
|
|
|
if (VNI == nullptr)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Map to local representant ID.
|
|
|
|
unsigned LocalID = SRInfo.ConEQ.getEqClass(VNI);
|
|
|
|
// Global ID
|
|
|
|
unsigned ID = LocalID + SRInfo.Index;
|
|
|
|
// Merge other sets
|
|
|
|
MergedID = MergedID == ~0u ? ID : Classes.join(MergedID, ID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Early exit if we ended up with a single equivalence class.
|
|
|
|
Classes.compress();
|
|
|
|
unsigned NumClasses = Classes.getNumClasses();
|
|
|
|
return NumClasses > 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenameIndependentSubregs::rewriteOperands(const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const {
|
|
|
|
const TargetRegisterInfo &TRI = *MRI->getTargetRegisterInfo();
|
2020-09-16 05:54:38 +08:00
|
|
|
unsigned Reg = Intervals[0]->reg();
|
2016-06-01 06:38:06 +08:00
|
|
|
for (MachineRegisterInfo::reg_nodbg_iterator I = MRI->reg_nodbg_begin(Reg),
|
|
|
|
E = MRI->reg_nodbg_end(); I != E; ) {
|
|
|
|
MachineOperand &MO = *I++;
|
|
|
|
if (!MO.isDef() && !MO.readsReg())
|
|
|
|
continue;
|
|
|
|
|
2018-07-10 04:07:03 +08:00
|
|
|
auto *MI = MO.getParent();
|
|
|
|
SlotIndex Pos = LIS->getInstructionIndex(*MI);
|
2016-08-04 06:37:47 +08:00
|
|
|
Pos = MO.isDef() ? Pos.getRegSlot(MO.isEarlyClobber())
|
|
|
|
: Pos.getBaseIndex();
|
2016-06-01 06:38:06 +08:00
|
|
|
unsigned SubRegIdx = MO.getSubReg();
|
|
|
|
LaneBitmask LaneMask = TRI.getSubRegIndexLaneMask(SubRegIdx);
|
|
|
|
|
|
|
|
unsigned ID = ~0u;
|
|
|
|
for (const SubRangeInfo &SRInfo : SubRangeInfos) {
|
|
|
|
const LiveInterval::SubRange &SR = *SRInfo.SR;
|
2016-12-15 22:36:06 +08:00
|
|
|
if ((SR.LaneMask & LaneMask).none())
|
2016-06-01 06:38:06 +08:00
|
|
|
continue;
|
2016-08-04 06:37:47 +08:00
|
|
|
const VNInfo *VNI = SR.getVNInfoAt(Pos);
|
|
|
|
if (VNI == nullptr)
|
2016-06-01 06:38:06 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Map to local representant ID.
|
2016-08-04 06:37:47 +08:00
|
|
|
unsigned LocalID = SRInfo.ConEQ.getEqClass(VNI);
|
2016-06-01 06:38:06 +08:00
|
|
|
// Global ID
|
|
|
|
ID = Classes[LocalID + SRInfo.Index];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-09-16 05:54:38 +08:00
|
|
|
unsigned VReg = Intervals[ID]->reg();
|
2016-06-01 06:38:06 +08:00
|
|
|
MO.setReg(VReg);
|
2017-06-28 02:28:10 +08:00
|
|
|
|
|
|
|
if (MO.isTied() && Reg != VReg) {
|
2018-07-10 04:07:03 +08:00
|
|
|
/// Undef use operands are not tracked in the equivalence class,
|
|
|
|
/// but need to be updated if they are tied; take care to only
|
|
|
|
/// update the tied operand.
|
|
|
|
unsigned OperandNo = MI->getOperandNo(&MO);
|
|
|
|
unsigned TiedIdx = MI->findTiedOperandIdx(OperandNo);
|
|
|
|
MI->getOperand(TiedIdx).setReg(VReg);
|
|
|
|
|
|
|
|
// above substitution breaks the iterator, so restart.
|
2017-06-27 05:33:36 +08:00
|
|
|
I = MRI->reg_nodbg_begin(Reg);
|
2017-06-06 06:58:57 +08:00
|
|
|
}
|
2016-06-01 06:38:06 +08:00
|
|
|
}
|
|
|
|
// TODO: We could attempt to recompute new register classes while visiting
|
|
|
|
// the operands: Some of the split register may be fine with less constraint
|
|
|
|
// classes than the original vreg.
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenameIndependentSubregs::distribute(const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const {
|
|
|
|
unsigned NumClasses = Classes.getNumClasses();
|
|
|
|
SmallVector<unsigned, 8> VNIMapping;
|
|
|
|
SmallVector<LiveInterval::SubRange*, 8> SubRanges;
|
|
|
|
BumpPtrAllocator &Allocator = LIS->getVNInfoAllocator();
|
|
|
|
for (const SubRangeInfo &SRInfo : SubRangeInfos) {
|
|
|
|
LiveInterval::SubRange &SR = *SRInfo.SR;
|
|
|
|
unsigned NumValNos = SR.valnos.size();
|
|
|
|
VNIMapping.clear();
|
|
|
|
VNIMapping.reserve(NumValNos);
|
|
|
|
SubRanges.clear();
|
|
|
|
SubRanges.resize(NumClasses-1, nullptr);
|
|
|
|
for (unsigned I = 0; I < NumValNos; ++I) {
|
|
|
|
const VNInfo &VNI = *SR.valnos[I];
|
|
|
|
unsigned LocalID = SRInfo.ConEQ.getEqClass(&VNI);
|
|
|
|
unsigned ID = Classes[LocalID + SRInfo.Index];
|
|
|
|
VNIMapping.push_back(ID);
|
|
|
|
if (ID > 0 && SubRanges[ID-1] == nullptr)
|
|
|
|
SubRanges[ID-1] = Intervals[ID]->createSubRange(Allocator, SR.LaneMask);
|
|
|
|
}
|
|
|
|
DistributeRange(SR, SubRanges.data(), VNIMapping);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool subRangeLiveAt(const LiveInterval &LI, SlotIndex Pos) {
|
|
|
|
for (const LiveInterval::SubRange &SR : LI.subranges()) {
|
|
|
|
if (SR.liveAt(Pos))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenameIndependentSubregs::computeMainRangesFixFlags(
|
|
|
|
const IntEqClasses &Classes,
|
|
|
|
const SmallVectorImpl<SubRangeInfo> &SubRangeInfos,
|
|
|
|
const SmallVectorImpl<LiveInterval*> &Intervals) const {
|
|
|
|
BumpPtrAllocator &Allocator = LIS->getVNInfoAllocator();
|
|
|
|
const SlotIndexes &Indexes = *LIS->getSlotIndexes();
|
|
|
|
for (size_t I = 0, E = Intervals.size(); I < E; ++I) {
|
|
|
|
LiveInterval &LI = *Intervals[I];
|
2020-09-16 05:54:38 +08:00
|
|
|
unsigned Reg = LI.reg();
|
2016-06-01 06:38:06 +08:00
|
|
|
|
|
|
|
LI.removeEmptySubRanges();
|
|
|
|
|
|
|
|
// There must be a def (or live-in) before every use. Splitting vregs may
|
|
|
|
// violate this principle as the splitted vreg may not have a definition on
|
|
|
|
// every path. Fix this by creating IMPLICIT_DEF instruction as necessary.
|
|
|
|
for (const LiveInterval::SubRange &SR : LI.subranges()) {
|
|
|
|
// Search for "PHI" value numbers in the subranges. We must find a live
|
|
|
|
// value in each predecessor block, add an IMPLICIT_DEF where it is
|
|
|
|
// missing.
|
|
|
|
for (unsigned I = 0; I < SR.valnos.size(); ++I) {
|
|
|
|
const VNInfo &VNI = *SR.valnos[I];
|
|
|
|
if (VNI.isUnused() || !VNI.isPHIDef())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
SlotIndex Def = VNI.def;
|
|
|
|
MachineBasicBlock &MBB = *Indexes.getMBBFromIndex(Def);
|
|
|
|
for (MachineBasicBlock *PredMBB : MBB.predecessors()) {
|
|
|
|
SlotIndex PredEnd = Indexes.getMBBEndIdx(PredMBB);
|
|
|
|
if (subRangeLiveAt(LI, PredEnd.getPrevSlot()))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MachineBasicBlock::iterator InsertPos =
|
|
|
|
llvm::findPHICopyInsertPoint(PredMBB, &MBB, Reg);
|
|
|
|
const MCInstrDesc &MCDesc = TII->get(TargetOpcode::IMPLICIT_DEF);
|
|
|
|
MachineInstrBuilder ImpDef = BuildMI(*PredMBB, InsertPos,
|
|
|
|
DebugLoc(), MCDesc, Reg);
|
|
|
|
SlotIndex DefIdx = LIS->InsertMachineInstrInMaps(*ImpDef);
|
|
|
|
SlotIndex RegDefIdx = DefIdx.getRegSlot();
|
|
|
|
for (LiveInterval::SubRange &SR : LI.subranges()) {
|
|
|
|
VNInfo *SRVNI = SR.getNextValue(RegDefIdx, Allocator);
|
|
|
|
SR.addSegment(LiveRange::Segment(RegDefIdx, PredEnd, SRVNI));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (MachineOperand &MO : MRI->reg_nodbg_operands(Reg)) {
|
|
|
|
if (!MO.isDef())
|
|
|
|
continue;
|
|
|
|
unsigned SubRegIdx = MO.getSubReg();
|
|
|
|
if (SubRegIdx == 0)
|
|
|
|
continue;
|
|
|
|
// After assigning the new vreg we may not have any other sublanes living
|
|
|
|
// in and out of the instruction anymore. We need to add new dead and
|
|
|
|
// undef flags in these cases.
|
|
|
|
if (!MO.isUndef()) {
|
|
|
|
SlotIndex Pos = LIS->getInstructionIndex(*MO.getParent());
|
|
|
|
if (!subRangeLiveAt(LI, Pos))
|
|
|
|
MO.setIsUndef();
|
|
|
|
}
|
|
|
|
if (!MO.isDead()) {
|
|
|
|
SlotIndex Pos = LIS->getInstructionIndex(*MO.getParent()).getDeadSlot();
|
|
|
|
if (!subRangeLiveAt(LI, Pos))
|
|
|
|
MO.setIsDead();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (I == 0)
|
|
|
|
LI.clear();
|
|
|
|
LIS->constructMainRangeFromSubranges(LI);
|
2016-08-24 21:37:55 +08:00
|
|
|
// A def of a subregister may be a use of other register lanes. Replacing
|
|
|
|
// such a def with a def of a different register will eliminate the use,
|
|
|
|
// and may cause the recorded live range to be larger than the actual
|
|
|
|
// liveness in the program IR.
|
|
|
|
LIS->shrinkToUses(&LI);
|
2016-06-01 06:38:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RenameIndependentSubregs::runOnMachineFunction(MachineFunction &MF) {
|
|
|
|
// Skip renaming if liveness of subregister is not tracked.
|
2016-08-25 06:17:45 +08:00
|
|
|
MRI = &MF.getRegInfo();
|
|
|
|
if (!MRI->subRegLivenessEnabled())
|
2016-06-01 06:38:06 +08:00
|
|
|
return false;
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(dbgs() << "Renaming independent subregister live ranges in "
|
|
|
|
<< MF.getName() << '\n');
|
2016-06-01 06:38:06 +08:00
|
|
|
|
|
|
|
LIS = &getAnalysis<LiveIntervals>();
|
|
|
|
TII = MF.getSubtarget().getInstrInfo();
|
|
|
|
|
|
|
|
// Iterate over all vregs. Note that we query getNumVirtRegs() the newly
|
|
|
|
// created vregs end up with higher numbers but do not need to be visited as
|
|
|
|
// there can't be any further splitting.
|
|
|
|
bool Changed = false;
|
|
|
|
for (size_t I = 0, E = MRI->getNumVirtRegs(); I < E; ++I) {
|
2019-08-02 07:27:28 +08:00
|
|
|
unsigned Reg = Register::index2VirtReg(I);
|
2016-06-01 06:38:06 +08:00
|
|
|
if (!LIS->hasInterval(Reg))
|
|
|
|
continue;
|
|
|
|
LiveInterval &LI = LIS->getInterval(Reg);
|
|
|
|
if (!LI.hasSubRanges())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Changed |= renameComponents(LI);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Changed;
|
|
|
|
}
|