spice/lib/compile/ir.ml

240 lines
6.0 KiB
OCaml

module Ast = Spice_syntax.Ast
module Value = Spice_runtime.Value
exception Error of string
let compile_error f =
Fmt.kstr (fun msg -> raise (Error msg)) f
type imm = Value.t
type id = Id of int [@@unboxed]
type uop =
| Not
and bop =
| Add
| Sub
| Mul
| Div
| Mod
| Eql
| Grt
| Lst
type exp =
| Lit of Value.t
| Var of id
| Get of path
| Set of path * exp
| Let of id * exp * exp
| Seq of exp * exp
| If of exp * exp * exp
| Uop of uop * exp
| Bop of bop * exp * exp
| Call of path * exp list
| Obj of obj
and path = id * string
and obj = {
vals : string list;
funs : (string * lambda) list;
(* clos : id list; *)
}
and lambda = {
self : id;
args : id list;
body : exp;
}
(* lower *)
let make_id_dispenser () =
let i = ref (-1) in fun () -> (incr i; Id !i)
module Env = struct
type t =
| Empty
| Cons of t * t
| Args of (string * id) list
| Obj of {
self : id;
elems : string list;
}
let rec find name = function
| Empty -> raise Not_found
| Args args ->
List.assoc name args, None
| Cons (e1, e2) ->
begin
try find name e1 with
Not_found -> find name e2
end
| Obj { self; elems } ->
if List.mem name elems then
self, Some name
else
raise Not_found
end
let seq_r a b = Seq (b, a)
let lower ~lib (modl : Ast.modl) =
let new_id = make_id_dispenser () in
let rec lower_exp env = function
| Ast.Literal (Int n) -> Lit (Int n)
| Ast.Literal True -> Lit True
| Ast.Literal False -> Lit False
| Ast.Literal Nil -> Lit Nil
| Ast.Path path ->
lower_path env path
(function
| `Var id -> Var id
| `Get (obj, fld) -> Get (obj, fld))
| Ast.Binop (op, e1, e2) ->
let not e = Uop (Not, e) in
let bop, uop = match op with
| Ast.Add -> Add, Fun.id
| Ast.Sub -> Sub, Fun.id
| Ast.Mul -> Mul, Fun.id
| Ast.Div -> Div, Fun.id
| Ast.Mod -> Mod, Fun.id
| Ast.Eql -> Eql, Fun.id
| Ast.Grt -> Grt, Fun.id
| Ast.Lst -> Lst, Fun.id
| Ast.Not_eql -> Eql, not
| Ast.Grt_eql -> Lst, not
| Ast.Lst_eql -> Grt, not
in
uop (Bop (bop, lower_exp env e1, lower_exp env e2))
| Ast.Call (fn, args) ->
lower_path env fn
(fun fn ->
let fn_path = match fn with
| `Var _ -> failwith "Ir.lower_exp: TODO(fcf calls)"
| `Get (obj, mth) -> (obj, mth)
in
let args = List.map (lower_exp env) args in
Call (fn_path, args))
| Ast.If (e1, e2, e3) ->
If (lower_exp env e1, lower_exp env e2, lower_exp env e3)
| Ast.Obj items ->
lower_block env items
| Ast.Scope items ->
lower_block env items ~is_scope:true
| Ast.Fun (_, _) ->
failwith "Ir.lower_exp: TODO(Fun)"
and lower_path env path k =
match path with
| Ast.Ele (obj, fld) ->
let rhs = lower_exp env obj in
let lhs = new_id () in
Let (lhs, rhs, k (`Get (lhs, fld)))
| Ast.Var name ->
match Env.find name env with
| id, None -> k (`Var id)
| obj, Some fld -> k (`Get (obj, fld))
| exception Not_found ->
compile_error "unbound variable %S" name
and lower_block ?(is_scope = false) env items =
(* collect names of bindings to form the new environment; also check if a scope ends
with an expression, if not then it is an error *)
let elems, ends_with_exp =
List.fold_left
(fun (elems, _) -> function
| Ast.Item_exp _ -> elems, true
| Ast.Item_val (name, _)
| Ast.Item_obj (name, _)
| Ast.Item_fun (name, _, _) -> name :: elems, false)
([], false)
items
in
if is_scope && not ends_with_exp then
compile_error "scope does not end in expression";
(* build environment for field initializers; NOT for lambda capture *)
let self = new_id () in
let env_in = Env.Cons (Obj { self; elems }, env) in
let funs_r, vals_r, inits_r =
List.fold_left
(fun (fns, vls, ins) -> function
| Ast.Item_exp exp ->
let init = lower_exp env_in exp in
fns, vls, init :: ins
| Ast.Item_val (name, exp) ->
let init = Set ((self, name), lower_exp env_in exp) in
fns, name :: vls, init :: ins
| Ast.Item_obj (name, items) ->
(* TODO: it would be ideal if we could construct the empty versions of obj's
in a sort of "pre-init" phase, before assigning field values. but for now,
obj items are identical to val's where the rhs is an obj expression. *)
let init = Set ((self, name), lower_block env_in items) in
fns, name :: vls, init :: ins
| Ast.Item_fun (name, args, body) ->
let fn = name, compile_lambda env args body in
fn :: fns, vls, ins)
([], [], [])
items
in
(* TODO: closure conversion *)
(* if [is_scope], return the last expr, otherwise return the object itself *)
let ret, inits_r = match is_scope, inits_r with
| true, init :: inits -> init, inits
| _, inits -> Var self, inits
in
(* reverse order of inits and decls since they are cons'ed backwards *)
Let (
self,
Obj {
funs = List.rev funs_r;
vals = List.rev vals_r;
},
List.fold_left
(fun a b -> Seq (b, a))
ret
inits_r
)
and compile_lambda env args body =
let self = new_id () in
if args <> [] then
failwith "Ir.compile_lambda: TODO(args non-empty)";
(* FIXME: capture environment *)
let env = ignore env; Env.Empty in
let args = [] in
let body = lower_exp env body in
{ self; args; body }
in
let self = new_id () in
let env =
(* TODO: lib entries *)
let _ = lib in
Env.Empty
in
let args = [] in
let body = lower_block env modl.items in
{ self; args; body }