2020-12-26 08:33:34 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "kuroko.h"
|
|
|
|
#include "object.h"
|
|
|
|
#include "value.h"
|
|
|
|
#include "memory.h"
|
|
|
|
#include "table.h"
|
2021-01-01 10:23:43 +03:00
|
|
|
#include "vm.h"
|
2020-12-26 08:33:34 +03:00
|
|
|
|
|
|
|
#define TABLE_MAX_LOAD 0.75
|
|
|
|
|
|
|
|
void krk_initTable(KrkTable * table) {
|
|
|
|
table->count = 0;
|
|
|
|
table->capacity = 0;
|
|
|
|
table->entries = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void krk_freeTable(KrkTable * table) {
|
|
|
|
FREE_ARRAY(KrkTableEntry, table->entries, table->capacity);
|
|
|
|
krk_initTable(table);
|
|
|
|
}
|
|
|
|
|
2021-01-12 16:30:06 +03:00
|
|
|
static uint32_t hashTupleValues(KrkTuple *tuple);
|
|
|
|
|
2020-12-26 08:33:34 +03:00
|
|
|
static uint32_t hashValue(KrkValue value) {
|
|
|
|
if (IS_STRING(value)) return (AS_STRING(value))->hash;
|
|
|
|
if (IS_INTEGER(value)) return (uint32_t)(AS_INTEGER(value));
|
|
|
|
if (IS_FLOATING(value)) return (uint32_t)(AS_FLOATING(value) * 1000); /* arbitrary; what's a good way to hash floats? */
|
|
|
|
if (IS_BOOLEAN(value)) return (uint32_t)(AS_BOOLEAN(value));
|
|
|
|
if (IS_NONE(value)) return 0;
|
2021-01-12 13:23:14 +03:00
|
|
|
if (IS_BYTES(value)) return (AS_BYTES(value))->hash; /* Same as strings, but we don't have an interning table */
|
2021-01-12 16:30:06 +03:00
|
|
|
if (IS_TUPLE(value)) return hashTupleValues(AS_TUPLE(value));
|
2021-01-02 06:21:11 +03:00
|
|
|
return (((uint32_t)(intptr_t)AS_OBJECT(value)) >> 4)| (((uint32_t)(intptr_t)AS_OBJECT(value)) << 28);
|
2020-12-26 08:33:34 +03:00
|
|
|
}
|
|
|
|
|
2021-01-12 16:30:06 +03:00
|
|
|
static uint32_t hashTupleValues(KrkTuple *tuple) {
|
|
|
|
uint32_t hash = 0;
|
|
|
|
for (size_t i = 0; i < tuple->values.count; ++i) {
|
|
|
|
hash += hashValue(tuple->values.values[i]);
|
|
|
|
}
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
2021-01-05 05:39:20 +03:00
|
|
|
KrkTableEntry * krk_findEntry(KrkTableEntry * entries, size_t capacity, KrkValue key) {
|
2020-12-26 08:33:34 +03:00
|
|
|
uint32_t index = hashValue(key) % capacity;
|
|
|
|
KrkTableEntry * tombstone = NULL;
|
|
|
|
for (;;) {
|
|
|
|
KrkTableEntry * entry = &entries[index];
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type == VAL_KWARGS) {
|
2020-12-26 08:33:34 +03:00
|
|
|
if (IS_NONE(entry->value)) {
|
|
|
|
return tombstone != NULL ? tombstone : entry;
|
|
|
|
} else {
|
|
|
|
if (tombstone == NULL) tombstone = entry;
|
|
|
|
}
|
|
|
|
} else if (krk_valuesEqual(entry->key, key)) {
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
index = (index + 1) % capacity;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void adjustCapacity(KrkTable * table, size_t capacity) {
|
|
|
|
KrkTableEntry * entries = ALLOCATE(KrkTableEntry, capacity);
|
|
|
|
for (size_t i = 0; i < capacity; ++i) {
|
2021-01-10 12:23:28 +03:00
|
|
|
entries[i].key = KWARGS_VAL(0);
|
2020-12-26 08:33:34 +03:00
|
|
|
entries[i].value = NONE_VAL();
|
|
|
|
}
|
|
|
|
|
|
|
|
table->count = 0;
|
|
|
|
for (size_t i = 0; i < table->capacity; ++i) {
|
|
|
|
KrkTableEntry * entry = &table->entries[i];
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type == VAL_KWARGS) continue;
|
2021-01-05 05:39:20 +03:00
|
|
|
KrkTableEntry * dest = krk_findEntry(entries, capacity, entry->key);
|
2020-12-26 08:33:34 +03:00
|
|
|
dest->key = entry->key;
|
|
|
|
dest->value = entry->value;
|
|
|
|
table->count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
FREE_ARRAY(KrkTableEntry, table->entries, table->capacity);
|
|
|
|
table->entries = entries;
|
|
|
|
table->capacity = capacity;
|
|
|
|
}
|
|
|
|
|
|
|
|
int krk_tableSet(KrkTable * table, KrkValue key, KrkValue value) {
|
|
|
|
if (table->count + 1 > table->capacity * TABLE_MAX_LOAD) {
|
|
|
|
size_t capacity = GROW_CAPACITY(table->capacity);
|
|
|
|
adjustCapacity(table, capacity);
|
|
|
|
}
|
2021-01-05 05:39:20 +03:00
|
|
|
KrkTableEntry * entry = krk_findEntry(table->entries, table->capacity, key);
|
2021-01-10 12:23:28 +03:00
|
|
|
int isNewKey = entry->key.type == VAL_KWARGS;
|
2020-12-26 08:33:34 +03:00
|
|
|
if (isNewKey && IS_NONE(entry->value)) table->count++;
|
|
|
|
entry->key = key;
|
|
|
|
entry->value = value;
|
|
|
|
return isNewKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
void krk_tableAddAll(KrkTable * from, KrkTable * to) {
|
|
|
|
for (size_t i = 0; i < from->capacity; ++i) {
|
|
|
|
KrkTableEntry * entry = &from->entries[i];
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type != VAL_KWARGS) {
|
2020-12-26 08:33:34 +03:00
|
|
|
krk_tableSet(to, entry->key, entry->value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int krk_tableGet(KrkTable * table, KrkValue key, KrkValue * value) {
|
|
|
|
if (table->count == 0) return 0;
|
2021-01-05 05:39:20 +03:00
|
|
|
KrkTableEntry * entry = krk_findEntry(table->entries, table->capacity, key);
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type == VAL_KWARGS) return 0;
|
2020-12-26 08:33:34 +03:00
|
|
|
*value = entry->value;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int krk_tableDelete(KrkTable * table, KrkValue key) {
|
|
|
|
if (table->count == 0) return 0;
|
2021-01-05 05:39:20 +03:00
|
|
|
KrkTableEntry * entry = krk_findEntry(table->entries, table->capacity, key);
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type == VAL_KWARGS) return 0;
|
|
|
|
entry->key = KWARGS_VAL(0);
|
2020-12-26 08:33:34 +03:00
|
|
|
entry->value = BOOLEAN_VAL(1);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-12-28 05:11:50 +03:00
|
|
|
KrkString * krk_tableFindString(KrkTable * table, const char * chars, size_t length, uint32_t hash) {
|
2020-12-26 08:33:34 +03:00
|
|
|
if (table->count == 0) return NULL;
|
|
|
|
|
|
|
|
uint32_t index = hash % table->capacity;
|
|
|
|
for (;;) {
|
|
|
|
KrkTableEntry * entry = &table->entries[index];
|
2021-01-10 12:23:28 +03:00
|
|
|
if (entry->key.type == VAL_KWARGS) {
|
2020-12-26 08:33:34 +03:00
|
|
|
if (IS_NONE(entry->value)) return NULL;
|
|
|
|
} else if (AS_STRING(entry->key)->length == length &&
|
|
|
|
AS_STRING(entry->key)->hash == hash &&
|
|
|
|
memcmp(AS_STRING(entry->key)->chars, chars, length) == 0) {
|
|
|
|
return AS_STRING(entry->key);
|
|
|
|
}
|
|
|
|
index = (index + 1) % table->capacity;
|
|
|
|
}
|
|
|
|
}
|