我正在編寫代碼來使用LLVM Go bindings從自定義VM字節碼生成LLVM字節碼;該代碼然後被JIT編譯並在進程中執行。JIT的LLVM代碼如何回調到Go函數?
自定義VM字節碼有幾個操作無法直接在LLVM中實現,因爲它們需要更改外部狀態;這些操作碼的功能實現爲Go功能。
有關於從Go生成LLVM字節碼入門的excellent guides,但沒有解決回調或導出函數的問題。是否有可能生成LLVM指令來回調Go功能?如果是這樣,怎麼樣?
我試過了@arrowd下面描述的方式,它看起來不起作用。源代碼,改編自費利克斯·安格爾的博客文章:
package main
import (
"C"
"fmt"
"llvm.org/llvm/final/bindings/go/llvm"
)
// export AddInts
func AddInts(arg1, arg2 int) int {
return arg1 + arg2;
}
func main() {
// setup our builder and module
builder := llvm.NewBuilder()
mod := llvm.NewModule("my_module")
// create our function prologue
main := llvm.FunctionType(llvm.Int32Type(), []llvm.Type{}, false)
llvm.AddFunction(mod, "main", main)
block := llvm.AddBasicBlock(mod.NamedFunction("main"), "entry")
builder.SetInsertPoint(block, block.FirstInstruction())
// int a = 32
a := builder.CreateAlloca(llvm.Int32Type(), "a")
builder.CreateStore(llvm.ConstInt(llvm.Int32Type(), 32, false), a)
// int b = 16
b := builder.CreateAlloca(llvm.Int32Type(), "b")
builder.CreateStore(llvm.ConstInt(llvm.Int32Type(), 16, false), b)
// return a + b
bVal := builder.CreateLoad(b, "b_val")
aVal := builder.CreateLoad(a, "a_val")
addIntsType := llvm.FunctionType(llvm.Int32Type(), []llvm.Type{llvm.Int32Type(), llvm.Int32Type()}, false)
addInts := llvm.AddFunction(mod, "AddInts", addIntsType)
call := builder.CreateCall(addInts, []llvm.Value{aVal, bVal}, "AddInts")
builder.CreateRet(call)
// verify it's all good
if ok := llvm.VerifyModule(mod, llvm.ReturnStatusAction); ok != nil {
fmt.Println(ok.Error())
}
mod.Dump()
// create our exe engine
engine, err := llvm.NewExecutionEngine(mod)
if err != nil {
fmt.Println(err.Error())
}
// run the function!
funcResult := engine.RunFunction(mod.NamedFunction("main"), []llvm.GenericValue{})
fmt.Printf("%d\n", funcResult.Int(false))
}
返回:
; ModuleID = 'my_module'
define i32 @main() {
entry:
%a = alloca i32
store i32 32, i32* %a
%b = alloca i32
store i32 16, i32* %b
%b_val = load i32* %b
%a_val = load i32* %a
%AddInts = call i32 @AddInts(i32 %a_val, i32 %b_val)
ret i32 %AddInts
}
declare i32 @AddInts(i32, i32)
LLVM ERROR: Tried to execute an unknown external function: AddInts
exit status 1
Haskell提供了一種將代碼導出爲C函數的方法,因此可以從任何東西(包括LLVM)中調用Haskell。 Go有這樣的功能嗎? – arrowd
@arrowd我的理解是,CGo提供了這一點(而本機Go實現沒有,因爲它使用自己的調用約定)。這是朝着正確方向邁出的一步,但我仍然不確定如何從LLVM實際調用該功能。 –