| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): none yet. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-04-16 22:40:48 +00:00
										 |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:37:56 +00:00
										 |  |  | /** \file blender/blenlib/intern/BLI_ghash.c
 | 
					
						
							|  |  |  |  *  \ingroup bli | 
					
						
							| 
									
										
										
										
											2013-06-22 22:30:56 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-08-18 01:00:52 +00:00
										 |  |  |  * A general (pointer -> pointer) hash table ADT | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-06-22 22:30:56 +00:00
										 |  |  |  * \note edgehash.c is based on this, make sure they stay in sync. | 
					
						
							| 
									
										
										
										
											2011-02-27 20:37:56 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-20 13:50:24 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2013-07-21 17:05:41 +00:00
										 |  |  | #include <limits.h>
 | 
					
						
							| 
									
										
										
										
											2011-10-20 13:50:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-17 11:10:48 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-28 19:35:26 +00:00
										 |  |  | #include "BLI_sys_types.h"  /* for intptr_t support */
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2011-10-24 08:18:20 +00:00
										 |  |  | #include "BLI_mempool.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BLI_ghash.h"
 | 
					
						
							| 
									
										
										
										
											2011-10-20 13:50:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | /***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:51 +00:00
										 |  |  | #ifdef __GNUC__
 | 
					
						
							|  |  |  | #  pragma GCC diagnostic error "-Wsign-conversion"
 | 
					
						
							| 
									
										
										
										
											2013-06-22 20:20:06 +00:00
										 |  |  | #  if (__GNUC__ * 100 + __GNUC_MINOR__) >= 406  /* gcc4.6+ only */
 | 
					
						
							|  |  |  | #    pragma GCC diagnostic error "-Wsign-compare"
 | 
					
						
							|  |  |  | #    pragma GCC diagnostic error "-Wconversion"
 | 
					
						
							|  |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											2013-06-22 20:00:17 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-08 12:54:07 +00:00
										 |  |  | const unsigned int hashsizes[] = { | 
					
						
							| 
									
										
										
										
											2010-01-23 11:25:20 +00:00
										 |  |  | 	5, 11, 17, 37, 67, 131, 257, 521, 1031, 2053, 4099, 8209,  | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	16411, 32771, 65537, 131101, 262147, 524309, 1048583, 2097169,  | 
					
						
							|  |  |  | 	4194319, 8388617, 16777259, 33554467, 67108879, 134217757,  | 
					
						
							|  |  |  | 	268435459 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 00:36:04 +00:00
										 |  |  | typedef struct Entry { | 
					
						
							|  |  |  | 	struct Entry *next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void *key, *val; | 
					
						
							|  |  |  | } Entry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct GHash { | 
					
						
							|  |  |  | 	GHashHashFP hashfp; | 
					
						
							|  |  |  | 	GHashCmpFP cmpfp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Entry **buckets; | 
					
						
							|  |  |  | 	struct BLI_mempool *entrypool; | 
					
						
							|  |  |  | 	unsigned int nbuckets; | 
					
						
							|  |  |  | 	unsigned int nentries; | 
					
						
							|  |  |  | 	unsigned short cursize, flag; | 
					
						
							|  |  |  | } GHash; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* GHash API */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | /* internal utility API */ | 
					
						
							| 
									
										
										
										
											2009-02-12 18:05:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | BLI_INLINE unsigned int ghash_keyhash(GHash *gh, const void *key) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return gh->hashfp(key) % gh->nbuckets; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | BLI_INLINE Entry *ghash_lookup_entry_ex(GHash *gh, const void *key, | 
					
						
							|  |  |  |                                         const unsigned int hash) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Entry *e; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	for (e = gh->buckets[hash]; e; e = e->next) { | 
					
						
							|  |  |  | 		if (gh->cmpfp(key, e->key) == 0) { | 
					
						
							|  |  |  | 			return e; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | BLI_INLINE Entry *ghash_lookup_entry(GHash *gh, const void *key) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	const unsigned int hash = ghash_keyhash(gh, key); | 
					
						
							|  |  |  | 	return ghash_lookup_entry_ex(gh, key, hash); | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | static void ghash_insert_ex(GHash *gh, void *key, void *val, | 
					
						
							|  |  |  |                             unsigned int hash) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-12 15:02:10 +00:00
										 |  |  | 	Entry *e = (Entry *)BLI_mempool_alloc(gh->entrypool); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 00:36:04 +00:00
										 |  |  | 	BLI_assert((gh->flag & GHASH_FLAG_ALLOW_DUPES) || (BLI_ghash_haskey(gh, key) == 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-03 15:03:55 +00:00
										 |  |  | 	e->next = gh->buckets[hash]; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	e->key = key; | 
					
						
							|  |  |  | 	e->val = val; | 
					
						
							|  |  |  | 	gh->buckets[hash] = e; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-22 22:30:56 +00:00
										 |  |  | 	if (UNLIKELY(++gh->nentries > gh->nbuckets / 2)) { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		Entry **old = gh->buckets; | 
					
						
							| 
									
										
										
										
											2013-06-22 22:30:56 +00:00
										 |  |  | 		const unsigned nold = gh->nbuckets; | 
					
						
							|  |  |  | 		unsigned int i; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		gh->nbuckets = hashsizes[++gh->cursize]; | 
					
						
							| 
									
										
										
										
											2013-03-18 16:34:57 +00:00
										 |  |  | 		gh->buckets = (Entry **)MEM_callocN(gh->nbuckets * sizeof(*gh->buckets), "buckets"); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		for (i = 0; i < nold; i++) { | 
					
						
							| 
									
										
										
										
											2013-06-22 22:30:56 +00:00
										 |  |  | 			Entry *e_next; | 
					
						
							|  |  |  | 			for (e = old[i]; e; e = e_next) { | 
					
						
							|  |  |  | 				e_next = e->next; | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 				hash = ghash_keyhash(gh, e->key); | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 				e->next = gh->buckets[hash]; | 
					
						
							|  |  |  | 				gh->buckets[hash] = e; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MEM_freeN(old); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Public API */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GHash *BLI_ghash_new(GHashHashFP hashfp, GHashCmpFP cmpfp, const char *info) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	GHash *gh = MEM_mallocN(sizeof(*gh), info); | 
					
						
							|  |  |  | 	gh->hashfp = hashfp; | 
					
						
							|  |  |  | 	gh->cmpfp = cmpfp; | 
					
						
							|  |  |  | 	gh->entrypool = BLI_mempool_create(sizeof(Entry), 64, 64, 0); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	gh->cursize = 0; | 
					
						
							|  |  |  | 	gh->nentries = 0; | 
					
						
							|  |  |  | 	gh->nbuckets = hashsizes[gh->cursize]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gh->buckets = MEM_callocN(gh->nbuckets * sizeof(*gh->buckets), "buckets"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return gh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int BLI_ghash_size(GHash *gh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (int)gh->nentries; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_ghash_insert(GHash *gh, void *key, void *val) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const unsigned int hash = ghash_keyhash(gh, key); | 
					
						
							|  |  |  | 	ghash_insert_ex(gh, key, val, hash); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Assign a new value to a key that may already be in ghash. | 
					
						
							|  |  |  |  * Avoids #BLI_ghash_remove, #BLI_ghash_insert calls (double lookups) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note We may want to have 'BLI_ghash_assign_ex' function that takes | 
					
						
							|  |  |  |  * GHashKeyFreeFP & GHashValFreeFP args. for now aren't needed. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void BLI_ghash_assign(GHash *gh, void *key, void *val, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const unsigned int hash = ghash_keyhash(gh, key); | 
					
						
							|  |  |  | 	Entry *e = ghash_lookup_entry_ex(gh, key, hash); | 
					
						
							|  |  |  | 	if (e) { | 
					
						
							|  |  |  | 		if (keyfreefp) keyfreefp(e->key); | 
					
						
							|  |  |  | 		if (valfreefp) valfreefp(e->val); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		e->key = key; | 
					
						
							|  |  |  | 		e->val = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		ghash_insert_ex(gh, key, val, hash); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 01:00:52 +00:00
										 |  |  | void *BLI_ghash_lookup(GHash *gh, const void *key) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Entry *e = ghash_lookup_entry(gh, key); | 
					
						
							|  |  |  | 	return e ? e->val : NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void **BLI_ghash_lookup_p(GHash *gh, const void *key) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Entry *e = ghash_lookup_entry(gh, key); | 
					
						
							|  |  |  | 	return e ? &e->val : NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 19:27:51 +00:00
										 |  |  | bool BLI_ghash_remove(GHash *gh, void *key, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp) | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	const unsigned int hash = ghash_keyhash(gh, key); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 	Entry *e; | 
					
						
							|  |  |  | 	Entry *p = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	for (e = gh->buckets[hash]; e; e = e->next) { | 
					
						
							|  |  |  | 		if (gh->cmpfp(key, e->key) == 0) { | 
					
						
							|  |  |  | 			Entry *n = e->next; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-28 19:33:14 +00:00
										 |  |  | 			if (keyfreefp) keyfreefp(e->key); | 
					
						
							|  |  |  | 			if (valfreefp) valfreefp(e->val); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 			BLI_mempool_free(gh->entrypool, e); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-18 07:38:51 +00:00
										 |  |  | 			/* correct but 'e' isn't used before return */ | 
					
						
							| 
									
										
										
										
											2012-10-23 03:38:26 +00:00
										 |  |  | 			/* e = n; *//*UNUSED*/ | 
					
						
							| 
									
										
										
										
											2012-05-28 19:33:14 +00:00
										 |  |  | 			if (p) p->next = n; | 
					
						
							|  |  |  | 			else   gh->buckets[hash] = n; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-09 09:24:15 +00:00
										 |  |  | 			gh->nentries--; | 
					
						
							| 
									
										
										
										
											2013-03-04 19:27:51 +00:00
										 |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		p = e; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 19:27:51 +00:00
										 |  |  | 	return false; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-28 19:33:14 +00:00
										 |  |  | /* same as above but return the value,
 | 
					
						
							|  |  |  |  * no free value argument since it will be returned */ | 
					
						
							|  |  |  | void *BLI_ghash_pop(GHash *gh, void *key, GHashKeyFreeFP keyfreefp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-18 03:41:39 +00:00
										 |  |  | 	const unsigned int hash = ghash_keyhash(gh, key); | 
					
						
							| 
									
										
										
										
											2012-05-28 19:33:14 +00:00
										 |  |  | 	Entry *e; | 
					
						
							|  |  |  | 	Entry *p = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (e = gh->buckets[hash]; e; e = e->next) { | 
					
						
							|  |  |  | 		if (gh->cmpfp(key, e->key) == 0) { | 
					
						
							|  |  |  | 			Entry *n = e->next; | 
					
						
							|  |  |  | 			void *value = e->val; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (keyfreefp) keyfreefp(e->key); | 
					
						
							|  |  |  | 			BLI_mempool_free(gh->entrypool, e); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* correct but 'e' isn't used before return */ | 
					
						
							| 
									
										
										
										
											2012-10-23 03:38:26 +00:00
										 |  |  | 			/* e = n; *//*UNUSED*/ | 
					
						
							| 
									
										
										
										
											2012-05-28 19:33:14 +00:00
										 |  |  | 			if (p) p->next = n; | 
					
						
							|  |  |  | 			else   gh->buckets[hash] = n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			gh->nentries--; | 
					
						
							|  |  |  | 			return value; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		p = e; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 19:27:51 +00:00
										 |  |  | bool BLI_ghash_haskey(GHash *gh, const void *key) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-18 01:00:52 +00:00
										 |  |  | 	return (ghash_lookup_entry(gh, key) != NULL); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 01:00:52 +00:00
										 |  |  | void BLI_ghash_clear(GHash *gh, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 01:00:52 +00:00
										 |  |  | 	if (keyfreefp || valfreefp) { | 
					
						
							|  |  |  | 		for (i = 0; i < gh->nbuckets; i++) { | 
					
						
							|  |  |  | 			Entry *e; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for (e = gh->buckets[i]; e; ) { | 
					
						
							|  |  |  | 				Entry *n = e->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (keyfreefp) keyfreefp(e->key); | 
					
						
							|  |  |  | 				if (valfreefp) valfreefp(e->val); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				e = n; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gh->cursize = 0; | 
					
						
							|  |  |  | 	gh->nentries = 0; | 
					
						
							|  |  |  | 	gh->nbuckets = hashsizes[gh->cursize]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(gh->buckets); | 
					
						
							|  |  |  | 	gh->buckets = MEM_callocN(gh->nbuckets * sizeof(*gh->buckets), "buckets"); | 
					
						
							| 
									
										
										
										
											2011-08-12 02:23:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void BLI_ghash_free(GHash *gh, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-20 19:39:29 +00:00
										 |  |  | 	unsigned int i; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-06 06:47:59 +00:00
										 |  |  | 	if (keyfreefp || valfreefp) { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		for (i = 0; i < gh->nbuckets; i++) { | 
					
						
							| 
									
										
										
										
											2009-09-06 06:47:59 +00:00
										 |  |  | 			Entry *e; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-12 15:02:10 +00:00
										 |  |  | 			for (e = gh->buckets[i]; e; ) { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 				Entry *n = e->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-06 06:47:59 +00:00
										 |  |  | 				if (keyfreefp) keyfreefp(e->key); | 
					
						
							|  |  |  | 				if (valfreefp) valfreefp(e->val); | 
					
						
							| 
									
										
										
										
											2009-02-13 03:27:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 				e = n; | 
					
						
							| 
									
										
										
										
											2009-09-06 06:47:59 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-23 11:25:20 +00:00
										 |  |  | 	MEM_freeN(gh->buckets); | 
					
						
							| 
									
										
										
										
											2009-02-12 18:05:34 +00:00
										 |  |  | 	BLI_mempool_destroy(gh->entrypool); | 
					
						
							| 
									
										
										
										
											2011-02-13 10:52:18 +00:00
										 |  |  | 	gh->buckets = NULL; | 
					
						
							| 
									
										
										
										
											2005-12-21 22:21:43 +00:00
										 |  |  | 	gh->nentries = 0; | 
					
						
							|  |  |  | 	gh->nbuckets = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	MEM_freeN(gh); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 00:36:04 +00:00
										 |  |  | void BLI_ghash_flag_set(GHash *gh, unsigned short flag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	gh->flag |= flag; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BLI_ghash_flag_clear(GHash *gh, unsigned short flag) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	gh->flag &= (unsigned short)~flag; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* GHash Iterator API */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | GHashIterator *BLI_ghashIterator_new(GHash *gh) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	GHashIterator *ghi = MEM_mallocN(sizeof(*ghi), "ghash iterator"); | 
					
						
							| 
									
										
										
										
											2013-07-21 17:05:41 +00:00
										 |  |  | 	BLI_ghashIterator_init(ghi, gh); | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | 	return ghi; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void BLI_ghashIterator_init(GHashIterator *ghi, GHash *gh) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	ghi->gh = gh; | 
					
						
							|  |  |  | 	ghi->curEntry = NULL; | 
					
						
							| 
									
										
										
										
											2013-07-21 17:05:41 +00:00
										 |  |  | 	ghi->curBucket = UINT_MAX;  /* wraps to zero */ | 
					
						
							| 
									
										
										
										
											2008-05-27 13:22:17 +00:00
										 |  |  | 	while (!ghi->curEntry) { | 
					
						
							|  |  |  | 		ghi->curBucket++; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		if (ghi->curBucket == ghi->gh->nbuckets) | 
					
						
							| 
									
										
										
										
											2008-05-27 13:22:17 +00:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		ghi->curEntry = ghi->gh->buckets[ghi->curBucket]; | 
					
						
							| 
									
										
										
										
											2008-05-27 13:22:17 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void BLI_ghashIterator_free(GHashIterator *ghi) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-23 11:25:20 +00:00
										 |  |  | 	MEM_freeN(ghi); | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void *BLI_ghashIterator_getKey(GHashIterator *ghi) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	return ghi->curEntry ? ghi->curEntry->key : NULL; | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void *BLI_ghashIterator_getValue(GHashIterator *ghi) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	return ghi->curEntry ? ghi->curEntry->val : NULL; | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void BLI_ghashIterator_step(GHashIterator *ghi) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | 	if (ghi->curEntry) { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		ghi->curEntry = ghi->curEntry->next; | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | 		while (!ghi->curEntry) { | 
					
						
							|  |  |  | 			ghi->curBucket++; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 			if (ghi->curBucket == ghi->gh->nbuckets) | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 			ghi->curEntry = ghi->gh->buckets[ghi->curBucket]; | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-05-08 12:58:11 +00:00
										 |  |  | bool BLI_ghashIterator_done(GHashIterator *ghi) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-08 12:58:11 +00:00
										 |  |  | 	return ghi->curEntry == NULL; | 
					
						
							| 
									
										
										
										
											2004-01-07 06:13:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | unsigned int BLI_ghashutil_ptrhash(const void *key) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-17 00:51:42 +00:00
										 |  |  | 	return (unsigned int)(intptr_t)key; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | int BLI_ghashutil_ptrcmp(const void *a, const void *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	if (a == b) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		return (a < b) ? -1 : 1; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | unsigned int BLI_ghashutil_inthash(const void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-08-17 17:08:00 +00:00
										 |  |  | 	uintptr_t key = (uintptr_t)ptr; | 
					
						
							| 
									
										
										
										
											2005-12-21 22:21:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	key += ~(key << 16); | 
					
						
							|  |  |  | 	key ^=  (key >>  5); | 
					
						
							|  |  |  | 	key +=  (key <<  3); | 
					
						
							|  |  |  | 	key ^=  (key >> 13); | 
					
						
							|  |  |  | 	key += ~(key <<  9); | 
					
						
							|  |  |  | 	key ^=  (key >> 17); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-21 15:53:30 +00:00
										 |  |  | 	return (unsigned int)(key & 0xffffffff); | 
					
						
							| 
									
										
										
										
											2005-12-21 22:21:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | int BLI_ghashutil_intcmp(const void *a, const void *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	if (a == b) | 
					
						
							| 
									
										
										
										
											2005-12-21 22:21:43 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 		return (a < b) ? -1 : 1; | 
					
						
							| 
									
										
										
										
											2005-12-21 22:21:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 13:57:58 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * This function implements the widely used "djb" hash apparently posted | 
					
						
							|  |  |  |  * by Daniel Bernstein to comp.lang.c some time ago.  The 32 bit | 
					
						
							|  |  |  |  * unsigned hash value starts at 5381 and for each byte 'c' in the | 
					
						
							|  |  |  |  * string, is updated: <literal>hash = hash * 33 + c</literal>.  This | 
					
						
							|  |  |  |  * function uses the signed value of each byte. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * note: this is the same hash method that glib 2.34.0 uses. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | unsigned int BLI_ghashutil_strhash(const void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-11 13:57:58 +00:00
										 |  |  | 	const signed char *p; | 
					
						
							|  |  |  | 	unsigned int h = 5381; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 13:57:58 +00:00
										 |  |  | 	for (p = ptr; *p != '\0'; p++) { | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:51 +00:00
										 |  |  | 		h = (h << 5) + h + (unsigned int)*p; | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 13:57:58 +00:00
										 |  |  | 	return h; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | int BLI_ghashutil_strcmp(const void *a, const void *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	return strcmp(a, b); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-16 00:51:36 +00:00
										 |  |  | GHash *BLI_ghash_ptr_new(const char *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return BLI_ghash_new(BLI_ghashutil_ptrhash, BLI_ghashutil_ptrcmp, info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | GHash *BLI_ghash_str_new(const char *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return BLI_ghash_new(BLI_ghashutil_strhash, BLI_ghashutil_strcmp, info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | GHash *BLI_ghash_int_new(const char *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return BLI_ghash_new(BLI_ghashutil_inthash, BLI_ghashutil_intcmp, info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | GHash *BLI_ghash_pair_new(const char *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return BLI_ghash_new(BLI_ghashutil_pairhash, BLI_ghashutil_paircmp, info); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-04 12:46:51 +00:00
										 |  |  | GHashPair *BLI_ghashutil_pairalloc(const void *first, const void *second) | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | 	GHashPair *pair = MEM_mallocN(sizeof(GHashPair), "GHashPair"); | 
					
						
							|  |  |  | 	pair->first = first; | 
					
						
							|  |  |  | 	pair->second = second; | 
					
						
							|  |  |  | 	return pair; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | unsigned int BLI_ghashutil_pairhash(const void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | 	const GHashPair *pair = ptr; | 
					
						
							|  |  |  | 	unsigned int hash = BLI_ghashutil_ptrhash(pair->first); | 
					
						
							| 
									
										
										
										
											2012-05-04 12:46:51 +00:00
										 |  |  | 	return hash ^ BLI_ghashutil_ptrhash(pair->second); | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | int BLI_ghashutil_paircmp(const void *a, const void *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | 	const GHashPair *A = a; | 
					
						
							|  |  |  | 	const GHashPair *B = b; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int cmp = BLI_ghashutil_ptrcmp(A->first, B->first); | 
					
						
							| 
									
										
										
										
											2012-03-11 23:47:41 +00:00
										 |  |  | 	if (cmp == 0) | 
					
						
							| 
									
										
										
										
											2012-05-04 12:46:51 +00:00
										 |  |  | 		return BLI_ghashutil_ptrcmp(A->second, B->second); | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | 	return cmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-17 00:52:36 +00:00
										 |  |  | void BLI_ghashutil_pairfree(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-21 08:45:10 +00:00
										 |  |  | 	MEM_freeN(ptr); | 
					
						
							| 
									
										
										
										
											2011-12-16 23:56:18 +00:00
										 |  |  | } |