//===--------------------- Backend.h ----------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// /// \file /// /// This file implements an OoO backend for the llvm-mca tool. /// //===----------------------------------------------------------------------===// #ifndef LLVM_TOOLS_LLVM_MCA_BACKEND_H #define LLVM_TOOLS_LLVM_MCA_BACKEND_H #include "Dispatch.h" #include "InstrBuilder.h" #include "Scheduler.h" #include "SourceMgr.h" namespace mca { class HWEventListener; class HWInstructionEvent; class HWStallEvent; /// \brief An out of order backend for a specific subtarget. /// /// It emulates an out-of-order execution of instructions. Instructions are /// fetched from a MCInst sequence managed by an object of class SourceMgr. /// Instructions are firstly dispatched to the schedulers and then executed. /// This class tracks the lifetime of an instruction from the moment where /// it gets dispatched to the schedulers, to the moment where it finishes /// executing and register writes are architecturally committed. /// In particular, it monitors changes in the state of every instruction /// in flight. /// Instructions are executed in a loop of iterations. The number of iterations /// is defined by the SourceMgr object. /// The Backend entrypoint is method 'Run()' which execute cycles in a loop /// until there are new instructions to dispatch, and not every instruction /// has been retired. /// Internally, the Backend collects statistical information in the form of /// histograms. For example, it tracks how the dispatch group size changes /// over time. class Backend { const llvm::MCSubtargetInfo &STI; InstrBuilder &IB; std::unique_ptr HWS; std::unique_ptr DU; SourceMgr &SM; unsigned Cycles; llvm::DenseMap> Instructions; std::set Listeners; void runCycle(unsigned Cycle); public: Backend(const llvm::MCSubtargetInfo &Subtarget, const llvm::MCRegisterInfo &MRI, InstrBuilder &B, SourceMgr &Source, unsigned DispatchWidth = 0, unsigned RegisterFileSize = 0, unsigned LoadQueueSize = 0, unsigned StoreQueueSize = 0, bool AssumeNoAlias = false) : STI(Subtarget), IB(B), HWS(llvm::make_unique(this, Subtarget.getSchedModel(), LoadQueueSize, StoreQueueSize, AssumeNoAlias)), DU(llvm::make_unique(this, Subtarget.getSchedModel(), MRI, RegisterFileSize, DispatchWidth, HWS.get())), SM(Source), Cycles(0) { HWS->setDispatchUnit(DU.get()); } void run() { while (SM.hasNext() || !DU->isRCUEmpty()) runCycle(Cycles++); } const Instruction &getInstruction(unsigned Index) const { const auto It = Instructions.find(Index); assert(It != Instructions.end() && "no running instructions with index"); assert(It->second); return *It->second; } void eraseInstruction(unsigned Index) { Instructions.erase(Index); } void addEventListener(HWEventListener *Listener); void notifyCycleBegin(unsigned Cycle); void notifyInstructionEvent(const HWInstructionEvent &Event); void notifyStallEvent(const HWStallEvent &Event); void notifyResourceAvailable(const ResourceRef &RR); void notifyReservedBuffers(llvm::ArrayRef Buffers); void notifyReleasedBuffers(llvm::ArrayRef Buffers); void notifyCycleEnd(unsigned Cycle); }; } // namespace mca #endif