xref: /freebsd-src/contrib/llvm-project/llvm/tools/llvm-mca/PipelinePrinter.cpp (revision 349cc55c9796c4596a5b9904cd3281af295f878f)
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