17 const ParmVarDecl *ParmVar,
18 const TemplateTypeParmDecl *TypeParmDecl,
19 DiagnosticBuilder &Diag,
20 const ASTContext &Context) {
21 const SourceManager &SM = Context.getSourceManager();
22 const LangOptions &LangOpts = Context.getLangOpts();
24 CharSourceRange CallRange =
25 Lexer::makeFileCharRange(CharSourceRange::getTokenRange(
26 Callee->getBeginLoc(), Callee->getEndLoc()),
29 if (CallRange.isValid()) {
30 const std::string TypeName =
31 (TypeParmDecl->getIdentifier() && !TypeParmDecl->isImplicit())
32 ? TypeParmDecl->getName().str()
33 : (llvm::Twine(
"decltype(") + ParmVar->getName() +
")").str();
35 const std::string ForwardName =
36 (llvm::Twine(
"forward<") + TypeName +
">").str();
42 NestedNameSpecifier NNS = Callee->getQualifier();
43 switch (NNS.getKind()) {
44 case NestedNameSpecifier::Kind::Null:
48 Diag << FixItHint::CreateReplacement(CallRange,
"std::" + ForwardName);
50 case NestedNameSpecifier::Kind::Namespace: {
51 auto [Namespace, Prefix] = NNS.getAsNamespaceAndPrefix();
52 if (Namespace->getName() ==
"std") {
55 Diag << FixItHint::CreateReplacement(CallRange,
56 "std::" + ForwardName);
57 }
else if (Prefix.getKind() == NestedNameSpecifier::Kind::Global) {
59 Diag << FixItHint::CreateReplacement(CallRange,
60 "::std::" + ForwardName);
74 auto ForwardingReferenceParmMatcher =
76 hasType(qualType(rValueReferenceType(),
77 references(templateTypeParmType(hasDeclaration(
78 templateTypeParmDecl().bind(
"type-parm-decl")))),
79 unless(references(qualType(isConstQualified()))))))
83 callExpr(callee(unresolvedLookupExpr(
84 hasAnyDeclaration(namedDecl(
85 hasUnderlyingDecl(hasName(
"::std::move")))))
88 hasArgument(0, ignoringParenImpCasts(declRefExpr(
89 to(ForwardingReferenceParmMatcher)))))
95 const MatchFinder::MatchResult &Result) {
96 const auto *CallMove = Result.Nodes.getNodeAs<CallExpr>(
"call-move");
97 const auto *UnresolvedLookup =
98 Result.Nodes.getNodeAs<UnresolvedLookupExpr>(
"lookup");
99 const auto *ParmVar = Result.Nodes.getNodeAs<ParmVarDecl>(
"parm-var");
100 const auto *TypeParmDecl =
101 Result.Nodes.getNodeAs<TemplateTypeParmDecl>(
"type-parm-decl");
105 const auto *FuncForParam = dyn_cast<FunctionDecl>(ParmVar->getDeclContext());
108 const FunctionTemplateDecl *FuncTemplate =
109 FuncForParam->getDescribedFunctionTemplate();
116 const TemplateParameterList *Params = FuncTemplate->getTemplateParameters();
117 if (!llvm::is_contained(*Params, TypeParmDecl))
120 auto Diag = diag(CallMove->getExprLoc(),
121 "forwarding reference passed to std::move(), which may "
122 "unexpectedly cause lvalues to be moved; use "
123 "std::forward() instead");