20 if (list->
ids == NULL) abort();
38 if (list->
ids == NULL)
return false;
50 assert(index < list->capacity);
53 list->
ids[index] = id;
62 for (
size_t index = 0; index < list->
size; index++) {
63 if (list->
ids[index] ==
id)
return true;
73 if (list->
ids != NULL) {
83pm_constant_pool_hash(
const uint8_t *start,
size_t length) {
85 uint32_t value = 5381;
87 for (
size_t index = 0; index < length; index++) {
88 value = ((value << 5) + value) + start[index];
98next_power_of_two(uint32_t v) {
116is_power_of_two(uint32_t size) {
117 return (size & (size - 1)) == 0;
126 assert(is_power_of_two(pool->
capacity));
128 uint32_t next_capacity = pool->
capacity * 2;
129 if (next_capacity < pool->capacity)
return false;
131 const uint32_t mask = next_capacity - 1;
134 void *next =
xcalloc(next_capacity, element_size);
135 if (next == NULL)
return false;
142 for (uint32_t index = 0; index < pool->
capacity; index++) {
148 uint32_t next_index = bucket->
hash & mask;
154 next_index = (next_index + 1) & mask;
159 next_buckets[next_index] = *bucket;
180 const uint32_t maximum = (~((uint32_t) 0));
181 if (capacity >= ((maximum / 2) + 1))
return false;
183 capacity = next_power_of_two(capacity);
185 void *memory =
xcalloc(capacity, element_size);
186 if (memory == NULL)
return false;
201 return &pool->
constants[constant_id - 1];
210 assert(is_power_of_two(pool->
capacity));
211 const uint32_t mask = pool->
capacity - 1;
213 uint32_t hash = pm_constant_pool_hash(start, length);
214 uint32_t index = hash & mask;
219 if ((constant->
length == length) && memcmp(constant->
start, start, length) == 0) {
223 index = (index + 1) & mask;
238 assert(is_power_of_two(pool->
capacity));
239 const uint32_t mask = pool->
capacity - 1;
241 uint32_t hash = pm_constant_pool_hash(start, length);
242 uint32_t index = hash & mask;
251 if ((constant->
length == length) && memcmp(constant->
start, start, length) == 0) {
260 xfree((
void *) start);
266 constant->
start = start;
273 index = (index + 1) & mask;
278 uint32_t
id = ++pool->
size;
279 assert(pool->
size < ((uint32_t) (1 << 30)));
282 .
id = (
unsigned int) (
id & 0x3fffffff),
283 .type = (
unsigned int) (
type & 0x3),
331 for (uint32_t index = 0; index < pool->
capacity; index++) {
#define xfree
Old name of ruby_xfree.
#define xrealloc
Old name of ruby_xrealloc.
#define xcalloc
Old name of ruby_xcalloc.
VALUE type(ANYARGS)
ANYARGS-ed function type.
A data structure that stores a set of strings.
static const pm_constant_pool_bucket_type_t PM_CONSTANT_POOL_BUCKET_DEFAULT
By default, each constant is a slice of the source.
#define PM_CONSTANT_ID_UNSET
When we allocate constants into the pool, we reserve 0 to mean that the slot is not yet filled.
unsigned int pm_constant_pool_bucket_type_t
The type of bucket in the constant pool hash map.
uint32_t pm_constant_id_t
A constant id is a unique identifier for a constant in the constant pool.
static const pm_constant_pool_bucket_type_t PM_CONSTANT_POOL_BUCKET_OWNED
An owned constant is one for which memory has been allocated.
static const pm_constant_pool_bucket_type_t PM_CONSTANT_POOL_BUCKET_CONSTANT
A constant constant is known at compile time.
size_t size
The number of constant ids in the list.
size_t capacity
The number of constant ids that have been allocated in the list.
pm_constant_id_t * ids
The constant ids in the list.
A bucket in the hash map.
uint32_t hash
The hash of the bucket.
unsigned int id
The incremental ID used for indexing back into the pool.
pm_constant_pool_bucket_type_t type
The type of the bucket, which determines how to free it.
The overall constant pool, which stores constants found while parsing.
uint32_t capacity
The number of buckets that have been allocated in the hash map.
pm_constant_pool_bucket_t * buckets
The buckets in the hash map.
uint32_t size
The number of buckets in the hash map.
pm_constant_t * constants
The constants that are stored in the buckets.
A constant in the pool which effectively stores a string.
size_t length
The length of the string.
const uint8_t * start
A pointer to the start of the string.