25 anyOf(cxxOperatorCallExpr(
26 hasOverloadedOperatorName(
"<<"),
27 hasRHS(declRefExpr(to(namedDecl(hasName(
"::std::endl"))))
29 callExpr(argumentCountIs(1),
30 callee(functionDecl(hasName(
"::std::endl"))))
36 const auto *Expression = Result.Nodes.getNodeAs<Expr>(
"expr");
38 assert(isa<DeclRefExpr>(Expression) || isa<CallExpr>(Expression));
44 if (isa<DeclRefExpr>(Expression)) {
46 const CharSourceRange TokenRange =
47 CharSourceRange::getTokenRange(Expression->getSourceRange());
48 StringRef SourceText = Lexer::getSourceText(
49 TokenRange, *Result.SourceManager, Result.Context->getLangOpts());
50 if (SourceText.empty())
51 SourceText =
"std::endl";
52 auto Diag = diag(Expression->getBeginLoc(),
53 "do not use '%0' with streams; use '\\n' instead")
55 if (TokenRange.isValid())
56 Diag << FixItHint::CreateReplacement(TokenRange,
"'\\n'");
59 const auto *CallExpression = cast<CallExpr>(Expression);
60 assert(CallExpression->getNumArgs() == 1);
62 StringRef SourceText = Lexer::getSourceText(
63 CharSourceRange::getTokenRange(
64 CallExpression->getCallee()->getSourceRange()),
65 *Result.SourceManager, Result.Context->getLangOpts());
66 if (SourceText.empty())
67 SourceText =
"std::endl";
68 auto Diag = diag(CallExpression->getBeginLoc(),
69 "do not use '%0' with streams; use '\\n' instead")
72 const CharSourceRange ArgTokenRange = CharSourceRange::getTokenRange(
73 CallExpression->getArg(0)->getSourceRange());
74 const StringRef ArgSourceText = Lexer::getSourceText(
75 ArgTokenRange, *Result.SourceManager, Result.Context->getLangOpts());
76 const CharSourceRange ReplacementRange =
77 CharSourceRange::getTokenRange(CallExpression->getSourceRange());
78 if (!ArgSourceText.empty() && ReplacementRange.isValid()) {
79 const std::string ReplacementString =
80 std::string(ArgSourceText) +
" << '\\n'";
81 Diag << FixItHint::CreateReplacement(ReplacementRange, ReplacementString);