schala/src/compilation.rs

177 lines
5.0 KiB
Rust
Raw Normal View History

2016-02-09 23:52:57 -08:00
extern crate llvm_sys;
use self::llvm_sys::prelude::*;
use self::llvm_sys::core;
use std::ptr;
use std::ffi::CString;
2016-02-10 03:25:37 -08:00
2016-02-15 23:41:00 -08:00
use parser::{ParseResult, AST, ASTNode, Prototype, Function, Expression};
2016-02-10 03:25:37 -08:00
2016-12-21 03:18:30 -08:00
mod LLVMWrap {
extern crate llvm_sys;
use self::llvm_sys::prelude::*;
use self::llvm_sys::core;
use std::ptr;
2016-12-25 11:19:17 -08:00
use std::ffi::CString;
2016-12-22 18:40:01 -08:00
pub fn create_context() -> LLVMContextRef {
2016-12-21 03:18:30 -08:00
unsafe {
core::LLVMContextCreate()
}
}
2016-12-22 18:40:01 -08:00
pub fn module_create_with_name(name: &str) -> LLVMModuleRef {
2016-12-21 03:18:30 -08:00
unsafe {
let n = name.as_ptr() as *const _;
core::LLVMModuleCreateWithName(n)
}
}
2016-12-21 17:50:13 -08:00
pub fn CreateBuilderInContext(context: LLVMContextRef) -> LLVMBuilderRef {
unsafe {
core::LLVMCreateBuilderInContext(context)
}
}
2016-12-25 11:19:17 -08:00
pub fn AppendBasicBlockInContext(context: LLVMContextRef, function: LLVMValueRef, name: &str) -> LLVMBasicBlockRef {
let c_name = CString::new(name).unwrap();
unsafe {
core::LLVMAppendBasicBlockInContext(context, function, c_name.as_ptr())
}
}
pub fn AddFunction(module: LLVMModuleRef, name: &str, function_type: LLVMTypeRef) -> LLVMValueRef {
let c_name = CString::new(name).unwrap();
unsafe {
core::LLVMAddFunction(module, c_name.as_ptr(), function_type)
}
}
2016-12-25 12:05:24 -08:00
//NOTE this is incomplete
pub fn FunctionType(return_type: LLVMTypeRef, param_types: &[LLVMTypeRef], is_var_rag: bool) -> LLVMTypeRef {
let len = param_types.len();
2016-12-25 12:05:24 -08:00
unsafe {
core::LLVMFunctionType(return_type, ptr::null_mut(), len as u32, if is_var_rag { 1 } else { 0 })
2016-12-25 12:05:24 -08:00
}
}
pub fn VoidTypeInContext(context: LLVMContextRef) -> LLVMTypeRef {
unsafe {
core::LLVMVoidTypeInContext(context)
}
}
2016-12-21 03:18:30 -08:00
pub fn DisposeBuilder(builder: LLVMBuilderRef) {
unsafe {
core::LLVMDisposeBuilder(builder)
}
}
pub fn DisposeModule(module: LLVMModuleRef) {
unsafe {
core::LLVMDisposeModule(module)
}
}
pub fn ContextDispose(context: LLVMContextRef) {
unsafe {
core::LLVMContextDispose(context)
}
}
pub fn PositionBuilderAtEnd(builder: LLVMBuilderRef, basic_block: LLVMBasicBlockRef) {
unsafe {
core::LLVMPositionBuilderAtEnd(builder, basic_block)
}
}
pub fn BuildRet(builder: LLVMBuilderRef, val: LLVMValueRef) -> LLVMValueRef {
unsafe {
core::LLVMBuildRet(builder, val)
}
}
pub fn BuildRetVoid(builder: LLVMBuilderRef) -> LLVMValueRef {
unsafe {
core::LLVMBuildRetVoid(builder)
}
}
pub fn DumpModule(module: LLVMModuleRef) {
unsafe {
core::LLVMDumpModule(module)
}
}
pub fn Int64TypeInContext(context: LLVMContextRef) -> LLVMTypeRef {
unsafe {
core::LLVMInt64TypeInContext(context)
}
}
pub fn ConstInt(int_type: LLVMTypeRef, n: u64, sign_extend: bool) -> LLVMValueRef {
unsafe {
core::LLVMConstInt(int_type, n, if sign_extend { 1 } else { 0 })
}
}
}
2016-12-25 12:05:24 -08:00
2016-03-04 14:32:22 -08:00
pub fn compile_ast(ast: AST) {
println!("Compiling!");
2016-12-22 18:40:01 -08:00
let context = LLVMWrap::create_context();
let module = LLVMWrap::module_create_with_name("example module");
2016-12-21 17:50:13 -08:00
let builder = LLVMWrap::CreateBuilderInContext(context);
2016-03-04 14:32:22 -08:00
//let void = LLVMWrap::VoidTypeInContext(context);
2016-12-25 12:05:24 -08:00
let int_type = LLVMWrap::Int64TypeInContext(context);
let function_type = LLVMWrap::FunctionType(int_type, &Vec::new(), false);
let function = LLVMWrap::AddFunction(module, "main", function_type);
2016-12-25 11:19:17 -08:00
let bb = LLVMWrap::AppendBasicBlockInContext(context, function, "entry");
LLVMWrap::PositionBuilderAtEnd(builder, bb);
let int_value: u64 = 84;
let int_value = LLVMWrap::ConstInt(int_type, int_value, false);
LLVMWrap::BuildRet(builder, int_value);
unsafe {
let out_file = CString::new("out.ll").unwrap();
core::LLVMPrintModuleToFile(module, out_file.as_ptr(), ptr::null_mut());
}
// Clean up. Values created in the context mostly get cleaned up there.
LLVMWrap::DisposeBuilder(builder);
LLVMWrap::DisposeModule(module);
LLVMWrap::ContextDispose(context);
2016-03-04 14:32:22 -08:00
}
trait CodeGen {
fn codegen(&self, LLVMContextRef) -> LLVMValueRef;
2016-02-10 03:25:37 -08:00
}
impl CodeGen for ASTNode {
fn codegen(&self, context: LLVMContextRef) -> LLVMValueRef {
use self::ASTNode::*;
2016-02-11 10:49:45 -08:00
match self {
&ExprNode(ref expr) => expr.codegen(context),
&FuncNode(ref func) => unimplemented!(),
2016-02-11 10:49:45 -08:00
}
}
}
impl CodeGen for Expression {
fn codegen(&self, context: LLVMContextRef) -> LLVMValueRef {
use self::Expression::*;
2016-02-12 23:14:09 -08:00
match self {
2016-12-22 18:40:01 -08:00
&BinExp(ref op, ref left, ref right) => {
unimplemented!()
},
&Number(ref n) => {
unimplemented!()
},
_ => unimplemented!(),
2016-02-12 23:14:09 -08:00
}
}
}