зеркало из https://github.com/github/ruby.git
148 строки
4.9 KiB
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);
|
|
}
|