1 //===- CheckerRegistry.cpp - Maintains all available checkers -------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "clang/StaticAnalyzer/Frontend/CheckerRegistry.h"
10 #include "clang/Basic/Diagnostic.h"
11 #include "clang/Basic/LLVM.h"
12 #include "clang/Driver/DriverDiagnostic.h"
13 #include "clang/Frontend/FrontendDiagnostic.h"
14 #include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"
15 #include "clang/StaticAnalyzer/Core/AnalyzerOptions.h"
16 #include "clang/StaticAnalyzer/Core/CheckerManager.h"
17 #include "llvm/ADT/STLExtras.h"
18 #include "llvm/ADT/SetVector.h"
19 #include "llvm/ADT/StringMap.h"
20 #include "llvm/ADT/StringRef.h"
21 #include "llvm/Support/DynamicLibrary.h"
22 #include "llvm/Support/Path.h"
23 #include "llvm/Support/raw_ostream.h"
24 #include <algorithm>
25 
26 using namespace clang;
27 using namespace ento;
28 using llvm::sys::DynamicLibrary;
29 
30 //===----------------------------------------------------------------------===//
31 // Utilities.
32 //===----------------------------------------------------------------------===//
33 
34 using RegisterCheckersFn = void (*)(CheckerRegistry &);
35 
36 static bool isCompatibleAPIVersion(const char *VersionString) {
37   // If the version string is null, its not an analyzer plugin.
38   if (!VersionString)
39     return false;
40 
41   // For now, none of the static analyzer API is considered stable.
42   // Versions must match exactly.
43   return strcmp(VersionString, CLANG_ANALYZER_API_VERSION_STRING) == 0;
44 }
45 
46 namespace {
47 template <class T> struct FullNameLT {
48   bool operator()(const T &Lhs, const T &Rhs) {
49     return Lhs.FullName < Rhs.FullName;
50   }
51 };
52 
53 using PackageNameLT = FullNameLT<CheckerRegistry::PackageInfo>;
54 using CheckerNameLT = FullNameLT<CheckerRegistry::CheckerInfo>;
55 } // end of anonymous namespace
56 
57 template <class CheckerOrPackageInfoList>
58 static std::conditional_t<std::is_const<CheckerOrPackageInfoList>::value,
59                           typename CheckerOrPackageInfoList::const_iterator,
60                           typename CheckerOrPackageInfoList::iterator>
61 binaryFind(CheckerOrPackageInfoList &Collection, StringRef FullName) {
62 
63   using CheckerOrPackage = typename CheckerOrPackageInfoList::value_type;
64   using CheckerOrPackageFullNameLT = FullNameLT<CheckerOrPackage>;
65 
66   assert(llvm::is_sorted(Collection, CheckerOrPackageFullNameLT{}) &&
67          "In order to efficiently gather checkers/packages, this function "
68          "expects them to be already sorted!");
69 
70   return llvm::lower_bound(Collection, CheckerOrPackage(FullName),
71                            CheckerOrPackageFullNameLT{});
72 }
73 
74 static constexpr char PackageSeparator = '.';
75 
76 static bool isInPackage(const CheckerRegistry::CheckerInfo &Checker,
77                         StringRef PackageName) {
78   // Does the checker's full name have the package as a prefix?
79   if (!Checker.FullName.startswith(PackageName))
80     return false;
81 
82   // Is the package actually just the name of a specific checker?
83   if (Checker.FullName.size() == PackageName.size())
84     return true;
85 
86   // Is the checker in the package (or a subpackage)?
87   if (Checker.FullName[PackageName.size()] == PackageSeparator)
88     return true;
89 
90   return false;
91 }
92 
93 //===----------------------------------------------------------------------===//
94 // Methods of CmdLineOption, PackageInfo and CheckerInfo.
95 //===----------------------------------------------------------------------===//
96 
97 LLVM_DUMP_METHOD void CheckerRegistry::CmdLineOption::dump() const {
98   dumpToStream(llvm::errs());
99 }
100 
101 LLVM_DUMP_METHOD void
102 CheckerRegistry::CmdLineOption::dumpToStream(llvm::raw_ostream &Out) const {
103   // The description can be just checked in Checkers.inc, the point here is to
104   // debug whether we succeeded in parsing it.
105   Out << OptionName << " (" << OptionType << ", "
106       << (IsHidden ? "hidden, " : "") << DevelopmentStatus << ") default: \""
107       << DefaultValStr;
108 }
109 
110 static StringRef toString(CheckerRegistry::StateFromCmdLine Kind) {
111   switch (Kind) {
112   case CheckerRegistry::StateFromCmdLine::State_Disabled:
113     return "Disabled";
114   case CheckerRegistry::StateFromCmdLine::State_Enabled:
115     return "Enabled";
116   case CheckerRegistry::StateFromCmdLine::State_Unspecified:
117     return "Unspecified";
118   }
119   llvm_unreachable("Unhandled CheckerRegistry::StateFromCmdLine enum");
120 }
121 
122 LLVM_DUMP_METHOD void CheckerRegistry::CheckerInfo::dump() const {
123   dumpToStream(llvm::errs());
124 }
125 
126 LLVM_DUMP_METHOD void
127 CheckerRegistry::CheckerInfo::dumpToStream(llvm::raw_ostream &Out) const {
128   // The description can be just checked in Checkers.inc, the point here is to
129   // debug whether we succeeded in parsing it. Same with documentation uri.
130   Out << FullName << " (" << toString(State) << (IsHidden ? ", hidden" : "")
131       << ")\n";
132   Out << "  Options:\n";
133   for (const CmdLineOption &Option : CmdLineOptions) {
134     Out << "    ";
135     Option.dumpToStream(Out);
136     Out << '\n';
137   }
138   Out << "  Dependencies:\n";
139   for (const CheckerInfo *Dependency : Dependencies) {
140     Out << "  " << Dependency->FullName << '\n';
141   }
142   Out << "  Weak dependencies:\n";
143   for (const CheckerInfo *Dependency : WeakDependencies) {
144     Out << "    " << Dependency->FullName << '\n';
145   }
146 }
147 
148 LLVM_DUMP_METHOD void CheckerRegistry::PackageInfo::dump() const {
149   dumpToStream(llvm::errs());
150 }
151 
152 LLVM_DUMP_METHOD void
153 CheckerRegistry::PackageInfo::dumpToStream(llvm::raw_ostream &Out) const {
154   Out << FullName << "\n";
155   Out << "  Options:\n";
156   for (const CmdLineOption &Option : CmdLineOptions) {
157     Out << "    ";
158     Option.dumpToStream(Out);
159     Out << '\n';
160   }
161 }
162 
163 //===----------------------------------------------------------------------===//
164 // Methods of CheckerRegistry.
165 //===----------------------------------------------------------------------===//
166 
167 CheckerRegistry::CheckerInfoListRange
168 CheckerRegistry::getMutableCheckersForCmdLineArg(StringRef CmdLineArg) {
169   auto It = binaryFind(Checkers, CmdLineArg);
170 
171   if (!isInPackage(*It, CmdLineArg))
172     return {Checkers.end(), Checkers.end()};
173 
174   // See how large the package is.
175   // If the package doesn't exist, assume the option refers to a single
176   // checker.
177   size_t Size = 1;
178   llvm::StringMap<size_t>::const_iterator PackageSize =
179       PackageSizes.find(CmdLineArg);
180 
181   if (PackageSize != PackageSizes.end())
182     Size = PackageSize->getValue();
183 
184   return {It, It + Size};
185 }
186 
187 CheckerRegistry::CheckerRegistry(
188     ArrayRef<std::string> Plugins, DiagnosticsEngine &Diags,
189     AnalyzerOptions &AnOpts,
190     ArrayRef<std::function<void(CheckerRegistry &)>> CheckerRegistrationFns)
191     : Diags(Diags), AnOpts(AnOpts) {
192 
193   // Register builtin checkers.
194 #define GET_CHECKERS
195 #define CHECKER(FULLNAME, CLASS, HELPTEXT, DOC_URI, IS_HIDDEN)                 \
196   addChecker(register##CLASS, shouldRegister##CLASS, FULLNAME, HELPTEXT,       \
197              DOC_URI, IS_HIDDEN);
198 
199 #define GET_PACKAGES
200 #define PACKAGE(FULLNAME) addPackage(FULLNAME);
201 
202 #include "clang/StaticAnalyzer/Checkers/Checkers.inc"
203 #undef CHECKER
204 #undef GET_CHECKERS
205 #undef PACKAGE
206 #undef GET_PACKAGES
207 
208   // Register checkers from plugins.
209   for (const std::string &Plugin : Plugins) {
210     // Get access to the plugin.
211     std::string ErrorMsg;
212     DynamicLibrary Lib =
213         DynamicLibrary::getPermanentLibrary(Plugin.c_str(), &ErrorMsg);
214     if (!Lib.isValid()) {
215       Diags.Report(diag::err_fe_unable_to_load_plugin) << Plugin << ErrorMsg;
216       continue;
217     }
218 
219     // See if its compatible with this build of clang.
220     const char *PluginAPIVersion = static_cast<const char *>(
221         Lib.getAddressOfSymbol("clang_analyzerAPIVersionString"));
222 
223     if (!isCompatibleAPIVersion(PluginAPIVersion)) {
224       Diags.Report(diag::warn_incompatible_analyzer_plugin_api)
225           << llvm::sys::path::filename(Plugin);
226       Diags.Report(diag::note_incompatible_analyzer_plugin_api)
227           << CLANG_ANALYZER_API_VERSION_STRING << PluginAPIVersion;
228       continue;
229     }
230 
231     // Register its checkers.
232     RegisterCheckersFn RegisterPluginCheckers =
233         reinterpret_cast<RegisterCheckersFn>(
234             Lib.getAddressOfSymbol("clang_registerCheckers"));
235     if (RegisterPluginCheckers)
236       RegisterPluginCheckers(*this);
237   }
238 
239   // Register statically linked checkers, that aren't generated from the tblgen
240   // file, but rather passed their registry function as a parameter in
241   // checkerRegistrationFns.
242 
243   for (const auto &Fn : CheckerRegistrationFns)
244     Fn(*this);
245 
246   // Sort checkers for efficient collection.
247   // FIXME: Alphabetical sort puts 'experimental' in the middle.
248   // Would it be better to name it '~experimental' or something else
249   // that's ASCIIbetically last?
250   llvm::sort(Packages, PackageNameLT{});
251   llvm::sort(Checkers, CheckerNameLT{});
252 
253 #define GET_CHECKER_DEPENDENCIES
254 
255 #define CHECKER_DEPENDENCY(FULLNAME, DEPENDENCY)                               \
256   addDependency(FULLNAME, DEPENDENCY);
257 
258 #define GET_CHECKER_WEAK_DEPENDENCIES
259 
260 #define CHECKER_WEAK_DEPENDENCY(FULLNAME, DEPENDENCY)                          \
261   addWeakDependency(FULLNAME, DEPENDENCY);
262 
263 #define GET_CHECKER_OPTIONS
264 #define CHECKER_OPTION(TYPE, FULLNAME, CMDFLAG, DESC, DEFAULT_VAL,             \
265                        DEVELOPMENT_STATUS, IS_HIDDEN)                          \
266   addCheckerOption(TYPE, FULLNAME, CMDFLAG, DEFAULT_VAL, DESC,                 \
267                    DEVELOPMENT_STATUS, IS_HIDDEN);
268 
269 #define GET_PACKAGE_OPTIONS
270 #define PACKAGE_OPTION(TYPE, FULLNAME, CMDFLAG, DESC, DEFAULT_VAL,             \
271                        DEVELOPMENT_STATUS, IS_HIDDEN)                          \
272   addPackageOption(TYPE, FULLNAME, CMDFLAG, DEFAULT_VAL, DESC,                 \
273                    DEVELOPMENT_STATUS, IS_HIDDEN);
274 
275 #include "clang/StaticAnalyzer/Checkers/Checkers.inc"
276 #undef CHECKER_DEPENDENCY
277 #undef GET_CHECKER_DEPENDENCIES
278 #undef CHECKER_WEAK_DEPENDENCY
279 #undef GET_CHECKER_WEAK_DEPENDENCIES
280 #undef CHECKER_OPTION
281 #undef GET_CHECKER_OPTIONS
282 #undef PACKAGE_OPTION
283 #undef GET_PACKAGE_OPTIONS
284 
285   resolveDependencies<true>();
286   resolveDependencies<false>();
287 
288   for (auto &DepPair : Dependencies) {
289     for (auto &WeakDepPair : WeakDependencies) {
290       // Some assertions to enforce that strong dependencies are relations in
291       // between purely modeling checkers, and weak dependencies are about
292       // diagnostics.
293       assert(WeakDepPair != DepPair &&
294              "A checker cannot strong and weak depend on the same checker!");
295       assert(WeakDepPair.first != DepPair.second &&
296              "A strong dependency mustn't have weak dependencies!");
297       assert(WeakDepPair.second != DepPair.second &&
298              "A strong dependency mustn't be a weak dependency as well!");
299     }
300   }
301 
302   resolveCheckerAndPackageOptions();
303 
304   // Parse '-analyzer-checker' and '-analyzer-disable-checker' options from the
305   // command line.
306   for (const std::pair<std::string, bool> &Opt : AnOpts.CheckersAndPackages) {
307     CheckerInfoListRange CheckerForCmdLineArg =
308         getMutableCheckersForCmdLineArg(Opt.first);
309 
310     if (CheckerForCmdLineArg.begin() == CheckerForCmdLineArg.end()) {
311       Diags.Report(diag::err_unknown_analyzer_checker_or_package) << Opt.first;
312       Diags.Report(diag::note_suggest_disabling_all_checkers);
313     }
314 
315     for (CheckerInfo &checker : CheckerForCmdLineArg) {
316       checker.State = Opt.second ? StateFromCmdLine::State_Enabled
317                                  : StateFromCmdLine::State_Disabled;
318     }
319   }
320   validateCheckerOptions();
321 }
322 
323 //===----------------------------------------------------------------------===//
324 // Dependency resolving.
325 //===----------------------------------------------------------------------===//
326 
327 template <typename IsEnabledFn>
328 static bool
329 collectStrongDependencies(const CheckerRegistry::ConstCheckerInfoList &Deps,
330                           const CheckerManager &Mgr,
331                           CheckerRegistry::CheckerInfoSet &Ret,
332                           IsEnabledFn IsEnabled);
333 
334 /// Collects weak dependencies in \p enabledCheckers.
335 template <typename IsEnabledFn>
336 static void
337 collectWeakDependencies(const CheckerRegistry::ConstCheckerInfoList &Deps,
338                         const CheckerManager &Mgr,
339                         CheckerRegistry::CheckerInfoSet &Ret,
340                         IsEnabledFn IsEnabled);
341 
342 void CheckerRegistry::initializeRegistry(const CheckerManager &Mgr) {
343   // First, we calculate the list of enabled checkers as specified by the
344   // invocation. Weak dependencies will not enable their unspecified strong
345   // depenencies, but its only after resolving strong dependencies for all
346   // checkers when we know whether they will be enabled.
347   CheckerInfoSet Tmp;
348   auto IsEnabledFromCmdLine = [&](const CheckerInfo *Checker) {
349     return !Checker->isDisabled(Mgr);
350   };
351   for (const CheckerInfo &Checker : Checkers) {
352     if (!Checker.isEnabled(Mgr))
353       continue;
354 
355     CheckerInfoSet Deps;
356     if (!collectStrongDependencies(Checker.Dependencies, Mgr, Deps,
357                                    IsEnabledFromCmdLine)) {
358       // If we failed to enable any of the dependencies, don't enable this
359       // checker.
360       continue;
361     }
362 
363     Tmp.insert(Deps.begin(), Deps.end());
364 
365     // Enable the checker.
366     Tmp.insert(&Checker);
367   }
368 
369   // Calculate enabled checkers with the correct registration order. As this is
370   // done recursively, its arguably cheaper, but for sure less error prone to
371   // recalculate from scratch.
372   auto IsEnabled = [&](const CheckerInfo *Checker) {
373     return llvm::is_contained(Tmp, Checker);
374   };
375   for (const CheckerInfo &Checker : Checkers) {
376     if (!Checker.isEnabled(Mgr))
377       continue;
378 
379     CheckerInfoSet Deps;
380 
381     collectWeakDependencies(Checker.WeakDependencies, Mgr, Deps, IsEnabled);
382 
383     if (!collectStrongDependencies(Checker.Dependencies, Mgr, Deps,
384                                    IsEnabledFromCmdLine)) {
385       // If we failed to enable any of the dependencies, don't enable this
386       // checker.
387       continue;
388     }
389 
390     // Note that set_union also preserves the order of insertion.
391     EnabledCheckers.set_union(Deps);
392     EnabledCheckers.insert(&Checker);
393   }
394 }
395 
396 template <typename IsEnabledFn>
397 static bool
398 collectStrongDependencies(const CheckerRegistry::ConstCheckerInfoList &Deps,
399                           const CheckerManager &Mgr,
400                           CheckerRegistry::CheckerInfoSet &Ret,
401                           IsEnabledFn IsEnabled) {
402 
403   for (const CheckerRegistry::CheckerInfo *Dependency : Deps) {
404     if (!IsEnabled(Dependency))
405       return false;
406 
407     // Collect dependencies recursively.
408     if (!collectStrongDependencies(Dependency->Dependencies, Mgr, Ret,
409                                    IsEnabled))
410       return false;
411     Ret.insert(Dependency);
412   }
413 
414   return true;
415 }
416 
417 template <typename IsEnabledFn>
418 static void
419 collectWeakDependencies(const CheckerRegistry::ConstCheckerInfoList &WeakDeps,
420                         const CheckerManager &Mgr,
421                         CheckerRegistry::CheckerInfoSet &Ret,
422                         IsEnabledFn IsEnabled) {
423 
424   for (const CheckerRegistry::CheckerInfo *Dependency : WeakDeps) {
425     // Don't enable this checker if strong dependencies are unsatisfied, but
426     // assume that weak dependencies are transitive.
427     collectWeakDependencies(Dependency->WeakDependencies, Mgr, Ret, IsEnabled);
428 
429     if (IsEnabled(Dependency) &&
430         collectStrongDependencies(Dependency->Dependencies, Mgr, Ret,
431                                   IsEnabled))
432       Ret.insert(Dependency);
433   }
434 }
435 
436 template <bool IsWeak> void CheckerRegistry::resolveDependencies() {
437   for (const std::pair<StringRef, StringRef> &Entry :
438        (IsWeak ? WeakDependencies : Dependencies)) {
439 
440     auto CheckerIt = binaryFind(Checkers, Entry.first);
441     assert(CheckerIt != Checkers.end() && CheckerIt->FullName == Entry.first &&
442            "Failed to find the checker while attempting to set up its "
443            "dependencies!");
444 
445     auto DependencyIt = binaryFind(Checkers, Entry.second);
446     assert(DependencyIt != Checkers.end() &&
447            DependencyIt->FullName == Entry.second &&
448            "Failed to find the dependency of a checker!");
449 
450     if (IsWeak)
451       CheckerIt->WeakDependencies.emplace_back(&*DependencyIt);
452     else
453       CheckerIt->Dependencies.emplace_back(&*DependencyIt);
454   }
455 }
456 
457 void CheckerRegistry::addDependency(StringRef FullName, StringRef Dependency) {
458   Dependencies.emplace_back(FullName, Dependency);
459 }
460 
461 void CheckerRegistry::addWeakDependency(StringRef FullName,
462                                         StringRef Dependency) {
463   WeakDependencies.emplace_back(FullName, Dependency);
464 }
465 
466 //===----------------------------------------------------------------------===//
467 // Checker option resolving and validating.
468 //===----------------------------------------------------------------------===//
469 
470 /// Insert the checker/package option to AnalyzerOptions' config table, and
471 /// validate it, if the user supplied it on the command line.
472 static void insertAndValidate(StringRef FullName,
473                               const CheckerRegistry::CmdLineOption &Option,
474                               AnalyzerOptions &AnOpts,
475                               DiagnosticsEngine &Diags) {
476 
477   std::string FullOption = (FullName + ":" + Option.OptionName).str();
478 
479   auto It =
480       AnOpts.Config.insert({FullOption, std::string(Option.DefaultValStr)});
481 
482   // Insertation was successful -- CmdLineOption's constructor will validate
483   // whether values received from plugins or TableGen files are correct.
484   if (It.second)
485     return;
486 
487   // Insertion failed, the user supplied this package/checker option on the
488   // command line. If the supplied value is invalid, we'll restore the option
489   // to it's default value, and if we're in non-compatibility mode, we'll also
490   // emit an error.
491 
492   StringRef SuppliedValue = It.first->getValue();
493 
494   if (Option.OptionType == "bool") {
495     if (SuppliedValue != "true" && SuppliedValue != "false") {
496       if (AnOpts.ShouldEmitErrorsOnInvalidConfigValue) {
497         Diags.Report(diag::err_analyzer_checker_option_invalid_input)
498             << FullOption << "a boolean value";
499       }
500 
501       It.first->setValue(std::string(Option.DefaultValStr));
502     }
503     return;
504   }
505 
506   if (Option.OptionType == "int") {
507     int Tmp;
508     bool HasFailed = SuppliedValue.getAsInteger(0, Tmp);
509     if (HasFailed) {
510       if (AnOpts.ShouldEmitErrorsOnInvalidConfigValue) {
511         Diags.Report(diag::err_analyzer_checker_option_invalid_input)
512             << FullOption << "an integer value";
513       }
514 
515       It.first->setValue(std::string(Option.DefaultValStr));
516     }
517     return;
518   }
519 }
520 
521 template <class T>
522 static void
523 insertOptionToCollection(StringRef FullName, T &Collection,
524                          const CheckerRegistry::CmdLineOption &Option,
525                          AnalyzerOptions &AnOpts, DiagnosticsEngine &Diags) {
526   auto It = binaryFind(Collection, FullName);
527   assert(It != Collection.end() &&
528          "Failed to find the checker while attempting to add a command line "
529          "option to it!");
530 
531   insertAndValidate(FullName, Option, AnOpts, Diags);
532 
533   It->CmdLineOptions.emplace_back(Option);
534 }
535 
536 void CheckerRegistry::resolveCheckerAndPackageOptions() {
537   for (const std::pair<StringRef, CmdLineOption> &CheckerOptEntry :
538        CheckerOptions) {
539     insertOptionToCollection(CheckerOptEntry.first, Checkers,
540                              CheckerOptEntry.second, AnOpts, Diags);
541   }
542 
543   for (const std::pair<StringRef, CmdLineOption> &PackageOptEntry :
544        PackageOptions) {
545     insertOptionToCollection(PackageOptEntry.first, Packages,
546                              PackageOptEntry.second, AnOpts, Diags);
547   }
548 }
549 
550 void CheckerRegistry::addPackage(StringRef FullName) {
551   Packages.emplace_back(PackageInfo(FullName));
552 }
553 
554 void CheckerRegistry::addPackageOption(StringRef OptionType,
555                                        StringRef PackageFullName,
556                                        StringRef OptionName,
557                                        StringRef DefaultValStr,
558                                        StringRef Description,
559                                        StringRef DevelopmentStatus,
560                                        bool IsHidden) {
561   PackageOptions.emplace_back(
562       PackageFullName, CmdLineOption{OptionType, OptionName, DefaultValStr,
563                                      Description, DevelopmentStatus, IsHidden});
564 }
565 
566 void CheckerRegistry::addChecker(InitializationFunction Rfn,
567                                  ShouldRegisterFunction Sfn, StringRef Name,
568                                  StringRef Desc, StringRef DocsUri,
569                                  bool IsHidden) {
570   Checkers.emplace_back(Rfn, Sfn, Name, Desc, DocsUri, IsHidden);
571 
572   // Record the presence of the checker in its packages.
573   StringRef PackageName, LeafName;
574   std::tie(PackageName, LeafName) = Name.rsplit(PackageSeparator);
575   while (!LeafName.empty()) {
576     PackageSizes[PackageName] += 1;
577     std::tie(PackageName, LeafName) = PackageName.rsplit(PackageSeparator);
578   }
579 }
580 
581 void CheckerRegistry::addCheckerOption(StringRef OptionType,
582                                        StringRef CheckerFullName,
583                                        StringRef OptionName,
584                                        StringRef DefaultValStr,
585                                        StringRef Description,
586                                        StringRef DevelopmentStatus,
587                                        bool IsHidden) {
588   CheckerOptions.emplace_back(
589       CheckerFullName, CmdLineOption{OptionType, OptionName, DefaultValStr,
590                                      Description, DevelopmentStatus, IsHidden});
591 }
592 
593 void CheckerRegistry::initializeManager(CheckerManager &CheckerMgr) const {
594   // Initialize the CheckerManager with all enabled checkers.
595   for (const auto *Checker : EnabledCheckers) {
596     CheckerMgr.setCurrentCheckerName(CheckerNameRef(Checker->FullName));
597     Checker->Initialize(CheckerMgr);
598   }
599 }
600 
601 static void
602 isOptionContainedIn(const CheckerRegistry::CmdLineOptionList &OptionList,
603                     StringRef SuppliedChecker, StringRef SuppliedOption,
604                     const AnalyzerOptions &AnOpts, DiagnosticsEngine &Diags) {
605 
606   if (!AnOpts.ShouldEmitErrorsOnInvalidConfigValue)
607     return;
608 
609   using CmdLineOption = CheckerRegistry::CmdLineOption;
610 
611   auto SameOptName = [SuppliedOption](const CmdLineOption &Opt) {
612     return Opt.OptionName == SuppliedOption;
613   };
614 
615   const auto *OptionIt = llvm::find_if(OptionList, SameOptName);
616 
617   if (OptionIt == OptionList.end()) {
618     Diags.Report(diag::err_analyzer_checker_option_unknown)
619         << SuppliedChecker << SuppliedOption;
620     return;
621   }
622 }
623 
624 void CheckerRegistry::validateCheckerOptions() const {
625   for (const auto &Config : AnOpts.Config) {
626 
627     StringRef SuppliedCheckerOrPackage;
628     StringRef SuppliedOption;
629     std::tie(SuppliedCheckerOrPackage, SuppliedOption) =
630         Config.getKey().split(':');
631 
632     if (SuppliedOption.empty())
633       continue;
634 
635     // AnalyzerOptions' config table contains the user input, so an entry could
636     // look like this:
637     //
638     //   cor:NoFalsePositives=true
639     //
640     // Since lower_bound would look for the first element *not less* than "cor",
641     // it would return with an iterator to the first checker in the core, so we
642     // we really have to use find here, which uses operator==.
643     auto CheckerIt =
644         llvm::find(Checkers, CheckerInfo(SuppliedCheckerOrPackage));
645     if (CheckerIt != Checkers.end()) {
646       isOptionContainedIn(CheckerIt->CmdLineOptions, SuppliedCheckerOrPackage,
647                           SuppliedOption, AnOpts, Diags);
648       continue;
649     }
650 
651     const auto *PackageIt =
652         llvm::find(Packages, PackageInfo(SuppliedCheckerOrPackage));
653     if (PackageIt != Packages.end()) {
654       isOptionContainedIn(PackageIt->CmdLineOptions, SuppliedCheckerOrPackage,
655                           SuppliedOption, AnOpts, Diags);
656       continue;
657     }
658 
659     Diags.Report(diag::err_unknown_analyzer_checker_or_package)
660         << SuppliedCheckerOrPackage;
661   }
662 }
663 
664 //===----------------------------------------------------------------------===//
665 // Printing functions.
666 //===----------------------------------------------------------------------===//
667 
668 void CheckerRegistry::printCheckerWithDescList(raw_ostream &Out,
669                                                size_t MaxNameChars) const {
670   // FIXME: Print available packages.
671 
672   Out << "CHECKERS:\n";
673 
674   // Find the maximum option length.
675   size_t OptionFieldWidth = 0;
676   for (const auto &Checker : Checkers) {
677     // Limit the amount of padding we are willing to give up for alignment.
678     //   Package.Name     Description  [Hidden]
679     size_t NameLength = Checker.FullName.size();
680     if (NameLength <= MaxNameChars)
681       OptionFieldWidth = std::max(OptionFieldWidth, NameLength);
682   }
683 
684   const size_t InitialPad = 2;
685 
686   auto Print = [=](llvm::raw_ostream &Out, const CheckerInfo &Checker,
687                    StringRef Description) {
688     AnalyzerOptions::printFormattedEntry(Out, {Checker.FullName, Description},
689                                          InitialPad, OptionFieldWidth);
690     Out << '\n';
691   };
692 
693   for (const auto &Checker : Checkers) {
694     // The order of this if branches is significant, we wouldn't like to display
695     // developer checkers even in the alpha output. For example,
696     // alpha.cplusplus.IteratorModeling is a modeling checker, hence it's hidden
697     // by default, and users (even when the user is a developer of an alpha
698     // checker) shouldn't normally tinker with whether they should be enabled.
699 
700     if (Checker.IsHidden) {
701       if (AnOpts.ShowCheckerHelpDeveloper)
702         Print(Out, Checker, Checker.Desc);
703       continue;
704     }
705 
706     if (Checker.FullName.startswith("alpha")) {
707       if (AnOpts.ShowCheckerHelpAlpha)
708         Print(Out, Checker,
709               ("(Enable only for development!) " + Checker.Desc).str());
710       continue;
711     }
712 
713     if (AnOpts.ShowCheckerHelp)
714         Print(Out, Checker, Checker.Desc);
715   }
716 }
717 
718 void CheckerRegistry::printEnabledCheckerList(raw_ostream &Out) const {
719   for (const auto *i : EnabledCheckers)
720     Out << i->FullName << '\n';
721 }
722 
723 void CheckerRegistry::printCheckerOptionList(raw_ostream &Out) const {
724   Out << "OVERVIEW: Clang Static Analyzer Checker and Package Option List\n\n";
725   Out << "USAGE: -analyzer-config <OPTION1=VALUE,OPTION2=VALUE,...>\n\n";
726   Out << "       -analyzer-config OPTION1=VALUE, -analyzer-config "
727          "OPTION2=VALUE, ...\n\n";
728   Out << "OPTIONS:\n\n";
729 
730   std::multimap<StringRef, const CmdLineOption &> OptionMap;
731 
732   for (const CheckerInfo &Checker : Checkers) {
733     for (const CmdLineOption &Option : Checker.CmdLineOptions) {
734       OptionMap.insert({Checker.FullName, Option});
735     }
736   }
737 
738   for (const PackageInfo &Package : Packages) {
739     for (const CmdLineOption &Option : Package.CmdLineOptions) {
740       OptionMap.insert({Package.FullName, Option});
741     }
742   }
743 
744   auto Print = [] (llvm::raw_ostream &Out, StringRef FullOption, StringRef Desc) {
745     AnalyzerOptions::printFormattedEntry(Out, {FullOption, Desc},
746                                          /*InitialPad*/ 2,
747                                          /*EntryWidth*/ 50,
748                                          /*MinLineWidth*/ 90);
749     Out << "\n\n";
750   };
751   for (const std::pair<const StringRef, const CmdLineOption &> &Entry :
752        OptionMap) {
753     const CmdLineOption &Option = Entry.second;
754     std::string FullOption = (Entry.first + ":" + Option.OptionName).str();
755 
756     std::string Desc =
757         ("(" + Option.OptionType + ") " + Option.Description + " (default: " +
758          (Option.DefaultValStr.empty() ? "\"\"" : Option.DefaultValStr) + ")")
759             .str();
760 
761     // The list of these if branches is significant, we wouldn't like to
762     // display hidden alpha checker options for
763     // -analyzer-checker-option-help-alpha.
764 
765     if (Option.IsHidden) {
766       if (AnOpts.ShowCheckerOptionDeveloperList)
767         Print(Out, FullOption, Desc);
768       continue;
769     }
770 
771     if (Option.DevelopmentStatus == "alpha" ||
772         Entry.first.startswith("alpha")) {
773       if (AnOpts.ShowCheckerOptionAlphaList)
774         Print(Out, FullOption,
775               llvm::Twine("(Enable only for development!) " + Desc).str());
776       continue;
777     }
778 
779     if (AnOpts.ShowCheckerOptionList)
780       Print(Out, FullOption, Desc);
781   }
782 }
783