2016-08-30 03:42:52 +08:00
|
|
|
//===-- GCNSchedStrategy.h - GCN Scheduler Strategy -*- C++ -*-------------===//
|
|
|
|
//
|
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-08-30 03:42:52 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
/// \file
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_LIB_TARGET_AMDGPU_GCNSCHEDSTRATEGY_H
|
|
|
|
#define LLVM_LIB_TARGET_AMDGPU_GCNSCHEDSTRATEGY_H
|
|
|
|
|
2017-05-16 23:43:52 +08:00
|
|
|
#include "GCNRegPressure.h"
|
2016-08-30 03:42:52 +08:00
|
|
|
#include "llvm/CodeGen/MachineScheduler.h"
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2017-03-01 03:20:33 +08:00
|
|
|
class SIMachineFunctionInfo;
|
2016-08-30 03:42:52 +08:00
|
|
|
class SIRegisterInfo;
|
2018-07-12 04:59:01 +08:00
|
|
|
class GCNSubtarget;
|
2016-08-30 03:42:52 +08:00
|
|
|
|
|
|
|
/// This is a minimal scheduler strategy. The main difference between this
|
|
|
|
/// and the GenericScheduler is that GCNSchedStrategy uses different
|
|
|
|
/// heuristics to determine excess/critical pressure sets. Its goal is to
|
|
|
|
/// maximize kernel occupancy (i.e. maximum number of waves per simd).
|
2019-05-09 06:10:04 +08:00
|
|
|
class GCNMaxOccupancySchedStrategy final : public GenericScheduler {
|
2017-02-16 01:19:50 +08:00
|
|
|
friend class GCNScheduleDAGMILive;
|
2016-08-30 03:42:52 +08:00
|
|
|
|
|
|
|
SUnit *pickNodeBidirectional(bool &IsTopNode);
|
|
|
|
|
|
|
|
void pickNodeFromQueue(SchedBoundary &Zone, const CandPolicy &ZonePolicy,
|
|
|
|
const RegPressureTracker &RPTracker,
|
|
|
|
SchedCandidate &Cand);
|
|
|
|
|
|
|
|
void initCandidate(SchedCandidate &Cand, SUnit *SU,
|
|
|
|
bool AtTop, const RegPressureTracker &RPTracker,
|
|
|
|
const SIRegisterInfo *SRI,
|
2017-02-16 01:19:50 +08:00
|
|
|
unsigned SGPRPressure, unsigned VGPRPressure);
|
2016-08-30 03:42:52 +08:00
|
|
|
|
2019-09-06 06:44:06 +08:00
|
|
|
std::vector<unsigned> Pressure;
|
|
|
|
std::vector<unsigned> MaxPressure;
|
|
|
|
|
2017-02-16 01:19:50 +08:00
|
|
|
unsigned SGPRExcessLimit;
|
|
|
|
unsigned VGPRExcessLimit;
|
|
|
|
unsigned SGPRCriticalLimit;
|
|
|
|
unsigned VGPRCriticalLimit;
|
2016-08-30 03:42:52 +08:00
|
|
|
|
2017-03-01 03:20:33 +08:00
|
|
|
unsigned TargetOccupancy;
|
|
|
|
|
|
|
|
MachineFunction *MF;
|
|
|
|
|
2016-08-30 03:42:52 +08:00
|
|
|
public:
|
|
|
|
GCNMaxOccupancySchedStrategy(const MachineSchedContext *C);
|
|
|
|
|
|
|
|
SUnit *pickNode(bool &IsTopNode) override;
|
2017-02-16 01:19:50 +08:00
|
|
|
|
|
|
|
void initialize(ScheduleDAGMI *DAG) override;
|
2017-03-21 21:15:46 +08:00
|
|
|
|
|
|
|
void setTargetOccupancy(unsigned Occ) { TargetOccupancy = Occ; }
|
2017-02-16 01:19:50 +08:00
|
|
|
};
|
|
|
|
|
2019-05-09 06:10:04 +08:00
|
|
|
class GCNScheduleDAGMILive final : public ScheduleDAGMILive {
|
2017-03-01 01:22:39 +08:00
|
|
|
|
2020-01-24 08:18:16 +08:00
|
|
|
enum : unsigned {
|
|
|
|
Collect,
|
|
|
|
InitialSchedule,
|
|
|
|
UnclusteredReschedule,
|
|
|
|
ClusteredLowOccupancyReschedule,
|
|
|
|
LastStage = ClusteredLowOccupancyReschedule
|
|
|
|
};
|
|
|
|
|
2018-07-12 04:59:01 +08:00
|
|
|
const GCNSubtarget &ST;
|
2017-03-01 03:20:33 +08:00
|
|
|
|
2018-05-31 13:36:04 +08:00
|
|
|
SIMachineFunctionInfo &MFI;
|
2017-03-01 03:20:33 +08:00
|
|
|
|
2017-07-13 14:48:39 +08:00
|
|
|
// Occupancy target at the beginning of function scheduling cycle.
|
2017-03-01 03:20:33 +08:00
|
|
|
unsigned StartingOccupancy;
|
|
|
|
|
|
|
|
// Minimal real occupancy recorder for the function.
|
|
|
|
unsigned MinOccupancy;
|
|
|
|
|
|
|
|
// Scheduling stage number.
|
|
|
|
unsigned Stage;
|
|
|
|
|
2017-05-17 00:11:26 +08:00
|
|
|
// Current region index.
|
|
|
|
size_t RegionIdx;
|
|
|
|
|
2019-04-25 07:32:21 +08:00
|
|
|
// Vector of regions recorder for later rescheduling
|
2017-03-29 05:48:54 +08:00
|
|
|
SmallVector<std::pair<MachineBasicBlock::iterator,
|
|
|
|
MachineBasicBlock::iterator>, 32> Regions;
|
2017-03-01 03:20:33 +08:00
|
|
|
|
2020-01-24 08:18:16 +08:00
|
|
|
// Records if a region is not yet scheduled, or schedule has been reverted,
|
|
|
|
// or we generally desire to reschedule it.
|
|
|
|
BitVector RescheduleRegions;
|
|
|
|
|
2017-05-17 00:11:26 +08:00
|
|
|
// Region live-in cache.
|
|
|
|
SmallVector<GCNRPTracker::LiveRegSet, 32> LiveIns;
|
|
|
|
|
|
|
|
// Region pressure cache.
|
|
|
|
SmallVector<GCNRegPressure, 32> Pressure;
|
2017-03-01 01:22:39 +08:00
|
|
|
|
2017-05-17 00:11:26 +08:00
|
|
|
// Temporary basic block live-in cache.
|
|
|
|
DenseMap<const MachineBasicBlock*, GCNRPTracker::LiveRegSet> MBBLiveIns;
|
2017-03-01 01:22:39 +08:00
|
|
|
|
2019-06-18 19:43:17 +08:00
|
|
|
DenseMap<MachineInstr *, GCNRPTracker::LiveRegSet> BBLiveInMap;
|
|
|
|
DenseMap<MachineInstr *, GCNRPTracker::LiveRegSet> getBBLiveInMap() const;
|
|
|
|
|
2017-05-16 23:43:52 +08:00
|
|
|
// Return current region pressure.
|
|
|
|
GCNRegPressure getRealRegPressure() const;
|
2017-03-01 01:22:39 +08:00
|
|
|
|
2017-05-17 00:11:26 +08:00
|
|
|
// Compute and cache live-ins and pressure for all regions in block.
|
|
|
|
void computeBlockPressure(const MachineBasicBlock *MBB);
|
|
|
|
|
|
|
|
|
2017-02-16 01:19:50 +08:00
|
|
|
public:
|
|
|
|
GCNScheduleDAGMILive(MachineSchedContext *C,
|
2017-03-01 03:20:33 +08:00
|
|
|
std::unique_ptr<MachineSchedStrategy> S);
|
|
|
|
|
2017-02-16 01:19:50 +08:00
|
|
|
void schedule() override;
|
2017-03-01 01:22:39 +08:00
|
|
|
|
|
|
|
void finalizeSchedule() override;
|
2016-08-30 03:42:52 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // End namespace llvm
|
|
|
|
|
|
|
|
#endif // GCNSCHEDSTRATEGY_H
|