Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(29)

Unified Diff: src/IceRegAlloc.cpp

Issue 720343003: Subzero: Simplify the FakeKill instruction. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: Remove the Nonpoints code in LiveRange Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: src/IceRegAlloc.cpp
diff --git a/src/IceRegAlloc.cpp b/src/IceRegAlloc.cpp
index 1b4d0c249d0d872f2aecb705081dce83e25a04c8..57661124447fca8b6613eafd89cfd8c504fef0d8 100644
--- a/src/IceRegAlloc.cpp
+++ b/src/IceRegAlloc.cpp
@@ -14,6 +14,7 @@
//===----------------------------------------------------------------------===//
#include "IceCfg.h"
+#include "IceCfgNode.h"
#include "IceInst.h"
#include "IceOperand.h"
#include "IceRegAlloc.h"
@@ -72,6 +73,63 @@ void dumpLiveRange(const Variable *Var, const Cfg *Func) {
} // end of anonymous namespace
+void LinearScan::initForGlobalAlloc() {
+ TimerMarker T(TimerStack::TT_initUnhandled, Func);
+ Unhandled.clear();
+ UnhandledPrecolored.clear();
+ Handled.clear();
+ Inactive.clear();
+ Active.clear();
+ // Gather the live ranges of all variables and add them to the
+ // Unhandled set.
+ const VarList &Vars = Func->getVariables();
+ Unhandled.reserve(Vars.size());
+ for (Variable *Var : Vars) {
+ // Explicitly don't consider zero-weight variables, which are
+ // meant to be spill slots.
+ if (Var->getWeight() == RegWeight::Zero)
+ continue;
+ // Don't bother if the variable has a null live range, which means
+ // it was never referenced.
+ if (Var->getLiveRange().isEmpty())
+ continue;
+ Var->untrimLiveRange();
+ Unhandled.push_back(Var);
+ if (Var->hasReg()) {
+ Var->setRegNumTmp(Var->getRegNum());
+ Var->setLiveRangeInfiniteWeight();
+ UnhandledPrecolored.push_back(Var);
+ }
+ }
+ struct CompareRanges {
+ bool operator()(const Variable *L, const Variable *R) {
+ InstNumberT Lstart = L->getLiveRange().getStart();
+ InstNumberT Rstart = R->getLiveRange().getStart();
+ if (Lstart == Rstart)
+ return L->getIndex() < R->getIndex();
+ return Lstart < Rstart;
+ }
+ };
+ // Do a reverse sort so that erasing elements (from the end) is fast.
+ std::sort(Unhandled.rbegin(), Unhandled.rend(), CompareRanges());
+ std::sort(UnhandledPrecolored.rbegin(), UnhandledPrecolored.rend(),
+ CompareRanges());
+
+ // Build the (ordered) list of FakeKill instruction numbers.
+ Kills.clear();
+ for (CfgNode *Node : Func->getNodes()) {
+ for (auto I = Node->getInsts().begin(), E = Node->getInsts().end(); I != E;
+ ++I) {
+ if (I->isDeleted())
+ continue;
+ if (auto Kill = llvm::dyn_cast<InstFakeKill>(I)) {
+ if (!Kill->getLinked()->isDeleted())
+ Kills.push_back(I->getNumber());
+ }
+ }
+ }
+}
+
// Implements the linear-scan algorithm. Based on "Linear Scan
// Register Allocation in the Context of SSA Form and Register
// Constraints" by Hanspeter Mössenböck and Michael Pfeiffer,
@@ -86,53 +144,16 @@ void dumpLiveRange(const Variable *Var, const Cfg *Func) {
void LinearScan::scan(const llvm::SmallBitVector &RegMaskFull) {
TimerMarker T(TimerStack::TT_linearScan, Func);
assert(RegMaskFull.any()); // Sanity check
- Unhandled.clear();
- UnhandledPrecolored.clear();
- Handled.clear();
- Inactive.clear();
- Active.clear();
Ostream &Str = Func->getContext()->getStrDump();
bool Verbose = Func->getContext()->isVerbose(IceV_LinearScan);
Func->resetCurrentNode();
VariablesMetadata *VMetadata = Func->getVMetadata();
- // Gather the live ranges of all variables and add them to the
- // Unhandled set.
- const VarList &Vars = Func->getVariables();
- {
- TimerMarker T(TimerStack::TT_initUnhandled, Func);
- Unhandled.reserve(Vars.size());
- for (Variable *Var : Vars) {
- // Explicitly don't consider zero-weight variables, which are
- // meant to be spill slots.
- if (Var->getWeight() == RegWeight::Zero)
- continue;
- // Don't bother if the variable has a null live range, which means
- // it was never referenced.
- if (Var->getLiveRange().isEmpty())
- continue;
- Var->untrimLiveRange();
- Unhandled.push_back(Var);
- if (Var->hasReg()) {
- Var->setRegNumTmp(Var->getRegNum());
- Var->setLiveRangeInfiniteWeight();
- UnhandledPrecolored.push_back(Var);
- }
- }
- struct CompareRanges {
- bool operator()(const Variable *L, const Variable *R) {
- InstNumberT Lstart = L->getLiveRange().getStart();
- InstNumberT Rstart = R->getLiveRange().getStart();
- if (Lstart == Rstart)
- return L->getIndex() < R->getIndex();
- return Lstart < Rstart;
- }
- };
- // Do a reverse sort so that erasing elements (from the end) is fast.
- std::sort(Unhandled.rbegin(), Unhandled.rend(), CompareRanges());
- std::sort(UnhandledPrecolored.rbegin(), UnhandledPrecolored.rend(),
- CompareRanges());
- }
+ // Build a LiveRange representing the Kills list.
+ LiveRange KillsRange;
+ for (InstNumberT I : Kills)
+ KillsRange.addSegment(I, I);
+ KillsRange.untrim();
// RegUses[I] is the number of live ranges (variables) that register
// I is currently assigned to. It can be greater than 1 as a result
@@ -155,6 +176,9 @@ void LinearScan::scan(const llvm::SmallBitVector &RegMaskFull) {
}
const llvm::SmallBitVector RegMask =
RegMaskFull & Func->getTarget()->getRegisterSetForType(Cur->getType());
+ KillsRange.trim(Cur->getLiveRange().getStart());
+ llvm::SmallBitVector KillsMask = Func->getTarget()->getRegisterSet(
jvoung (off chromium) 2014/11/14 16:47:23 Can the kills mask be lifted out of the while loop
Jim Stichnoth 2014/11/14 18:27:03 Good point, done.
+ TargetLowering::RegSet_CallerSave, TargetLowering::RegSet_None);
// Check for precolored ranges. If Cur is precolored, it
// definitely gets that register. Previously processed live
@@ -220,15 +244,6 @@ void LinearScan::scan(const llvm::SmallBitVector &RegMaskFull) {
++Next;
Variable *Item = *I;
Item->trimLiveRange(Cur->getLiveRange().getStart());
- // As an optimization, don't bother checking pure point-valued
- // Inactive ranges, because the overlapsStart() test will never
- // succeed, and the rangeEndsBefore() test will generally only
- // succeed after the last call instruction, which statistically
- // happens near the end. TODO(stichnot): Consider suppressing
- // this check every N iterations in case calls are only at the
- // beginning of the function.
- if (!Item->getLiveRange().isNonpoints())
- continue;
if (Item->rangeEndsBefore(Cur)) {
// Move Item from Inactive to Handled list.
if (Verbose) {
@@ -374,6 +389,19 @@ void LinearScan::scan(const llvm::SmallBitVector &RegMaskFull) {
}
}
+ // Remove scratch registers from the Free[] list, and mark their
+ // Weights[] as infinite, if KillsRange overlaps Cur's live range.
+ const bool UseTrimmed = true;
+ if (Cur->getLiveRange().overlaps(KillsRange, UseTrimmed)) {
+ Free.reset(KillsMask);
+ for (int i = KillsMask.find_first(); i != -1;
+ i = KillsMask.find_next(i)) {
+ Weights[i].setWeight(RegWeight::Inf);
+ if (PreferReg == i)
+ AllowOverlap = false;
+ }
+ }
+
// Print info about physical register availability.
if (Verbose) {
for (SizeT i = 0; i < RegMask.size(); ++i) {
« src/IceOperand.h ('K') | « src/IceRegAlloc.h ('k') | src/IceTargetLowering.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698