在使用LLVM编写JIT时,如何重用C操作码实现?

时间:2009-02-02 18:49:13

标签: compiler-construction jit llvm language-implementation

在llvm教程和示例中,编译器通过这样的调用输出LLVM IR

return Builder.CreateAdd(L, R, "addtmp");

但许多口译员都是这样写的:

switch (opcode) {
     case ADD:
             result = L + R;
             break;
     ...

如何在不需要重新实现LLVM IR中的每个操作码的情况下,如何提取这些代码片段以使用LLVM进行JIT?

1 个答案:

答案 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。