2016-04-27 22:27:05 +08:00
|
|
|
//===-- FindAllSymbols.cpp - find all symbols -----------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "FindAllSymbols.h"
|
|
|
|
#include "SymbolInfo.h"
|
|
|
|
#include "clang/AST/Decl.h"
|
|
|
|
#include "clang/AST/DeclCXX.h"
|
|
|
|
#include "clang/AST/Type.h"
|
|
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
|
|
|
#include "clang/ASTMatchers/ASTMatchers.h"
|
|
|
|
#include "clang/Tooling/Tooling.h"
|
2016-05-11 16:38:21 +08:00
|
|
|
#include "llvm/ADT/Optional.h"
|
2016-04-27 22:27:05 +08:00
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace find_all_symbols {
|
|
|
|
namespace {
|
2016-05-11 16:38:21 +08:00
|
|
|
std::vector<SymbolInfo::Context> GetContexts(const NamedDecl *ND) {
|
|
|
|
std::vector<SymbolInfo::Context> Contexts;
|
2016-04-27 22:27:05 +08:00
|
|
|
for (const auto *Context = ND->getDeclContext(); Context;
|
|
|
|
Context = Context->getParent()) {
|
|
|
|
if (llvm::isa<TranslationUnitDecl>(Context) ||
|
|
|
|
llvm::isa<LinkageSpecDecl>(Context))
|
|
|
|
break;
|
|
|
|
|
|
|
|
assert(llvm::isa<NamedDecl>(Context) &&
|
|
|
|
"Expect Context to be a NamedDecl");
|
|
|
|
if (const auto *NSD = dyn_cast<NamespaceDecl>(Context)) {
|
2016-05-11 16:38:21 +08:00
|
|
|
Contexts.emplace_back(SymbolInfo::Namespace, NSD->isAnonymousNamespace()
|
|
|
|
? ""
|
|
|
|
: NSD->getName().str());
|
2016-04-27 22:27:05 +08:00
|
|
|
} else {
|
|
|
|
const auto *RD = cast<RecordDecl>(Context);
|
2016-05-11 16:38:21 +08:00
|
|
|
Contexts.emplace_back(SymbolInfo::Record, RD->getName().str());
|
2016-04-27 22:27:05 +08:00
|
|
|
}
|
|
|
|
}
|
2016-05-11 16:38:21 +08:00
|
|
|
return Contexts;
|
2016-04-27 22:27:05 +08:00
|
|
|
}
|
|
|
|
|
2016-05-11 16:38:21 +08:00
|
|
|
llvm::Optional<SymbolInfo> CreateSymbolInfo(const NamedDecl *ND,
|
|
|
|
const SourceManager &SM) {
|
|
|
|
SymbolInfo::SymbolKind Type;
|
|
|
|
if (llvm::isa<VarDecl>(ND)) {
|
|
|
|
Type = SymbolInfo::Variable;
|
|
|
|
} else if (llvm::isa<FunctionDecl>(ND)) {
|
|
|
|
Type = SymbolInfo::Function;
|
|
|
|
} else if (llvm::isa<TypedefNameDecl>(ND)) {
|
|
|
|
Type = SymbolInfo::TypedefName;
|
|
|
|
} else {
|
|
|
|
assert(llvm::isa<RecordDecl>(ND) && "Matched decl must be one of VarDecl, "
|
|
|
|
"FunctionDecl, TypedefNameDecl and "
|
|
|
|
"RecordDecl!");
|
|
|
|
// C-style record decl can have empty name, e.g "struct { ... } var;".
|
|
|
|
if (ND->getName().empty())
|
|
|
|
return llvm::None;
|
|
|
|
Type = SymbolInfo::Class;
|
|
|
|
}
|
2016-04-29 17:45:09 +08:00
|
|
|
|
2016-05-11 16:38:21 +08:00
|
|
|
SourceLocation Loc = SM.getExpansionLoc(ND->getLocation());
|
2016-04-27 22:27:05 +08:00
|
|
|
if (!Loc.isValid()) {
|
|
|
|
llvm::errs() << "Declaration " << ND->getNameAsString() << "("
|
|
|
|
<< ND->getDeclKindName()
|
|
|
|
<< ") has invalid declaration location.";
|
2016-05-11 16:38:21 +08:00
|
|
|
return llvm::None;
|
2016-04-27 22:27:05 +08:00
|
|
|
}
|
2016-05-11 16:38:21 +08:00
|
|
|
llvm::StringRef FilePath = SM.getFilename(Loc);
|
2016-04-27 22:27:05 +08:00
|
|
|
if (FilePath.empty())
|
2016-05-11 16:38:21 +08:00
|
|
|
return llvm::None;
|
2016-04-27 22:27:05 +08:00
|
|
|
|
2016-05-11 19:50:08 +08:00
|
|
|
return SymbolInfo(ND->getNameAsString(), Type, FilePath.str(),
|
2016-05-11 16:38:21 +08:00
|
|
|
GetContexts(ND), SM.getExpansionLineNumber(Loc));
|
2016-04-27 22:27:05 +08:00
|
|
|
}
|
2016-05-11 16:38:21 +08:00
|
|
|
|
2016-04-27 22:27:05 +08:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void FindAllSymbols::registerMatchers(MatchFinder *MatchFinder) {
|
|
|
|
// FIXME: Handle specialization.
|
|
|
|
auto IsInSpecialization = hasAncestor(
|
|
|
|
decl(anyOf(cxxRecordDecl(isExplicitTemplateSpecialization()),
|
|
|
|
functionDecl(isExplicitTemplateSpecialization()))));
|
|
|
|
|
|
|
|
// Matchers for both C and C++.
|
|
|
|
// We only match symbols from header files, i.e. not from main files (see
|
|
|
|
// function's comment for detailed explanation).
|
|
|
|
auto CommonFilter =
|
|
|
|
allOf(unless(isImplicit()), unless(isExpansionInMainFile()));
|
|
|
|
|
|
|
|
auto HasNSOrTUCtxMatcher =
|
|
|
|
hasDeclContext(anyOf(namespaceDecl(), translationUnitDecl()));
|
|
|
|
|
|
|
|
// We need seperate rules for C record types and C++ record types since some
|
|
|
|
// template related matchers are inapplicable on C record declarations.
|
|
|
|
//
|
|
|
|
// Matchers specific to C++ code.
|
|
|
|
// All declarations should be in namespace or translation unit.
|
|
|
|
auto CCMatcher =
|
|
|
|
allOf(HasNSOrTUCtxMatcher, unless(IsInSpecialization),
|
|
|
|
unless(ast_matchers::isTemplateInstantiation()),
|
|
|
|
unless(isInstantiated()), unless(classTemplateSpecializationDecl()),
|
|
|
|
unless(isExplicitTemplateSpecialization()));
|
|
|
|
|
|
|
|
// Matchers specific to code in extern "C" {...}.
|
|
|
|
auto ExternCMatcher = hasDeclContext(linkageSpecDecl());
|
|
|
|
|
|
|
|
// Matchers for variable declarations.
|
|
|
|
//
|
|
|
|
// In most cases, `ParmVarDecl` is filtered out by hasDeclContext(...)
|
|
|
|
// matcher since the declaration context is usually `MethodDecl`. However,
|
|
|
|
// this assumption does not hold for parameters of a function pointer
|
|
|
|
// parameter.
|
|
|
|
// For example, consider a function declaration:
|
|
|
|
// void Func(void (*)(float), int);
|
|
|
|
// The float parameter of the function pointer has an empty name, and its
|
|
|
|
// declaration context is an anonymous namespace; therefore, it won't be
|
|
|
|
// filtered out by our matchers above.
|
|
|
|
MatchFinder->addMatcher(varDecl(CommonFilter,
|
|
|
|
anyOf(ExternCMatcher, CCMatcher),
|
|
|
|
unless(parmVarDecl()))
|
|
|
|
.bind("decl"),
|
|
|
|
this);
|
|
|
|
|
|
|
|
// Matchers for C-style record declarations in extern "C" {...}.
|
|
|
|
MatchFinder->addMatcher(
|
|
|
|
recordDecl(CommonFilter, ExternCMatcher, isDefinition()).bind("decl"),
|
|
|
|
this);
|
|
|
|
|
|
|
|
// Matchers for C++ record declarations.
|
|
|
|
auto CxxRecordDecl =
|
|
|
|
cxxRecordDecl(CommonFilter, CCMatcher, isDefinition(),
|
|
|
|
unless(isExplicitTemplateSpecialization()));
|
|
|
|
MatchFinder->addMatcher(CxxRecordDecl.bind("decl"), this);
|
|
|
|
|
|
|
|
// Matchers for function declarations.
|
|
|
|
MatchFinder->addMatcher(
|
|
|
|
functionDecl(CommonFilter, anyOf(ExternCMatcher, CCMatcher)).bind("decl"),
|
|
|
|
this);
|
|
|
|
|
|
|
|
// Matcher for typedef and type alias declarations.
|
|
|
|
//
|
|
|
|
// typedef and type alias can come from C-style headers and C++ heaeders.
|
|
|
|
// For C-style header, `DeclContxet` can be either `TranslationUnitDecl`
|
|
|
|
// or `LinkageSpecDecl`.
|
|
|
|
// For C++ header, `DeclContext ` can be one of `TranslationUnitDecl`,
|
|
|
|
// `NamespaceDecl`.
|
|
|
|
// With the following context matcher, we can match `typedefNameDecl` from
|
|
|
|
// both C-style header and C++ header (except for those in classes).
|
|
|
|
// "cc_matchers" are not included since template-related matchers are not
|
|
|
|
// applicable on `TypedefNameDecl`.
|
|
|
|
MatchFinder->addMatcher(
|
|
|
|
typedefNameDecl(CommonFilter, anyOf(HasNSOrTUCtxMatcher,
|
|
|
|
hasDeclContext(linkageSpecDecl())))
|
|
|
|
.bind("decl"),
|
|
|
|
this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FindAllSymbols::run(const MatchFinder::MatchResult &Result) {
|
|
|
|
// Ignore Results in failing TUs.
|
|
|
|
if (Result.Context->getDiagnostics().hasErrorOccurred()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const NamedDecl *ND = Result.Nodes.getNodeAs<NamedDecl>("decl");
|
|
|
|
assert(ND && "Matched declaration must be a NamedDecl!");
|
|
|
|
const SourceManager *SM = Result.SourceManager;
|
|
|
|
|
2016-05-11 16:38:21 +08:00
|
|
|
llvm::Optional<SymbolInfo> Symbol = CreateSymbolInfo(ND, *SM);
|
|
|
|
if (Symbol)
|
|
|
|
Reporter->reportResult(
|
|
|
|
SM->getFileEntryForID(SM->getMainFileID())->getName(), *Symbol);
|
2016-04-27 22:27:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace find_all_symbols
|
|
|
|
} // namespace clang
|