2012-09-04 14:49:56 +00:00
|
|
|
//===-- HexagonMachineScheduler.h - Custom Hexagon MI scheduler. ----===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Custom Hexagon MI scheduler.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef HEXAGONASMPRINTER_H
|
|
|
|
#define HEXAGONASMPRINTER_H
|
|
|
|
|
2012-12-04 07:12:27 +00:00
|
|
|
#include "llvm/ADT/OwningPtr.h"
|
|
|
|
#include "llvm/ADT/PriorityQueue.h"
|
|
|
|
#include "llvm/Analysis/AliasAnalysis.h"
|
2012-09-04 14:49:56 +00:00
|
|
|
#include "llvm/CodeGen/LiveIntervalAnalysis.h"
|
|
|
|
#include "llvm/CodeGen/MachineScheduler.h"
|
|
|
|
#include "llvm/CodeGen/Passes.h"
|
|
|
|
#include "llvm/CodeGen/RegisterClassInfo.h"
|
|
|
|
#include "llvm/CodeGen/RegisterPressure.h"
|
|
|
|
#include "llvm/CodeGen/ResourcePriorityQueue.h"
|
|
|
|
#include "llvm/CodeGen/ScheduleDAGInstrs.h"
|
|
|
|
#include "llvm/CodeGen/ScheduleHazardRecognizer.h"
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2012-12-04 07:12:27 +00:00
|
|
|
#include "llvm/Target/TargetInstrInfo.h"
|
2012-09-04 14:49:56 +00:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
//===----------------------------------------------------------------------===//
|
2012-09-10 17:31:34 +00:00
|
|
|
// ConvergingVLIWScheduler - Implementation of the standard
|
|
|
|
// MachineSchedStrategy.
|
2012-09-04 14:49:56 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
class VLIWResourceModel {
|
|
|
|
/// ResourcesModel - Represents VLIW state.
|
|
|
|
/// Not limited to VLIW targets per say, but assumes
|
|
|
|
/// definition of DFA by a target.
|
|
|
|
DFAPacketizer *ResourcesModel;
|
|
|
|
|
2012-10-10 05:43:09 +00:00
|
|
|
const TargetSchedModel *SchedModel;
|
2012-09-04 14:49:56 +00:00
|
|
|
|
|
|
|
/// Local packet/bundle model. Purely
|
|
|
|
/// internal to the MI schedulre at the time.
|
|
|
|
std::vector<SUnit*> Packet;
|
|
|
|
|
|
|
|
/// Total packets created.
|
|
|
|
unsigned TotalPackets;
|
|
|
|
|
|
|
|
public:
|
2012-10-10 05:43:09 +00:00
|
|
|
VLIWResourceModel(const TargetMachine &TM, const TargetSchedModel *SM) :
|
|
|
|
SchedModel(SM), TotalPackets(0) {
|
2012-09-04 14:49:56 +00:00
|
|
|
ResourcesModel = TM.getInstrInfo()->CreateTargetScheduleState(&TM,NULL);
|
|
|
|
|
2012-09-10 17:31:34 +00:00
|
|
|
// This hard requirement could be relaxed,
|
|
|
|
// but for now do not let it proceed.
|
|
|
|
assert(ResourcesModel && "Unimplemented CreateTargetScheduleState.");
|
|
|
|
|
2012-10-10 05:43:09 +00:00
|
|
|
Packet.resize(SchedModel->getIssueWidth());
|
2012-09-04 14:49:56 +00:00
|
|
|
Packet.clear();
|
|
|
|
ResourcesModel->clearResources();
|
|
|
|
}
|
|
|
|
|
|
|
|
~VLIWResourceModel() {
|
|
|
|
delete ResourcesModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetPacketState() {
|
|
|
|
Packet.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetDFA() {
|
|
|
|
ResourcesModel->clearResources();
|
|
|
|
}
|
|
|
|
|
2012-09-10 17:31:34 +00:00
|
|
|
void reset() {
|
|
|
|
Packet.clear();
|
|
|
|
ResourcesModel->clearResources();
|
|
|
|
}
|
|
|
|
|
2012-09-04 14:49:56 +00:00
|
|
|
bool isResourceAvailable(SUnit *SU);
|
2012-09-10 17:31:34 +00:00
|
|
|
bool reserveResources(SUnit *SU);
|
2012-09-04 14:49:56 +00:00
|
|
|
unsigned getTotalPackets() const { return TotalPackets; }
|
|
|
|
};
|
|
|
|
|
2012-09-11 00:39:15 +00:00
|
|
|
/// Extend the standard ScheduleDAGMI to provide more context and override the
|
|
|
|
/// top-level schedule() driver.
|
|
|
|
class VLIWMachineScheduler : public ScheduleDAGMI {
|
2012-09-04 14:49:56 +00:00
|
|
|
public:
|
|
|
|
VLIWMachineScheduler(MachineSchedContext *C, MachineSchedStrategy *S):
|
2012-09-14 12:19:58 +00:00
|
|
|
ScheduleDAGMI(C, S) {}
|
2012-09-04 14:49:56 +00:00
|
|
|
|
|
|
|
/// Schedule - This is called back from ScheduleDAGInstrs::Run() when it's
|
|
|
|
/// time to do some work.
|
2012-09-11 00:39:15 +00:00
|
|
|
virtual void schedule();
|
2012-09-14 15:07:59 +00:00
|
|
|
/// Perform platform specific DAG postprocessing.
|
|
|
|
void postprocessDAG();
|
2012-09-04 14:49:56 +00:00
|
|
|
};
|
2012-09-10 17:31:34 +00:00
|
|
|
|
|
|
|
/// ConvergingVLIWScheduler shrinks the unscheduled zone using heuristics
|
|
|
|
/// to balance the schedule.
|
|
|
|
class ConvergingVLIWScheduler : public MachineSchedStrategy {
|
|
|
|
|
|
|
|
/// Store the state used by ConvergingVLIWScheduler heuristics, required
|
|
|
|
/// for the lifetime of one invocation of pickNode().
|
|
|
|
struct SchedCandidate {
|
|
|
|
// The best SUnit candidate.
|
|
|
|
SUnit *SU;
|
|
|
|
|
|
|
|
// Register pressure values for the best candidate.
|
|
|
|
RegPressureDelta RPDelta;
|
|
|
|
|
|
|
|
// Best scheduling cost.
|
|
|
|
int SCost;
|
|
|
|
|
|
|
|
SchedCandidate(): SU(NULL), SCost(0) {}
|
|
|
|
};
|
|
|
|
/// Represent the type of SchedCandidate found within a single queue.
|
|
|
|
enum CandResult {
|
|
|
|
NoCand, NodeOrder, SingleExcess, SingleCritical, SingleMax, MultiPressure,
|
|
|
|
BestCost};
|
|
|
|
|
|
|
|
/// Each Scheduling boundary is associated with ready queues. It tracks the
|
|
|
|
/// current cycle in whichever direction at has moved, and maintains the state
|
|
|
|
/// of "hazards" and other interlocks at the current cycle.
|
|
|
|
struct SchedBoundary {
|
|
|
|
VLIWMachineScheduler *DAG;
|
2012-10-10 05:43:09 +00:00
|
|
|
const TargetSchedModel *SchedModel;
|
2012-09-10 17:31:34 +00:00
|
|
|
|
|
|
|
ReadyQueue Available;
|
|
|
|
ReadyQueue Pending;
|
|
|
|
bool CheckPending;
|
|
|
|
|
|
|
|
ScheduleHazardRecognizer *HazardRec;
|
|
|
|
VLIWResourceModel *ResourceModel;
|
|
|
|
|
|
|
|
unsigned CurrCycle;
|
|
|
|
unsigned IssueCount;
|
|
|
|
|
|
|
|
/// MinReadyCycle - Cycle of the soonest available instruction.
|
|
|
|
unsigned MinReadyCycle;
|
|
|
|
|
|
|
|
// Remember the greatest min operand latency.
|
|
|
|
unsigned MaxMinLatency;
|
|
|
|
|
|
|
|
/// Pending queues extend the ready queues with the same ID and the
|
|
|
|
/// PendingFlag set.
|
|
|
|
SchedBoundary(unsigned ID, const Twine &Name):
|
2012-10-10 05:43:09 +00:00
|
|
|
DAG(0), SchedModel(0), Available(ID, Name+".A"),
|
2012-09-10 17:31:34 +00:00
|
|
|
Pending(ID << ConvergingVLIWScheduler::LogMaxQID, Name+".P"),
|
|
|
|
CheckPending(false), HazardRec(0), ResourceModel(0),
|
|
|
|
CurrCycle(0), IssueCount(0),
|
|
|
|
MinReadyCycle(UINT_MAX), MaxMinLatency(0) {}
|
|
|
|
|
|
|
|
~SchedBoundary() {
|
|
|
|
delete ResourceModel;
|
|
|
|
delete HazardRec;
|
|
|
|
}
|
|
|
|
|
2012-10-10 05:43:09 +00:00
|
|
|
void init(VLIWMachineScheduler *dag, const TargetSchedModel *smodel) {
|
|
|
|
DAG = dag;
|
|
|
|
SchedModel = smodel;
|
|
|
|
}
|
|
|
|
|
2012-09-10 17:31:34 +00:00
|
|
|
bool isTop() const {
|
|
|
|
return Available.getID() == ConvergingVLIWScheduler::TopQID;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool checkHazard(SUnit *SU);
|
|
|
|
|
|
|
|
void releaseNode(SUnit *SU, unsigned ReadyCycle);
|
|
|
|
|
|
|
|
void bumpCycle();
|
|
|
|
|
|
|
|
void bumpNode(SUnit *SU);
|
|
|
|
|
|
|
|
void releasePending();
|
|
|
|
|
|
|
|
void removeReady(SUnit *SU);
|
|
|
|
|
|
|
|
SUnit *pickOnlyChoice();
|
|
|
|
};
|
|
|
|
|
|
|
|
VLIWMachineScheduler *DAG;
|
2012-10-10 05:43:09 +00:00
|
|
|
const TargetSchedModel *SchedModel;
|
2012-09-10 17:31:34 +00:00
|
|
|
const TargetRegisterInfo *TRI;
|
|
|
|
|
|
|
|
// State of the top and bottom scheduled instruction boundaries.
|
|
|
|
SchedBoundary Top;
|
|
|
|
SchedBoundary Bot;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// SUnit::NodeQueueId: 0 (none), 1 (top), 2 (bot), 3 (both)
|
|
|
|
enum {
|
|
|
|
TopQID = 1,
|
|
|
|
BotQID = 2,
|
|
|
|
LogMaxQID = 2
|
|
|
|
};
|
|
|
|
|
|
|
|
ConvergingVLIWScheduler():
|
2012-10-10 05:43:09 +00:00
|
|
|
DAG(0), SchedModel(0), TRI(0), Top(TopQID, "TopQ"), Bot(BotQID, "BotQ") {}
|
2012-09-10 17:31:34 +00:00
|
|
|
|
2012-09-11 00:39:15 +00:00
|
|
|
virtual void initialize(ScheduleDAGMI *dag);
|
2012-09-10 17:31:34 +00:00
|
|
|
|
|
|
|
virtual SUnit *pickNode(bool &IsTopNode);
|
|
|
|
|
|
|
|
virtual void schedNode(SUnit *SU, bool IsTopNode);
|
|
|
|
|
|
|
|
virtual void releaseTopNode(SUnit *SU);
|
|
|
|
|
|
|
|
virtual void releaseBottomNode(SUnit *SU);
|
|
|
|
|
2012-09-14 15:07:59 +00:00
|
|
|
unsigned ReportPackets() {
|
|
|
|
return Top.ResourceModel->getTotalPackets() +
|
|
|
|
Bot.ResourceModel->getTotalPackets();
|
|
|
|
}
|
|
|
|
|
2012-09-10 17:31:34 +00:00
|
|
|
protected:
|
|
|
|
SUnit *pickNodeBidrectional(bool &IsTopNode);
|
|
|
|
|
|
|
|
int SchedulingCost(ReadyQueue &Q,
|
|
|
|
SUnit *SU, SchedCandidate &Candidate,
|
|
|
|
RegPressureDelta &Delta, bool verbose);
|
|
|
|
|
|
|
|
CandResult pickNodeFromQueue(ReadyQueue &Q,
|
|
|
|
const RegPressureTracker &RPTracker,
|
|
|
|
SchedCandidate &Candidate);
|
|
|
|
#ifndef NDEBUG
|
|
|
|
void traceCandidate(const char *Label, const ReadyQueue &Q, SUnit *SU,
|
|
|
|
PressureElement P = PressureElement());
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2012-09-04 14:49:56 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|