2008-11-20 00:18:57 +01:00
|
|
|
//===---- ScheduleDAG.cpp - Implement the ScheduleDAG class ---------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This implements the ScheduleDAG class, which is a base class used by
|
|
|
|
// scheduling implementation classes.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "pre-RA-sched"
|
|
|
|
#include "llvm/CodeGen/ScheduleDAG.h"
|
2009-01-15 23:18:12 +01:00
|
|
|
#include "llvm/CodeGen/ScheduleHazardRecognizer.h"
|
Various bits of framework needed for precise machine-level selection
DAG scheduling during isel. Most new functionality is currently
guarded by -enable-sched-cycles and -enable-sched-hazard.
Added InstrItineraryData::IssueWidth field, currently derived from
ARM itineraries, but could be initialized differently on other targets.
Added ScheduleHazardRecognizer::MaxLookAhead to indicate whether it is
active, and if so how many cycles of state it holds.
Added SchedulingPriorityQueue::HasReadyFilter to allowing gating entry
into the scheduler's available queue.
ScoreboardHazardRecognizer now accesses the ScheduleDAG in order to
get information about it's SUnits, provides RecedeCycle for bottom-up
scheduling, correctly computes scoreboard depth, tracks IssueCount, and
considers potential stall cycles when checking for hazards.
ScheduleDAGRRList now models machine cycles and hazards (under
flags). It tracks MinAvailableCycle, drives the hazard recognizer and
priority queue's ready filter, manages a new PendingQueue, properly
accounts for stall cycles, etc.
llvm-svn: 122541
2010-12-24 06:03:26 +01:00
|
|
|
#include "llvm/CodeGen/SelectionDAGNodes.h"
|
2008-11-20 00:18:57 +01:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
#include "llvm/Target/TargetInstrInfo.h"
|
|
|
|
#include "llvm/Target/TargetRegisterInfo.h"
|
2011-06-15 19:16:12 +02:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2008-11-20 00:18:57 +01:00
|
|
|
#include "llvm/Support/Debug.h"
|
2009-07-24 11:53:24 +02:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2008-11-20 02:41:34 +01:00
|
|
|
#include <climits>
|
2008-11-20 00:18:57 +01:00
|
|
|
using namespace llvm;
|
|
|
|
|
2011-06-15 19:16:12 +02:00
|
|
|
#ifndef NDEBUG
|
|
|
|
cl::opt<bool> StressSchedOpt(
|
|
|
|
"stress-sched", cl::Hidden, cl::init(false),
|
|
|
|
cl::desc("Stress test instruction scheduling"));
|
|
|
|
#endif
|
|
|
|
|
2009-01-15 20:20:50 +01:00
|
|
|
ScheduleDAG::ScheduleDAG(MachineFunction &mf)
|
2009-02-11 05:27:20 +01:00
|
|
|
: TM(mf.getTarget()),
|
2009-01-15 20:20:50 +01:00
|
|
|
TII(TM.getInstrInfo()),
|
|
|
|
TRI(TM.getRegisterInfo()),
|
|
|
|
MF(mf), MRI(mf.getRegInfo()),
|
2009-02-11 00:27:53 +01:00
|
|
|
EntrySU(), ExitSU() {
|
2011-06-15 19:16:12 +02:00
|
|
|
#ifndef NDEBUG
|
|
|
|
StressSched = StressSchedOpt;
|
|
|
|
#endif
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ScheduleDAG::~ScheduleDAG() {}
|
|
|
|
|
Various bits of framework needed for precise machine-level selection
DAG scheduling during isel. Most new functionality is currently
guarded by -enable-sched-cycles and -enable-sched-hazard.
Added InstrItineraryData::IssueWidth field, currently derived from
ARM itineraries, but could be initialized differently on other targets.
Added ScheduleHazardRecognizer::MaxLookAhead to indicate whether it is
active, and if so how many cycles of state it holds.
Added SchedulingPriorityQueue::HasReadyFilter to allowing gating entry
into the scheduler's available queue.
ScoreboardHazardRecognizer now accesses the ScheduleDAG in order to
get information about it's SUnits, provides RecedeCycle for bottom-up
scheduling, correctly computes scoreboard depth, tracks IssueCount, and
considers potential stall cycles when checking for hazards.
ScheduleDAGRRList now models machine cycles and hazards (under
flags). It tracks MinAvailableCycle, drives the hazard recognizer and
priority queue's ready filter, manages a new PendingQueue, properly
accounts for stall cycles, etc.
llvm-svn: 122541
2010-12-24 06:03:26 +01:00
|
|
|
/// getInstrDesc helper to handle SDNodes.
|
|
|
|
const TargetInstrDesc *ScheduleDAG::getNodeDesc(const SDNode *Node) const {
|
2010-12-24 07:46:50 +01:00
|
|
|
if (!Node || !Node->isMachineOpcode()) return NULL;
|
Various bits of framework needed for precise machine-level selection
DAG scheduling during isel. Most new functionality is currently
guarded by -enable-sched-cycles and -enable-sched-hazard.
Added InstrItineraryData::IssueWidth field, currently derived from
ARM itineraries, but could be initialized differently on other targets.
Added ScheduleHazardRecognizer::MaxLookAhead to indicate whether it is
active, and if so how many cycles of state it holds.
Added SchedulingPriorityQueue::HasReadyFilter to allowing gating entry
into the scheduler's available queue.
ScoreboardHazardRecognizer now accesses the ScheduleDAG in order to
get information about it's SUnits, provides RecedeCycle for bottom-up
scheduling, correctly computes scoreboard depth, tracks IssueCount, and
considers potential stall cycles when checking for hazards.
ScheduleDAGRRList now models machine cycles and hazards (under
flags). It tracks MinAvailableCycle, drives the hazard recognizer and
priority queue's ready filter, manages a new PendingQueue, properly
accounts for stall cycles, etc.
llvm-svn: 122541
2010-12-24 06:03:26 +01:00
|
|
|
return &TII->get(Node->getMachineOpcode());
|
|
|
|
}
|
|
|
|
|
2008-11-20 00:18:57 +01:00
|
|
|
/// dump - dump the schedule.
|
|
|
|
void ScheduleDAG::dumpSchedule() const {
|
|
|
|
for (unsigned i = 0, e = Sequence.size(); i != e; i++) {
|
|
|
|
if (SUnit *SU = Sequence[i])
|
|
|
|
SU->dump(this);
|
|
|
|
else
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "**** NOOP ****\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Run - perform scheduling.
|
|
|
|
///
|
2009-02-11 05:27:20 +01:00
|
|
|
void ScheduleDAG::Run(MachineBasicBlock *bb,
|
|
|
|
MachineBasicBlock::iterator insertPos) {
|
|
|
|
BB = bb;
|
|
|
|
InsertPos = insertPos;
|
2009-01-16 23:10:20 +01:00
|
|
|
|
2009-01-15 20:20:50 +01:00
|
|
|
SUnits.clear();
|
|
|
|
Sequence.clear();
|
2009-02-11 00:27:53 +01:00
|
|
|
EntrySU = SUnit();
|
|
|
|
ExitSU = SUnit();
|
2009-01-15 20:20:50 +01:00
|
|
|
|
2008-11-20 00:18:57 +01:00
|
|
|
Schedule();
|
2009-02-11 05:27:20 +01:00
|
|
|
|
2009-08-22 22:41:06 +02:00
|
|
|
DEBUG({
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "*** Final schedule ***\n";
|
2009-08-22 22:41:06 +02:00
|
|
|
dumpSchedule();
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << '\n';
|
2009-08-22 22:41:06 +02:00
|
|
|
});
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
|
|
|
|
2008-12-16 02:05:52 +01:00
|
|
|
/// addPred - This adds the specified edge as a pred of the current node if
|
|
|
|
/// not already. It also adds the current node as a successor of the
|
|
|
|
/// specified node.
|
2011-02-04 04:18:17 +01:00
|
|
|
bool SUnit::addPred(const SDep &D) {
|
2008-12-16 02:05:52 +01:00
|
|
|
// If this node already has this depenence, don't add a redundant one.
|
2009-02-11 01:12:28 +01:00
|
|
|
for (SmallVector<SDep, 4>::const_iterator I = Preds.begin(), E = Preds.end();
|
|
|
|
I != E; ++I)
|
|
|
|
if (*I == D)
|
2011-02-04 04:18:17 +01:00
|
|
|
return false;
|
2008-12-16 02:05:52 +01:00
|
|
|
// Now add a corresponding succ to N.
|
|
|
|
SDep P = D;
|
|
|
|
P.setSUnit(this);
|
|
|
|
SUnit *N = D.getSUnit();
|
|
|
|
// Update the bookkeeping.
|
|
|
|
if (D.getKind() == SDep::Data) {
|
2009-09-30 22:15:38 +02:00
|
|
|
assert(NumPreds < UINT_MAX && "NumPreds will overflow!");
|
|
|
|
assert(N->NumSuccs < UINT_MAX && "NumSuccs will overflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
++NumPreds;
|
|
|
|
++N->NumSuccs;
|
|
|
|
}
|
2009-09-30 22:15:38 +02:00
|
|
|
if (!N->isScheduled) {
|
|
|
|
assert(NumPredsLeft < UINT_MAX && "NumPredsLeft will overflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
++NumPredsLeft;
|
2009-09-30 22:15:38 +02:00
|
|
|
}
|
|
|
|
if (!isScheduled) {
|
|
|
|
assert(N->NumSuccsLeft < UINT_MAX && "NumSuccsLeft will overflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
++N->NumSuccsLeft;
|
2009-09-30 22:15:38 +02:00
|
|
|
}
|
2008-12-16 04:25:46 +01:00
|
|
|
Preds.push_back(D);
|
2009-01-13 20:08:45 +01:00
|
|
|
N->Succs.push_back(P);
|
2009-01-05 23:40:26 +01:00
|
|
|
if (P.getLatency() != 0) {
|
|
|
|
this->setDepthDirty();
|
|
|
|
N->setHeightDirty();
|
|
|
|
}
|
2011-02-04 04:18:17 +01:00
|
|
|
return true;
|
2008-12-16 02:05:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// removePred - This removes the specified edge as a pred of the current
|
|
|
|
/// node if it exists. It also removes the current node as a successor of
|
|
|
|
/// the specified node.
|
|
|
|
void SUnit::removePred(const SDep &D) {
|
|
|
|
// Find the matching predecessor.
|
|
|
|
for (SmallVector<SDep, 4>::iterator I = Preds.begin(), E = Preds.end();
|
|
|
|
I != E; ++I)
|
|
|
|
if (*I == D) {
|
|
|
|
bool FoundSucc = false;
|
|
|
|
// Find the corresponding successor in N.
|
|
|
|
SDep P = D;
|
|
|
|
P.setSUnit(this);
|
|
|
|
SUnit *N = D.getSUnit();
|
|
|
|
for (SmallVector<SDep, 4>::iterator II = N->Succs.begin(),
|
|
|
|
EE = N->Succs.end(); II != EE; ++II)
|
|
|
|
if (*II == P) {
|
|
|
|
FoundSucc = true;
|
|
|
|
N->Succs.erase(II);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
assert(FoundSucc && "Mismatching preds / succs lists!");
|
|
|
|
Preds.erase(I);
|
2009-01-13 20:08:45 +01:00
|
|
|
// Update the bookkeeping.
|
|
|
|
if (P.getKind() == SDep::Data) {
|
2009-09-30 22:15:38 +02:00
|
|
|
assert(NumPreds > 0 && "NumPreds will underflow!");
|
|
|
|
assert(N->NumSuccs > 0 && "NumSuccs will underflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
--NumPreds;
|
|
|
|
--N->NumSuccs;
|
|
|
|
}
|
2009-09-30 22:15:38 +02:00
|
|
|
if (!N->isScheduled) {
|
|
|
|
assert(NumPredsLeft > 0 && "NumPredsLeft will underflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
--NumPredsLeft;
|
2009-09-30 22:15:38 +02:00
|
|
|
}
|
|
|
|
if (!isScheduled) {
|
|
|
|
assert(N->NumSuccsLeft > 0 && "NumSuccsLeft will underflow!");
|
2008-12-16 02:05:52 +01:00
|
|
|
--N->NumSuccsLeft;
|
2009-09-30 22:15:38 +02:00
|
|
|
}
|
2009-01-05 23:40:26 +01:00
|
|
|
if (P.getLatency() != 0) {
|
|
|
|
this->setDepthDirty();
|
|
|
|
N->setHeightDirty();
|
|
|
|
}
|
2008-12-16 02:05:52 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-12-16 04:25:46 +01:00
|
|
|
void SUnit::setDepthDirty() {
|
2008-12-22 22:11:33 +01:00
|
|
|
if (!isDepthCurrent) return;
|
2008-12-16 04:25:46 +01:00
|
|
|
SmallVector<SUnit*, 8> WorkList;
|
|
|
|
WorkList.push_back(this);
|
2008-12-22 22:11:33 +01:00
|
|
|
do {
|
2008-12-20 17:42:33 +01:00
|
|
|
SUnit *SU = WorkList.pop_back_val();
|
2008-12-16 04:25:46 +01:00
|
|
|
SU->isDepthCurrent = false;
|
2008-12-20 17:34:57 +01:00
|
|
|
for (SUnit::const_succ_iterator I = SU->Succs.begin(),
|
2008-12-22 22:11:33 +01:00
|
|
|
E = SU->Succs.end(); I != E; ++I) {
|
|
|
|
SUnit *SuccSU = I->getSUnit();
|
|
|
|
if (SuccSU->isDepthCurrent)
|
|
|
|
WorkList.push_back(SuccSU);
|
|
|
|
}
|
|
|
|
} while (!WorkList.empty());
|
2008-12-16 04:25:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void SUnit::setHeightDirty() {
|
2008-12-22 22:11:33 +01:00
|
|
|
if (!isHeightCurrent) return;
|
2008-12-16 04:25:46 +01:00
|
|
|
SmallVector<SUnit*, 8> WorkList;
|
|
|
|
WorkList.push_back(this);
|
2008-12-22 22:11:33 +01:00
|
|
|
do {
|
2008-12-20 17:42:33 +01:00
|
|
|
SUnit *SU = WorkList.pop_back_val();
|
2008-12-16 04:25:46 +01:00
|
|
|
SU->isHeightCurrent = false;
|
2008-12-20 17:34:57 +01:00
|
|
|
for (SUnit::const_pred_iterator I = SU->Preds.begin(),
|
2008-12-22 22:11:33 +01:00
|
|
|
E = SU->Preds.end(); I != E; ++I) {
|
|
|
|
SUnit *PredSU = I->getSUnit();
|
|
|
|
if (PredSU->isHeightCurrent)
|
|
|
|
WorkList.push_back(PredSU);
|
|
|
|
}
|
|
|
|
} while (!WorkList.empty());
|
2008-12-16 04:25:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// setDepthToAtLeast - Update this node's successors to reflect the
|
|
|
|
/// fact that this node's depth just increased.
|
|
|
|
///
|
2009-11-20 20:32:48 +01:00
|
|
|
void SUnit::setDepthToAtLeast(unsigned NewDepth) {
|
|
|
|
if (NewDepth <= getDepth())
|
2008-12-16 04:25:46 +01:00
|
|
|
return;
|
|
|
|
setDepthDirty();
|
|
|
|
Depth = NewDepth;
|
|
|
|
isDepthCurrent = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// setHeightToAtLeast - Update this node's predecessors to reflect the
|
|
|
|
/// fact that this node's height just increased.
|
|
|
|
///
|
2009-11-20 20:32:48 +01:00
|
|
|
void SUnit::setHeightToAtLeast(unsigned NewHeight) {
|
|
|
|
if (NewHeight <= getHeight())
|
2008-12-16 04:25:46 +01:00
|
|
|
return;
|
|
|
|
setHeightDirty();
|
|
|
|
Height = NewHeight;
|
|
|
|
isHeightCurrent = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ComputeDepth - Calculate the maximal path from the node to the exit.
|
|
|
|
///
|
2009-11-20 20:32:48 +01:00
|
|
|
void SUnit::ComputeDepth() {
|
2008-12-16 04:25:46 +01:00
|
|
|
SmallVector<SUnit*, 8> WorkList;
|
|
|
|
WorkList.push_back(this);
|
2008-12-23 18:22:32 +01:00
|
|
|
do {
|
2008-12-16 04:25:46 +01:00
|
|
|
SUnit *Cur = WorkList.back();
|
|
|
|
|
|
|
|
bool Done = true;
|
|
|
|
unsigned MaxPredDepth = 0;
|
|
|
|
for (SUnit::const_pred_iterator I = Cur->Preds.begin(),
|
|
|
|
E = Cur->Preds.end(); I != E; ++I) {
|
|
|
|
SUnit *PredSU = I->getSUnit();
|
|
|
|
if (PredSU->isDepthCurrent)
|
|
|
|
MaxPredDepth = std::max(MaxPredDepth,
|
|
|
|
PredSU->Depth + I->getLatency());
|
|
|
|
else {
|
|
|
|
Done = false;
|
|
|
|
WorkList.push_back(PredSU);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Done) {
|
|
|
|
WorkList.pop_back();
|
|
|
|
if (MaxPredDepth != Cur->Depth) {
|
|
|
|
Cur->setDepthDirty();
|
|
|
|
Cur->Depth = MaxPredDepth;
|
|
|
|
}
|
|
|
|
Cur->isDepthCurrent = true;
|
|
|
|
}
|
2008-12-23 18:22:32 +01:00
|
|
|
} while (!WorkList.empty());
|
2008-12-16 04:25:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// ComputeHeight - Calculate the maximal path from the node to the entry.
|
|
|
|
///
|
2009-11-20 20:32:48 +01:00
|
|
|
void SUnit::ComputeHeight() {
|
2008-12-16 04:25:46 +01:00
|
|
|
SmallVector<SUnit*, 8> WorkList;
|
|
|
|
WorkList.push_back(this);
|
2008-12-23 18:22:32 +01:00
|
|
|
do {
|
2008-12-16 04:25:46 +01:00
|
|
|
SUnit *Cur = WorkList.back();
|
|
|
|
|
|
|
|
bool Done = true;
|
|
|
|
unsigned MaxSuccHeight = 0;
|
|
|
|
for (SUnit::const_succ_iterator I = Cur->Succs.begin(),
|
|
|
|
E = Cur->Succs.end(); I != E; ++I) {
|
|
|
|
SUnit *SuccSU = I->getSUnit();
|
|
|
|
if (SuccSU->isHeightCurrent)
|
|
|
|
MaxSuccHeight = std::max(MaxSuccHeight,
|
|
|
|
SuccSU->Height + I->getLatency());
|
|
|
|
else {
|
|
|
|
Done = false;
|
|
|
|
WorkList.push_back(SuccSU);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Done) {
|
|
|
|
WorkList.pop_back();
|
|
|
|
if (MaxSuccHeight != Cur->Height) {
|
|
|
|
Cur->setHeightDirty();
|
|
|
|
Cur->Height = MaxSuccHeight;
|
|
|
|
}
|
|
|
|
Cur->isHeightCurrent = true;
|
|
|
|
}
|
2008-12-23 18:22:32 +01:00
|
|
|
} while (!WorkList.empty());
|
2008-12-16 04:25:46 +01:00
|
|
|
}
|
|
|
|
|
2008-11-20 00:18:57 +01:00
|
|
|
/// SUnit - Scheduling unit. It's an wrapper around either a single SDNode or
|
|
|
|
/// a group of nodes flagged together.
|
|
|
|
void SUnit::dump(const ScheduleDAG *G) const {
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "SU(" << NodeNum << "): ";
|
2008-11-20 00:18:57 +01:00
|
|
|
G->dumpNode(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SUnit::dumpAll(const ScheduleDAG *G) const {
|
|
|
|
dump(G);
|
|
|
|
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " # preds left : " << NumPredsLeft << "\n";
|
|
|
|
dbgs() << " # succs left : " << NumSuccsLeft << "\n";
|
2011-02-04 04:18:17 +01:00
|
|
|
dbgs() << " # rdefs left : " << NumRegDefsLeft << "\n";
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " Latency : " << Latency << "\n";
|
|
|
|
dbgs() << " Depth : " << Depth << "\n";
|
|
|
|
dbgs() << " Height : " << Height << "\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
|
|
|
|
if (Preds.size() != 0) {
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " Predecessors:\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
for (SUnit::const_succ_iterator I = Preds.begin(), E = Preds.end();
|
|
|
|
I != E; ++I) {
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " ";
|
2008-12-09 23:54:47 +01:00
|
|
|
switch (I->getKind()) {
|
2010-01-05 02:25:41 +01:00
|
|
|
case SDep::Data: dbgs() << "val "; break;
|
|
|
|
case SDep::Anti: dbgs() << "anti"; break;
|
|
|
|
case SDep::Output: dbgs() << "out "; break;
|
|
|
|
case SDep::Order: dbgs() << "ch "; break;
|
2008-12-09 23:54:47 +01:00
|
|
|
}
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "#";
|
|
|
|
dbgs() << I->getSUnit() << " - SU(" << I->getSUnit()->NodeNum << ")";
|
2008-12-09 23:54:47 +01:00
|
|
|
if (I->isArtificial())
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " *";
|
|
|
|
dbgs() << ": Latency=" << I->getLatency();
|
2011-06-15 19:16:12 +02:00
|
|
|
if (I->isAssignedRegDep())
|
|
|
|
dbgs() << " Reg=" << G->TRI->getName(I->getReg());
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (Succs.size() != 0) {
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " Successors:\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
for (SUnit::const_succ_iterator I = Succs.begin(), E = Succs.end();
|
|
|
|
I != E; ++I) {
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " ";
|
2008-12-09 23:54:47 +01:00
|
|
|
switch (I->getKind()) {
|
2010-01-05 02:25:41 +01:00
|
|
|
case SDep::Data: dbgs() << "val "; break;
|
|
|
|
case SDep::Anti: dbgs() << "anti"; break;
|
|
|
|
case SDep::Output: dbgs() << "out "; break;
|
|
|
|
case SDep::Order: dbgs() << "ch "; break;
|
2008-12-09 23:54:47 +01:00
|
|
|
}
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "#";
|
|
|
|
dbgs() << I->getSUnit() << " - SU(" << I->getSUnit()->NodeNum << ")";
|
2008-12-09 23:54:47 +01:00
|
|
|
if (I->isArtificial())
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << " *";
|
|
|
|
dbgs() << ": Latency=" << I->getLatency();
|
|
|
|
dbgs() << "\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
|
|
|
}
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "\n";
|
2008-11-20 00:18:57 +01:00
|
|
|
}
|
2008-11-20 02:26:25 +01:00
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
/// VerifySchedule - Verify that all SUnits were scheduled and that
|
|
|
|
/// their state is consistent.
|
|
|
|
///
|
|
|
|
void ScheduleDAG::VerifySchedule(bool isBottomUp) {
|
|
|
|
bool AnyNotSched = false;
|
|
|
|
unsigned DeadNodes = 0;
|
|
|
|
unsigned Noops = 0;
|
|
|
|
for (unsigned i = 0, e = SUnits.size(); i != e; ++i) {
|
|
|
|
if (!SUnits[i].isScheduled) {
|
|
|
|
if (SUnits[i].NumPreds == 0 && SUnits[i].NumSuccs == 0) {
|
|
|
|
++DeadNodes;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!AnyNotSched)
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
SUnits[i].dump(this);
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "has not been scheduled!\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
AnyNotSched = true;
|
|
|
|
}
|
2008-12-16 04:25:46 +01:00
|
|
|
if (SUnits[i].isScheduled &&
|
2009-11-03 21:57:50 +01:00
|
|
|
(isBottomUp ? SUnits[i].getHeight() : SUnits[i].getDepth()) >
|
2008-12-16 04:25:46 +01:00
|
|
|
unsigned(INT_MAX)) {
|
2008-11-20 02:26:25 +01:00
|
|
|
if (!AnyNotSched)
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
SUnits[i].dump(this);
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "has an unexpected "
|
2008-12-16 04:25:46 +01:00
|
|
|
<< (isBottomUp ? "Height" : "Depth") << " value!\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
AnyNotSched = true;
|
|
|
|
}
|
|
|
|
if (isBottomUp) {
|
|
|
|
if (SUnits[i].NumSuccsLeft != 0) {
|
|
|
|
if (!AnyNotSched)
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
SUnits[i].dump(this);
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "has successors left!\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
AnyNotSched = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (SUnits[i].NumPredsLeft != 0) {
|
|
|
|
if (!AnyNotSched)
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
SUnits[i].dump(this);
|
2010-01-05 02:25:41 +01:00
|
|
|
dbgs() << "has predecessors left!\n";
|
2008-11-20 02:26:25 +01:00
|
|
|
AnyNotSched = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (unsigned i = 0, e = Sequence.size(); i != e; ++i)
|
|
|
|
if (!Sequence[i])
|
|
|
|
++Noops;
|
|
|
|
assert(!AnyNotSched);
|
|
|
|
assert(Sequence.size() + DeadNodes - Noops == SUnits.size() &&
|
|
|
|
"The number of nodes scheduled doesn't match the expected number!");
|
|
|
|
}
|
|
|
|
#endif
|
2008-11-25 01:52:40 +01:00
|
|
|
|
2010-06-30 05:40:54 +02:00
|
|
|
/// InitDAGTopologicalSorting - create the initial topological
|
2008-11-25 01:52:40 +01:00
|
|
|
/// ordering from the DAG to be scheduled.
|
|
|
|
///
|
2010-06-30 05:40:54 +02:00
|
|
|
/// The idea of the algorithm is taken from
|
2008-11-25 01:52:40 +01:00
|
|
|
/// "Online algorithms for managing the topological order of
|
|
|
|
/// a directed acyclic graph" by David J. Pearce and Paul H.J. Kelly
|
2010-06-30 05:40:54 +02:00
|
|
|
/// This is the MNR algorithm, which was first introduced by
|
|
|
|
/// A. Marchetti-Spaccamela, U. Nanni and H. Rohnert in
|
2008-11-25 01:52:40 +01:00
|
|
|
/// "Maintaining a topological order under edge insertions".
|
|
|
|
///
|
2010-06-30 05:40:54 +02:00
|
|
|
/// Short description of the algorithm:
|
2008-11-25 01:52:40 +01:00
|
|
|
///
|
|
|
|
/// Topological ordering, ord, of a DAG maps each node to a topological
|
|
|
|
/// index so that for all edges X->Y it is the case that ord(X) < ord(Y).
|
|
|
|
///
|
2010-06-30 05:40:54 +02:00
|
|
|
/// This means that if there is a path from the node X to the node Z,
|
2008-11-25 01:52:40 +01:00
|
|
|
/// then ord(X) < ord(Z).
|
|
|
|
///
|
|
|
|
/// This property can be used to check for reachability of nodes:
|
2010-06-30 05:40:54 +02:00
|
|
|
/// if Z is reachable from X, then an insertion of the edge Z->X would
|
2008-11-25 01:52:40 +01:00
|
|
|
/// create a cycle.
|
|
|
|
///
|
|
|
|
/// The algorithm first computes a topological ordering for the DAG by
|
|
|
|
/// initializing the Index2Node and Node2Index arrays and then tries to keep
|
|
|
|
/// the ordering up-to-date after edge insertions by reordering the DAG.
|
|
|
|
///
|
|
|
|
/// On insertion of the edge X->Y, the algorithm first marks by calling DFS
|
|
|
|
/// the nodes reachable from Y, and then shifts them using Shift to lie
|
|
|
|
/// immediately after X in Index2Node.
|
|
|
|
void ScheduleDAGTopologicalSort::InitDAGTopologicalSorting() {
|
|
|
|
unsigned DAGSize = SUnits.size();
|
|
|
|
std::vector<SUnit*> WorkList;
|
|
|
|
WorkList.reserve(DAGSize);
|
|
|
|
|
|
|
|
Index2Node.resize(DAGSize);
|
|
|
|
Node2Index.resize(DAGSize);
|
|
|
|
|
|
|
|
// Initialize the data structures.
|
|
|
|
for (unsigned i = 0, e = DAGSize; i != e; ++i) {
|
|
|
|
SUnit *SU = &SUnits[i];
|
|
|
|
int NodeNum = SU->NodeNum;
|
|
|
|
unsigned Degree = SU->Succs.size();
|
|
|
|
// Temporarily use the Node2Index array as scratch space for degree counts.
|
|
|
|
Node2Index[NodeNum] = Degree;
|
|
|
|
|
|
|
|
// Is it a node without dependencies?
|
|
|
|
if (Degree == 0) {
|
|
|
|
assert(SU->Succs.empty() && "SUnit should have no successors");
|
|
|
|
// Collect leaf nodes.
|
|
|
|
WorkList.push_back(SU);
|
|
|
|
}
|
2010-06-30 05:40:54 +02:00
|
|
|
}
|
2008-11-25 01:52:40 +01:00
|
|
|
|
|
|
|
int Id = DAGSize;
|
|
|
|
while (!WorkList.empty()) {
|
|
|
|
SUnit *SU = WorkList.back();
|
|
|
|
WorkList.pop_back();
|
|
|
|
Allocate(SU->NodeNum, --Id);
|
|
|
|
for (SUnit::const_pred_iterator I = SU->Preds.begin(), E = SU->Preds.end();
|
|
|
|
I != E; ++I) {
|
2008-12-09 23:54:47 +01:00
|
|
|
SUnit *SU = I->getSUnit();
|
2008-11-25 01:52:40 +01:00
|
|
|
if (!--Node2Index[SU->NodeNum])
|
|
|
|
// If all dependencies of the node are processed already,
|
|
|
|
// then the node can be computed now.
|
|
|
|
WorkList.push_back(SU);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Visited.resize(DAGSize);
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
// Check correctness of the ordering
|
|
|
|
for (unsigned i = 0, e = DAGSize; i != e; ++i) {
|
|
|
|
SUnit *SU = &SUnits[i];
|
|
|
|
for (SUnit::const_pred_iterator I = SU->Preds.begin(), E = SU->Preds.end();
|
|
|
|
I != E; ++I) {
|
2010-06-30 05:40:54 +02:00
|
|
|
assert(Node2Index[SU->NodeNum] > Node2Index[I->getSUnit()->NodeNum] &&
|
2008-11-25 01:52:40 +01:00
|
|
|
"Wrong topological sorting");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-04-15 07:18:47 +02:00
|
|
|
/// AddPred - Updates the topological ordering to accommodate an edge
|
2008-11-25 01:52:40 +01:00
|
|
|
/// to be added from SUnit X to SUnit Y.
|
|
|
|
void ScheduleDAGTopologicalSort::AddPred(SUnit *Y, SUnit *X) {
|
|
|
|
int UpperBound, LowerBound;
|
|
|
|
LowerBound = Node2Index[Y->NodeNum];
|
|
|
|
UpperBound = Node2Index[X->NodeNum];
|
|
|
|
bool HasLoop = false;
|
|
|
|
// Is Ord(X) < Ord(Y) ?
|
|
|
|
if (LowerBound < UpperBound) {
|
|
|
|
// Update the topological order.
|
|
|
|
Visited.reset();
|
|
|
|
DFS(Y, UpperBound, HasLoop);
|
|
|
|
assert(!HasLoop && "Inserted edge creates a loop!");
|
|
|
|
// Recompute topological indexes.
|
|
|
|
Shift(Visited, LowerBound, UpperBound);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-15 07:18:47 +02:00
|
|
|
/// RemovePred - Updates the topological ordering to accommodate an
|
2008-11-25 01:52:40 +01:00
|
|
|
/// an edge to be removed from the specified node N from the predecessors
|
|
|
|
/// of the current node M.
|
|
|
|
void ScheduleDAGTopologicalSort::RemovePred(SUnit *M, SUnit *N) {
|
|
|
|
// InitDAGTopologicalSorting();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// DFS - Make a DFS traversal to mark all nodes reachable from SU and mark
|
|
|
|
/// all nodes affected by the edge insertion. These nodes will later get new
|
|
|
|
/// topological indexes by means of the Shift method.
|
2008-12-09 17:37:48 +01:00
|
|
|
void ScheduleDAGTopologicalSort::DFS(const SUnit *SU, int UpperBound,
|
2010-12-20 01:50:16 +01:00
|
|
|
bool &HasLoop) {
|
2008-11-25 01:52:40 +01:00
|
|
|
std::vector<const SUnit*> WorkList;
|
2010-06-30 05:40:54 +02:00
|
|
|
WorkList.reserve(SUnits.size());
|
2008-11-25 01:52:40 +01:00
|
|
|
|
|
|
|
WorkList.push_back(SU);
|
2008-12-23 18:22:32 +01:00
|
|
|
do {
|
2008-11-25 01:52:40 +01:00
|
|
|
SU = WorkList.back();
|
|
|
|
WorkList.pop_back();
|
|
|
|
Visited.set(SU->NodeNum);
|
|
|
|
for (int I = SU->Succs.size()-1; I >= 0; --I) {
|
2008-12-09 23:54:47 +01:00
|
|
|
int s = SU->Succs[I].getSUnit()->NodeNum;
|
2008-11-25 01:52:40 +01:00
|
|
|
if (Node2Index[s] == UpperBound) {
|
2010-06-30 05:40:54 +02:00
|
|
|
HasLoop = true;
|
2008-11-25 01:52:40 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Visit successors if not already and in affected region.
|
|
|
|
if (!Visited.test(s) && Node2Index[s] < UpperBound) {
|
2008-12-09 23:54:47 +01:00
|
|
|
WorkList.push_back(SU->Succs[I].getSUnit());
|
2010-06-30 05:40:54 +02:00
|
|
|
}
|
|
|
|
}
|
2008-12-23 18:22:32 +01:00
|
|
|
} while (!WorkList.empty());
|
2008-11-25 01:52:40 +01:00
|
|
|
}
|
|
|
|
|
2010-06-30 05:40:54 +02:00
|
|
|
/// Shift - Renumber the nodes so that the topological ordering is
|
2008-11-25 01:52:40 +01:00
|
|
|
/// preserved.
|
2010-06-30 05:40:54 +02:00
|
|
|
void ScheduleDAGTopologicalSort::Shift(BitVector& Visited, int LowerBound,
|
2008-12-09 17:37:48 +01:00
|
|
|
int UpperBound) {
|
2008-11-25 01:52:40 +01:00
|
|
|
std::vector<int> L;
|
|
|
|
int shift = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = LowerBound; i <= UpperBound; ++i) {
|
|
|
|
// w is node at topological index i.
|
|
|
|
int w = Index2Node[i];
|
|
|
|
if (Visited.test(w)) {
|
|
|
|
// Unmark.
|
|
|
|
Visited.reset(w);
|
|
|
|
L.push_back(w);
|
|
|
|
shift = shift + 1;
|
|
|
|
} else {
|
|
|
|
Allocate(w, i - shift);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned j = 0; j < L.size(); ++j) {
|
|
|
|
Allocate(L[j], i - shift);
|
|
|
|
i = i + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// WillCreateCycle - Returns true if adding an edge from SU to TargetSU will
|
|
|
|
/// create a cycle.
|
|
|
|
bool ScheduleDAGTopologicalSort::WillCreateCycle(SUnit *SU, SUnit *TargetSU) {
|
|
|
|
if (IsReachable(TargetSU, SU))
|
|
|
|
return true;
|
|
|
|
for (SUnit::pred_iterator I = SU->Preds.begin(), E = SU->Preds.end();
|
|
|
|
I != E; ++I)
|
2008-12-09 23:54:47 +01:00
|
|
|
if (I->isAssignedRegDep() &&
|
|
|
|
IsReachable(TargetSU, I->getSUnit()))
|
2008-11-25 01:52:40 +01:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// IsReachable - Checks if SU is reachable from TargetSU.
|
2008-12-09 17:37:48 +01:00
|
|
|
bool ScheduleDAGTopologicalSort::IsReachable(const SUnit *SU,
|
|
|
|
const SUnit *TargetSU) {
|
2008-11-25 01:52:40 +01:00
|
|
|
// If insertion of the edge SU->TargetSU would create a cycle
|
|
|
|
// then there is a path from TargetSU to SU.
|
|
|
|
int UpperBound, LowerBound;
|
|
|
|
LowerBound = Node2Index[TargetSU->NodeNum];
|
|
|
|
UpperBound = Node2Index[SU->NodeNum];
|
|
|
|
bool HasLoop = false;
|
|
|
|
// Is Ord(TargetSU) < Ord(SU) ?
|
|
|
|
if (LowerBound < UpperBound) {
|
|
|
|
Visited.reset();
|
2010-06-30 05:40:54 +02:00
|
|
|
// There may be a path from TargetSU to SU. Check for it.
|
2008-11-25 01:52:40 +01:00
|
|
|
DFS(TargetSU, UpperBound, HasLoop);
|
|
|
|
}
|
|
|
|
return HasLoop;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Allocate - assign the topological index to the node n.
|
|
|
|
void ScheduleDAGTopologicalSort::Allocate(int n, int index) {
|
|
|
|
Node2Index[n] = index;
|
|
|
|
Index2Node[index] = n;
|
|
|
|
}
|
|
|
|
|
2010-06-30 05:40:54 +02:00
|
|
|
ScheduleDAGTopologicalSort::
|
|
|
|
ScheduleDAGTopologicalSort(std::vector<SUnit> &sunits) : SUnits(sunits) {}
|
2009-01-15 23:18:12 +01:00
|
|
|
|
|
|
|
ScheduleHazardRecognizer::~ScheduleHazardRecognizer() {}
|