shithub: femtolisp

ref: dc50df083ca50561084bf572f538ca76a9dd100e
dir: /femtolisp/table.c/

View raw version
#include <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#include <assert.h>
#include <sys/types.h>
#include "llt.h"
#include "flisp.h"
#include "equalhash.h"

static value_t tablesym;
static fltype_t *tabletype;

void print_htable(value_t v, ios_t *f, int princ)
{
    htable_t *h = (htable_t*)cv_data((cvalue_t*)ptr(v));
    size_t i;
    int first=1;
    fl_print_str("#table(", f);
    for(i=0; i < h->size; i+=2) {
        if (h->table[i+1] != HT_NOTFOUND) {
            if (!first) fl_print_str("  ", f);
            fl_print_child(f, (value_t)h->table[i], princ);
            fl_print_chr(' ', f);
            fl_print_child(f, (value_t)h->table[i+1], princ);
            first = 0;
        }
    }
    fl_print_chr(')', f);
}

void print_traverse_htable(value_t self)
{
    htable_t *h = (htable_t*)cv_data((cvalue_t*)ptr(self));
    size_t i;
    for(i=0; i < h->size; i+=2) {
        if (h->table[i+1] != HT_NOTFOUND) {
            print_traverse((value_t)h->table[i]);
            print_traverse((value_t)h->table[i+1]);
        }
    }
}

void free_htable(value_t self)
{
    htable_t *h = (htable_t*)cv_data((cvalue_t*)ptr(self));
    htable_free(h);
}

void relocate_htable(value_t oldv, value_t newv)
{
    htable_t *oldh = (htable_t*)cv_data((cvalue_t*)ptr(oldv));
    htable_t *h = (htable_t*)cv_data((cvalue_t*)ptr(newv));
    if (oldh->table == &oldh->_space[0])
        h->table = &h->_space[0];
    size_t i;
    for(i=0; i < h->size; i++) {
        if (h->table[i] != HT_NOTFOUND)
            h->table[i] = (void*)relocate_lispvalue((value_t)h->table[i]);
    }
}

cvtable_t table_vtable = { print_htable, relocate_htable, free_htable,
                           print_traverse_htable };

int ishashtable(value_t v)
{
    return iscvalue(v) && cv_class((cvalue_t*)ptr(v)) == tabletype;
}

value_t fl_tablep(value_t *args, uint32_t nargs)
{
    argcount("tablep", nargs, 1);
    return ishashtable(args[0]) ? T : NIL;
}

static htable_t *totable(value_t v, char *fname)
{
    if (ishashtable(v))
        return (htable_t*)cv_data((cvalue_t*)ptr(v));
    type_error(fname, "table", v);
    return NULL;
}

value_t fl_table(value_t *args, uint32_t nargs)
{
    if (nargs & 1)
        lerror(ArgError, "table: arguments must come in pairs");
    value_t nt;
    // prevent small tables from being added to finalizer list
    if (nargs <= HT_N_INLINE) {
        tabletype->vtable->finalize = NULL;
        nt = cvalue(tabletype, sizeof(htable_t));
        tabletype->vtable->finalize = free_htable;
    }
    else {
        nt = cvalue(tabletype, 2*sizeof(void*));
    }
    htable_t *h = (htable_t*)cv_data((cvalue_t*)ptr(nt));
    htable_new(h, nargs/2);
    uint32_t i;
    for(i=0; i < nargs; i+=2)
        equalhash_put(h, (void*)args[i], (void*)args[i+1]);
    return nt;
}

// (put table key value)
value_t fl_table_put(value_t *args, uint32_t nargs)
{
    argcount("put", nargs, 3);
    htable_t *h = totable(args[0], "put");
    void **table0 = h->table;
    equalhash_put(h, (void*)args[1], (void*)args[2]);
    // register finalizer if we outgrew inline space
    if (table0 == &h->_space[0] && h->table != &h->_space[0]) {
        cvalue_t *cv = (cvalue_t*)ptr(args[0]);
        add_finalizer(cv);
        cv->len = 2*sizeof(void*);
    }
    return args[0];
}

// (get table key [default])
value_t fl_table_get(value_t *args, uint32_t nargs)
{
    if (nargs != 3)
        argcount("get", nargs, 2);
    htable_t *h = totable(args[0], "get");
    value_t v = (value_t)equalhash_get(h, (void*)args[1]);
    if (v == (value_t)HT_NOTFOUND) {
        if (nargs == 3)
            return args[2];
        lerror(KeyError, "get: key not found");
    }
    return v;
}

// (has table key)
value_t fl_table_has(value_t *args, uint32_t nargs)
{
    argcount("has", nargs, 2);
    htable_t *h = totable(args[0], "has");
    return equalhash_has(h, (void*)args[1]) ? T : NIL;
}

// (del table key)
value_t fl_table_del(value_t *args, uint32_t nargs)
{
    argcount("del", nargs, 2);
    htable_t *h = totable(args[0], "del");
    if (!equalhash_remove(h, (void*)args[1]))
        lerror(KeyError, "del: key not found");
    return args[0];
}

value_t fl_table_foldl(value_t *args, uint32_t nargs)
{
    argcount("table.foldl", nargs, 3);
    PUSH(listn(3, NIL, NIL, NIL));
    htable_t *h = totable(args[2], "table.foldl");
    size_t i, n = h->size;
    void **table = h->table;
    value_t c;
    for(i=0; i < n; i+=2) {
        if (table[i+1] != HT_NOTFOUND) {
            c = Stack[SP-1];
            car_(c) = (value_t)table[i];
            car_(cdr_(c)) = (value_t)table[i+1];
            car_(cdr_(cdr_(c))) = args[1];
            args[1] = apply(args[0], c);
            // reload pointer
            table = ((htable_t*)cv_data((cvalue_t*)ptr(args[2])))->table;
        }
    }
    (void)POP();
    return args[1];
}

static builtinspec_t tablefunc_info[] = {
    { "table", fl_table },
    { "tablep", fl_tablep },
    { "put", fl_table_put },
    { "get", fl_table_get },
    { "has", fl_table_has },
    { "del", fl_table_del },
    { "table.foldl", fl_table_foldl },
    { NULL, NULL }
};

void table_init()
{
    tablesym = symbol("table");
    tabletype = define_opaque_type(tablesym, sizeof(htable_t),
                                   &table_vtable, NULL);
    assign_global_builtins(tablefunc_info);
}