2011-02-15 02:13:31 +08:00
|
|
|
//===--- CheckerRegistration.cpp - Registration for the Analyzer Checkers -===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Defines the registration function for the analyzer checkers.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-02-16 00:54:12 +08:00
|
|
|
#include "clang/StaticAnalyzer/Frontend/CheckerRegistration.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Basic/Diagnostic.h"
|
|
|
|
#include "clang/Frontend/FrontendDiagnostic.h"
|
2011-08-17 05:24:21 +08:00
|
|
|
#include "clang/StaticAnalyzer/Checkers/ClangCheckers.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/AnalyzerOptions.h"
|
2011-02-15 02:13:31 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/CheckerManager.h"
|
2011-08-17 05:24:21 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/CheckerOptInfo.h"
|
|
|
|
#include "clang/StaticAnalyzer/Core/CheckerRegistry.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/StaticAnalyzer/Frontend/FrontendActions.h"
|
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2011-08-17 09:30:59 +08:00
|
|
|
#include "llvm/Support/DynamicLibrary.h"
|
2018-11-02 23:59:37 +08:00
|
|
|
#include "llvm/Support/FormattedStream.h"
|
2011-08-17 12:56:03 +08:00
|
|
|
#include "llvm/Support/Path.h"
|
2011-02-25 08:09:51 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2014-03-09 19:36:40 +08:00
|
|
|
#include <memory>
|
2011-02-15 02:13:31 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
using namespace ento;
|
2011-08-17 09:30:59 +08:00
|
|
|
using llvm::sys::DynamicLibrary;
|
2011-02-15 02:13:31 +08:00
|
|
|
|
2011-08-17 09:30:59 +08:00
|
|
|
namespace {
|
|
|
|
class ClangCheckerRegistry : public CheckerRegistry {
|
|
|
|
typedef void (*RegisterCheckersFn)(CheckerRegistry &);
|
2011-08-17 05:24:21 +08:00
|
|
|
|
2011-08-17 09:30:59 +08:00
|
|
|
static bool isCompatibleAPIVersion(const char *versionString);
|
2011-09-26 07:23:43 +08:00
|
|
|
static void warnIncompatible(DiagnosticsEngine *diags, StringRef pluginPath,
|
2011-08-17 12:56:03 +08:00
|
|
|
const char *pluginAPIVersion);
|
|
|
|
|
|
|
|
public:
|
2011-09-26 07:23:43 +08:00
|
|
|
ClangCheckerRegistry(ArrayRef<std::string> plugins,
|
2014-05-27 10:45:47 +08:00
|
|
|
DiagnosticsEngine *diags = nullptr);
|
2011-08-17 09:30:59 +08:00
|
|
|
};
|
2015-09-08 11:50:52 +08:00
|
|
|
|
2011-08-17 09:30:59 +08:00
|
|
|
} // end anonymous namespace
|
|
|
|
|
2011-08-17 12:56:03 +08:00
|
|
|
ClangCheckerRegistry::ClangCheckerRegistry(ArrayRef<std::string> plugins,
|
2011-09-26 07:23:43 +08:00
|
|
|
DiagnosticsEngine *diags) {
|
2011-08-17 09:30:59 +08:00
|
|
|
registerBuiltinCheckers(*this);
|
|
|
|
|
|
|
|
for (ArrayRef<std::string>::iterator i = plugins.begin(), e = plugins.end();
|
|
|
|
i != e; ++i) {
|
|
|
|
// Get access to the plugin.
|
2015-07-16 04:32:07 +08:00
|
|
|
std::string err;
|
|
|
|
DynamicLibrary lib = DynamicLibrary::getPermanentLibrary(i->c_str(), &err);
|
|
|
|
if (!lib.isValid()) {
|
|
|
|
diags->Report(diag::err_fe_unable_to_load_plugin) << *i << err;
|
|
|
|
continue;
|
|
|
|
}
|
2011-08-17 09:30:59 +08:00
|
|
|
|
|
|
|
// See if it's compatible with this build of clang.
|
|
|
|
const char *pluginAPIVersion =
|
|
|
|
(const char *) lib.getAddressOfSymbol("clang_analyzerAPIVersionString");
|
2011-08-17 12:56:03 +08:00
|
|
|
if (!isCompatibleAPIVersion(pluginAPIVersion)) {
|
|
|
|
warnIncompatible(diags, *i, pluginAPIVersion);
|
2011-08-17 09:30:59 +08:00
|
|
|
continue;
|
2011-08-17 12:56:03 +08:00
|
|
|
}
|
2011-08-17 09:30:59 +08:00
|
|
|
|
|
|
|
// Register its checkers.
|
|
|
|
RegisterCheckersFn registerPluginCheckers =
|
2011-08-17 12:22:25 +08:00
|
|
|
(RegisterCheckersFn) (intptr_t) lib.getAddressOfSymbol(
|
|
|
|
"clang_registerCheckers");
|
2011-08-17 09:30:59 +08:00
|
|
|
if (registerPluginCheckers)
|
|
|
|
registerPluginCheckers(*this);
|
|
|
|
}
|
2011-08-17 05:24:21 +08:00
|
|
|
}
|
|
|
|
|
2011-08-17 09:30:59 +08:00
|
|
|
bool ClangCheckerRegistry::isCompatibleAPIVersion(const char *versionString) {
|
|
|
|
// If the version string is null, it's not an analyzer plugin.
|
2014-05-27 10:45:47 +08:00
|
|
|
if (!versionString)
|
2011-08-17 09:30:59 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// For now, none of the static analyzer API is considered stable.
|
|
|
|
// Versions must match exactly.
|
2015-12-28 23:19:39 +08:00
|
|
|
return strcmp(versionString, CLANG_ANALYZER_API_VERSION_STRING) == 0;
|
2011-08-17 09:30:59 +08:00
|
|
|
}
|
|
|
|
|
2011-09-26 07:23:43 +08:00
|
|
|
void ClangCheckerRegistry::warnIncompatible(DiagnosticsEngine *diags,
|
2011-08-17 12:56:03 +08:00
|
|
|
StringRef pluginPath,
|
|
|
|
const char *pluginAPIVersion) {
|
|
|
|
if (!diags)
|
|
|
|
return;
|
|
|
|
if (!pluginAPIVersion)
|
|
|
|
return;
|
|
|
|
|
|
|
|
diags->Report(diag::warn_incompatible_analyzer_plugin_api)
|
|
|
|
<< llvm::sys::path::filename(pluginPath);
|
|
|
|
diags->Report(diag::note_incompatible_analyzer_plugin_api)
|
|
|
|
<< CLANG_ANALYZER_API_VERSION_STRING
|
|
|
|
<< pluginAPIVersion;
|
|
|
|
}
|
|
|
|
|
2016-08-08 21:41:04 +08:00
|
|
|
static SmallVector<CheckerOptInfo, 8>
|
|
|
|
getCheckerOptList(const AnalyzerOptions &opts) {
|
|
|
|
SmallVector<CheckerOptInfo, 8> checkerOpts;
|
|
|
|
for (unsigned i = 0, e = opts.CheckersControlList.size(); i != e; ++i) {
|
|
|
|
const std::pair<std::string, bool> &opt = opts.CheckersControlList[i];
|
2016-11-02 18:39:27 +08:00
|
|
|
checkerOpts.push_back(CheckerOptInfo(opt.first, opt.second));
|
2016-08-08 21:41:04 +08:00
|
|
|
}
|
|
|
|
return checkerOpts;
|
|
|
|
}
|
|
|
|
|
2018-06-27 22:56:12 +08:00
|
|
|
std::unique_ptr<CheckerManager> ento::createCheckerManager(
|
2018-08-07 07:09:07 +08:00
|
|
|
ASTContext &context,
|
|
|
|
AnalyzerOptions &opts,
|
2018-06-27 22:56:12 +08:00
|
|
|
ArrayRef<std::string> plugins,
|
|
|
|
ArrayRef<std::function<void(CheckerRegistry &)>> checkerRegistrationFns,
|
|
|
|
DiagnosticsEngine &diags) {
|
2018-08-07 07:09:07 +08:00
|
|
|
auto checkerMgr = llvm::make_unique<CheckerManager>(context, opts);
|
2011-02-15 02:13:31 +08:00
|
|
|
|
2016-08-08 21:41:04 +08:00
|
|
|
SmallVector<CheckerOptInfo, 8> checkerOpts = getCheckerOptList(opts);
|
2011-02-15 02:13:31 +08:00
|
|
|
|
2011-08-17 12:56:03 +08:00
|
|
|
ClangCheckerRegistry allCheckers(plugins, &diags);
|
2018-06-27 22:56:12 +08:00
|
|
|
|
|
|
|
for (const auto &Fn : checkerRegistrationFns)
|
|
|
|
Fn(allCheckers);
|
|
|
|
|
2011-08-17 12:56:03 +08:00
|
|
|
allCheckers.initializeManager(*checkerMgr, checkerOpts);
|
2015-07-10 05:43:45 +08:00
|
|
|
allCheckers.validateCheckerOptions(opts, diags);
|
2011-03-01 01:36:09 +08:00
|
|
|
checkerMgr->finishedCheckerRegistration();
|
|
|
|
|
2011-02-15 02:13:31 +08:00
|
|
|
for (unsigned i = 0, e = checkerOpts.size(); i != e; ++i) {
|
2014-08-30 04:01:38 +08:00
|
|
|
if (checkerOpts[i].isUnclaimed()) {
|
2012-07-25 15:12:13 +08:00
|
|
|
diags.Report(diag::err_unknown_analyzer_checker)
|
2011-02-15 02:13:31 +08:00
|
|
|
<< checkerOpts[i].getName();
|
2014-08-30 04:01:38 +08:00
|
|
|
diags.Report(diag::note_suggest_disabling_all_checkers);
|
|
|
|
}
|
|
|
|
|
2011-02-15 02:13:31 +08:00
|
|
|
}
|
|
|
|
|
2015-01-17 08:46:55 +08:00
|
|
|
return checkerMgr;
|
2011-02-15 02:13:31 +08:00
|
|
|
}
|
2011-02-25 08:09:51 +08:00
|
|
|
|
2011-08-17 05:24:21 +08:00
|
|
|
void ento::printCheckerHelp(raw_ostream &out, ArrayRef<std::string> plugins) {
|
|
|
|
out << "OVERVIEW: Clang Static Analyzer Checkers List\n\n";
|
|
|
|
out << "USAGE: -analyzer-checker <CHECKER or PACKAGE,...>\n\n";
|
2011-02-25 08:09:51 +08:00
|
|
|
|
2011-08-17 09:30:59 +08:00
|
|
|
ClangCheckerRegistry(plugins).printHelp(out);
|
2011-02-25 08:09:51 +08:00
|
|
|
}
|
2016-08-08 21:41:04 +08:00
|
|
|
|
|
|
|
void ento::printEnabledCheckerList(raw_ostream &out,
|
|
|
|
ArrayRef<std::string> plugins,
|
|
|
|
const AnalyzerOptions &opts) {
|
|
|
|
out << "OVERVIEW: Clang Static Analyzer Enabled Checkers List\n\n";
|
|
|
|
|
|
|
|
SmallVector<CheckerOptInfo, 8> checkerOpts = getCheckerOptList(opts);
|
|
|
|
ClangCheckerRegistry(plugins).printList(out, checkerOpts);
|
|
|
|
}
|
2018-11-02 23:59:37 +08:00
|
|
|
|
|
|
|
void ento::printAnalyzerConfigList(raw_ostream &out) {
|
|
|
|
out << "OVERVIEW: Clang Static Analyzer -analyzer-config Option List\n\n";
|
|
|
|
out << "USAGE: clang -cc1 [CLANG_OPTIONS] -analyzer-config "
|
|
|
|
"<OPTION1=VALUE,OPTION2=VALUE,...>\n\n";
|
|
|
|
out << " clang -cc1 [CLANG_OPTIONS] -analyzer-config OPTION1=VALUE, "
|
|
|
|
"-analyzer-config OPTION2=VALUE, ...\n\n";
|
|
|
|
out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang"
|
|
|
|
"<OPTION1=VALUE,OPTION2=VALUE,...>\n\n";
|
|
|
|
out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang "
|
|
|
|
"OPTION1=VALUE, -Xclang -analyzer-config -Xclang "
|
|
|
|
"OPTION2=VALUE, ...\n\n";
|
|
|
|
out << "OPTIONS:\n\n";
|
|
|
|
|
|
|
|
using OptionAndDescriptionTy = std::pair<StringRef, std::string>;
|
|
|
|
OptionAndDescriptionTy PrintableOptions[] = {
|
|
|
|
#define ANALYZER_OPTION(TYPE, NAME, CMDFLAG, DESC, DEFAULT_VAL) \
|
|
|
|
{ \
|
|
|
|
CMDFLAG, \
|
|
|
|
llvm::Twine(llvm::Twine() + "(" + \
|
|
|
|
(#TYPE == "StringRef" ? "string" : #TYPE ) + ") " DESC \
|
|
|
|
" (default: " #DEFAULT_VAL ")").str() \
|
|
|
|
},
|
|
|
|
|
|
|
|
#define ANALYZER_OPTION_DEPENDS_ON_USER_MODE(TYPE, NAME, CMDFLAG, DESC, \
|
|
|
|
SHALLOW_VAL, DEEP_VAL) \
|
|
|
|
{ \
|
|
|
|
CMDFLAG, \
|
|
|
|
llvm::Twine(llvm::Twine() + "(" + \
|
|
|
|
(#TYPE == "StringRef" ? "string" : #TYPE ) + ") " DESC \
|
|
|
|
" (default: " #SHALLOW_VAL " in shallow mode, " #DEEP_VAL \
|
|
|
|
" in deep mode)").str() \
|
|
|
|
},
|
|
|
|
#include "clang/StaticAnalyzer/Core/AnalyzerOptions.def"
|
|
|
|
#undef ANALYZER_OPTION
|
|
|
|
#undef ANALYZER_OPTION_DEPENDS_ON_USER_MODE
|
|
|
|
};
|
|
|
|
|
|
|
|
llvm::sort(PrintableOptions, [](const OptionAndDescriptionTy &LHS,
|
|
|
|
const OptionAndDescriptionTy &RHS) {
|
|
|
|
return LHS.first < RHS.first;
|
|
|
|
});
|
|
|
|
|
|
|
|
constexpr size_t MinLineWidth = 70;
|
|
|
|
constexpr size_t PadForOpt = 2;
|
|
|
|
constexpr size_t OptionWidth = 30;
|
|
|
|
constexpr size_t PadForDesc = PadForOpt + OptionWidth;
|
|
|
|
static_assert(MinLineWidth > PadForDesc, "MinLineWidth must be greater!");
|
|
|
|
|
|
|
|
llvm::formatted_raw_ostream FOut(out);
|
|
|
|
|
|
|
|
for (const auto &Pair : PrintableOptions) {
|
|
|
|
FOut.PadToColumn(PadForOpt) << Pair.first;
|
|
|
|
|
|
|
|
// If the buffer's length is greater then PadForDesc, print a newline.
|
|
|
|
if (FOut.getColumn() > PadForDesc)
|
|
|
|
FOut << '\n';
|
|
|
|
|
|
|
|
FOut.PadToColumn(PadForDesc);
|
|
|
|
|
|
|
|
for (char C : Pair.second) {
|
|
|
|
if (FOut.getColumn() > MinLineWidth && C == ' ') {
|
|
|
|
FOut << '\n';
|
|
|
|
FOut.PadToColumn(PadForDesc);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
FOut << C;
|
|
|
|
}
|
|
|
|
FOut << "\n\n";
|
|
|
|
}
|
|
|
|
}
|