在llvm教程和示例中,编译器通过这样的调用输出LLVM IR
return Builder.CreateAdd(L, R, "addtmp");
但许多口译员都是这样写的:
switch (opcode) {
case ADD:
result = L + R;
break;
...
如何在不需要重新实现LLVM IR中的每个操作码的情况下,如何提取这些代码片段以使用LLVM进行JIT?
答案 0 :(得分:9)
好的,首先获取所有代码片段并将它们重构为自己的函数。所以你的代码转到:
void addOpcode(uint32_t *result, uint32_t L, uint32_t R) {
*result = L + R;
}
switch (opcode) {
case ADD:
addOpcode(&result, L, R);
break;
....
好的,所以这样做后你的翻译应该仍然运行。现在获取所有新功能并将它们放在自己的文件中。现在使用llvm-gcc或clang编译该文件,而不是生成本机代码,使用"cpp" backend(-march -cpp)编译它。这将生成实例化编译单元的字节代码的C ++代码。您可以指定选项以将其限制为特定功能等。您可能希望使用“-cppgen module”。
现在回到你的解释器循环粘合在一起调用生成的C ++代码而不是直接执行原始代码,然后将它传递给一些优化器和本机代码生成器。关于JIT的Gratz ;-)您可以在几个LLVM项目中看到这样的示例,例如llvm-lua中的vm_ops。