aboutsummaryrefslogtreecommitdiff
path: root/src/reducer.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/reducer.c')
-rw-r--r--src/reducer.c305
1 files changed, 238 insertions, 67 deletions
diff --git a/src/reducer.c b/src/reducer.c
index 546a550..c5d8a89 100644
--- a/src/reducer.c
+++ b/src/reducer.c
@@ -7,27 +7,35 @@
#include <string.h>
#include <reducer.h>
-#include <gc.h>
#include <murmur3.h>
-#include <hamt.h>
+#include <store.h>
#include <term.h>
+struct tracked {
+ int tracker;
+ void *stuff;
+};
+
struct stack {
+ int tracker;
void *data;
struct stack *next;
};
struct closure {
+ int tracker;
struct term *term;
- HAMT store;
+ struct store *store;
};
struct box {
+ int tracker;
enum { TODO, DONE } state;
struct term *term;
};
struct cache {
+ int tracker;
struct box *box;
struct term *term;
};
@@ -37,7 +45,7 @@ struct conf {
union {
struct { // closure
struct term *term;
- HAMT store;
+ struct store *store;
struct stack *stack;
} econf; // blue
@@ -54,9 +62,124 @@ static int name_generator(void)
return current++;
}
+static void *track(void *ptr)
+{
+ struct tracked *tracked = ptr;
+ tracked->tracker = 0;
+ return ptr;
+}
+
+static void release(void **ptr)
+{
+ struct tracked *tracked = *ptr;
+ tracked->tracker--;
+ if (tracked->tracker == 0) {
+ /* printf("Release %p (free)\n", tracked); */
+ free(*ptr);
+ } else {
+ /* printf("Release %p (%d)\n", tracked, tracked->tracker); */
+ }
+ /* *ptr = 0; // TODO: ? */
+}
+
+static void *acquire(void *ptr)
+{
+ struct tracked *tracked = ptr;
+ tracked->tracker++;
+ /* printf("Acquire %p (%d)\n", ptr, tracked->tracker); */
+ return ptr;
+}
+
+static void release_term(struct term **term)
+{
+ if (!(*term)) // e.g. for empty boxes
+ return;
+
+ struct term *deref = *term;
+ switch (deref->type) {
+ case ABS:
+ release_term(&deref->u.abs.term);
+ break;
+ case APP:
+ release_term(&deref->u.app.lhs);
+ release_term(&deref->u.app.rhs);
+ break;
+ case VAR:
+ break;
+ case CLOSURE:
+ release_term(&((struct closure *)deref->u.other)->term);
+ store_release(&((struct closure *)deref->u.other)->store);
+ release((void **)&deref->u.other);
+ break;
+ case CACHE:
+ release_term(&((struct cache *)deref->u.other)->box->term);
+ release((void **)&((struct cache *)deref->u.other)->box);
+ release_term(&((struct cache *)deref->u.other)->term);
+ release((void **)&deref->u.other);
+ break;
+ default:
+ fprintf(stderr, "Invalid type %d\n", deref->type);
+ }
+ release((void **)term);
+}
+
+static void release_box(struct box **box)
+{
+ release_term(&(*box)->term);
+ release((void **)box);
+}
+
+static void release_cache(struct cache **cache)
+{
+ release_box(&(*cache)->box);
+ release_term(&(*cache)->term);
+ release((void **)cache);
+}
+
+// expensive, only use when necessary
+static struct term *acquire_term(struct term *term)
+{
+ if (!term) // e.g. for empty boxes
+ return term;
+
+ acquire(term);
+ switch (term->type) {
+ case ABS:
+ acquire_term(term->u.abs.term);
+ break;
+ case APP:
+ acquire_term(term->u.app.lhs);
+ acquire_term(term->u.app.rhs);
+ break;
+ case VAR:
+ break;
+ case CLOSURE:
+ acquire(term->u.other);
+ acquire_term(((struct closure *)term->u.other)->term);
+ store_acquire(((struct closure *)term->u.other)->store);
+ break;
+ case CACHE:
+ acquire(term->u.other);
+ acquire(((struct cache *)term->u.other)->box);
+ acquire_term(((struct cache *)term->u.other)->box->term);
+ acquire_term(((struct cache *)term->u.other)->term);
+ break;
+ default:
+ fprintf(stderr, "Invalid type %d\n", term->type);
+ }
+ return term;
+}
+
+static struct box *acquire_box(struct box *box)
+{
+ acquire(box);
+ acquire_term(box->term);
+ return box;
+}
+
static struct stack *stack_push(struct stack *stack, void *data)
{
- struct stack *new = malloc(sizeof(*new));
+ struct stack *new = acquire(track(malloc(sizeof(*new))));
new->data = data;
new->next = stack;
return new;
@@ -64,10 +187,12 @@ static struct stack *stack_push(struct stack *stack, void *data)
static struct stack *stack_next(struct stack *stack)
{
- return stack->next;
+ struct stack *next = stack->next;
+ release((void **)&stack);
+ return next;
}
-static void econf(struct conf *conf, struct term *term, HAMT store,
+static void econf(struct conf *conf, struct term *term, struct store *store,
struct stack *stack)
{
conf->type = ECONF;
@@ -83,9 +208,12 @@ static void cconf(struct conf *conf, struct stack *stack, struct term *term)
conf->u.cconf.term = term;
}
-static int transition_1(struct term **term, HAMT *store, struct stack **stack)
+static int transition_1(struct term **term, struct store **store,
+ struct stack **stack)
{
- struct closure *closure = malloc(sizeof(*closure));
+ struct term *orig = *term;
+
+ struct closure *closure = track(malloc(sizeof(*closure)));
closure->term = (*term)->u.app.rhs;
closure->store = *store;
@@ -93,25 +221,28 @@ static int transition_1(struct term **term, HAMT *store, struct stack **stack)
app->u.app.lhs = new_term(VAR);
app->u.app.rhs = new_term(CLOSURE);
app->u.app.rhs->u.other = closure;
+ acquire_term(app);
- *term = (*term)->u.app.lhs;
+ *term = acquire_term((*term)->u.app.lhs);
*store = *store;
*stack = stack_push(*stack, app);
+ release_term(&orig);
return 0;
}
-static int transition_2(struct stack **stack, struct term **term, HAMT store)
+static int transition_2(struct stack **stack, struct term **term,
+ struct store *store)
{
- struct box *box = malloc(sizeof(*box));
+ struct box *box = track(malloc(sizeof(*box)));
box->state = TODO;
box->term = 0;
- struct closure *closure = malloc(sizeof(*closure));
+ struct closure *closure = track(malloc(sizeof(*closure)));
closure->term = *term;
closure->store = store;
- struct cache *cache = malloc(sizeof(*cache));
+ struct cache *cache = track(malloc(sizeof(*cache)));
cache->box = box;
cache->term = new_term(CLOSURE);
cache->term->u.other = closure;
@@ -119,37 +250,47 @@ static int transition_2(struct stack **stack, struct term **term, HAMT store)
*stack = *stack;
*term = new_term(CACHE);
(*term)->u.other = cache;
+ acquire_term(*term);
+ store_release(&store);
return 0;
}
-static int transition_3(struct term **term, HAMT *store, struct stack **stack,
- struct box *box)
+static int transition_3(struct term **term, struct store **store,
+ struct stack **stack, struct box *box)
{
+ struct term *orig = *term;
assert(box->term->type == CLOSURE);
struct closure *closure = box->term->u.other;
- struct cache *cache = malloc(sizeof(*cache));
+ struct cache *cache = track(malloc(sizeof(*cache)));
cache->box = box;
cache->term = new_term(VAR);
struct term *cache_term = new_term(CACHE);
cache_term->u.other = cache;
+ acquire_term(cache_term);
- *term = closure->term;
- *store = closure->store;
+ *term = acquire_term(closure->term);
+ *store = store_acquire(closure->store);
*stack = stack_push(*stack, cache_term);
- free(closure);
+ release_term(&orig);
+ store_release(store);
return 0;
}
static int transition_4(struct stack **stack, struct term **term,
- struct box *box)
+ struct store *store, struct box *box)
{
+ struct term *orig = *term;
*stack = *stack;
- *term = box->term;
+ *term = acquire_term(box->term);
+
+ store_release(&store);
+ release_term(&orig);
+ /* release_box(&box); */
return 0;
}
@@ -158,98 +299,139 @@ static int transition_5(struct stack **stack, struct term **term,
{
cache->box->state = DONE;
cache->box->term = *term;
+ /* acquire_term(*term); */
+ acquire_box(cache->box); // TODO: ?
*stack = stack_next(*stack);
- *term = *term;
+ *term = acquire_term(*term);
+
+ release_cache(&cache);
return 0;
}
-static int transition_6(struct term **term, HAMT *store, struct stack **stack,
- struct term *peek_term, struct closure *closure)
+static int transition_6(struct term **term, struct store **store,
+ struct stack **stack, struct term *peek_term,
+ struct closure *closure)
{
- struct box *box = malloc(sizeof(*box));
+ struct term *orig = *term;
+
+ struct box *box = track(malloc(sizeof(*box)));
box->state = TODO;
box->term = peek_term->u.app.rhs;
+ acquire_box(box);
- *term = closure->term->u.abs.term;
- *store = hamt_pset(closure->store, &closure->term->u.abs.name, box);
+ *term = acquire_term(closure->term->u.abs.term);
+ *store = store_acquire(
+ store_set(closure->store, &closure->term->u.abs.name, box, 0));
*stack = stack_next(*stack);
+
+ release_term(&orig);
return 0;
}
-static int transition_7(struct term **term, HAMT *store, struct stack **stack,
- struct box *box, struct closure *closure)
+static int transition_7(struct term **term, struct store **store,
+ struct stack **stack, struct box *box,
+ struct closure *closure)
{
+ struct term *orig = *term;
int x = name_generator();
- struct box *var_box = malloc(sizeof(*var_box));
+ struct box *var_box = track(malloc(sizeof(*var_box)));
var_box->state = DONE;
var_box->term = new_term(VAR);
var_box->term->u.var.name = x;
+ acquire_box(var_box);
- struct cache *cache = malloc(sizeof(*cache));
- cache->box = box;
+ struct cache *cache = track(malloc(sizeof(*cache)));
+ cache->box = acquire_box(box);
cache->term = new_term(VAR);
struct term *cache_term = new_term(CACHE);
cache_term->u.other = cache;
+ acquire_term(cache_term);
struct term *abs = new_term(ABS);
abs->u.abs.name = x;
abs->u.abs.term = new_term(VAR);
+ acquire_term(abs);
- *term = closure->term->u.abs.term;
- *store = hamt_pset(closure->store, (void *)&closure->term->u.abs.name,
- var_box);
+ *term = acquire_term(closure->term->u.abs.term);
+ *store = store_acquire(store_set(closure->store,
+ (void *)&closure->term->u.abs.name,
+ var_box, 0));
*stack = stack_push(*stack, cache_term);
*stack = stack_push(*stack, abs);
+
+ release_term(&orig);
return 0;
}
static int transition_8(struct stack **stack, struct term **term,
struct box *box)
{
+ struct term *orig = *term;
+
*stack = *stack;
- *term = box->term;
+ *term = acquire_term(box->term);
+
+ release_term(&orig);
return 0;
}
-static int transition_9(struct term **term, HAMT *store, struct stack **stack,
- struct term *peek_term)
+static int transition_9(struct term **term, struct store **store,
+ struct stack **stack, struct term *peek_term)
{
+ struct term *orig = *term;
+
struct closure *closure = peek_term->u.app.rhs->u.other;
struct term *app = new_term(APP);
app->u.app.lhs = *term;
app->u.app.rhs = new_term(VAR);
+ acquire_term(app);
- *term = closure->term;
- *store = closure->store;
+ *term = acquire_term(closure->term);
+ *store = store_acquire(closure->store);
*stack = stack_push(stack_next(*stack), app);
+
+ release_term(&orig);
+ release_term(&peek_term);
return 0;
}
static int transition_10(struct stack **stack, struct term **term,
struct term *peek_term)
{
+ struct term *orig = *term;
+
struct term *app = new_term(APP);
app->u.app.lhs = peek_term->u.app.lhs;
app->u.app.rhs = *term;
+ acquire_term(app);
*stack = stack_next(*stack);
*term = app;
+
+ release_term(&orig);
+ release_term(&peek_term);
return 0;
}
static int transition_11(struct stack **stack, struct term **term,
struct term *peek_term)
{
+ struct term *orig = *term;
+
struct term *abs = new_term(ABS);
abs->u.abs.name = peek_term->u.abs.name;
abs->u.abs.term = *term;
+ acquire_term(abs);
*stack = stack_next(*stack);
*term = abs;
+
+ release_term(&orig);
+ release_term(&peek_term);
return 0;
}
@@ -257,7 +439,7 @@ static int transition_closure(struct conf *conf, int i,
void (*callback)(int, char))
{
struct term *term = conf->u.econf.term;
- HAMT store = conf->u.econf.store;
+ struct store *store = conf->u.econf.store;
struct stack *stack = conf->u.econf.stack;
int ret = 1;
@@ -273,12 +455,12 @@ static int transition_closure(struct conf *conf, int i,
cconf(conf, stack, term);
return ret;
case VAR:;
- /* printf("%p\n", hamt_get); */
- struct box *box = hamt_get(store, &term->u.var.name);
+ struct box *box = store_get(store, &term->u.var.name, 0);
if (!box) {
- box = malloc(sizeof(*box));
+ box = track(malloc(sizeof(*box)));
box->state = DONE;
box->term = term;
+ acquire_box(box);
}
if (box->state == TODO) { // (3)
callback(i, '3');
@@ -287,7 +469,7 @@ static int transition_closure(struct conf *conf, int i,
return ret;
} else if (box->state == DONE) { // (4)
callback(i, '4');
- ret = transition_4(&stack, &term, box);
+ ret = transition_4(&stack, &term, store, box);
cconf(conf, stack, term);
return ret;
}
@@ -328,7 +510,7 @@ static int transition_computed(struct conf *conf, int i,
((struct cache *)term->u.other)->term->u.other;
if (closure->term->type == ABS) {
callback(i, '6');
- HAMT store;
+ struct store *store;
ret = transition_6(&term, &store, &stack, peek_term,
closure);
econf(conf, term, store, stack);
@@ -343,7 +525,7 @@ static int transition_computed(struct conf *conf, int i,
if (closure->term->type == ABS && box->state == TODO &&
!box->term) { // (7)
callback(i, '7');
- HAMT store;
+ struct store *store;
ret = transition_7(&term, &store, &stack, box, closure);
econf(conf, term, store, stack);
return ret;
@@ -360,7 +542,7 @@ static int transition_computed(struct conf *conf, int i,
!peek_term->u.app.lhs->u.var.name &&
peek_term->u.app.rhs->type == CLOSURE) { // (9)
callback(i, '9');
- HAMT store;
+ struct store *store;
ret = transition_9(&term, &store, &stack, peek_term);
econf(conf, term, store, stack);
return ret;
@@ -412,33 +594,22 @@ static struct conf *for_each_state(struct conf *conf, int i,
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);
+ return *(const int *)lhs == *(const int *)rhs;
}
-static void nop(void *addr)
+static uint32_t hash_var(const void *key)
{
- (void)nop;
+ return murmur3_32((const uint8_t *)key, sizeof(int), 0);
}
struct term *reduce(struct term *term, void (*callback)(int, char))
{
- /* GC_set_find_leak(1); */
- /* GC_INIT(); */
-
struct stack stack = { 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 store *store =
+ store_acquire(store_new(hash_var, hash_var_equal));
struct conf conf = {
.type = ECONF,
- .u.econf.term = term,
+ .u.econf.term = acquire_term(term),
.u.econf.store = store,
.u.econf.stack = &stack,
};
@@ -446,6 +617,6 @@ struct term *reduce(struct term *term, void (*callback)(int, char))
assert(conf.type == CCONF);
struct term *ret = duplicate_term(conf.u.cconf.term);
- hamt_delete(store);
+ /* store_destroy(&store); // should already be freed? TODO */
return ret;
}