[llvm] [ctx_prof] test tool: generate ctxprof bistream from json (PR #100379)

Mircea Trofin via llvm-commits llvm-commits at lists.llvm.org
Thu Jul 25 12:37:06 PDT 2024


https://github.com/mtrofin updated https://github.com/llvm/llvm-project/pull/100379

>From 5f7ff322b46919bc9835fd584a7c234272600348 Mon Sep 17 00:00:00 2001
From: Mircea Trofin <mtrofin at google.com>
Date: Thu, 18 Jul 2024 13:31:52 -0700
Subject: [PATCH 1/3] [ctx_prof] test tool: generate ctxprof bistream from json

This is a tool to simplify testing. It generates a valid contextual
profile file from a json representation.

The tool is authored to allow for future evolution, e.g. if we want to
support profile merging or other tasks, not necessarily scoped to testing.
---
 llvm/test/CMakeLists.txt                      |   1 +
 llvm/test/lit.cfg.py                          |   1 +
 .../tools/llvm-ctxprof-util/Inputs/bad.json   |   1 +
 .../tools/llvm-ctxprof-util/Inputs/empty.json |   1 +
 .../Inputs/invalid-bad-subctx.json            |   8 +
 .../Inputs/invalid-no-counters.json           |   5 +
 .../Inputs/invalid-no-ctx.json                |   1 +
 .../Inputs/invalid-no-vector.json             |   1 +
 .../tools/llvm-ctxprof-util/Inputs/valid.json |  29 ++++
 .../llvm-ctxprof-util-negative.test           |  20 +++
 .../llvm-ctxprof-util/llvm-ctxprof-util.test  |  39 +++++
 llvm/tools/CMakeLists.txt                     |   1 +
 llvm/tools/llvm-ctxprof-util/CMakeLists.txt   |  14 ++
 .../llvm-ctxprof-util/llvm-ctxprof-util.cpp   | 148 ++++++++++++++++++
 14 files changed, 270 insertions(+)
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/bad.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/empty.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-bad-subctx.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-counters.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-ctx.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-vector.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
 create mode 100644 llvm/tools/llvm-ctxprof-util/CMakeLists.txt
 create mode 100644 llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp

diff --git a/llvm/test/CMakeLists.txt b/llvm/test/CMakeLists.txt
index 6b7f2b58e603e..8abc153336251 100644
--- a/llvm/test/CMakeLists.txt
+++ b/llvm/test/CMakeLists.txt
@@ -76,6 +76,7 @@ set(LLVM_TEST_DEPENDS
           llvm-cfi-verify
           llvm-config
           llvm-cov
+          llvm-ctxprof-util
           llvm-cvtres
           llvm-cxxdump
           llvm-cxxfilt
diff --git a/llvm/test/lit.cfg.py b/llvm/test/lit.cfg.py
index fe1262893212f..e5e3dc7e1b4bd 100644
--- a/llvm/test/lit.cfg.py
+++ b/llvm/test/lit.cfg.py
@@ -182,6 +182,7 @@ def get_asan_rtlib():
         "llvm-bitcode-strip",
         "llvm-config",
         "llvm-cov",
+        "llvm-ctxprof-util",
         "llvm-cxxdump",
         "llvm-cvtres",
         "llvm-debuginfod-find",
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/bad.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/bad.json
new file mode 100644
index 0000000000000..35c169002386e
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/bad.json
@@ -0,0 +1 @@
+[{]
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/empty.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/empty.json
new file mode 100644
index 0000000000000..fe51488c7066f
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/empty.json
@@ -0,0 +1 @@
+[]
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-bad-subctx.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-bad-subctx.json
new file mode 100644
index 0000000000000..b47e0ee1a04ba
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-bad-subctx.json
@@ -0,0 +1,8 @@
+[{
+  "Guid": 123,
+  "Counters": [1, 2],
+  "Callsites":
+  [
+    {"Guid": 1}
+  ]
+}]
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-counters.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-counters.json
new file mode 100644
index 0000000000000..95cdd45a5a0f7
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-counters.json
@@ -0,0 +1,5 @@
+[
+  {
+    "Guid": 1231
+  }
+]
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-ctx.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-ctx.json
new file mode 100644
index 0000000000000..93d51406d63fb
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-ctx.json
@@ -0,0 +1 @@
+[{}]
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-vector.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-vector.json
new file mode 100644
index 0000000000000..0967ef424bce6
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/invalid-no-vector.json
@@ -0,0 +1 @@
+{}
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
new file mode 100644
index 0000000000000..96e9f3767ef08
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
@@ -0,0 +1,29 @@
+[
+  {
+    "Guid": 1000,
+    "Counters": [1, 2, 3],
+    "Callsites": [
+      [],
+      [
+        {
+          "Guid": 2000,
+          "Counters": [4, 5]
+        },
+        {
+          "Guid": 18446744073709551613,
+          "Counters": [6, 7, 8]
+        }
+    ], 
+    [
+      {
+        "Guid": 3000,
+        "Counters": [40, 50]
+      }
+    ]
+  ]
+  },
+  {
+    "Guid": 18446744073709551612,
+    "Counters": [5, 9, 10]
+  }
+]
diff --git a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
new file mode 100644
index 0000000000000..97dd19be24f64
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
@@ -0,0 +1,20 @@
+; RUN: not llvm-ctxprof-util nofile.json 2>&1 | FileCheck %s --check-prefix=NO_CMD
+; RUN: not llvm-ctxprof-util fromJSON nofile.json 2>&1 | FileCheck %s --check-prefix=NO_FLAG
+; RUN: not llvm-ctxprof-util fromJSON --input nofile.json 2>&1 | FileCheck %s --check-prefix=NO_FILE
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/bad.json 2>&1 | FileCheck %s --check-prefix=BAD_JSON
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/invalid-no-vector.json 2>&1 | FileCheck %s --check-prefix=NO_VECTOR
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/invalid-no-ctx.json 2>&1 | FileCheck %s --check-prefix=NO_CTX
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/invalid-no-counters.json 2>&1 | FileCheck %s --check-prefix=NO_COUNTERS
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/invalid-bad-subctx.json 2>&1 | FileCheck %s --check-prefix=BAD_SUBCTX
+; RUN: rm -rf %t
+; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/valid.json --output %t/output.bitstream 2>&1 | FileCheck %s --check-prefix=NO_DIR
+
+; NO_CMD: Unknown subcommand 'nofile.json'
+; NO_FLAG: Unknown command line argument 'nofile.json'. 
+; NO_FILE: 'nofile.json': No such file or directory
+; BAD_JSON: Expected object key
+; NO_VECTOR: expected array
+; NO_CTX: missing value at (root)[0].Guid
+; NO_COUNTERS: missing value at (root)[0].Counters
+; BAD_SUBCTX: expected array at (root)[0].Callsites[0]
+; NO_DIR: failed to open output
diff --git a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
new file mode 100644
index 0000000000000..00b44a5de14d2
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
@@ -0,0 +1,39 @@
+; RUN: mkdir -p %t
+; RUN: llvm-ctxprof-util fromJSON --input %S/Inputs/empty.json -output %t/empty.bitstream
+; RUN: llvm-bcanalyzer --dump %t/empty.bitstream | FileCheck %s --check-prefix=EMPTY
+
+; RUN: llvm-ctxprof-util fromJSON --input %S/Inputs/valid.json -output %t/valid.bitstream
+; RUN: llvm-bcanalyzer --dump %t/valid.bitstream | FileCheck %s --check-prefix=VALID
+
+; EMPTY: <BLOCKINFO_BLOCK/>
+; EMPTY-NEXT: <Metadata NumWords=1 BlockCodeSize=2>
+; EMPTY-NEXT:   <Version op0=1/>
+; EMPTY-NEXT: </Metadata>
+
+; Note that uin64_t are printed as signed values by llvm-bcanalyzer
+
+; VALID: <BLOCKINFO_BLOCK/>
+; VALID: <Context
+; VALID-NEXT: <GUID op0=1000/>
+; VALID-NEXT: <Counters op0=1 op1=2 op2=3/>
+; VALID-NEXT: <Context
+
+; We have no callee/context at index 0, 2 callsites for index 1, and one for
+; index 2
+
+; VALID-NEXT:   <GUID op0=-3/>
+; VALID-NEXT:   <CalleeIndex op0=1/>
+; VALID-NEXT:   <Counters op0=6 op1=7 op2=8/>
+; VALID:      </Context>
+; VALID-NEXT: <Context
+; VALID-NEXT:   <GUID op0=2000/>
+; VALID-NEXT:   <CalleeIndex op0=1/>
+; VALID:      </Context>
+; VALID-NEXT: <Context
+; VALID-NEXT: <GUID op0=3000/>
+; VALID-NEXT:   <CalleeIndex op0=2/>
+; VALID-NEXT:   <Counters op0=40 op1=50/>
+; VALID-NEXT: </Context>
+; VALID-NEXT: </Context>
+; VALID-NEXT: <Context
+; VALID-NEXT: <GUID op0=-4/>
diff --git a/llvm/tools/CMakeLists.txt b/llvm/tools/CMakeLists.txt
index db66dad5dc0db..b9c5a79849ec8 100644
--- a/llvm/tools/CMakeLists.txt
+++ b/llvm/tools/CMakeLists.txt
@@ -32,6 +32,7 @@ add_llvm_tool_subdirectory(lto)
 add_llvm_tool_subdirectory(gold)
 add_llvm_tool_subdirectory(llvm-ar)
 add_llvm_tool_subdirectory(llvm-config)
+add_llvm_tool_subdirectory(llvm-ctxprof-util)
 add_llvm_tool_subdirectory(llvm-lto)
 add_llvm_tool_subdirectory(llvm-profdata)
 
diff --git a/llvm/tools/llvm-ctxprof-util/CMakeLists.txt b/llvm/tools/llvm-ctxprof-util/CMakeLists.txt
new file mode 100644
index 0000000000000..abf8e1aa0651f
--- /dev/null
+++ b/llvm/tools/llvm-ctxprof-util/CMakeLists.txt
@@ -0,0 +1,14 @@
+set(LLVM_LINK_COMPONENTS
+  Core
+  Object
+  ProfileData
+  Support
+  )
+
+add_llvm_tool(llvm-ctxprof-util
+llvm-ctxprof-util.cpp
+
+  DEPENDS
+  intrinsics_gen
+  GENERATE_DRIVER
+  )
diff --git a/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp b/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp
new file mode 100644
index 0000000000000..eb2567f18346f
--- /dev/null
+++ b/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp
@@ -0,0 +1,148 @@
+//===--- PGOCtxProfJSONReader.h - JSON format  ------------------*- C++ -*-===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+///
+/// \file
+///
+/// JSON format for the contextual profile for testing.
+///
+//===----------------------------------------------------------------------===//
+
+#include "llvm/ADT/STLExtras.h"
+#include "llvm/IR/GlobalValue.h"
+#include "llvm/ProfileData/CtxInstrContextNode.h"
+#include "llvm/ProfileData/PGOCtxProfWriter.h"
+#include "llvm/Support/CommandLine.h"
+#include "llvm/Support/Error.h"
+#include "llvm/Support/JSON.h"
+#include "llvm/Support/LLVMDriver.h"
+#include "llvm/Support/MemoryBuffer.h"
+#include "llvm/Support/raw_ostream.h"
+
+using namespace llvm;
+
+static cl::SubCommand FromJSON("fromJSON", "Convert from json");
+
+static cl::opt<std::string> InputFilename(
+    "input", cl::value_desc("input"), cl::init("-"),
+    cl::desc(
+        "Input file. The format is an array of contexts.\n"
+        "Each context is a dictionary with the following keys:\n"
+        "'Guid', mandatory. The value is a 64-bit integer.\n"
+        "'Counters', mandatory. An array of 32-bit ints. These are the "
+        "counter values.\n"
+        "'Contexts', optional. An array containing arrays of contexts. The "
+        "context array at a position 'i' is the set of callees at that "
+        "callsite index. Use an empty array to indicate no callees."),
+    cl::sub(FromJSON));
+
+static cl::opt<std::string> OutputFilename("output", cl::value_desc("output"),
+                                           cl::init("-"),
+                                           cl::desc("Output file"),
+                                           cl::sub(FromJSON));
+
+namespace {
+// A structural representation of the JSON input.
+struct DeserializableCtx {
+  GlobalValue::GUID Guid = 0;
+  std::vector<uint64_t> Counters;
+  std::vector<std::vector<DeserializableCtx>> Callsites;
+};
+
+ctx_profile::ContextNode *
+createNode(std::vector<std::unique_ptr<char[]>> &Nodes,
+           const std::vector<DeserializableCtx> &DCList);
+
+// Convert a DeserializableCtx into a ContextNode, potentially linking it to
+// its sibling (e.g. callee at same callsite) "Next".
+ctx_profile::ContextNode *
+createNode(std::vector<std::unique_ptr<char[]>> &Nodes,
+           const DeserializableCtx &DC,
+           ctx_profile::ContextNode *Next = nullptr) {
+  auto AllocSize = ctx_profile::ContextNode::getAllocSize(DC.Counters.size(),
+                                                          DC.Callsites.size());
+  auto *Mem = Nodes.emplace_back(std::make_unique<char[]>(AllocSize)).get();
+  std::memset(Mem, 0, AllocSize);
+  auto *Ret = new (Mem) ctx_profile::ContextNode(DC.Guid, DC.Counters.size(),
+                                                 DC.Callsites.size(), Next);
+  std::memcpy(Ret->counters(), DC.Counters.data(),
+              sizeof(uint64_t) * DC.Counters.size());
+  for (const auto &[I, DCList] : llvm::enumerate(DC.Callsites))
+    Ret->subContexts()[I] = createNode(Nodes, DCList);
+  return Ret;
+}
+
+// Convert a list of DeserializableCtx into a linked list of ContextNodes.
+ctx_profile::ContextNode *
+createNode(std::vector<std::unique_ptr<char[]>> &Nodes,
+           const std::vector<DeserializableCtx> &DCList) {
+  ctx_profile::ContextNode *List = nullptr;
+  for (const auto &DC : DCList)
+    List = createNode(Nodes, DC, List);
+  return List;
+}
+} // namespace
+
+namespace llvm {
+namespace json {
+// Hook into the JSON deserialization.
+bool fromJSON(const Value &E, DeserializableCtx &R, Path P) {
+  json::ObjectMapper Mapper(E, P);
+  return Mapper && Mapper.map("Guid", R.Guid) &&
+         Mapper.map("Counters", R.Counters) &&
+         Mapper.mapOptional("Callsites", R.Callsites);
+}
+} // namespace json
+} // namespace llvm
+
+// Save the bitstream profile from the JSON representation.
+Error convertFromJSON() {
+  auto BufOrError = MemoryBuffer::getFileOrSTDIN(InputFilename);
+  if (!BufOrError)
+    return createFileError(InputFilename, BufOrError.getError());
+  auto P = json::parse(BufOrError.get()->getBuffer());
+  if (!P)
+    return P.takeError();
+
+  std::vector<DeserializableCtx> DCList;
+  json::Path::Root R("");
+  if (!fromJSON(*P, DCList, R))
+    return R.getError();
+  // Nodes provides memory backing for the ContextualNodes.
+  std::vector<std::unique_ptr<char[]>> Nodes;
+  std::error_code EC;
+  raw_fd_stream Out(OutputFilename, EC);
+  if (EC)
+    return createStringError(EC, "failed to open output");
+  PGOCtxProfileWriter Writer(Out);
+  for (const auto &DC : DCList) {
+    auto *TopList = createNode(Nodes, DC);
+    if (!TopList)
+      return createStringError(
+          "Unexpected error converting internal structure to ctx profile");
+    Writer.write(*TopList);
+  }
+  if (EC)
+    return createStringError(EC, "failed to write output");
+  return Error::success();
+}
+
+int llvm_ctxprof_util_main(int argc, char **argv, const llvm::ToolContext &) {
+  cl::ParseCommandLineOptions(argc, argv, "LLVM Contextual Profile Utils\n");
+  ExitOnError ExitOnErr("llvm-ctxprof-util: ");
+  if (FromJSON) {
+    if (auto E = convertFromJSON()) {
+      handleAllErrors(std::move(E), [&](const ErrorInfoBase &E) {
+        E.log(errs());
+        errs() << "\n";
+      });
+      return 1;
+    }
+    return 0;
+  }
+  return 1;
+}

>From 6400f05dd77cfde98cfa3d0962fa30ebf107944b Mon Sep 17 00:00:00 2001
From: Mircea Trofin <mtrofin at google.com>
Date: Thu, 25 Jul 2024 04:53:00 -0700
Subject: [PATCH 2/3] full output

---
 .../llvm-ctxprof-util/Inputs/valid.expected   | 27 +++++++++++++
 .../llvm-ctxprof-util-negative.test           |  2 +
 .../llvm-ctxprof-util/llvm-ctxprof-util.test  | 38 +++++--------------
 3 files changed, 38 insertions(+), 29 deletions(-)
 create mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected

diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected
new file mode 100644
index 0000000000000..00e5206394177
--- /dev/null
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected
@@ -0,0 +1,27 @@
+<BLOCKINFO_BLOCK/>
+<Metadata NumWords=30 BlockCodeSize=2>
+  <Version op0=1/>
+  <Context NumWords=20 BlockCodeSize=2>
+    <GUID op0=1000/>
+    <Counters op0=1 op1=2 op2=3/>
+    <Context NumWords=5 BlockCodeSize=2>
+      <GUID op0=-3/>
+      <CalleeIndex op0=1/>
+      <Counters op0=6 op1=7 op2=8/>
+    </Context>
+    <Context NumWords=3 BlockCodeSize=2>
+      <GUID op0=2000/>
+      <CalleeIndex op0=1/>
+      <Counters op0=4 op1=5/>
+    </Context>
+    <Context NumWords=3 BlockCodeSize=2>
+      <GUID op0=3000/>
+      <CalleeIndex op0=2/>
+      <Counters op0=40 op1=50/>
+    </Context>
+  </Context>
+  <Context NumWords=4 BlockCodeSize=2>
+    <GUID op0=-4/>
+    <Counters op0=5 op1=9 op2=10/>
+  </Context>
+</Metadata>
diff --git a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
index 97dd19be24f64..97598c11979fa 100644
--- a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
+++ b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util-negative.test
@@ -1,4 +1,5 @@
 ; RUN: not llvm-ctxprof-util nofile.json 2>&1 | FileCheck %s --check-prefix=NO_CMD
+; RUN: not llvm-ctxprof-util invalidCmd --input nofile.json 2>&1 | FileCheck %s --check-prefix=INVALID_CMD
 ; RUN: not llvm-ctxprof-util fromJSON nofile.json 2>&1 | FileCheck %s --check-prefix=NO_FLAG
 ; RUN: not llvm-ctxprof-util fromJSON --input nofile.json 2>&1 | FileCheck %s --check-prefix=NO_FILE
 ; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/bad.json 2>&1 | FileCheck %s --check-prefix=BAD_JSON
@@ -10,6 +11,7 @@
 ; RUN: not llvm-ctxprof-util fromJSON --input %S/Inputs/valid.json --output %t/output.bitstream 2>&1 | FileCheck %s --check-prefix=NO_DIR
 
 ; NO_CMD: Unknown subcommand 'nofile.json'
+; INVALID_CMD: Unknown subcommand 'invalidCmd'
 ; NO_FLAG: Unknown command line argument 'nofile.json'. 
 ; NO_FILE: 'nofile.json': No such file or directory
 ; BAD_JSON: Expected object key
diff --git a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
index 00b44a5de14d2..143c2e24e7e71 100644
--- a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
+++ b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
@@ -3,37 +3,17 @@
 ; RUN: llvm-bcanalyzer --dump %t/empty.bitstream | FileCheck %s --check-prefix=EMPTY
 
 ; RUN: llvm-ctxprof-util fromJSON --input %S/Inputs/valid.json -output %t/valid.bitstream
-; RUN: llvm-bcanalyzer --dump %t/valid.bitstream | FileCheck %s --check-prefix=VALID
+
+; For the valid case, check against a reference output.
+; Note that uint64_t are printed as signed values by llvm-bcanalyzer:
+;  * 18446744073709551613 in json is -3 in the output
+;  * 18446744073709551612 in json is -4 in the output
+; Also we have no callee/context at index 0, 2 callsites for index 1, and one for
+; index 2.
+; RUN: llvm-bcanalyzer --dump %t/valid.bitstream | head -27 > %t.valid.in
+; RUN: diff %S/Inputs/valid.expected %t.valid.in
 
 ; EMPTY: <BLOCKINFO_BLOCK/>
 ; EMPTY-NEXT: <Metadata NumWords=1 BlockCodeSize=2>
 ; EMPTY-NEXT:   <Version op0=1/>
 ; EMPTY-NEXT: </Metadata>
-
-; Note that uin64_t are printed as signed values by llvm-bcanalyzer
-
-; VALID: <BLOCKINFO_BLOCK/>
-; VALID: <Context
-; VALID-NEXT: <GUID op0=1000/>
-; VALID-NEXT: <Counters op0=1 op1=2 op2=3/>
-; VALID-NEXT: <Context
-
-; We have no callee/context at index 0, 2 callsites for index 1, and one for
-; index 2
-
-; VALID-NEXT:   <GUID op0=-3/>
-; VALID-NEXT:   <CalleeIndex op0=1/>
-; VALID-NEXT:   <Counters op0=6 op1=7 op2=8/>
-; VALID:      </Context>
-; VALID-NEXT: <Context
-; VALID-NEXT:   <GUID op0=2000/>
-; VALID-NEXT:   <CalleeIndex op0=1/>
-; VALID:      </Context>
-; VALID-NEXT: <Context
-; VALID-NEXT: <GUID op0=3000/>
-; VALID-NEXT:   <CalleeIndex op0=2/>
-; VALID-NEXT:   <Counters op0=40 op1=50/>
-; VALID-NEXT: </Context>
-; VALID-NEXT: </Context>
-; VALID-NEXT: <Context
-; VALID-NEXT: <GUID op0=-4/>

>From b26498e375a0edcda30d99fd867c1547d49a093c Mon Sep 17 00:00:00 2001
From: Mircea Trofin <mtrofin at google.com>
Date: Thu, 25 Jul 2024 12:36:35 -0700
Subject: [PATCH 3/3] inlined valid test, reformat, llvm_unreachable

---
 .../llvm-ctxprof-util/Inputs/valid.expected   | 27 ------------
 .../tools/llvm-ctxprof-util/Inputs/valid.json | 42 +++++++++++++------
 .../llvm-ctxprof-util/llvm-ctxprof-util.test  | 31 +++++++++++++-
 .../llvm-ctxprof-util/llvm-ctxprof-util.cpp   |  4 +-
 4 files changed, 62 insertions(+), 42 deletions(-)
 delete mode 100644 llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected

diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected
deleted file mode 100644
index 00e5206394177..0000000000000
--- a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.expected
+++ /dev/null
@@ -1,27 +0,0 @@
-<BLOCKINFO_BLOCK/>
-<Metadata NumWords=30 BlockCodeSize=2>
-  <Version op0=1/>
-  <Context NumWords=20 BlockCodeSize=2>
-    <GUID op0=1000/>
-    <Counters op0=1 op1=2 op2=3/>
-    <Context NumWords=5 BlockCodeSize=2>
-      <GUID op0=-3/>
-      <CalleeIndex op0=1/>
-      <Counters op0=6 op1=7 op2=8/>
-    </Context>
-    <Context NumWords=3 BlockCodeSize=2>
-      <GUID op0=2000/>
-      <CalleeIndex op0=1/>
-      <Counters op0=4 op1=5/>
-    </Context>
-    <Context NumWords=3 BlockCodeSize=2>
-      <GUID op0=3000/>
-      <CalleeIndex op0=2/>
-      <Counters op0=40 op1=50/>
-    </Context>
-  </Context>
-  <Context NumWords=4 BlockCodeSize=2>
-    <GUID op0=-4/>
-    <Counters op0=5 op1=9 op2=10/>
-  </Context>
-</Metadata>
diff --git a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
index 96e9f3767ef08..15d6f52a1c191 100644
--- a/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
+++ b/llvm/test/tools/llvm-ctxprof-util/Inputs/valid.json
@@ -1,29 +1,47 @@
 [
   {
     "Guid": 1000,
-    "Counters": [1, 2, 3],
+    "Counters": [
+      1,
+      2,
+      3
+    ],
     "Callsites": [
       [],
       [
         {
           "Guid": 2000,
-          "Counters": [4, 5]
+          "Counters": [
+            4,
+            5
+          ]
         },
         {
           "Guid": 18446744073709551613,
-          "Counters": [6, 7, 8]
+          "Counters": [
+            6,
+            7,
+            8
+          ]
         }
-    ], 
-    [
-      {
-        "Guid": 3000,
-        "Counters": [40, 50]
-      }
+      ],
+      [
+        {
+          "Guid": 3000,
+          "Counters": [
+            40,
+            50
+          ]
+        }
+      ]
     ]
-  ]
   },
   {
     "Guid": 18446744073709551612,
-    "Counters": [5, 9, 10]
+    "Counters": [
+      5,
+      9,
+      10
+    ]
   }
-]
+]
\ No newline at end of file
diff --git a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
index 143c2e24e7e71..d430818d05442 100644
--- a/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
+++ b/llvm/test/tools/llvm-ctxprof-util/llvm-ctxprof-util.test
@@ -10,10 +10,37 @@
 ;  * 18446744073709551612 in json is -4 in the output
 ; Also we have no callee/context at index 0, 2 callsites for index 1, and one for
 ; index 2.
-; RUN: llvm-bcanalyzer --dump %t/valid.bitstream | head -27 > %t.valid.in
-; RUN: diff %S/Inputs/valid.expected %t.valid.in
+; RUN: llvm-bcanalyzer --dump %t/valid.bitstream | FileCheck %s --check-prefix=VALID
 
 ; EMPTY: <BLOCKINFO_BLOCK/>
 ; EMPTY-NEXT: <Metadata NumWords=1 BlockCodeSize=2>
 ; EMPTY-NEXT:   <Version op0=1/>
 ; EMPTY-NEXT: </Metadata>
+
+; VALID:      <BLOCKINFO_BLOCK/>
+; VALID-NEXT: <Metadata NumWords=30 BlockCodeSize=2>
+; VALID-NEXT:   <Version op0=1/>
+; VALID-NEXT:   <Context NumWords=20 BlockCodeSize=2>
+; VALID-NEXT:     <GUID op0=1000/>
+; VALID-NEXT:     <Counters op0=1 op1=2 op2=3/>
+; VALID-NEXT:     <Context NumWords=5 BlockCodeSize=2>
+; VALID-NEXT:       <GUID op0=-3/>
+; VALID-NEXT:       <CalleeIndex op0=1/>
+; VALID-NEXT:       <Counters op0=6 op1=7 op2=8/>
+; VALID-NEXT:     </Context>
+; VALID-NEXT:     <Context NumWords=3 BlockCodeSize=2>
+; VALID-NEXT:       <GUID op0=2000/>
+; VALID-NEXT:       <CalleeIndex op0=1/>
+; VALID-NEXT:       <Counters op0=4 op1=5/>
+; VALID-NEXT:     </Context>
+; VALID-NEXT:     <Context NumWords=3 BlockCodeSize=2>
+; VALID-NEXT:       <GUID op0=3000/>
+; VALID-NEXT:       <CalleeIndex op0=2/>
+; VALID-NEXT:       <Counters op0=40 op1=50/>
+; VALID-NEXT:     </Context>
+; VALID-NEXT:   </Context>
+; VALID-NEXT:   <Context NumWords=4 BlockCodeSize=2>
+; VALID-NEXT:     <GUID op0=-4/>
+; VALID-NEXT:     <Counters op0=5 op1=9 op2=10/>
+; VALID-NEXT:   </Context>
+; VALID-NEXT: </Metadata>
diff --git a/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp b/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp
index eb2567f18346f..ded8c8a6e4332 100644
--- a/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp
+++ b/llvm/tools/llvm-ctxprof-util/llvm-ctxprof-util.cpp
@@ -18,6 +18,7 @@
 #include "llvm/ProfileData/PGOCtxProfWriter.h"
 #include "llvm/Support/CommandLine.h"
 #include "llvm/Support/Error.h"
+#include "llvm/Support/ErrorHandling.h"
 #include "llvm/Support/JSON.h"
 #include "llvm/Support/LLVMDriver.h"
 #include "llvm/Support/MemoryBuffer.h"
@@ -144,5 +145,6 @@ int llvm_ctxprof_util_main(int argc, char **argv, const llvm::ToolContext &) {
     }
     return 0;
   }
-  return 1;
+  llvm_unreachable("Unknown subcommands should have been handled by the "
+                   "command line parser.");
 }



More information about the llvm-commits mailing list