はじめに
Go言語で利用するLLVM入門をRustで書き直してみたメモです。
全ソースはこちら。
Cargo.toml
[dependencies]
llvm-sys = "60"
main.rs
extern crate llvm_sys;
use llvm_sys::core::*;
use llvm_sys::target;
use llvm_sys::analysis::{LLVMVerifyModule, LLVMVerifierFailureAction};
use llvm_sys::execution_engine::*;
use std::ffi::CString;
use std::os::raw::{c_char};
/// Initialise LLVM
///
/// Makes sure that the parts of LLVM we are going to use are
/// initialised before we do anything with them.
fn initialise_llvm() {
unsafe {
if target::LLVM_InitializeNativeTarget() != 0 {
panic!("Could not initialise target");
}
if target::LLVM_InitializeNativeAsmPrinter() != 0 {
panic!("Could not initialise ASM Printer");
}
}
}
fn main() {
let llvm_error = 1;
let val1 = 32;
let val2 = 16;
initialise_llvm();
// setup our builder and module
let builder = unsafe { LLVMCreateBuilder() };
let mod_name = CString::new("my_module").unwrap();
let module = unsafe { LLVMModuleCreateWithName(mod_name.as_ptr()) };
// create our function prologue
let function_type = unsafe {
let mut param_types = [];
LLVMFunctionType(LLVMInt32Type(), param_types.as_mut_ptr(), param_types.len() as u32, 0)
};
let function_name = CString::new("main").unwrap();
let function = unsafe { LLVMAddFunction(module, function_name.as_ptr(), function_type) };
let entry_name = CString::new("entry").unwrap();
let entry_block = unsafe { LLVMAppendBasicBlock(function, entry_name.as_ptr()) };
unsafe { LLVMPositionBuilderAtEnd(builder, entry_block); }
// int a = 32
let a_name = CString::new("a").unwrap();
let a = unsafe { LLVMBuildAlloca(builder, LLVMInt32Type(), a_name.as_ptr()) };
unsafe { LLVMBuildStore(builder, LLVMConstInt(LLVMInt32Type(), val1, 0), a); }
// int b = 16
let b_name = CString::new("b").unwrap();
let b = unsafe { LLVMBuildAlloca(builder, LLVMInt32Type(), b_name.as_ptr()) };
unsafe { LLVMBuildStore(builder, LLVMConstInt(LLVMInt32Type(), val2, 0), b); }
// return a + b
let b_val_name = CString::new("b_val").unwrap();
let b_val = unsafe { LLVMBuildLoad(builder, b, b_val_name.as_ptr()) };
let a_val_name = CString::new("a_val").unwrap();
let a_val = unsafe { LLVMBuildLoad(builder, a, a_val_name.as_ptr()) };
let ab_val_name = CString::new("ab_val").unwrap();
unsafe {
let res = LLVMBuildAdd(builder, a_val, b_val, ab_val_name.as_ptr());
LLVMBuildRet(builder, res);
}
// verify it's all good
let mut error: *mut c_char = 0 as *mut c_char;
let ok = unsafe {
let buf: *mut *mut c_char = &mut error;
LLVMVerifyModule(module, LLVMVerifierFailureAction::LLVMReturnStatusAction, buf)
};
if ok == llvm_error {
let err_msg = unsafe { CString::from_raw(error).into_string().unwrap() };
panic!("cannot verify module '{:?}'.\nError: {}", mod_name, err_msg);
}
// Clean up the builder now that we are finished using it.
unsafe { LLVMDisposeBuilder(builder) }
// Dump the LLVM IR to stdout so we can see what we've created
unsafe { LLVMDumpModule(module) }
// create our exe engine
let mut engine: LLVMExecutionEngineRef = 0 as LLVMExecutionEngineRef;
let ok = unsafe {
error = 0 as *mut c_char;
let buf: *mut *mut c_char = &mut error;
let engine_ref: *mut LLVMExecutionEngineRef = &mut engine;
LLVMLinkInInterpreter();
LLVMCreateInterpreterForModule(engine_ref, module, buf)
};
if ok == llvm_error {
let err_msg = unsafe { CString::from_raw(error).into_string().unwrap() };
println!("Execution error: {}", err_msg);
}else{
// run the function!
let func_name = CString::new("main").unwrap();
let named_function = unsafe { LLVMGetNamedFunction(module, func_name.as_ptr()) };
let mut params = [];
let func_result = unsafe { LLVMRunFunction(engine, named_function, params.len() as u32, params.as_mut_ptr()) };
let result = unsafe{ LLVMGenericValueToInt(func_result, 0) };
println!("{} + {} = {}", val1, val2, result);
}
// Clean up the module after we're done with it.
unsafe { LLVMDisposeModule(module) }
}
LLVMCreateExecutionEngineForModuleは"Interpreter has not been linked in."というエラーが出るのでコメントアウト
追記(2018/03/14):"Interpreter has not been linked in."というエラーについて。
"Interpreter has not been linked in."というエラーは、libffiがリンクされていなかったせいで出ていたようだ。
( それ以外にも、 LLVMLinkInInterpreter()
を呼ぶのも必要だった。)
この問題は、llvm-sysに存在するもののようで、Not linking to libffi + typoというイシューでも言及されている。
上記のイシューでは、 LLVM_SYS_<LLVM VERSION>_FFI_WORKAROUND
という環境変数をセットするとリンクされるように書いてあるが、うまくいかなかった。
結局、 build.rs
を使って強制的にlibffiをリンクさせることでなんとか解決出来た。
fn main() {
println!("cargo:rustc-link-lib=dylib={}", "ffi");
}
環境によっては、こんなことをしなくてもリンクされるのかもしれない。
実行結果のビルド
上記のコードを実行して得られたコードを、'compiled.ll'として保存した場合。
$ llvm-as compiled.ll
$ lli compiled.bc
$ echo $?
とビルド&実行し、48と表示されればオッケー。
「32 + 16」を計算するだけで、これぐらいのコードになります。