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