forked from OSchip/llvm-project
70 lines
2.4 KiB
C++
70 lines
2.4 KiB
C++
|
//===--- UniqueptrDeleteReleaseCheck.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 "UniqueptrDeleteReleaseCheck.h"
|
||
|
#include "clang/AST/ASTContext.h"
|
||
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
||
|
#include "clang/Lex/Lexer.h"
|
||
|
|
||
|
using namespace clang::ast_matchers;
|
||
|
|
||
|
namespace clang {
|
||
|
namespace tidy {
|
||
|
|
||
|
void UniqueptrDeleteReleaseCheck::registerMatchers(MatchFinder *Finder) {
|
||
|
auto IsSusbstituted = qualType(anyOf(
|
||
|
substTemplateTypeParmType(), hasDescendant(substTemplateTypeParmType())));
|
||
|
|
||
|
auto UniquePtrWithDefaultDelete = classTemplateSpecializationDecl(
|
||
|
hasName("std::unique_ptr"),
|
||
|
hasTemplateArgument(1, refersToType(qualType(hasDeclaration(cxxRecordDecl(
|
||
|
hasName("std::default_delete")))))));
|
||
|
|
||
|
Finder->addMatcher(
|
||
|
cxxDeleteExpr(
|
||
|
has(cxxMemberCallExpr(on(expr(hasType(UniquePtrWithDefaultDelete),
|
||
|
unless(hasType(IsSusbstituted)))
|
||
|
.bind("uptr")),
|
||
|
callee(cxxMethodDecl(hasName("release"))))))
|
||
|
.bind("delete"),
|
||
|
this);
|
||
|
}
|
||
|
|
||
|
void UniqueptrDeleteReleaseCheck::check(
|
||
|
const MatchFinder::MatchResult &Result) {
|
||
|
const auto *PtrExpr = Result.Nodes.getNodeAs<Expr>("uptr");
|
||
|
const auto *DeleteExpr = Result.Nodes.getNodeAs<Expr>("delete");
|
||
|
|
||
|
if (PtrExpr->getLocStart().isMacroID())
|
||
|
return;
|
||
|
|
||
|
// Ignore dependent types.
|
||
|
// It can give us false positives, so we go with false negatives instead to
|
||
|
// be safe.
|
||
|
if (PtrExpr->getType()->isDependentType())
|
||
|
return;
|
||
|
|
||
|
SourceLocation AfterPtr =
|
||
|
Lexer::getLocForEndOfToken(PtrExpr->getLocEnd(), 0, *Result.SourceManager,
|
||
|
Result.Context->getLangOpts());
|
||
|
|
||
|
diag(DeleteExpr->getLocStart(),
|
||
|
"prefer '= nullptr' to 'delete x.release()' to reset unique_ptr<> "
|
||
|
"objects")
|
||
|
<< FixItHint::CreateRemoval(CharSourceRange::getCharRange(
|
||
|
DeleteExpr->getLocStart(), PtrExpr->getLocStart()))
|
||
|
<< FixItHint::CreateReplacement(
|
||
|
CharSourceRange::getTokenRange(AfterPtr, DeleteExpr->getLocEnd()),
|
||
|
" = nullptr");
|
||
|
}
|
||
|
|
||
|
} // namespace tidy
|
||
|
} // namespace clang
|
||
|
|