open Ast open Ast.IR1 open Ast.V1 open Baselib open Errors let analyze_value = function | Syntax.Void -> Void, Void_t | Syntax.Int n -> Int n, Int_t | Syntax.Bool b -> Bool b, Bool_t | Syntax.Str s -> Str s, Str_t ;; let rec analyze_expr env ua t = function | Syntax.Val v -> let checked_value, new_t = analyze_value v.value in if not (List.exists (fun t2 -> List.mem t2 [ new_t; Magic_t ]) t) then errt t [ new_t ] v.pos; Val checked_value, new_t | Syntax.Var v -> if not (Env.mem v.name env) then raise (SemanticsError ("Unbound variable \"" ^ v.name ^ "\"", v.pos)); if List.mem v.name ua then warn ("Unassigned variable \"" ^ v.name ^ "\"") v.pos; let new_t = Env.find v.name env in if not (List.mem new_t t) then errt t [ new_t ] v.pos; Var v.name, new_t | Syntax.Call c -> if not (Env.mem c.func env) then raise (SemanticsError ("Unbound function \"" ^ c.func ^ "\"", c.pos)); (match Env.find c.func env with | Func_t (ret_t, t_list) -> if not (List.exists (fun t2 -> List.mem t2 [ ret_t; Magic_t ]) t) then errt [ ret_t ] t c.pos; if List.length t_list != List.length c.args then raise (SemanticsError ( Printf.sprintf "Function \"%s\" expects %d arguments but %d was given" c.func (List.length t_list) (List.length c.args) , c.pos )); let args = List.map2 (fun t2 e -> let e2, new_t = analyze_expr env ua [ t2 ] e in if new_t = t2 then e2 else errt [ t2 ] [ new_t ] (match e with | Syntax.Val v -> v.pos | Syntax.Var v -> v.pos | Syntax.Call c -> c.pos)) t_list c.args in Call (c.func, args), ret_t | _ -> raise (SemanticsError ("\"" ^ c.func ^ "\" isn't a function", c.pos))) ;; let rec analyze_instr env ua ret_t = function | Syntax.Decl d -> Decl d.name, Env.add d.name d.type_t env, [ d.name ] @ ua | Syntax.Assign a -> if not (Env.mem a.var env) then raise (SemanticsError ("Unbound variable \"" ^ a.var ^ "\"", a.pos)); let checked_expr, _ = analyze_expr env ua [ Env.find a.var env ] a.expr in Assign (a.var, checked_expr), env, List.filter (fun x -> x <> a.var) ua | Syntax.Do d -> let checked_expr, _ = analyze_expr env ua [ Magic_t ] d.expr in Do checked_expr, env, [] | Syntax.Cond c -> let cond, _ = analyze_expr env ua [ Bool_t; Int_t ] c.expr in let if_b, _ = analyze_block env ua Magic_t c.pos c.if_b in let else_b, _ = analyze_block env ua Magic_t c.pos c.else_b in Cond (cond, if_b, else_b), env, [] | Syntax.Loop l -> let cond, _ = analyze_expr env ua [ Bool_t; Int_t ] l.expr in let block, _ = analyze_block env ua Magic_t l.pos l.block in Loop (cond, block), env, [] | Syntax.Return r -> let checked_expr, _ = analyze_expr env ua [ ret_t ] r.expr in Return checked_expr, env, [] and analyze_block env ua ret_t pos = function | [] -> if ret_t != Void_t && ret_t != Magic_t then warn "Non-void function without return" pos; [], ua | instr :: new_block -> let new_instr, new_env, new_ua = analyze_instr env ua ret_t instr in (match new_instr with | Return _ -> [ new_instr ], new_ua | _ -> let new_block, new_ua2 = analyze_block new_env new_ua ret_t pos new_block in new_instr :: new_block, new_ua2) ;; let analyze_func env ua = function | Syntax.Func f -> let add_fn = let rec add_args env2 = function | [] -> env2 | h :: t -> (match h with | Syntax.Arg a -> add_args (Env.add a.name a.type_t env2) t) in Env.add f.func (Func_t ( f.type_t , List.map (fun a -> match a with | Syntax.Arg a -> a.type_t) f.args )) (add_args env f.args) in let block, _ = analyze_block add_fn ua f.type_t f.pos f.code in ( Func ( f.func , List.map (fun a -> match a with | Syntax.Arg a -> a.name) f.args , block ) , Env.add f.func (Func_t ( f.type_t , List.map (fun a -> match a with | Syntax.Arg a -> a.type_t) f.args )) env ) ;; let rec analyze_prog env ua default = function | [] -> if fst default then [] else raise (SemanticsError ("No " ^ snd default ^ " function", Lexing.dummy_pos)) | fn :: suite -> let fn, new_env = analyze_func env ua fn in let main_lbl = snd default in fn :: analyze_prog new_env ua (if fst default then default else Env.mem main_lbl new_env, main_lbl) suite ;; let analyze parsed = analyze_prog _types_ [] (false, "main") parsed