forked from OSchip/llvm-project
213 lines
8.2 KiB
C++
213 lines
8.2 KiB
C++
//===--------------------- Scheduler.h ------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
/// \file
|
|
///
|
|
/// A scheduler for Processor Resource Units and Processor Resource Groups.
|
|
///
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_TOOLS_LLVM_MCA_SCHEDULER_H
|
|
#define LLVM_TOOLS_LLVM_MCA_SCHEDULER_H
|
|
|
|
#include "HardwareUnits/HardwareUnit.h"
|
|
#include "HardwareUnits/LSUnit.h"
|
|
#include "ResourceManager.h"
|
|
#include "llvm/ADT/SmallVector.h"
|
|
#include "llvm/MC/MCSchedule.h"
|
|
|
|
namespace mca {
|
|
|
|
class SchedulerStrategy {
|
|
public:
|
|
SchedulerStrategy() = default;
|
|
virtual ~SchedulerStrategy();
|
|
|
|
/// Returns true if Lhs should take priority over Rhs.
|
|
///
|
|
/// This method is used by class Scheduler to select the "best" ready
|
|
/// instruction to issue to the underlying pipelines.
|
|
virtual bool compare(const InstRef &Lhs, const InstRef &Rhs) const = 0;
|
|
};
|
|
|
|
/// Default instruction selection strategy used by class Scheduler.
|
|
class DefaultSchedulerStrategy : public SchedulerStrategy {
|
|
/// This method ranks instructions based on their age, and the number of known
|
|
/// users. The lower the rank value, the better.
|
|
int computeRank(const InstRef &Lhs) const {
|
|
return Lhs.getSourceIndex() - Lhs.getInstruction()->getNumUsers();
|
|
}
|
|
|
|
public:
|
|
DefaultSchedulerStrategy() = default;
|
|
virtual ~DefaultSchedulerStrategy();
|
|
|
|
bool compare(const InstRef &Lhs, const InstRef &Rhs) const override {
|
|
int LhsRank = computeRank(Lhs);
|
|
int RhsRank = computeRank(Rhs);
|
|
|
|
/// Prioritize older instructions over younger instructions to minimize the
|
|
/// pressure on the reorder buffer.
|
|
if (LhsRank == RhsRank)
|
|
return Lhs.getSourceIndex() < Rhs.getSourceIndex();
|
|
return LhsRank < RhsRank;
|
|
}
|
|
};
|
|
|
|
/// Class Scheduler is responsible for issuing instructions to pipeline
|
|
/// resources.
|
|
///
|
|
/// Internally, it delegates to a ResourceManager the management of processor
|
|
/// resources. This class is also responsible for tracking the progress of
|
|
/// instructions from the dispatch stage, until the write-back stage.
|
|
///
|
|
/// An instruction dispatched to the Scheduler is initially placed into either
|
|
/// the 'WaitSet' or the 'ReadySet' depending on the availability of the input
|
|
/// operands.
|
|
///
|
|
/// An instruction is moved from the WaitSet to the ReadySet when register
|
|
/// operands become available, and all memory dependencies are met.
|
|
/// Instructions that are moved from the WaitSet to the ReadySet transition
|
|
/// in state from 'IS_AVAILABLE' to 'IS_READY'.
|
|
///
|
|
/// On every cycle, the Scheduler checks if it can promote instructions from the
|
|
/// WaitSet to the ReadySet.
|
|
///
|
|
/// An Instruction is moved from the ReadySet the `IssuedSet` when it is issued
|
|
/// to a (one or more) pipeline(s). This event also causes an instruction state
|
|
/// transition (i.e. from state IS_READY, to state IS_EXECUTING). An Instruction
|
|
/// leaves the IssuedSet when it reaches the write-back stage.
|
|
class Scheduler : public HardwareUnit {
|
|
LSUnit *LSU;
|
|
|
|
// Instruction selection strategy for this Scheduler.
|
|
std::unique_ptr<SchedulerStrategy> Strategy;
|
|
|
|
// Hardware resources that are managed by this scheduler.
|
|
std::unique_ptr<ResourceManager> Resources;
|
|
|
|
std::vector<InstRef> WaitSet;
|
|
std::vector<InstRef> ReadySet;
|
|
std::vector<InstRef> IssuedSet;
|
|
|
|
/// Verify the given selection strategy and set the Strategy member
|
|
/// accordingly. If no strategy is provided, the DefaultSchedulerStrategy is
|
|
/// used.
|
|
void initializeStrategy(std::unique_ptr<SchedulerStrategy> S);
|
|
|
|
/// Issue an instruction without updating the ready queue.
|
|
void issueInstructionImpl(
|
|
InstRef &IR,
|
|
llvm::SmallVectorImpl<std::pair<ResourceRef, double>> &Pipes);
|
|
|
|
// Identify instructions that have finished executing, and remove them from
|
|
// the IssuedSet. References to executed instructions are added to input
|
|
// vector 'Executed'.
|
|
void updateIssuedSet(llvm::SmallVectorImpl<InstRef> &Executed);
|
|
|
|
// Try to promote instructions from WaitSet to ReadySet.
|
|
// Add promoted instructions to the 'Ready' vector in input.
|
|
void promoteToReadySet(llvm::SmallVectorImpl<InstRef> &Ready);
|
|
|
|
public:
|
|
Scheduler(const llvm::MCSchedModel &Model, LSUnit *Lsu)
|
|
: LSU(Lsu), Resources(llvm::make_unique<ResourceManager>(Model)) {
|
|
initializeStrategy(nullptr);
|
|
}
|
|
Scheduler(const llvm::MCSchedModel &Model, LSUnit *Lsu,
|
|
std::unique_ptr<SchedulerStrategy> SelectStrategy)
|
|
: LSU(Lsu), Resources(llvm::make_unique<ResourceManager>(Model)) {
|
|
initializeStrategy(std::move(SelectStrategy));
|
|
}
|
|
Scheduler(std::unique_ptr<ResourceManager> RM, LSUnit *Lsu,
|
|
std::unique_ptr<SchedulerStrategy> SelectStrategy)
|
|
: LSU(Lsu), Resources(std::move(RM)) {
|
|
initializeStrategy(std::move(SelectStrategy));
|
|
}
|
|
|
|
// Stalls generated by the scheduler.
|
|
enum Status {
|
|
SC_AVAILABLE,
|
|
SC_LOAD_QUEUE_FULL,
|
|
SC_STORE_QUEUE_FULL,
|
|
SC_BUFFERS_FULL,
|
|
SC_DISPATCH_GROUP_STALL,
|
|
};
|
|
|
|
/// Check if the instruction in 'IR' can be dispatched and returns an answer
|
|
/// in the form of a Status value.
|
|
///
|
|
/// The DispatchStage is responsible for querying the Scheduler before
|
|
/// dispatching new instructions. This routine is used for performing such
|
|
/// a query. If the instruction 'IR' can be dispatched, then true is
|
|
/// returned, otherwise false is returned with Event set to the stall type.
|
|
/// Internally, it also checks if the load/store unit is available.
|
|
Status isAvailable(const InstRef &IR) const;
|
|
|
|
/// Reserves buffer and LSUnit queue resources that are necessary to issue
|
|
/// this instruction.
|
|
///
|
|
/// Returns true if instruction IR is ready to be issued to the underlying
|
|
/// pipelines. Note that this operation cannot fail; it assumes that a
|
|
/// previous call to method `isAvailable(IR)` returned `SC_AVAILABLE`.
|
|
void dispatch(const InstRef &IR);
|
|
|
|
/// Returns true if IR is ready to be executed by the underlying pipelines.
|
|
/// This method assumes that IR has been previously dispatched.
|
|
bool isReady(const InstRef &IR) const;
|
|
|
|
/// Issue an instruction and populates a vector of used pipeline resources,
|
|
/// and a vector of instructions that transitioned to the ready state as a
|
|
/// result of this event.
|
|
void
|
|
issueInstruction(InstRef &IR,
|
|
llvm::SmallVectorImpl<std::pair<ResourceRef, double>> &Used,
|
|
llvm::SmallVectorImpl<InstRef> &Ready);
|
|
|
|
/// Returns true if IR has to be issued immediately, or if IR is a zero
|
|
/// latency instruction.
|
|
bool mustIssueImmediately(const InstRef &IR) const;
|
|
|
|
/// This routine notifies the Scheduler that a new cycle just started.
|
|
///
|
|
/// It notifies the underlying ResourceManager that a new cycle just started.
|
|
/// Vector `Freed` is populated with resourceRef related to resources that
|
|
/// have changed in state, and that are now available to new instructions.
|
|
/// Instructions executed are added to vector Executed, while vector Ready is
|
|
/// populated with instructions that have become ready in this new cycle.
|
|
void cycleEvent(llvm::SmallVectorImpl<ResourceRef> &Freed,
|
|
llvm::SmallVectorImpl<InstRef> &Ready,
|
|
llvm::SmallVectorImpl<InstRef> &Executed);
|
|
|
|
/// Convert a resource mask into a valid llvm processor resource identifier.
|
|
unsigned getResourceID(uint64_t Mask) const {
|
|
return Resources->resolveResourceMask(Mask);
|
|
}
|
|
|
|
/// Select the next instruction to issue from the ReadySet. Returns an invalid
|
|
/// instruction reference if there are no ready instructions, or if processor
|
|
/// resources are not available.
|
|
InstRef select();
|
|
|
|
#ifndef NDEBUG
|
|
// Update the ready queues.
|
|
void dump() const;
|
|
|
|
// This routine performs a sanity check. This routine should only be called
|
|
// when we know that 'IR' is not in the scheduler's instruction queues.
|
|
void sanityCheck(const InstRef &IR) const {
|
|
assert(llvm::find(WaitSet, IR) == WaitSet.end());
|
|
assert(llvm::find(ReadySet, IR) == ReadySet.end());
|
|
assert(llvm::find(IssuedSet, IR) == IssuedSet.end());
|
|
}
|
|
#endif // !NDEBUG
|
|
};
|
|
} // namespace mca
|
|
|
|
#endif // LLVM_TOOLS_LLVM_MCA_SCHEDULER_H
|