You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			674 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			674 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===--- SimplifyBooleanExpr.cpp clang-tidy ---------------------*- C++ -*-===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "SimplifyBooleanExprCheck.h"
 | |
| #include "clang/AST/RecursiveASTVisitor.h"
 | |
| #include "clang/Lex/Lexer.h"
 | |
| 
 | |
| #include <cassert>
 | |
| #include <string>
 | |
| #include <utility>
 | |
| 
 | |
| using namespace clang::ast_matchers;
 | |
| 
 | |
| namespace clang {
 | |
| namespace tidy {
 | |
| namespace readability {
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| StringRef getText(const MatchFinder::MatchResult &Result, SourceRange Range) {
 | |
|   return Lexer::getSourceText(CharSourceRange::getTokenRange(Range),
 | |
|                               *Result.SourceManager,
 | |
|                               Result.Context->getLangOpts());
 | |
| }
 | |
| 
 | |
| template <typename T>
 | |
| StringRef getText(const MatchFinder::MatchResult &Result, T &Node) {
 | |
|   return getText(Result, Node.getSourceRange());
 | |
| }
 | |
| 
 | |
| const char ConditionThenStmtId[] = "if-bool-yields-then";
 | |
| const char ConditionElseStmtId[] = "if-bool-yields-else";
 | |
| const char TernaryId[] = "ternary-bool-yields-condition";
 | |
| const char TernaryNegatedId[] = "ternary-bool-yields-not-condition";
 | |
| const char IfReturnsBoolId[] = "if-return";
 | |
| const char IfReturnsNotBoolId[] = "if-not-return";
 | |
| const char ThenLiteralId[] = "then-literal";
 | |
| const char IfAssignVariableId[] = "if-assign-lvalue";
 | |
| const char IfAssignLocId[] = "if-assign-loc";
 | |
| const char IfAssignBoolId[] = "if-assign";
 | |
| const char IfAssignNotBoolId[] = "if-assign-not";
 | |
| const char IfAssignObjId[] = "if-assign-obj";
 | |
| const char CompoundReturnId[] = "compound-return";
 | |
| const char CompoundBoolId[] = "compound-bool";
 | |
| const char CompoundNotBoolId[] = "compound-bool-not";
 | |
| 
 | |
| const char IfStmtId[] = "if";
 | |
| 
 | |
| const char SimplifyOperatorDiagnostic[] =
 | |
|     "redundant boolean literal supplied to boolean operator";
 | |
| const char SimplifyConditionDiagnostic[] =
 | |
|     "redundant boolean literal in if statement condition";
 | |
| const char SimplifyConditionalReturnDiagnostic[] =
 | |
|     "redundant boolean literal in conditional return statement";
 | |
| 
 | |
| const CXXBoolLiteralExpr *getBoolLiteral(const MatchFinder::MatchResult &Result,
 | |
|                                          StringRef Id) {
 | |
|   const auto *Literal = Result.Nodes.getNodeAs<CXXBoolLiteralExpr>(Id);
 | |
|   return (Literal && Literal->getLocStart().isMacroID()) ? nullptr : Literal;
 | |
| }
 | |
| 
 | |
| internal::Matcher<Stmt> returnsBool(bool Value, StringRef Id = "ignored") {
 | |
|   auto SimpleReturnsBool =
 | |
|       returnStmt(has(cxxBoolLiteral(equals(Value)).bind(Id)))
 | |
|           .bind("returns-bool");
 | |
|   return anyOf(SimpleReturnsBool,
 | |
|                compoundStmt(statementCountIs(1), has(SimpleReturnsBool)));
 | |
| }
 | |
| 
 | |
| bool needsParensAfterUnaryNegation(const Expr *E) {
 | |
|   E = E->IgnoreImpCasts();
 | |
|   if (isa<BinaryOperator>(E) || isa<ConditionalOperator>(E))
 | |
|     return true;
 | |
| 
 | |
|   if (const auto *Op = dyn_cast<CXXOperatorCallExpr>(E))
 | |
|     return Op->getNumArgs() == 2 && Op->getOperator() != OO_Call &&
 | |
|            Op->getOperator() != OO_Subscript;
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| std::pair<BinaryOperatorKind, BinaryOperatorKind> Opposites[] = {
 | |
|     {BO_LT, BO_GE}, {BO_GT, BO_LE}, {BO_EQ, BO_NE}};
 | |
| 
 | |
| StringRef negatedOperator(const BinaryOperator *BinOp) {
 | |
|   const BinaryOperatorKind Opcode = BinOp->getOpcode();
 | |
|   for (auto NegatableOp : Opposites) {
 | |
|     if (Opcode == NegatableOp.first)
 | |
|       return BinOp->getOpcodeStr(NegatableOp.second);
 | |
|     if (Opcode == NegatableOp.second)
 | |
|       return BinOp->getOpcodeStr(NegatableOp.first);
 | |
|   }
 | |
|   return StringRef();
 | |
| }
 | |
| 
 | |
| std::pair<OverloadedOperatorKind, StringRef> OperatorNames[] = {
 | |
|     {OO_EqualEqual, "=="},   {OO_ExclaimEqual, "!="}, {OO_Less, "<"},
 | |
|     {OO_GreaterEqual, ">="}, {OO_Greater, ">"},       {OO_LessEqual, "<="}};
 | |
| 
 | |
| StringRef getOperatorName(OverloadedOperatorKind OpKind) {
 | |
|   for (auto Name : OperatorNames) {
 | |
|     if (Name.first == OpKind)
 | |
|       return Name.second;
 | |
|   }
 | |
| 
 | |
|   return StringRef();
 | |
| }
 | |
| 
 | |
| std::pair<OverloadedOperatorKind, OverloadedOperatorKind> OppositeOverloads[] =
 | |
|     {{OO_EqualEqual, OO_ExclaimEqual},
 | |
|      {OO_Less, OO_GreaterEqual},
 | |
|      {OO_Greater, OO_LessEqual}};
 | |
| 
 | |
| StringRef negatedOperator(const CXXOperatorCallExpr *OpCall) {
 | |
|   const OverloadedOperatorKind Opcode = OpCall->getOperator();
 | |
|   for (auto NegatableOp : OppositeOverloads) {
 | |
|     if (Opcode == NegatableOp.first)
 | |
|       return getOperatorName(NegatableOp.second);
 | |
|     if (Opcode == NegatableOp.second)
 | |
|       return getOperatorName(NegatableOp.first);
 | |
|   }
 | |
|   return StringRef();
 | |
| }
 | |
| 
 | |
| std::string asBool(StringRef text, bool NeedsStaticCast) {
 | |
|   if (NeedsStaticCast)
 | |
|     return ("static_cast<bool>(" + text + ")").str();
 | |
| 
 | |
|   return text;
 | |
| }
 | |
| 
 | |
| bool needsNullPtrComparison(const Expr *E) {
 | |
|   if (const auto *ImpCast = dyn_cast<ImplicitCastExpr>(E))
 | |
|     return ImpCast->getCastKind() == CK_PointerToBoolean ||
 | |
|            ImpCast->getCastKind() == CK_MemberPointerToBoolean;
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool needsZeroComparison(const Expr *E) {
 | |
|   if (const auto *ImpCast = dyn_cast<ImplicitCastExpr>(E))
 | |
|     return ImpCast->getCastKind() == CK_IntegralToBoolean;
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool needsStaticCast(const Expr *E) {
 | |
|   if (const auto *ImpCast = dyn_cast<ImplicitCastExpr>(E)) {
 | |
|     if (ImpCast->getCastKind() == CK_UserDefinedConversion &&
 | |
|         ImpCast->getSubExpr()->getType()->isBooleanType()) {
 | |
|       if (const auto *MemCall =
 | |
|               dyn_cast<CXXMemberCallExpr>(ImpCast->getSubExpr())) {
 | |
|         if (const auto *MemDecl =
 | |
|                 dyn_cast<CXXConversionDecl>(MemCall->getMethodDecl())) {
 | |
|           if (MemDecl->isExplicit())
 | |
|             return true;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   E = E->IgnoreImpCasts();
 | |
|   return !E->getType()->isBooleanType();
 | |
| }
 | |
| 
 | |
| std::string compareExpressionToConstant(const MatchFinder::MatchResult &Result,
 | |
|                                         const Expr *E, bool Negated,
 | |
|                                         const char *Constant) {
 | |
|   E = E->IgnoreImpCasts();
 | |
|   const std::string ExprText =
 | |
|       (isa<BinaryOperator>(E) ? ("(" + getText(Result, *E) + ")")
 | |
|                               : getText(Result, *E))
 | |
|           .str();
 | |
|   return ExprText + " " + (Negated ? "!=" : "==") + " " + Constant;
 | |
| }
 | |
| 
 | |
| std::string compareExpressionToNullPtr(const MatchFinder::MatchResult &Result,
 | |
|                                        const Expr *E, bool Negated) {
 | |
|   const char *NullPtr =
 | |
|       Result.Context->getLangOpts().CPlusPlus11 ? "nullptr" : "NULL";
 | |
|   return compareExpressionToConstant(Result, E, Negated, NullPtr);
 | |
| }
 | |
| 
 | |
| std::string compareExpressionToZero(const MatchFinder::MatchResult &Result,
 | |
|                                     const Expr *E, bool Negated) {
 | |
|   return compareExpressionToConstant(Result, E, Negated, "0");
 | |
| }
 | |
| 
 | |
| std::string replacementExpression(const MatchFinder::MatchResult &Result,
 | |
|                                   bool Negated, const Expr *E) {
 | |
|   E = E->ignoreParenBaseCasts();
 | |
|   const bool NeedsStaticCast = needsStaticCast(E);
 | |
|   if (Negated) {
 | |
|     if (const auto *UnOp = dyn_cast<UnaryOperator>(E)) {
 | |
|       if (UnOp->getOpcode() == UO_LNot) {
 | |
|         if (needsNullPtrComparison(UnOp->getSubExpr()))
 | |
|           return compareExpressionToNullPtr(Result, UnOp->getSubExpr(), true);
 | |
| 
 | |
|         if (needsZeroComparison(UnOp->getSubExpr()))
 | |
|           return compareExpressionToZero(Result, UnOp->getSubExpr(), true);
 | |
| 
 | |
|         return replacementExpression(Result, false, UnOp->getSubExpr());
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (needsNullPtrComparison(E))
 | |
|       return compareExpressionToNullPtr(Result, E, false);
 | |
| 
 | |
|     if (needsZeroComparison(E))
 | |
|       return compareExpressionToZero(Result, E, false);
 | |
| 
 | |
|     StringRef NegatedOperator;
 | |
|     const Expr *LHS = nullptr;
 | |
|     const Expr *RHS = nullptr;
 | |
|     if (const auto *BinOp = dyn_cast<BinaryOperator>(E)) {
 | |
|       NegatedOperator = negatedOperator(BinOp);
 | |
|       LHS = BinOp->getLHS();
 | |
|       RHS = BinOp->getRHS();
 | |
|     } else if (const auto *OpExpr = dyn_cast<CXXOperatorCallExpr>(E)) {
 | |
|       if (OpExpr->getNumArgs() == 2) {
 | |
|         NegatedOperator = negatedOperator(OpExpr);
 | |
|         LHS = OpExpr->getArg(0);
 | |
|         RHS = OpExpr->getArg(1);
 | |
|       }
 | |
|     }
 | |
|     if (!NegatedOperator.empty() && LHS && RHS)
 | |
|       return (asBool((getText(Result, *LHS) + " " + NegatedOperator + " " +
 | |
|                       getText(Result, *RHS))
 | |
|                          .str(),
 | |
|                      NeedsStaticCast));
 | |
| 
 | |
|     StringRef Text = getText(Result, *E);
 | |
|     if (!NeedsStaticCast && needsParensAfterUnaryNegation(E))
 | |
|       return ("!(" + Text + ")").str();
 | |
| 
 | |
|     if (needsNullPtrComparison(E))
 | |
|       return compareExpressionToNullPtr(Result, E, false);
 | |
| 
 | |
|     if (needsZeroComparison(E))
 | |
|       return compareExpressionToZero(Result, E, false);
 | |
| 
 | |
|     return ("!" + asBool(Text, NeedsStaticCast));
 | |
|   }
 | |
| 
 | |
|   if (const auto *UnOp = dyn_cast<UnaryOperator>(E)) {
 | |
|     if (UnOp->getOpcode() == UO_LNot) {
 | |
|       if (needsNullPtrComparison(UnOp->getSubExpr()))
 | |
|         return compareExpressionToNullPtr(Result, UnOp->getSubExpr(), false);
 | |
| 
 | |
|       if (needsZeroComparison(UnOp->getSubExpr()))
 | |
|         return compareExpressionToZero(Result, UnOp->getSubExpr(), false);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (needsNullPtrComparison(E))
 | |
|     return compareExpressionToNullPtr(Result, E, true);
 | |
| 
 | |
|   if (needsZeroComparison(E))
 | |
|     return compareExpressionToZero(Result, E, true);
 | |
| 
 | |
|   return asBool(getText(Result, *E), NeedsStaticCast);
 | |
| }
 | |
| 
 | |
| const CXXBoolLiteralExpr *stmtReturnsBool(const ReturnStmt *Ret, bool Negated) {
 | |
|   if (const auto *Bool = dyn_cast<CXXBoolLiteralExpr>(Ret->getRetValue())) {
 | |
|     if (Bool->getValue() == !Negated)
 | |
|       return Bool;
 | |
|   }
 | |
| 
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| const CXXBoolLiteralExpr *stmtReturnsBool(const IfStmt *IfRet, bool Negated) {
 | |
|   if (IfRet->getElse() != nullptr)
 | |
|     return nullptr;
 | |
| 
 | |
|   if (const auto *Ret = dyn_cast<ReturnStmt>(IfRet->getThen()))
 | |
|     return stmtReturnsBool(Ret, Negated);
 | |
| 
 | |
|   if (const auto *Compound = dyn_cast<CompoundStmt>(IfRet->getThen())) {
 | |
|     if (Compound->size() == 1) {
 | |
|       if (const auto *CompoundRet = dyn_cast<ReturnStmt>(Compound->body_back()))
 | |
|         return stmtReturnsBool(CompoundRet, Negated);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| bool containsDiscardedTokens(const MatchFinder::MatchResult &Result,
 | |
|                              CharSourceRange CharRange) {
 | |
|   std::string ReplacementText =
 | |
|       Lexer::getSourceText(CharRange, *Result.SourceManager,
 | |
|                            Result.Context->getLangOpts())
 | |
|           .str();
 | |
|   Lexer Lex(CharRange.getBegin(), Result.Context->getLangOpts(),
 | |
|             ReplacementText.data(), ReplacementText.data(),
 | |
|             ReplacementText.data() + ReplacementText.size());
 | |
|   Lex.SetCommentRetentionState(true);
 | |
| 
 | |
|   Token Tok;
 | |
|   while (!Lex.LexFromRawLexer(Tok)) {
 | |
|     if (Tok.is(tok::TokenKind::comment) || Tok.is(tok::TokenKind::hash))
 | |
|       return true;
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| class SimplifyBooleanExprCheck::Visitor : public RecursiveASTVisitor<Visitor> {
 | |
|   using Base = RecursiveASTVisitor<Visitor>;
 | |
| 
 | |
|  public:
 | |
|   Visitor(SimplifyBooleanExprCheck *Check,
 | |
|           const MatchFinder::MatchResult &Result)
 | |
|       : Check(Check), Result(Result) {}
 | |
| 
 | |
|   bool VisitBinaryOperator(BinaryOperator *Op) {
 | |
|     Check->reportBinOp(Result, Op);
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   SimplifyBooleanExprCheck *Check;
 | |
|   const MatchFinder::MatchResult &Result;
 | |
| };
 | |
| 
 | |
| 
 | |
| SimplifyBooleanExprCheck::SimplifyBooleanExprCheck(StringRef Name,
 | |
|                                                    ClangTidyContext *Context)
 | |
|     : ClangTidyCheck(Name, Context),
 | |
|       ChainedConditionalReturn(Options.get("ChainedConditionalReturn", 0U)),
 | |
|       ChainedConditionalAssignment(
 | |
|           Options.get("ChainedConditionalAssignment", 0U)) {}
 | |
| 
 | |
| bool containsBoolLiteral(const Expr *E) {
 | |
|   if (!E)
 | |
|     return false;
 | |
|   E = E->IgnoreParenImpCasts();
 | |
|   if (isa<CXXBoolLiteralExpr>(E))
 | |
|     return true;
 | |
|   if (const auto *BinOp = dyn_cast<BinaryOperator>(E))
 | |
|     return containsBoolLiteral(BinOp->getLHS()) ||
 | |
|            containsBoolLiteral(BinOp->getRHS());
 | |
|   if (const auto *UnaryOp = dyn_cast<UnaryOperator>(E))
 | |
|     return containsBoolLiteral(UnaryOp->getSubExpr());
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::reportBinOp(
 | |
|     const MatchFinder::MatchResult &Result, const BinaryOperator *Op) {
 | |
|   const auto *LHS = Op->getLHS()->IgnoreParenImpCasts();
 | |
|   const auto *RHS = Op->getRHS()->IgnoreParenImpCasts();
 | |
| 
 | |
|   const CXXBoolLiteralExpr *Bool = nullptr;
 | |
|   const Expr *Other = nullptr;
 | |
|   if ((Bool = dyn_cast<CXXBoolLiteralExpr>(LHS)))
 | |
|     Other = RHS;
 | |
|   else if ((Bool = dyn_cast<CXXBoolLiteralExpr>(RHS)))
 | |
|     Other = LHS;
 | |
|   else
 | |
|     return;
 | |
| 
 | |
|   if (Bool->getLocStart().isMacroID())
 | |
|     return;
 | |
| 
 | |
|   // FIXME: why do we need this?
 | |
|   if (!isa<CXXBoolLiteralExpr>(Other) && containsBoolLiteral(Other))
 | |
|     return;
 | |
| 
 | |
|   bool BoolValue = Bool->getValue();
 | |
| 
 | |
|   auto replaceWithExpression = [this, &Result, LHS, RHS, Bool](
 | |
|                                    const Expr *ReplaceWith, bool Negated) {
 | |
|     std::string Replacement =
 | |
|         replacementExpression(Result, Negated, ReplaceWith);
 | |
|     SourceRange Range(LHS->getLocStart(), RHS->getLocEnd());
 | |
|     issueDiag(Result, Bool->getLocStart(), SimplifyOperatorDiagnostic, Range,
 | |
|               Replacement);
 | |
|   };
 | |
| 
 | |
|   switch (Op->getOpcode()) {
 | |
|     case BO_LAnd:
 | |
|       if (BoolValue) {
 | |
|         // expr && true -> expr
 | |
|         replaceWithExpression(Other, /*Negated=*/false);
 | |
|       } else {
 | |
|         // expr && false -> false
 | |
|         replaceWithExpression(Bool, /*Negated=*/false);
 | |
|       }
 | |
|       break;
 | |
|     case BO_LOr:
 | |
|       if (BoolValue) {
 | |
|         // expr || true -> true
 | |
|         replaceWithExpression(Bool, /*Negated=*/false);
 | |
|       } else {
 | |
|         // expr || false -> expr
 | |
|         replaceWithExpression(Other, /*Negated=*/false);
 | |
|       }
 | |
|       break;
 | |
|     case BO_EQ:
 | |
|       // expr == true -> expr, expr == false -> !expr
 | |
|       replaceWithExpression(Other, /*Negated=*/!BoolValue);
 | |
|       break;
 | |
|     case BO_NE:
 | |
|       // expr != true -> !expr, expr != false -> expr
 | |
|       replaceWithExpression(Other, /*Negated=*/BoolValue);
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::matchBoolCondition(MatchFinder *Finder,
 | |
|                                                   bool Value,
 | |
|                                                   StringRef BooleanId) {
 | |
|   Finder->addMatcher(
 | |
|       ifStmt(isExpansionInMainFile(),
 | |
|              hasCondition(cxxBoolLiteral(equals(Value)).bind(BooleanId)))
 | |
|           .bind(IfStmtId),
 | |
|       this);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::matchTernaryResult(MatchFinder *Finder,
 | |
|                                                   bool Value,
 | |
|                                                   StringRef TernaryId) {
 | |
|   Finder->addMatcher(
 | |
|       conditionalOperator(isExpansionInMainFile(),
 | |
|                           hasTrueExpression(cxxBoolLiteral(equals(Value))),
 | |
|                           hasFalseExpression(cxxBoolLiteral(equals(!Value))))
 | |
|           .bind(TernaryId),
 | |
|       this);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::matchIfReturnsBool(MatchFinder *Finder,
 | |
|                                                   bool Value, StringRef Id) {
 | |
|   if (ChainedConditionalReturn)
 | |
|     Finder->addMatcher(ifStmt(isExpansionInMainFile(),
 | |
|                               hasThen(returnsBool(Value, ThenLiteralId)),
 | |
|                               hasElse(returnsBool(!Value)))
 | |
|                            .bind(Id),
 | |
|                        this);
 | |
|   else
 | |
|     Finder->addMatcher(ifStmt(isExpansionInMainFile(),
 | |
|                               unless(hasParent(ifStmt())),
 | |
|                               hasThen(returnsBool(Value, ThenLiteralId)),
 | |
|                               hasElse(returnsBool(!Value)))
 | |
|                            .bind(Id),
 | |
|                        this);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::matchIfAssignsBool(MatchFinder *Finder,
 | |
|                                                   bool Value, StringRef Id) {
 | |
|   auto SimpleThen = binaryOperator(
 | |
|       hasOperatorName("="),
 | |
|       hasLHS(declRefExpr(hasDeclaration(decl().bind(IfAssignObjId)))),
 | |
|       hasLHS(expr().bind(IfAssignVariableId)),
 | |
|       hasRHS(cxxBoolLiteral(equals(Value)).bind(IfAssignLocId)));
 | |
|   auto Then = anyOf(SimpleThen, compoundStmt(statementCountIs(1),
 | |
|                                              hasAnySubstatement(SimpleThen)));
 | |
|   auto SimpleElse = binaryOperator(
 | |
|       hasOperatorName("="),
 | |
|       hasLHS(declRefExpr(hasDeclaration(equalsBoundNode(IfAssignObjId)))),
 | |
|       hasRHS(cxxBoolLiteral(equals(!Value))));
 | |
|   auto Else = anyOf(SimpleElse, compoundStmt(statementCountIs(1),
 | |
|                                              hasAnySubstatement(SimpleElse)));
 | |
|   if (ChainedConditionalAssignment)
 | |
|     Finder->addMatcher(
 | |
|         ifStmt(isExpansionInMainFile(), hasThen(Then), hasElse(Else)).bind(Id),
 | |
|         this);
 | |
|   else
 | |
|     Finder->addMatcher(ifStmt(isExpansionInMainFile(),
 | |
|                               unless(hasParent(ifStmt())), hasThen(Then),
 | |
|                               hasElse(Else))
 | |
|                            .bind(Id),
 | |
|                        this);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::matchCompoundIfReturnsBool(MatchFinder *Finder,
 | |
|                                                           bool Value,
 | |
|                                                           StringRef Id) {
 | |
|   Finder->addMatcher(
 | |
|       compoundStmt(allOf(hasAnySubstatement(ifStmt(hasThen(returnsBool(Value)),
 | |
|                                                    unless(hasElse(stmt())))),
 | |
|                          hasAnySubstatement(
 | |
|                              returnStmt(has(ignoringParenImpCasts(
 | |
|                                             cxxBoolLiteral(equals(!Value)))))
 | |
|                                  .bind(CompoundReturnId))))
 | |
|           .bind(Id),
 | |
|       this);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
 | |
|   Options.store(Opts, "ChainedConditionalReturn", ChainedConditionalReturn);
 | |
|   Options.store(Opts, "ChainedConditionalAssignment",
 | |
|                 ChainedConditionalAssignment);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::registerMatchers(MatchFinder *Finder) {
 | |
|   Finder->addMatcher(translationUnitDecl().bind("top"), this);
 | |
| 
 | |
|   matchBoolCondition(Finder, true, ConditionThenStmtId);
 | |
|   matchBoolCondition(Finder, false, ConditionElseStmtId);
 | |
| 
 | |
|   matchTernaryResult(Finder, true, TernaryId);
 | |
|   matchTernaryResult(Finder, false, TernaryNegatedId);
 | |
| 
 | |
|   matchIfReturnsBool(Finder, true, IfReturnsBoolId);
 | |
|   matchIfReturnsBool(Finder, false, IfReturnsNotBoolId);
 | |
| 
 | |
|   matchIfAssignsBool(Finder, true, IfAssignBoolId);
 | |
|   matchIfAssignsBool(Finder, false, IfAssignNotBoolId);
 | |
| 
 | |
|   matchCompoundIfReturnsBool(Finder, true, CompoundBoolId);
 | |
|   matchCompoundIfReturnsBool(Finder, false, CompoundNotBoolId);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::check(const MatchFinder::MatchResult &Result) {
 | |
|   if (const CXXBoolLiteralExpr *TrueConditionRemoved =
 | |
|           getBoolLiteral(Result, ConditionThenStmtId))
 | |
|     replaceWithThenStatement(Result, TrueConditionRemoved);
 | |
|   else if (const CXXBoolLiteralExpr *FalseConditionRemoved =
 | |
|                getBoolLiteral(Result, ConditionElseStmtId))
 | |
|     replaceWithElseStatement(Result, FalseConditionRemoved);
 | |
|   else if (const auto *Ternary =
 | |
|                Result.Nodes.getNodeAs<ConditionalOperator>(TernaryId))
 | |
|     replaceWithCondition(Result, Ternary);
 | |
|   else if (const auto *TernaryNegated =
 | |
|                Result.Nodes.getNodeAs<ConditionalOperator>(TernaryNegatedId))
 | |
|     replaceWithCondition(Result, TernaryNegated, true);
 | |
|   else if (const auto *If = Result.Nodes.getNodeAs<IfStmt>(IfReturnsBoolId))
 | |
|     replaceWithReturnCondition(Result, If);
 | |
|   else if (const auto *IfNot =
 | |
|                Result.Nodes.getNodeAs<IfStmt>(IfReturnsNotBoolId))
 | |
|     replaceWithReturnCondition(Result, IfNot, true);
 | |
|   else if (const auto *IfAssign =
 | |
|                Result.Nodes.getNodeAs<IfStmt>(IfAssignBoolId))
 | |
|     replaceWithAssignment(Result, IfAssign);
 | |
|   else if (const auto *IfAssignNot =
 | |
|                Result.Nodes.getNodeAs<IfStmt>(IfAssignNotBoolId))
 | |
|     replaceWithAssignment(Result, IfAssignNot, true);
 | |
|   else if (const auto *Compound =
 | |
|                Result.Nodes.getNodeAs<CompoundStmt>(CompoundBoolId))
 | |
|     replaceCompoundReturnWithCondition(Result, Compound);
 | |
|   else if (const auto *Compound =
 | |
|                Result.Nodes.getNodeAs<CompoundStmt>(CompoundNotBoolId))
 | |
|     replaceCompoundReturnWithCondition(Result, Compound, true);
 | |
|   else if (const auto TU = Result.Nodes.getNodeAs<Decl>("top"))
 | |
|     Visitor(this, Result).TraverseDecl(const_cast<Decl*>(TU));
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::issueDiag(
 | |
|     const ast_matchers::MatchFinder::MatchResult &Result, SourceLocation Loc,
 | |
|     StringRef Description, SourceRange ReplacementRange,
 | |
|     StringRef Replacement) {
 | |
|   CharSourceRange CharRange =
 | |
|       Lexer::makeFileCharRange(CharSourceRange::getTokenRange(ReplacementRange),
 | |
|                                *Result.SourceManager, getLangOpts());
 | |
| 
 | |
|   DiagnosticBuilder Diag = diag(Loc, Description);
 | |
|   if (!containsDiscardedTokens(Result, CharRange))
 | |
|     Diag << FixItHint::CreateReplacement(CharRange, Replacement);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceWithThenStatement(
 | |
|     const MatchFinder::MatchResult &Result,
 | |
|     const CXXBoolLiteralExpr *TrueConditionRemoved) {
 | |
|   const auto *IfStatement = Result.Nodes.getNodeAs<IfStmt>(IfStmtId);
 | |
|   issueDiag(Result, TrueConditionRemoved->getLocStart(),
 | |
|             SimplifyConditionDiagnostic, IfStatement->getSourceRange(),
 | |
|             getText(Result, *IfStatement->getThen()));
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceWithElseStatement(
 | |
|     const MatchFinder::MatchResult &Result,
 | |
|     const CXXBoolLiteralExpr *FalseConditionRemoved) {
 | |
|   const auto *IfStatement = Result.Nodes.getNodeAs<IfStmt>(IfStmtId);
 | |
|   const Stmt *ElseStatement = IfStatement->getElse();
 | |
|   issueDiag(Result, FalseConditionRemoved->getLocStart(),
 | |
|             SimplifyConditionDiagnostic, IfStatement->getSourceRange(),
 | |
|             ElseStatement ? getText(Result, *ElseStatement) : "");
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceWithCondition(
 | |
|     const MatchFinder::MatchResult &Result, const ConditionalOperator *Ternary,
 | |
|     bool Negated) {
 | |
|   std::string Replacement =
 | |
|       replacementExpression(Result, Negated, Ternary->getCond());
 | |
|   issueDiag(Result, Ternary->getTrueExpr()->getLocStart(),
 | |
|             "redundant boolean literal in ternary expression result",
 | |
|             Ternary->getSourceRange(), Replacement);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceWithReturnCondition(
 | |
|     const MatchFinder::MatchResult &Result, const IfStmt *If, bool Negated) {
 | |
|   StringRef Terminator = isa<CompoundStmt>(If->getElse()) ? ";" : "";
 | |
|   std::string Condition = replacementExpression(Result, Negated, If->getCond());
 | |
|   std::string Replacement = ("return " + Condition + Terminator).str();
 | |
|   SourceLocation Start =
 | |
|       Result.Nodes.getNodeAs<CXXBoolLiteralExpr>(ThenLiteralId)->getLocStart();
 | |
|   issueDiag(Result, Start, SimplifyConditionalReturnDiagnostic,
 | |
|             If->getSourceRange(), Replacement);
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceCompoundReturnWithCondition(
 | |
|     const MatchFinder::MatchResult &Result, const CompoundStmt *Compound,
 | |
|     bool Negated) {
 | |
|   const auto *Ret = Result.Nodes.getNodeAs<ReturnStmt>(CompoundReturnId);
 | |
| 
 | |
|   // The body shouldn't be empty because the matcher ensures that it must
 | |
|   // contain at least two statements:
 | |
|   // 1) A `return` statement returning a boolean literal `false` or `true`
 | |
|   // 2) An `if` statement with no `else` clause that consists of a single
 | |
|   //    `return` statement returning the opposite boolean literal `true` or
 | |
|   //    `false`.
 | |
|   assert(Compound->size() >= 2);
 | |
|   const IfStmt *BeforeIf = nullptr;
 | |
|   CompoundStmt::const_body_iterator Current = Compound->body_begin();
 | |
|   CompoundStmt::const_body_iterator After = Compound->body_begin();
 | |
|   for (++After; After != Compound->body_end() && *Current != Ret;
 | |
|        ++Current, ++After) {
 | |
|     if (const auto *If = dyn_cast<IfStmt>(*Current)) {
 | |
|       if (const CXXBoolLiteralExpr *Lit = stmtReturnsBool(If, Negated)) {
 | |
|         if (*After == Ret) {
 | |
|           if (!ChainedConditionalReturn && BeforeIf)
 | |
|             continue;
 | |
| 
 | |
|           const Expr *Condition = If->getCond();
 | |
|           std::string Replacement =
 | |
|               "return " + replacementExpression(Result, Negated, Condition);
 | |
|           issueDiag(
 | |
|               Result, Lit->getLocStart(), SimplifyConditionalReturnDiagnostic,
 | |
|               SourceRange(If->getLocStart(), Ret->getLocEnd()), Replacement);
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         BeforeIf = If;
 | |
|       }
 | |
|     } else {
 | |
|       BeforeIf = nullptr;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SimplifyBooleanExprCheck::replaceWithAssignment(
 | |
|     const MatchFinder::MatchResult &Result, const IfStmt *IfAssign,
 | |
|     bool Negated) {
 | |
|   SourceRange Range = IfAssign->getSourceRange();
 | |
|   StringRef VariableName =
 | |
|       getText(Result, *Result.Nodes.getNodeAs<Expr>(IfAssignVariableId));
 | |
|   StringRef Terminator = isa<CompoundStmt>(IfAssign->getElse()) ? ";" : "";
 | |
|   std::string Condition =
 | |
|       replacementExpression(Result, Negated, IfAssign->getCond());
 | |
|   std::string Replacement =
 | |
|       (VariableName + " = " + Condition + Terminator).str();
 | |
|   SourceLocation Location =
 | |
|       Result.Nodes.getNodeAs<CXXBoolLiteralExpr>(IfAssignLocId)->getLocStart();
 | |
|   issueDiag(Result, Location,
 | |
|             "redundant boolean literal in conditional assignment", Range,
 | |
|             Replacement);
 | |
| }
 | |
| 
 | |
| } // namespace readability
 | |
| } // namespace tidy
 | |
| } // namespace clang
 |