LLVM IR,简单的程序给出segfault

LLVM IR, simple program gives segfault

本文关键字:segfault 程序 IR 简单 LLVM      更新时间:2023-10-16

我在运行时生成了以下llvm ir,我希望立即执行函数

;other declarations/functions
define %gen__struct__81ty_struct.catch_val @gen__fun__115(void**) {
entry:
%1 = getelementptr void*, void** %0, i32 0
%2 = load void*, void** %1, align 8
%3 = bitcast void* %2 to i64*
store i64 1, i64* %3, align 8
%4 = load i64, i64* %3, align 8
%5 = getelementptr void*, void** %0, i32 1
%6 = load void*, void** %5, align 8
%7 = bitcast void* %6 to i64*
store i64 1, i64* %7, align 8
%8 = load i64, i64* %7, align 8
%9 = getelementptr void*, void** %0, i32 2
%10 = load void*, void** %9, align 8
%11 = bitcast void* %10 to i64*
store i64 1, i64* %11, align 8
ret %gen__struct__81ty_struct.catch_val zeroinitializer
}

然后,我使用万花筒jit和下面的C++代码编译并执行上面的函数,但我一直得到segfault。

Function* out = 0;
//code that generates the function ...
runFunc fptr = 0;
jit->addModule(std::move(gblDevice->lmod));
fptr = (runFunc)jit->getPointerToFunction(out);
vector<void*> params;
uint64_t a = 0;
uint64_t b = 0;
uint64_t c = 0;
params.push_back(&a);
params.push_back(&b);
params.push_back(&c);
catch_val_pt cval = fptr(&params[0]);

我已经试了几个小时来找出上面的代码有什么问题,但似乎一切都解决了。

getPointerToFunction函数为.

class OrcJIT {
// ...
JITSymbol findSymbol(const std::string Name) {
std::string MangledName;
raw_string_ostream MangledNameStream(MangledName);
Mangler::getNameWithPrefix(MangledNameStream, Name, DL);
return CODLayer.findSymbol(MangledNameStream.str(), true);
}
inline void* getPointerToFunction(Function* F) {
return (void*)findSymbol(F->getName().data()).getAddress();
}
//...
}

最终的答案非常简单。

llvm返回值和gcc返回值的方式之间必须存在二进制不兼容。简短的回答是gen__struct__81ty_struct.catch_val返回值导致segfault。

简单地说,在生成的函数中,您需要通过传递给生成函数的指针将其复制到结构中。除了最简单的情况(也称为int)外,您不能依赖生成的函数的返回值进行匹配。