1 //===--------------------- Pipeline.cpp -------------------------*- C++ -*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
10 /// This file implements an ordered container of stages that simulate the
11 /// pipeline of a hardware backend.
13 //===----------------------------------------------------------------------===//
15 #include "llvm/MCA/Pipeline.h"
16 #include "llvm/MCA/HWEventListener.h"
17 #include "llvm/Support/Debug.h"
22 #define DEBUG_TYPE "llvm-mca"
24 void Pipeline::addEventListener(HWEventListener
*Listener
) {
26 Listeners
.insert(Listener
);
27 for (auto &S
: Stages
)
28 S
->addListener(Listener
);
31 bool Pipeline::hasWorkToProcess() {
32 return any_of(Stages
, [](const std::unique_ptr
<Stage
> &S
) {
33 return S
->hasWorkToComplete();
37 Expected
<unsigned> Pipeline::run() {
38 assert(!Stages
.empty() && "Unexpected empty pipeline found!");
43 if (Error Err
= runCycle())
44 return std::move(Err
);
47 } while (hasWorkToProcess());
52 Error
Pipeline::runCycle() {
53 Error Err
= ErrorSuccess();
54 // Update stages before we start processing new instructions.
55 for (auto I
= Stages
.rbegin(), E
= Stages
.rend(); I
!= E
&& !Err
; ++I
) {
56 const std::unique_ptr
<Stage
> &S
= *I
;
58 Err
= S
->cycleResume();
60 Err
= S
->cycleStart();
63 CurrentState
= State::Started
;
65 // Now fetch and execute new instructions.
67 Stage
&FirstStage
= *Stages
[0];
68 while (!Err
&& FirstStage
.isAvailable(IR
))
69 Err
= FirstStage
.execute(IR
);
71 if (Err
.isA
<InstStreamPause
>()) {
72 CurrentState
= State::Paused
;
76 // Update stages in preparation for a new cycle.
77 for (const std::unique_ptr
<Stage
> &S
: Stages
) {
86 void Pipeline::appendStage(std::unique_ptr
<Stage
> S
) {
87 assert(S
&& "Invalid null stage in input!");
88 if (!Stages
.empty()) {
89 Stage
*Last
= Stages
.back().get();
90 Last
->setNextInSequence(S
.get());
93 Stages
.push_back(std::move(S
));
96 void Pipeline::notifyCycleBegin() {
97 LLVM_DEBUG(dbgs() << "\n[E] Cycle begin: " << Cycles
<< '\n');
98 for (HWEventListener
*Listener
: Listeners
)
99 Listener
->onCycleBegin();
102 void Pipeline::notifyCycleEnd() {
103 LLVM_DEBUG(dbgs() << "[E] Cycle end: " << Cycles
<< "\n");
104 for (HWEventListener
*Listener
: Listeners
)
105 Listener
->onCycleEnd();