1 /*
2 * CDDL HEADER START
3 *
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License, Version 1.0 only
6 * (the "License"). You may not use this file except in compliance
7 * with the License.
8 *
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
13 *
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
19 *
20 * CDDL HEADER END
21 */
22 #ifdef HAVE_NBTOOL_CONFIG_H
23 #include "nbtool_config.h"
24 #endif
25
26 /*
27 * Copyright (c) 2001 by Sun Microsystems, Inc.
28 * All rights reserved.
29 */
30
31 #pragma ident "%Z%%M% %I% %E% SMI"
32
33 #include <sys/types.h>
34 #include <sys/sysmacros.h>
35 #include <strings.h>
36 #include <stdlib.h>
37 #include <stdio.h>
38
39 #include "strtab.h"
40 #include "memory.h"
41
42 #define STRTAB_HASHSZ 211 /* use a prime number of hash buckets */
43 #define STRTAB_BUFSZ (64 * 1024) /* use 64K data buffers by default */
44
45 static void
strtab_grow(strtab_t * sp)46 strtab_grow(strtab_t *sp)
47 {
48 sp->str_nbufs++;
49 sp->str_bufs = xrealloc(sp->str_bufs, sp->str_nbufs * sizeof (char *));
50 sp->str_ptr = xmalloc(sp->str_bufsz);
51 sp->str_bufs[sp->str_nbufs - 1] = sp->str_ptr;
52 }
53
54 void
strtab_create(strtab_t * sp)55 strtab_create(strtab_t *sp)
56 {
57 sp->str_hash = xcalloc(STRTAB_HASHSZ * sizeof (strhash_t *));
58 sp->str_hashsz = STRTAB_HASHSZ;
59 sp->str_bufs = NULL;
60 sp->str_ptr = NULL;
61 sp->str_nbufs = 0;
62 sp->str_bufsz = STRTAB_BUFSZ;
63 sp->str_nstrs = 1;
64 sp->str_size = 1;
65
66 strtab_grow(sp);
67 *sp->str_ptr++ = '\0';
68 }
69
70 void
strtab_destroy(strtab_t * sp)71 strtab_destroy(strtab_t *sp)
72 {
73 strhash_t *hp, *hq;
74 ulong_t i;
75
76 for (i = 0; i < sp->str_hashsz; i++) {
77 for (hp = sp->str_hash[i]; hp != NULL; hp = hq) {
78 hq = hp->str_next;
79 free(hp);
80 }
81 }
82
83 for (i = 0; i < sp->str_nbufs; i++)
84 free(sp->str_bufs[i]);
85
86 free(sp->str_hash);
87 free(sp->str_bufs);
88 }
89
90 static ulong_t
strtab_hash(const char * key,size_t * len)91 strtab_hash(const char *key, size_t *len)
92 {
93 ulong_t g, h = 0;
94 const char *p;
95 size_t n = 0;
96
97 for (p = key; *p != '\0'; p++, n++) {
98 h = (h << 4) + *p;
99
100 if ((g = (h & 0xf0000000)) != 0) {
101 h ^= (g >> 24);
102 h ^= g;
103 }
104 }
105
106 *len = n;
107 return (h);
108 }
109
110 static int
strtab_compare(strtab_t * sp,strhash_t * hp,const char * str,size_t len)111 strtab_compare(strtab_t *sp, strhash_t *hp, const char *str, size_t len)
112 {
113 ulong_t b = hp->str_buf;
114 const char *buf = hp->str_data;
115 size_t resid, n;
116 int rv;
117
118 while (len != 0) {
119 if (buf == sp->str_bufs[b] + sp->str_bufsz)
120 buf = sp->str_bufs[++b];
121
122 resid = sp->str_bufs[b] + sp->str_bufsz - buf;
123 n = MIN(resid, len);
124
125 if ((rv = strncmp(buf, str, n)) != 0)
126 return (rv);
127
128 buf += n;
129 str += n;
130 len -= n;
131 }
132
133 return (0);
134 }
135
136 static void
strtab_copyin(strtab_t * sp,const char * str,size_t len)137 strtab_copyin(strtab_t *sp, const char *str, size_t len)
138 {
139 ulong_t b = sp->str_nbufs - 1;
140 size_t resid, n;
141
142 while (len != 0) {
143 if (sp->str_ptr == sp->str_bufs[b] + sp->str_bufsz) {
144 strtab_grow(sp);
145 b++;
146 }
147
148 resid = sp->str_bufs[b] + sp->str_bufsz - sp->str_ptr;
149 n = MIN(resid, len);
150 bcopy(str, sp->str_ptr, n);
151
152 sp->str_ptr += n;
153 str += n;
154 len -= n;
155 }
156 }
157
158 size_t
strtab_insert(strtab_t * sp,const char * str)159 strtab_insert(strtab_t *sp, const char *str)
160 {
161 strhash_t *hp;
162 size_t len;
163 ulong_t h;
164
165 if (str == NULL || str[0] == '\0')
166 return (0); /* we keep a \0 at offset 0 to simplify things */
167
168 h = strtab_hash(str, &len) % sp->str_hashsz;
169
170 /*
171 * If the string is already in our hash table, just return the offset
172 * of the existing string element and do not add a duplicate string.
173 */
174 for (hp = sp->str_hash[h]; hp != NULL; hp = hp->str_next) {
175 if (strtab_compare(sp, hp, str, len + 1) == 0)
176 return (hp->str_off);
177 }
178
179 /*
180 * Create a new hash bucket, initialize it, and insert it at the front
181 * of the hash chain for the appropriate bucket.
182 */
183 hp = xmalloc(sizeof (strhash_t));
184
185 hp->str_data = sp->str_ptr;
186 hp->str_buf = sp->str_nbufs - 1;
187 hp->str_off = sp->str_size;
188 hp->str_len = len;
189 hp->str_next = sp->str_hash[h];
190
191 sp->str_hash[h] = hp;
192
193 /*
194 * Now copy the string data into our buffer list, and then update
195 * the global counts of strings and bytes. Return str's byte offset.
196 */
197 strtab_copyin(sp, str, len + 1);
198 sp->str_nstrs++;
199 sp->str_size += len + 1;
200
201 return (hp->str_off);
202 }
203
204 size_t
strtab_size(const strtab_t * sp)205 strtab_size(const strtab_t *sp)
206 {
207 return (sp->str_size);
208 }
209
210 ssize_t
strtab_write(const strtab_t * sp,ssize_t (* func)(void *,size_t,void *),void * priv)211 strtab_write(const strtab_t *sp,
212 ssize_t (*func)(void *, size_t, void *), void *priv)
213 {
214 ssize_t res, total = 0;
215 ulong_t i;
216 size_t n;
217
218 for (i = 0; i < sp->str_nbufs; i++, total += res) {
219 if (i == sp->str_nbufs - 1)
220 n = sp->str_ptr - sp->str_bufs[i];
221 else
222 n = sp->str_bufsz;
223
224 if ((res = func(sp->str_bufs[i], n, priv)) <= 0)
225 break;
226 }
227
228 if (total == 0 && sp->str_size != 0)
229 return (-1);
230
231 return (total);
232 }
233
234 void
strtab_print(const strtab_t * sp)235 strtab_print(const strtab_t *sp)
236 {
237 const strhash_t *hp;
238 ulong_t i;
239
240 for (i = 0; i < sp->str_hashsz; i++) {
241 for (hp = sp->str_hash[i]; hp != NULL; hp = hp->str_next) {
242 const char *buf = hp->str_data;
243 ulong_t b = hp->str_buf;
244 size_t resid, len, n;
245
246 (void) printf("[%lu] %lu \"", (ulong_t)hp->str_off, b);
247
248 for (len = hp->str_len; len != 0; len -= n) {
249 if (buf == sp->str_bufs[b] + sp->str_bufsz)
250 buf = sp->str_bufs[++b];
251
252 resid = sp->str_bufs[b] + sp->str_bufsz - buf;
253 n = MIN(resid, len);
254
255 (void) printf("%.*s", (int)n, buf);
256 buf += n;
257 }
258
259 (void) printf("\"\n");
260 }
261 }
262 }
263