forked from OSchip/llvm-project
cpp11-migrate: New mechanism for overriding file contents
Next step toward supporting migrating headers. Instead of using ClangTool's ability to override all files at once, use a custom FrontendAction and override only the source (and eventually headers) the action is about to parse. Use of newFrontendActionFactory() is replaced with a new factory maker provided by Transform. llvm-svn: 183855
This commit is contained in:
parent
31c60f740e
commit
a6bbcdd707
|
@ -37,12 +37,6 @@ int AddOverrideTransform::apply(const FileContentsByPath &InputStates,
|
|||
FileContentsByPath &ResultStates) {
|
||||
RefactoringTool AddOverrideTool(Database, SourcePaths);
|
||||
|
||||
for (FileContentsByPath::const_iterator I = InputStates.begin(),
|
||||
E = InputStates.end();
|
||||
I != E; ++I) {
|
||||
AddOverrideTool.mapVirtualFile(I->first, I->second);
|
||||
}
|
||||
|
||||
unsigned AcceptedChanges = 0;
|
||||
|
||||
MatchFinder Finder;
|
||||
|
@ -53,8 +47,8 @@ int AddOverrideTransform::apply(const FileContentsByPath &InputStates,
|
|||
|
||||
Finder.addMatcher(makeCandidateForOverrideAttrMatcher(), &Fixer);
|
||||
|
||||
if (int result = AddOverrideTool.run(
|
||||
newFrontendActionFactory(&Finder, /*Callbacks=*/ this))) {
|
||||
if (int result =
|
||||
AddOverrideTool.run(createActionFactory(Finder, InputStates))) {
|
||||
llvm::errs() << "Error encountered during translation.\n";
|
||||
return result;
|
||||
}
|
||||
|
|
|
@ -1,11 +1,79 @@
|
|||
#include "Core/Transform.h"
|
||||
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
||||
#include "clang/Basic/FileManager.h"
|
||||
#include "clang/Basic/SourceManager.h"
|
||||
#include "clang/Frontend/CompilerInstance.h"
|
||||
#include "clang/Rewrite/Core/Rewriter.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
|
||||
using namespace clang;
|
||||
|
||||
namespace {
|
||||
|
||||
using namespace tooling;
|
||||
using namespace ast_matchers;
|
||||
|
||||
/// \brief Custom FrontendActionFactory to produce FrontendActions that handle
|
||||
/// overriding source file contents before parsing.
|
||||
///
|
||||
/// The nested class FactoryAdaptor overrides BeginSourceFileAction to override
|
||||
/// source file contents before parsing happens. Both Begin and
|
||||
/// EndSourceFileAction call corresponding callbacks provided by
|
||||
/// SourceFileCallbacks.
|
||||
class ActionFactory : public clang::tooling::FrontendActionFactory {
|
||||
public:
|
||||
ActionFactory(MatchFinder &Finder, const FileContentsByPath &Overrides,
|
||||
SourceFileCallbacks &Callbacks)
|
||||
: Finder(Finder), Overrides(Overrides), Callbacks(Callbacks) {}
|
||||
|
||||
virtual FrontendAction *create() LLVM_OVERRIDE {
|
||||
return new FactoryAdaptor(Finder, Overrides, Callbacks);
|
||||
}
|
||||
|
||||
private:
|
||||
class FactoryAdaptor : public ASTFrontendAction {
|
||||
public:
|
||||
FactoryAdaptor(MatchFinder &Finder, const FileContentsByPath &Overrides,
|
||||
SourceFileCallbacks &Callbacks)
|
||||
: Finder(Finder), Overrides(Overrides), Callbacks(Callbacks) {}
|
||||
|
||||
ASTConsumer *CreateASTConsumer(CompilerInstance &, StringRef) {
|
||||
return Finder.newASTConsumer();
|
||||
}
|
||||
|
||||
virtual bool BeginSourceFileAction(CompilerInstance &CI,
|
||||
StringRef Filename) LLVM_OVERRIDE {
|
||||
if (!ASTFrontendAction::BeginSourceFileAction(CI, Filename))
|
||||
return false;
|
||||
|
||||
FileContentsByPath::const_iterator I = Overrides.find(Filename.str());
|
||||
if (I != Overrides.end())
|
||||
// If an override exists, use it.
|
||||
CI.getSourceManager()
|
||||
.overrideFileContents(CI.getFileManager().getFile(I->first),
|
||||
llvm::MemoryBuffer::getMemBuffer(I->second));
|
||||
|
||||
return Callbacks.handleBeginSource(CI, Filename);
|
||||
}
|
||||
|
||||
virtual void EndSourceFileAction() LLVM_OVERRIDE {
|
||||
Callbacks.handleEndSource();
|
||||
return ASTFrontendAction::EndSourceFileAction();
|
||||
}
|
||||
|
||||
private:
|
||||
MatchFinder &Finder;
|
||||
const FileContentsByPath &Overrides;
|
||||
SourceFileCallbacks &Callbacks;
|
||||
};
|
||||
|
||||
MatchFinder &Finder;
|
||||
const FileContentsByPath &Overrides;
|
||||
SourceFileCallbacks &Callbacks;
|
||||
};
|
||||
|
||||
} // namespace
|
||||
|
||||
void collectResults(clang::Rewriter &Rewrite,
|
||||
const FileContentsByPath &InputStates,
|
||||
FileContentsByPath &Results) {
|
||||
|
@ -55,3 +123,9 @@ void Transform::handleEndSource() {
|
|||
void Transform::addTiming(llvm::StringRef Label, llvm::TimeRecord Duration) {
|
||||
Timings.push_back(std::make_pair(Label.str(), Duration));
|
||||
}
|
||||
|
||||
FrontendActionFactory *
|
||||
Transform::createActionFactory(MatchFinder &Finder,
|
||||
const FileContentsByPath &InputStates) {
|
||||
return new ActionFactory(Finder, InputStates, *this);
|
||||
}
|
||||
|
|
|
@ -49,6 +49,9 @@ namespace clang {
|
|||
namespace tooling {
|
||||
class CompilationDatabase;
|
||||
} // namespace tooling
|
||||
namespace ast_matchers {
|
||||
class MatchFinder;
|
||||
} // namespace ast_matchers
|
||||
} // namespace clang
|
||||
|
||||
/// \brief The key is the path of a file, which is mapped to a
|
||||
|
@ -233,6 +236,13 @@ protected:
|
|||
|
||||
const TransformOptions &Options() { return GlobalOptions; }
|
||||
|
||||
/// \brief Subclasses call this function to create a FrontendActionFactory to
|
||||
/// pass to ClangTool. The factory returned by this function is responsible
|
||||
/// for overriding source file contents with results of previous transforms.
|
||||
clang::tooling::FrontendActionFactory *
|
||||
createActionFactory(clang::ast_matchers::MatchFinder &Finder,
|
||||
const FileContentsByPath &InputStates);
|
||||
|
||||
private:
|
||||
const std::string Name;
|
||||
const TransformOptions &GlobalOptions;
|
||||
|
|
|
@ -31,12 +31,6 @@ int LoopConvertTransform::apply(const FileContentsByPath &InputStates,
|
|||
FileContentsByPath &ResultStates) {
|
||||
RefactoringTool LoopTool(Database, SourcePaths);
|
||||
|
||||
for (FileContentsByPath::const_iterator I = InputStates.begin(),
|
||||
E = InputStates.end();
|
||||
I != E; ++I) {
|
||||
LoopTool.mapVirtualFile(I->first, I->second);
|
||||
}
|
||||
|
||||
StmtAncestorASTVisitor ParentFinder;
|
||||
StmtGeneratedVarNameMap GeneratedDecls;
|
||||
ReplacedVarsMap ReplacedVars;
|
||||
|
@ -63,8 +57,7 @@ int LoopConvertTransform::apply(const FileContentsByPath &InputStates,
|
|||
Options().MaxRiskLevel, LFK_PseudoArray);
|
||||
Finder.addMatcher(makePseudoArrayLoopMatcher(), &PseudoarrrayLoopFixer);
|
||||
|
||||
if (int result = LoopTool.run(
|
||||
newFrontendActionFactory(&Finder, /*Callbacks=*/ this))) {
|
||||
if (int result = LoopTool.run(createActionFactory(Finder, InputStates))) {
|
||||
llvm::errs() << "Error encountered during translation.\n";
|
||||
return result;
|
||||
}
|
||||
|
|
|
@ -26,11 +26,6 @@ int UseAutoTransform::apply(const FileContentsByPath &InputStates,
|
|||
FileContentsByPath &ResultStates) {
|
||||
RefactoringTool UseAutoTool(Database, SourcePaths);
|
||||
|
||||
for (FileContentsByPath::const_iterator I = InputStates.begin(),
|
||||
E = InputStates.end();
|
||||
I != E; ++I)
|
||||
UseAutoTool.mapVirtualFile(I->first, I->second);
|
||||
|
||||
unsigned AcceptedChanges = 0;
|
||||
|
||||
MatchFinder Finder;
|
||||
|
@ -42,8 +37,7 @@ int UseAutoTransform::apply(const FileContentsByPath &InputStates,
|
|||
Finder.addMatcher(makeIteratorDeclMatcher(), &ReplaceIterators);
|
||||
Finder.addMatcher(makeDeclWithNewMatcher(), &ReplaceNew);
|
||||
|
||||
if (int Result = UseAutoTool.run(
|
||||
newFrontendActionFactory(&Finder, /*Callbacks=*/ this))) {
|
||||
if (int Result = UseAutoTool.run(createActionFactory(Finder, InputStates))) {
|
||||
llvm::errs() << "Error encountered during translation.\n";
|
||||
return Result;
|
||||
}
|
||||
|
|
|
@ -31,12 +31,6 @@ int UseNullptrTransform::apply(const FileContentsByPath &InputStates,
|
|||
FileContentsByPath &ResultStates) {
|
||||
RefactoringTool UseNullptrTool(Database, SourcePaths);
|
||||
|
||||
for (FileContentsByPath::const_iterator I = InputStates.begin(),
|
||||
E = InputStates.end();
|
||||
I != E; ++I) {
|
||||
UseNullptrTool.mapVirtualFile(I->first, I->second);
|
||||
}
|
||||
|
||||
unsigned AcceptedChanges = 0;
|
||||
|
||||
MatchFinder Finder;
|
||||
|
@ -46,8 +40,8 @@ int UseNullptrTransform::apply(const FileContentsByPath &InputStates,
|
|||
|
||||
Finder.addMatcher(makeCastSequenceMatcher(), &Fixer);
|
||||
|
||||
if (int result = UseNullptrTool.run(
|
||||
newFrontendActionFactory(&Finder, /*Callbacks=*/ this))) {
|
||||
if (int result =
|
||||
UseNullptrTool.run(createActionFactory(Finder, InputStates))) {
|
||||
llvm::errs() << "Error encountered during translation.\n";
|
||||
return result;
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue