summaryrefslogtreecommitdiff
path: root/src/runtime.c
diff options
context:
space:
mode:
authorLLLL Colonq <llll@colonq>2025-09-21 02:56:07 -0400
committerLLLL Colonq <llll@colonq>2025-09-21 02:56:07 -0400
commit6530799bb08a1e2e367d12972fdd4210243ebb39 (patch)
tree59bda1bc25c7991d97ac5d2875afbef78eb36cf5 /src/runtime.c
parent85a67a25ac9757e694166b3c9e9e2c8cdeefc6da (diff)
Evaluation!
Diffstat (limited to 'src/runtime.c')
-rw-r--r--src/runtime.c110
1 files changed, 103 insertions, 7 deletions
diff --git a/src/runtime.c b/src/runtime.c
index ba5d131..abfd480 100644
--- a/src/runtime.c
+++ b/src/runtime.c
@@ -231,22 +231,41 @@ pit_value_heavy *pit_deref(pit_runtime *rt, pit_ref p) {
return pit_arena_idx(rt->values, p);
}
-bool pit_is_cons(pit_runtime *rt, pit_value a) {
+bool pit_is_integer(pit_runtime *rt, pit_value a) {
+ (void) rt;
+ return pit_value_sort(a) == PIT_VALUE_SORT_INTEGER;
+}
+bool pit_is_double(pit_runtime *rt, pit_value a) {
+ (void) rt;
+ return pit_value_sort(a) == PIT_VALUE_SORT_DOUBLE;
+}
+bool pit_is_symbol(pit_runtime *rt, pit_value a) {
+ (void) rt;
+ return pit_value_sort(a) == PIT_VALUE_SORT_SYMBOL;
+}
+bool pit_is_value_heavy_sort(pit_runtime *rt, pit_value a, enum pit_value_heavy_sort e) {
switch (pit_value_sort(a)) {
case PIT_VALUE_SORT_REF:
pit_value_heavy *ha = pit_deref(rt, a);
if (!ha) { pit_error(rt, "bad ref"); return false; }
- switch (ha->hsort) {
- case PIT_VALUE_HEAVY_SORT_CONS:
- return true;
- default:
- break;
- }
+ return ha->hsort == e;
default:
break;
}
return false;
}
+bool pit_is_cons(pit_runtime *rt, pit_value a) {
+ return pit_is_value_heavy_sort(rt, a, PIT_VALUE_HEAVY_SORT_CONS);
+}
+bool pit_is_array(pit_runtime *rt, pit_value a) {
+ return pit_is_value_heavy_sort(rt, a, PIT_VALUE_HEAVY_SORT_ARRAY);
+}
+bool pit_is_bytes(pit_runtime *rt, pit_value a) {
+ return pit_is_value_heavy_sort(rt, a, PIT_VALUE_HEAVY_SORT_BYTES);
+}
+bool pit_is_nativefunc(pit_runtime *rt, pit_value a) {
+ return pit_is_value_heavy_sort(rt, a, PIT_VALUE_HEAVY_SORT_NATIVEFUNC);
+}
bool pit_eq(pit_value a, pit_value b) {
return a == b;
}
@@ -427,3 +446,80 @@ pit_value pit_apply(pit_runtime *rt, pit_value f, pit_value args) {
return PIT_NIL;
}
}
+
+struct eval_stack {
+ i64 top, cap;
+ pit_value *data;
+};
+static struct eval_stack *eval_stack_new() {
+ struct eval_stack *ret = malloc(sizeof(*ret));
+ ret->top = 0;
+ ret->cap = 32;
+ ret->data = calloc(ret->cap, sizeof(pit_value));
+ return ret;
+}
+static void eval_stack_destroy(struct eval_stack *s) {
+ if (s) {
+ if (s->data) free(s->data);
+ free(s);
+ }
+}
+static void eval_stack_push(pit_runtime *rt, struct eval_stack *s, pit_value x) {
+ (void) rt;
+ s->data[s->top++] = x;
+ if (s->top >= s->cap) s->data = realloc(s->data, (s->cap <<= 1) * sizeof(pit_value));
+}
+static pit_value eval_stack_pop(pit_runtime *rt, struct eval_stack *s) {
+ if (s->top == 0) { pit_error(rt, "evaluation stack underflow"); return PIT_NIL; }
+ return s->data[--s->top];
+}
+
+pit_value pit_eval(pit_runtime *rt, pit_value top) {
+ struct eval_stack *expr_stack = eval_stack_new();
+ struct eval_stack *program = eval_stack_new();
+ eval_stack_push(rt, expr_stack, top);
+ // first, convert the expression tree into "polish notation" in program
+ while (expr_stack->top > 0) {
+ pit_value cur = eval_stack_pop(rt, expr_stack);
+ if (pit_is_cons(rt, cur)) { // compound expressions: function/macro application special forms
+ pit_value fsym = pit_car(rt, cur);
+ pit_value f = pit_fget(rt, fsym);
+ pit_value args = pit_cdr(rt, cur);
+ i64 argcount = 0;
+ while (args != PIT_NIL) {
+ eval_stack_push(rt, expr_stack, pit_car(rt, args));
+ args = pit_cdr(rt, args);
+ argcount += 1;
+ }
+ eval_stack_push(rt, program, pit_cons(rt, f, pit_integer_new(rt, argcount)));
+ } else if (pit_is_symbol(rt, cur)) { // unquoted symbols: variable lookup
+ eval_stack_push(rt, program, pit_get(rt, cur));
+ } else if (pit_is_double(rt, cur)) { // double literals
+ eval_stack_push(rt, program, cur);
+ } else if (pit_is_integer(rt, cur)) { // integer literals
+ eval_stack_push(rt, program, cur);
+ } else {
+ pit_error(rt, "unknown expression in eval");
+ goto end;
+ }
+ }
+ struct eval_stack *result_stack = eval_stack_new();
+ // then, execute the polish notation program from right to left
+ for (i64 idx = program->top - 1; idx >= 0; --idx) {
+ pit_value expr = program->data[idx];
+ if (pit_is_cons(rt, expr)) { // this is a function call
+ pit_value args = PIT_NIL;
+ for (i64 i = 0; i < pit_as_integer(rt, pit_cdr(rt, expr)); ++i) {
+ args = pit_cons(rt, eval_stack_pop(rt, result_stack), args);
+ }
+ eval_stack_push(rt, result_stack, pit_apply(rt, pit_car(rt, expr), args));
+ } else { // this is an atom
+ eval_stack_push(rt, result_stack, expr);
+ }
+ }
+end:
+ pit_value ret = eval_stack_pop(rt, result_stack);
+ eval_stack_destroy(expr_stack);
+ eval_stack_destroy(result_stack);
+ return ret;
+}