#include "common.h"
#include "memory.h"
#include "value.h"
#include "vm.h"
#include "object.h"


#define ALLOCATE_OBJ(vm, type, objectType) \
    (type*)allocateObject(vm, sizeof(type), objectType)

static Obj* allocateObject(VM* vm, size_t size, ObjType type) {
    Obj* object = (Obj*)reallocate(NULL, 0, size);
    object->type = type;
    object->next = vm->objects;
    vm->objects = object;
    return object;
}

static ObjString* allocateString(VM* vm, char* chars, int length, const char* source, uint32_t hash) {
    ObjString* string = ALLOCATE_OBJ(vm, ObjString, OBJ_STRING);
    string->obj.type = OBJ_STRING;
    string->length = length;
    string->chars = chars;
    string->hash = hash;
    tableSet(&vm->strings, string, NIL_VAL);
    return string;
}

static uint32_t hashString(const char* key, int length) {
    // FNV-1a hash
    uint32_t hash = 2166136261u;
    for (int i = 0; i < length; i++) {
        hash ^= (uint8_t)key[i];
        hash *= 16777619;
    }
    return hash;
}

ObjString* takeString(VM* vm, char* chars, int length, uint32_t hash) {
    uint32_t hash = hashString(chars, length);
    return allocateString(vm, chars, length, NULL, hash);
}

ObjString* copyString(VM* vm, const char* chars, int length) {
    char* heapChars = ALLOCATE(char, length + 1);
    if (heapChars == NULL) return NULL;
    memcpy(heapChars, chars, length);
    heapChars[length] = '\0';
    uint32_t hash = hashString(chars, length);
    ObjString* interned = tableFindString(&vm->strings, chars, length, hash);
    if (interned != NULL) {
        FREE_ARRAY(char, heapChars, length + 1);
        return interned;
    }

    return allocateString(vm, heapChars, length, chars, hash);
}

void printObject(VM* vm, Value value) {
    switch (OBJ_TYPE(value)) {
        case OBJ_STRING:
            printf("%s", AS_CSTRING(value));
            break;
        case OBJ_FUNCTION:
            printFunction(vm, AS_FUNCTION(value));
            break;
        default:
            printf("<object>");
    }
}

ObjFunction* newFunction(VM* vm) {
    ObjFunction* function = ALLOCATE_OBJ(vm, ObjFunction, OBJ_FUNCTION);
    function->arity = 0;
    function->chunk = NULL;
    function->name = NULL;
    initChunk(function->chunk);
    return function;
}