blob: 603401a6920ff3185a20faa59dbb9553227e19bc [file] [log] [blame]
Alexander Kornienko5b982e52015-03-09 11:48:54 +00001//===--- BoolPointerImplicitConversionCheck.cpp - clang-tidy --------------===//
Benjamin Kramer1c8b3172014-07-11 08:08:47 +00002//
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
Alexander Kornienko5b982e52015-03-09 11:48:54 +000010#include "BoolPointerImplicitConversionCheck.h"
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000011
12using namespace clang::ast_matchers;
13
14namespace clang {
Alexander Kornienko50d7f4612015-06-17 13:11:37 +000015namespace {
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000016
17AST_MATCHER(CastExpr, isPointerToBoolean) {
18 return Node.getCastKind() == CK_PointerToBoolean;
19}
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000020
Alexander Kornienko50d7f4612015-06-17 13:11:37 +000021} // namespace
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000022
23namespace tidy {
Alexander Kornienko2b3124202015-03-02 12:25:03 +000024namespace misc {
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000025
Alexander Kornienko5b982e52015-03-09 11:48:54 +000026void BoolPointerImplicitConversionCheck::registerMatchers(MatchFinder *Finder) {
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000027 // Look for ifs that have an implicit bool* to bool conversion in the
28 // condition. Filter negations.
29 Finder->addMatcher(
30 ifStmt(hasCondition(findAll(implicitCastExpr(
31 allOf(unless(hasParent(unaryOperator(hasOperatorName("!")))),
32 hasSourceExpression(expr(
Etienne Bergeron9d265992016-04-21 16:57:56 +000033 hasType(pointerType(pointee(booleanType()))),
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000034 ignoringParenImpCasts(declRefExpr().bind("expr")))),
Benjamin Kramer242b5b82014-08-12 12:12:57 +000035 isPointerToBoolean())))),
Benjamin Kramerb7f59d62014-09-03 13:21:51 +000036 unless(isInTemplateInstantiation())).bind("if"),
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000037 this);
38}
39
Alexander Kornienko5b982e52015-03-09 11:48:54 +000040void BoolPointerImplicitConversionCheck::check(
41 const MatchFinder::MatchResult &Result) {
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000042 auto *If = Result.Nodes.getStmtAs<IfStmt>("if");
43 auto *Var = Result.Nodes.getStmtAs<DeclRefExpr>("expr");
44
Benjamin Kramer242b5b82014-08-12 12:12:57 +000045 // Ignore macros.
46 if (Var->getLocStart().isMacroID())
47 return;
48
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000049 // Only allow variable accesses for now, no function calls or member exprs.
50 // Check that we don't dereference the variable anywhere within the if. This
51 // avoids false positives for checks of the pointer for nullptr before it is
52 // dereferenced. If there is a dereferencing operator on this variable don't
53 // emit a diagnostic. Also ignore array subscripts.
54 const Decl *D = Var->getDecl();
55 auto DeclRef = ignoringParenImpCasts(declRefExpr(to(equalsNode(D))));
56 if (!match(findAll(
57 unaryOperator(hasOperatorName("*"), hasUnaryOperand(DeclRef))),
58 *If, *Result.Context).empty() ||
59 !match(findAll(arraySubscriptExpr(hasBase(DeclRef))), *If,
60 *Result.Context).empty() ||
61 // FIXME: We should still warn if the paremater is implicitly converted to
62 // bool.
Gabor Horvath5273f612016-04-06 12:04:51 +000063 !match(findAll(callExpr(hasAnyArgument(ignoringParenImpCasts(DeclRef)))),
64 *If, *Result.Context)
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000065 .empty() ||
Aaron Ballmanb9ea09c2015-09-17 13:31:25 +000066 !match(findAll(cxxDeleteExpr(has(expr(DeclRef)))), *If, *Result.Context)
Benjamin Kramer1c8b3172014-07-11 08:08:47 +000067 .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