468663ddbb
Former-commit-id: 1d6753294b2993e1fbf92de9366bb9544db4189b
220 lines
8.4 KiB
C++
220 lines
8.4 KiB
C++
//===--- SuspiciousEnumUsageCheck.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 "SuspiciousEnumUsageCheck.h"
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
|
#include <algorithm>
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
namespace clang {
|
|
namespace tidy {
|
|
namespace misc {
|
|
|
|
static const char DifferentEnumErrorMessage[] =
|
|
"enum values are from different enum types";
|
|
|
|
static const char BitmaskErrorMessage[] =
|
|
"enum type seems like a bitmask (contains mostly "
|
|
"power-of-2 literals), but this literal is not a "
|
|
"power-of-2";
|
|
|
|
static const char BitmaskVarErrorMessage[] =
|
|
"enum type seems like a bitmask (contains mostly "
|
|
"power-of-2 literals) but %plural{1:a literal is|:some literals are}0 not "
|
|
"power-of-2";
|
|
|
|
static const char BitmaskNoteMessage[] = "used here as a bitmask";
|
|
|
|
/// Stores a min and a max value which describe an interval.
|
|
struct ValueRange {
|
|
llvm::APSInt MinVal;
|
|
llvm::APSInt MaxVal;
|
|
|
|
ValueRange(const EnumDecl *EnumDec) {
|
|
const auto MinMaxVal = std::minmax_element(
|
|
EnumDec->enumerator_begin(), EnumDec->enumerator_end(),
|
|
[](const EnumConstantDecl *E1, const EnumConstantDecl *E2) {
|
|
return llvm::APSInt::compareValues(E1->getInitVal(),
|
|
E2->getInitVal()) < 0;
|
|
});
|
|
MinVal = MinMaxVal.first->getInitVal();
|
|
MaxVal = MinMaxVal.second->getInitVal();
|
|
}
|
|
};
|
|
|
|
/// Return the number of EnumConstantDecls in an EnumDecl.
|
|
static int enumLength(const EnumDecl *EnumDec) {
|
|
return std::distance(EnumDec->enumerator_begin(), EnumDec->enumerator_end());
|
|
}
|
|
|
|
static bool hasDisjointValueRange(const EnumDecl *Enum1,
|
|
const EnumDecl *Enum2) {
|
|
ValueRange Range1(Enum1), Range2(Enum2);
|
|
return llvm::APSInt::compareValues(Range1.MaxVal, Range2.MinVal) < 0 ||
|
|
llvm::APSInt::compareValues(Range2.MaxVal, Range1.MinVal) < 0;
|
|
}
|
|
|
|
static bool isNonPowerOf2NorNullLiteral(const EnumConstantDecl *EnumConst) {
|
|
llvm::APSInt Val = EnumConst->getInitVal();
|
|
if (Val.isPowerOf2() || !Val.getBoolValue())
|
|
return false;
|
|
const Expr *InitExpr = EnumConst->getInitExpr();
|
|
if (!InitExpr)
|
|
return true;
|
|
return isa<IntegerLiteral>(InitExpr->IgnoreImpCasts());
|
|
}
|
|
|
|
static bool isMaxValAllBitSetLiteral(const EnumDecl *EnumDec) {
|
|
auto EnumConst = std::max_element(
|
|
EnumDec->enumerator_begin(), EnumDec->enumerator_end(),
|
|
[](const EnumConstantDecl *E1, const EnumConstantDecl *E2) {
|
|
return E1->getInitVal() < E2->getInitVal();
|
|
});
|
|
|
|
if (const Expr *InitExpr = EnumConst->getInitExpr()) {
|
|
return EnumConst->getInitVal().countTrailingOnes() ==
|
|
EnumConst->getInitVal().getActiveBits() &&
|
|
isa<IntegerLiteral>(InitExpr->IgnoreImpCasts());
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int countNonPowOfTwoLiteralNum(const EnumDecl *EnumDec) {
|
|
return std::count_if(
|
|
EnumDec->enumerator_begin(), EnumDec->enumerator_end(),
|
|
[](const EnumConstantDecl *E) { return isNonPowerOf2NorNullLiteral(E); });
|
|
}
|
|
|
|
/// Check if there is one or two enumerators that are not a power of 2 and are
|
|
/// initialized by a literal in the enum type, and that the enumeration contains
|
|
/// enough elements to reasonably act as a bitmask. Exclude the case where the
|
|
/// last enumerator is the sum of the lesser values (and initialized by a
|
|
/// literal) or when it could contain consecutive values.
|
|
static bool isPossiblyBitMask(const EnumDecl *EnumDec) {
|
|
ValueRange VR(EnumDec);
|
|
int EnumLen = enumLength(EnumDec);
|
|
int NonPowOfTwoCounter = countNonPowOfTwoLiteralNum(EnumDec);
|
|
return NonPowOfTwoCounter >= 1 && NonPowOfTwoCounter <= 2 &&
|
|
NonPowOfTwoCounter < EnumLen / 2 &&
|
|
(VR.MaxVal - VR.MinVal != EnumLen - 1) &&
|
|
!(NonPowOfTwoCounter == 1 && isMaxValAllBitSetLiteral(EnumDec));
|
|
}
|
|
|
|
SuspiciousEnumUsageCheck::SuspiciousEnumUsageCheck(StringRef Name,
|
|
ClangTidyContext *Context)
|
|
: ClangTidyCheck(Name, Context),
|
|
StrictMode(Options.getLocalOrGlobal("StrictMode", 0)) {}
|
|
|
|
void SuspiciousEnumUsageCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
|
|
Options.store(Opts, "StrictMode", StrictMode);
|
|
}
|
|
|
|
void SuspiciousEnumUsageCheck::registerMatchers(MatchFinder *Finder) {
|
|
const auto enumExpr = [](StringRef RefName, StringRef DeclName) {
|
|
return allOf(ignoringImpCasts(expr().bind(RefName)),
|
|
ignoringImpCasts(hasType(enumDecl().bind(DeclName))));
|
|
};
|
|
|
|
Finder->addMatcher(
|
|
binaryOperator(hasOperatorName("|"), hasLHS(enumExpr("", "enumDecl")),
|
|
hasRHS(allOf(enumExpr("", "otherEnumDecl"),
|
|
ignoringImpCasts(hasType(enumDecl(
|
|
unless(equalsBoundNode("enumDecl"))))))))
|
|
.bind("diffEnumOp"),
|
|
this);
|
|
|
|
Finder->addMatcher(
|
|
binaryOperator(anyOf(hasOperatorName("+"), hasOperatorName("|")),
|
|
hasLHS(enumExpr("lhsExpr", "enumDecl")),
|
|
hasRHS(allOf(enumExpr("rhsExpr", ""),
|
|
ignoringImpCasts(hasType(enumDecl(
|
|
equalsBoundNode("enumDecl"))))))),
|
|
this);
|
|
|
|
Finder->addMatcher(
|
|
binaryOperator(anyOf(hasOperatorName("+"), hasOperatorName("|")),
|
|
hasEitherOperand(
|
|
allOf(hasType(isInteger()), unless(enumExpr("", "")))),
|
|
hasEitherOperand(enumExpr("enumExpr", "enumDecl"))),
|
|
this);
|
|
|
|
Finder->addMatcher(
|
|
binaryOperator(anyOf(hasOperatorName("|="), hasOperatorName("+=")),
|
|
hasRHS(enumExpr("enumExpr", "enumDecl"))),
|
|
this);
|
|
}
|
|
|
|
void SuspiciousEnumUsageCheck::checkSuspiciousBitmaskUsage(
|
|
const Expr *NodeExpr, const EnumDecl *EnumDec) {
|
|
const auto *EnumExpr = dyn_cast<DeclRefExpr>(NodeExpr);
|
|
const auto *EnumConst =
|
|
EnumExpr ? dyn_cast<EnumConstantDecl>(EnumExpr->getDecl()) : nullptr;
|
|
|
|
// Report the parameter if neccessary.
|
|
if (!EnumConst) {
|
|
diag(EnumDec->getInnerLocStart(), BitmaskVarErrorMessage)
|
|
<< countNonPowOfTwoLiteralNum(EnumDec);
|
|
diag(EnumExpr->getExprLoc(), BitmaskNoteMessage, DiagnosticIDs::Note);
|
|
} else if (isNonPowerOf2NorNullLiteral(EnumConst)) {
|
|
diag(EnumConst->getSourceRange().getBegin(), BitmaskErrorMessage);
|
|
diag(EnumExpr->getExprLoc(), BitmaskNoteMessage, DiagnosticIDs::Note);
|
|
}
|
|
}
|
|
|
|
void SuspiciousEnumUsageCheck::check(const MatchFinder::MatchResult &Result) {
|
|
// Case 1: The two enum values come from different types.
|
|
if (const auto *DiffEnumOp =
|
|
Result.Nodes.getNodeAs<BinaryOperator>("diffEnumOp")) {
|
|
const auto *EnumDec = Result.Nodes.getNodeAs<EnumDecl>("enumDecl");
|
|
const auto *OtherEnumDec =
|
|
Result.Nodes.getNodeAs<EnumDecl>("otherEnumDecl");
|
|
// Skip when one of the parameters is an empty enum. The
|
|
// hasDisjointValueRange function could not decide the values properly in
|
|
// case of an empty enum.
|
|
if (EnumDec->enumerator_begin() == EnumDec->enumerator_end() ||
|
|
OtherEnumDec->enumerator_begin() == OtherEnumDec->enumerator_end())
|
|
return;
|
|
|
|
if (!hasDisjointValueRange(EnumDec, OtherEnumDec))
|
|
diag(DiffEnumOp->getOperatorLoc(), DifferentEnumErrorMessage);
|
|
return;
|
|
}
|
|
|
|
// Case 2 and 3 only checked in strict mode. The checker tries to detect
|
|
// suspicious bitmasks which contains values initialized by non power-of-2
|
|
// literals.
|
|
if (!StrictMode)
|
|
return;
|
|
const auto *EnumDec = Result.Nodes.getNodeAs<EnumDecl>("enumDecl");
|
|
if (!isPossiblyBitMask(EnumDec))
|
|
return;
|
|
|
|
// Case 2:
|
|
// a. Investigating the right hand side of `+=` or `|=` operator.
|
|
// b. When the operator is `|` or `+` but only one of them is an EnumExpr
|
|
if (const auto *EnumExpr = Result.Nodes.getNodeAs<Expr>("enumExpr")) {
|
|
checkSuspiciousBitmaskUsage(EnumExpr, EnumDec);
|
|
return;
|
|
}
|
|
|
|
// Case 3:
|
|
// '|' or '+' operator where both argument comes from the same enum type
|
|
const auto *LhsExpr = Result.Nodes.getNodeAs<Expr>("lhsExpr");
|
|
checkSuspiciousBitmaskUsage(LhsExpr, EnumDec);
|
|
|
|
const auto *RhsExpr = Result.Nodes.getNodeAs<Expr>("rhsExpr");
|
|
checkSuspiciousBitmaskUsage(RhsExpr, EnumDec);
|
|
}
|
|
|
|
} // namespace misc
|
|
} // namespace tidy
|
|
} // namespace clang
|