2017-08-10 08:46:15 +08:00
|
|
|
//===- GCNIterativeScheduler.h - GCN Scheduler ------------------*- C++ -*-===//
|
2017-03-21 21:15:46 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_LIB_TARGET_AMDGPU_GCNITERATIVESCHEDULER_H
|
|
|
|
#define LLVM_LIB_TARGET_AMDGPU_GCNITERATIVESCHEDULER_H
|
|
|
|
|
|
|
|
#include "GCNRegPressure.h"
|
2017-08-10 08:46:15 +08:00
|
|
|
#include "llvm/ADT/ArrayRef.h"
|
|
|
|
#include "llvm/CodeGen/MachineBasicBlock.h"
|
2017-03-21 21:15:46 +08:00
|
|
|
#include "llvm/CodeGen/MachineScheduler.h"
|
2017-08-10 08:46:15 +08:00
|
|
|
#include "llvm/Support/Allocator.h"
|
|
|
|
#include <limits>
|
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
2017-03-21 21:15:46 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2017-08-10 08:46:15 +08:00
|
|
|
class MachineInstr;
|
|
|
|
class SUnit;
|
|
|
|
class raw_ostream;
|
|
|
|
|
2017-03-21 21:15:46 +08:00
|
|
|
class GCNIterativeScheduler : public ScheduleDAGMILive {
|
2017-08-10 08:46:15 +08:00
|
|
|
using BaseClass = ScheduleDAGMILive;
|
|
|
|
|
2017-03-21 21:15:46 +08:00
|
|
|
public:
|
|
|
|
enum StrategyKind {
|
|
|
|
SCHEDULE_MINREGONLY,
|
|
|
|
SCHEDULE_MINREGFORCED,
|
2017-11-20 22:35:53 +08:00
|
|
|
SCHEDULE_LEGACYMAXOCCUPANCY,
|
|
|
|
SCHEDULE_ILP
|
2017-03-21 21:15:46 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
GCNIterativeScheduler(MachineSchedContext *C,
|
|
|
|
StrategyKind S);
|
|
|
|
|
|
|
|
void schedule() override;
|
|
|
|
|
|
|
|
void enterRegion(MachineBasicBlock *BB,
|
|
|
|
MachineBasicBlock::iterator Begin,
|
|
|
|
MachineBasicBlock::iterator End,
|
|
|
|
unsigned RegionInstrs) override;
|
|
|
|
|
|
|
|
void finalizeSchedule() override;
|
|
|
|
|
|
|
|
protected:
|
2017-08-10 08:46:15 +08:00
|
|
|
using ScheduleRef = ArrayRef<const SUnit *>;
|
2017-03-21 21:15:46 +08:00
|
|
|
|
|
|
|
struct TentativeSchedule {
|
2017-08-10 08:46:15 +08:00
|
|
|
std::vector<MachineInstr *> Schedule;
|
2017-03-21 21:15:46 +08:00
|
|
|
GCNRegPressure MaxPressure;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Region {
|
|
|
|
// Fields except for BestSchedule are supposed to reflect current IR state
|
|
|
|
// `const` fields are to emphasize they shouldn't change for any schedule.
|
|
|
|
MachineBasicBlock::iterator Begin;
|
|
|
|
// End is either a boundary instruction or end of basic block
|
|
|
|
const MachineBasicBlock::iterator End;
|
|
|
|
const unsigned NumRegionInstrs;
|
|
|
|
GCNRegPressure MaxPressure;
|
|
|
|
|
|
|
|
// best schedule for the region so far (not scheduled yet)
|
|
|
|
std::unique_ptr<TentativeSchedule> BestSchedule;
|
|
|
|
};
|
|
|
|
|
|
|
|
SpecificBumpPtrAllocator<Region> Alloc;
|
|
|
|
std::vector<Region*> Regions;
|
|
|
|
|
|
|
|
MachineSchedContext *Context;
|
|
|
|
const StrategyKind Strategy;
|
|
|
|
mutable GCNUpwardRPTracker UPTracker;
|
|
|
|
|
|
|
|
class BuildDAG;
|
|
|
|
class OverrideLegacyStrategy;
|
|
|
|
|
|
|
|
template <typename Range>
|
|
|
|
GCNRegPressure getSchedulePressure(const Region &R,
|
|
|
|
Range &&Schedule) const;
|
|
|
|
|
|
|
|
GCNRegPressure getRegionPressure(MachineBasicBlock::iterator Begin,
|
|
|
|
MachineBasicBlock::iterator End) const;
|
|
|
|
|
|
|
|
GCNRegPressure getRegionPressure(const Region &R) const {
|
|
|
|
return getRegionPressure(R.Begin, R.End);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setBestSchedule(Region &R,
|
|
|
|
ScheduleRef Schedule,
|
|
|
|
const GCNRegPressure &MaxRP = GCNRegPressure());
|
|
|
|
|
|
|
|
void scheduleBest(Region &R);
|
|
|
|
|
|
|
|
std::vector<MachineInstr*> detachSchedule(ScheduleRef Schedule) const;
|
|
|
|
|
|
|
|
void sortRegionsByPressure(unsigned TargetOcc);
|
|
|
|
|
|
|
|
template <typename Range>
|
|
|
|
void scheduleRegion(Region &R, Range &&Schedule,
|
|
|
|
const GCNRegPressure &MaxRP = GCNRegPressure());
|
|
|
|
|
|
|
|
unsigned tryMaximizeOccupancy(unsigned TargetOcc =
|
|
|
|
std::numeric_limits<unsigned>::max());
|
|
|
|
|
|
|
|
void scheduleLegacyMaxOccupancy(bool TryMaximizeOccupancy = true);
|
|
|
|
void scheduleMinReg(bool force = false);
|
2017-11-20 22:35:53 +08:00
|
|
|
void scheduleILP(bool TryMaximizeOccupancy = true);
|
2017-03-21 21:15:46 +08:00
|
|
|
|
|
|
|
void printRegions(raw_ostream &OS) const;
|
|
|
|
void printSchedResult(raw_ostream &OS,
|
|
|
|
const Region *R,
|
|
|
|
const GCNRegPressure &RP) const;
|
|
|
|
void printSchedRP(raw_ostream &OS,
|
|
|
|
const GCNRegPressure &Before,
|
|
|
|
const GCNRegPressure &After) const;
|
|
|
|
};
|
|
|
|
|
2017-08-10 08:46:15 +08:00
|
|
|
} // end namespace llvm
|
2017-03-21 21:15:46 +08:00
|
|
|
|
|
|
|
#endif // LLVM_LIB_TARGET_AMDGPU_GCNITERATIVESCHEDULER_H
|