// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// Clang tool to change calls to scoper::Pass() to just use std::move().
#include <memory>
#include <string>
#include "clang/AST/ASTContext.h"
#include "clang/ASTMatchers/ASTMatchFinder.h"
#include "clang/ASTMatchers/ASTMatchers.h"
#include "clang/ASTMatchers/ASTMatchersMacros.h"
#include "clang/Basic/SourceManager.h"
#include "clang/Frontend/FrontendActions.h"
#include "clang/Lex/Lexer.h"
#include "clang/Tooling/CommonOptionsParser.h"
#include "clang/Tooling/Refactoring.h"
#include "clang/Tooling/Tooling.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/TargetSelect.h"
using namespace clang::ast_matchers;
using clang::tooling::CommonOptionsParser;
using clang::tooling::Replacement;
using clang::tooling::Replacements;
using llvm::StringRef;
namespace {
class RewriterCallback : public MatchFinder::MatchCallback {
public:
explicit RewriterCallback(Replacements* replacements)
: replacements_(replacements) {}
virtual void run(const MatchFinder::MatchResult& result) override;
private:
Replacements* const replacements_;
};
void RewriterCallback::run(const MatchFinder::MatchResult& result) {
const clang::CXXMemberCallExpr* call_expr =
result.Nodes.getNodeAs<clang::CXXMemberCallExpr>("expr");
const clang::MemberExpr* callee =
clang::dyn_cast<clang::MemberExpr>(call_expr->getCallee());
const bool is_arrow = callee->isArrow();
const clang::Expr* arg = result.Nodes.getNodeAs<clang::Expr>("arg");
const char kMoveRefText[] = "std::move(";
const char kMovePtrText[] = "std::move(*";
auto err = replacements_->add(
Replacement(*result.SourceManager,
result.SourceManager->getSpellingLoc(arg->getLocStart()), 0,
is_arrow ? kMovePtrText : kMoveRefText));
assert(!err);
// Delete everything but the closing parentheses from the original call to
// Pass(): the closing parantheses is left to match up with the parantheses
// just inserted with std::move.
err = replacements_->add(Replacement(
*result.SourceManager,
clang::CharSourceRange::getCharRange(
result.SourceManager->getSpellingLoc(callee->getOperatorLoc()),
result.SourceManager->getSpellingLoc(call_expr->getRParenLoc())),
""));
assert(!err);
}
} // namespace
static llvm::cl::extrahelp common_help(CommonOptionsParser::HelpMessage);
int main(int argc, const char* argv[]) {
// TODO(dcheng): Clang tooling should do this itself.
// http://llvm.org/bugs/show_bug.cgi?id=21627
llvm::InitializeNativeTarget();
llvm::InitializeNativeTargetAsmParser();
llvm::cl::OptionCategory category(
"C++11 modernization: change scoped::Pass() to std::move()");
CommonOptionsParser options(argc, argv, category);
clang::tooling::ClangTool tool(options.getCompilations(),
options.getSourcePathList());
MatchFinder match_finder;
Replacements replacements;
auto pass_matcher = id(
"expr",
cxxMemberCallExpr(
argumentCountIs(0),
callee(functionDecl(hasName("Pass"), returns(rValueReferenceType()))),
on(id("arg", expr()))));
RewriterCallback callback(&replacements);
match_finder.addMatcher(pass_matcher, &callback);
std::unique_ptr<clang::tooling::FrontendActionFactory> factory =
clang::tooling::newFrontendActionFactory(&match_finder);
int result = tool.run(factory.get());
if (result != 0)
return result;
// Serialization format is documented in tools/clang/scripts/run_tool.py
llvm::outs() << "==== BEGIN EDITS ====\n";
for (const auto& r : replacements) {
std::string replacement_text = r.getReplacementText().str();
std::replace(replacement_text.begin(), replacement_text.end(), '\n', '\0');
llvm::outs() << "r:::" << r.getFilePath() << ":::" << r.getOffset()
<< ":::" << r.getLength() << ":::" << replacement_text << "\n";
}
llvm::outs() << "==== END EDITS ====\n";
return 0;
}