1*9880d681SAndroid Build Coastguard Worker //===- llvm-profdata.cpp - LLVM profile data tool -------------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker //
10*9880d681SAndroid Build Coastguard Worker // llvm-profdata merges .profdata files.
11*9880d681SAndroid Build Coastguard Worker //
12*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
13*9880d681SAndroid Build Coastguard Worker
14*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/SmallSet.h"
15*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/SmallVector.h"
16*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/StringRef.h"
17*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/LLVMContext.h"
18*9880d681SAndroid Build Coastguard Worker #include "llvm/ProfileData/InstrProfReader.h"
19*9880d681SAndroid Build Coastguard Worker #include "llvm/ProfileData/InstrProfWriter.h"
20*9880d681SAndroid Build Coastguard Worker #include "llvm/ProfileData/ProfileCommon.h"
21*9880d681SAndroid Build Coastguard Worker #include "llvm/ProfileData/SampleProfReader.h"
22*9880d681SAndroid Build Coastguard Worker #include "llvm/ProfileData/SampleProfWriter.h"
23*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/CommandLine.h"
24*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Errc.h"
25*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/FileSystem.h"
26*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Format.h"
27*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ManagedStatic.h"
28*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MemoryBuffer.h"
29*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Path.h"
30*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/PrettyStackTrace.h"
31*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/Signals.h"
32*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/raw_ostream.h"
33*9880d681SAndroid Build Coastguard Worker #include <algorithm>
34*9880d681SAndroid Build Coastguard Worker
35*9880d681SAndroid Build Coastguard Worker using namespace llvm;
36*9880d681SAndroid Build Coastguard Worker
37*9880d681SAndroid Build Coastguard Worker enum ProfileFormat { PF_None = 0, PF_Text, PF_Binary, PF_GCC };
38*9880d681SAndroid Build Coastguard Worker
exitWithError(const Twine & Message,StringRef Whence="",StringRef Hint="")39*9880d681SAndroid Build Coastguard Worker static void exitWithError(const Twine &Message, StringRef Whence = "",
40*9880d681SAndroid Build Coastguard Worker StringRef Hint = "") {
41*9880d681SAndroid Build Coastguard Worker errs() << "error: ";
42*9880d681SAndroid Build Coastguard Worker if (!Whence.empty())
43*9880d681SAndroid Build Coastguard Worker errs() << Whence << ": ";
44*9880d681SAndroid Build Coastguard Worker errs() << Message << "\n";
45*9880d681SAndroid Build Coastguard Worker if (!Hint.empty())
46*9880d681SAndroid Build Coastguard Worker errs() << Hint << "\n";
47*9880d681SAndroid Build Coastguard Worker ::exit(1);
48*9880d681SAndroid Build Coastguard Worker }
49*9880d681SAndroid Build Coastguard Worker
exitWithError(Error E,StringRef Whence="")50*9880d681SAndroid Build Coastguard Worker static void exitWithError(Error E, StringRef Whence = "") {
51*9880d681SAndroid Build Coastguard Worker if (E.isA<InstrProfError>()) {
52*9880d681SAndroid Build Coastguard Worker handleAllErrors(std::move(E), [&](const InstrProfError &IPE) {
53*9880d681SAndroid Build Coastguard Worker instrprof_error instrError = IPE.get();
54*9880d681SAndroid Build Coastguard Worker StringRef Hint = "";
55*9880d681SAndroid Build Coastguard Worker if (instrError == instrprof_error::unrecognized_format) {
56*9880d681SAndroid Build Coastguard Worker // Hint for common error of forgetting -sample for sample profiles.
57*9880d681SAndroid Build Coastguard Worker Hint = "Perhaps you forgot to use the -sample option?";
58*9880d681SAndroid Build Coastguard Worker }
59*9880d681SAndroid Build Coastguard Worker exitWithError(IPE.message(), Whence, Hint);
60*9880d681SAndroid Build Coastguard Worker });
61*9880d681SAndroid Build Coastguard Worker }
62*9880d681SAndroid Build Coastguard Worker
63*9880d681SAndroid Build Coastguard Worker exitWithError(toString(std::move(E)), Whence);
64*9880d681SAndroid Build Coastguard Worker }
65*9880d681SAndroid Build Coastguard Worker
exitWithErrorCode(std::error_code EC,StringRef Whence="")66*9880d681SAndroid Build Coastguard Worker static void exitWithErrorCode(std::error_code EC, StringRef Whence = "") {
67*9880d681SAndroid Build Coastguard Worker exitWithError(EC.message(), Whence);
68*9880d681SAndroid Build Coastguard Worker }
69*9880d681SAndroid Build Coastguard Worker
70*9880d681SAndroid Build Coastguard Worker namespace {
71*9880d681SAndroid Build Coastguard Worker enum ProfileKinds { instr, sample };
72*9880d681SAndroid Build Coastguard Worker }
73*9880d681SAndroid Build Coastguard Worker
handleMergeWriterError(Error E,StringRef WhenceFile="",StringRef WhenceFunction="",bool ShowHint=true)74*9880d681SAndroid Build Coastguard Worker static void handleMergeWriterError(Error E, StringRef WhenceFile = "",
75*9880d681SAndroid Build Coastguard Worker StringRef WhenceFunction = "",
76*9880d681SAndroid Build Coastguard Worker bool ShowHint = true) {
77*9880d681SAndroid Build Coastguard Worker if (!WhenceFile.empty())
78*9880d681SAndroid Build Coastguard Worker errs() << WhenceFile << ": ";
79*9880d681SAndroid Build Coastguard Worker if (!WhenceFunction.empty())
80*9880d681SAndroid Build Coastguard Worker errs() << WhenceFunction << ": ";
81*9880d681SAndroid Build Coastguard Worker
82*9880d681SAndroid Build Coastguard Worker auto IPE = instrprof_error::success;
83*9880d681SAndroid Build Coastguard Worker E = handleErrors(std::move(E),
84*9880d681SAndroid Build Coastguard Worker [&IPE](std::unique_ptr<InstrProfError> E) -> Error {
85*9880d681SAndroid Build Coastguard Worker IPE = E->get();
86*9880d681SAndroid Build Coastguard Worker return Error(std::move(E));
87*9880d681SAndroid Build Coastguard Worker });
88*9880d681SAndroid Build Coastguard Worker errs() << toString(std::move(E)) << "\n";
89*9880d681SAndroid Build Coastguard Worker
90*9880d681SAndroid Build Coastguard Worker if (ShowHint) {
91*9880d681SAndroid Build Coastguard Worker StringRef Hint = "";
92*9880d681SAndroid Build Coastguard Worker if (IPE != instrprof_error::success) {
93*9880d681SAndroid Build Coastguard Worker switch (IPE) {
94*9880d681SAndroid Build Coastguard Worker case instrprof_error::hash_mismatch:
95*9880d681SAndroid Build Coastguard Worker case instrprof_error::count_mismatch:
96*9880d681SAndroid Build Coastguard Worker case instrprof_error::value_site_count_mismatch:
97*9880d681SAndroid Build Coastguard Worker Hint = "Make sure that all profile data to be merged is generated "
98*9880d681SAndroid Build Coastguard Worker "from the same binary.";
99*9880d681SAndroid Build Coastguard Worker break;
100*9880d681SAndroid Build Coastguard Worker default:
101*9880d681SAndroid Build Coastguard Worker break;
102*9880d681SAndroid Build Coastguard Worker }
103*9880d681SAndroid Build Coastguard Worker }
104*9880d681SAndroid Build Coastguard Worker
105*9880d681SAndroid Build Coastguard Worker if (!Hint.empty())
106*9880d681SAndroid Build Coastguard Worker errs() << Hint << "\n";
107*9880d681SAndroid Build Coastguard Worker }
108*9880d681SAndroid Build Coastguard Worker }
109*9880d681SAndroid Build Coastguard Worker
110*9880d681SAndroid Build Coastguard Worker struct WeightedFile {
111*9880d681SAndroid Build Coastguard Worker StringRef Filename;
112*9880d681SAndroid Build Coastguard Worker uint64_t Weight;
113*9880d681SAndroid Build Coastguard Worker
WeightedFileWeightedFile114*9880d681SAndroid Build Coastguard Worker WeightedFile() {}
115*9880d681SAndroid Build Coastguard Worker
WeightedFileWeightedFile116*9880d681SAndroid Build Coastguard Worker WeightedFile(StringRef F, uint64_t W) : Filename{F}, Weight{W} {}
117*9880d681SAndroid Build Coastguard Worker };
118*9880d681SAndroid Build Coastguard Worker typedef SmallVector<WeightedFile, 5> WeightedFileVector;
119*9880d681SAndroid Build Coastguard Worker
mergeInstrProfile(const WeightedFileVector & Inputs,StringRef OutputFilename,ProfileFormat OutputFormat,bool OutputSparse)120*9880d681SAndroid Build Coastguard Worker static void mergeInstrProfile(const WeightedFileVector &Inputs,
121*9880d681SAndroid Build Coastguard Worker StringRef OutputFilename,
122*9880d681SAndroid Build Coastguard Worker ProfileFormat OutputFormat, bool OutputSparse) {
123*9880d681SAndroid Build Coastguard Worker if (OutputFilename.compare("-") == 0)
124*9880d681SAndroid Build Coastguard Worker exitWithError("Cannot write indexed profdata format to stdout.");
125*9880d681SAndroid Build Coastguard Worker
126*9880d681SAndroid Build Coastguard Worker if (OutputFormat != PF_Binary && OutputFormat != PF_Text)
127*9880d681SAndroid Build Coastguard Worker exitWithError("Unknown format is specified.");
128*9880d681SAndroid Build Coastguard Worker
129*9880d681SAndroid Build Coastguard Worker std::error_code EC;
130*9880d681SAndroid Build Coastguard Worker raw_fd_ostream Output(OutputFilename.data(), EC, sys::fs::F_None);
131*9880d681SAndroid Build Coastguard Worker if (EC)
132*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, OutputFilename);
133*9880d681SAndroid Build Coastguard Worker
134*9880d681SAndroid Build Coastguard Worker InstrProfWriter Writer(OutputSparse);
135*9880d681SAndroid Build Coastguard Worker SmallSet<instrprof_error, 4> WriterErrorCodes;
136*9880d681SAndroid Build Coastguard Worker for (const auto &Input : Inputs) {
137*9880d681SAndroid Build Coastguard Worker auto ReaderOrErr = InstrProfReader::create(Input.Filename);
138*9880d681SAndroid Build Coastguard Worker if (Error E = ReaderOrErr.takeError())
139*9880d681SAndroid Build Coastguard Worker exitWithError(std::move(E), Input.Filename);
140*9880d681SAndroid Build Coastguard Worker
141*9880d681SAndroid Build Coastguard Worker auto Reader = std::move(ReaderOrErr.get());
142*9880d681SAndroid Build Coastguard Worker bool IsIRProfile = Reader->isIRLevelProfile();
143*9880d681SAndroid Build Coastguard Worker if (Writer.setIsIRLevelProfile(IsIRProfile))
144*9880d681SAndroid Build Coastguard Worker exitWithError("Merge IR generated profile with Clang generated profile.");
145*9880d681SAndroid Build Coastguard Worker
146*9880d681SAndroid Build Coastguard Worker for (auto &I : *Reader) {
147*9880d681SAndroid Build Coastguard Worker if (Error E = Writer.addRecord(std::move(I), Input.Weight)) {
148*9880d681SAndroid Build Coastguard Worker // Only show hint the first time an error occurs.
149*9880d681SAndroid Build Coastguard Worker instrprof_error IPE = InstrProfError::take(std::move(E));
150*9880d681SAndroid Build Coastguard Worker bool firstTime = WriterErrorCodes.insert(IPE).second;
151*9880d681SAndroid Build Coastguard Worker handleMergeWriterError(make_error<InstrProfError>(IPE), Input.Filename,
152*9880d681SAndroid Build Coastguard Worker I.Name, firstTime);
153*9880d681SAndroid Build Coastguard Worker }
154*9880d681SAndroid Build Coastguard Worker }
155*9880d681SAndroid Build Coastguard Worker if (Reader->hasError())
156*9880d681SAndroid Build Coastguard Worker exitWithError(Reader->getError(), Input.Filename);
157*9880d681SAndroid Build Coastguard Worker }
158*9880d681SAndroid Build Coastguard Worker if (OutputFormat == PF_Text)
159*9880d681SAndroid Build Coastguard Worker Writer.writeText(Output);
160*9880d681SAndroid Build Coastguard Worker else
161*9880d681SAndroid Build Coastguard Worker Writer.write(Output);
162*9880d681SAndroid Build Coastguard Worker }
163*9880d681SAndroid Build Coastguard Worker
164*9880d681SAndroid Build Coastguard Worker static sampleprof::SampleProfileFormat FormatMap[] = {
165*9880d681SAndroid Build Coastguard Worker sampleprof::SPF_None, sampleprof::SPF_Text, sampleprof::SPF_Binary,
166*9880d681SAndroid Build Coastguard Worker sampleprof::SPF_GCC};
167*9880d681SAndroid Build Coastguard Worker
mergeSampleProfile(const WeightedFileVector & Inputs,StringRef OutputFilename,ProfileFormat OutputFormat)168*9880d681SAndroid Build Coastguard Worker static void mergeSampleProfile(const WeightedFileVector &Inputs,
169*9880d681SAndroid Build Coastguard Worker StringRef OutputFilename,
170*9880d681SAndroid Build Coastguard Worker ProfileFormat OutputFormat) {
171*9880d681SAndroid Build Coastguard Worker using namespace sampleprof;
172*9880d681SAndroid Build Coastguard Worker auto WriterOrErr =
173*9880d681SAndroid Build Coastguard Worker SampleProfileWriter::create(OutputFilename, FormatMap[OutputFormat]);
174*9880d681SAndroid Build Coastguard Worker if (std::error_code EC = WriterOrErr.getError())
175*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, OutputFilename);
176*9880d681SAndroid Build Coastguard Worker
177*9880d681SAndroid Build Coastguard Worker auto Writer = std::move(WriterOrErr.get());
178*9880d681SAndroid Build Coastguard Worker StringMap<FunctionSamples> ProfileMap;
179*9880d681SAndroid Build Coastguard Worker SmallVector<std::unique_ptr<sampleprof::SampleProfileReader>, 5> Readers;
180*9880d681SAndroid Build Coastguard Worker LLVMContext Context;
181*9880d681SAndroid Build Coastguard Worker for (const auto &Input : Inputs) {
182*9880d681SAndroid Build Coastguard Worker auto ReaderOrErr = SampleProfileReader::create(Input.Filename, Context);
183*9880d681SAndroid Build Coastguard Worker if (std::error_code EC = ReaderOrErr.getError())
184*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, Input.Filename);
185*9880d681SAndroid Build Coastguard Worker
186*9880d681SAndroid Build Coastguard Worker // We need to keep the readers around until after all the files are
187*9880d681SAndroid Build Coastguard Worker // read so that we do not lose the function names stored in each
188*9880d681SAndroid Build Coastguard Worker // reader's memory. The function names are needed to write out the
189*9880d681SAndroid Build Coastguard Worker // merged profile map.
190*9880d681SAndroid Build Coastguard Worker Readers.push_back(std::move(ReaderOrErr.get()));
191*9880d681SAndroid Build Coastguard Worker const auto Reader = Readers.back().get();
192*9880d681SAndroid Build Coastguard Worker if (std::error_code EC = Reader->read())
193*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, Input.Filename);
194*9880d681SAndroid Build Coastguard Worker
195*9880d681SAndroid Build Coastguard Worker StringMap<FunctionSamples> &Profiles = Reader->getProfiles();
196*9880d681SAndroid Build Coastguard Worker for (StringMap<FunctionSamples>::iterator I = Profiles.begin(),
197*9880d681SAndroid Build Coastguard Worker E = Profiles.end();
198*9880d681SAndroid Build Coastguard Worker I != E; ++I) {
199*9880d681SAndroid Build Coastguard Worker StringRef FName = I->first();
200*9880d681SAndroid Build Coastguard Worker FunctionSamples &Samples = I->second;
201*9880d681SAndroid Build Coastguard Worker sampleprof_error Result = ProfileMap[FName].merge(Samples, Input.Weight);
202*9880d681SAndroid Build Coastguard Worker if (Result != sampleprof_error::success) {
203*9880d681SAndroid Build Coastguard Worker std::error_code EC = make_error_code(Result);
204*9880d681SAndroid Build Coastguard Worker handleMergeWriterError(errorCodeToError(EC), Input.Filename, FName);
205*9880d681SAndroid Build Coastguard Worker }
206*9880d681SAndroid Build Coastguard Worker }
207*9880d681SAndroid Build Coastguard Worker }
208*9880d681SAndroid Build Coastguard Worker Writer->write(ProfileMap);
209*9880d681SAndroid Build Coastguard Worker }
210*9880d681SAndroid Build Coastguard Worker
parseWeightedFile(const StringRef & WeightedFilename)211*9880d681SAndroid Build Coastguard Worker static WeightedFile parseWeightedFile(const StringRef &WeightedFilename) {
212*9880d681SAndroid Build Coastguard Worker StringRef WeightStr, FileName;
213*9880d681SAndroid Build Coastguard Worker std::tie(WeightStr, FileName) = WeightedFilename.split(',');
214*9880d681SAndroid Build Coastguard Worker
215*9880d681SAndroid Build Coastguard Worker uint64_t Weight;
216*9880d681SAndroid Build Coastguard Worker if (WeightStr.getAsInteger(10, Weight) || Weight < 1)
217*9880d681SAndroid Build Coastguard Worker exitWithError("Input weight must be a positive integer.");
218*9880d681SAndroid Build Coastguard Worker
219*9880d681SAndroid Build Coastguard Worker if (!sys::fs::exists(FileName))
220*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(make_error_code(errc::no_such_file_or_directory),
221*9880d681SAndroid Build Coastguard Worker FileName);
222*9880d681SAndroid Build Coastguard Worker
223*9880d681SAndroid Build Coastguard Worker return WeightedFile(FileName, Weight);
224*9880d681SAndroid Build Coastguard Worker }
225*9880d681SAndroid Build Coastguard Worker
226*9880d681SAndroid Build Coastguard Worker static std::unique_ptr<MemoryBuffer>
getInputFilenamesFileBuf(const StringRef & InputFilenamesFile)227*9880d681SAndroid Build Coastguard Worker getInputFilenamesFileBuf(const StringRef &InputFilenamesFile) {
228*9880d681SAndroid Build Coastguard Worker if (InputFilenamesFile == "")
229*9880d681SAndroid Build Coastguard Worker return {};
230*9880d681SAndroid Build Coastguard Worker
231*9880d681SAndroid Build Coastguard Worker auto BufOrError = MemoryBuffer::getFileOrSTDIN(InputFilenamesFile);
232*9880d681SAndroid Build Coastguard Worker if (!BufOrError)
233*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(BufOrError.getError(), InputFilenamesFile);
234*9880d681SAndroid Build Coastguard Worker
235*9880d681SAndroid Build Coastguard Worker return std::move(*BufOrError);
236*9880d681SAndroid Build Coastguard Worker }
237*9880d681SAndroid Build Coastguard Worker
parseInputFilenamesFile(MemoryBuffer * Buffer,WeightedFileVector & WFV)238*9880d681SAndroid Build Coastguard Worker static void parseInputFilenamesFile(MemoryBuffer *Buffer,
239*9880d681SAndroid Build Coastguard Worker WeightedFileVector &WFV) {
240*9880d681SAndroid Build Coastguard Worker if (!Buffer)
241*9880d681SAndroid Build Coastguard Worker return;
242*9880d681SAndroid Build Coastguard Worker
243*9880d681SAndroid Build Coastguard Worker SmallVector<StringRef, 8> Entries;
244*9880d681SAndroid Build Coastguard Worker StringRef Data = Buffer->getBuffer();
245*9880d681SAndroid Build Coastguard Worker Data.split(Entries, '\n', /*MaxSplit=*/-1, /*KeepEmpty=*/false);
246*9880d681SAndroid Build Coastguard Worker for (const StringRef &FileWeightEntry : Entries) {
247*9880d681SAndroid Build Coastguard Worker StringRef SanitizedEntry = FileWeightEntry.trim(" \t\v\f\r");
248*9880d681SAndroid Build Coastguard Worker // Skip comments.
249*9880d681SAndroid Build Coastguard Worker if (SanitizedEntry.startswith("#"))
250*9880d681SAndroid Build Coastguard Worker continue;
251*9880d681SAndroid Build Coastguard Worker // If there's no comma, it's an unweighted profile.
252*9880d681SAndroid Build Coastguard Worker else if (SanitizedEntry.find(',') == StringRef::npos)
253*9880d681SAndroid Build Coastguard Worker WFV.emplace_back(SanitizedEntry, 1);
254*9880d681SAndroid Build Coastguard Worker else
255*9880d681SAndroid Build Coastguard Worker WFV.emplace_back(parseWeightedFile(SanitizedEntry));
256*9880d681SAndroid Build Coastguard Worker }
257*9880d681SAndroid Build Coastguard Worker }
258*9880d681SAndroid Build Coastguard Worker
merge_main(int argc,const char * argv[])259*9880d681SAndroid Build Coastguard Worker static int merge_main(int argc, const char *argv[]) {
260*9880d681SAndroid Build Coastguard Worker cl::list<std::string> InputFilenames(cl::Positional,
261*9880d681SAndroid Build Coastguard Worker cl::desc("<filename...>"));
262*9880d681SAndroid Build Coastguard Worker cl::list<std::string> WeightedInputFilenames("weighted-input",
263*9880d681SAndroid Build Coastguard Worker cl::desc("<weight>,<filename>"));
264*9880d681SAndroid Build Coastguard Worker cl::opt<std::string> InputFilenamesFile(
265*9880d681SAndroid Build Coastguard Worker "input-files", cl::init(""),
266*9880d681SAndroid Build Coastguard Worker cl::desc("Path to file containing newline-separated "
267*9880d681SAndroid Build Coastguard Worker "[<weight>,]<filename> entries"));
268*9880d681SAndroid Build Coastguard Worker cl::alias InputFilenamesFileA("f", cl::desc("Alias for --input-files"),
269*9880d681SAndroid Build Coastguard Worker cl::aliasopt(InputFilenamesFile));
270*9880d681SAndroid Build Coastguard Worker cl::opt<bool> DumpInputFileList(
271*9880d681SAndroid Build Coastguard Worker "dump-input-file-list", cl::init(false), cl::Hidden,
272*9880d681SAndroid Build Coastguard Worker cl::desc("Dump the list of input files and their weights, then exit"));
273*9880d681SAndroid Build Coastguard Worker cl::opt<std::string> OutputFilename("output", cl::value_desc("output"),
274*9880d681SAndroid Build Coastguard Worker cl::init("-"), cl::Required,
275*9880d681SAndroid Build Coastguard Worker cl::desc("Output file"));
276*9880d681SAndroid Build Coastguard Worker cl::alias OutputFilenameA("o", cl::desc("Alias for --output"),
277*9880d681SAndroid Build Coastguard Worker cl::aliasopt(OutputFilename));
278*9880d681SAndroid Build Coastguard Worker cl::opt<ProfileKinds> ProfileKind(
279*9880d681SAndroid Build Coastguard Worker cl::desc("Profile kind:"), cl::init(instr),
280*9880d681SAndroid Build Coastguard Worker cl::values(clEnumVal(instr, "Instrumentation profile (default)"),
281*9880d681SAndroid Build Coastguard Worker clEnumVal(sample, "Sample profile"), clEnumValEnd));
282*9880d681SAndroid Build Coastguard Worker cl::opt<ProfileFormat> OutputFormat(
283*9880d681SAndroid Build Coastguard Worker cl::desc("Format of output profile"), cl::init(PF_Binary),
284*9880d681SAndroid Build Coastguard Worker cl::values(clEnumValN(PF_Binary, "binary", "Binary encoding (default)"),
285*9880d681SAndroid Build Coastguard Worker clEnumValN(PF_Text, "text", "Text encoding"),
286*9880d681SAndroid Build Coastguard Worker clEnumValN(PF_GCC, "gcc",
287*9880d681SAndroid Build Coastguard Worker "GCC encoding (only meaningful for -sample)"),
288*9880d681SAndroid Build Coastguard Worker clEnumValEnd));
289*9880d681SAndroid Build Coastguard Worker cl::opt<bool> OutputSparse("sparse", cl::init(false),
290*9880d681SAndroid Build Coastguard Worker cl::desc("Generate a sparse profile (only meaningful for -instr)"));
291*9880d681SAndroid Build Coastguard Worker
292*9880d681SAndroid Build Coastguard Worker cl::ParseCommandLineOptions(argc, argv, "LLVM profile data merger\n");
293*9880d681SAndroid Build Coastguard Worker
294*9880d681SAndroid Build Coastguard Worker WeightedFileVector WeightedInputs;
295*9880d681SAndroid Build Coastguard Worker for (StringRef Filename : InputFilenames)
296*9880d681SAndroid Build Coastguard Worker WeightedInputs.emplace_back(Filename, 1);
297*9880d681SAndroid Build Coastguard Worker for (StringRef WeightedFilename : WeightedInputFilenames)
298*9880d681SAndroid Build Coastguard Worker WeightedInputs.emplace_back(parseWeightedFile(WeightedFilename));
299*9880d681SAndroid Build Coastguard Worker
300*9880d681SAndroid Build Coastguard Worker // Make sure that the file buffer stays alive for the duration of the
301*9880d681SAndroid Build Coastguard Worker // weighted input vector's lifetime.
302*9880d681SAndroid Build Coastguard Worker auto Buffer = getInputFilenamesFileBuf(InputFilenamesFile);
303*9880d681SAndroid Build Coastguard Worker parseInputFilenamesFile(Buffer.get(), WeightedInputs);
304*9880d681SAndroid Build Coastguard Worker
305*9880d681SAndroid Build Coastguard Worker if (WeightedInputs.empty())
306*9880d681SAndroid Build Coastguard Worker exitWithError("No input files specified. See " +
307*9880d681SAndroid Build Coastguard Worker sys::path::filename(argv[0]) + " -help");
308*9880d681SAndroid Build Coastguard Worker
309*9880d681SAndroid Build Coastguard Worker if (DumpInputFileList) {
310*9880d681SAndroid Build Coastguard Worker for (auto &WF : WeightedInputs)
311*9880d681SAndroid Build Coastguard Worker outs() << WF.Weight << "," << WF.Filename << "\n";
312*9880d681SAndroid Build Coastguard Worker return 0;
313*9880d681SAndroid Build Coastguard Worker }
314*9880d681SAndroid Build Coastguard Worker
315*9880d681SAndroid Build Coastguard Worker if (ProfileKind == instr)
316*9880d681SAndroid Build Coastguard Worker mergeInstrProfile(WeightedInputs, OutputFilename, OutputFormat,
317*9880d681SAndroid Build Coastguard Worker OutputSparse);
318*9880d681SAndroid Build Coastguard Worker else
319*9880d681SAndroid Build Coastguard Worker mergeSampleProfile(WeightedInputs, OutputFilename, OutputFormat);
320*9880d681SAndroid Build Coastguard Worker
321*9880d681SAndroid Build Coastguard Worker return 0;
322*9880d681SAndroid Build Coastguard Worker }
323*9880d681SAndroid Build Coastguard Worker
showInstrProfile(const std::string & Filename,bool ShowCounts,bool ShowIndirectCallTargets,bool ShowDetailedSummary,std::vector<uint32_t> DetailedSummaryCutoffs,bool ShowAllFunctions,const std::string & ShowFunction,bool TextFormat,raw_fd_ostream & OS)324*9880d681SAndroid Build Coastguard Worker static int showInstrProfile(const std::string &Filename, bool ShowCounts,
325*9880d681SAndroid Build Coastguard Worker bool ShowIndirectCallTargets,
326*9880d681SAndroid Build Coastguard Worker bool ShowDetailedSummary,
327*9880d681SAndroid Build Coastguard Worker std::vector<uint32_t> DetailedSummaryCutoffs,
328*9880d681SAndroid Build Coastguard Worker bool ShowAllFunctions,
329*9880d681SAndroid Build Coastguard Worker const std::string &ShowFunction, bool TextFormat,
330*9880d681SAndroid Build Coastguard Worker raw_fd_ostream &OS) {
331*9880d681SAndroid Build Coastguard Worker auto ReaderOrErr = InstrProfReader::create(Filename);
332*9880d681SAndroid Build Coastguard Worker std::vector<uint32_t> Cutoffs = std::move(DetailedSummaryCutoffs);
333*9880d681SAndroid Build Coastguard Worker if (ShowDetailedSummary && Cutoffs.empty()) {
334*9880d681SAndroid Build Coastguard Worker Cutoffs = {800000, 900000, 950000, 990000, 999000, 999900, 999990};
335*9880d681SAndroid Build Coastguard Worker }
336*9880d681SAndroid Build Coastguard Worker InstrProfSummaryBuilder Builder(std::move(Cutoffs));
337*9880d681SAndroid Build Coastguard Worker if (Error E = ReaderOrErr.takeError())
338*9880d681SAndroid Build Coastguard Worker exitWithError(std::move(E), Filename);
339*9880d681SAndroid Build Coastguard Worker
340*9880d681SAndroid Build Coastguard Worker auto Reader = std::move(ReaderOrErr.get());
341*9880d681SAndroid Build Coastguard Worker bool IsIRInstr = Reader->isIRLevelProfile();
342*9880d681SAndroid Build Coastguard Worker size_t ShownFunctions = 0;
343*9880d681SAndroid Build Coastguard Worker uint64_t TotalNumValueSites = 0;
344*9880d681SAndroid Build Coastguard Worker uint64_t TotalNumValueSitesWithValueProfile = 0;
345*9880d681SAndroid Build Coastguard Worker uint64_t TotalNumValues = 0;
346*9880d681SAndroid Build Coastguard Worker for (const auto &Func : *Reader) {
347*9880d681SAndroid Build Coastguard Worker bool Show =
348*9880d681SAndroid Build Coastguard Worker ShowAllFunctions || (!ShowFunction.empty() &&
349*9880d681SAndroid Build Coastguard Worker Func.Name.find(ShowFunction) != Func.Name.npos);
350*9880d681SAndroid Build Coastguard Worker
351*9880d681SAndroid Build Coastguard Worker bool doTextFormatDump = (Show && ShowCounts && TextFormat);
352*9880d681SAndroid Build Coastguard Worker
353*9880d681SAndroid Build Coastguard Worker if (doTextFormatDump) {
354*9880d681SAndroid Build Coastguard Worker InstrProfSymtab &Symtab = Reader->getSymtab();
355*9880d681SAndroid Build Coastguard Worker InstrProfWriter::writeRecordInText(Func, Symtab, OS);
356*9880d681SAndroid Build Coastguard Worker continue;
357*9880d681SAndroid Build Coastguard Worker }
358*9880d681SAndroid Build Coastguard Worker
359*9880d681SAndroid Build Coastguard Worker assert(Func.Counts.size() > 0 && "function missing entry counter");
360*9880d681SAndroid Build Coastguard Worker Builder.addRecord(Func);
361*9880d681SAndroid Build Coastguard Worker
362*9880d681SAndroid Build Coastguard Worker if (Show) {
363*9880d681SAndroid Build Coastguard Worker
364*9880d681SAndroid Build Coastguard Worker if (!ShownFunctions)
365*9880d681SAndroid Build Coastguard Worker OS << "Counters:\n";
366*9880d681SAndroid Build Coastguard Worker
367*9880d681SAndroid Build Coastguard Worker ++ShownFunctions;
368*9880d681SAndroid Build Coastguard Worker
369*9880d681SAndroid Build Coastguard Worker OS << " " << Func.Name << ":\n"
370*9880d681SAndroid Build Coastguard Worker << " Hash: " << format("0x%016" PRIx64, Func.Hash) << "\n"
371*9880d681SAndroid Build Coastguard Worker << " Counters: " << Func.Counts.size() << "\n";
372*9880d681SAndroid Build Coastguard Worker if (!IsIRInstr)
373*9880d681SAndroid Build Coastguard Worker OS << " Function count: " << Func.Counts[0] << "\n";
374*9880d681SAndroid Build Coastguard Worker
375*9880d681SAndroid Build Coastguard Worker if (ShowIndirectCallTargets)
376*9880d681SAndroid Build Coastguard Worker OS << " Indirect Call Site Count: "
377*9880d681SAndroid Build Coastguard Worker << Func.getNumValueSites(IPVK_IndirectCallTarget) << "\n";
378*9880d681SAndroid Build Coastguard Worker
379*9880d681SAndroid Build Coastguard Worker if (ShowCounts) {
380*9880d681SAndroid Build Coastguard Worker OS << " Block counts: [";
381*9880d681SAndroid Build Coastguard Worker size_t Start = (IsIRInstr ? 0 : 1);
382*9880d681SAndroid Build Coastguard Worker for (size_t I = Start, E = Func.Counts.size(); I < E; ++I) {
383*9880d681SAndroid Build Coastguard Worker OS << (I == Start ? "" : ", ") << Func.Counts[I];
384*9880d681SAndroid Build Coastguard Worker }
385*9880d681SAndroid Build Coastguard Worker OS << "]\n";
386*9880d681SAndroid Build Coastguard Worker }
387*9880d681SAndroid Build Coastguard Worker
388*9880d681SAndroid Build Coastguard Worker if (ShowIndirectCallTargets) {
389*9880d681SAndroid Build Coastguard Worker InstrProfSymtab &Symtab = Reader->getSymtab();
390*9880d681SAndroid Build Coastguard Worker uint32_t NS = Func.getNumValueSites(IPVK_IndirectCallTarget);
391*9880d681SAndroid Build Coastguard Worker OS << " Indirect Target Results: \n";
392*9880d681SAndroid Build Coastguard Worker TotalNumValueSites += NS;
393*9880d681SAndroid Build Coastguard Worker for (size_t I = 0; I < NS; ++I) {
394*9880d681SAndroid Build Coastguard Worker uint32_t NV = Func.getNumValueDataForSite(IPVK_IndirectCallTarget, I);
395*9880d681SAndroid Build Coastguard Worker std::unique_ptr<InstrProfValueData[]> VD =
396*9880d681SAndroid Build Coastguard Worker Func.getValueForSite(IPVK_IndirectCallTarget, I);
397*9880d681SAndroid Build Coastguard Worker TotalNumValues += NV;
398*9880d681SAndroid Build Coastguard Worker if (NV)
399*9880d681SAndroid Build Coastguard Worker TotalNumValueSitesWithValueProfile++;
400*9880d681SAndroid Build Coastguard Worker for (uint32_t V = 0; V < NV; V++) {
401*9880d681SAndroid Build Coastguard Worker OS << "\t[ " << I << ", ";
402*9880d681SAndroid Build Coastguard Worker OS << Symtab.getFuncName(VD[V].Value) << ", " << VD[V].Count
403*9880d681SAndroid Build Coastguard Worker << " ]\n";
404*9880d681SAndroid Build Coastguard Worker }
405*9880d681SAndroid Build Coastguard Worker }
406*9880d681SAndroid Build Coastguard Worker }
407*9880d681SAndroid Build Coastguard Worker }
408*9880d681SAndroid Build Coastguard Worker }
409*9880d681SAndroid Build Coastguard Worker if (Reader->hasError())
410*9880d681SAndroid Build Coastguard Worker exitWithError(Reader->getError(), Filename);
411*9880d681SAndroid Build Coastguard Worker
412*9880d681SAndroid Build Coastguard Worker if (ShowCounts && TextFormat)
413*9880d681SAndroid Build Coastguard Worker return 0;
414*9880d681SAndroid Build Coastguard Worker std::unique_ptr<ProfileSummary> PS(Builder.getSummary());
415*9880d681SAndroid Build Coastguard Worker if (ShowAllFunctions || !ShowFunction.empty())
416*9880d681SAndroid Build Coastguard Worker OS << "Functions shown: " << ShownFunctions << "\n";
417*9880d681SAndroid Build Coastguard Worker OS << "Total functions: " << PS->getNumFunctions() << "\n";
418*9880d681SAndroid Build Coastguard Worker OS << "Maximum function count: " << PS->getMaxFunctionCount() << "\n";
419*9880d681SAndroid Build Coastguard Worker OS << "Maximum internal block count: " << PS->getMaxInternalCount() << "\n";
420*9880d681SAndroid Build Coastguard Worker if (ShownFunctions && ShowIndirectCallTargets) {
421*9880d681SAndroid Build Coastguard Worker OS << "Total Number of Indirect Call Sites : " << TotalNumValueSites
422*9880d681SAndroid Build Coastguard Worker << "\n";
423*9880d681SAndroid Build Coastguard Worker OS << "Total Number of Sites With Values : "
424*9880d681SAndroid Build Coastguard Worker << TotalNumValueSitesWithValueProfile << "\n";
425*9880d681SAndroid Build Coastguard Worker OS << "Total Number of Profiled Values : " << TotalNumValues << "\n";
426*9880d681SAndroid Build Coastguard Worker }
427*9880d681SAndroid Build Coastguard Worker
428*9880d681SAndroid Build Coastguard Worker if (ShowDetailedSummary) {
429*9880d681SAndroid Build Coastguard Worker OS << "Detailed summary:\n";
430*9880d681SAndroid Build Coastguard Worker OS << "Total number of blocks: " << PS->getNumCounts() << "\n";
431*9880d681SAndroid Build Coastguard Worker OS << "Total count: " << PS->getTotalCount() << "\n";
432*9880d681SAndroid Build Coastguard Worker for (auto Entry : PS->getDetailedSummary()) {
433*9880d681SAndroid Build Coastguard Worker OS << Entry.NumCounts << " blocks with count >= " << Entry.MinCount
434*9880d681SAndroid Build Coastguard Worker << " account for "
435*9880d681SAndroid Build Coastguard Worker << format("%0.6g", (float)Entry.Cutoff / ProfileSummary::Scale * 100)
436*9880d681SAndroid Build Coastguard Worker << " percentage of the total counts.\n";
437*9880d681SAndroid Build Coastguard Worker }
438*9880d681SAndroid Build Coastguard Worker }
439*9880d681SAndroid Build Coastguard Worker return 0;
440*9880d681SAndroid Build Coastguard Worker }
441*9880d681SAndroid Build Coastguard Worker
showSampleProfile(const std::string & Filename,bool ShowCounts,bool ShowAllFunctions,const std::string & ShowFunction,raw_fd_ostream & OS)442*9880d681SAndroid Build Coastguard Worker static int showSampleProfile(const std::string &Filename, bool ShowCounts,
443*9880d681SAndroid Build Coastguard Worker bool ShowAllFunctions,
444*9880d681SAndroid Build Coastguard Worker const std::string &ShowFunction,
445*9880d681SAndroid Build Coastguard Worker raw_fd_ostream &OS) {
446*9880d681SAndroid Build Coastguard Worker using namespace sampleprof;
447*9880d681SAndroid Build Coastguard Worker LLVMContext Context;
448*9880d681SAndroid Build Coastguard Worker auto ReaderOrErr = SampleProfileReader::create(Filename, Context);
449*9880d681SAndroid Build Coastguard Worker if (std::error_code EC = ReaderOrErr.getError())
450*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, Filename);
451*9880d681SAndroid Build Coastguard Worker
452*9880d681SAndroid Build Coastguard Worker auto Reader = std::move(ReaderOrErr.get());
453*9880d681SAndroid Build Coastguard Worker if (std::error_code EC = Reader->read())
454*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, Filename);
455*9880d681SAndroid Build Coastguard Worker
456*9880d681SAndroid Build Coastguard Worker if (ShowAllFunctions || ShowFunction.empty())
457*9880d681SAndroid Build Coastguard Worker Reader->dump(OS);
458*9880d681SAndroid Build Coastguard Worker else
459*9880d681SAndroid Build Coastguard Worker Reader->dumpFunctionProfile(ShowFunction, OS);
460*9880d681SAndroid Build Coastguard Worker
461*9880d681SAndroid Build Coastguard Worker return 0;
462*9880d681SAndroid Build Coastguard Worker }
463*9880d681SAndroid Build Coastguard Worker
show_main(int argc,const char * argv[])464*9880d681SAndroid Build Coastguard Worker static int show_main(int argc, const char *argv[]) {
465*9880d681SAndroid Build Coastguard Worker cl::opt<std::string> Filename(cl::Positional, cl::Required,
466*9880d681SAndroid Build Coastguard Worker cl::desc("<profdata-file>"));
467*9880d681SAndroid Build Coastguard Worker
468*9880d681SAndroid Build Coastguard Worker cl::opt<bool> ShowCounts("counts", cl::init(false),
469*9880d681SAndroid Build Coastguard Worker cl::desc("Show counter values for shown functions"));
470*9880d681SAndroid Build Coastguard Worker cl::opt<bool> TextFormat(
471*9880d681SAndroid Build Coastguard Worker "text", cl::init(false),
472*9880d681SAndroid Build Coastguard Worker cl::desc("Show instr profile data in text dump format"));
473*9880d681SAndroid Build Coastguard Worker cl::opt<bool> ShowIndirectCallTargets(
474*9880d681SAndroid Build Coastguard Worker "ic-targets", cl::init(false),
475*9880d681SAndroid Build Coastguard Worker cl::desc("Show indirect call site target values for shown functions"));
476*9880d681SAndroid Build Coastguard Worker cl::opt<bool> ShowDetailedSummary("detailed-summary", cl::init(false),
477*9880d681SAndroid Build Coastguard Worker cl::desc("Show detailed profile summary"));
478*9880d681SAndroid Build Coastguard Worker cl::list<uint32_t> DetailedSummaryCutoffs(
479*9880d681SAndroid Build Coastguard Worker cl::CommaSeparated, "detailed-summary-cutoffs",
480*9880d681SAndroid Build Coastguard Worker cl::desc(
481*9880d681SAndroid Build Coastguard Worker "Cutoff percentages (times 10000) for generating detailed summary"),
482*9880d681SAndroid Build Coastguard Worker cl::value_desc("800000,901000,999999"));
483*9880d681SAndroid Build Coastguard Worker cl::opt<bool> ShowAllFunctions("all-functions", cl::init(false),
484*9880d681SAndroid Build Coastguard Worker cl::desc("Details for every function"));
485*9880d681SAndroid Build Coastguard Worker cl::opt<std::string> ShowFunction("function",
486*9880d681SAndroid Build Coastguard Worker cl::desc("Details for matching functions"));
487*9880d681SAndroid Build Coastguard Worker
488*9880d681SAndroid Build Coastguard Worker cl::opt<std::string> OutputFilename("output", cl::value_desc("output"),
489*9880d681SAndroid Build Coastguard Worker cl::init("-"), cl::desc("Output file"));
490*9880d681SAndroid Build Coastguard Worker cl::alias OutputFilenameA("o", cl::desc("Alias for --output"),
491*9880d681SAndroid Build Coastguard Worker cl::aliasopt(OutputFilename));
492*9880d681SAndroid Build Coastguard Worker cl::opt<ProfileKinds> ProfileKind(
493*9880d681SAndroid Build Coastguard Worker cl::desc("Profile kind:"), cl::init(instr),
494*9880d681SAndroid Build Coastguard Worker cl::values(clEnumVal(instr, "Instrumentation profile (default)"),
495*9880d681SAndroid Build Coastguard Worker clEnumVal(sample, "Sample profile"), clEnumValEnd));
496*9880d681SAndroid Build Coastguard Worker
497*9880d681SAndroid Build Coastguard Worker cl::ParseCommandLineOptions(argc, argv, "LLVM profile data summary\n");
498*9880d681SAndroid Build Coastguard Worker
499*9880d681SAndroid Build Coastguard Worker if (OutputFilename.empty())
500*9880d681SAndroid Build Coastguard Worker OutputFilename = "-";
501*9880d681SAndroid Build Coastguard Worker
502*9880d681SAndroid Build Coastguard Worker std::error_code EC;
503*9880d681SAndroid Build Coastguard Worker raw_fd_ostream OS(OutputFilename.data(), EC, sys::fs::F_Text);
504*9880d681SAndroid Build Coastguard Worker if (EC)
505*9880d681SAndroid Build Coastguard Worker exitWithErrorCode(EC, OutputFilename);
506*9880d681SAndroid Build Coastguard Worker
507*9880d681SAndroid Build Coastguard Worker if (ShowAllFunctions && !ShowFunction.empty())
508*9880d681SAndroid Build Coastguard Worker errs() << "warning: -function argument ignored: showing all functions\n";
509*9880d681SAndroid Build Coastguard Worker
510*9880d681SAndroid Build Coastguard Worker std::vector<uint32_t> Cutoffs(DetailedSummaryCutoffs.begin(),
511*9880d681SAndroid Build Coastguard Worker DetailedSummaryCutoffs.end());
512*9880d681SAndroid Build Coastguard Worker if (ProfileKind == instr)
513*9880d681SAndroid Build Coastguard Worker return showInstrProfile(Filename, ShowCounts, ShowIndirectCallTargets,
514*9880d681SAndroid Build Coastguard Worker ShowDetailedSummary, DetailedSummaryCutoffs,
515*9880d681SAndroid Build Coastguard Worker ShowAllFunctions, ShowFunction, TextFormat, OS);
516*9880d681SAndroid Build Coastguard Worker else
517*9880d681SAndroid Build Coastguard Worker return showSampleProfile(Filename, ShowCounts, ShowAllFunctions,
518*9880d681SAndroid Build Coastguard Worker ShowFunction, OS);
519*9880d681SAndroid Build Coastguard Worker }
520*9880d681SAndroid Build Coastguard Worker
main(int argc,const char * argv[])521*9880d681SAndroid Build Coastguard Worker int main(int argc, const char *argv[]) {
522*9880d681SAndroid Build Coastguard Worker // Print a stack trace if we signal out.
523*9880d681SAndroid Build Coastguard Worker sys::PrintStackTraceOnErrorSignal(argv[0]);
524*9880d681SAndroid Build Coastguard Worker PrettyStackTraceProgram X(argc, argv);
525*9880d681SAndroid Build Coastguard Worker llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
526*9880d681SAndroid Build Coastguard Worker
527*9880d681SAndroid Build Coastguard Worker StringRef ProgName(sys::path::filename(argv[0]));
528*9880d681SAndroid Build Coastguard Worker if (argc > 1) {
529*9880d681SAndroid Build Coastguard Worker int (*func)(int, const char *[]) = nullptr;
530*9880d681SAndroid Build Coastguard Worker
531*9880d681SAndroid Build Coastguard Worker if (strcmp(argv[1], "merge") == 0)
532*9880d681SAndroid Build Coastguard Worker func = merge_main;
533*9880d681SAndroid Build Coastguard Worker else if (strcmp(argv[1], "show") == 0)
534*9880d681SAndroid Build Coastguard Worker func = show_main;
535*9880d681SAndroid Build Coastguard Worker
536*9880d681SAndroid Build Coastguard Worker if (func) {
537*9880d681SAndroid Build Coastguard Worker std::string Invocation(ProgName.str() + " " + argv[1]);
538*9880d681SAndroid Build Coastguard Worker argv[1] = Invocation.c_str();
539*9880d681SAndroid Build Coastguard Worker return func(argc - 1, argv + 1);
540*9880d681SAndroid Build Coastguard Worker }
541*9880d681SAndroid Build Coastguard Worker
542*9880d681SAndroid Build Coastguard Worker if (strcmp(argv[1], "-h") == 0 || strcmp(argv[1], "-help") == 0 ||
543*9880d681SAndroid Build Coastguard Worker strcmp(argv[1], "--help") == 0) {
544*9880d681SAndroid Build Coastguard Worker
545*9880d681SAndroid Build Coastguard Worker errs() << "OVERVIEW: LLVM profile data tools\n\n"
546*9880d681SAndroid Build Coastguard Worker << "USAGE: " << ProgName << " <command> [args...]\n"
547*9880d681SAndroid Build Coastguard Worker << "USAGE: " << ProgName << " <command> -help\n\n"
548*9880d681SAndroid Build Coastguard Worker << "Available commands: merge, show\n";
549*9880d681SAndroid Build Coastguard Worker return 0;
550*9880d681SAndroid Build Coastguard Worker }
551*9880d681SAndroid Build Coastguard Worker }
552*9880d681SAndroid Build Coastguard Worker
553*9880d681SAndroid Build Coastguard Worker if (argc < 2)
554*9880d681SAndroid Build Coastguard Worker errs() << ProgName << ": No command specified!\n";
555*9880d681SAndroid Build Coastguard Worker else
556*9880d681SAndroid Build Coastguard Worker errs() << ProgName << ": Unknown command!\n";
557*9880d681SAndroid Build Coastguard Worker
558*9880d681SAndroid Build Coastguard Worker errs() << "USAGE: " << ProgName << " <merge|show> [args...]\n";
559*9880d681SAndroid Build Coastguard Worker return 1;
560*9880d681SAndroid Build Coastguard Worker }
561