25 const ParmVarDecl *ParmVar,
26 const TemplateTypeParmDecl *TypeParmDecl,
27 DiagnosticBuilder &Diag,
28 const ASTContext &Context) {
29 const SourceManager &SM = Context.getSourceManager();
30 const LangOptions &LangOpts = Context.getLangOpts();
32 const CharSourceRange CallRange =
33 Lexer::makeFileCharRange(CharSourceRange::getTokenRange(
34 Callee->getBeginLoc(), Callee->getEndLoc()),
37 if (CallRange.isValid()) {
38 const std::string TypeName =
39 (TypeParmDecl->getIdentifier() && !TypeParmDecl->isImplicit())
40 ? TypeParmDecl->getName().str()
41 : (llvm::Twine(
"decltype(") + ParmVar->getName() +
")").str();
43 const std::string ForwardName =
44 (llvm::Twine(
"forward<") + TypeName +
">").str();
50 const NestedNameSpecifier NNS = Callee->getQualifier();
51 switch (NNS.getKind()) {
52 case NestedNameSpecifier::Kind::Null:
56 Diag << FixItHint::CreateReplacement(CallRange,
"std::" + ForwardName);
58 case NestedNameSpecifier::Kind::Namespace: {
59 auto [Namespace, Prefix] = NNS.getAsNamespaceAndPrefix();
60 if (Namespace->getName() ==
"std") {
63 Diag << FixItHint::CreateReplacement(CallRange,
64 "std::" + ForwardName);
65 }
else if (Prefix.getKind() == NestedNameSpecifier::Kind::Global) {
67 Diag << FixItHint::CreateReplacement(CallRange,
68 "::std::" + ForwardName);
82 auto ForwardingReferenceParmMatcher =
84 hasType(qualType(rValueReferenceType(),
85 references(templateTypeParmType(hasDeclaration(
86 templateTypeParmDecl().bind(
"type-parm-decl")))),
87 unless(references(qualType(isConstQualified()))))))
92 callee(unresolvedLookupExpr(
94 namedDecl(hasUnderlyingDecl(hasName(
"::std::move")))))
97 hasArgument(0, ignoringParenImpCasts(declRefExpr(
98 to(ForwardingReferenceParmMatcher),
99 unless(refersToEnclosingVariableOrCapture())))))
105 const MatchFinder::MatchResult &Result) {
106 const auto *CallMove = Result.Nodes.getNodeAs<CallExpr>(
"call-move");
107 const auto *UnresolvedLookup =
108 Result.Nodes.getNodeAs<UnresolvedLookupExpr>(
"lookup");
109 const auto *ParmVar = Result.Nodes.getNodeAs<ParmVarDecl>(
"parm-var");
110 const auto *TypeParmDecl =
111 Result.Nodes.getNodeAs<TemplateTypeParmDecl>(
"type-parm-decl");
115 const auto *FuncForParam = dyn_cast<FunctionDecl>(ParmVar->getDeclContext());
118 const FunctionTemplateDecl *FuncTemplate =
119 FuncForParam->getDescribedFunctionTemplate();
126 const TemplateParameterList *Params = FuncTemplate->getTemplateParameters();
127 if (!llvm::is_contained(*Params, TypeParmDecl))
130 auto Diag = diag(CallMove->getExprLoc(),
131 "forwarding reference passed to std::move(), which may "
132 "unexpectedly cause lvalues to be moved; use "
133 "std::forward() instead");