ruby/yarp/util/yp_constant_pool.c

148 строки
4.9 KiB
C

#include "yarp/util/yp_constant_pool.h"
// Initialize a list of constant ids.
void
yp_constant_id_list_init(yp_constant_id_list_t *list) {
list->ids = NULL;
list->size = 0;
list->capacity = 0;
}
// Append a constant id to a list of constant ids. Returns false if any
// potential reallocations fail.
bool
yp_constant_id_list_append(yp_constant_id_list_t *list, yp_constant_id_t id) {
if (list->size >= list->capacity) {
list->capacity = list->capacity == 0 ? 8 : list->capacity * 2;
list->ids = (yp_constant_id_t *) realloc(list->ids, sizeof(yp_constant_id_t) * list->capacity);
if (list->ids == NULL) return false;
}
list->ids[list->size++] = id;
return true;
}
// Checks if the current constant id list includes the given constant id.
bool
yp_constant_id_list_includes(yp_constant_id_list_t *list, yp_constant_id_t id) {
for (size_t index = 0; index < list->size; index++) {
if (list->ids[index] == id) return true;
}
return false;
}
// Get the memory size of a list of constant ids.
size_t
yp_constant_id_list_memsize(yp_constant_id_list_t *list) {
return sizeof(yp_constant_id_list_t) + (list->capacity * sizeof(yp_constant_id_t));
}
// Free the memory associated with a list of constant ids.
void
yp_constant_id_list_free(yp_constant_id_list_t *list) {
if (list->ids != NULL) {
free(list->ids);
}
}
// A relatively simple hash function (djb2) that is used to hash strings. We are
// optimizing here for simplicity and speed.
static inline size_t
yp_constant_pool_hash(const char *start, size_t length) {
// This is a prime number used as the initial value for the hash function.
size_t value = 5381;
for (size_t index = 0; index < length; index++) {
value = ((value << 5) + value) + ((unsigned char) start[index]);
}
return value;
}
// Resize a constant pool to a given capacity.
static inline bool
yp_constant_pool_resize(yp_constant_pool_t *pool) {
size_t next_capacity = pool->capacity * 2;
yp_constant_t *next_constants = calloc(next_capacity, sizeof(yp_constant_t));
if (next_constants == NULL) return false;
// For each constant in the current constant pool, rehash the content, find
// the index in the next constant pool, and insert it.
for (size_t index = 0; index < pool->capacity; index++) {
yp_constant_t *constant = &pool->constants[index];
// If an id is set on this constant, then we know we have content here.
// In this case we need to insert it into the next constant pool.
if (constant->id != 0) {
size_t next_index = constant->hash % next_capacity;
// This implements linear scanning to find the next available slot
// in case this index is already taken. We don't need to bother
// comparing the values since we know that the hash is unique.
while (next_constants[next_index].id != 0) {
next_index = (next_index + 1) % next_capacity;
}
// Here we copy over the entire constant, which includes the id so
// that they are consistent between resizes.
next_constants[next_index] = *constant;
}
}
free(pool->constants);
pool->constants = next_constants;
pool->capacity = next_capacity;
return true;
}
// Initialize a new constant pool with a given capacity.
bool
yp_constant_pool_init(yp_constant_pool_t *pool, size_t capacity) {
pool->constants = calloc(capacity, sizeof(yp_constant_t));
if (pool->constants == NULL) return false;
pool->size = 0;
pool->capacity = capacity;
return true;
}
// Insert a constant into a constant pool. Returns the id of the constant, or 0
// if any potential calls to resize fail.
yp_constant_id_t
yp_constant_pool_insert(yp_constant_pool_t *pool, const char *start, size_t length) {
if (pool->size >= pool->capacity * 0.75) {
if (!yp_constant_pool_resize(pool)) return 0;
}
size_t hash = yp_constant_pool_hash(start, length);
size_t index = hash % pool->capacity;
yp_constant_t *constant;
while (constant = &pool->constants[index], constant->id != 0) {
// If there is a collision, then we need to check if the content is the
// same as the content we are trying to insert. If it is, then we can
// return the id of the existing constant.
if ((constant->length == length) && strncmp(constant->start, start, length) == 0) {
return pool->constants[index].id;
}
index = (index + 1) % pool->capacity;
}
yp_constant_id_t id = (yp_constant_id_t)++pool->size;
pool->constants[index] = (yp_constant_t) {
.id = id,
.start = start,
.length = length,
.hash = hash
};
return id;
}
// Free the memory associated with a constant pool.
void
yp_constant_pool_free(yp_constant_pool_t *pool) {
free(pool->constants);
}