123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- //===--- TooSmallLoopVariableCheck.cpp - clang-tidy -----------------------===//
- //
- // 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 "TooSmallLoopVariableCheck.h"
- #include "clang/AST/ASTContext.h"
- #include "clang/ASTMatchers/ASTMatchFinder.h"
- using namespace clang::ast_matchers;
- namespace clang::tidy::bugprone {
- static constexpr llvm::StringLiteral LoopName =
- llvm::StringLiteral("forLoopName");
- static constexpr llvm::StringLiteral LoopVarName =
- llvm::StringLiteral("loopVar");
- static constexpr llvm::StringLiteral LoopVarCastName =
- llvm::StringLiteral("loopVarCast");
- static constexpr llvm::StringLiteral LoopUpperBoundName =
- llvm::StringLiteral("loopUpperBound");
- static constexpr llvm::StringLiteral LoopIncrementName =
- llvm::StringLiteral("loopIncrement");
- TooSmallLoopVariableCheck::TooSmallLoopVariableCheck(StringRef Name,
- ClangTidyContext *Context)
- : ClangTidyCheck(Name, Context),
- MagnitudeBitsUpperLimit(Options.get("MagnitudeBitsUpperLimit", 16U)) {}
- void TooSmallLoopVariableCheck::storeOptions(
- ClangTidyOptions::OptionMap &Opts) {
- Options.store(Opts, "MagnitudeBitsUpperLimit", MagnitudeBitsUpperLimit);
- }
- /// The matcher for loops with suspicious integer loop variable.
- ///
- /// In this general example, assuming 'j' and 'k' are of integral type:
- /// \code
- /// for (...; j < 3 + 2; ++k) { ... }
- /// \endcode
- /// The following string identifiers are bound to these parts of the AST:
- /// LoopVarName: 'j' (as a VarDecl)
- /// LoopVarCastName: 'j' (after implicit conversion)
- /// LoopUpperBoundName: '3 + 2' (as an Expr)
- /// LoopIncrementName: 'k' (as an Expr)
- /// LoopName: The entire for loop (as a ForStmt)
- ///
- void TooSmallLoopVariableCheck::registerMatchers(MatchFinder *Finder) {
- StatementMatcher LoopVarMatcher =
- expr(
- ignoringParenImpCasts(declRefExpr(to(varDecl(hasType(isInteger()))))))
- .bind(LoopVarName);
- // We need to catch only those comparisons which contain any integer cast.
- StatementMatcher LoopVarConversionMatcher = traverse(
- TK_AsIs, implicitCastExpr(hasImplicitDestinationType(isInteger()),
- has(ignoringParenImpCasts(LoopVarMatcher)))
- .bind(LoopVarCastName));
- // We are interested in only those cases when the loop bound is a variable
- // value (not const, enum, etc.).
- StatementMatcher LoopBoundMatcher =
- expr(ignoringParenImpCasts(allOf(hasType(isInteger()),
- unless(integerLiteral()),
- unless(hasType(isConstQualified())),
- unless(hasType(enumType())))))
- .bind(LoopUpperBoundName);
- // We use the loop increment expression only to make sure we found the right
- // loop variable.
- StatementMatcher IncrementMatcher =
- expr(ignoringParenImpCasts(hasType(isInteger()))).bind(LoopIncrementName);
- Finder->addMatcher(
- forStmt(
- hasCondition(anyOf(
- binaryOperator(hasOperatorName("<"),
- hasLHS(LoopVarConversionMatcher),
- hasRHS(LoopBoundMatcher)),
- binaryOperator(hasOperatorName("<="),
- hasLHS(LoopVarConversionMatcher),
- hasRHS(LoopBoundMatcher)),
- binaryOperator(hasOperatorName(">"), hasLHS(LoopBoundMatcher),
- hasRHS(LoopVarConversionMatcher)),
- binaryOperator(hasOperatorName(">="), hasLHS(LoopBoundMatcher),
- hasRHS(LoopVarConversionMatcher)))),
- hasIncrement(IncrementMatcher))
- .bind(LoopName),
- this);
- }
- /// Returns the magnitude bits of an integer type.
- static unsigned calcMagnitudeBits(const ASTContext &Context,
- const QualType &IntExprType) {
- assert(IntExprType->isIntegerType());
- return IntExprType->isUnsignedIntegerType()
- ? Context.getIntWidth(IntExprType)
- : Context.getIntWidth(IntExprType) - 1;
- }
- /// Calculate the upper bound expression's magnitude bits, but ignore
- /// constant like values to reduce false positives.
- static unsigned calcUpperBoundMagnitudeBits(const ASTContext &Context,
- const Expr *UpperBound,
- const QualType &UpperBoundType) {
- // Ignore casting caused by constant values inside a binary operator.
- // We are interested in variable values' magnitude bits.
- if (const auto *BinOperator = dyn_cast<BinaryOperator>(UpperBound)) {
- const Expr *RHSE = BinOperator->getRHS()->IgnoreParenImpCasts();
- const Expr *LHSE = BinOperator->getLHS()->IgnoreParenImpCasts();
- QualType RHSEType = RHSE->getType();
- QualType LHSEType = LHSE->getType();
- if (!RHSEType->isIntegerType() || !LHSEType->isIntegerType())
- return 0;
- bool RHSEIsConstantValue = RHSEType->isEnumeralType() ||
- RHSEType.isConstQualified() ||
- isa<IntegerLiteral>(RHSE);
- bool LHSEIsConstantValue = LHSEType->isEnumeralType() ||
- LHSEType.isConstQualified() ||
- isa<IntegerLiteral>(LHSE);
- // Avoid false positives produced by two constant values.
- if (RHSEIsConstantValue && LHSEIsConstantValue)
- return 0;
- if (RHSEIsConstantValue)
- return calcMagnitudeBits(Context, LHSEType);
- if (LHSEIsConstantValue)
- return calcMagnitudeBits(Context, RHSEType);
- return std::max(calcMagnitudeBits(Context, LHSEType),
- calcMagnitudeBits(Context, RHSEType));
- }
- return calcMagnitudeBits(Context, UpperBoundType);
- }
- void TooSmallLoopVariableCheck::check(const MatchFinder::MatchResult &Result) {
- const auto *LoopVar = Result.Nodes.getNodeAs<Expr>(LoopVarName);
- const auto *UpperBound =
- Result.Nodes.getNodeAs<Expr>(LoopUpperBoundName)->IgnoreParenImpCasts();
- const auto *LoopIncrement =
- Result.Nodes.getNodeAs<Expr>(LoopIncrementName)->IgnoreParenImpCasts();
- // We matched the loop variable incorrectly.
- if (LoopVar->getType() != LoopIncrement->getType())
- return;
- QualType LoopVarType = LoopVar->getType();
- QualType UpperBoundType = UpperBound->getType();
- ASTContext &Context = *Result.Context;
- unsigned LoopVarMagnitudeBits = calcMagnitudeBits(Context, LoopVarType);
- unsigned UpperBoundMagnitudeBits =
- calcUpperBoundMagnitudeBits(Context, UpperBound, UpperBoundType);
- if (UpperBoundMagnitudeBits == 0)
- return;
- if (LoopVarMagnitudeBits > MagnitudeBitsUpperLimit)
- return;
- if (LoopVarMagnitudeBits < UpperBoundMagnitudeBits)
- diag(LoopVar->getBeginLoc(), "loop variable has narrower type %0 than "
- "iteration's upper bound %1")
- << LoopVarType << UpperBoundType;
- }
- } // namespace clang::tidy::bugprone
|