Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/thorpej_scsipi]: src/usr.sbin/config bump ALIGNBYTES up to 7 -- that way...
details: https://anonhg.NetBSD.org/src/rev/526958c7437a
branches: thorpej_scsipi
changeset: 477318:526958c7437a
user: cgd <cgd%NetBSD.org@localhost>
date: Mon Oct 02 19:51:08 2000 +0000
description:
bump ALIGNBYTES up to 7 -- that way this'll have more of a hope
of working on non-NetBSD systems with 64-bit pointers. (affects only
the MAKE_BOOTSTRAP case, and then only if BSD not defied.)
diffstat:
usr.sbin/config/hash.c | 307 +++++++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 307 insertions(+), 0 deletions(-)
diffs (truncated from 311 to 300 lines):
diff -r bdf6ae0978e3 -r 526958c7437a usr.sbin/config/hash.c
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/usr.sbin/config/hash.c Mon Oct 02 19:51:08 2000 +0000
@@ -0,0 +1,307 @@
+/* $NetBSD: hash.c,v 1.8.2.2 2000/10/02 19:51:08 cgd Exp $ */
+
+/*
+ * Copyright (c) 1992, 1993
+ * The Regents of the University of California. All rights reserved.
+ *
+ * This software was developed by the Computer Systems Engineering group
+ * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
+ * contributed to Berkeley.
+ *
+ * All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Lawrence Berkeley Laboratories.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * from: @(#)hash.c 8.1 (Berkeley) 6/6/93
+ */
+
+#include <sys/param.h>
+#include <stdlib.h>
+#include <string.h>
+#include "config.h"
+
+/*
+ * These are really for MAKE_BOOTSTRAP but harmless.
+ * XXX - Why not just use malloc in here, anyway?
+ */
+#ifndef ALIGNBYTES
+#define ALIGNBYTES 7
+#endif
+#ifndef ALIGN
+#define ALIGN(p) (((long)(p) + ALIGNBYTES) &~ ALIGNBYTES)
+#endif
+
+/*
+ * Interned strings are kept in a hash table. By making each string
+ * unique, the program can compare strings by comparing pointers.
+ */
+struct hashent {
+ struct hashent *h_next; /* hash buckets are chained */
+ const char *h_name; /* the string */
+ u_int h_hash; /* its hash value */
+ void *h_value; /* other values (for name=value) */
+};
+struct hashtab {
+ size_t ht_size; /* size (power of 2) */
+ u_int ht_mask; /* == ht_size - 1 */
+ u_int ht_used; /* number of entries used */
+ u_int ht_lim; /* when to expand */
+ struct hashent **ht_tab; /* base of table */
+};
+static struct hashtab strings;
+
+/*
+ * HASHFRACTION controls ht_lim, which in turn controls the average chain
+ * length. We allow a few entries, on average, as comparing them is usually
+ * cheap (the h_hash values prevent a strcmp).
+ */
+#define HASHFRACTION(sz) ((sz) * 3 / 2)
+
+/* round up to next multiple of y, where y is a power of 2 */
+#define ROUND(x, y) (((x) + (y) - 1) & ~((y) - 1))
+
+static void *poolalloc(size_t);
+static void ht_expand(struct hashtab *);
+static void ht_init(struct hashtab *, size_t);
+static inline u_int hash(const char *);
+static inline struct hashent *newhashent(const char *, u_int);
+
+/*
+ * Allocate space that will never be freed.
+ */
+static void *
+poolalloc(size_t size)
+{
+ char *p;
+ size_t alloc;
+ static char *pool;
+ static size_t nleft;
+
+ if (nleft < size) {
+ /*
+ * Compute a `good' size to allocate via malloc.
+ * 16384 is a guess at a good page size for malloc;
+ * 32 is a guess at malloc's overhead.
+ */
+ alloc = ROUND(size + 32, 16384) - 32;
+ p = emalloc(alloc);
+ nleft = alloc - size;
+ } else {
+ p = pool;
+ nleft -= size;
+ }
+ pool = p + size;
+ return (p);
+}
+
+/*
+ * Initialize a new hash table. The size must be a power of 2.
+ */
+static void
+ht_init(struct hashtab *ht, size_t sz)
+{
+ struct hashent **h;
+ u_int n;
+
+ h = emalloc(sz * sizeof *h);
+ ht->ht_tab = h;
+ ht->ht_size = sz;
+ ht->ht_mask = sz - 1;
+ for (n = 0; n < sz; n++)
+ *h++ = NULL;
+ ht->ht_used = 0;
+ ht->ht_lim = HASHFRACTION(sz);
+}
+
+/*
+ * Expand an existing hash table.
+ */
+static void
+ht_expand(struct hashtab *ht)
+{
+ struct hashent *p, **h, **oldh, *q;
+ u_int n, i;
+
+ n = ht->ht_size * 2;
+ h = emalloc(n * sizeof *h);
+ for (i = 0; i < n; i++)
+ h[i] = NULL;
+ oldh = ht->ht_tab;
+ n--;
+ for (i = ht->ht_size; i != 0; i--) {
+ for (p = *oldh++; p != NULL; p = q) {
+ q = p->h_next;
+ p->h_next = h[p->h_hash & n];
+ h[p->h_hash & n] = p;
+ }
+ }
+ free(ht->ht_tab);
+ ht->ht_tab = h;
+ ht->ht_mask = n;
+ ht->ht_size = ++n;
+ ht->ht_lim = HASHFRACTION(n);
+}
+
+/*
+ * Make a new hash entry, setting its h_next to NULL.
+ */
+static inline struct hashent *
+newhashent(const char *name, u_int h)
+{
+ struct hashent *hp;
+ char *m;
+
+ m = poolalloc(sizeof(*hp) + ALIGNBYTES);
+ hp = (struct hashent *)ALIGN(m);
+ hp->h_name = name;
+ hp->h_hash = h;
+ hp->h_next = NULL;
+ return (hp);
+}
+
+/*
+ * Hash a string.
+ */
+static inline u_int
+hash(const char *str)
+{
+ u_int h;
+
+ for (h = 0; *str;)
+ h = (h << 5) + h + *str++;
+ return (h);
+}
+
+void
+initintern(void)
+{
+
+ ht_init(&strings, 128);
+}
+
+/*
+ * Generate a single unique copy of the given string. We expect this
+ * function to be used frequently, so it should be fast.
+ */
+const char *
+intern(const char *s)
+{
+ struct hashtab *ht;
+ struct hashent *hp, **hpp;
+ u_int h;
+ char *p;
+ size_t l;
+
+ ht = &strings;
+ h = hash(s);
+ hpp = &ht->ht_tab[h & ht->ht_mask];
+ for (; (hp = *hpp) != NULL; hpp = &hp->h_next)
+ if (hp->h_hash == h && strcmp(hp->h_name, s) == 0)
+ return (hp->h_name);
+ l = strlen(s) + 1;
+ p = poolalloc(l);
+ memmove(p, s, l);
+ *hpp = newhashent(p, h);
+ if (++ht->ht_used > ht->ht_lim)
+ ht_expand(ht);
+ return (p);
+}
+
+struct hashtab *
+ht_new(void)
+{
+ struct hashtab *ht;
+
+ ht = emalloc(sizeof *ht);
+ ht_init(ht, 8);
+ return (ht);
+}
+
+/*
+ * Insert and/or replace.
+ */
+int
+ht_insrep(struct hashtab *ht, const char *nam, void *val, int replace)
+{
+ struct hashent *hp, **hpp;
+ u_int h;
+
+ h = hash(nam);
+ hpp = &ht->ht_tab[h & ht->ht_mask];
+ for (; (hp = *hpp) != NULL; hpp = &hp->h_next) {
+ if (hp->h_name == nam) {
+ if (replace)
+ hp->h_value = val;
+ return (1);
+ }
+ }
+ *hpp = hp = newhashent(nam, h);
+ hp->h_value = val;
+ if (++ht->ht_used > ht->ht_lim)
+ ht_expand(ht);
+ return (0);
+}
+
+void *
+ht_lookup(struct hashtab *ht, const char *nam)
+{
+ struct hashent *hp, **hpp;
+ u_int h;
+
+ h = hash(nam);
+ hpp = &ht->ht_tab[h & ht->ht_mask];
+ for (; (hp = *hpp) != NULL; hpp = &hp->h_next)
+ if (hp->h_name == nam)
+ return (hp->h_value);
+ return (NULL);
+}
+
+/*
+ * first parameter to callback is the entry name from the hash table
+ * second parameter is the value from the hash table
+ * third argument is passed through from the "arg" parameter to ht_enumerate()
+ */
+
+int
+ht_enumerate(struct hashtab *ht, ht_callback cbfunc, void *arg)
+{
Home |
Main Index |
Thread Index |
Old Index