X-Git-Url: https://git.m6w6.name/?a=blobdiff_plain;f=libmemcached%2Fmemcached_hash.c;h=395020a39afb2de33616be39633bf4be03cf2c0e;hb=fa11f4a5d1a9c92eda13f8a5aa11dcff542b8815;hp=5deb5c755f2708c353756125b31c43ac35b28a8b;hpb=34fc4157bda6f458cc603cf32c93507dab9c4f8a;p=m6w6%2Flibmemcached diff --git a/libmemcached/memcached_hash.c b/libmemcached/memcached_hash.c index 5deb5c75..395020a3 100644 --- a/libmemcached/memcached_hash.c +++ b/libmemcached/memcached_hash.c @@ -12,7 +12,7 @@ static uint32_t FNV_32_PRIME= 16777619; static uint32_t internal_generate_hash(const char *key, size_t key_length); static uint32_t internal_generate_md5(const char *key, size_t key_length); -uint32_t memcached_generate_hash_value(const char *key, size_t key_length, memcached_hash hash_algorithm) +uint32_t memcached_generate_hash_value(const char *key, size_t key_length, memcached_hash_t hash_algorithm) { uint32_t hash= 1; /* Just here to remove compile warning */ uint32_t x= 0; @@ -31,46 +31,49 @@ uint32_t memcached_generate_hash_value(const char *key, size_t key_length, memca hash= 1; break; /* FNV hash'es lifted from Dustin Sallings work */ - case MEMCACHED_HASH_FNV1_64: + case MEMCACHED_HASH_FNV1_64: { /* Thanks to pierre@demartines.com for the pointer */ uint64_t temp_hash; temp_hash= FNV_64_INIT; - for (x= 0; x < key_length; x++) + for (x= 0; x < key_length; x++) { temp_hash *= FNV_64_PRIME; - temp_hash ^= key[x]; + temp_hash ^= (uint64_t)key[x]; } hash= (uint32_t)temp_hash; } break; - case MEMCACHED_HASH_FNV1A_64: + case MEMCACHED_HASH_FNV1A_64: { - hash= FNV_64_INIT; - for (x= 0; x < key_length; x++) + hash= (uint32_t) FNV_64_INIT; + for (x= 0; x < key_length; x++) { - hash ^= key[x]; - hash *= FNV_64_PRIME; + uint32_t val= (uint32_t)key[x]; + hash ^= val; + hash *= (uint32_t) FNV_64_PRIME; } } break; - case MEMCACHED_HASH_FNV1_32: + case MEMCACHED_HASH_FNV1_32: { hash= FNV_32_INIT; - for (x= 0; x < key_length; x++) + for (x= 0; x < key_length; x++) { + uint32_t val= (uint32_t)key[x]; hash *= FNV_32_PRIME; - hash ^= key[x]; + hash ^= val; } } break; - case MEMCACHED_HASH_FNV1A_32: + case MEMCACHED_HASH_FNV1A_32: { hash= FNV_32_INIT; - for (x= 0; x < key_length; x++) + for (x= 0; x < key_length; x++) { - hash ^= key[x]; + uint32_t val= (uint32_t)key[x]; + hash ^= val; hash *= FNV_32_PRIME; } } @@ -118,10 +121,11 @@ uint32_t generate_hash(memcached_st *ptr, const char *key, size_t key_length) static uint32_t dispatch_host(memcached_st *ptr, uint32_t hash) { - switch (ptr->distribution) + switch (ptr->distribution) { case MEMCACHED_DISTRIBUTION_CONSISTENT: case MEMCACHED_DISTRIBUTION_CONSISTENT_KETAMA: + case MEMCACHED_DISTRIBUTION_CONSISTENT_KETAMA_SPY: { uint32_t num= ptr->continuum_points_counter; WATCHPOINT_ASSERT(ptr->continuum); @@ -142,11 +146,11 @@ static uint32_t dispatch_host(memcached_st *ptr, uint32_t hash) if (right == end) right= begin; return right->index; - } + } case MEMCACHED_DISTRIBUTION_MODULA: return hash % ptr->number_of_hosts; case MEMCACHED_DISTRIBUTION_RANDOM: - return random() % ptr->number_of_hosts; + return (uint32_t) random() % ptr->number_of_hosts; default: WATCHPOINT_ASSERT(0); /* We have added a distribution without extending the logic */ return hash % ptr->number_of_hosts; @@ -155,8 +159,8 @@ static uint32_t dispatch_host(memcached_st *ptr, uint32_t hash) /* NOTREACHED */ } -/* - One day make this public, and have it return the actual memcached_server_st +/* + One day make this public, and have it return the actual memcached_server_st to the calling application. */ uint32_t memcached_generate_hash(memcached_st *ptr, const char *key, size_t key_length) @@ -168,14 +172,17 @@ uint32_t memcached_generate_hash(memcached_st *ptr, const char *key, size_t key_ if (ptr->number_of_hosts == 1) return 0; - if (ptr->flags & MEM_HASH_WITH_PREFIX_KEY) + if (ptr->flags.hash_with_prefix_key) { - int temp_len= ptr->prefix_key_length + key_length; - char *temp= (char *)malloc(temp_len); + size_t temp_length= ptr->prefix_key_length + key_length; + char temp[temp_length]; + + if (temp_length > MEMCACHED_MAX_KEY -1) + return 0; + strncpy(temp, ptr->prefix_key, ptr->prefix_key_length); strncpy(temp + ptr->prefix_key_length, key, key_length); - hash= generate_hash(ptr, temp, temp_len); - free(temp); + hash= generate_hash(ptr, temp, temp_length); } else { @@ -200,17 +207,18 @@ static uint32_t internal_generate_hash(const char *key, size_t key_length) const char *ptr= key; uint32_t value= 0; - while (key_length--) + while (key_length--) { - value += *ptr++; + uint32_t val= (uint32_t) *ptr++; + value += val; value += (value << 10); value ^= (value >> 6); } value += (value << 3); value ^= (value >> 11); - value += (value << 15); + value += (value << 15); - return value == 0 ? 1 : value; + return value == 0 ? 1 : (uint32_t) value; } static uint32_t internal_generate_md5(const char *key, size_t key_length)