#include <sys/types.h> #include <assert.h> #include <math.h> #include <setjmp.h> #include <stdarg.h> #include <stdint.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include "scheme.h" #include "equalhash.h" static value_t tablesym; static struct fltype *tabletype; void print_htable(value_t v, struct ios *f) { struct htable *h = (struct htable *)cv_data((struct cvalue *)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]); fl_print_chr(' ', f); fl_print_child(f, (value_t)h->table[i + 1]); first = 0; } } fl_print_chr(')', f); } void print_traverse_htable(value_t self) { struct htable *h = (struct htable *)cv_data((struct cvalue *)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) { struct htable *h = (struct htable *)cv_data((struct cvalue *)ptr(self)); htable_free(h); } void relocate_htable(value_t oldv, value_t newv) { size_t i; struct htable *oldh; struct htable *h; oldh = (struct htable *)cv_data((struct cvalue *)ptr(oldv)); h = (struct htable *)cv_data((struct cvalue *)ptr(newv)); if (oldh->table == &oldh->_space[0]) h->table = &h->_space[0]; for (i = 0; i < h->size; i++) { if (h->table[i] != HT_NOTFOUND) h->table[i] = (void *)relocate_lispvalue((value_t)h->table[i]); } } struct cvtable table_vtable = { print_htable, relocate_htable, free_htable, print_traverse_htable }; int ishashtable(value_t v) { return iscvalue(v) && cv_class((struct cvalue *)ptr(v)) == tabletype; } value_t fl_tablep(value_t *args, uint32_t nargs) { argcount("table?", nargs, 1); return ishashtable(args[0]) ? FL_T : FL_F; } static struct htable *totable(value_t v, char *fname) { if (!ishashtable(v)) type_error(fname, "table", v); return (struct htable *)cv_data((struct cvalue *)ptr(v)); } value_t fl_table(value_t *args, uint32_t nargs) { struct htable *h; value_t nt, k, arg; size_t cnt; uint32_t i; cnt = (size_t)nargs; if (cnt & 1) lerror(ArgError, "table: arguments must come in pairs"); // prevent small tables from being added to finalizer list if (cnt <= HT_N_INLINE) { tabletype->vtable->finalize = NULL; nt = cvalue(tabletype, sizeof(struct htable)); tabletype->vtable->finalize = free_htable; } else { nt = cvalue(tabletype, 2 * sizeof(void *)); } h = (struct htable *)cv_data((struct cvalue *)ptr(nt)); htable_new(h, cnt / 2); k = FL_NIL; arg = FL_NIL; FOR_ARGS(i, 0, arg, args) { if (i & 1) equalhash_put(h, (void *)k, (void *)arg); else k = arg; } return nt; } // (put! table key value) value_t fl_table_put(value_t *args, uint32_t nargs) { struct htable *h; void **table0; argcount("put!", nargs, 3); h = totable(args[0], "put!"); 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]) { struct cvalue *cv = (struct cvalue *)ptr(args[0]); add_finalizer(cv); cv->len = 2 * sizeof(void *); } return args[0]; } static void key_error(char *fname, value_t key) { lerrorf(fl_list2(KeyError, key), "%s: key not found", fname); } // (get table key [default]) value_t fl_table_get(value_t *args, uint32_t nargs) { struct htable *h; value_t v; if (nargs != 3) argcount("get", nargs, 2); h = totable(args[0], "get"); v = (value_t)equalhash_get(h, (void *)args[1]); if (v == (value_t)HT_NOTFOUND) { if (nargs == 3) return args[2]; key_error("get", args[1]); } return v; } // (has? table key) value_t fl_table_has(value_t *args, uint32_t nargs) { struct htable *h; argcount("has", nargs, 2); h = totable(args[0], "has"); return equalhash_has(h, (void *)args[1]) ? FL_T : FL_F; } // (del! table key) value_t fl_table_del(value_t *args, uint32_t nargs) { struct htable *h; argcount("del!", nargs, 2); h = totable(args[0], "del!"); if (!equalhash_remove(h, (void *)args[1])) key_error("del!", args[1]); return args[0]; } value_t fl_table_foldl(value_t *args, uint32_t nargs) { struct htable *h; void **table; size_t i, n; value_t f, zero, t; argcount("table.foldl", nargs, 3); f = args[0]; zero = args[1]; t = args[2]; h = totable(t, "table.foldl"); n = h->size; table = h->table; fl_gc_handle(&f); fl_gc_handle(&zero); fl_gc_handle(&t); for (i = 0; i < n; i += 2) { if (table[i + 1] != HT_NOTFOUND) { zero = fl_applyn(3, f, (value_t)table[i], (value_t)table[i + 1], zero); // reload pointer h = (struct htable *)cv_data((struct cvalue *)ptr(t)); if (h->size != n) lerror(EnumerationError, "table.foldl: table modified"); table = h->table; } } fl_free_gc_handles(3); return zero; } static struct builtinspec tablefunc_info[] = { { "table", fl_table }, { "table?", 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(void) { tablesym = symbol("table"); tabletype = define_opaque_type(tablesym, sizeof(struct htable), &table_vtable, NULL); assign_global_builtins(tablefunc_info); }