From f5da908c4936ea384becd6a8f05e75f8ff01cef0 Mon Sep 17 00:00:00 2001 From: Hallvard Furuseth Date: Wed, 9 May 2007 22:22:44 +0000 Subject: [PATCH] thread_keys is a (poor) open-addessed hash table, but it lacked a "deleted item" mark. Add DELETED_THREAD_CTX. Also improve the hash function a bit, and make the hash unsigned. --- libraries/libldap_r/tpool.c | 41 ++++++++++++++++++++++--------------- 1 file changed, 25 insertions(+), 16 deletions(-) diff --git a/libraries/libldap_r/tpool.c b/libraries/libldap_r/tpool.c index 162415ca8a..1a62f3f0f9 100644 --- a/libraries/libldap_r/tpool.c +++ b/libraries/libldap_r/tpool.c @@ -56,15 +56,23 @@ typedef struct ldap_int_thread_userctx_s { static ldap_pvt_thread_t tid_zero; -/* Thread ID -> context mapping. +/* Thread ID -> context mapping (poor open-addressed hash table). * Protected by ldap_pvt_thread_pool_mutex except during pauses, * when it is reserved for ldap_pvt_thread_pool_purgekey(). */ static struct { ldap_pvt_thread_t id; - ldap_int_thread_userctx_t *ctx; + ldap_int_thread_userctx_t *ctx; /* set when id != tid_zero */ +# define DELETED_THREAD_CTX (&ldap_int_main_thrctx + 1) /* dummy addr */ } thread_keys[LDAP_MAXTHR]; - + +#define TID_HASH(tid, hash) do { \ + unsigned const char *ptr_ = (unsigned const char *)&(tid); \ + unsigned i_; \ + for (i_ = 0, (hash) = ptr_[0]; ++i_ < sizeof(tid);) \ + (hash) += ((hash) << 5) ^ ptr_[i_]; \ +} while(0) + typedef struct ldap_int_thread_ctx_s { union { @@ -324,9 +332,6 @@ ldap_pvt_thread_pool_init ( return(0); } -#define TID_HASH(tid, hash) do { unsigned i; \ - unsigned char *ptr = (unsigned char *)&(tid); \ - for (i=0, hash=0; iltp_cond, &pool->ltp_mutex); ldap_pvt_thread_mutex_lock(&ldap_pvt_thread_pool_mutex); - thread_keys[keyslot].ctx = NULL; + thread_keys[keyslot].ctx = DELETED_THREAD_CTX; thread_keys[keyslot].id = tid_zero; ldap_pvt_thread_mutex_unlock(&ldap_pvt_thread_pool_mutex); @@ -876,8 +881,8 @@ void ldap_pvt_thread_pool_purgekey( void *key ) ldap_int_thread_userctx_t *ctx; for ( i=0; iltu_key[j].ltk_key == key ) { if (ctx->ltu_key[j].ltk_free) @@ -902,7 +907,7 @@ void ldap_pvt_thread_pool_purgekey( void *key ) void *ldap_pvt_thread_pool_context( ) { ldap_pvt_thread_t tid; - int i, hash; + unsigned i, hash; ldap_int_thread_userctx_t *ctx; tid = ldap_pvt_thread_self(); @@ -910,12 +915,16 @@ void *ldap_pvt_thread_pool_context( ) return &ldap_int_main_thrctx; TID_HASH( tid, hash ); + i = hash &= (LDAP_MAXTHR-1); ldap_pvt_thread_mutex_lock(&ldap_pvt_thread_pool_mutex); - for (i = hash & (LDAP_MAXTHR-1); - !ldap_pvt_thread_equal(thread_keys[i].id, tid_zero) && - !ldap_pvt_thread_equal(thread_keys[i].id, tid); - i = (i+1) & (LDAP_MAXTHR-1)); - ctx = thread_keys[i].ctx; + do { + ctx = thread_keys[i].ctx; + if ( ctx != DELETED_THREAD_CTX ) + if ( ldap_pvt_thread_equal(thread_keys[i].id, tid) || !ctx ) + goto done; + } while ( (i = (i+1) & (LDAP_MAXTHR-1)) != hash ); + ctx = NULL; + done: ldap_pvt_thread_mutex_unlock(&ldap_pvt_thread_pool_mutex); return ctx;