- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我需要使用 LLVM/Clang 作为后端来实现一个将 C 代码编译为 eBPF 字节码的库。这些代码将从内存中读取,我也需要在内存中获取生成的汇编代码。
到目前为止,我已经能够使用以下代码编译为 LLVM IR:
#include <string>
#include <vector>
#include <clang/Frontend/CompilerInstance.h>
#include <clang/Basic/DiagnosticOptions.h>
#include <clang/Frontend/TextDiagnosticPrinter.h>
#include <clang/CodeGen/CodeGenAction.h>
#include <clang/Basic/TargetInfo.h>
#include <llvm/Support/TargetSelect.h>
using namespace std;
using namespace clang;
using namespace llvm;
int main() {
constexpr auto testCodeFileName = "test.cpp";
constexpr auto testCode = "int test() { return 2+2; }";
// Prepare compilation arguments
vector<const char *> args;
args.push_back(testCodeFileName);
// Prepare DiagnosticEngine
DiagnosticOptions DiagOpts;
TextDiagnosticPrinter *textDiagPrinter =
new clang::TextDiagnosticPrinter(errs(),
&DiagOpts);
IntrusiveRefCntPtr<clang::DiagnosticIDs> pDiagIDs;
DiagnosticsEngine *pDiagnosticsEngine =
new DiagnosticsEngine(pDiagIDs,
&DiagOpts,
textDiagPrinter);
// Initialize CompilerInvocation
CompilerInvocation *CI = new CompilerInvocation();
CompilerInvocation::CreateFromArgs(*CI, &args[0], &args[0] + args.size(), *pDiagnosticsEngine);
// Map code filename to a memoryBuffer
StringRef testCodeData(testCode);
unique_ptr<MemoryBuffer> buffer = MemoryBuffer::getMemBufferCopy(testCodeData);
CI->getPreprocessorOpts().addRemappedFile(testCodeFileName, buffer.get());
// Create and initialize CompilerInstance
CompilerInstance Clang;
Clang.setInvocation(CI);
Clang.createDiagnostics();
// Set target (I guess I can initialize only the BPF target, but I don't know how)
InitializeAllTargets();
const std::shared_ptr<clang::TargetOptions> targetOptions = std::make_shared<clang::TargetOptions>();
targetOptions->Triple = string("bpf");
TargetInfo *pTargetInfo = TargetInfo::CreateTargetInfo(*pDiagnosticsEngine,targetOptions);
Clang.setTarget(pTargetInfo);
// Create and execute action
// CodeGenAction *compilerAction = new EmitLLVMOnlyAction();
CodeGenAction *compilerAction = new EmitAssemblyAction();
Clang.ExecuteAction(*compilerAction);
buffer.release();
}
为了编译,我使用以下 CMakeLists.txt:
cmake_minimum_required(VERSION 3.3.2)
project(clang_backend CXX)
set(CMAKE_CXX_COMPILER "clang++")
execute_process(COMMAND llvm-config --cxxflags OUTPUT_VARIABLE LLVM_CONFIG OUTPUT_STRIP_TRAILING_WHITESPACE)
execute_process(COMMAND llvm-config --libs OUTPUT_VARIABLE LLVM_LIBS OUTPUT_STRIP_TRAILING_WHITESPACE)
set(CMAKE_CXX_FLAGS ${LLVM_CONFIG})
set(CLANG_LIBS clang clangFrontend clangDriver clangSerialization clangParse
clangCodeGen clangSema clangAnalysis clangEdit clangAST clangLex
clangBasic )
add_executable(clang_backend main.cpp)
target_link_libraries(clang_backend ${CLANG_LIBS})
target_link_libraries(clang_backend ${LLVM_LIBS})
如果我理解正确,如果我将编译器操作更改为 EmitAssemblyAction(),我应该能够生成汇编代码,但我可能没有初始化某些东西,因为我在 llvm::TargetPassConfig::中遇到段错误:在/tmp/llvm-3.7.1.src/lib/CodeGen/Passes.cpp:419 添加PassesToHandleExceptions (this=this@entry=0x6d8d30)
这一行的代码是:
switch (TM->getMCAsmInfo()->getExceptionHandlingType()) {
有没有人有例子或知道我错过了什么?
最佳答案
所以,如果你编译 LLVM 时启用断言,错误就更清楚了,它实际上会告诉你你需要做什么:
x: .../src/llvm/lib/CodeGen/LLVMTargetMachine.cpp:63:
void llvm::LLVMTargetMachine::initAsmInfo():
Assertion `TmpAsmInfo && "MCAsmInfo not initialized. "
"Make sure you include the correct TargetSelect.h"
"and that InitializeAllTargetMCs() is being invoked!"' failed.
(我添加了一些换行符,因为它打印为一个长行)。
在 main
的开头添加所需的 InitializeAllTargetMCs()
后,我又遇到了一个错误。查看我的编译器生成的目标文件,我“猜测”这是另一个 InitializeAll*
调用的问题。进行了一些测试,结果表明您还需要 InitializeAllAsmPrinters();
- 如果您要生成汇编代码,这是有道理的。
我不完全确定如何从您的代码中“查看”结果,但是将这两个添加到 main
的开头使其运行完成而不是断言、退出并出现错误或崩溃 - 这通常是朝着正确方向迈出的良好一步。
所以这是 main
在“我的”代码中的样子:
int main() {
constexpr auto testCodeFileName = "test.cpp";
constexpr auto testCode = "int test() { return 2+2; }";
InitializeAllTargetMCs();
InitializeAllAsmPrinters();
// Prepare compilation arguments
vector<const char *> args;
args.push_back(testCodeFileName);
// Prepare DiagnosticEngine
DiagnosticOptions DiagOpts;
TextDiagnosticPrinter *textDiagPrinter =
new clang::TextDiagnosticPrinter(errs(),
&DiagOpts);
IntrusiveRefCntPtr<clang::DiagnosticIDs> pDiagIDs;
DiagnosticsEngine *pDiagnosticsEngine =
new DiagnosticsEngine(pDiagIDs,
&DiagOpts,
textDiagPrinter);
// Initialize CompilerInvocation
CompilerInvocation *CI = new CompilerInvocation();
CompilerInvocation::CreateFromArgs(*CI, &args[0], &args[0] + args.size(), *pDiagnosticsEngine);
// Map code filename to a memoryBuffer
StringRef testCodeData(testCode);
unique_ptr<MemoryBuffer> buffer = MemoryBuffer::getMemBufferCopy(testCodeData);
CI->getPreprocessorOpts().addRemappedFile(testCodeFileName, buffer.get());
// Create and initialize CompilerInstance
CompilerInstance Clang;
Clang.setInvocation(CI);
Clang.createDiagnostics();
// Set target (I guess I can initialize only the BPF target, but I don't know how)
InitializeAllTargets();
const std::shared_ptr<clang::TargetOptions> targetOptions = std::make_shared<clang::TargetOptions>();
targetOptions->Triple = string("bpf");
TargetInfo *pTargetInfo = TargetInfo::CreateTargetInfo(*pDiagnosticsEngine,targetOptions);
Clang.setTarget(pTargetInfo);
// Create and execute action
// CodeGenAction *compilerAction = new EmitLLVMOnlyAction();
CodeGenAction *compilerAction = new EmitAssemblyAction();
Clang.ExecuteAction(*compilerAction);
buffer.release();
}
我强烈建议,如果您想使用 clang&LLVM 进行开发,请构建 Clang&LLVM 的调试版本 - 这将有助于追踪“原因”并及早发现问题以及更明显的问题。将 -DCMAKE_BUILD_TYPE=Debug
与 cmake
一起使用以获得那种 flavor 。
我用于构建 LLVM 和 Clang 的完整脚本:
export CC=clang
export CXX=clang++
cmake -DCMAKE_BUILD_TYPE=Debug -DCMAKE_INSTALL_PREFIX=/usr/local/llvm-debug -DLLVM_TAR
GETS_TO_BUILD=X86 ../llvm
[我使用的是 3.8 的后期预发布版本来测试它,但我非常怀疑它在这方面与 3.7.1 有很大不同]
关于c++ - 使用 libclang 从内存中的 C 代码生成程序集,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34828480/
如果我在一个大项目中有多个文件,所有文件共享大量包含的头文件,有什么办法可以共享解析头文件的工作吗?我曾希望创建一个索引,然后向其中添加多个 translationUnits 可能会导致一些工作被共享
如果我在一个大项目中有多个文件,所有这些文件共享大量包含的头文件,有没有办法分担解析头文件的工作?我曾希望创建一个 Index 然后向其添加多个 translationUnits 可能会导致一些工作被
我使用 libclang 来解析源文件并获取对某种类型的引用,如 CXType ,假设它是“const std::__1::basic_string ”(由 clang_getTypeSpelling
我正在使用最新的 LibClang 来解析一些 C 头文件。我处理的代码来自 CXUnsavedFile 的(它都是动态生成的,没有任何内容存在于磁盘上)。例如: FileA.h 包含: struct
有时,当我尝试在Xcode中构建项目时(大约十分之一),它崩溃并退出。然后,我得到一个错误框,其中显示:“使用libclang.dylib插件时Xcode意外退出”。 Xcode有时也会随机崩溃,没有
我使用的是 Linux Mint,并且使用 Clang Complete 中的 makefile 安装了 clang_complete ,但它不起作用。当我打开 cpp 文件时,出现错误消息: Loa
我正在尝试使用 libclang 构建一个小型解析程序。 要解析的源文件(Node.h): #pragma once struct Node { int value; struct N
Currentlty 我正在做一个项目,用 libclang 转储 C++ 代码的类信息。还有一些关于类型限定符的悲惨经历:const、volatile、&、&& 以及它们的组合。以下是转储函数删除参
有什么方法可以从 libclang 中获取信息,了解源文件中的 C++ 代码是否具有正确的语法?即使使用无效的 C++ 代码,Libclang 也会尝试创建翻译单元。 最佳答案 一般问题的一般答案是肯
如何使用 libclang 获取原始文字的值? 例如,如果我有一个游标类型为 CXCursor_IntegerLiteral 的 CXCursor,我该如何提取文字值。 更新: 我在使用 libcla
我有以下使用 clang-c API 的代码。 #include #include #include CXChildVisitResult printVisitor(CXCursor curso
我有一个 c++ 文件,它会更改,其余所有头文件保持不变。但是每当我重新解析一个翻译单元时,libclang 最终会消耗大量的 cpu 和 ram。虽然它使用预编译头和所有(我可以看到生成的前导文件)
class aclass{ public: int num; }; int main() { aclass *ok; ok->num = 4; return 0; }
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 4年前关闭。 Improve this qu
我正在使用 LibClang 列出所有函数调用及其相应的定义。 以下是执行此操作的python脚本: def traverse(node): if node.kind == CALL_EXPR:
我尝试使用 libClang 解析 C++ 方法,但在尝试获取函数的参数/参数时,有时会给出错误的类型。 例子: 我有两种不同的方法 std::string Method::exportMethod(
我尝试使用 libClang 解析 C++ 方法,但在尝试获取函数的参数/参数时,有时会给出错误的类型。 例子: 我有两种不同的方法 std::string Method::exportMethod(
我试图了解如何使用 libclang 完成代码。我看过“超越编译器的思考”,并且查看了 c-index-test,我发现了一个简单的示例程序 here 我编译了该程序,并在我制作的示例文件上运行它,该
我需要解析一个 C++ 代码文件并找到其中所有具有完全限定名称的函数调用。我正在使用 libclang 的 Python 绑定(bind),因为它似乎比编写我自己的 C++ 解析器更容易,即使文档很少
我正在学习使用 Python + libclang 解析 C++ 文件,并借助 Eli Bendersky 提供的这个信息量很大(但有点过时)的教程. 我的目标是解析 C++ 文件并识别这些文件中存在
我是一名优秀的程序员,十分优秀!