vm and compiler: removed automatic currying, and several other modifications to the language
ci/woodpecker/push/debian Pipeline was successful
ci/woodpecker/push/fedora Pipeline was successful
ci/woodpecker/push/nix Pipeline was successful
ci/woodpecker/push/publish Pipeline was successful

This commit is contained in:
2026-05-10 17:23:42 +03:00
parent 947d2274bb
commit c9694af826
7 changed files with 84 additions and 69 deletions
+2 -2
View File
@@ -5,8 +5,8 @@
*)
open Types
let builtin_print (v : Types.value) =
print_endline (print_value v);
let builtin_print (v : Types.value ref list) =
List.iter (fun r -> print_endline (print_value !r)) v;
Types.Nil
let table = [|
+6 -6
View File
@@ -6,7 +6,7 @@ type value =
| Nil
| Cons of value * value
| Symbol of string
| Closure of int * value ref list
| Closure of int * int * value ref list
| Native of int (* This is basically a syscall, each ID represents a primitive operation
that should have a well-defined effect. These will be further detailed
in the language documentation
@@ -20,8 +20,8 @@ type instr =
| StoreGlobal of int
| MakeCons
| Pop (* discards top of stack *)
| Apply
| MakeClosure of int
| Apply of int (* arg count *)
| MakeClosure of int * int (* arg count, code pointer *)
| Jump of int
| JumpF of int (* jump if false. *)
| End
@@ -47,7 +47,7 @@ let rec print_value = function
| Nil -> p "'()"
| Cons (a, b) -> p "(%s . %s)" (print_value a) (print_value b)
| Symbol x -> p "'%s" x
| Closure (i, _) -> p "<closure %d>" i
| Closure (a, i, _) -> p "<closure of %d args at %d>" a i
| Native i -> p "<native %d>" i
@@ -59,8 +59,8 @@ let print_one = function
| StoreGlobal i -> p "STORE_GLOBAL %d\n" i
| MakeCons -> p "CONS\n"
| Pop -> p "POP\n"
| Apply -> p "APPLY\n"
| MakeClosure i -> p "MKCLOSURE %d\n" i
| Apply i -> p "APPLY %d\n" i
| MakeClosure (a, i) -> p "MKCLOSURE %d, %d\n" a i
| Jump i -> p "JMP %d\n" i
| JumpF i -> p "JMPF %d\n" i
| End -> p "END\n"
+13 -7
View File
@@ -16,6 +16,11 @@ let pop_one state =
match state.stack with
| v :: rest -> state.stack <- rest; v
| [] -> failwith ("VM error: cannot pop from empty stack! " )
let pop_args state count =
let rec aux acc i =
if i <= 0 then acc
else aux ((ref (pop_one state)) :: acc) (i - 1)
in aux [] count
let peek_one state =
match state.stack with
| v :: _ -> v
@@ -29,19 +34,20 @@ let trace state =
let env () = List.fold_left (fun acc x -> acc ^ " " ^ (Types.print_value !x)) "" state.env in
Printf.printf "%d: \n\tstack: [%s ]\n\tenv:[%s]\n" state.i (stack ()) (env ())
let rec do_apply state =
let rec do_apply state arg_count =
let cur_env = state.env in
let cur_i = state.i in
let arg = pop_one state in
let args = pop_args state arg_count in
let f = pop_one state in
match f with
| Closure (x, e) ->
| Closure (a, _, _) when a != arg_count -> failwith "Wrong argument count to function"
| Closure (_, x, e) ->
state.call_stack <- (cur_i, cur_env) :: state.call_stack;
state.i <- x;
state.env <- (ref arg) :: e;
state.env <- List.append args e;
interpret state
| Native x ->
push state (Native.table.(x) arg);
push state (Native.table.(x) args);
interpret state
| _ -> failwith "Cannot apply non-closure object"
@@ -60,8 +66,8 @@ and interpret state =
let car = pop_one state in
push state (Cons (car, cdr))
| Pop -> ignore (pop_one state) ; interpret state
| Apply -> do_apply state
| MakeClosure x -> push state (Closure (x, state.env)); interpret state
| Apply a -> do_apply state a
| MakeClosure (args, x) -> push state (Closure (args, x, state.env)); interpret state
| Jump target -> state.i <- target ; interpret state
| JumpF target ->
(match (pop_one state) with