10b57cec5SDimitry Andric //===--------------------- PipelinePrinter.cpp ------------------*- C++ -*-===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric /// \file
90b57cec5SDimitry Andric ///
100b57cec5SDimitry Andric /// This file implements the PipelinePrinter interface.
110b57cec5SDimitry Andric ///
120b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
130b57cec5SDimitry Andric
140b57cec5SDimitry Andric #include "PipelinePrinter.h"
15*fe6060f1SDimitry Andric #include "CodeRegion.h"
16*fe6060f1SDimitry Andric #include "Views/InstructionView.h"
170b57cec5SDimitry Andric
180b57cec5SDimitry Andric namespace llvm {
190b57cec5SDimitry Andric namespace mca {
200b57cec5SDimitry Andric
printRegionHeader(llvm::raw_ostream & OS) const21*fe6060f1SDimitry Andric void PipelinePrinter::printRegionHeader(llvm::raw_ostream &OS) const {
22*fe6060f1SDimitry Andric StringRef RegionName;
23*fe6060f1SDimitry Andric if (!Region.getDescription().empty())
24*fe6060f1SDimitry Andric RegionName = Region.getDescription();
25*fe6060f1SDimitry Andric
26*fe6060f1SDimitry Andric OS << "\n[" << RegionIdx << "] Code Region";
27*fe6060f1SDimitry Andric if (!RegionName.empty())
28*fe6060f1SDimitry Andric OS << " - " << RegionName;
29*fe6060f1SDimitry Andric OS << "\n\n";
300b57cec5SDimitry Andric }
31*fe6060f1SDimitry Andric
getJSONReportRegion() const32*fe6060f1SDimitry Andric json::Object PipelinePrinter::getJSONReportRegion() const {
33*fe6060f1SDimitry Andric json::Object JO;
34*fe6060f1SDimitry Andric
35*fe6060f1SDimitry Andric StringRef RegionName = "";
36*fe6060f1SDimitry Andric if (!Region.getDescription().empty())
37*fe6060f1SDimitry Andric RegionName = Region.getDescription();
38*fe6060f1SDimitry Andric
39*fe6060f1SDimitry Andric JO.try_emplace("Name", RegionName);
40*fe6060f1SDimitry Andric for (const auto &V : Views)
41*fe6060f1SDimitry Andric if (V->isSerializable())
42*fe6060f1SDimitry Andric JO.try_emplace(V->getNameAsString().str(), V->toJSON());
43*fe6060f1SDimitry Andric
44*fe6060f1SDimitry Andric return JO;
45*fe6060f1SDimitry Andric }
46*fe6060f1SDimitry Andric
getJSONSimulationParameters() const47*fe6060f1SDimitry Andric json::Object PipelinePrinter::getJSONSimulationParameters() const {
48*fe6060f1SDimitry Andric json::Object SimParameters({{"-mcpu", STI.getCPU()},
49*fe6060f1SDimitry Andric {"-mtriple", STI.getTargetTriple().getTriple()},
50*fe6060f1SDimitry Andric {"-march", STI.getTargetTriple().getArchName()}});
51*fe6060f1SDimitry Andric
52*fe6060f1SDimitry Andric const MCSchedModel &SM = STI.getSchedModel();
53*fe6060f1SDimitry Andric if (!SM.isOutOfOrder())
54*fe6060f1SDimitry Andric return SimParameters;
55*fe6060f1SDimitry Andric
56*fe6060f1SDimitry Andric if (PO.RegisterFileSize)
57*fe6060f1SDimitry Andric SimParameters.try_emplace("-register-file-size", PO.RegisterFileSize);
58*fe6060f1SDimitry Andric
59*fe6060f1SDimitry Andric if (!PO.AssumeNoAlias)
60*fe6060f1SDimitry Andric SimParameters.try_emplace("-noalias", PO.AssumeNoAlias);
61*fe6060f1SDimitry Andric
62*fe6060f1SDimitry Andric if (PO.DecodersThroughput)
63*fe6060f1SDimitry Andric SimParameters.try_emplace("-decoder-throughput", PO.DecodersThroughput);
64*fe6060f1SDimitry Andric
65*fe6060f1SDimitry Andric if (PO.MicroOpQueueSize)
66*fe6060f1SDimitry Andric SimParameters.try_emplace("-micro-op-queue-size", PO.MicroOpQueueSize);
67*fe6060f1SDimitry Andric
68*fe6060f1SDimitry Andric if (PO.DispatchWidth)
69*fe6060f1SDimitry Andric SimParameters.try_emplace("-dispatch", PO.DispatchWidth);
70*fe6060f1SDimitry Andric
71*fe6060f1SDimitry Andric if (PO.LoadQueueSize)
72*fe6060f1SDimitry Andric SimParameters.try_emplace("-lqueue", PO.LoadQueueSize);
73*fe6060f1SDimitry Andric
74*fe6060f1SDimitry Andric if (PO.StoreQueueSize)
75*fe6060f1SDimitry Andric SimParameters.try_emplace("-squeue", PO.StoreQueueSize);
76*fe6060f1SDimitry Andric
77*fe6060f1SDimitry Andric return SimParameters;
78*fe6060f1SDimitry Andric }
79*fe6060f1SDimitry Andric
getJSONTargetInfo() const80*fe6060f1SDimitry Andric json::Object PipelinePrinter::getJSONTargetInfo() const {
81*fe6060f1SDimitry Andric json::Array Resources;
82*fe6060f1SDimitry Andric const MCSchedModel &SM = STI.getSchedModel();
83*fe6060f1SDimitry Andric StringRef MCPU = STI.getCPU();
84*fe6060f1SDimitry Andric
85*fe6060f1SDimitry Andric for (unsigned I = 1, E = SM.getNumProcResourceKinds(); I < E; ++I) {
86*fe6060f1SDimitry Andric const MCProcResourceDesc &ProcResource = *SM.getProcResource(I);
87*fe6060f1SDimitry Andric unsigned NumUnits = ProcResource.NumUnits;
88*fe6060f1SDimitry Andric if (ProcResource.SubUnitsIdxBegin || !NumUnits)
89*fe6060f1SDimitry Andric continue;
90*fe6060f1SDimitry Andric
91*fe6060f1SDimitry Andric for (unsigned J = 0; J < NumUnits; ++J) {
92*fe6060f1SDimitry Andric std::string ResourceName = ProcResource.Name;
93*fe6060f1SDimitry Andric if (NumUnits > 1) {
94*fe6060f1SDimitry Andric ResourceName += ".";
95*fe6060f1SDimitry Andric ResourceName += J;
96*fe6060f1SDimitry Andric }
97*fe6060f1SDimitry Andric
98*fe6060f1SDimitry Andric Resources.push_back(ResourceName);
99*fe6060f1SDimitry Andric }
100*fe6060f1SDimitry Andric }
101*fe6060f1SDimitry Andric
102*fe6060f1SDimitry Andric return json::Object({{"CPUName", MCPU}, {"Resources", std::move(Resources)}});
103*fe6060f1SDimitry Andric }
104*fe6060f1SDimitry Andric
printReport(json::Object & JO) const105*fe6060f1SDimitry Andric void PipelinePrinter::printReport(json::Object &JO) const {
106*fe6060f1SDimitry Andric if (!RegionIdx) {
107*fe6060f1SDimitry Andric JO.try_emplace("TargetInfo", getJSONTargetInfo());
108*fe6060f1SDimitry Andric JO.try_emplace("SimulationParameters", getJSONSimulationParameters());
109*fe6060f1SDimitry Andric // Construct an array of regions.
110*fe6060f1SDimitry Andric JO.try_emplace("CodeRegions", json::Array());
111*fe6060f1SDimitry Andric }
112*fe6060f1SDimitry Andric
113*fe6060f1SDimitry Andric json::Array *Regions = JO.getArray("CodeRegions");
114*fe6060f1SDimitry Andric assert(Regions && "This array must exist!");
115*fe6060f1SDimitry Andric Regions->push_back(getJSONReportRegion());
116*fe6060f1SDimitry Andric }
117*fe6060f1SDimitry Andric
printReport(llvm::raw_ostream & OS) const118*fe6060f1SDimitry Andric void PipelinePrinter::printReport(llvm::raw_ostream &OS) const {
119*fe6060f1SDimitry Andric // Don't print the header of this region if it is the default region, and if
120*fe6060f1SDimitry Andric // it doesn't have an end location.
121*fe6060f1SDimitry Andric if (Region.startLoc().isValid() || Region.endLoc().isValid())
122*fe6060f1SDimitry Andric printRegionHeader(OS);
123*fe6060f1SDimitry Andric
124*fe6060f1SDimitry Andric for (const auto &V : Views)
125*fe6060f1SDimitry Andric V->printView(OS);
126*fe6060f1SDimitry Andric }
127*fe6060f1SDimitry Andric
128*fe6060f1SDimitry Andric } // namespace mca
1290b57cec5SDimitry Andric } // namespace llvm
130