ref: ccff1a68e10909fb73e471772d1d603e3fa72297
dir: /dict.c/
#include <stdio.h> #include <stdlib.h> #include <string.h> #include "roff.h" #define DHASH(d, s) ((d)->level2 && (s)[0] ? (((unsigned char) (s)[1]) << 8) | (unsigned char) (s)[0] : (unsigned char) s[0]) #define CNTMIN (1 << 10) struct dict { struct iset *map; char **key; int *val; int size; int n; int notfound; /* the value returned for missing keys */ int level2; /* use two characters for hashing */ int dupkeys; /* duplicate keys if set */ }; static void dict_extend(struct dict *d, int size) { d->key = mextend(d->key, d->size, size, sizeof(d->key[0])); d->val = mextend(d->val, d->size, size, sizeof(d->val[0])); d->size = size; } /* * initialise a dictionary * * notfound: the value returned for missing keys. * dupkeys: if nonzero, store a copy of keys inserted via dict_put(). * level2: use two characters for hasing */ struct dict *dict_make(int notfound, int dupkeys, int level2) { struct dict *d = xmalloc(sizeof(*d)); memset(d, 0, sizeof(*d)); d->n = 1; d->level2 = level2; d->dupkeys = dupkeys; d->notfound = notfound; d->map = iset_make(); dict_extend(d, CNTMIN); return d; } void dict_free(struct dict *d) { int i; if (d->dupkeys) for (i = 0; i < d->size; i++) free(d->key[i]); free(d->val); free(d->key); iset_free(d->map); free(d); } void dict_put(struct dict *d, char *key, int val) { int idx; if (d->n >= d->size) dict_extend(d, d->n + CNTMIN); if (d->dupkeys) { int len = strlen(key) + 1; char *dup = xmalloc(len); memcpy(dup, key, len); key = dup; } idx = d->n++; d->key[idx] = key; d->val[idx] = val; iset_put(d->map, DHASH(d, key), idx); } /* return the index of key in d */ int dict_idx(struct dict *d, char *key) { int h = DHASH(d, key); int *b = iset_get(d->map, h); int *r = b + iset_len(d->map, h); while (b && --r >= b) if (!strcmp(d->key[*r], key)) return *r; return -1; } char *dict_key(struct dict *d, int idx) { return d->key[idx]; } int dict_val(struct dict *d, int idx) { return d->val[idx]; } int dict_get(struct dict *d, char *key) { int idx = dict_idx(d, key); return idx >= 0 ? d->val[idx] : d->notfound; } /* match a prefix of key; in the first call, *idx should be -1 */ int dict_prefix(struct dict *d, char *key, int *pos) { int *r = iset_get(d->map, DHASH(d, key)); while (r && r[++*pos] >= 0) { int idx = r[*pos]; int plen = strlen(d->key[idx]); if (!strncmp(d->key[idx], key, plen)) return d->val[idx]; } return d->notfound; }