2 个版本
0.2.1 | 2020 年 6 月 22 日 |
---|---|
0.2.0 | 2020 年 6 月 22 日 |
#530 in 编程语言
650KB
16K SLoC
Cilk
Rust 编写的受 LLVM 影响的玩具编译器基础设施。
不要期望太多功能!
待办事项
- 优化
- 简单
- 将寄存器溢出保存到调用者保存的寄存器,如 ebx,而不是堆栈。(llc 也是这样做的)
- 考虑物理寄存器的分配顺序。
- 困难
- ....
- 简单
- 精炼代码
- 详细编写文档
- 编写测试(因为我最近删除了大部分测试)
构建
要求:Rust nightly
cargo test --features x86_64 # build for x86_64
cargo test brainfuxk --features x86_64 -- --nocapture # this is fun. just try it.
cargo test --features riscv64 # build for riscv64. very few features are implemented.
示例
- 斐波那契(以下代码可能不起作用。请查看 ./tests)
use cilk::{
codegen::x64::{exec},
exec::interpreter::interp,
ir::{builder::Builder, module::Module, value::{Value, ImmediateValue}, types::Type},
};
let mut m = Module::new("cilk");
let fibo = m.create_function(
"fibo", Type::Int32, vec![Type::Int32]
);
let mut builder = Builder::new(&mut m, fibo);
{
let entry = builder.append_basic_block();
let br1 = builder.append_basic_block();
let br2 = builder.append_basic_block();
builder.set_insert_point(entry);
let arg0 = builder.get_param(0).unwrap();
let eq1 = builder.build_icmp(
ICmpKind::Le,
arg0,
Value::Immediate(ImmediateValue::Int32(2)),
);
builder.build_cond_br(eq1, br1, br2);
builder.set_insert_point(br1);
builder.build_ret(Value::Immediate(ImmediateValue::Int32(1)));
builder.set_insert_point(br2);
let fibo1arg = builder.build_sub(
arg0,
Value::Immediate(ImmediateValue::Int32(1)),
);
let fibo1 = builder.build_call(Value::Function(fibo), vec![fibo1arg]);
let fibo2arg = builder.build_sub(
arg0,
Value::Immediate(ImmediateValue::Int32(2)),
);
let fibo2 = builder.build_call(Value::Function(fibo), vec![fibo2arg]);
let add = builder.build_add(fibo1, fibo2);
builder.build_ret(add);
println!("Function dump:\n{}", m.dump(fibo));
}
// Function dump:
// define i32 fibo(i32) {
// label.0:
// %0 = icmp le i32 %arg.0, i32 2
// br i1 %0 %label.1, %label.2
//
// label.1:
// ret i32 1
//
// label.2:
// %3 = sub i32 %arg.0, i32 1
// %4 = call i32 fibo(i32 %3, )
// %5 = sub i32 %arg.0, i32 2
// %6 = call i32 fibo(i32 %5, )
// %7 = add i32 %4, i32 %6
// ret i32 %7
//
// }
// In this branch, this may not work correctly.
let mut interp = interp::Interpreter::new(&m);
let ret = interp.run_function(fibo, vec![interp::ConcreteValue::Int32(10)]);
println!("fibo(10) = {:?}", ret); // fibo(10) = Int32(55)
// JIT suppports for only x86_64
let mut jit = exec::jit::JITExecutor::new(&m);
let func = jit.find_function_by_name("func").unwrap();
println!( "fibo(10) = {:?}",
jit.run(func, vec![exec::jit::GenericValue::Int32(10)])); // fibo(10) = 55
- 有可用的宏来描述 IR
let fibo = cilk_ir!(m; define [i32] f (i32) {
entry:
cond = icmp le (%arg.0), (i32 2);
br (%cond) l1, l2;
l1:
ret (i32 1);
l2:
a1 = sub (%arg.0), (i32 1);
r1 = call f [(%a1)];
a2 = sub (%arg.0), (i32 2);
r2 = call f [(%a2)];
r3 = add (%r1), (%r2);
ret (%r3);
});
依赖关系
~3.5MB
~70K SLoC