font: unmap glyphs with .fmap
[neatroff.git] / iset.c
blob496468887fa167b837f1b7ce19cba0b06c251b83
1 #include <stdlib.h>
2 #include <string.h>
3 #include "roff.h"
5 #define ALIGN(n, a) (((n) + (a) - 1) & ~((a) - 1))
6 #define CNTMIN (1 << 10)
7 #define CNTMAX (1 << 20)
9 /* iset structure to map integers to sets */
10 struct iset {
11 int **set;
12 int *sz;
13 int *len;
14 int cnt;
17 static void iset_extend(struct iset *iset, int cnt)
19 iset->set = mextend(iset->set, iset->cnt, cnt, sizeof(iset->set[0]));
20 iset->sz = mextend(iset->sz, iset->cnt, cnt, sizeof(iset->sz[0]));
21 iset->len = mextend(iset->len, iset->cnt, cnt, sizeof(iset->len[0]));
22 iset->cnt = cnt;
25 struct iset *iset_make(void)
27 struct iset *iset = xmalloc(sizeof(*iset));
28 memset(iset, 0, sizeof(*iset));
29 iset_extend(iset, CNTMIN);
30 return iset;
33 void iset_free(struct iset *iset)
35 int i;
36 for (i = 0; i < iset->cnt; i++)
37 free(iset->set[i]);
38 free(iset->set);
39 free(iset->len);
40 free(iset->sz);
41 free(iset);
44 int *iset_get(struct iset *iset, int key)
46 return key >= 0 && key < iset->cnt ? iset->set[key] : NULL;
49 int iset_len(struct iset *iset, int key)
51 return key >= 0 && key < iset->cnt ? iset->len[key] : 0;
54 void iset_put(struct iset *iset, int key, int ent)
56 if (key < 0 || key >= CNTMAX)
57 return;
58 if (key >= iset->cnt)
59 iset_extend(iset, ALIGN(key + 1, CNTMIN));
60 if (key >= 0 && key < iset->cnt && iset->len[key] + 1 >= iset->sz[key]) {
61 int olen = iset->sz[key];
62 int nlen = iset->sz[key] * 2 + 8;
63 void *nset = xmalloc(nlen * sizeof(iset->set[key][0]));
64 if (iset->set[key]) {
65 memcpy(nset, iset->set[key],
66 olen * sizeof(iset->set[key][0]));
67 free(iset->set[key]);
69 iset->sz[key] = nlen;
70 iset->set[key] = nset;
72 iset->set[key][iset->len[key]++] = ent;
73 iset->set[key][iset->len[key]] = -1;