2016-02-09 23:52:57 -08:00
|
|
|
extern crate llvm_sys;
|
|
|
|
extern crate iron_llvm;
|
|
|
|
|
|
|
|
use std::collections::HashMap;
|
2016-02-11 10:49:45 -08:00
|
|
|
use self::llvm_sys::prelude::LLVMValueRef;
|
2016-02-10 03:25:37 -08:00
|
|
|
|
2016-02-11 10:49:45 -08:00
|
|
|
use self::iron_llvm::core;
|
|
|
|
use self::iron_llvm::core::types::{RealTypeCtor, RealTypeRef};
|
|
|
|
use self::iron_llvm::{LLVMRef, LLVMRefCtor};
|
|
|
|
|
2016-02-12 23:14:09 -08:00
|
|
|
use parser::{ParseResult, AST, ASTNode, Prototype, Expression};
|
2016-02-10 03:25:37 -08:00
|
|
|
|
|
|
|
pub struct Context {
|
|
|
|
context: core::Context,
|
|
|
|
builder: core::Builder,
|
|
|
|
named_values: HashMap<String, LLVMValueRef>,
|
|
|
|
ty: RealTypeRef,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Context {
|
|
|
|
pub fn new() -> Context {
|
|
|
|
let context = core::Context::get_global();
|
|
|
|
let builder = core::Builder::new();
|
|
|
|
let named_values = HashMap::new();
|
|
|
|
let ty = RealTypeRef::get_double();
|
|
|
|
Context {
|
|
|
|
context: context,
|
|
|
|
builder: builder,
|
|
|
|
named_values: named_values,
|
|
|
|
ty: ty,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait ModuleProvider {
|
|
|
|
fn dump(&self);
|
|
|
|
fn get_module(&mut self) -> &mut core::Module;
|
|
|
|
fn get_function(&mut self, name: &str) -> Option<(FunctionRef, bool)>;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct SimpleModuleProvider {
|
|
|
|
module: core::Module,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SimpleModuleProvider {
|
|
|
|
pub fn new(name: &str) -> SimpleModuleProvider {
|
|
|
|
let module = core::Module::new(name);
|
|
|
|
SimpleModuleProvider {
|
|
|
|
module: module,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ModuleProvider for SimpleModuleProvider {
|
|
|
|
fn dump(&self) {
|
|
|
|
self.module.dump();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_module(&mut self) -> &mut core::Module {
|
|
|
|
&mut self.module
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_function(&mut self, name: &str) -> Option<(FunctionRef, bool)> {
|
2016-02-11 10:49:45 -08:00
|
|
|
match self.module.get_function_by_name(name) {
|
2016-02-10 03:25:37 -08:00
|
|
|
Some(f) => Some((f, f.count_basic_block() > 0)),
|
|
|
|
None => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-11 10:49:45 -08:00
|
|
|
pub type IRBuildingResult = Result<(LLVMValueRef, bool), String>;
|
2016-02-10 03:25:37 -08:00
|
|
|
|
|
|
|
fn error(msg: &str) -> IRBuildingResult {
|
|
|
|
Err(msg.to_string())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait IRBuilder {
|
|
|
|
fn codegen(&self, context: &mut Context, module_provider: &mut ModuleProvider) -> IRBuildingResult;
|
|
|
|
}
|
2016-02-11 10:49:45 -08:00
|
|
|
|
|
|
|
impl IRBuilder for ParseResult<AST> {
|
|
|
|
fn codegen(&self, context: &mut Context, module_provider: &mut ModuleProvider) -> IRBuildingResult {
|
|
|
|
match self {
|
|
|
|
&Ok(ast) => ast.codegen(context, module_provider),
|
|
|
|
&Err(err) => Err(err.msg.clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IRBuilder for AST {
|
|
|
|
fn codegen(&self, context: &mut Context, module_provider: &mut ModuleProvider) -> IRBuildingResult {
|
|
|
|
let mut result = error("empty AST");
|
|
|
|
for node in self.iter() {
|
|
|
|
result = Ok(try!(node.codegen(context, module_provider)));
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
}
|
2016-02-12 23:14:09 -08:00
|
|
|
|
|
|
|
impl IRBuilder for ASTNode {
|
|
|
|
fn codegen(&self, context: &mut Context, module_provider: &mut ModuleProvider) -> IRBuildingResult {
|
|
|
|
match self {
|
|
|
|
&ASTNode::ExprNode(ref expression) => expression.codegen(context, module_provider),
|
|
|
|
&ASTNode::FuncNode(ref function) => function.codegen(context, module_provider),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IRBuilder for Function {
|
|
|
|
fn codegen(&self, context: &mut Context, module_provider: &mut ModuleProvider) -> IRBuildingResult {
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|