blob: 8a26735b007e2b6e3419137dd2eafa33a21e2d39 [file] [log] [blame]
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +00001//===--- CheckerRegistration.cpp - Registration for the Analyzer Checkers -===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// Defines the registration function for the analyzer checkers.
11//
12//===----------------------------------------------------------------------===//
13
Argyrios Kyrtzidis6fa0d202011-02-15 16:54:12 +000014#include "clang/StaticAnalyzer/Frontend/CheckerRegistration.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000015#include "clang/Basic/Diagnostic.h"
16#include "clang/Frontend/FrontendDiagnostic.h"
Jordy Rose59cce712011-08-16 21:24:21 +000017#include "clang/StaticAnalyzer/Checkers/ClangCheckers.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000018#include "clang/StaticAnalyzer/Core/AnalyzerOptions.h"
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +000019#include "clang/StaticAnalyzer/Core/CheckerManager.h"
Jordy Rose59cce712011-08-16 21:24:21 +000020#include "clang/StaticAnalyzer/Core/CheckerOptInfo.h"
21#include "clang/StaticAnalyzer/Core/CheckerRegistry.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000022#include "clang/StaticAnalyzer/Frontend/FrontendActions.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000023#include "llvm/ADT/SmallVector.h"
Jordy Rose93b86e42011-08-17 01:30:59 +000024#include "llvm/Support/DynamicLibrary.h"
Kristof Umannf1f351c2018-11-02 15:59:37 +000025#include "llvm/Support/FormattedStream.h"
Jordy Rose075d73b2011-08-17 04:56:03 +000026#include "llvm/Support/Path.h"
Argyrios Kyrtzidis17bee3e2011-02-25 00:09:51 +000027#include "llvm/Support/raw_ostream.h"
Ahmed Charlesdfca6f92014-03-09 11:36:40 +000028#include <memory>
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +000029
30using namespace clang;
31using namespace ento;
Jordy Rose93b86e42011-08-17 01:30:59 +000032using llvm::sys::DynamicLibrary;
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +000033
Jordy Rose93b86e42011-08-17 01:30:59 +000034namespace {
35class ClangCheckerRegistry : public CheckerRegistry {
36 typedef void (*RegisterCheckersFn)(CheckerRegistry &);
Jordy Rose59cce712011-08-16 21:24:21 +000037
Jordy Rose93b86e42011-08-17 01:30:59 +000038 static bool isCompatibleAPIVersion(const char *versionString);
David Blaikie9c902b52011-09-25 23:23:43 +000039 static void warnIncompatible(DiagnosticsEngine *diags, StringRef pluginPath,
Jordy Rose075d73b2011-08-17 04:56:03 +000040 const char *pluginAPIVersion);
41
42public:
David Blaikie9c902b52011-09-25 23:23:43 +000043 ClangCheckerRegistry(ArrayRef<std::string> plugins,
Craig Topper0dbb7832014-05-27 02:45:47 +000044 DiagnosticsEngine *diags = nullptr);
Jordy Rose93b86e42011-08-17 01:30:59 +000045};
Ted Kremenek3a0678e2015-09-08 03:50:52 +000046
Jordy Rose93b86e42011-08-17 01:30:59 +000047} // end anonymous namespace
48
Jordy Rose075d73b2011-08-17 04:56:03 +000049ClangCheckerRegistry::ClangCheckerRegistry(ArrayRef<std::string> plugins,
David Blaikie9c902b52011-09-25 23:23:43 +000050 DiagnosticsEngine *diags) {
Jordy Rose93b86e42011-08-17 01:30:59 +000051 registerBuiltinCheckers(*this);
52
53 for (ArrayRef<std::string>::iterator i = plugins.begin(), e = plugins.end();
54 i != e; ++i) {
55 // Get access to the plugin.
Gabor Horvatha61bb642015-07-15 20:32:07 +000056 std::string err;
57 DynamicLibrary lib = DynamicLibrary::getPermanentLibrary(i->c_str(), &err);
58 if (!lib.isValid()) {
59 diags->Report(diag::err_fe_unable_to_load_plugin) << *i << err;
60 continue;
61 }
Jordy Rose93b86e42011-08-17 01:30:59 +000062
63 // See if it's compatible with this build of clang.
64 const char *pluginAPIVersion =
65 (const char *) lib.getAddressOfSymbol("clang_analyzerAPIVersionString");
Jordy Rose075d73b2011-08-17 04:56:03 +000066 if (!isCompatibleAPIVersion(pluginAPIVersion)) {
67 warnIncompatible(diags, *i, pluginAPIVersion);
Jordy Rose93b86e42011-08-17 01:30:59 +000068 continue;
Jordy Rose075d73b2011-08-17 04:56:03 +000069 }
Jordy Rose93b86e42011-08-17 01:30:59 +000070
71 // Register its checkers.
72 RegisterCheckersFn registerPluginCheckers =
Benjamin Kramer8b3929d2011-08-17 04:22:25 +000073 (RegisterCheckersFn) (intptr_t) lib.getAddressOfSymbol(
74 "clang_registerCheckers");
Jordy Rose93b86e42011-08-17 01:30:59 +000075 if (registerPluginCheckers)
76 registerPluginCheckers(*this);
77 }
Jordy Rose59cce712011-08-16 21:24:21 +000078}
79
Jordy Rose93b86e42011-08-17 01:30:59 +000080bool ClangCheckerRegistry::isCompatibleAPIVersion(const char *versionString) {
81 // If the version string is null, it's not an analyzer plugin.
Craig Topper0dbb7832014-05-27 02:45:47 +000082 if (!versionString)
Jordy Rose93b86e42011-08-17 01:30:59 +000083 return false;
84
85 // For now, none of the static analyzer API is considered stable.
86 // Versions must match exactly.
Alexander Kornienko44a784f2015-12-28 15:19:39 +000087 return strcmp(versionString, CLANG_ANALYZER_API_VERSION_STRING) == 0;
Jordy Rose93b86e42011-08-17 01:30:59 +000088}
89
David Blaikie9c902b52011-09-25 23:23:43 +000090void ClangCheckerRegistry::warnIncompatible(DiagnosticsEngine *diags,
Jordy Rose075d73b2011-08-17 04:56:03 +000091 StringRef pluginPath,
92 const char *pluginAPIVersion) {
93 if (!diags)
94 return;
95 if (!pluginAPIVersion)
96 return;
97
98 diags->Report(diag::warn_incompatible_analyzer_plugin_api)
99 << llvm::sys::path::filename(pluginPath);
100 diags->Report(diag::note_incompatible_analyzer_plugin_api)
101 << CLANG_ANALYZER_API_VERSION_STRING
102 << pluginAPIVersion;
103}
104
Gabor Horvathc4309902016-08-08 13:41:04 +0000105static SmallVector<CheckerOptInfo, 8>
106getCheckerOptList(const AnalyzerOptions &opts) {
107 SmallVector<CheckerOptInfo, 8> checkerOpts;
108 for (unsigned i = 0, e = opts.CheckersControlList.size(); i != e; ++i) {
109 const std::pair<std::string, bool> &opt = opts.CheckersControlList[i];
Malcolm Parsonsf76f6502016-11-02 10:39:27 +0000110 checkerOpts.push_back(CheckerOptInfo(opt.first, opt.second));
Gabor Horvathc4309902016-08-08 13:41:04 +0000111 }
112 return checkerOpts;
113}
114
Alexander Kornienkod00ed8e2018-06-27 14:56:12 +0000115std::unique_ptr<CheckerManager> ento::createCheckerManager(
George Karpenkov4ece68a2018-08-06 23:09:07 +0000116 ASTContext &context,
117 AnalyzerOptions &opts,
Alexander Kornienkod00ed8e2018-06-27 14:56:12 +0000118 ArrayRef<std::string> plugins,
119 ArrayRef<std::function<void(CheckerRegistry &)>> checkerRegistrationFns,
120 DiagnosticsEngine &diags) {
George Karpenkov4ece68a2018-08-06 23:09:07 +0000121 auto checkerMgr = llvm::make_unique<CheckerManager>(context, opts);
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000122
Gabor Horvathc4309902016-08-08 13:41:04 +0000123 SmallVector<CheckerOptInfo, 8> checkerOpts = getCheckerOptList(opts);
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000124
Jordy Rose075d73b2011-08-17 04:56:03 +0000125 ClangCheckerRegistry allCheckers(plugins, &diags);
Alexander Kornienkod00ed8e2018-06-27 14:56:12 +0000126
127 for (const auto &Fn : checkerRegistrationFns)
128 Fn(allCheckers);
129
Jordy Rose075d73b2011-08-17 04:56:03 +0000130 allCheckers.initializeManager(*checkerMgr, checkerOpts);
Gabor Horvathfc4c4d42015-07-09 21:43:45 +0000131 allCheckers.validateCheckerOptions(opts, diags);
Argyrios Kyrtzidisa15dfec2011-02-28 17:36:09 +0000132 checkerMgr->finishedCheckerRegistration();
133
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000134 for (unsigned i = 0, e = checkerOpts.size(); i != e; ++i) {
Anna Zaksd5478fd2014-08-29 20:01:38 +0000135 if (checkerOpts[i].isUnclaimed()) {
Ted Kremenek2b619662012-07-25 07:12:13 +0000136 diags.Report(diag::err_unknown_analyzer_checker)
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000137 << checkerOpts[i].getName();
Anna Zaksd5478fd2014-08-29 20:01:38 +0000138 diags.Report(diag::note_suggest_disabling_all_checkers);
139 }
140
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000141 }
142
Richard Trieud4b05ce2015-01-17 00:46:55 +0000143 return checkerMgr;
Argyrios Kyrtzidis556c45e2011-02-14 18:13:31 +0000144}
Argyrios Kyrtzidis17bee3e2011-02-25 00:09:51 +0000145
Jordy Rose59cce712011-08-16 21:24:21 +0000146void ento::printCheckerHelp(raw_ostream &out, ArrayRef<std::string> plugins) {
147 out << "OVERVIEW: Clang Static Analyzer Checkers List\n\n";
148 out << "USAGE: -analyzer-checker <CHECKER or PACKAGE,...>\n\n";
Argyrios Kyrtzidis17bee3e2011-02-25 00:09:51 +0000149
Jordy Rose93b86e42011-08-17 01:30:59 +0000150 ClangCheckerRegistry(plugins).printHelp(out);
Argyrios Kyrtzidis17bee3e2011-02-25 00:09:51 +0000151}
Gabor Horvathc4309902016-08-08 13:41:04 +0000152
153void ento::printEnabledCheckerList(raw_ostream &out,
154 ArrayRef<std::string> plugins,
155 const AnalyzerOptions &opts) {
156 out << "OVERVIEW: Clang Static Analyzer Enabled Checkers List\n\n";
157
158 SmallVector<CheckerOptInfo, 8> checkerOpts = getCheckerOptList(opts);
159 ClangCheckerRegistry(plugins).printList(out, checkerOpts);
160}
Kristof Umannf1f351c2018-11-02 15:59:37 +0000161
162void ento::printAnalyzerConfigList(raw_ostream &out) {
163 out << "OVERVIEW: Clang Static Analyzer -analyzer-config Option List\n\n";
164 out << "USAGE: clang -cc1 [CLANG_OPTIONS] -analyzer-config "
165 "<OPTION1=VALUE,OPTION2=VALUE,...>\n\n";
166 out << " clang -cc1 [CLANG_OPTIONS] -analyzer-config OPTION1=VALUE, "
167 "-analyzer-config OPTION2=VALUE, ...\n\n";
168 out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang"
169 "<OPTION1=VALUE,OPTION2=VALUE,...>\n\n";
170 out << " clang [CLANG_OPTIONS] -Xclang -analyzer-config -Xclang "
171 "OPTION1=VALUE, -Xclang -analyzer-config -Xclang "
172 "OPTION2=VALUE, ...\n\n";
173 out << "OPTIONS:\n\n";
174
175 using OptionAndDescriptionTy = std::pair<StringRef, std::string>;
176 OptionAndDescriptionTy PrintableOptions[] = {
177#define ANALYZER_OPTION(TYPE, NAME, CMDFLAG, DESC, DEFAULT_VAL) \
178 { \
179 CMDFLAG, \
180 llvm::Twine(llvm::Twine() + "(" + \
Kristof Umann37829b52018-11-02 19:48:56 +0000181 (StringRef(#TYPE) == "StringRef" ? "string" : #TYPE ) + \
182 ") " DESC \
Kristof Umannf1f351c2018-11-02 15:59:37 +0000183 " (default: " #DEFAULT_VAL ")").str() \
184 },
185
186#define ANALYZER_OPTION_DEPENDS_ON_USER_MODE(TYPE, NAME, CMDFLAG, DESC, \
187 SHALLOW_VAL, DEEP_VAL) \
188 { \
189 CMDFLAG, \
190 llvm::Twine(llvm::Twine() + "(" + \
Kristof Umann37829b52018-11-02 19:48:56 +0000191 (StringRef(#TYPE) == "StringRef" ? "string" : #TYPE ) + \
192 ") " DESC \
Kristof Umannf1f351c2018-11-02 15:59:37 +0000193 " (default: " #SHALLOW_VAL " in shallow mode, " #DEEP_VAL \
194 " in deep mode)").str() \
195 },
196#include "clang/StaticAnalyzer/Core/AnalyzerOptions.def"
197#undef ANALYZER_OPTION
198#undef ANALYZER_OPTION_DEPENDS_ON_USER_MODE
199 };
200
201 llvm::sort(PrintableOptions, [](const OptionAndDescriptionTy &LHS,
202 const OptionAndDescriptionTy &RHS) {
203 return LHS.first < RHS.first;
204 });
205
206 constexpr size_t MinLineWidth = 70;
207 constexpr size_t PadForOpt = 2;
208 constexpr size_t OptionWidth = 30;
209 constexpr size_t PadForDesc = PadForOpt + OptionWidth;
210 static_assert(MinLineWidth > PadForDesc, "MinLineWidth must be greater!");
211
212 llvm::formatted_raw_ostream FOut(out);
213
214 for (const auto &Pair : PrintableOptions) {
215 FOut.PadToColumn(PadForOpt) << Pair.first;
216
217 // If the buffer's length is greater then PadForDesc, print a newline.
218 if (FOut.getColumn() > PadForDesc)
219 FOut << '\n';
220
221 FOut.PadToColumn(PadForDesc);
222
223 for (char C : Pair.second) {
224 if (FOut.getColumn() > MinLineWidth && C == ' ') {
225 FOut << '\n';
226 FOut.PadToColumn(PadForDesc);
227 continue;
228 }
229 FOut << C;
230 }
231 FOut << "\n\n";
232 }
233}