//===- unittests/StaticAnalyzer/RegisterCustomCheckersTest.cpp ------------===// // // 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 // //===----------------------------------------------------------------------===// #include "CheckerRegistration.h" #include "clang/Frontend/CompilerInstance.h" #include "clang/StaticAnalyzer/Core/BugReporter/BugReporter.h" #include "clang/StaticAnalyzer/Core/BugReporter/BugType.h" #include "clang/StaticAnalyzer/Core/Checker.h" #include "clang/StaticAnalyzer/Core/PathSensitive/AnalysisManager.h" #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h" #include "clang/StaticAnalyzer/Frontend/AnalysisConsumer.h" #include "clang/StaticAnalyzer/Frontend/CheckerRegistry.h" #include "clang/Tooling/Tooling.h" #include "gtest/gtest.h" namespace clang { namespace ento { namespace { //===----------------------------------------------------------------------===// // Just a minimal test for how checker registration works with statically // linked, non TableGen generated checkers. //===----------------------------------------------------------------------===// class CustomChecker : public Checker { public: void checkASTCodeBody(const Decl *D, AnalysisManager &Mgr, BugReporter &BR) const { BR.EmitBasicReport(D, this, "Custom diagnostic", categories::LogicError, "Custom diagnostic description", PathDiagnosticLocation(D, Mgr.getSourceManager()), {}); } }; void addCustomChecker(AnalysisASTConsumer &AnalysisConsumer, AnalyzerOptions &AnOpts) { AnOpts.CheckersAndPackages = {{"custom.CustomChecker", true}}; AnalysisConsumer.AddCheckerRegistrationFn([](CheckerRegistry &Registry) { Registry.addChecker("custom.CustomChecker", "Description", ""); }); } TEST(RegisterCustomCheckers, RegisterChecker) { std::string Diags; EXPECT_TRUE(runCheckerOnCode("void f() {;}", Diags)); EXPECT_EQ(Diags, "custom.CustomChecker:Custom diagnostic description\n"); } //===----------------------------------------------------------------------===// // Pretty much the same. //===----------------------------------------------------------------------===// class LocIncDecChecker : public Checker { public: void checkLocation(SVal Loc, bool IsLoad, const Stmt *S, CheckerContext &C) const { auto UnaryOp = dyn_cast(S); if (UnaryOp && !IsLoad) { EXPECT_FALSE(UnaryOp->isIncrementOp()); } } }; void addLocIncDecChecker(AnalysisASTConsumer &AnalysisConsumer, AnalyzerOptions &AnOpts) { AnOpts.CheckersAndPackages = {{"test.LocIncDecChecker", true}}; AnalysisConsumer.AddCheckerRegistrationFn([](CheckerRegistry &Registry) { Registry.addChecker("test.LocIncDecChecker", "Description", ""); }); } TEST(RegisterCustomCheckers, CheckLocationIncDec) { EXPECT_TRUE( runCheckerOnCode("void f() { int *p; (*p)++; }")); } //===----------------------------------------------------------------------===// // Unsatisfied checker dependency //===----------------------------------------------------------------------===// class PrerequisiteChecker : public Checker { public: void checkASTCodeBody(const Decl *D, AnalysisManager &Mgr, BugReporter &BR) const { BR.EmitBasicReport(D, this, "Prerequisite", categories::LogicError, "This is the prerequisite checker", PathDiagnosticLocation(D, Mgr.getSourceManager()), {}); } }; void registerPrerequisiteChecker(CheckerManager &mgr) { mgr.registerChecker(); } bool shouldRegisterPrerequisiteChecker(const CheckerManager &mgr) { return false; } class DependentChecker : public Checker { public: void checkASTCodeBody(const Decl *D, AnalysisManager &Mgr, BugReporter &BR) const { BR.EmitBasicReport(D, this, "Dependent", categories::LogicError, "This is the Dependent Checker", PathDiagnosticLocation(D, Mgr.getSourceManager()), {}); } }; void registerDependentChecker(CheckerManager &mgr) { mgr.registerChecker(); } bool shouldRegisterDependentChecker(const CheckerManager &mgr) { return true; } void addDependentChecker(AnalysisASTConsumer &AnalysisConsumer, AnalyzerOptions &AnOpts) { AnOpts.CheckersAndPackages = {{"custom.Dependent", true}}; AnalysisConsumer.AddCheckerRegistrationFn([](CheckerRegistry &Registry) { Registry.addChecker(registerPrerequisiteChecker, shouldRegisterPrerequisiteChecker, "custom.Prerequisite", "Description", "", false); Registry.addChecker(registerDependentChecker, shouldRegisterDependentChecker, "custom.Dependent", "Description", "", false); Registry.addDependency("custom.Dependent", "custom.Prerequisite"); }); } TEST(RegisterDependentCheckers, RegisterChecker) { std::string Diags; EXPECT_TRUE(runCheckerOnCode("void f() {;}", Diags)); EXPECT_EQ(Diags, ""); } } // namespace } // namespace ento } // namespace clang