Create environment module

This commit is contained in:
2025-08-10 02:05:16 +01:00
parent a03ef58eca
commit fade9395fa
6 changed files with 172 additions and 1 deletions

View File

@@ -1,5 +1,6 @@
add_library(imp
am.c
env.c
memory_stream.c
parse.c
store.c

80
lib/env.c Normal file
View File

@@ -0,0 +1,80 @@
#include "env.h"
#include <assert.h>
#include <string.h>
static bool symbol_eq(symbol_t *s, symbol_t *t)
{
return s->len == t->len && memcmp(s->buf, t->buf, s->len) == 0;
}
static expr_t **lookup(am_t *am, bool *found)
{
assert(am->expr != NULL);
assert(am->expr->is_atom);
assert(am->expr->atom.type == ATOM_TYPE_SYMBOL);
if (am->env->is_atom) {
assert(am->env->atom.type == ATOM_TYPE_EMPTY_LIST);
*found = false;
return &am->env;
}
expr_t *prev;
for (expr_t *list = am->env; !list->is_atom; list = list->pair.cdr) {
assert(list != NULL);
expr_t *entry = list->pair.car;
assert(!entry->is_atom);
expr_t *key = entry->pair.car;
assert(key != NULL);
assert(key->is_atom);
assert(key->atom.type == ATOM_TYPE_SYMBOL);
if (symbol_eq(&am->expr->atom.symbol, &key->atom.symbol)) {
*found = true;
return &entry->pair.cdr;
}
prev = list;
}
assert(!prev->is_atom);
assert(prev->pair.cdr->is_atom);
assert(prev->pair.cdr->atom.type == ATOM_TYPE_EMPTY_LIST);
*found = false;
return &prev->pair.cdr;
}
void env_init(am_t *am, store_t *store)
{
am->env = store_alloc(store);
am->env->is_atom = true;
am->env->atom.type = ATOM_TYPE_EMPTY_LIST;
}
void env_fetch(am_t *am)
{
bool found;
expr_t *val = *lookup(am, &found);
am->val = found ? val : NULL;
}
void env_set(am_t *am, store_t *store)
{
bool found;
expr_t **loc = lookup(am, &found);
if (found) {
*loc = am->val;
} else {
(*loc)->is_atom = false;
(*loc)->pair.cdr = store_alloc(store);
(*loc)->pair.cdr->is_atom = true;
(*loc)->pair.cdr->atom.type = ATOM_TYPE_EMPTY_LIST;
expr_t *entry = (*loc)->pair.car = store_alloc(store);
entry->pair.car = am->expr;
entry->pair.cdr = am->val;
}
}

View File

@@ -6,7 +6,7 @@
#define AM_STACK_SIZE 128U
typedef struct {
expr_t *expr;
expr_t *env, *expr, *val;
expr_t **sp, *stack[AM_STACK_SIZE];
} am_t;

11
lib/include/env.h Normal file
View File

@@ -0,0 +1,11 @@
#ifndef ENV_H
#define ENV_H
#include "am.h"
#include "store.h"
void env_init(am_t *am, store_t *store);
void env_fetch(am_t *am);
void env_set(am_t *am, store_t *store);
#endif