在一些LLVM教程中,我看到将C函数绑定到基于LLVM的自定义语言变得非常简单. LLVM将程序员指向一个可以与LLVM生成的代码混合的指针.
C库最好的方法是什么?假设我有一个相当复杂的库,如Qt或Boost,我想绑定到我的自定义语言.我需要创建一个存根库(如Python或Lua要求),还是LLVM提供某种外部函数接口(FFI)?
解决方法
在我的LLVM代码中,我为此创建了extern“C”包装函数,并将LLVM函数声明插入到模块中以便调用它们.那么,使LLVM知道函数的一个好方法是不要让它使用dlopen并在执行的二进制文件中搜索函数名(这是一个痛苦的屁股,因为函数名需要在.dynsym部分,而且也很慢),而是手动进行映射,使用
ExecutionEngine::addGlobalMapping.
只需得到该声明的llvm :: Function *,并将C函数中的函数地址转换为void *,并将这两个东西传递给LLVM.执行你的东西的JIT然后将知道在哪里找到该功能.
例如,如果要包装QString,您可以创建几个创建,销毁和调用这样一个对象的函数的函数
extern "C" void createQString(void *p,char const*v) { new (p) QString(v); // placement-new } extern "C" int32_t countQString(void *p) { QString *q = static_cast<QString*>(p); return q->count(); } extern "C" void destroyQString(void *p) { QString *q = static_cast<QString*>(p); q->~QString(); }
并创建适当的声明和映射.然后,您可以调用这些函数,沿着一个适合对齐的内存区域传递,并为QString(可能分配)设置一个大小,并指定一个指向C字符串数据进行初始化的i8 *.