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