1 //===- llvm-cxxmap.cpp ----------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // llvm-cxxmap computes a correspondence between old symbol names and new
10 // symbol names based on a symbol equivalence file.
12 //===----------------------------------------------------------------------===//
14 #include "llvm/ADT/DenseMap.h"
15 #include "llvm/ADT/DenseSet.h"
16 #include "llvm/ADT/StringRef.h"
17 #include "llvm/ProfileData/SymbolRemappingReader.h"
18 #include "llvm/Support/CommandLine.h"
19 #include "llvm/Support/FileSystem.h"
20 #include "llvm/Support/InitLLVM.h"
21 #include "llvm/Support/LineIterator.h"
22 #include "llvm/Support/MemoryBuffer.h"
23 #include "llvm/Support/WithColor.h"
24 #include "llvm/Support/raw_ostream.h"
28 cl::OptionCategory
CXXMapCategory("CXX Map Options");
30 cl::opt
<std::string
> OldSymbolFile(cl::Positional
, cl::Required
,
31 cl::desc("<symbol-file>"),
32 cl::cat(CXXMapCategory
));
33 cl::opt
<std::string
> NewSymbolFile(cl::Positional
, cl::Required
,
34 cl::desc("<symbol-file>"),
35 cl::cat(CXXMapCategory
));
36 cl::opt
<std::string
> RemappingFile("remapping-file", cl::Required
,
37 cl::desc("Remapping file"),
38 cl::cat(CXXMapCategory
));
39 cl::alias
RemappingFileA("r", cl::aliasopt(RemappingFile
),
40 cl::cat(CXXMapCategory
));
41 cl::opt
<std::string
> OutputFilename("output", cl::value_desc("output"),
42 cl::init("-"), cl::desc("Output file"),
43 cl::cat(CXXMapCategory
));
44 cl::alias
OutputFilenameA("o", cl::aliasopt(OutputFilename
),
45 cl::cat(CXXMapCategory
));
47 cl::opt
<bool> WarnAmbiguous(
49 cl::desc("Warn on equivalent symbols in the output symbol list"),
50 cl::cat(CXXMapCategory
));
51 cl::opt
<bool> WarnIncomplete(
53 cl::desc("Warn on input symbols missing from output symbol list"),
54 cl::cat(CXXMapCategory
));
56 static void warn(Twine Message
, Twine Whence
= "",
57 std::string Hint
= "") {
59 std::string WhenceStr
= Whence
.str();
60 if (!WhenceStr
.empty())
61 errs() << WhenceStr
<< ": ";
62 errs() << Message
<< "\n";
64 WithColor::note() << Hint
<< "\n";
67 static void exitWithError(Twine Message
, Twine Whence
= "",
68 std::string Hint
= "") {
70 std::string WhenceStr
= Whence
.str();
71 if (!WhenceStr
.empty())
72 errs() << WhenceStr
<< ": ";
73 errs() << Message
<< "\n";
75 WithColor::note() << Hint
<< "\n";
79 static void exitWithError(Error E
, StringRef Whence
= "") {
80 exitWithError(toString(std::move(E
)), Whence
);
83 static void exitWithErrorCode(std::error_code EC
, StringRef Whence
= "") {
84 exitWithError(EC
.message(), Whence
);
87 static void remapSymbols(MemoryBuffer
&OldSymbolFile
,
88 MemoryBuffer
&NewSymbolFile
,
89 MemoryBuffer
&RemappingFile
,
91 // Load the remapping file and prepare to canonicalize symbols.
92 SymbolRemappingReader Reader
;
93 if (Error E
= Reader
.read(RemappingFile
))
94 exitWithError(std::move(E
));
96 // Canonicalize the new symbols.
97 DenseMap
<SymbolRemappingReader::Key
, StringRef
> MappedNames
;
98 DenseSet
<StringRef
> UnparseableSymbols
;
99 for (line_iterator
LineIt(NewSymbolFile
, /*SkipBlanks=*/true, '#');
100 !LineIt
.is_at_eof(); ++LineIt
) {
101 StringRef Symbol
= *LineIt
;
103 auto K
= Reader
.insert(Symbol
);
105 UnparseableSymbols
.insert(Symbol
);
109 auto ItAndIsNew
= MappedNames
.insert({K
, Symbol
});
110 if (WarnAmbiguous
&& !ItAndIsNew
.second
&&
111 ItAndIsNew
.first
->second
!= Symbol
) {
112 warn("symbol " + Symbol
+ " is equivalent to earlier symbol " +
113 ItAndIsNew
.first
->second
,
114 NewSymbolFile
.getBufferIdentifier() + ":" +
115 Twine(LineIt
.line_number()),
116 "later symbol will not be the target of any remappings");
120 // Figure out which new symbol each old symbol is equivalent to.
121 for (line_iterator
LineIt(OldSymbolFile
, /*SkipBlanks=*/true, '#');
122 !LineIt
.is_at_eof(); ++LineIt
) {
123 StringRef Symbol
= *LineIt
;
125 auto K
= Reader
.lookup(Symbol
);
126 StringRef NewSymbol
= MappedNames
.lookup(K
);
128 if (NewSymbol
.empty()) {
129 if (WarnIncomplete
&& !UnparseableSymbols
.count(Symbol
)) {
130 warn("no new symbol matches old symbol " + Symbol
,
131 OldSymbolFile
.getBufferIdentifier() + ":" +
132 Twine(LineIt
.line_number()));
137 Out
<< Symbol
<< " " << NewSymbol
<< "\n";
141 int main(int argc
, const char *argv
[]) {
142 InitLLVM
X(argc
, argv
);
144 cl::HideUnrelatedOptions({&CXXMapCategory
, &getColorCategory()});
145 cl::ParseCommandLineOptions(argc
, argv
, "LLVM C++ mangled name remapper\n");
147 auto OldSymbolBufOrError
= MemoryBuffer::getFileOrSTDIN(OldSymbolFile
);
148 if (!OldSymbolBufOrError
)
149 exitWithErrorCode(OldSymbolBufOrError
.getError(), OldSymbolFile
);
151 auto NewSymbolBufOrError
= MemoryBuffer::getFileOrSTDIN(NewSymbolFile
);
152 if (!NewSymbolBufOrError
)
153 exitWithErrorCode(NewSymbolBufOrError
.getError(), NewSymbolFile
);
155 auto RemappingBufOrError
= MemoryBuffer::getFileOrSTDIN(RemappingFile
);
156 if (!RemappingBufOrError
)
157 exitWithErrorCode(RemappingBufOrError
.getError(), RemappingFile
);
160 raw_fd_ostream
OS(OutputFilename
.data(), EC
, sys::fs::OF_TextWithCRLF
);
162 exitWithErrorCode(EC
, OutputFilename
);
164 remapSymbols(*OldSymbolBufOrError
.get(), *NewSymbolBufOrError
.get(),
165 *RemappingBufOrError
.get(), OS
);