⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 strhash.c

📁 package of develop dns
💻 C
字号:
#ifndef lintstatic char *rcsid = "$Id: strhash.c,v 1.1.1.1 2003/06/04 00:26:13 marka Exp $";#endif/* * Copyright (c) 2000 Japan Network Information Center.  All rights reserved. *   * By using this file, you agree to the terms and conditions set forth bellow. *  * 			LICENSE TERMS AND CONDITIONS  *  * The following License Terms and Conditions apply, unless a different * license is obtained from Japan Network Information Center ("JPNIC"), * a Japanese association, Kokusai-Kougyou-Kanda Bldg 6F, 2-3-4 Uchi-Kanda, * Chiyoda-ku, Tokyo 101-0047, Japan. *  * 1. Use, Modification and Redistribution (including distribution of any *    modified or derived work) in source and/or binary forms is permitted *    under this License Terms and Conditions. *  * 2. Redistribution of source code must retain the copyright notices as they *    appear in each source code file, this License Terms and Conditions. *  * 3. Redistribution in binary form must reproduce the Copyright Notice, *    this License Terms and Conditions, in the documentation and/or other *    materials provided with the distribution.  For the purposes of binary *    distribution the "Copyright Notice" refers to the following language: *    "Copyright (c) 2000-2002 Japan Network Information Center.  All rights reserved." *  * 4. The name of JPNIC may not be used to endorse or promote products *    derived from this Software without specific prior written approval of *    JPNIC. *  * 5. Disclaimer/Limitation of Liability: THIS SOFTWARE IS PROVIDED BY JPNIC *    "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 JPNIC 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 DAMAGES. */#include <config.h>#include <stddef.h>#include <stdlib.h>#include <string.h>#include <idn/assert.h>#include <idn/logmacro.h>#include <idn/result.h>#include <idn/strhash.h>/* * Initially, the number of hash buckets is INITIAL_HASH_SIZE. * As the more elements are put in the hash, the number of elements * per bucket will exceed THRESHOLD eventually.  When it happens, * the number of buckets will be multiplied by FACTOR. */#define INITIAL_HASH_SIZE	67#define FACTOR			7#define THRESHOLD		5#define HASH_MULT		31typedef struct strhash_entry {	struct strhash_entry *next;	unsigned long hash_value;	char *key;	void *value;} strhash_entry_t;struct idn__strhash {	int nbins;	int nelements;	strhash_entry_t **bins;};static unsigned long	hash_value(const char *key);static strhash_entry_t	*find_entry(strhash_entry_t *entry, const char *key,				    unsigned long hash);static strhash_entry_t	*new_entry(const char *key, void *value);static idn_result_t	expand_bins(idn__strhash_t hash, int new_size);idn_result_tidn__strhash_create(idn__strhash_t *hashp) {	idn__strhash_t hash;	idn_result_t r;	TRACE(("idn__strhash_create()\n"));	assert(hashp != NULL);	*hashp = NULL;	if ((hash = malloc(sizeof(struct idn__strhash))) == NULL) {		WARNING(("idn__strhash_create: malloc failed (hash)\n"));		return (idn_nomemory);	}	hash->nbins = 0;	hash->nelements = 0;	hash->bins = NULL;	if ((r = expand_bins(hash, INITIAL_HASH_SIZE)) != idn_success) {		WARNING(("idn__strhash_create: malloc failed (bins)\n"));		free(hash);		return (r);	}	*hashp = hash;	return (idn_success);}voididn__strhash_destroy(idn__strhash_t hash, idn__strhash_freeproc_t proc) {	int i;	assert(hash != NULL && hash->bins != NULL);	for (i = 0; i < hash->nbins; i++) {		strhash_entry_t *bin = hash->bins[i];		strhash_entry_t *next;		while (bin != NULL) {			next = bin->next;			if (proc != NULL)				(*proc)(bin->value);			free(bin);			bin = next;		}	}	free(hash->bins);	free(hash);}idn_result_tidn__strhash_put(idn__strhash_t hash, const char *key, void *value) {	unsigned long h, h_index;	strhash_entry_t *entry;	assert(hash != NULL && key != NULL);	h = hash_value(key);	h_index = h % hash->nbins;	if ((entry = find_entry(hash->bins[h_index], key, h)) != NULL) {		/* Entry exists.  Replace the value. */		entry->value = value;	} else {		/* Create new entry. */		if ((entry = new_entry(key, value)) == NULL) {			return (idn_nomemory);		}		/* Insert it to the list. */		entry->next = hash->bins[h_index];		hash->bins[h_index] = entry;		hash->nelements++;		if (hash->nelements > hash->nbins * THRESHOLD) {			idn_result_t r;			r = expand_bins(hash, hash->nbins * FACTOR);			if (r != idn_success) {				TRACE(("idn__strhash_put: hash table "					"expansion failed\n"));			}		}	}	return (idn_success);}idn_result_tidn__strhash_get(idn__strhash_t hash, const char *key, void **valuep) {	unsigned long h;	strhash_entry_t *entry;	assert(hash != NULL && key != NULL && valuep != NULL);	h = hash_value(key);	entry = find_entry(hash->bins[h % hash->nbins], key, h);	if (entry == NULL)		return (idn_noentry);	*valuep = entry->value;	return (idn_success);}intidn__strhash_exists(idn__strhash_t hash, const char *key) {	unsigned long h;	assert(hash != NULL && key != NULL);	h = hash_value(key);	return (find_entry(hash->bins[h % hash->nbins], key, h) != NULL);}static unsigned longhash_value(const char *key) {	unsigned long h = 0;	unsigned char *p = (unsigned char *)key;	int c;	while ((c = *p++) != '\0') {		h = h * HASH_MULT + c;	}	return (h);}static strhash_entry_t *find_entry(strhash_entry_t *entry, const char *key, unsigned long hash) {	assert(key != NULL);	while (entry != NULL) {		if (entry->hash_value == hash && strcmp(key, entry->key) == 0)			return (entry);		entry = entry->next;	}	return (NULL);}static strhash_entry_t *new_entry(const char *key, void *value) {	strhash_entry_t *entry;	int len;	assert(key != NULL);	len = strlen(key) + 1;	if ((entry = malloc(sizeof(strhash_entry_t) + len)) == NULL) {		return (NULL);	}	entry->next = NULL;	entry->hash_value = hash_value(key);	entry->key = (char *)(entry + 1);	(void)strcpy(entry->key, key);	entry->value = value;	return (entry);}static idn_result_texpand_bins(idn__strhash_t hash, int new_size) {	strhash_entry_t **old_bins, **new_bins;	int old_size;	int old_index, new_index;	new_bins = malloc(sizeof(strhash_entry_t *) * new_size);	if (new_bins == NULL)		return (idn_nomemory);	memset(new_bins, 0, sizeof(strhash_entry_t *) * new_size);	old_bins = hash->bins;	old_size = hash->nbins;	for (old_index = 0; old_index < old_size; old_index++) {		strhash_entry_t *entries = old_bins[old_index];		while (entries != NULL) {			strhash_entry_t *e = entries;			/* Remove the top element from the linked list. */			entries = entries->next;			/* ..and move to the new hash. */			new_index = e->hash_value % new_size;			e->next = new_bins[new_index];			new_bins[new_index] = e;		}	}	hash->nbins = new_size;	hash->bins = new_bins;	if (old_bins != NULL)		free(old_bins);	return (idn_success);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -