forked from OSchip/llvm-project
177 lines
6.1 KiB
C++
177 lines
6.1 KiB
C++
|
//===--- tools/extra/clang-tidy/ClangTidy.cpp - Clang tidy tool -----------===//
|
||
|
//
|
||
|
// The LLVM Compiler Infrastructure
|
||
|
//
|
||
|
// This file is distributed under the University of Illinois Open Source
|
||
|
// License. See LICENSE.TXT for details.
|
||
|
//
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
///
|
||
|
/// \file This file implements a clang-tidy tool.
|
||
|
///
|
||
|
/// This tool uses the Clang Tooling infrastructure, see
|
||
|
/// http://clang.llvm.org/docs/HowToSetupToolingForLLVM.html
|
||
|
/// for details on setting it up with LLVM source tree.
|
||
|
///
|
||
|
//===----------------------------------------------------------------------===//
|
||
|
|
||
|
#include "ClangTidy.h"
|
||
|
#include "ClangTidyDiagnosticConsumer.h"
|
||
|
#include "ClangTidyModuleRegistry.h"
|
||
|
#include "clang/AST/ASTConsumer.h"
|
||
|
#include "clang/AST/ASTContext.h"
|
||
|
#include "clang/AST/Decl.h"
|
||
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
||
|
#include "clang/Lex/PPCallbacks.h"
|
||
|
#include "clang/Lex/Preprocessor.h"
|
||
|
#include "clang/Frontend/ASTConsumers.h"
|
||
|
#include "clang/Frontend/CompilerInstance.h"
|
||
|
#include "clang/Frontend/FrontendActions.h"
|
||
|
#include "clang/Frontend/TextDiagnosticPrinter.h"
|
||
|
#include "clang/Rewrite/Frontend/FixItRewriter.h"
|
||
|
#include "clang/Rewrite/Frontend/FrontendActions.h"
|
||
|
#include "clang/Tooling/Tooling.h"
|
||
|
#include "clang/Tooling/Refactoring.h"
|
||
|
#include "llvm/Support/Path.h"
|
||
|
#include "llvm/Support/Signals.h"
|
||
|
#include <vector>
|
||
|
|
||
|
using namespace clang::ast_matchers;
|
||
|
using namespace clang::driver;
|
||
|
using namespace clang::tooling;
|
||
|
using namespace llvm;
|
||
|
|
||
|
namespace clang {
|
||
|
namespace tidy {
|
||
|
namespace {
|
||
|
|
||
|
class ClangTidyPPAction : public PreprocessOnlyAction {
|
||
|
public:
|
||
|
ClangTidyPPAction(SmallVectorImpl<ClangTidyCheck *> &Checks,
|
||
|
ClangTidyContext &Context)
|
||
|
: Checks(Checks), Context(Context) {}
|
||
|
|
||
|
private:
|
||
|
virtual bool BeginSourceFileAction(CompilerInstance &Compiler,
|
||
|
llvm::StringRef file_name) {
|
||
|
Context.setSourceManager(&Compiler.getSourceManager());
|
||
|
for (SmallVectorImpl<ClangTidyCheck *>::iterator I = Checks.begin(),
|
||
|
E = Checks.end();
|
||
|
I != E; ++I)
|
||
|
(*I)->registerPPCallbacks(Compiler);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
SmallVectorImpl<ClangTidyCheck *> &Checks;
|
||
|
ClangTidyContext &Context;
|
||
|
};
|
||
|
|
||
|
class ClangTidyPPActionFactory : public FrontendActionFactory {
|
||
|
public:
|
||
|
ClangTidyPPActionFactory(SmallVectorImpl<ClangTidyCheck *> &Checks,
|
||
|
ClangTidyContext &Context)
|
||
|
: Checks(Checks), Context(Context) {}
|
||
|
|
||
|
virtual FrontendAction *create() {
|
||
|
return new ClangTidyPPAction(Checks, Context);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
SmallVectorImpl<ClangTidyCheck *> &Checks;
|
||
|
ClangTidyContext &Context;
|
||
|
};
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
ClangTidyError::ClangTidyError(const SourceManager &Sources, SourceLocation Loc,
|
||
|
StringRef Message,
|
||
|
const tooling::Replacements &Fix)
|
||
|
: Message(Message), Fix(Fix) {
|
||
|
FilePath = Sources.getFilename(Loc);
|
||
|
FileOffset = Sources.getFileOffset(Loc);
|
||
|
}
|
||
|
|
||
|
DiagnosticBuilder ClangTidyContext::Diag(SourceLocation Loc,
|
||
|
StringRef Message) {
|
||
|
return DiagEngine->Report(
|
||
|
Loc, DiagEngine->getCustomDiagID(DiagnosticsEngine::Warning, Message));
|
||
|
}
|
||
|
|
||
|
void ClangTidyContext::setDiagnosticsEngine(DiagnosticsEngine *Engine) {
|
||
|
DiagEngine = Engine;
|
||
|
}
|
||
|
|
||
|
void ClangTidyContext::setSourceManager(SourceManager *SourceMgr) {
|
||
|
DiagEngine->setSourceManager(SourceMgr);
|
||
|
}
|
||
|
|
||
|
/// \brief Store a \c ClangTidyError.
|
||
|
void ClangTidyContext::storeError(const ClangTidyError &Error) {
|
||
|
Errors->push_back(Error);
|
||
|
}
|
||
|
|
||
|
void ClangTidyCheck::run(const ast_matchers::MatchFinder::MatchResult &Result) {
|
||
|
Context->setSourceManager(Result.SourceManager);
|
||
|
check(Result);
|
||
|
}
|
||
|
|
||
|
void runClangTidy(StringRef CheckRegexString,
|
||
|
const tooling::CompilationDatabase &Compilations,
|
||
|
ArrayRef<std::string> Ranges,
|
||
|
SmallVectorImpl<ClangTidyError> *Errors) {
|
||
|
// FIXME: Ranges are currently full files. Support selecting specific
|
||
|
// (line-)ranges.
|
||
|
ClangTool Tool(Compilations, Ranges);
|
||
|
clang::tidy::ClangTidyContext Context(Errors);
|
||
|
ClangTidyDiagnosticConsumer DiagConsumer(Context);
|
||
|
ClangTidyCheckFactories CheckFactories;
|
||
|
for (ClangTidyModuleRegistry::iterator I = ClangTidyModuleRegistry::begin(),
|
||
|
E = ClangTidyModuleRegistry::end();
|
||
|
I != E; ++I) {
|
||
|
OwningPtr<ClangTidyModule> Module(I->instantiate());
|
||
|
Module->addCheckFactories(CheckFactories);
|
||
|
}
|
||
|
|
||
|
SmallVector<ClangTidyCheck *, 16> Checks;
|
||
|
CheckFactories.createChecks(CheckRegexString, Checks);
|
||
|
|
||
|
MatchFinder Finder;
|
||
|
for (SmallVectorImpl<ClangTidyCheck *>::iterator I = Checks.begin(),
|
||
|
E = Checks.end();
|
||
|
I != E; ++I) {
|
||
|
(*I)->setContext(&Context);
|
||
|
(*I)->registerMatchers(&Finder);
|
||
|
}
|
||
|
|
||
|
Tool.run(new ClangTidyPPActionFactory(Checks, Context));
|
||
|
Tool.run(newFrontendActionFactory(&Finder));
|
||
|
}
|
||
|
|
||
|
void handleErrors(SmallVectorImpl<ClangTidyError> &Errors, bool Fix) {
|
||
|
FileManager Files((FileSystemOptions()));
|
||
|
IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions();
|
||
|
DiagnosticConsumer *DiagPrinter =
|
||
|
new TextDiagnosticPrinter(llvm::outs(), &*DiagOpts);
|
||
|
DiagnosticsEngine Diags(IntrusiveRefCntPtr<DiagnosticIDs>(new DiagnosticIDs),
|
||
|
&*DiagOpts, DiagPrinter);
|
||
|
DiagPrinter->BeginSourceFile(LangOptions());
|
||
|
SourceManager SourceMgr(Diags, Files);
|
||
|
Rewriter Rewrite(SourceMgr, LangOptions());
|
||
|
for (SmallVectorImpl<ClangTidyError>::iterator I = Errors.begin(),
|
||
|
E = Errors.end();
|
||
|
I != E; ++I) {
|
||
|
const FileEntry *File = Files.getFile(I->FilePath);
|
||
|
FileID ID = SourceMgr.createFileID(File, SourceLocation(), SrcMgr::C_User);
|
||
|
SourceLocation Loc = SourceMgr.getLocForStartOfFile(ID);
|
||
|
Diags.Report(Loc.getLocWithOffset(I->FileOffset),
|
||
|
Diags.getCustomDiagID(DiagnosticsEngine::Warning, I->Message));
|
||
|
tooling::applyAllReplacements(I->Fix, Rewrite);
|
||
|
}
|
||
|
// FIXME: Run clang-format on changes.
|
||
|
if (Fix)
|
||
|
Rewrite.overwriteChangedFiles();
|
||
|
}
|
||
|
|
||
|
} // namespace tidy
|
||
|
} // namespace clang
|