forked from OSchip/llvm-project
472 lines
17 KiB
C++
472 lines
17 KiB
C++
//===-- ClangIncludeFixer.cpp - Standalone include fixer ------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "FuzzySymbolIndex.h"
|
|
#include "InMemorySymbolIndex.h"
|
|
#include "IncludeFixer.h"
|
|
#include "IncludeFixerContext.h"
|
|
#include "SymbolIndexManager.h"
|
|
#include "YamlSymbolIndex.h"
|
|
#include "clang/Format/Format.h"
|
|
#include "clang/Frontend/TextDiagnosticPrinter.h"
|
|
#include "clang/Rewrite/Core/Rewriter.h"
|
|
#include "clang/Tooling/CommonOptionsParser.h"
|
|
#include "clang/Tooling/Core/Replacement.h"
|
|
#include "clang/Tooling/Tooling.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/YAMLTraits.h"
|
|
|
|
using namespace clang;
|
|
using namespace llvm;
|
|
using clang::include_fixer::IncludeFixerContext;
|
|
|
|
LLVM_YAML_IS_DOCUMENT_LIST_VECTOR(IncludeFixerContext)
|
|
LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(IncludeFixerContext::HeaderInfo)
|
|
LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(IncludeFixerContext::QuerySymbolInfo)
|
|
|
|
namespace llvm {
|
|
namespace yaml {
|
|
|
|
template <> struct MappingTraits<tooling::Range> {
|
|
struct NormalizedRange {
|
|
NormalizedRange(const IO &) : Offset(0), Length(0) {}
|
|
|
|
NormalizedRange(const IO &, const tooling::Range &R)
|
|
: Offset(R.getOffset()), Length(R.getLength()) {}
|
|
|
|
tooling::Range denormalize(const IO &) {
|
|
return tooling::Range(Offset, Length);
|
|
}
|
|
|
|
unsigned Offset;
|
|
unsigned Length;
|
|
};
|
|
static void mapping(IO &IO, tooling::Range &Info) {
|
|
MappingNormalization<NormalizedRange, tooling::Range> Keys(IO, Info);
|
|
IO.mapRequired("Offset", Keys->Offset);
|
|
IO.mapRequired("Length", Keys->Length);
|
|
}
|
|
};
|
|
|
|
template <> struct MappingTraits<IncludeFixerContext::HeaderInfo> {
|
|
static void mapping(IO &io, IncludeFixerContext::HeaderInfo &Info) {
|
|
io.mapRequired("Header", Info.Header);
|
|
io.mapRequired("QualifiedName", Info.QualifiedName);
|
|
}
|
|
};
|
|
|
|
template <> struct MappingTraits<IncludeFixerContext::QuerySymbolInfo> {
|
|
static void mapping(IO &io, IncludeFixerContext::QuerySymbolInfo &Info) {
|
|
io.mapRequired("RawIdentifier", Info.RawIdentifier);
|
|
io.mapRequired("Range", Info.Range);
|
|
}
|
|
};
|
|
|
|
template <> struct MappingTraits<IncludeFixerContext> {
|
|
static void mapping(IO &IO, IncludeFixerContext &Context) {
|
|
IO.mapRequired("QuerySymbolInfos", Context.QuerySymbolInfos);
|
|
IO.mapRequired("HeaderInfos", Context.HeaderInfos);
|
|
IO.mapRequired("FilePath", Context.FilePath);
|
|
}
|
|
};
|
|
} // namespace yaml
|
|
} // namespace llvm
|
|
|
|
namespace {
|
|
cl::OptionCategory IncludeFixerCategory("Tool options");
|
|
|
|
enum DatabaseFormatTy {
|
|
fixed, ///< Hard-coded mapping.
|
|
yaml, ///< Yaml database created by find-all-symbols.
|
|
fuzzyYaml, ///< Yaml database with fuzzy-matched identifiers.
|
|
};
|
|
|
|
cl::opt<DatabaseFormatTy> DatabaseFormat(
|
|
"db", cl::desc("Specify input format"),
|
|
cl::values(clEnumVal(fixed, "Hard-coded mapping"),
|
|
clEnumVal(yaml, "Yaml database created by find-all-symbols"),
|
|
clEnumVal(fuzzyYaml, "Yaml database, with fuzzy-matched names")),
|
|
cl::init(yaml), cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<std::string> Input("input",
|
|
cl::desc("String to initialize the database"),
|
|
cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<std::string>
|
|
QuerySymbol("query-symbol",
|
|
cl::desc("Query a given symbol (e.g. \"a::b::foo\") in\n"
|
|
"database directly without parsing the file."),
|
|
cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<bool>
|
|
MinimizeIncludePaths("minimize-paths",
|
|
cl::desc("Whether to minimize added include paths"),
|
|
cl::init(true), cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<bool> Quiet("q", cl::desc("Reduce terminal output"), cl::init(false),
|
|
cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<bool>
|
|
STDINMode("stdin",
|
|
cl::desc("Override source file's content (in the overlaying\n"
|
|
"virtual file system) with input from <stdin> and run\n"
|
|
"the tool on the new content with the compilation\n"
|
|
"options of the source file. This mode is currently\n"
|
|
"used for editor integration."),
|
|
cl::init(false), cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<bool> OutputHeaders(
|
|
"output-headers",
|
|
cl::desc("Print the symbol being queried and all its relevant headers in\n"
|
|
"JSON format to stdout:\n"
|
|
" {\n"
|
|
" \"FilePath\": \"/path/to/foo.cc\",\n"
|
|
" \"QuerySymbolInfos\": [\n"
|
|
" {\"RawIdentifier\": \"foo\",\n"
|
|
" \"Range\": {\"Offset\": 0, \"Length\": 3}}\n"
|
|
" ],\n"
|
|
" \"HeaderInfos\": [ {\"Header\": \"\\\"foo_a.h\\\"\",\n"
|
|
" \"QualifiedName\": \"a::foo\"} ]\n"
|
|
" }"),
|
|
cl::init(false), cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<std::string> InsertHeader(
|
|
"insert-header",
|
|
cl::desc("Insert a specific header. This should run with STDIN mode.\n"
|
|
"The result is written to stdout. It is currently used for\n"
|
|
"editor integration. Support YAML/JSON format:\n"
|
|
" -insert-header=\"{\n"
|
|
" FilePath: \"/path/to/foo.cc\",\n"
|
|
" QuerySymbolInfos: [\n"
|
|
" {RawIdentifier: foo,\n"
|
|
" Range: {Offset: 0, Length: 3}}\n"
|
|
" ],\n"
|
|
" HeaderInfos: [ {Headers: \"\\\"foo_a.h\\\"\",\n"
|
|
" QualifiedName: \"a::foo\"} ]}\""),
|
|
cl::init(""), cl::cat(IncludeFixerCategory));
|
|
|
|
cl::opt<std::string>
|
|
Style("style",
|
|
cl::desc("Fallback style for reformatting after inserting new\n"
|
|
"headers if there is no clang-format config file found."),
|
|
cl::init("llvm"), cl::cat(IncludeFixerCategory));
|
|
|
|
std::unique_ptr<include_fixer::SymbolIndexManager>
|
|
createSymbolIndexManager(StringRef FilePath) {
|
|
using find_all_symbols::SymbolInfo;
|
|
|
|
auto SymbolIndexMgr = llvm::make_unique<include_fixer::SymbolIndexManager>();
|
|
switch (DatabaseFormat) {
|
|
case fixed: {
|
|
// Parse input and fill the database with it.
|
|
// <symbol>=<header><, header...>
|
|
// Multiple symbols can be given, separated by semicolons.
|
|
std::map<std::string, std::vector<std::string>> SymbolsMap;
|
|
SmallVector<StringRef, 4> SemicolonSplits;
|
|
StringRef(Input).split(SemicolonSplits, ";");
|
|
std::vector<find_all_symbols::SymbolAndSignals> Symbols;
|
|
for (StringRef Pair : SemicolonSplits) {
|
|
auto Split = Pair.split('=');
|
|
std::vector<std::string> Headers;
|
|
SmallVector<StringRef, 4> CommaSplits;
|
|
Split.second.split(CommaSplits, ",");
|
|
for (size_t I = 0, E = CommaSplits.size(); I != E; ++I)
|
|
Symbols.push_back(
|
|
{SymbolInfo(Split.first.trim(), SymbolInfo::SymbolKind::Unknown,
|
|
CommaSplits[I].trim(), {}),
|
|
// Use fake "seen" signal for tests, so first header wins.
|
|
SymbolInfo::Signals(/*Seen=*/static_cast<unsigned>(E - I),
|
|
/*Used=*/0)});
|
|
}
|
|
SymbolIndexMgr->addSymbolIndex([=]() {
|
|
return llvm::make_unique<include_fixer::InMemorySymbolIndex>(Symbols);
|
|
});
|
|
break;
|
|
}
|
|
case yaml: {
|
|
auto CreateYamlIdx = [=]() -> std::unique_ptr<include_fixer::SymbolIndex> {
|
|
llvm::ErrorOr<std::unique_ptr<include_fixer::YamlSymbolIndex>> DB(
|
|
nullptr);
|
|
if (!Input.empty()) {
|
|
DB = include_fixer::YamlSymbolIndex::createFromFile(Input);
|
|
} else {
|
|
// If we don't have any input file, look in the directory of the
|
|
// first
|
|
// file and its parents.
|
|
SmallString<128> AbsolutePath(tooling::getAbsolutePath(FilePath));
|
|
StringRef Directory = llvm::sys::path::parent_path(AbsolutePath);
|
|
DB = include_fixer::YamlSymbolIndex::createFromDirectory(
|
|
Directory, "find_all_symbols_db.yaml");
|
|
}
|
|
|
|
if (!DB) {
|
|
llvm::errs() << "Couldn't find YAML db: " << DB.getError().message()
|
|
<< '\n';
|
|
return nullptr;
|
|
}
|
|
return std::move(*DB);
|
|
};
|
|
|
|
SymbolIndexMgr->addSymbolIndex(std::move(CreateYamlIdx));
|
|
break;
|
|
}
|
|
case fuzzyYaml: {
|
|
// This mode is not very useful, because we don't correct the identifier.
|
|
// It's main purpose is to expose FuzzySymbolIndex to tests.
|
|
SymbolIndexMgr->addSymbolIndex(
|
|
[]() -> std::unique_ptr<include_fixer::SymbolIndex> {
|
|
auto DB = include_fixer::FuzzySymbolIndex::createFromYAML(Input);
|
|
if (!DB) {
|
|
llvm::errs() << "Couldn't load fuzzy YAML db: "
|
|
<< llvm::toString(DB.takeError()) << '\n';
|
|
return nullptr;
|
|
}
|
|
return std::move(*DB);
|
|
});
|
|
break;
|
|
}
|
|
}
|
|
return SymbolIndexMgr;
|
|
}
|
|
|
|
void writeToJson(llvm::raw_ostream &OS, const IncludeFixerContext& Context) {
|
|
OS << "{\n"
|
|
<< " \"FilePath\": \""
|
|
<< llvm::yaml::escape(Context.getFilePath()) << "\",\n"
|
|
<< " \"QuerySymbolInfos\": [\n";
|
|
for (const auto &Info : Context.getQuerySymbolInfos()) {
|
|
OS << " {\"RawIdentifier\": \"" << Info.RawIdentifier << "\",\n";
|
|
OS << " \"Range\":{";
|
|
OS << "\"Offset\":" << Info.Range.getOffset() << ",";
|
|
OS << "\"Length\":" << Info.Range.getLength() << "}}";
|
|
if (&Info != &Context.getQuerySymbolInfos().back())
|
|
OS << ",\n";
|
|
}
|
|
OS << "\n ],\n";
|
|
OS << " \"HeaderInfos\": [\n";
|
|
const auto &HeaderInfos = Context.getHeaderInfos();
|
|
for (const auto &Info : HeaderInfos) {
|
|
OS << " {\"Header\": \"" << llvm::yaml::escape(Info.Header) << "\",\n"
|
|
<< " \"QualifiedName\": \"" << Info.QualifiedName << "\"}";
|
|
if (&Info != &HeaderInfos.back())
|
|
OS << ",\n";
|
|
}
|
|
OS << "\n";
|
|
OS << " ]\n";
|
|
OS << "}\n";
|
|
}
|
|
|
|
int includeFixerMain(int argc, const char **argv) {
|
|
tooling::CommonOptionsParser options(argc, argv, IncludeFixerCategory);
|
|
tooling::ClangTool tool(options.getCompilations(),
|
|
options.getSourcePathList());
|
|
|
|
llvm::StringRef SourceFilePath = options.getSourcePathList().front();
|
|
// In STDINMode, we override the file content with the <stdin> input.
|
|
// Since `tool.mapVirtualFile` takes `StringRef`, we define `Code` outside of
|
|
// the if-block so that `Code` is not released after the if-block.
|
|
std::unique_ptr<llvm::MemoryBuffer> Code;
|
|
if (STDINMode) {
|
|
assert(options.getSourcePathList().size() == 1 &&
|
|
"Expect exactly one file path in STDINMode.");
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> CodeOrErr =
|
|
MemoryBuffer::getSTDIN();
|
|
if (std::error_code EC = CodeOrErr.getError()) {
|
|
errs() << EC.message() << "\n";
|
|
return 1;
|
|
}
|
|
Code = std::move(CodeOrErr.get());
|
|
if (Code->getBufferSize() == 0)
|
|
return 0; // Skip empty files.
|
|
|
|
tool.mapVirtualFile(SourceFilePath, Code->getBuffer());
|
|
}
|
|
|
|
if (!InsertHeader.empty()) {
|
|
if (!STDINMode) {
|
|
errs() << "Should be running in STDIN mode\n";
|
|
return 1;
|
|
}
|
|
|
|
llvm::yaml::Input yin(InsertHeader);
|
|
IncludeFixerContext Context;
|
|
yin >> Context;
|
|
|
|
const auto &HeaderInfos = Context.getHeaderInfos();
|
|
assert(!HeaderInfos.empty());
|
|
// We only accept one unique header.
|
|
// Check all elements in HeaderInfos have the same header.
|
|
bool IsUniqueHeader = std::equal(
|
|
HeaderInfos.begin()+1, HeaderInfos.end(), HeaderInfos.begin(),
|
|
[](const IncludeFixerContext::HeaderInfo &LHS,
|
|
const IncludeFixerContext::HeaderInfo &RHS) {
|
|
return LHS.Header == RHS.Header;
|
|
});
|
|
if (!IsUniqueHeader) {
|
|
errs() << "Expect exactly one unique header.\n";
|
|
return 1;
|
|
}
|
|
|
|
// If a header has multiple symbols, we won't add the missing namespace
|
|
// qualifiers because we don't know which one is exactly used.
|
|
//
|
|
// Check whether all elements in HeaderInfos have the same qualified name.
|
|
bool IsUniqueQualifiedName = std::equal(
|
|
HeaderInfos.begin() + 1, HeaderInfos.end(), HeaderInfos.begin(),
|
|
[](const IncludeFixerContext::HeaderInfo &LHS,
|
|
const IncludeFixerContext::HeaderInfo &RHS) {
|
|
return LHS.QualifiedName == RHS.QualifiedName;
|
|
});
|
|
auto InsertStyle = format::getStyle("file", Context.getFilePath(), Style);
|
|
if (!InsertStyle) {
|
|
llvm::errs() << llvm::toString(InsertStyle.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
auto Replacements = clang::include_fixer::createIncludeFixerReplacements(
|
|
Code->getBuffer(), Context, *InsertStyle,
|
|
/*AddQualifiers=*/IsUniqueQualifiedName);
|
|
if (!Replacements) {
|
|
errs() << "Failed to create replacements: "
|
|
<< llvm::toString(Replacements.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
|
|
auto ChangedCode =
|
|
tooling::applyAllReplacements(Code->getBuffer(), *Replacements);
|
|
if (!ChangedCode) {
|
|
llvm::errs() << llvm::toString(ChangedCode.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
llvm::outs() << *ChangedCode;
|
|
return 0;
|
|
}
|
|
|
|
// Set up data source.
|
|
std::unique_ptr<include_fixer::SymbolIndexManager> SymbolIndexMgr =
|
|
createSymbolIndexManager(SourceFilePath);
|
|
if (!SymbolIndexMgr)
|
|
return 1;
|
|
|
|
// Query symbol mode.
|
|
if (!QuerySymbol.empty()) {
|
|
auto MatchedSymbols = SymbolIndexMgr->search(
|
|
QuerySymbol, /*IsNestedSearch=*/true, SourceFilePath);
|
|
for (auto &Symbol : MatchedSymbols) {
|
|
std::string HeaderPath = Symbol.getFilePath().str();
|
|
Symbol.SetFilePath(((HeaderPath[0] == '"' || HeaderPath[0] == '<')
|
|
? HeaderPath
|
|
: "\"" + HeaderPath + "\""));
|
|
}
|
|
|
|
// We leave an empty symbol range as we don't know the range of the symbol
|
|
// being queried in this mode. include-fixer won't add namespace qualifiers
|
|
// if the symbol range is empty, which also fits this case.
|
|
IncludeFixerContext::QuerySymbolInfo Symbol;
|
|
Symbol.RawIdentifier = QuerySymbol;
|
|
auto Context =
|
|
IncludeFixerContext(SourceFilePath, {Symbol}, MatchedSymbols);
|
|
writeToJson(llvm::outs(), Context);
|
|
return 0;
|
|
}
|
|
|
|
// Now run our tool.
|
|
std::vector<include_fixer::IncludeFixerContext> Contexts;
|
|
include_fixer::IncludeFixerActionFactory Factory(*SymbolIndexMgr, Contexts,
|
|
Style, MinimizeIncludePaths);
|
|
|
|
if (tool.run(&Factory) != 0) {
|
|
// We suppress all Clang diagnostics (because they would be wrong,
|
|
// include-fixer does custom recovery) but still want to give some feedback
|
|
// in case there was a compiler error we couldn't recover from. The most
|
|
// common case for this is a #include in the file that couldn't be found.
|
|
llvm::errs() << "Fatal compiler error occurred while parsing file!"
|
|
" (incorrect include paths?)\n";
|
|
return 1;
|
|
}
|
|
|
|
assert(!Contexts.empty());
|
|
|
|
if (OutputHeaders) {
|
|
// FIXME: Print contexts of all processing files instead of the first one.
|
|
writeToJson(llvm::outs(), Contexts.front());
|
|
return 0;
|
|
}
|
|
|
|
std::vector<tooling::Replacements> FixerReplacements;
|
|
for (const auto &Context : Contexts) {
|
|
StringRef FilePath = Context.getFilePath();
|
|
auto InsertStyle = format::getStyle("file", FilePath, Style);
|
|
if (!InsertStyle) {
|
|
llvm::errs() << llvm::toString(InsertStyle.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
auto Buffer = llvm::MemoryBuffer::getFile(FilePath);
|
|
if (!Buffer) {
|
|
errs() << "Couldn't open file: " + FilePath.str() + ": "
|
|
<< Buffer.getError().message() + "\n";
|
|
return 1;
|
|
}
|
|
|
|
auto Replacements = clang::include_fixer::createIncludeFixerReplacements(
|
|
Buffer.get()->getBuffer(), Context, *InsertStyle);
|
|
if (!Replacements) {
|
|
errs() << "Failed to create replacement: "
|
|
<< llvm::toString(Replacements.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
FixerReplacements.push_back(*Replacements);
|
|
}
|
|
|
|
if (!Quiet) {
|
|
for (const auto &Context : Contexts) {
|
|
if (!Context.getHeaderInfos().empty()) {
|
|
llvm::errs() << "Added #include "
|
|
<< Context.getHeaderInfos().front().Header << " for "
|
|
<< Context.getFilePath() << "\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
if (STDINMode) {
|
|
assert(FixerReplacements.size() == 1);
|
|
auto ChangedCode = tooling::applyAllReplacements(Code->getBuffer(),
|
|
FixerReplacements.front());
|
|
if (!ChangedCode) {
|
|
llvm::errs() << llvm::toString(ChangedCode.takeError()) << "\n";
|
|
return 1;
|
|
}
|
|
llvm::outs() << *ChangedCode;
|
|
return 0;
|
|
}
|
|
|
|
// Set up a new source manager for applying the resulting replacements.
|
|
IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts(new DiagnosticOptions);
|
|
DiagnosticsEngine Diagnostics(new DiagnosticIDs, &*DiagOpts);
|
|
TextDiagnosticPrinter DiagnosticPrinter(outs(), &*DiagOpts);
|
|
SourceManager SM(Diagnostics, tool.getFiles());
|
|
Diagnostics.setClient(&DiagnosticPrinter, false);
|
|
|
|
// Write replacements to disk.
|
|
Rewriter Rewrites(SM, LangOptions());
|
|
for (const auto &Replacement : FixerReplacements) {
|
|
if (!tooling::applyAllReplacements(Replacement, Rewrites)) {
|
|
llvm::errs() << "Failed to apply replacements.\n";
|
|
return 1;
|
|
}
|
|
}
|
|
return Rewrites.overwriteChangedFiles();
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int main(int argc, const char **argv) {
|
|
return includeFixerMain(argc, argv);
|
|
}
|