[llvm] [MCA] New option -scheduling-info (PR #130574)
Andrea Di Biagio via llvm-commits
llvm-commits at lists.llvm.org
Wed Mar 12 07:50:32 PDT 2025
================
@@ -14,10 +14,121 @@
#include "Views/InstructionInfoView.h"
#include "llvm/Support/FormattedStream.h"
#include "llvm/Support/JSON.h"
+#include "llvm/Support/WithColor.h"
namespace llvm {
namespace mca {
+void InstructionInfoView::getComment(const MCInst &MCI,
+ std::string &CommentString) const {
+ StringRef s = MCI.getLoc().getPointer();
+ std::string InstrStr;
+ size_t pos = 0, pos_cmt = 0;
+
+ // Recognized comments are after assembly instructions on the same line.
+ // It is usefull to add in comment scheduling information from architecture
+ // specification.
+ // '#' comment mark is not supported by llvm-mca
+
+ CommentString = "";
+ if ((pos = s.find("\n")) != std::string::npos) {
+ InstrStr = s.substr(0, pos);
+ // C style comment
+ if (((pos_cmt = InstrStr.find("/*")) != std::string::npos) &&
+ ((pos = InstrStr.find("*/")) != std::string::npos)) {
+ CommentString = InstrStr.substr(pos_cmt, pos);
+ return;
+ }
+ // C++ style comment
+ if ((pos_cmt = InstrStr.find("//")) != std::string::npos) {
+ CommentString = InstrStr.substr(pos_cmt, pos);
+ return;
+ }
+ }
+ return;
+}
+
+void InstructionInfoView::printSchedulingInfoView(raw_ostream &OS) const {
+ std::string Buffer;
+ std::string CommentString;
+ raw_string_ostream TempStream(Buffer);
+ formatted_raw_ostream FOS(TempStream);
+
+ ArrayRef<MCInst> Source = getSource();
+ if (!Source.size())
+ return;
+
+ IIVDVec IIVD(Source.size());
+ collectData(IIVD);
+
+ FOS << "\n\nResources:\n";
+ const MCSchedModel &SM = getSubTargetInfo().getSchedModel();
+ for (unsigned I = 1, ResourceIndex = 0, E = SM.getNumProcResourceKinds();
+ I < E; ++I) {
+ const MCProcResourceDesc &ProcResource = *SM.getProcResource(I);
+ unsigned NumUnits = ProcResource.NumUnits;
+ // Skip invalid resources with zero units.
+ if (!NumUnits)
+ continue;
+
+ FOS << '[' << ResourceIndex << ']';
+ FOS.PadToColumn(6);
+ FOS << "- " << ProcResource.Name << ':' << NumUnits;
+ if (ProcResource.SubUnitsIdxBegin) {
+ FOS.PadToColumn(20);
+ for (unsigned U = 0; U < NumUnits; ++U) {
+ FOS << SM.getProcResource(ProcResource.SubUnitsIdxBegin[U])->Name;
+ if ((U + 1) < NumUnits) {
+ FOS << ", ";
+ }
+ }
+ }
+ FOS << '\n';
+ ResourceIndex++;
+ }
+
+ FOS << "\n\nScheduling Info:\n";
+ FOS << "[1]: #uOps\n[2]: Latency\n[3]: Bypass Latency\n"
+ << "[4]: Throughput\n[5]: Resources\n"
+ << "[6]: LLVM OpcodeName\n";
+
+ // paddings for each scheduling info output. Start at [2]
+ std::vector<unsigned> paddings = {7, 12, 18, 27, 94, 113, 150};
+ for (unsigned i = 0; i < paddings.size() - 1; i++) {
+ FOS << "[" << i + 1 << "]";
+ FOS.PadToColumn(paddings[i]);
+ }
+ FOS << "Instructions:\n";
+
+ for (const auto &[Index, IIVDEntry, Inst] : enumerate(IIVD, Source)) {
+ getComment(Inst, CommentString);
+
+ FOS << " " << IIVDEntry.NumMicroOpcodes;
+ FOS.PadToColumn(paddings[0]);
+ FOS << " " << IIVDEntry.Latency;
+ FOS.PadToColumn(paddings[1]);
+ FOS << " " << IIVDEntry.Bypass;
+ FOS.PadToColumn(paddings[2]);
+ if (IIVDEntry.RThroughput) {
+ double RT = 1.0 / *IIVDEntry.RThroughput;
+ FOS << " " << format("%.2f", RT);
+ } else {
+ FOS << " -";
+ }
----------------
adibiagio wrote:
Maybe add a flag for controlling whether to print it as reciprocal throughput (default) or normal throughput.
https://github.com/llvm/llvm-project/pull/130574
More information about the llvm-commits
mailing list