xref: /aosp_15_r20/external/llvm/examples/OCaml-Kaleidoscope/Chapter4/codegen.ml (revision 9880d6810fe72a1726cb53787c6711e909410d58)
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