2012-08-25 07:46:42 +08:00
|
|
|
//===-- loop-convert/LoopConvert.cpp - C++11 For loop migration -*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements a tool that migrates for loops to take advantage of the
|
|
|
|
// range-basead syntax new to C++11.
|
|
|
|
//
|
|
|
|
// Usage:
|
|
|
|
// loop-convert <cmake-output-dir> <file1> <file2> ...
|
|
|
|
//
|
|
|
|
// Where <cmake-output-dir> is a CMake build directory containing a file named
|
|
|
|
// compile_commands.json.
|
|
|
|
//
|
|
|
|
// <file1>... specify the pahs of files in the CMake source tree, with the same
|
|
|
|
// requirements as other tools built on LibTooling.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "LoopActions.h"
|
|
|
|
#include "LoopMatchers.h"
|
|
|
|
#include "clang/Basic/FileManager.h"
|
|
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
|
|
#include "clang/Frontend/FrontendActions.h"
|
|
|
|
#include "clang/Tooling/Refactoring.h"
|
2012-12-04 17:21:50 +08:00
|
|
|
#include "clang/Tooling/Tooling.h"
|
2012-08-25 07:46:42 +08:00
|
|
|
|
|
|
|
using clang::ast_matchers::MatchFinder;
|
|
|
|
namespace cl = llvm::cl;
|
|
|
|
using namespace clang::tooling;
|
|
|
|
using namespace clang::loop_migrate;
|
|
|
|
|
|
|
|
static cl::opt<std::string> BuildPath(
|
|
|
|
cl::Positional,
|
|
|
|
cl::desc("<build-path>"));
|
|
|
|
|
|
|
|
static cl::list<std::string> SourcePaths(
|
|
|
|
cl::Positional,
|
|
|
|
cl::desc("<source0> [... <sourceN>]"),
|
|
|
|
cl::OneOrMore);
|
|
|
|
|
|
|
|
// General options go here:
|
|
|
|
static cl::opt<bool> CountOnly(
|
|
|
|
"count-only", cl::desc("Do not apply transformations; only count them."));
|
|
|
|
|
|
|
|
static cl::opt<TranslationConfidenceKind> TransformationLevel(
|
|
|
|
cl::desc("Choose safety requirements for transformations:"),
|
|
|
|
cl::values(clEnumValN(TCK_Safe, "A0", "Enable safe transformations"),
|
|
|
|
clEnumValN(TCK_Reasonable, "A1",
|
|
|
|
"Enable transformations that might change semantics "
|
|
|
|
"(default)"),
|
|
|
|
clEnumValN(TCK_Risky, "A2",
|
|
|
|
"Enable transformations that are likely "
|
|
|
|
"to change semantics"),
|
|
|
|
clEnumValEnd),
|
|
|
|
cl::init(TCK_Reasonable));
|
|
|
|
|
|
|
|
int main(int argc, const char **argv) {
|
|
|
|
llvm::OwningPtr<CompilationDatabase> Compilations(
|
|
|
|
FixedCompilationDatabase::loadFromCommandLine(argc, argv));
|
|
|
|
cl::ParseCommandLineOptions(argc, argv);
|
|
|
|
if (!Compilations) {
|
|
|
|
std::string ErrorMessage;
|
|
|
|
Compilations.reset(
|
|
|
|
!BuildPath.empty() ?
|
|
|
|
CompilationDatabase::autoDetectFromDirectory(BuildPath, ErrorMessage) :
|
|
|
|
CompilationDatabase::autoDetectFromSource(SourcePaths[0],
|
|
|
|
ErrorMessage));
|
|
|
|
if (!Compilations)
|
|
|
|
llvm::report_fatal_error(ErrorMessage);
|
|
|
|
}
|
|
|
|
ClangTool SyntaxTool(*Compilations, SourcePaths);
|
|
|
|
|
|
|
|
// First, let's check to make sure there were no errors.
|
|
|
|
if (int result =
|
|
|
|
SyntaxTool.run(newFrontendActionFactory<clang::SyntaxOnlyAction>())) {
|
|
|
|
llvm::errs() << "Error compiling files.\n";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
RefactoringTool LoopTool(*Compilations, SourcePaths);
|
|
|
|
StmtAncestorASTVisitor ParentFinder;
|
|
|
|
StmtGeneratedVarNameMap GeneratedDecls;
|
|
|
|
ReplacedVarsMap ReplacedVars;
|
|
|
|
unsigned AcceptedChanges = 0;
|
|
|
|
unsigned DeferredChanges = 0;
|
|
|
|
unsigned RejectedChanges = 0;
|
|
|
|
|
|
|
|
MatchFinder Finder;
|
|
|
|
LoopFixer ArrayLoopFixer(&ParentFinder, &LoopTool.getReplacements(),
|
|
|
|
&GeneratedDecls, &ReplacedVars, &AcceptedChanges,
|
|
|
|
&DeferredChanges, &RejectedChanges,
|
|
|
|
CountOnly, TransformationLevel, LFK_Array);
|
|
|
|
Finder.addMatcher(makeArrayLoopMatcher(), &ArrayLoopFixer);
|
|
|
|
LoopFixer IteratorLoopFixer(&ParentFinder, &LoopTool.getReplacements(),
|
|
|
|
&GeneratedDecls, &ReplacedVars, &AcceptedChanges,
|
|
|
|
&DeferredChanges, &RejectedChanges,
|
|
|
|
CountOnly, TransformationLevel, LFK_Iterator);
|
|
|
|
Finder.addMatcher(makeIteratorLoopMatcher(), &IteratorLoopFixer);
|
|
|
|
LoopFixer PseudoarrrayLoopFixer(&ParentFinder, &LoopTool.getReplacements(),
|
|
|
|
&GeneratedDecls, &ReplacedVars,
|
|
|
|
&AcceptedChanges, &DeferredChanges,
|
|
|
|
&RejectedChanges, CountOnly,
|
|
|
|
TransformationLevel, LFK_PseudoArray);
|
|
|
|
Finder.addMatcher(makePseudoArrayLoopMatcher(), &PseudoarrrayLoopFixer);
|
|
|
|
if (int result = LoopTool.run(newFrontendActionFactory(&Finder))) {
|
|
|
|
llvm::errs() << "Error encountered during translation.\n";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::outs() << "\nFor Loop Conversion:\n\t" << AcceptedChanges
|
|
|
|
<< " converted loop(s)\n\t" << DeferredChanges
|
|
|
|
<< " potentially conflicting change(s) deferred.\n\t"
|
|
|
|
<< RejectedChanges << " change(s) rejected.\n";
|
|
|
|
if (DeferredChanges > 0)
|
|
|
|
llvm::outs() << "Re-run this tool to attempt applying deferred changes.\n";
|
|
|
|
if (RejectedChanges > 0)
|
|
|
|
llvm::outs() << "Re-run this tool with a lower required confidence level "
|
|
|
|
"to apply rejected changes.\n";
|
|
|
|
|
|
|
|
if (AcceptedChanges > 0) {
|
|
|
|
// Check to see if the changes introduced any new errors.
|
|
|
|
ClangTool EndSyntaxTool(*Compilations, SourcePaths);
|
|
|
|
if (int result = EndSyntaxTool.run(
|
|
|
|
newFrontendActionFactory<clang::SyntaxOnlyAction>())) {
|
|
|
|
llvm::errs() << "Error compiling files after translation.\n";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|