summaryrefslogtreecommitdiffstats
path: root/clang-tools-extra/cpp11-migrate/ReplaceAutoPtr/ReplaceAutoPtrMatchers.cpp
blob: 64bff2c2211b5fb21fca916e2a84f42dfb361dfc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
//===-- ReplaceAutoPtrMatchers.cpp - std::auto_ptr replacement -*- C++ -*--===//
//
//                     The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
///
/// \file
/// \brief This file contains the definitions for matcher-generating functions
/// and names for bound nodes found by AST matchers.
///
//===----------------------------------------------------------------------===//

#include "ReplaceAutoPtrMatchers.h"

const char *AutoPtrTokenId = "AutoPtrTokenId";
const char *AutoPtrOwnershipTransferId = "AutoPtrOwnershipTransferId";

namespace clang {
namespace ast_matchers {

/// \brief Matches expressions that are lvalues.
///
/// In the following example, a[0] matches expr(isLValue()):
/// \code
///     std::string a[2];
///     std::string b;
///     b = a[0];
///     b = "this string won't match";
/// \endcode
AST_MATCHER(Expr, isLValue) {
  return Node.getValueKind() == VK_LValue;
}

/// \brief Matches declarations whose declaration context is the C++ standard
/// library namespace \c std.
///
/// Note that inline namespaces are silently ignored during the lookup since
/// both libstdc++ and libc++ are known to use them for versioning purposes.
///
/// Given
/// \code
///   namespace ns {
///     struct my_type {};
///     using namespace std;
///   }
///
///   using std::vector;
///   using ns::my_type;
///   using ns::list;
/// \endcode
/// usingDecl(hasAnyUsingShadowDecl(hasTargetDecl(isFromStdNamespace())))
///   matches "using std::vector" and "using ns::list".
AST_MATCHER(Decl, isFromStdNamespace) {
  const DeclContext *D = Node.getDeclContext();

  while (D->isInlineNamespace())
    D = D->getParent();

  if (!D->isNamespace() || !D->getParent()->isTranslationUnit())
    return false;

  const IdentifierInfo *Info = cast<NamespaceDecl>(D)->getIdentifier();

  return Info && Info->isStr("std");
}

} // end namespace ast_matchers
} // end namespace clang

using namespace clang;
using namespace clang::ast_matchers;

// shared matchers
static DeclarationMatcher AutoPtrDecl =
    recordDecl(hasName("auto_ptr"), isFromStdNamespace());

static TypeMatcher AutoPtrType = qualType(hasDeclaration(AutoPtrDecl));

// Matcher that finds expressions that are candidates to be wrapped with
// 'std::move()'.
//
// Binds the id \c AutoPtrOwnershipTransferId to the expression.
static StatementMatcher MovableArgumentMatcher = expr(
    allOf(isLValue(), hasType(AutoPtrType))).bind(AutoPtrOwnershipTransferId);

TypeLocMatcher makeAutoPtrTypeLocMatcher() {
  // skip elaboratedType() as the named type will match soon thereafter.
  return typeLoc(loc(qualType(AutoPtrType, unless(elaboratedType()))))
      .bind(AutoPtrTokenId);
}

DeclarationMatcher makeAutoPtrUsingDeclMatcher() {
  return usingDecl(hasAnyUsingShadowDecl(hasTargetDecl(
      allOf(hasName("auto_ptr"), isFromStdNamespace())))).bind(AutoPtrTokenId);
}

StatementMatcher makeTransferOwnershipExprMatcher() {
  StatementMatcher assignOperator =
    operatorCallExpr(allOf(
      hasOverloadedOperatorName("="),
      callee(methodDecl(ofClass(AutoPtrDecl))),
      hasArgument(1, MovableArgumentMatcher)));

  StatementMatcher copyCtor =
    constructExpr(allOf(hasType(AutoPtrType),
                        argumentCountIs(1),
                        hasArgument(0, MovableArgumentMatcher)));

  return anyOf(assignOperator, copyCtor);
}
OpenPOWER on IntegriCloud