From 65f3737325490e35501df9d64ca8a4728edc776d Mon Sep 17 00:00:00 2001 From: Eric Fischer Date: Thu, 2 Jul 2015 17:03:07 -0700 Subject: [PATCH] Use a hash table of binary trees for string pooling, not just a binary tree --- pool.c | 29 ++++++++++++++++++++--------- pool.h | 2 +- 2 files changed, 21 insertions(+), 10 deletions(-) diff --git a/pool.c b/pool.c index a36140d..e874d23 100644 --- a/pool.c +++ b/pool.c @@ -2,11 +2,23 @@ #include #include "pool.h" -static struct pool_val *pool1(struct pool *p, char *s, int type, int (*compare)(const char *, const char *)) { - struct pool_val **v = &(p->vals); +#define POOL_WIDTH 256 + +static int hash(char *s) { + int h = 0; + for (; *s; s++) { + h = h * 37 + *s; + } + h = h & 0xFF; + return h; +} + +struct pool_val *pool(struct pool *p, char *s, int type) { + int h = hash(s); + struct pool_val **v = &(p->vals[h]); while (*v != NULL) { - int cmp = compare(s, (*v)->s); + int cmp = strcmp(s, (*v)->s); if (cmp == 0) { cmp = type - (*v)->type; @@ -41,7 +53,8 @@ static struct pool_val *pool1(struct pool *p, char *s, int type, int (*compare)( } int is_pooled(struct pool *p, char *s, int type) { - struct pool_val **v = &(p->vals); + int h = hash(s); + struct pool_val **v = &(p->vals[h]); while (*v != NULL) { int cmp = strcmp(s, (*v)->s); @@ -62,10 +75,6 @@ int is_pooled(struct pool *p, char *s, int type) { return 0; } -struct pool_val *pool(struct pool *p, char *s, int type) { - return pool1(p, s, type, strcmp); -} - void pool_free1(struct pool *p, void (*func)(void *)) { while (p->head != NULL) { if (func != NULL) { @@ -79,6 +88,8 @@ void pool_free1(struct pool *p, void (*func)(void *)) { p->head = NULL; p->tail = NULL; + + free(p->vals); p->vals = NULL; } @@ -92,7 +103,7 @@ void pool_free_strings(struct pool *p) { void pool_init(struct pool *p, int n) { p->n = n; - p->vals = NULL; + p->vals = calloc(POOL_WIDTH, sizeof(struct pool_val *)); p->head = NULL; p->tail = NULL; } diff --git a/pool.h b/pool.h index d995e5b..147e8c8 100644 --- a/pool.h +++ b/pool.h @@ -10,7 +10,7 @@ struct pool_val { }; struct pool { - struct pool_val *vals; + struct pool_val **vals; struct pool_val *head; struct pool_val *tail;