aboutsummaryrefslogtreecommitdiff
path: root/src/reducer.c
diff options
context:
space:
mode:
authorMarvin Borner2023-02-13 16:52:38 +0100
committerMarvin Borner2023-02-13 16:52:38 +0100
commit4e106e78a98f7a241fc2681fefc0996a34207045 (patch)
tree0f504c180ca6d8759d0ec48aa5f7ac214d016c06 /src/reducer.c
parent373c4bdc9cc01e2986f518eccc54c9d3856b7d05 (diff)
Switched to HAMT and BDWGC
Diffstat (limited to 'src/reducer.c')
-rw-r--r--src/reducer.c243
1 files changed, 92 insertions, 151 deletions
diff --git a/src/reducer.c b/src/reducer.c
index 2a44fcc..546a550 100644
--- a/src/reducer.c
+++ b/src/reducer.c
@@ -1,9 +1,15 @@
+// Copyright (c) 2023, Marvin Borner <dev@marvinborner.de>
// based on the RKNL abstract machine
+
#include <stdlib.h>
#include <assert.h>
#include <stdio.h>
+#include <string.h>
#include <reducer.h>
+#include <gc.h>
+#include <murmur3.h>
+#include <hamt.h>
#include <term.h>
struct stack {
@@ -11,19 +17,9 @@ struct stack {
struct stack *next;
};
-#define STORE_SIZE 128 // hashmap // TODO: Optimize?
-struct store {
- struct stack *stack;
-};
-
-struct store_data {
- int key;
- void *data;
-};
-
struct closure {
struct term *term;
- struct store (*store)[STORE_SIZE];
+ HAMT store;
};
struct box {
@@ -41,7 +37,7 @@ struct conf {
union {
struct { // closure
struct term *term;
- struct store (*store)[STORE_SIZE];
+ HAMT store;
struct stack *stack;
} econf; // blue
@@ -63,91 +59,16 @@ static struct stack *stack_push(struct stack *stack, void *data)
struct stack *new = malloc(sizeof(*new));
new->data = data;
new->next = stack;
- printf("Pushing %p to %p => %p\n", data, (void *)stack, (void *)new);
return new;
}
static struct stack *stack_next(struct stack *stack)
{
- printf("Popping %p => %p\n", (void *)stack, (void *)stack->next);
return stack->next;
}
-// no deep cloning for now // TODO: deep clone needed?
-static struct stack *stack_clone(struct stack *stack)
-{
- struct stack *new = 0;
- struct stack *iterator = stack;
- while (iterator) {
- new = stack_push(new, iterator->data);
- iterator = stack_next(iterator);
- }
- return new;
-}
-
-// no deep cloning for now // TODO: deep clone needed?
-static struct store (*store_clone(struct store (*store)[STORE_SIZE]))[STORE_SIZE]
-{
- struct store(*new)[STORE_SIZE] =
- calloc(1, STORE_SIZE * sizeof(struct store));
-
- for (int key = 0; key < STORE_SIZE; key++) {
- struct store *list = &(*store)[key % STORE_SIZE];
- if (list->stack)
- (*new)[key].stack = stack_clone(list->stack);
- }
- return new;
-}
-
-static int store_replace(struct store (*store)[STORE_SIZE], int key, void *data)
-{
- struct store *elem = &(*store)[key % STORE_SIZE];
- struct stack *iterator = elem->stack;
- while (iterator) {
- struct store_data *keyed = iterator->data;
- if (keyed->key == key) {
- keyed->data = data;
- printf("Replaced %d\n", key);
- return 1;
- }
- iterator = stack_next(iterator);
- }
- return 0;
-}
-
-static struct store (*store_push(struct store (*store)[STORE_SIZE], int key,
- void *data))[STORE_SIZE]
-{
- printf("Storing %p with %d (%d)\n", data, key, key % STORE_SIZE);
- if (store_replace(store, key, data))
- return store;
- struct store *elem = &(*store)[key % STORE_SIZE];
- struct store_data *keyed = malloc(sizeof(*keyed));
- keyed->key = key;
- keyed->data = data;
- elem->stack = stack_push(elem->stack, keyed);
- return store;
-}
-
-static void *store_get(struct store (*store)[STORE_SIZE], int key)
-{
- printf("Wanting %d (%d)\n", key, key % STORE_SIZE);
- struct store *elem = &(*store)[key % STORE_SIZE];
- struct stack *iterator = elem->stack;
- while (iterator) {
- struct store_data *keyed = iterator->data;
- if (keyed->key == key) {
- printf("Got it: %p\n", keyed->data);
- return keyed->data;
- }
- iterator = stack_next(iterator);
- }
- printf("Couldn't find it. Anyways..\n");
- return 0; // => unbound variable
-}
-
-static void econf(struct conf *conf, struct term *term,
- struct store (*store)[STORE_SIZE], struct stack *stack)
+static void econf(struct conf *conf, struct term *term, HAMT store,
+ struct stack *stack)
{
conf->type = ECONF;
conf->u.econf.term = term;
@@ -162,12 +83,11 @@ static void cconf(struct conf *conf, struct stack *stack, struct term *term)
conf->u.cconf.term = term;
}
-static int transition_1(struct term **term, struct store (**store)[STORE_SIZE],
- struct stack **stack)
+static int transition_1(struct term **term, HAMT *store, struct stack **stack)
{
struct closure *closure = malloc(sizeof(*closure));
closure->term = (*term)->u.app.rhs;
- closure->store = store_clone(*store);
+ closure->store = *store;
struct term *app = new_term(APP);
app->u.app.lhs = new_term(VAR);
@@ -177,24 +97,19 @@ static int transition_1(struct term **term, struct store (**store)[STORE_SIZE],
*term = (*term)->u.app.lhs;
*store = *store;
*stack = stack_push(*stack, app);
+
return 0;
}
-static int transition_2(struct stack **stack, struct term **term,
- struct store (**store)[STORE_SIZE])
+static int transition_2(struct stack **stack, struct term **term, HAMT store)
{
struct box *box = malloc(sizeof(*box));
box->state = TODO;
box->term = 0;
- // TODO: necessary?
- struct term *abs = new_term(ABS);
- abs->u.abs.name = (*term)->u.abs.name;
- abs->u.abs.term = (*term)->u.abs.term;
-
struct closure *closure = malloc(sizeof(*closure));
- closure->term = abs;
- closure->store = store_clone(*store);
+ closure->term = *term;
+ closure->store = store;
struct cache *cache = malloc(sizeof(*cache));
cache->box = box;
@@ -204,17 +119,16 @@ static int transition_2(struct stack **stack, struct term **term,
*stack = *stack;
*term = new_term(CACHE);
(*term)->u.other = cache;
+
return 0;
}
-static int transition_3(struct term **term, struct store (**store)[STORE_SIZE],
- struct stack **stack, struct box *box)
+static int transition_3(struct term **term, HAMT *store, struct stack **stack,
+ struct box *box)
{
assert(box->term->type == CLOSURE);
struct closure *closure = box->term->u.other;
- print_term(closure->term);
- printf("\n");
struct cache *cache = malloc(sizeof(*cache));
cache->box = box;
@@ -226,6 +140,8 @@ static int transition_3(struct term **term, struct store (**store)[STORE_SIZE],
*term = closure->term;
*store = closure->store;
*stack = stack_push(*stack, cache_term);
+
+ free(closure);
return 0;
}
@@ -240,7 +156,6 @@ static int transition_4(struct stack **stack, struct term **term,
static int transition_5(struct stack **stack, struct term **term,
struct cache *cache)
{
- printf("Setting %p\n", (void *)cache->box);
cache->box->state = DONE;
cache->box->term = *term;
@@ -249,23 +164,21 @@ static int transition_5(struct stack **stack, struct term **term,
return 0;
}
-static int transition_6(struct term **term, struct store (**store)[STORE_SIZE],
- struct stack **stack, struct term *peek_term,
- struct closure *closure)
+static int transition_6(struct term **term, HAMT *store, struct stack **stack,
+ struct term *peek_term, struct closure *closure)
{
struct box *box = malloc(sizeof(*box));
box->state = TODO;
box->term = peek_term->u.app.rhs;
*term = closure->term->u.abs.term;
- *store = store_push(closure->store, closure->term->u.abs.name, box);
+ *store = hamt_pset(closure->store, &closure->term->u.abs.name, box);
*stack = stack_next(*stack);
return 0;
}
-static int transition_7(struct term **term, struct store (**store)[STORE_SIZE],
- struct stack **stack, struct box *box,
- struct closure *closure)
+static int transition_7(struct term **term, HAMT *store, struct stack **stack,
+ struct box *box, struct closure *closure)
{
int x = name_generator();
@@ -286,7 +199,8 @@ static int transition_7(struct term **term, struct store (**store)[STORE_SIZE],
abs->u.abs.term = new_term(VAR);
*term = closure->term->u.abs.term;
- *store = store_push(closure->store, closure->term->u.abs.name, var_box);
+ *store = hamt_pset(closure->store, (void *)&closure->term->u.abs.name,
+ var_box);
*stack = stack_push(*stack, cache_term);
*stack = stack_push(*stack, abs);
return 0;
@@ -300,8 +214,8 @@ static int transition_8(struct stack **stack, struct term **term,
return 0;
}
-static int transition_9(struct term **term, struct store (**store)[STORE_SIZE],
- struct stack **stack, struct term *peek_term)
+static int transition_9(struct term **term, HAMT *store, struct stack **stack,
+ struct term *peek_term)
{
struct closure *closure = peek_term->u.app.rhs->u.other;
@@ -339,41 +253,40 @@ static int transition_11(struct stack **stack, struct term **term,
return 0;
}
-static int transition_closure(struct conf *conf)
+static int transition_closure(struct conf *conf, int i,
+ void (*callback)(int, char))
{
struct term *term = conf->u.econf.term;
- struct store(*store)[STORE_SIZE] = conf->u.econf.store;
+ HAMT store = conf->u.econf.store;
struct stack *stack = conf->u.econf.stack;
- printf("ECONF: %p %p %p\n", (void *)term, (void *)store, (void *)stack);
int ret = 1;
switch (term->type) {
case APP: // (1)
- printf("(1)\n");
+ callback(i, '1');
ret = transition_1(&term, &store, &stack);
econf(conf, term, store, stack);
return ret;
case ABS: // (2)
- printf("(2)\n");
- ret = transition_2(&stack, &term, &store);
+ callback(i, '2');
+ ret = transition_2(&stack, &term, store);
cconf(conf, stack, term);
return ret;
case VAR:;
- struct box *box = store_get(store, term->u.var.name);
+ /* printf("%p\n", hamt_get); */
+ struct box *box = hamt_get(store, &term->u.var.name);
if (!box) {
box = malloc(sizeof(*box));
box->state = DONE;
- // TODO: Using term directly is probably fine
- box->term = new_term(VAR);
- box->term->u.var.name = term->u.var.name;
+ box->term = term;
}
if (box->state == TODO) { // (3)
- printf("(3)\n");
+ callback(i, '3');
ret = transition_3(&term, &store, &stack, box);
econf(conf, term, store, stack);
return ret;
} else if (box->state == DONE) { // (4)
- printf("(4)\n");
+ callback(i, '4');
ret = transition_4(&stack, &term, box);
cconf(conf, stack, term);
return ret;
@@ -386,11 +299,11 @@ static int transition_closure(struct conf *conf)
}
}
-static int transition_computed(struct conf *conf)
+static int transition_computed(struct conf *conf, int i,
+ void (*callback)(int, char))
{
struct stack *stack = conf->u.cconf.stack;
struct term *term = conf->u.cconf.term;
- printf("CCONF: %p %p\n", (void *)stack, (void *)term);
if (!stack) {
fprintf(stderr, "Invalid stack!\n");
return 1;
@@ -401,7 +314,7 @@ static int transition_computed(struct conf *conf)
struct cache *cache = peek_term->u.other;
struct term *cache_term = cache->term;
if (cache_term->type == VAR && !cache_term->u.var.name) {
- printf("(5)\n");
+ callback(i, '5');
ret = transition_5(&stack, &term, cache);
cconf(conf, stack, term);
return ret;
@@ -414,8 +327,8 @@ static int transition_computed(struct conf *conf)
struct closure *closure =
((struct cache *)term->u.other)->term->u.other;
if (closure->term->type == ABS) {
- printf("(6)\n");
- struct store(*store)[STORE_SIZE];
+ callback(i, '6');
+ HAMT store;
ret = transition_6(&term, &store, &stack, peek_term,
closure);
econf(conf, term, store, stack);
@@ -429,14 +342,14 @@ static int transition_computed(struct conf *conf)
((struct cache *)term->u.other)->term->u.other;
if (closure->term->type == ABS && box->state == TODO &&
!box->term) { // (7)
- printf("(7)\n");
- struct store(*store)[STORE_SIZE];
+ callback(i, '7');
+ HAMT store;
ret = transition_7(&term, &store, &stack, box, closure);
econf(conf, term, store, stack);
return ret;
}
if (closure->term->type == ABS && box->state == DONE) { // (8)
- printf("(8)\n");
+ callback(i, '8');
ret = transition_8(&stack, &term, box);
cconf(conf, stack, term);
return ret;
@@ -446,8 +359,8 @@ static int transition_computed(struct conf *conf)
peek_term->u.app.lhs->type == VAR &&
!peek_term->u.app.lhs->u.var.name &&
peek_term->u.app.rhs->type == CLOSURE) { // (9)
- printf("(9)\n");
- struct store(*store)[STORE_SIZE];
+ callback(i, '9');
+ HAMT store;
ret = transition_9(&term, &store, &stack, peek_term);
econf(conf, term, store, stack);
return ret;
@@ -455,7 +368,7 @@ static int transition_computed(struct conf *conf)
if (peek_term && peek_term->type == APP &&
peek_term->u.app.rhs->type == VAR &&
!peek_term->u.app.rhs->u.var.name) { // (10)
- printf("(10)\n");
+ callback(i, 'A');
ret = transition_10(&stack, &term, peek_term);
cconf(conf, stack, term);
return ret;
@@ -463,7 +376,7 @@ static int transition_computed(struct conf *conf)
if (peek_term && peek_term->type == ABS &&
peek_term->u.abs.term->type == VAR &&
!peek_term->u.abs.term->u.var.name) { // (11)
- printf("(11)\n");
+ callback(i, 'B');
ret = transition_11(&stack, &term, peek_term);
cconf(conf, stack, term);
return ret;
@@ -476,35 +389,63 @@ static int transition_computed(struct conf *conf)
return 1;
}
-static int transition(struct conf *conf)
+static int transition(struct conf *conf, int i, void (*callback)(int, char))
{
if (conf->type == ECONF) {
- return transition_closure(conf);
+ return transition_closure(conf, i, callback);
} else if (conf->type == CCONF) {
- return transition_computed(conf);
+ return transition_computed(conf, i, callback);
}
fprintf(stderr, "Invalid transition state %x\n", conf->type);
return 1;
}
-static struct conf *for_each_state(struct conf *conf)
+static struct conf *for_each_state(struct conf *conf, int i,
+ void (*callback)(int, char))
{
- int ret = transition(conf);
- return ret ? conf : for_each_state(conf);
+ int ret = 0;
+ while (!ret)
+ ret = transition(conf, i++, callback);
+ return conf;
}
-struct term *reduce(struct term *term)
+static int hash_var_equal(const void *lhs, const void *rhs)
{
+ /* return memcmp(lhs, rhs, sizeof(int)); */
+ return *(const int *)lhs != *(const int *)rhs;
+}
+
+static uint32_t hash_var(const void *key, const size_t gen)
+{
+ return murmur3_32((const uint8_t *)key, sizeof(int), gen);
+}
+
+static void nop(void *addr)
+{
+ (void)nop;
+}
+
+struct term *reduce(struct term *term, void (*callback)(int, char))
+{
+ /* GC_set_find_leak(1); */
+ /* GC_INIT(); */
+
struct stack stack = { 0 };
- struct store store[STORE_SIZE] = { 0 };
+
+ struct hamt_allocator allocator = { GC_malloc, GC_realloc, nop };
+ HAMT store = hamt_create(hash_var, hash_var_equal, &allocator);
+ /* HAMT store = hamt_create(hash_var, hash_var_equal, &hamt_allocator_default); */
struct conf conf = {
.type = ECONF,
.u.econf.term = term,
- .u.econf.store = &store,
+ .u.econf.store = store,
.u.econf.stack = &stack,
};
- for_each_state(&conf);
+ for_each_state(&conf, 0, callback);
assert(conf.type == CCONF);
- return duplicate_term(conf.u.cconf.term);
+
+ struct term *ret = duplicate_term(conf.u.cconf.term);
+ hamt_delete(store);
+ return ret;
}