ona/source/kym/environment.cpp

183 lines
4.2 KiB
C++
Raw Normal View History

2023-02-18 04:34:40 +01:00
export module kym.environment;
2023-02-19 17:50:29 +01:00
import coral;
import coral.sequence;
2023-02-18 04:34:40 +01:00
import kym;
2023-02-19 17:50:29 +01:00
using loggable = coral::callable<void(coral::slice<char const> const &)>;
2023-02-18 20:40:12 +01:00
2023-02-18 04:34:40 +01:00
export namespace kym {
struct vm;
}
enum class token_kind {
end,
};
struct token {
2023-02-19 17:50:29 +01:00
coral::slice<char const> text;
2023-02-18 04:34:40 +01:00
token_kind kind;
};
struct tokenizer {
2023-02-19 17:50:29 +01:00
coral::slice<char const> source;
2023-02-19 17:50:29 +01:00
tokenizer(coral::slice<char const> const & source) : source{source} {}
token next() {
2023-02-19 17:50:29 +01:00
coral::usize cursor = 0;
while (cursor < source.length) {
}
return token{
.kind = token_kind::end,
};
}
};
2023-02-18 04:34:40 +01:00
struct bytecode {
2023-02-19 17:50:29 +01:00
bytecode(coral::allocator * allocator) : error_message_buffer{allocator} {
2023-02-18 04:34:40 +01:00
}
2023-02-18 20:40:12 +01:00
bool compile(tokenizer bytecode_tokenizer, loggable const & log_error) {
2023-02-18 04:34:40 +01:00
for (;;) {
token const initial_token = bytecode_tokenizer.next();
2023-02-18 04:34:40 +01:00
switch (initial_token.kind) {
case token_kind::end: return true;
2023-02-19 17:50:29 +01:00
default: coral::unreachable();
2023-02-18 04:34:40 +01:00
}
}
}
2023-02-19 17:50:29 +01:00
kym::value execute(kym::vm & vm, coral::slice<kym::value const> const & arguments) {
2023-02-18 04:34:40 +01:00
return kym::nil;
}
private:
2023-02-19 17:50:29 +01:00
coral::stack<char> error_message_buffer;
2023-02-18 04:34:40 +01:00
};
export namespace kym {
2023-02-19 17:50:29 +01:00
value default_call(vm & owning_vm, void * userdata, coral::slice<value const> const & arguments) {
return nil;
}
coral::expected<coral::usize, coral::io_error> default_stringify(vm & owning_vm, void * userdata, coral::writer & output) {
return output.write(coral::slice{"[object]"}.as_bytes());
}
2023-02-18 04:34:40 +01:00
struct bound_object {
void * userdata;
2023-02-18 04:34:40 +01:00
struct {
void(*cleanup)(vm &, void *);
2023-02-18 04:34:40 +01:00
2023-02-19 17:50:29 +01:00
value(*call)(vm &, void *, coral::slice<value const> const &);
coral::expected<coral::usize, coral::io_error>(*stringify)(vm &, void *, coral::writer &);
} behavior;
bound_object(vm * owning_vm) : userdata{nullptr}, owning_vm{owning_vm}, behavior{
.cleanup = [](vm & owning_vm, void * userdata) {},
.call = default_call,
.stringify = default_stringify,
} {}
void cleanup() {
this->behavior.cleanup(*this->owning_vm, this->userdata);
2023-02-18 04:34:40 +01:00
}
2023-02-19 17:50:29 +01:00
value call(coral::slice<value const> const & arguments) {
return this->behavior.call(*this->owning_vm, this->userdata, arguments);
2023-02-18 04:34:40 +01:00
}
coral::expected<coral::usize, coral::io_error> stringify(coral::writer & output) {
2023-02-18 20:40:12 +01:00
return this->behavior.stringify(*this->owning_vm, this->userdata, output);
}
2023-02-19 17:50:29 +01:00
value get_field(coral::slice<char const> const & field_name) {
return nil;
2023-02-18 04:34:40 +01:00
}
private:
vm * owning_vm;
2023-02-18 04:34:40 +01:00
};
struct vm {
struct init_options {
2023-02-19 17:50:29 +01:00
coral::u16 datastack_size;
2023-02-18 04:34:40 +01:00
2023-02-19 17:50:29 +01:00
coral::u16 callstack_size;
2023-02-18 04:34:40 +01:00
};
2023-02-19 17:50:29 +01:00
vm(coral::allocator * allocator, auto log) : allocator{allocator}, log{log}, data_stack{} {}
2023-02-18 04:34:40 +01:00
~vm() {
if (this->data_stack.pointer != nullptr)
this->allocator->deallocate(this->data_stack.pointer);
}
bool init(init_options const & options) {
2023-02-19 17:50:29 +01:00
coral::u8 * const data_stack_buffer = this->allocator->reallocate(reinterpret_cast
<coral::u8 *>(this->data_stack.pointer), options.datastack_size * sizeof(value));
2023-02-18 04:34:40 +01:00
if (data_stack_buffer == nullptr) return false;
this->data_stack = {
reinterpret_cast<value * >(data_stack_buffer), options.datastack_size};
return true;
}
2023-02-19 17:50:29 +01:00
value compile(coral::slice<char const> const & source) {
2023-02-18 04:34:40 +01:00
bytecode * source_bytecode = new (*this->allocator) bytecode{allocator};
if (source_bytecode == nullptr) return nil;
2023-02-19 17:50:29 +01:00
if (!source_bytecode->compile(tokenizer{source}, [&](coral::slice<char const> error_message) {
2023-02-18 20:40:12 +01:00
this->log(error_message);
})) {
2023-02-18 04:34:40 +01:00
this->allocator->deallocate(source_bytecode);
return nil;
}
return this->new_object([this, source_bytecode](bound_object & object) {
object.userdata = source_bytecode;
object.behavior.cleanup = [](vm & owning_vm, void * userdata) {
owning_vm.allocator->deallocate(userdata);
2023-02-18 04:34:40 +01:00
};
2023-02-19 17:50:29 +01:00
object.behavior.call = [](vm & owning_vm, void * userdata, coral::slice<value const> const & arguments) -> value {
return reinterpret_cast<bytecode *>(userdata)->execute(owning_vm, arguments);
2023-02-18 04:34:40 +01:00
};
this->allocator->deallocate(source_bytecode);
});
}
2023-02-19 17:50:29 +01:00
value new_object(coral::callable<void(bound_object &)> const & then) {
2023-02-18 04:34:40 +01:00
return nil;
}
2023-02-19 17:50:29 +01:00
void with_object(value object_value, coral::callable<void(bound_object &)> const & then) {
2023-02-18 04:34:40 +01:00
}
private:
2023-02-19 17:50:29 +01:00
coral::slice<value> data_stack;
2023-02-18 04:34:40 +01:00
2023-02-18 20:40:12 +01:00
loggable log;
2023-02-18 04:34:40 +01:00
2023-02-19 17:50:29 +01:00
coral::allocator * allocator;
2023-02-18 04:34:40 +01:00
};
}