blob: b243ddb63a2a48b8cd413b255763d291c3ce5a79 [file] [log] [blame]
Benjamin Kramer1c8b3172014-07-11 08:08:47 +00001//===--- BoolPointerImplicitConversion.cpp - clang-tidy -------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "BoolPointerImplicitConversion.h"
11
12using namespace clang::ast_matchers;
13
14namespace clang {
15namespace ast_matchers {
16
17AST_MATCHER(CastExpr, isPointerToBoolean) {
18 return Node.getCastKind() == CK_PointerToBoolean;
19}
20AST_MATCHER(QualType, isBoolean) { return Node->isBooleanType(); }
21
22} // namespace ast_matchers
23
24namespace tidy {
Alexander Kornienko2b3124202015-03-02 12:25:03 +000025namespace misc {
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000026
27void BoolPointerImplicitConversion::registerMatchers(MatchFinder *Finder) {
28 // Look for ifs that have an implicit bool* to bool conversion in the
29 // condition. Filter negations.
30 Finder->addMatcher(
31 ifStmt(hasCondition(findAll(implicitCastExpr(
32 allOf(unless(hasParent(unaryOperator(hasOperatorName("!")))),
33 hasSourceExpression(expr(
34 hasType(pointerType(pointee(isBoolean()))),
35 ignoringParenImpCasts(declRefExpr().bind("expr")))),
Benjamin Kramer242b5b82014-08-12 12:12:57 +000036 isPointerToBoolean())))),
Benjamin Kramerb7f59d62014-09-03 13:21:51 +000037 unless(isInTemplateInstantiation())).bind("if"),
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000038 this);
39}
40
41void
42BoolPointerImplicitConversion::check(const MatchFinder::MatchResult &Result) {
43 auto *If = Result.Nodes.getStmtAs<IfStmt>("if");
44 auto *Var = Result.Nodes.getStmtAs<DeclRefExpr>("expr");
45
Benjamin Kramer242b5b82014-08-12 12:12:57 +000046 // Ignore macros.
47 if (Var->getLocStart().isMacroID())
48 return;
49
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000050 // Only allow variable accesses for now, no function calls or member exprs.
51 // Check that we don't dereference the variable anywhere within the if. This
52 // avoids false positives for checks of the pointer for nullptr before it is
53 // dereferenced. If there is a dereferencing operator on this variable don't
54 // emit a diagnostic. Also ignore array subscripts.
55 const Decl *D = Var->getDecl();
56 auto DeclRef = ignoringParenImpCasts(declRefExpr(to(equalsNode(D))));
57 if (!match(findAll(
58 unaryOperator(hasOperatorName("*"), hasUnaryOperand(DeclRef))),
59 *If, *Result.Context).empty() ||
60 !match(findAll(arraySubscriptExpr(hasBase(DeclRef))), *If,
61 *Result.Context).empty() ||
62 // FIXME: We should still warn if the paremater is implicitly converted to
63 // bool.
64 !match(findAll(callExpr(hasAnyArgument(DeclRef))), *If, *Result.Context)
65 .empty() ||
66 !match(findAll(deleteExpr(has(expr(DeclRef)))), *If, *Result.Context)
67 .empty())
68 return;
69
70 diag(Var->getLocStart(), "dubious check of 'bool *' against 'nullptr', did "
71 "you mean to dereference it?")
72 << FixItHint::CreateInsertion(Var->getLocStart(), "*");
73}
74
Alexander Kornienko2b3124202015-03-02 12:25:03 +000075} // namespace misc
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000076} // namespace tidy
77} // namespace clang