1*9880d681SAndroid Build Coastguard Worker(*===----------------------------------------------------------------------=== 2*9880d681SAndroid Build Coastguard Worker * Code Generation 3*9880d681SAndroid Build Coastguard Worker *===----------------------------------------------------------------------===*) 4*9880d681SAndroid Build Coastguard Worker 5*9880d681SAndroid Build Coastguard Workeropen Llvm 6*9880d681SAndroid Build Coastguard Worker 7*9880d681SAndroid Build Coastguard Workerexception Error of string 8*9880d681SAndroid Build Coastguard Worker 9*9880d681SAndroid Build Coastguard Workerlet context = global_context () 10*9880d681SAndroid Build Coastguard Workerlet the_module = create_module context "my cool jit" 11*9880d681SAndroid Build Coastguard Workerlet builder = builder context 12*9880d681SAndroid Build Coastguard Workerlet named_values:(string, llvalue) Hashtbl.t = Hashtbl.create 10 13*9880d681SAndroid Build Coastguard Workerlet double_type = double_type context 14*9880d681SAndroid Build Coastguard Worker 15*9880d681SAndroid Build Coastguard Workerlet rec codegen_expr = function 16*9880d681SAndroid Build Coastguard Worker | Ast.Number n -> const_float double_type n 17*9880d681SAndroid Build Coastguard Worker | Ast.Variable name -> 18*9880d681SAndroid Build Coastguard Worker (try Hashtbl.find named_values name with 19*9880d681SAndroid Build Coastguard Worker | Not_found -> raise (Error "unknown variable name")) 20*9880d681SAndroid Build Coastguard Worker | Ast.Binary (op, lhs, rhs) -> 21*9880d681SAndroid Build Coastguard Worker let lhs_val = codegen_expr lhs in 22*9880d681SAndroid Build Coastguard Worker let rhs_val = codegen_expr rhs in 23*9880d681SAndroid Build Coastguard Worker begin 24*9880d681SAndroid Build Coastguard Worker match op with 25*9880d681SAndroid Build Coastguard Worker | '+' -> build_fadd lhs_val rhs_val "addtmp" builder 26*9880d681SAndroid Build Coastguard Worker | '-' -> build_fsub lhs_val rhs_val "subtmp" builder 27*9880d681SAndroid Build Coastguard Worker | '*' -> build_fmul lhs_val rhs_val "multmp" builder 28*9880d681SAndroid Build Coastguard Worker | '<' -> 29*9880d681SAndroid Build Coastguard Worker (* Convert bool 0/1 to double 0.0 or 1.0 *) 30*9880d681SAndroid Build Coastguard Worker let i = build_fcmp Fcmp.Ult lhs_val rhs_val "cmptmp" builder in 31*9880d681SAndroid Build Coastguard Worker build_uitofp i double_type "booltmp" builder 32*9880d681SAndroid Build Coastguard Worker | _ -> raise (Error "invalid binary operator") 33*9880d681SAndroid Build Coastguard Worker end 34*9880d681SAndroid Build Coastguard Worker | Ast.Call (callee, args) -> 35*9880d681SAndroid Build Coastguard Worker (* Look up the name in the module table. *) 36*9880d681SAndroid Build Coastguard Worker let callee = 37*9880d681SAndroid Build Coastguard Worker match lookup_function callee the_module with 38*9880d681SAndroid Build Coastguard Worker | Some callee -> callee 39*9880d681SAndroid Build Coastguard Worker | None -> raise (Error "unknown function referenced") 40*9880d681SAndroid Build Coastguard Worker in 41*9880d681SAndroid Build Coastguard Worker let params = params callee in 42*9880d681SAndroid Build Coastguard Worker 43*9880d681SAndroid Build Coastguard Worker (* If argument mismatch error. *) 44*9880d681SAndroid Build Coastguard Worker if Array.length params == Array.length args then () else 45*9880d681SAndroid Build Coastguard Worker raise (Error "incorrect # arguments passed"); 46*9880d681SAndroid Build Coastguard Worker let args = Array.map codegen_expr args in 47*9880d681SAndroid Build Coastguard Worker build_call callee args "calltmp" builder 48*9880d681SAndroid Build Coastguard Worker 49*9880d681SAndroid Build Coastguard Workerlet codegen_proto = function 50*9880d681SAndroid Build Coastguard Worker | Ast.Prototype (name, args) -> 51*9880d681SAndroid Build Coastguard Worker (* Make the function type: double(double,double) etc. *) 52*9880d681SAndroid Build Coastguard Worker let doubles = Array.make (Array.length args) double_type in 53*9880d681SAndroid Build Coastguard Worker let ft = function_type double_type doubles in 54*9880d681SAndroid Build Coastguard Worker let f = 55*9880d681SAndroid Build Coastguard Worker match lookup_function name the_module with 56*9880d681SAndroid Build Coastguard Worker | None -> declare_function name ft the_module 57*9880d681SAndroid Build Coastguard Worker 58*9880d681SAndroid Build Coastguard Worker (* If 'f' conflicted, there was already something named 'name'. If it 59*9880d681SAndroid Build Coastguard Worker * has a body, don't allow redefinition or reextern. *) 60*9880d681SAndroid Build Coastguard Worker | Some f -> 61*9880d681SAndroid Build Coastguard Worker (* If 'f' already has a body, reject this. *) 62*9880d681SAndroid Build Coastguard Worker if block_begin f <> At_end f then 63*9880d681SAndroid Build Coastguard Worker raise (Error "redefinition of function"); 64*9880d681SAndroid Build Coastguard Worker 65*9880d681SAndroid Build Coastguard Worker (* If 'f' took a different number of arguments, reject. *) 66*9880d681SAndroid Build Coastguard Worker if element_type (type_of f) <> ft then 67*9880d681SAndroid Build Coastguard Worker raise (Error "redefinition of function with different # args"); 68*9880d681SAndroid Build Coastguard Worker f 69*9880d681SAndroid Build Coastguard Worker in 70*9880d681SAndroid Build Coastguard Worker 71*9880d681SAndroid Build Coastguard Worker (* Set names for all arguments. *) 72*9880d681SAndroid Build Coastguard Worker Array.iteri (fun i a -> 73*9880d681SAndroid Build Coastguard Worker let n = args.(i) in 74*9880d681SAndroid Build Coastguard Worker set_value_name n a; 75*9880d681SAndroid Build Coastguard Worker Hashtbl.add named_values n a; 76*9880d681SAndroid Build Coastguard Worker ) (params f); 77*9880d681SAndroid Build Coastguard Worker f 78*9880d681SAndroid Build Coastguard Worker 79*9880d681SAndroid Build Coastguard Workerlet codegen_func the_fpm = function 80*9880d681SAndroid Build Coastguard Worker | Ast.Function (proto, body) -> 81*9880d681SAndroid Build Coastguard Worker Hashtbl.clear named_values; 82*9880d681SAndroid Build Coastguard Worker let the_function = codegen_proto proto in 83*9880d681SAndroid Build Coastguard Worker 84*9880d681SAndroid Build Coastguard Worker (* Create a new basic block to start insertion into. *) 85*9880d681SAndroid Build Coastguard Worker let bb = append_block context "entry" the_function in 86*9880d681SAndroid Build Coastguard Worker position_at_end bb builder; 87*9880d681SAndroid Build Coastguard Worker 88*9880d681SAndroid Build Coastguard Worker try 89*9880d681SAndroid Build Coastguard Worker let ret_val = codegen_expr body in 90*9880d681SAndroid Build Coastguard Worker 91*9880d681SAndroid Build Coastguard Worker (* Finish off the function. *) 92*9880d681SAndroid Build Coastguard Worker let _ = build_ret ret_val builder in 93*9880d681SAndroid Build Coastguard Worker 94*9880d681SAndroid Build Coastguard Worker (* Validate the generated code, checking for consistency. *) 95*9880d681SAndroid Build Coastguard Worker Llvm_analysis.assert_valid_function the_function; 96*9880d681SAndroid Build Coastguard Worker 97*9880d681SAndroid Build Coastguard Worker (* Optimize the function. *) 98*9880d681SAndroid Build Coastguard Worker let _ = PassManager.run_function the_function the_fpm in 99*9880d681SAndroid Build Coastguard Worker 100*9880d681SAndroid Build Coastguard Worker the_function 101*9880d681SAndroid Build Coastguard Worker with e -> 102*9880d681SAndroid Build Coastguard Worker delete_function the_function; 103*9880d681SAndroid Build Coastguard Worker raise e 104