Инструменты Clang, как генерировать выходные данные препроцессора из модифицированных буферов? - PullRequest
0 голосов
/ 12 июня 2019

Я делаю преобразование источника в источник, используя инструменты Clang.В настоящее время я могу сгенерировать изменить исходный код и сохранить измененные файлы.Вот функция, которую я использую для вывода измененного файла.

rewriter.getEditBuffer(sm.getMainFileID()).write(llvm::outs());

// I can also save all modified files at once
rewriter.overwriteChangedFiles();

Однако вместо сохранения измененных файлов я хотел выдать вывод препроцессора.Я хотел добиться чего-то вроде кода ниже.

rewriter.emitPreprocessorOutput(rewriter.getBuffers());

Любые советы о том, как это сделать?

Вот небольшой пример кода ссылки, который я использую.


#include <sstream>
#include <iostream>
#include "clang/Driver/Options.h"
#include "clang/AST/AST.h"
#include "clang/AST/ASTContext.h"
#include "clang/AST/ASTConsumer.h"
#include "clang/AST/RecursiveASTVisitor.h"
#include "clang/Frontend/ASTConsumers.h"
#include "clang/Frontend/FrontendActions.h"
#include "clang/Frontend/CompilerInstance.h"
#include "clang/Tooling/CommonOptionsParser.h"
#include "clang/Tooling/Tooling.h"
#include "clang/Rewrite/Core/Rewriter.h"
#include "clang/AST/DeclCXX.h"
#include "llvm/ADT/Sequence.h"
#include <iostream>
#include <iterator>
#include <string>


using namespace std;
using namespace clang;
using namespace clang::driver;
using namespace clang::tooling;
using namespace llvm;


Rewriter rewriter;


class ExampleVisitor : public RecursiveASTVisitor<ExampleVisitor> {
private:
    ASTContext *astContext;

public:
    explicit ExampleVisitor(CompilerInstance *CI) : astContext(&(CI->getASTContext())) {}

    // Custom modifier visitors
    // ...
};



class ExampleASTConsumer : public ASTConsumer {
private:
    ExampleVisitor *visitor; 

public:

    explicit ExampleASTConsumer(CompilerInstance *CI) : visitor(new ExampleVisitor(CI)) { }
    virtual void HandleTranslationUnit(ASTContext &Context) {
        visitor->TraverseDecl(Context.getTranslationUnitDecl());
    }

};

class ExampleFrontendAction : public ASTFrontendAction {
public:
    virtual std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI, StringRef file) {
        rewriter.setSourceMgr(CI.getSourceManager(), CI.getLangOpts());
        return std::unique_ptr<clang::ASTConsumer>(new ExampleASTConsumer(&CI)); // pass CI pointer to ASTConsumer
    }

    void EndSourceFileAction() override {
        SourceManager &sm = rewriter.getSourceMgr();
        rewriter.InsertTextBefore(sm.getLocForStartOfFile(sm.getMainFileID()), "// This file has been modified");
        rewriter.getEditBuffer(sm.getMainFileID()).write(llvm::outs());
    }
};

static llvm::cl::OptionCategory ReplaceTypeCategory("Source-to-source options");

int main(int argc, const char **argv) {
    CommonOptionsParser op(argc, argv, ReplaceTypeCategory);        
    ClangTool Tool(op.getCompilations(), op.getSourcePathList());
    int result = Tool.run(newFrontendActionFactory<ExampleFrontendAction>().get());
    return result;
}

Краткие ссылки

  • Я использую код ссылки из этого блога , вот полный код .
...