123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===- CheckerRegistry.h - Maintains all available checkers -----*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- //
- // Contains the logic for parsing the TableGen file Checkers.td, and parsing the
- // specific invocation of the analyzer (which checker/package is enabled, values
- // of their options, etc). This is in the frontend library because checker
- // registry functions are called from here but are defined in the dependent
- // library libStaticAnalyzerCheckers, but the actual data structure that holds
- // the parsed information is in the Core library.
- //
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_CLANG_STATICANALYZER_FRONTEND_CHECKERREGISTRY_H
- #define LLVM_CLANG_STATICANALYZER_FRONTEND_CHECKERREGISTRY_H
- #include "clang/Basic/LLVM.h"
- #include "clang/StaticAnalyzer/Core/CheckerRegistryData.h"
- #include "llvm/ADT/StringRef.h"
- // FIXME: move this information to an HTML file in docs/.
- // At the very least, a checker plugin is a dynamic library that exports
- // clang_analyzerAPIVersionString. This should be defined as follows:
- //
- // extern "C"
- // const char clang_analyzerAPIVersionString[] =
- // CLANG_ANALYZER_API_VERSION_STRING;
- //
- // This is used to check whether the current version of the analyzer is known to
- // be incompatible with a plugin. Plugins with incompatible version strings,
- // or without a version string at all, will not be loaded.
- //
- // To add a custom checker to the analyzer, the plugin must also define the
- // function clang_registerCheckers. For example:
- //
- // extern "C"
- // void clang_registerCheckers (CheckerRegistry ®istry) {
- // registry.addChecker<MainCallChecker>("example.MainCallChecker",
- // "Disallows calls to functions called main");
- // }
- //
- // The first method argument is the full name of the checker, including its
- // enclosing package. By convention, the registered name of a checker is the
- // name of the associated class (the template argument).
- // The second method argument is a short human-readable description of the
- // checker.
- //
- // The clang_registerCheckers function may add any number of checkers to the
- // registry. If any checkers require additional initialization, use the three-
- // argument form of CheckerRegistry::addChecker.
- //
- // To load a checker plugin, specify the full path to the dynamic library as
- // the argument to the -load option in the cc1 frontend. You can then enable
- // your custom checker using the -analyzer-checker:
- //
- // clang -cc1 -load </path/to/plugin.dylib> -analyze
- // -analyzer-checker=<example.MainCallChecker>
- //
- // For a complete working example, see examples/analyzer-plugin.
- #ifndef CLANG_ANALYZER_API_VERSION_STRING
- // FIXME: The Clang version string is not particularly granular;
- // the analyzer infrastructure can change a lot between releases.
- // Unfortunately, this string has to be statically embedded in each plugin,
- // so we can't just use the functions defined in Version.h.
- #include "clang/Basic/Version.h"
- #define CLANG_ANALYZER_API_VERSION_STRING CLANG_VERSION_STRING
- #endif
- namespace clang {
- class AnalyzerOptions;
- class DiagnosticsEngine;
- namespace ento {
- class CheckerManager;
- /// Manages a set of available checkers for running a static analysis.
- /// The checkers are organized into packages by full name, where including
- /// a package will recursively include all subpackages and checkers within it.
- /// For example, the checker "core.builtin.NoReturnFunctionChecker" will be
- /// included if initializeManager() is called with an option of "core",
- /// "core.builtin", or the full name "core.builtin.NoReturnFunctionChecker".
- class CheckerRegistry {
- public:
- CheckerRegistry(CheckerRegistryData &Data, ArrayRef<std::string> Plugins,
- DiagnosticsEngine &Diags, AnalyzerOptions &AnOpts,
- ArrayRef<std::function<void(CheckerRegistry &)>>
- CheckerRegistrationFns = {});
- /// Collects all enabled checkers in the field EnabledCheckers. It preserves
- /// the order of insertion, as dependencies have to be enabled before the
- /// checkers that depend on them.
- void initializeRegistry(const CheckerManager &Mgr);
- private:
- /// Default initialization function for checkers -- since CheckerManager
- /// includes this header, we need to make it a template parameter, and since
- /// the checker must be a template parameter as well, we can't put this in the
- /// cpp file.
- template <typename MGR, typename T> static void initializeManager(MGR &mgr) {
- mgr.template registerChecker<T>();
- }
- template <typename T> static bool returnTrue(const CheckerManager &mgr) {
- return true;
- }
- public:
- /// Adds a checker to the registry. Use this non-templated overload when your
- /// checker requires custom initialization.
- void addChecker(RegisterCheckerFn Fn, ShouldRegisterFunction sfn,
- StringRef FullName, StringRef Desc, StringRef DocsUri,
- bool IsHidden);
- /// Adds a checker to the registry. Use this templated overload when your
- /// checker does not require any custom initialization.
- /// This function isn't really needed and probably causes more headaches than
- /// the tiny convenience that it provides, but external plugins might use it,
- /// and there isn't a strong incentive to remove it.
- template <class T>
- void addChecker(StringRef FullName, StringRef Desc, StringRef DocsUri,
- bool IsHidden = false) {
- // Avoid MSVC's Compiler Error C2276:
- // http://msdn.microsoft.com/en-us/library/850cstw1(v=VS.80).aspx
- addChecker(&CheckerRegistry::initializeManager<CheckerManager, T>,
- &CheckerRegistry::returnTrue<T>, FullName, Desc, DocsUri,
- IsHidden);
- }
- /// Makes the checker with the full name \p fullName depend on the checker
- /// called \p dependency.
- void addDependency(StringRef FullName, StringRef Dependency);
- /// Makes the checker with the full name \p fullName weak depend on the
- /// checker called \p dependency.
- void addWeakDependency(StringRef FullName, StringRef Dependency);
- /// Registers an option to a given checker. A checker option will always have
- /// the following format:
- /// CheckerFullName:OptionName=Value
- /// And can be specified from the command line like this:
- /// -analyzer-config CheckerFullName:OptionName=Value
- ///
- /// Options for unknown checkers, or unknown options for a given checker, or
- /// invalid value types for that given option are reported as an error in
- /// non-compatibility mode.
- void addCheckerOption(StringRef OptionType, StringRef CheckerFullName,
- StringRef OptionName, StringRef DefaultValStr,
- StringRef Description, StringRef DevelopmentStatus,
- bool IsHidden = false);
- /// Adds a package to the registry.
- void addPackage(StringRef FullName);
- /// Registers an option to a given package. A package option will always have
- /// the following format:
- /// PackageFullName:OptionName=Value
- /// And can be specified from the command line like this:
- /// -analyzer-config PackageFullName:OptionName=Value
- ///
- /// Options for unknown packages, or unknown options for a given package, or
- /// invalid value types for that given option are reported as an error in
- /// non-compatibility mode.
- void addPackageOption(StringRef OptionType, StringRef PackageFullName,
- StringRef OptionName, StringRef DefaultValStr,
- StringRef Description, StringRef DevelopmentStatus,
- bool IsHidden = false);
- // FIXME: This *really* should be added to the frontend flag descriptions.
- /// Initializes a CheckerManager by calling the initialization functions for
- /// all checkers specified by the given CheckerOptInfo list. The order of this
- /// list is significant; later options can be used to reverse earlier ones.
- /// This can be used to exclude certain checkers in an included package.
- void initializeManager(CheckerManager &CheckerMgr) const;
- /// Check if every option corresponds to a specific checker or package.
- void validateCheckerOptions() const;
- private:
- template <bool IsWeak> void resolveDependencies();
- void resolveCheckerAndPackageOptions();
- CheckerRegistryData &Data;
- DiagnosticsEngine &Diags;
- AnalyzerOptions &AnOpts;
- };
- } // namespace ento
- } // namespace clang
- #endif // LLVM_CLANG_STATICANALYZER_FRONTEND_CHECKERREGISTRY_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|