[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
//===--- PreferIsaOrDynCastInConditionalsCheck.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 "PreferIsaOrDynCastInConditionalsCheck.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
2020-06-29 23:05:51 +08:00
|
|
|
#include "clang/Lex/Lexer.h"
|
[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace ast_matchers {
|
|
|
|
AST_MATCHER(Expr, isMacroID) { return Node.getExprLoc().isMacroID(); }
|
|
|
|
} // namespace ast_matchers
|
|
|
|
|
|
|
|
namespace tidy {
|
2019-05-11 02:27:09 +08:00
|
|
|
namespace llvm_check {
|
[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
|
|
|
|
void PreferIsaOrDynCastInConditionalsCheck::registerMatchers(
|
|
|
|
MatchFinder *Finder) {
|
|
|
|
auto Condition = hasCondition(implicitCastExpr(has(
|
|
|
|
callExpr(
|
|
|
|
allOf(unless(isMacroID()), unless(cxxMemberCallExpr()),
|
|
|
|
anyOf(callee(namedDecl(hasName("cast"))),
|
|
|
|
callee(namedDecl(hasName("dyn_cast")).bind("dyn_cast")))))
|
|
|
|
.bind("call"))));
|
|
|
|
|
|
|
|
auto Any = anyOf(
|
|
|
|
has(declStmt(containsDeclaration(
|
|
|
|
0,
|
|
|
|
varDecl(hasInitializer(
|
|
|
|
callExpr(allOf(unless(isMacroID()), unless(cxxMemberCallExpr()),
|
|
|
|
callee(namedDecl(hasName("cast")))))
|
|
|
|
.bind("assign")))))),
|
|
|
|
Condition);
|
|
|
|
|
|
|
|
auto CallExpression =
|
|
|
|
callExpr(
|
2020-07-07 21:30:52 +08:00
|
|
|
allOf(
|
|
|
|
unless(isMacroID()), unless(cxxMemberCallExpr()),
|
|
|
|
allOf(callee(namedDecl(hasAnyName("isa", "cast", "cast_or_null",
|
|
|
|
"dyn_cast", "dyn_cast_or_null"))
|
|
|
|
.bind("func")),
|
|
|
|
hasArgument(0, anyOf(declRefExpr().bind("arg"),
|
|
|
|
cxxMemberCallExpr().bind("arg"))))))
|
[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
.bind("rhs");
|
|
|
|
|
|
|
|
Finder->addMatcher(
|
2019-11-12 23:15:56 +08:00
|
|
|
traverse(ast_type_traits::TK_AsIs,
|
|
|
|
stmt(anyOf(
|
|
|
|
ifStmt(Any), whileStmt(Any), doStmt(Condition),
|
|
|
|
binaryOperator(
|
|
|
|
allOf(unless(isExpansionInFileMatching(
|
|
|
|
"llvm/include/llvm/Support/Casting.h")),
|
|
|
|
hasOperatorName("&&"),
|
|
|
|
hasLHS(implicitCastExpr().bind("lhs")),
|
|
|
|
hasRHS(anyOf(implicitCastExpr(has(CallExpression)),
|
|
|
|
CallExpression))))
|
|
|
|
.bind("and")))),
|
[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PreferIsaOrDynCastInConditionalsCheck::check(
|
|
|
|
const MatchFinder::MatchResult &Result) {
|
|
|
|
if (const auto *MatchedDecl = Result.Nodes.getNodeAs<CallExpr>("assign")) {
|
|
|
|
SourceLocation StartLoc = MatchedDecl->getCallee()->getExprLoc();
|
|
|
|
SourceLocation EndLoc =
|
|
|
|
StartLoc.getLocWithOffset(StringRef("cast").size() - 1);
|
|
|
|
|
|
|
|
diag(MatchedDecl->getBeginLoc(),
|
|
|
|
"cast<> in conditional will assert rather than return a null pointer")
|
|
|
|
<< FixItHint::CreateReplacement(SourceRange(StartLoc, EndLoc),
|
|
|
|
"dyn_cast");
|
|
|
|
} else if (const auto *MatchedDecl =
|
|
|
|
Result.Nodes.getNodeAs<CallExpr>("call")) {
|
|
|
|
SourceLocation StartLoc = MatchedDecl->getCallee()->getExprLoc();
|
|
|
|
SourceLocation EndLoc =
|
|
|
|
StartLoc.getLocWithOffset(StringRef("cast").size() - 1);
|
|
|
|
|
|
|
|
StringRef Message =
|
|
|
|
"cast<> in conditional will assert rather than return a null pointer";
|
|
|
|
if (Result.Nodes.getNodeAs<NamedDecl>("dyn_cast"))
|
|
|
|
Message = "return value from dyn_cast<> not used";
|
|
|
|
|
|
|
|
diag(MatchedDecl->getBeginLoc(), Message)
|
|
|
|
<< FixItHint::CreateReplacement(SourceRange(StartLoc, EndLoc), "isa");
|
|
|
|
} else if (const auto *MatchedDecl =
|
|
|
|
Result.Nodes.getNodeAs<BinaryOperator>("and")) {
|
|
|
|
const auto *LHS = Result.Nodes.getNodeAs<ImplicitCastExpr>("lhs");
|
|
|
|
const auto *RHS = Result.Nodes.getNodeAs<CallExpr>("rhs");
|
|
|
|
const auto *Arg = Result.Nodes.getNodeAs<Expr>("arg");
|
|
|
|
const auto *Func = Result.Nodes.getNodeAs<NamedDecl>("func");
|
|
|
|
|
|
|
|
assert(LHS && "LHS is null");
|
|
|
|
assert(RHS && "RHS is null");
|
|
|
|
assert(Arg && "Arg is null");
|
|
|
|
assert(Func && "Func is null");
|
|
|
|
|
|
|
|
StringRef LHSString(Lexer::getSourceText(
|
|
|
|
CharSourceRange::getTokenRange(LHS->getSourceRange()),
|
|
|
|
*Result.SourceManager, getLangOpts()));
|
|
|
|
|
|
|
|
StringRef ArgString(Lexer::getSourceText(
|
|
|
|
CharSourceRange::getTokenRange(Arg->getSourceRange()),
|
|
|
|
*Result.SourceManager, getLangOpts()));
|
|
|
|
|
|
|
|
if (ArgString != LHSString)
|
|
|
|
return;
|
|
|
|
|
|
|
|
StringRef RHSString(Lexer::getSourceText(
|
|
|
|
CharSourceRange::getTokenRange(RHS->getSourceRange()),
|
|
|
|
*Result.SourceManager, getLangOpts()));
|
|
|
|
|
|
|
|
std::string Replacement("isa_and_nonnull");
|
|
|
|
Replacement += RHSString.substr(Func->getName().size());
|
|
|
|
|
|
|
|
diag(MatchedDecl->getBeginLoc(),
|
|
|
|
"isa_and_nonnull<> is preferred over an explicit test for null "
|
|
|
|
"followed by calling isa<>")
|
|
|
|
<< FixItHint::CreateReplacement(SourceRange(MatchedDecl->getBeginLoc(),
|
|
|
|
MatchedDecl->getEndLoc()),
|
|
|
|
Replacement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-11 02:27:09 +08:00
|
|
|
} // namespace llvm_check
|
[clang-tidy] Add new checker: llvm-prefer-isa-or-dyn-cast-in-conditionals
Summary:
Looks at conditionals and finds cases of ``cast<>``, which will
assert rather than return a null pointer, and ``dyn_cast<>`` where
the return value is not captured. Additionally, finds cases that
match the pattern ``var.foo() && isa<X>(var.foo())``, where the
method is called twice and could be expensive.
.. code-block:: c++
// Finds cases like these:
if (auto x = cast<X>(y)) <...>
if (cast<X>(y)) <...>
// But not cases like these:
if (auto f = cast<Z>(y)->foo()) <...>
if (cast<Z>(y)->foo()) <...>
Reviewers: alexfh, rjmccall, hokein, aaron.ballman, JonasToth
Reviewed By: aaron.ballman
Subscribers: xbolva00, Eugene.Zelenko, mgorny, xazax.hun, cfe-commits
Tags: #clang-tools-extra, #clang
Differential Revision: https://reviews.llvm.org/D59802
llvm-svn: 359142
2019-04-25 05:25:57 +08:00
|
|
|
} // namespace tidy
|
|
|
|
} // namespace clang
|