[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