diff options
Diffstat (limited to 'src/reducer.c')
-rw-r--r-- | src/reducer.c | 305 |
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; } |