You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			60 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			60 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===--- InterfacesGlobalInitCheck.cpp - clang-tidy------------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "InterfacesGlobalInitCheck.h"
 | |
| #include "clang/AST/ASTContext.h"
 | |
| #include "clang/ASTMatchers/ASTMatchFinder.h"
 | |
| 
 | |
| using namespace clang::ast_matchers;
 | |
| 
 | |
| namespace clang {
 | |
| namespace tidy {
 | |
| namespace cppcoreguidelines {
 | |
| 
 | |
| void InterfacesGlobalInitCheck::registerMatchers(MatchFinder *Finder) {
 | |
|   const auto IsGlobal =
 | |
|       allOf(hasGlobalStorage(),
 | |
|             hasDeclContext(anyOf(translationUnitDecl(), // Global scope.
 | |
|                                  namespaceDecl(),       // Namespace scope.
 | |
|                                  recordDecl())),        // Class scope.
 | |
|             unless(isConstexpr()));
 | |
| 
 | |
|   const auto ReferencesUndefinedGlobalVar = declRefExpr(hasDeclaration(
 | |
|       varDecl(IsGlobal, unless(isDefinition())).bind("referencee")));
 | |
| 
 | |
|   Finder->addMatcher(
 | |
|       varDecl(IsGlobal, isDefinition(),
 | |
|               hasInitializer(expr(hasDescendant(ReferencesUndefinedGlobalVar))))
 | |
|           .bind("var"),
 | |
|       this);
 | |
| }
 | |
| 
 | |
| void InterfacesGlobalInitCheck::check(const MatchFinder::MatchResult &Result) {
 | |
|   const auto *const Var = Result.Nodes.getNodeAs<VarDecl>("var");
 | |
|   // For now assume that people who write macros know what they're doing.
 | |
|   if (Var->getLocation().isMacroID())
 | |
|     return;
 | |
|   const auto *const Referencee = Result.Nodes.getNodeAs<VarDecl>("referencee");
 | |
|   // If the variable has been defined, we're good.
 | |
|   const auto *const ReferenceeDef = Referencee->getDefinition();
 | |
|   if (ReferenceeDef != nullptr &&
 | |
|       Result.SourceManager->isBeforeInTranslationUnit(
 | |
|           ReferenceeDef->getLocation(), Var->getLocation())) {
 | |
|     return;
 | |
|   }
 | |
|   diag(Var->getLocation(),
 | |
|        "initializing non-local variable with non-const expression depending on "
 | |
|        "uninitialized non-local variable %0")
 | |
|       << Referencee;
 | |
| }
 | |
| 
 | |
| } // namespace cppcoreguidelines
 | |
| } // namespace tidy
 | |
| } // namespace clang
 |