2015-01-31 09:13:33 +00:00
|
|
|
/**
|
|
|
|
r_assoc.c
|
|
|
|
|
|
|
|
This is an associative array implementation, using an open-chained
|
|
|
|
hash bucket technique.
|
|
|
|
|
|
|
|
Note that this implementation permits each data entry to have
|
|
|
|
separate copy constructors and destructors. This currently wastes
|
|
|
|
space, but could be implemented while saving space by using
|
|
|
|
the high order bit of the length value or somesuch.
|
|
|
|
|
|
|
|
The major problem with this code is it's not resizable, though it
|
|
|
|
could be made so.
|
2023-08-14 10:37:08 +00:00
|
|
|
|
2015-01-31 09:13:33 +00:00
|
|
|
|
|
|
|
Copyright (C) 1999-2000 RTFM, Inc.
|
|
|
|
All Rights Reserved
|
|
|
|
|
|
|
|
This package is a SSLv3/TLS protocol analyzer written by Eric Rescorla
|
|
|
|
<ekr@rtfm.com> and licensed by RTFM, Inc.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
3. All advertising materials mentioning features or use of this software
|
|
|
|
must display the following acknowledgement:
|
2023-08-14 10:37:08 +00:00
|
|
|
|
2015-01-31 09:13:33 +00:00
|
|
|
This product includes software developed by Eric Rescorla for
|
|
|
|
RTFM, Inc.
|
|
|
|
|
|
|
|
4. Neither the name of RTFM, Inc. nor the name of Eric Rescorla may be
|
|
|
|
used to endorse or promote products derived from this
|
|
|
|
software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY ERIC RESCORLA AND RTFM, INC. ``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 THE REGENTS OR CONTRIBUTORS 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
|
2023-08-14 10:37:08 +00:00
|
|
|
OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY SUCH
|
|
|
|
DAMAGE.
|
2015-01-31 09:13:33 +00:00
|
|
|
|
|
|
|
$Id: r_assoc.c,v 1.4 2001/12/24 06:06:26 ekr Exp $
|
|
|
|
|
|
|
|
|
|
|
|
ekr@rtfm.com Sun Jan 17 17:57:15 1999
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <r_common.h>
|
|
|
|
#include "r_assoc.h"
|
|
|
|
|
|
|
|
typedef struct r_assoc_el_ {
|
2023-08-14 10:37:08 +00:00
|
|
|
char *key;
|
|
|
|
int key_len;
|
|
|
|
void *data;
|
|
|
|
struct r_assoc_el_ *prev;
|
|
|
|
struct r_assoc_el_ *next;
|
|
|
|
int(*copy) PROTO_LIST((void **new, void *old));
|
|
|
|
int(*destroy) PROTO_LIST((void *ptr));
|
2015-01-31 09:13:33 +00:00
|
|
|
} r_assoc_el;
|
2023-08-14 10:37:08 +00:00
|
|
|
|
2015-01-31 09:13:33 +00:00
|
|
|
struct r_assoc_ {
|
2023-08-14 10:37:08 +00:00
|
|
|
int size;
|
|
|
|
int bits;
|
|
|
|
r_assoc_el **chains;
|
2015-01-31 09:13:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#define DEFAULT_TABLE_BITS 5
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
static int destroy_assoc_chain PROTO_LIST((r_assoc_el * chain));
|
|
|
|
static int r_assoc_fetch_bucket
|
|
|
|
PROTO_LIST((r_assoc * assoc, char *key, int len, r_assoc_el **bucketp));
|
|
|
|
UINT4 hash_compute PROTO_LIST((char *key, int len, int size));
|
|
|
|
static int copy_assoc_chain PROTO_LIST((r_assoc_el * *newp, r_assoc_el *old));
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
int r_assoc_create(r_assoc **assocp) {
|
|
|
|
r_assoc *assoc = 0;
|
|
|
|
int _status;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!(assoc = (r_assoc *)calloc(sizeof(r_assoc), 1)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
|
|
|
assoc->size = (1 << DEFAULT_TABLE_BITS);
|
|
|
|
assoc->bits = DEFAULT_TABLE_BITS;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!(assoc->chains =
|
|
|
|
(r_assoc_el **)calloc(sizeof(r_assoc_el *), assoc->size)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
*assocp = assoc;
|
2021-01-15 10:51:02 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
_status = 0;
|
|
|
|
abort:
|
|
|
|
if(_status) {
|
|
|
|
r_assoc_destroy(&assoc);
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
2023-08-14 10:37:08 +00:00
|
|
|
return (_status);
|
|
|
|
}
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
int r_assoc_destroy(r_assoc **assocp) {
|
|
|
|
r_assoc *assoc;
|
|
|
|
int i;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!assocp || !*assocp)
|
|
|
|
return (0);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
assoc = *assocp;
|
|
|
|
for(i = 0; i < assoc->size; i++)
|
|
|
|
destroy_assoc_chain(assoc->chains[i]);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
free(assoc->chains);
|
|
|
|
free(assoc);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
static int destroy_assoc_chain(r_assoc_el *chain) {
|
|
|
|
r_assoc_el *nxt;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
while(chain) {
|
|
|
|
nxt = chain->next;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(chain->destroy)
|
|
|
|
chain->destroy(chain->data);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
free(chain->key);
|
|
|
|
|
|
|
|
free(chain);
|
|
|
|
chain = nxt;
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
static int copy_assoc_chain(r_assoc_el **newp, r_assoc_el *old) {
|
|
|
|
r_assoc_el *new = 0, *ptr, *tmp;
|
|
|
|
int r, _status;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!old) {
|
|
|
|
*newp = 0;
|
|
|
|
return (0);
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
2023-08-14 10:37:08 +00:00
|
|
|
for(; old; old = old->next) {
|
|
|
|
if(!(tmp = (r_assoc_el *)calloc(sizeof(r_assoc_el), 1)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!new) {
|
|
|
|
new = tmp;
|
|
|
|
ptr = new;
|
|
|
|
} else {
|
|
|
|
ptr->next = tmp;
|
|
|
|
tmp->prev = ptr;
|
|
|
|
ptr = tmp;
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
ptr->destroy = old->destroy;
|
|
|
|
ptr->copy = old->copy;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(old->copy) {
|
|
|
|
if((r = old->copy(&ptr->data, old->data)))
|
|
|
|
ABORT(r);
|
|
|
|
} else
|
|
|
|
ptr->data = old->data;
|
|
|
|
|
|
|
|
if(!(ptr->key = (char *)malloc(old->key_len)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
|
|
|
memcpy(ptr->key, old->key, ptr->key_len = old->key_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
*newp = new;
|
|
|
|
|
|
|
|
_status = 0;
|
|
|
|
abort:
|
|
|
|
if(_status) {
|
|
|
|
destroy_assoc_chain(new);
|
|
|
|
}
|
|
|
|
return (_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int r_assoc_fetch_bucket(r_assoc *assoc,
|
|
|
|
char *key,
|
|
|
|
int len,
|
|
|
|
r_assoc_el **bucketp) {
|
|
|
|
UINT4 hash_value;
|
|
|
|
r_assoc_el *bucket;
|
|
|
|
|
|
|
|
hash_value = hash_compute(key, len, assoc->bits);
|
|
|
|
|
|
|
|
for(bucket = assoc->chains[hash_value]; bucket; bucket = bucket->next) {
|
|
|
|
if(bucket->key_len == len && !memcmp(bucket->key, key, len)) {
|
|
|
|
*bucketp = bucket;
|
|
|
|
return (0);
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
return (R_NOT_FOUND);
|
|
|
|
}
|
|
|
|
|
|
|
|
int r_assoc_fetch(r_assoc *assoc, char *key, int len, void **datap) {
|
|
|
|
r_assoc_el *bucket;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
if((r = r_assoc_fetch_bucket(assoc, key, len, &bucket))) {
|
|
|
|
if(r != R_NOT_FOUND)
|
|
|
|
ERETURN(r);
|
|
|
|
return (r);
|
|
|
|
}
|
|
|
|
|
|
|
|
*datap = bucket->data;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int r_assoc_insert(r_assoc *assoc,
|
|
|
|
char *key,
|
|
|
|
int len,
|
|
|
|
void *data,
|
|
|
|
int(*copy) PROTO_LIST((void **new, void *old)),
|
|
|
|
int(*destroy) PROTO_LIST((void *ptr)),
|
|
|
|
int how) {
|
|
|
|
r_assoc_el *bucket, *new_bucket = 0;
|
|
|
|
int r, _status;
|
|
|
|
|
|
|
|
if((r = r_assoc_fetch_bucket(assoc, key, len, &bucket))) {
|
|
|
|
/*Note that we compute the hash value twice*/
|
|
|
|
UINT4 hash_value;
|
|
|
|
|
|
|
|
if(r != R_NOT_FOUND)
|
|
|
|
ABORT(r);
|
|
|
|
hash_value = hash_compute(key, len, assoc->bits);
|
|
|
|
|
|
|
|
if(!(new_bucket = (r_assoc_el *)calloc(sizeof(r_assoc_el), 1)))
|
2015-05-13 13:44:38 +00:00
|
|
|
ABORT(R_NO_MEMORY);
|
2023-08-14 10:37:08 +00:00
|
|
|
if(!(new_bucket->key = (char *)malloc(len)))
|
2015-01-31 09:13:33 +00:00
|
|
|
ABORT(R_NO_MEMORY);
|
2023-08-14 10:37:08 +00:00
|
|
|
memcpy(new_bucket->key, key, len);
|
|
|
|
new_bucket->key_len = len;
|
|
|
|
|
|
|
|
/*Insert at the list head. Is FIFO a good algorithm?*/
|
|
|
|
if(assoc->chains[hash_value])
|
|
|
|
assoc->chains[hash_value]->prev = new_bucket;
|
|
|
|
new_bucket->next = assoc->chains[hash_value];
|
|
|
|
assoc->chains[hash_value] = new_bucket;
|
|
|
|
bucket = new_bucket;
|
|
|
|
} else {
|
|
|
|
if(!(how & R_ASSOC_REPLACE))
|
|
|
|
ABORT(R_ALREADY);
|
|
|
|
|
|
|
|
if(bucket->destroy)
|
|
|
|
bucket->destroy(bucket->data);
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
bucket->data = data;
|
|
|
|
bucket->copy = copy;
|
|
|
|
bucket->destroy = destroy;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
_status = 0;
|
|
|
|
abort:
|
|
|
|
if(_status && new_bucket) {
|
|
|
|
free(new_bucket->key);
|
|
|
|
free(new_bucket);
|
|
|
|
}
|
|
|
|
return (_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
int r_assoc_copy(r_assoc **newp, r_assoc *old) {
|
|
|
|
int r, _status, i;
|
|
|
|
r_assoc *new;
|
|
|
|
|
|
|
|
if(!(new = (r_assoc *)calloc(sizeof(r_assoc), 1)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
|
|
|
new->size = old->size;
|
|
|
|
new->bits = old->bits;
|
|
|
|
|
|
|
|
if(!(new->chains = (r_assoc_el **)calloc(sizeof(r_assoc_el), old->size)))
|
|
|
|
ABORT(R_NO_MEMORY);
|
|
|
|
for(i = 0; i < new->size; i++) {
|
|
|
|
if((r = copy_assoc_chain(new->chains + i, old->chains[i])))
|
|
|
|
ABORT(R_NO_MEMORY);
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
2023-08-14 10:37:08 +00:00
|
|
|
*newp = new;
|
|
|
|
|
|
|
|
_status = 0;
|
|
|
|
abort:
|
|
|
|
if(_status) {
|
|
|
|
r_assoc_destroy(&new);
|
|
|
|
}
|
|
|
|
return (_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
int r_assoc_init_iter(r_assoc *assoc, r_assoc_iterator *iter) {
|
|
|
|
int i;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
iter->assoc = assoc;
|
|
|
|
iter->prev_chain = -1;
|
|
|
|
iter->prev = 0;
|
|
|
|
|
|
|
|
iter->next_chain = assoc->size;
|
|
|
|
iter->next = 0;
|
|
|
|
|
|
|
|
for(i = 0; i < assoc->size; i++) {
|
|
|
|
if(assoc->chains[i] != 0) {
|
|
|
|
iter->next_chain = i;
|
|
|
|
iter->next = assoc->chains[i];
|
|
|
|
break;
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
2023-08-14 10:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int r_assoc_iter(r_assoc_iterator *iter, void **key, int *keyl, void **val) {
|
|
|
|
int i;
|
|
|
|
r_assoc_el *ret;
|
|
|
|
|
|
|
|
if(!iter->next)
|
|
|
|
return (R_EOD);
|
|
|
|
ret = iter->next;
|
|
|
|
|
|
|
|
*key = ret->key;
|
|
|
|
*keyl = ret->key_len;
|
|
|
|
*val = ret->data;
|
|
|
|
|
|
|
|
/* Now increment */
|
|
|
|
iter->prev_chain = iter->next_chain;
|
|
|
|
iter->prev = iter->next;
|
|
|
|
|
|
|
|
/* More on this chain */
|
|
|
|
if(iter->next->next) {
|
|
|
|
iter->next = iter->next->next;
|
|
|
|
} else {
|
|
|
|
iter->next = 0;
|
|
|
|
|
|
|
|
/* FInd the next occupied chain*/
|
|
|
|
for(i = iter->next_chain; i < iter->assoc->size; i++) {
|
|
|
|
if(iter->assoc->chains[i]) {
|
|
|
|
iter->next_chain = i;
|
|
|
|
iter->next = iter->assoc->chains[i];
|
|
|
|
break;
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
/* Delete the last returned value*/
|
|
|
|
int r_assoc_iter_delete(r_assoc_iterator *iter) {
|
|
|
|
/* First unhook it from the list*/
|
|
|
|
if(!iter->prev->prev) {
|
|
|
|
/* First element*/
|
|
|
|
iter->assoc->chains[iter->prev_chain] = iter->prev->next;
|
|
|
|
} else {
|
|
|
|
iter->prev->prev->next = iter->prev->next;
|
|
|
|
}
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
if(iter->prev->next) {
|
|
|
|
iter->prev->next->prev = iter->prev->prev;
|
2015-01-31 09:13:33 +00:00
|
|
|
}
|
2023-08-14 10:37:08 +00:00
|
|
|
|
|
|
|
iter->prev->destroy(iter->prev->data);
|
|
|
|
free(iter->prev->data);
|
|
|
|
free(iter->prev);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2015-01-31 09:13:33 +00:00
|
|
|
/*This is a hack from AMS. Supposedly, it's pretty good for strings, even
|
|
|
|
though it doesn't take into account all the data*/
|
2023-08-14 10:37:08 +00:00
|
|
|
UINT4
|
|
|
|
hash_compute(char *key, int len, int bits) {
|
|
|
|
UINT4 h = 0;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
h = key[0] + (key[len - 1] * len);
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
h &= (1 << bits) - 1;
|
2015-01-31 09:13:33 +00:00
|
|
|
|
2023-08-14 10:37:08 +00:00
|
|
|
return (h);
|
|
|
|
}
|