#include #include #include #include #include "runtime.h" struct node_base* alloc_node() { struct node_base* new_node = malloc(sizeof(struct node_app)); assert(new_node != NULL); return new_node; } struct node_app* alloc_app(struct node_base* l, struct node_base* r) { struct node_app* node = (struct node_app*) alloc_node(); node->base.tag = NODE_APP; node->left = l; node->right = r; return node; } struct node_num* alloc_num(int32_t n) { struct node_num* node = (struct node_num*) alloc_node(); node->base.tag = NODE_NUM; node->value = n; return node; } struct node_global* alloc_global(void (*f)(struct gmachine*), int32_t a) { struct node_global* node = (struct node_global*) alloc_node(); node->base.tag = NODE_GLOBAL; node->arity = a; node->function = f; return node; } struct node_ind* alloc_ind(struct node_base* n) { struct node_ind* node = (struct node_ind*) alloc_node(); node->base.tag = NODE_IND; node->next = n; return node; } void free_node_direct(struct node_base* n) { if(n->tag == NODE_DATA) { free(((struct node_data*) n)->array); } } void stack_init(struct stack* s) { s->size = 4; s->count = 0; s->data = malloc(sizeof(*s->data) * s->size); assert(s->data != NULL); } void stack_free(struct stack* s) { free(s->data); } void stack_push(struct stack* s, struct node_base* n) { while(s->count >= s->size) { s->data = realloc(s->data, sizeof(*s->data) * (s->size *= 2)); assert(s->data != NULL); } s->data[s->count++] = n; } struct node_base* stack_pop(struct stack* s) { assert(s->count > 0); return s->data[--s->count]; } struct node_base* stack_peek(struct stack* s, size_t o) { assert(s->count > o); return s->data[s->count - o - 1]; } void stack_popn(struct stack* s, size_t n) { assert(s->count >= n); s->count -= n; } void gmachine_init(struct gmachine* g) { stack_init(&g->stack); g->nodes = NULL; } void gmachine_free(struct gmachine* g) { stack_free(&g->stack); struct node_base* to_free = g->nodes; struct node_base* next; while(to_free) { next = to_free->gc_next; free_node_direct(to_free); free(to_free); to_free = next; } } void gmachine_slide(struct gmachine* g, size_t n) { assert(g->stack.count > n); g->stack.data[g->stack.count - n - 1] = g->stack.data[g->stack.count - 1]; g->stack.count -= n; } void gmachine_update(struct gmachine* g, size_t o) { assert(g->stack.count > o + 1); struct node_ind* ind = (struct node_ind*) g->stack.data[g->stack.count - o - 2]; ind->base.tag = NODE_IND; ind->next = g->stack.data[g->stack.count -= 1]; } void gmachine_alloc(struct gmachine* g, size_t o) { while(o--) { stack_push(&g->stack, gmachine_track(g, (struct node_base*) alloc_ind(NULL))); } } void gmachine_pack(struct gmachine* g, size_t n, int8_t t) { assert(g->stack.count >= n); struct node_base** data = malloc(sizeof(*data) * n); assert(data != NULL); memcpy(data, &g->stack.data[g->stack.count - n], n * sizeof(*data)); struct node_data* new_node = (struct node_data*) alloc_node(); new_node->array = data; new_node->base.tag = NODE_DATA; new_node->tag = t; stack_popn(&g->stack, n); stack_push(&g->stack, gmachine_track(g, (struct node_base*) new_node)); } void gmachine_split(struct gmachine* g, size_t n) { struct node_data* node = (struct node_data*) stack_pop(&g->stack); for(size_t i = 0; i < n; i++) { stack_push(&g->stack, node->array[i]); } } struct node_base* gmachine_track(struct gmachine* g, struct node_base* b) { b->gc_next = g->nodes; g->nodes = b; return b; } void gmachine_gc(struct gmachine* g) { } void unwind(struct gmachine* g) { struct stack* s = &g->stack; while(1) { struct node_base* peek = stack_peek(s, 0); if(peek->tag == NODE_APP) { struct node_app* n = (struct node_app*) peek; stack_push(s, n->left); } else if(peek->tag == NODE_GLOBAL) { struct node_global* n = (struct node_global*) peek; assert(s->count > n->arity); for(size_t i = 1; i <= n->arity; i++) { s->data[s->count - i] = ((struct node_app*) s->data[s->count - i - 1])->right; } n->function(g); } else if(peek->tag == NODE_IND) { struct node_ind* n = (struct node_ind*) peek; stack_pop(s); stack_push(s, n->next); } else { break; } } } extern void f_main(struct gmachine* s); void print_node(struct node_base* n) { if(n->tag == NODE_APP) { struct node_app* app = (struct node_app*) n; print_node(app->left); putchar(' '); print_node(app->right); } else if(n->tag == NODE_DATA) { printf("(Packed)"); } else if(n->tag == NODE_GLOBAL) { struct node_global* global = (struct node_global*) n; printf("(Global: %p)", global->function); } else if(n->tag == NODE_IND) { print_node(((struct node_ind*) n)->next); } else if(n->tag == NODE_NUM) { struct node_num* num = (struct node_num*) n; printf("%d", num->value); } } int main(int argc, char** argv) { struct gmachine gmachine; struct node_global* first_node = alloc_global(f_main, 0); struct node_base* result; gmachine_init(&gmachine); gmachine_track(&gmachine, (struct node_base*) first_node); stack_push(&gmachine.stack, (struct node_base*) first_node); unwind(&gmachine); result = stack_pop(&gmachine.stack); printf("Result: "); print_node(result); putchar('\n'); gmachine_free(&gmachine); }