You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			224 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			224 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===--- NonConstParameterCheck.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 "NonConstParameterCheck.h"
 | ||
|  | #include "clang/AST/ASTContext.h"
 | ||
|  | #include "clang/ASTMatchers/ASTMatchFinder.h"
 | ||
|  | 
 | ||
|  | using namespace clang::ast_matchers; | ||
|  | 
 | ||
|  | namespace clang { | ||
|  | namespace tidy { | ||
|  | namespace readability { | ||
|  | 
 | ||
|  | void NonConstParameterCheck::registerMatchers(MatchFinder *Finder) { | ||
|  |   // Add parameters to Parameters.
 | ||
|  |   Finder->addMatcher(parmVarDecl(unless(isInstantiated())).bind("Parm"), this); | ||
|  | 
 | ||
|  |   // C++ constructor.
 | ||
|  |   Finder->addMatcher(cxxConstructorDecl().bind("Ctor"), this); | ||
|  | 
 | ||
|  |   // Track unused parameters, there is Wunused-parameter about unused
 | ||
|  |   // parameters.
 | ||
|  |   Finder->addMatcher(declRefExpr().bind("Ref"), this); | ||
|  | 
 | ||
|  |   // Analyse parameter usage in function.
 | ||
|  |   Finder->addMatcher(stmt(anyOf(unaryOperator(anyOf(hasOperatorName("++"), | ||
|  |                                                     hasOperatorName("--"))), | ||
|  |                                 binaryOperator(), callExpr(), returnStmt(), | ||
|  |                                 cxxConstructExpr())) | ||
|  |                          .bind("Mark"), | ||
|  |                      this); | ||
|  |   Finder->addMatcher(varDecl(hasInitializer(anything())).bind("Mark"), this); | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::check(const MatchFinder::MatchResult &Result) { | ||
|  |   if (const auto *Parm = Result.Nodes.getNodeAs<ParmVarDecl>("Parm")) { | ||
|  |     if (const DeclContext *D = Parm->getParentFunctionOrMethod()) { | ||
|  |       if (const auto *M = dyn_cast<CXXMethodDecl>(D)) { | ||
|  |         if (M->isVirtual() || M->size_overridden_methods() != 0) | ||
|  |           return; | ||
|  |       } | ||
|  |     } | ||
|  |     addParm(Parm); | ||
|  |   } else if (const auto *Ctor = | ||
|  |                  Result.Nodes.getNodeAs<CXXConstructorDecl>("Ctor")) { | ||
|  |     for (const auto *Parm : Ctor->parameters()) | ||
|  |       addParm(Parm); | ||
|  |     for (const auto *Init : Ctor->inits()) | ||
|  |       markCanNotBeConst(Init->getInit(), true); | ||
|  |   } else if (const auto *Ref = Result.Nodes.getNodeAs<DeclRefExpr>("Ref")) { | ||
|  |     setReferenced(Ref); | ||
|  |   } else if (const auto *S = Result.Nodes.getNodeAs<Stmt>("Mark")) { | ||
|  |     if (const auto *B = dyn_cast<BinaryOperator>(S)) { | ||
|  |       if (B->isAssignmentOp()) | ||
|  |         markCanNotBeConst(B, false); | ||
|  |     } else if (const auto *CE = dyn_cast<CallExpr>(S)) { | ||
|  |       // Typically, if a parameter is const then it is fine to make the data
 | ||
|  |       // const. But sometimes the data is written even though the parameter
 | ||
|  |       // is const. Mark all data passed by address to the function.
 | ||
|  |       for (const auto *Arg : CE->arguments()) { | ||
|  |         markCanNotBeConst(Arg->IgnoreParenCasts(), true); | ||
|  |       } | ||
|  | 
 | ||
|  |       // Data passed by nonconst reference should not be made const.
 | ||
|  |       if (const FunctionDecl *FD = CE->getDirectCallee()) { | ||
|  |         unsigned ArgNr = 0U; | ||
|  |         for (const auto *Par : FD->parameters()) { | ||
|  |           if (ArgNr >= CE->getNumArgs()) | ||
|  |             break; | ||
|  |           const Expr *Arg = CE->getArg(ArgNr++); | ||
|  |           // Is this a non constant reference parameter?
 | ||
|  |           const Type *ParType = Par->getType().getTypePtr(); | ||
|  |           if (!ParType->isReferenceType() || Par->getType().isConstQualified()) | ||
|  |             continue; | ||
|  |           markCanNotBeConst(Arg->IgnoreParenCasts(), false); | ||
|  |         } | ||
|  |       } | ||
|  |     } else if (const auto *CE = dyn_cast<CXXConstructExpr>(S)) { | ||
|  |       for (const auto *Arg : CE->arguments()) { | ||
|  |         markCanNotBeConst(Arg->IgnoreParenCasts(), true); | ||
|  |       } | ||
|  |     } else if (const auto *R = dyn_cast<ReturnStmt>(S)) { | ||
|  |       markCanNotBeConst(R->getRetValue(), true); | ||
|  |     } else if (const auto *U = dyn_cast<UnaryOperator>(S)) { | ||
|  |       markCanNotBeConst(U, true); | ||
|  |     } | ||
|  |   } else if (const auto *VD = Result.Nodes.getNodeAs<VarDecl>("Mark")) { | ||
|  |     const QualType T = VD->getType(); | ||
|  |     if ((T->isPointerType() && !T->getPointeeType().isConstQualified()) || | ||
|  |         T->isArrayType()) | ||
|  |       markCanNotBeConst(VD->getInit(), true); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::addParm(const ParmVarDecl *Parm) { | ||
|  |   // Only add nonconst integer/float pointer parameters.
 | ||
|  |   const QualType T = Parm->getType(); | ||
|  |   if (!T->isPointerType() || T->getPointeeType().isConstQualified() || | ||
|  |       !(T->getPointeeType()->isIntegerType() || | ||
|  |         T->getPointeeType()->isFloatingType())) | ||
|  |     return; | ||
|  | 
 | ||
|  |   if (Parameters.find(Parm) != Parameters.end()) | ||
|  |     return; | ||
|  | 
 | ||
|  |   ParmInfo PI; | ||
|  |   PI.IsReferenced = false; | ||
|  |   PI.CanBeConst = true; | ||
|  |   Parameters[Parm] = PI; | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::setReferenced(const DeclRefExpr *Ref) { | ||
|  |   auto It = Parameters.find(dyn_cast<ParmVarDecl>(Ref->getDecl())); | ||
|  |   if (It != Parameters.end()) | ||
|  |     It->second.IsReferenced = true; | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::onEndOfTranslationUnit() { | ||
|  |   diagnoseNonConstParameters(); | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::diagnoseNonConstParameters() { | ||
|  |   for (const auto &It : Parameters) { | ||
|  |     const ParmVarDecl *Par = It.first; | ||
|  |     const ParmInfo &ParamInfo = It.second; | ||
|  | 
 | ||
|  |     // Unused parameter => there are other warnings about this.
 | ||
|  |     if (!ParamInfo.IsReferenced) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Parameter can't be const.
 | ||
|  |     if (!ParamInfo.CanBeConst) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     SmallVector<FixItHint, 8> Fixes; | ||
|  |     auto *Function = | ||
|  |         dyn_cast_or_null<const FunctionDecl>(Par->getParentFunctionOrMethod()); | ||
|  |     if (!Function) | ||
|  |       continue; | ||
|  |     unsigned Index = Par->getFunctionScopeIndex(); | ||
|  |     for (FunctionDecl *FnDecl : Function->redecls()) | ||
|  |       Fixes.push_back(FixItHint::CreateInsertion( | ||
|  |           FnDecl->getParamDecl(Index)->getLocStart(), "const ")); | ||
|  | 
 | ||
|  |     diag(Par->getLocation(), "pointer parameter '%0' can be pointer to const") | ||
|  |         << Par->getName() << Fixes; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void NonConstParameterCheck::markCanNotBeConst(const Expr *E, | ||
|  |                                                bool CanNotBeConst) { | ||
|  |   if (!E) | ||
|  |     return; | ||
|  | 
 | ||
|  |   if (const auto *Cast = dyn_cast<ImplicitCastExpr>(E)) { | ||
|  |     // If expression is const then ignore usage.
 | ||
|  |     const QualType T = Cast->getType(); | ||
|  |     if (T->isPointerType() && T->getPointeeType().isConstQualified()) | ||
|  |       return; | ||
|  |   } | ||
|  | 
 | ||
|  |   E = E->IgnoreParenCasts(); | ||
|  | 
 | ||
|  |   if (const auto *B = dyn_cast<BinaryOperator>(E)) { | ||
|  |     if (B->isAdditiveOp()) { | ||
|  |       // p + 2
 | ||
|  |       markCanNotBeConst(B->getLHS(), CanNotBeConst); | ||
|  |       markCanNotBeConst(B->getRHS(), CanNotBeConst); | ||
|  |     } else if (B->isAssignmentOp()) { | ||
|  |       markCanNotBeConst(B->getLHS(), false); | ||
|  | 
 | ||
|  |       // If LHS is not const then RHS can't be const.
 | ||
|  |       const QualType T = B->getLHS()->getType(); | ||
|  |       if (T->isPointerType() && !T->getPointeeType().isConstQualified()) | ||
|  |         markCanNotBeConst(B->getRHS(), true); | ||
|  |     } | ||
|  |   } else if (const auto *C = dyn_cast<ConditionalOperator>(E)) { | ||
|  |     markCanNotBeConst(C->getTrueExpr(), CanNotBeConst); | ||
|  |     markCanNotBeConst(C->getFalseExpr(), CanNotBeConst); | ||
|  |   } else if (const auto *U = dyn_cast<UnaryOperator>(E)) { | ||
|  |     if (U->getOpcode() == UO_PreInc || U->getOpcode() == UO_PreDec || | ||
|  |         U->getOpcode() == UO_PostInc || U->getOpcode() == UO_PostDec) { | ||
|  |       if (const auto *SubU = | ||
|  |               dyn_cast<UnaryOperator>(U->getSubExpr()->IgnoreParenCasts())) | ||
|  |         markCanNotBeConst(SubU->getSubExpr(), true); | ||
|  |       markCanNotBeConst(U->getSubExpr(), CanNotBeConst); | ||
|  |     } else if (U->getOpcode() == UO_Deref) { | ||
|  |       if (!CanNotBeConst) | ||
|  |         markCanNotBeConst(U->getSubExpr(), true); | ||
|  |     } else { | ||
|  |       markCanNotBeConst(U->getSubExpr(), CanNotBeConst); | ||
|  |     } | ||
|  |   } else if (const auto *A = dyn_cast<ArraySubscriptExpr>(E)) { | ||
|  |     markCanNotBeConst(A->getBase(), true); | ||
|  |   } else if (const auto *CLE = dyn_cast<CompoundLiteralExpr>(E)) { | ||
|  |     markCanNotBeConst(CLE->getInitializer(), true); | ||
|  |   } else if (const auto *Constr = dyn_cast<CXXConstructExpr>(E)) { | ||
|  |     for (const auto *Arg : Constr->arguments()) { | ||
|  |       if (const auto *M = dyn_cast<MaterializeTemporaryExpr>(Arg)) | ||
|  |         markCanNotBeConst(cast<Expr>(M->getTemporary()), CanNotBeConst); | ||
|  |     } | ||
|  |   } else if (const auto *ILE = dyn_cast<InitListExpr>(E)) { | ||
|  |     for (unsigned I = 0U; I < ILE->getNumInits(); ++I) | ||
|  |       markCanNotBeConst(ILE->getInit(I), true); | ||
|  |   } else if (CanNotBeConst) { | ||
|  |     // Referencing parameter.
 | ||
|  |     if (const auto *D = dyn_cast<DeclRefExpr>(E)) { | ||
|  |       auto It = Parameters.find(dyn_cast<ParmVarDecl>(D->getDecl())); | ||
|  |       if (It != Parameters.end()) | ||
|  |         It->second.CanBeConst = false; | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace readability
 | ||
|  | } // namespace tidy
 | ||
|  | } // namespace clang
 |