Compare commits

..

No commits in common. "ef537bef1419269b084a06cf8694f6bd93266b39" and "decc2799b09161f5fc286f46a9c405e50cd89f0e" have entirely different histories.

11 changed files with 1120 additions and 2281 deletions

View File

@ -57,7 +57,7 @@ pub const Stacking = struct {
}); });
} }
pub fn deinit(self: *Stacking) void { pub fn free(self: *Stacking) void {
while (self.head_region) |region| { while (self.head_region) |region| {
const next_region = region.next; const next_region = region.next;
@ -75,15 +75,6 @@ pub const Stacking = struct {
unreachable; unreachable;
} }
pub fn init(allocator: io.Allocator, min_region_size: usize) Stacking {
return Stacking{
.allocator = allocator,
.min_region_size = min_region_size,
.head_region = null,
.tail_region = null,
};
}
fn reallocate(self: *Stacking, _: usize, allocation: ?[]io.Byte, byte_size: usize) io.AllocationError![]io.Byte { fn reallocate(self: *Stacking, _: usize, allocation: ?[]io.Byte, byte_size: usize) io.AllocationError![]io.Byte {
if (allocation) |buffer| { if (allocation) |buffer| {
if (byte_size < buffer.len) { if (byte_size < buffer.len) {
@ -113,4 +104,13 @@ pub const Stacking = struct {
return @as([*]io.Byte, @ptrCast(tail_region.allocate(region_size)))[0 .. byte_size]; return @as([*]io.Byte, @ptrCast(tail_region.allocate(region_size)))[0 .. byte_size];
} }
pub fn make(allocator: io.Allocator, min_region_size: usize) Stacking {
return Stacking{
.allocator = allocator,
.min_region_size = min_region_size,
.head_region = null,
.tail_region = null,
};
}
}; };

View File

@ -16,16 +16,6 @@ pub fn Stack(comptime Value: type) type {
self.values = self.values[0 .. 0]; self.values = self.values[0 .. 0];
} }
pub fn deinit(self: *Self) void {
if (self.capacity == 0) {
return;
}
self.allocator.deallocate(self.values.ptr[0 .. self.capacity]);
self.values = &.{};
}
pub fn drop(self: *Self, count: usize) bool { pub fn drop(self: *Self, count: usize) bool {
if (math.checked_sub(self.values.len, count)) |updated_count| { if (math.checked_sub(self.values.len, count)) |updated_count| {
self.values = self.values[0 .. updated_count]; self.values = self.values[0 .. updated_count];
@ -36,6 +26,16 @@ pub fn Stack(comptime Value: type) type {
return false; return false;
} }
pub fn free(self: *Self) void {
if (self.capacity == 0) {
return;
}
self.allocator.deallocate(self.values.ptr[0 .. self.capacity]);
self.values = &.{};
}
pub fn grow(self: *Self, growth_amount: usize) io.AllocationError!void { pub fn grow(self: *Self, growth_amount: usize) io.AllocationError!void {
const grown_capacity = self.capacity + growth_amount; const grown_capacity = self.capacity + growth_amount;
const buffer = try self.allocator.reallocate(null, @sizeOf(Value) * grown_capacity); const buffer = try self.allocator.reallocate(null, @sizeOf(Value) * grown_capacity);
@ -51,7 +51,11 @@ pub fn Stack(comptime Value: type) type {
self.capacity = grown_capacity; self.capacity = grown_capacity;
} }
pub fn init(allocator: io.Allocator) Self { pub fn is_empty(self: Self) bool {
return self.values.len == 0;
}
pub fn make(allocator: io.Allocator) Self {
return .{ return .{
.allocator = allocator, .allocator = allocator,
.capacity = 0, .capacity = 0,
@ -59,11 +63,7 @@ pub fn Stack(comptime Value: type) type {
}; };
} }
pub fn is_empty(self: Self) bool { pub fn pack(self: *Self) io.AllocationError!void {
return self.values.len == 0;
}
pub fn pack(self: *Self) io.AllocationError![]Value {
const packed_size = self.values.len; const packed_size = self.values.len;
const buffer = try self.allocator.reallocate(null, @sizeOf(Value) * self.values.len); const buffer = try self.allocator.reallocate(null, @sizeOf(Value) * self.values.len);
@ -75,8 +75,6 @@ pub fn Stack(comptime Value: type) type {
self.values = @as([*]Value, @ptrCast(@alignCast(buffer)))[0 .. packed_size]; self.values = @as([*]Value, @ptrCast(@alignCast(buffer)))[0 .. packed_size];
self.capacity = packed_size; self.capacity = packed_size;
return self.values;
} }
pub fn peek(self: Self) ?Value { pub fn peek(self: Self) ?Value {

View File

@ -174,7 +174,7 @@ pub fn Table(comptime Key: type, comptime Value: type, comptime Traits: type) ty
self.count = 0; self.count = 0;
} }
pub fn deinit(self: *Self) void { pub fn free(self: *Self) void {
if (self.entries.len == 0) { if (self.entries.len == 0) {
return; return;
} }
@ -185,15 +185,6 @@ pub fn Table(comptime Key: type, comptime Value: type, comptime Traits: type) ty
self.count = 0; self.count = 0;
} }
pub fn init(allocator: io.Allocator, traits: Traits) Self {
return .{
.allocator = allocator,
.count = 0,
.entries = &.{},
.traits = traits,
};
}
pub fn insert(self: *Self, key: Key, value: Value) io.AllocationError!bool { pub fn insert(self: *Self, key: Key, value: Value) io.AllocationError!bool {
try self.rehash(load_max); try self.rehash(load_max);
@ -244,14 +235,23 @@ pub fn Table(comptime Key: type, comptime Value: type, comptime Traits: type) ty
return null; return null;
} }
pub fn make(allocator: io.Allocator, traits: Traits) Self {
return .{
.allocator = allocator,
.count = 0,
.entries = &.{},
.traits = traits,
};
}
pub fn rehash(self: *Self, max_load: f32) io.AllocationError!void { pub fn rehash(self: *Self, max_load: f32) io.AllocationError!void {
if (self.calculate_load_factor() <= max_load) { if (self.calculate_load_factor() <= max_load) {
return; return;
} }
var table = init(self.allocator, self.traits); var table = make(self.allocator, self.traits);
errdefer table.deinit(); errdefer table.free();
table.entries = allocate: { table.entries = allocate: {
const min_count = @max(1, self.count); const min_count = @max(1, self.count);
@ -266,7 +266,7 @@ pub fn Table(comptime Key: type, comptime Value: type, comptime Traits: type) ty
} }
} }
self.deinit(); self.free();
self.* = table; self.* = table;
} }

View File

@ -13,15 +13,15 @@ pub const Manifest = struct {
tick_rate: f32 = 60.0, tick_rate: f32 = 60.0,
pub fn load(self: *Manifest, env: *kym.RuntimeEnv) kym.RuntimeError!void { pub fn load(self: *Manifest, env: *kym.RuntimeEnv) kym.RuntimeError!void {
const manifest = (try env.import(file.Path.from(&.{"app.ona"}))).pop() orelse return; const manifest = try env.import(file.Path.from(&.{"app.ona"})) orelse return;
defer env.release(manifest); defer env.discard(manifest);
const width = @as(u16, get: { const width = @as(u16, get: {
if (try kym.get_field(env, manifest, "width")) |ref| { if (try kym.get_field(env, manifest, "width")) |ref| {
defer env.release(ref); defer env.discard(ref);
const fixed = try env.expect_fixed(ref); const fixed = try env.unwrap_fixed(ref);
if (fixed > 0 and fixed < coral.math.max_int(@typeInfo(@TypeOf(self.width)).Int)) { if (fixed > 0 and fixed < coral.math.max_int(@typeInfo(@TypeOf(self.width)).Int)) {
break: get @intCast(fixed); break: get @intCast(fixed);
@ -33,9 +33,9 @@ pub const Manifest = struct {
const height = @as(u16, get: { const height = @as(u16, get: {
if (try kym.get_field(env, manifest, "height")) |ref| { if (try kym.get_field(env, manifest, "height")) |ref| {
defer env.release(ref); defer env.discard(ref);
const fixed = try env.expect_fixed(ref); const fixed = try env.unwrap_fixed(ref);
if (fixed > 0 and fixed < coral.math.max_int(@typeInfo(@TypeOf(self.height)).Int)) { if (fixed > 0 and fixed < coral.math.max_int(@typeInfo(@TypeOf(self.height)).Int)) {
break: get @intCast(fixed); break: get @intCast(fixed);
@ -47,18 +47,18 @@ pub const Manifest = struct {
const tick_rate = @as(f32, get: { const tick_rate = @as(f32, get: {
if (try kym.get_field(env, manifest, "tick_rate")) |ref| { if (try kym.get_field(env, manifest, "tick_rate")) |ref| {
defer env.release(ref); defer env.discard(ref);
break: get @floatCast(try env.expect_float(ref)); break: get @floatCast(try env.unwrap_float(ref));
} }
break: get self.tick_rate; break: get self.tick_rate;
}); });
if (try kym.get_field(env, manifest, "title")) |ref| { if (try kym.get_field(env, manifest, "title")) |ref| {
defer env.release(ref); defer env.discard(ref);
const title_string = try env.expect_string(ref); const title_string = try env.unwrap_string(ref);
const limited_title_len = @min(title_string.len, self.title.len); const limited_title_len = @min(title_string.len, self.title.len);
coral.io.copy(&self.title, title_string[0 .. limited_title_len]); coral.io.copy(&self.title, title_string[0 .. limited_title_len]);

View File

@ -1,2 +0,0 @@
pub const lina = @import("./gfx/lina.zig");

View File

@ -1,181 +0,0 @@
pub const Vector2 = struct {
x: f32,
y: f32,
pub const Scalars = [2]f32;
pub fn equals(self: Vector2, vector: Vector2) bool {
return self.x == vector.x and self.y == vector.y;
}
pub fn from_scalar(scalar: f32) Vector2 {
return .{
.x = scalar,
.y = scalar,
};
}
pub fn from_scalars(scalars: Scalars) Vector2 {
return .{
.x = scalars[0],
.y = scalars[1],
};
}
pub fn scalar_added(self: Vector2, scalar: f32) Vector2 {
return .{
.x = self.x + scalar,
.y = self.y + scalar,
};
}
pub fn scalar_divided(self: Vector2, scalar: f32) Vector2 {
return .{
.x = self.x / scalar,
.y = self.y / scalar,
};
}
pub fn scalar_multiplied(self: Vector2, scalar: f32) Vector2 {
return .{
.x = self.x * scalar,
.y = self.y * scalar,
};
}
pub fn to_scalars(self: Vector2) Scalars {
return .{self.x, self.y};
}
pub fn scalar_subtracted(self: Vector2, scalar: f32) Vector2 {
return .{
.x = self.x - scalar,
.y = self.y - scalar,
};
}
pub fn vector_added(self: Vector2, vector: Vector2) Vector2 {
return .{
.x = self.x + vector.x,
.y = self.y + vector.y,
};
}
pub fn vector_divided(self: Vector2, vector: Vector2) Vector2 {
return .{
.x = self.x / vector.x,
.y = self.y / vector.y,
};
}
pub fn vector_multiplied(self: Vector2, vector: Vector2) Vector2 {
return .{
.x = self.x * vector.x,
.y = self.y * vector.y,
};
}
pub fn vector_subtracted(self: Vector2, vector: Vector2) Vector2 {
return .{
.x = self.x - vector.x,
.y = self.y - vector.y,
};
}
};
pub const Vector3 = struct {
x: f32,
y: f32,
z: f32,
pub const Scalars = [3]f32;
pub fn equals(self: Vector3, vector: Vector3) bool {
return self.x == vector.x and self.y == vector.y and self.z == vector.z;
}
pub fn from_scalar(scalar: f32) Vector3 {
return .{
.x = scalar,
.y = scalar,
.z = scalar,
};
}
pub fn from_scalars(scalars: Scalars) Vector3 {
return .{
.x = scalars[0],
.y = scalars[1],
.z = scalars[2],
};
}
pub fn scalar_added(self: Vector3, scalar: f32) Vector3 {
return .{
.x = self.x + scalar,
.y = self.y + scalar,
.z = self.z + scalar,
};
}
pub fn scalar_divided(self: Vector3, scalar: f32) Vector3 {
return .{
.x = self.x / scalar,
.y = self.y / scalar,
.z = self.z / scalar,
};
}
pub fn scalar_multiplied(self: Vector3, scalar: f32) Vector3 {
return .{
.x = self.x * scalar,
.y = self.y * scalar,
.z = self.z * scalar,
};
}
pub fn scalar_subtracted(self: Vector3, scalar: f32) Vector3 {
return .{
.x = self.x - scalar,
.y = self.y - scalar,
.z = self.z - scalar,
};
}
pub fn to_scalars(self: Vector3) Scalars {
return .{self.x, self.y, self.z};
}
pub fn vector_added(self: Vector3, other: Vector3) Vector3 {
return .{
.x = self.x + other.x,
.y = self.y + other.y,
.z = self.z + other.z,
};
}
pub fn vector_divided(self: Vector3, other: Vector3) Vector3 {
return .{
.x = self.x / other.x,
.y = self.y / other.y,
.z = self.z / other.z,
};
}
pub fn vector_multiplied(self: Vector3, other: Vector3) Vector3 {
return .{
.x = self.x * other.x,
.y = self.y * other.y,
.z = self.z * other.z,
};
}
pub fn vector_subtracted(self: Vector3, other: Vector3) Vector3 {
return .{
.x = self.x - other.x,
.y = self.y - other.y,
.z = self.z - other.z,
};
}
};

File diff suppressed because it is too large Load Diff

View File

@ -10,13 +10,13 @@ const tokens = @import("./tokens.zig");
const tree = @import("./tree.zig"); const tree = @import("./tree.zig");
name: *kym.RuntimeObj, name: *kym.RuntimeRef,
arity: u8, arity: u8,
opcodes: OpcodeList, opcodes: OpcodeList,
lines: LineList, lines: LineList,
cursor: usize, cursor: usize,
constants: ConstList, constants: ConstList,
bindings: []?*kym.RuntimeObj, bindings: []?*kym.RuntimeRef,
const Builtin = enum { const Builtin = enum {
import, import,
@ -118,9 +118,9 @@ const Compiler = struct {
}, },
.lambda_construct => |lambda_construct| { .lambda_construct => |lambda_construct| {
var chunk = try Self.init(self.env, name orelse "<lambda>", lambda_construct.environment); var chunk = try Self.make(self.env, name orelse "<lambda>", lambda_construct.environment);
errdefer chunk.deinit(self.env); errdefer chunk.free(self.env);
if (lambda_construct.environment.capture_count == 0) { if (lambda_construct.environment.capture_count == 0) {
try self.chunk.write(expression.line, .{.push_const = try self.declare_chunk(chunk)}); try self.chunk.write(expression.line, .{.push_const = try self.declare_chunk(chunk)});
@ -193,7 +193,7 @@ const Compiler = struct {
return; return;
} }
if (try get_binding_index(environment, declaration_get.declaration)) |index| { if (try self.get_binding_index(environment, declaration_get.declaration)) |index| {
try self.chunk.write(expression.line, .{.push_binding = index}); try self.chunk.write(expression.line, .{.push_binding = index});
if (is_declaration_boxed(declaration_get.declaration)) { if (is_declaration_boxed(declaration_get.declaration)) {
@ -208,19 +208,19 @@ const Compiler = struct {
.declaration_set => |declaration_set| { .declaration_set => |declaration_set| {
if (get_local_index(environment, declaration_set.declaration)) |index| { if (get_local_index(environment, declaration_set.declaration)) |index| {
try self.compile_expression(environment, declaration_set.assign, null);
if (is_declaration_boxed(declaration_set.declaration)) { if (is_declaration_boxed(declaration_set.declaration)) {
try self.chunk.write(expression.line, .{.push_local = index}); try self.chunk.write(expression.line, .{.push_local = index});
try self.compile_expression(environment, declaration_set.assign, null);
try self.chunk.write(expression.line, .set_box); try self.chunk.write(expression.line, .set_box);
} else { } else {
try self.compile_expression(environment, declaration_set.assign, null);
try self.chunk.write(expression.line, .{.set_local = index}); try self.chunk.write(expression.line, .{.set_local = index});
} }
return; return;
} }
if (try get_binding_index(environment, declaration_set.declaration)) |index| { if (try self.get_binding_index(environment, declaration_set.declaration)) |index| {
try self.compile_expression(environment, declaration_set.assign, null); try self.compile_expression(environment, declaration_set.assign, null);
try self.chunk.write(expression.line, .{.push_binding = index}); try self.chunk.write(expression.line, .{.push_binding = index});
@ -345,9 +345,9 @@ const Compiler = struct {
}); });
} }
const constant = (try self.env.new_dynamic(coral.io.bytes_of(&chunk), typeinfo)).pop().?; const constant = try self.env.new_dynamic(coral.io.bytes_of(&chunk), typeinfo);
errdefer self.env.release(constant); errdefer self.env.discard(constant);
try self.chunk.constants.push_one(constant); try self.chunk.constants.push_one(constant);
@ -361,9 +361,9 @@ const Compiler = struct {
}); });
} }
const constant = (try self.env.new_fixed(fixed)).pop().?; const constant = try self.env.new_fixed(fixed);
errdefer self.env.release(constant); errdefer self.env.discard(constant);
try self.chunk.constants.push_one(constant); try self.chunk.constants.push_one(constant);
@ -377,9 +377,9 @@ const Compiler = struct {
}); });
} }
const constant = (try self.env.new_float(float)).pop().?; const constant = try self.env.new_float(float);
errdefer self.env.release(constant); errdefer self.env.discard(constant);
try self.chunk.constants.push_one(constant); try self.chunk.constants.push_one(constant);
@ -393,9 +393,9 @@ const Compiler = struct {
}); });
} }
const constant = (try self.env.new_string(string)).pop().?; const constant = try self.env.new_string(string);
errdefer self.env.release(constant); errdefer self.env.discard(constant);
try self.chunk.constants.push_one(constant); try self.chunk.constants.push_one(constant);
@ -409,16 +409,16 @@ const Compiler = struct {
}); });
} }
const constant = (try self.env.new_symbol(symbol)).pop().?; const constant = try self.env.new_symbol(symbol);
errdefer self.env.release(constant); errdefer self.env.discard(constant);
try self.chunk.constants.push_one(constant); try self.chunk.constants.push_one(constant);
return @intCast(self.chunk.constants.values.len - 1); return @intCast(self.chunk.constants.values.len - 1);
} }
fn get_binding_index(environment: *const tree.Environment, declaration: *const tree.Declaration) kym.RuntimeError!?u8 { fn get_binding_index(self: *const Compiler, environment: *const tree.Environment, declaration: *const tree.Declaration) kym.RuntimeError!?u8 {
var binding_index = @as(u8, 0); var binding_index = @as(u8, 0);
while (binding_index < environment.capture_count) : (binding_index += 1) { while (binding_index < environment.capture_count) : (binding_index += 1) {
@ -430,7 +430,7 @@ const Compiler = struct {
target_environment = target_environment.enclosing orelse return null; target_environment = target_environment.enclosing orelse return null;
} }
coral.debug.assert(capture.* == .declaration_index); try kym.assert(self.env, capture.* == .declaration_index);
if (&target_environment.declarations[capture.declaration_index] == declaration) { if (&target_environment.declarations[capture.declaration_index] == declaration) {
return binding_index; return binding_index;
@ -459,7 +459,7 @@ const Compiler = struct {
} }
}; };
const ConstList = coral.list.Stack(*kym.RuntimeObj); const ConstList = coral.list.Stack(*kym.RuntimeRef);
const LineList = coral.list.Stack(tokens.Line); const LineList = coral.list.Stack(tokens.Line);
@ -507,33 +507,11 @@ const OpcodeList = coral.list.Stack(Opcode);
const Self = @This(); const Self = @This();
pub fn deinit(self: *Self, env: *kym.RuntimeEnv) void { pub fn dump(chunk: Self, env: *kym.RuntimeEnv) kym.RuntimeError!*kym.RuntimeRef {
while (self.constants.pop()) |constant| {
env.release(constant);
}
self.constants.deinit();
self.opcodes.deinit();
env.release(self.name);
if (self.bindings.len != 0) {
for (self.bindings) |binding| {
if (binding) |value| {
env.release(value);
}
}
env.allocator.deallocate(self.bindings);
}
self.bindings = &.{};
}
pub fn dump(chunk: Self, env: *kym.RuntimeEnv) kym.RuntimeError!*kym.RuntimeObj {
var opcode_cursor = @as(u32, 0); var opcode_cursor = @as(u32, 0);
var buffer = coral.list.ByteStack.init(env.allocator); var buffer = coral.list.ByteStack.make(env.allocator);
defer buffer.deinit(); defer buffer.free();
const writer = coral.list.stack_as_writer(&buffer); const writer = coral.list.stack_as_writer(&buffer);
@ -550,11 +528,15 @@ pub fn dump(chunk: Self, env: *kym.RuntimeEnv) kym.RuntimeError!*kym.RuntimeObj
.push_false => coral.utf8.print_string(writer, "push false\n"), .push_false => coral.utf8.print_string(writer, "push false\n"),
.push_const => |push_const| print: { .push_const => |push_const| print: {
const string_ref = (try (try env.push(try chunk.get_constant(env, push_const))).to_string()).pop().?; try kym.assert(env, push_const < chunk.constants.values.len);
defer env.release(string_ref); const string_ref = try env.to_string(chunk.constants.values[push_const]);
const string = string_ref.is_string(); defer env.discard(string_ref);
const string = string_ref.as_string();
coral.debug.assert(string != null);
break: print coral.utf8.print_formatted(writer, "push const ({value})\n", .{.value = string.?}); break: print coral.utf8.print_formatted(writer, "push const ({value})\n", .{.value = string.?});
}, },
@ -617,247 +599,407 @@ pub fn dump(chunk: Self, env: *kym.RuntimeEnv) kym.RuntimeError!*kym.RuntimeObj
}; };
} }
return (try env.new_string(buffer.values)).pop().?; return env.new_string(buffer.values);
} }
pub fn execute(self: *Self, env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.RuntimeObj { pub fn execute(self: *Self, env: *kym.RuntimeEnv, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
self.cursor = 0; self.cursor = 0;
while (self.cursor < self.opcodes.values.len) : (self.cursor += 1) { while (self.cursor < self.opcodes.values.len) : (self.cursor += 1) {
switch (self.opcodes.values[self.cursor]) { switch (self.opcodes.values[self.cursor]) {
.ret => break, .ret => break,
.pop => env.discard(),
.push_nil => _ = try env.push(null), .pop => {
.push_true => _ = try env.new_boolean(true), if (try env.pop_local()) |ref| {
.push_false => _ = try env.new_boolean(false), env.discard(ref);
.push_const => |push_const| _ = try env.push(try self.get_constant(env, push_const)), }
.push_local => |push_local| _ = try env.local_get(push_local), },
.push_top => _ = try env.local_top(),
.push_nil => try env.locals.push_one(null),
.push_true => try env.locals.push_one(try env.new_boolean(true)),
.push_false => try env.locals.push_one(try env.new_boolean(false)),
.push_const => |push_const| {
try kym.assert(env, push_const < self.constants.values.len);
try env.locals.push_one(self.constants.values[push_const].acquire());
},
.push_local => |push_local| {
try kym.assert(env, push_local < (env.locals.values.len - frame.locals_top));
if (env.locals.values[frame.locals_top + push_local]) |local| {
try env.locals.push_one(local.acquire());
} else {
try env.locals.push_one(null);
}
},
.push_top => {
const frame_locals = env.locals.values[frame.locals_top ..];
try kym.assert(env, frame_locals.len != 0);
if (frame_locals[frame_locals.len - 1]) |local| {
try env.locals.push_one(local.acquire());
} else {
try env.locals.push_one(null);
}
},
.push_table => |push_table| { .push_table => |push_table| {
const table = (try env.new_table()).pop().?; const table = try env.new_table();
errdefer env.release(table); errdefer env.discard(table);
if (push_table != 0) { {
var popped = @as(usize, 0); var popped = @as(usize, 0);
while (popped < push_table) : (popped += 1) { while (popped < push_table) : (popped += 1) {
const index = try env.expect_object(env.pop()); const index = try env.expect(try env.pop_local());
defer env.release(index); defer env.discard(index);
if (env.pop()) |object| { if (try env.pop_local()) |value| {
defer env.release(object); defer env.discard(value);
try (try env.push(table)).index_set(index, object); try env.set(table, index, value);
} }
} }
} }
_ = try env.push(table); try env.locals.push_one(table);
}, },
.push_boxed => { .push_boxed => {
const value = env.pop(); const value = try env.pop_local();
defer { defer {
if (value) |object| { if (value) |ref| {
env.release(object); env.discard(ref);
} }
} }
_ = try env.new_boxed(value); const boxable = try env.new_boxed(value);
errdefer env.discard(boxable);
try env.locals.push_one(boxable);
}, },
.push_binding => |push_binding| _ = try env.push(try self.get_binding(env, push_binding)), .push_binding => |push_binding| {
.push_concat => |push_concat| _ = try env.concat(push_concat), try kym.assert(env, push_binding <= self.bindings.len);
try env.locals.push_one(if (self.bindings[push_binding]) |value| value.acquire() else null);
},
.push_concat => |push_concat| {
const frame_locals = env.locals.values[frame.locals_top ..];
try kym.assert(env, push_concat <= frame_locals.len);
const concat_locals = frame_locals[(frame_locals.len - push_concat) .. frame_locals.len];
for (concat_locals) |value| {
_ = try env.expect(value);
}
const concatenated_value = try env.concat(@ptrCast(concat_locals));
errdefer env.discard(concatenated_value);
var to_pop = concat_locals.len;
while (to_pop != 0) : (to_pop -= 1) {
const popped = env.pop_local() catch unreachable;
coral.debug.assert(popped != null);
env.discard(popped.?);
}
try env.locals.push_one(concatenated_value);
},
.bind => |bind| { .bind => |bind| {
const callable = try env.expect_object(env.pop()); const callable = try env.expect(try env.pop_local());
defer env.release(callable); errdefer env.discard(callable);
const chunk = @as(*Self, @ptrCast(@alignCast(try env.expect_dynamic(callable, typeinfo)))); const chunk = @as(*Self, @ptrCast(@alignCast(try env.unwrap_dynamic(callable, typeinfo))));
if (chunk.bindings.len != 0) { chunk.bindings = try coral.io.allocate_many(env.allocator, bind, @as(?*kym.RuntimeRef, null));
return env.raise(error.IllegalState, "cannot bind values to an already-bound chunk", .{});
}
chunk.bindings = try coral.io.allocate_many(env.allocator, bind, @as(?*kym.RuntimeObj, null));
for (0 .. bind) |index| { for (0 .. bind) |index| {
const value = env.pop(); const value = try env.pop_local();
errdefer { errdefer {
if (value) |object| { if (value) |ref| {
env.release(object); env.discard(ref);
} }
} }
chunk.bindings[index] = value; const binding = &chunk.bindings[index];
if (binding.*) |*existing_binding| {
env.discard(existing_binding.*);
} }
_ = try env.push(callable); binding.* = value;
}
try env.locals.push_one(callable);
}, },
.push_builtin => |push_builtin| _ = try env.new_syscall(switch (push_builtin) { .push_builtin => |push_builtin| {
const builtin_syscall = try env.new_syscall(switch (push_builtin) {
.import => syscall_import, .import => syscall_import,
.print => syscall_print, .print => syscall_print,
.vec2 => syscall_vec2, .vec2 => syscall_vec2,
.vec3 => syscall_vec3, .vec3 => syscall_vec3,
}), });
.set_local => |local_set| _ = try env.local_set(local_set, env.pop()), errdefer env.discard(builtin_syscall);
.get_box => _ = try env.boxed_get(),
try env.locals.push_one(builtin_syscall);
},
.set_local => |local_set| {
const local = &env.locals.values[frame.locals_top + local_set];
if (local.*) |previous_local| {
env.discard(previous_local);
}
local.* = try env.pop_local();
},
.get_box => {
const box = try env.expect(try env.pop_local());
defer env.discard(box);
if (try env.get_boxed(box)) |unboxed| {
errdefer env.discard(unboxed);
try env.locals.push_one(unboxed);
} else {
try env.locals.push_one(null);
}
},
.set_box => { .set_box => {
const value = env.pop(); const box = try env.expect(try env.pop_local());
defer { defer env.discard(box);
if (value) |object| {
env.release(object);
}
}
try env.boxed_set(value); const value = try env.expect(try env.pop_local());
errdefer env.discard(value);
try env.set_boxed(box, value);
}, },
.get_dynamic => { .get_dynamic => {
const index = try env.expect_object(env.pop()); const index = try env.expect(try env.pop_local());
defer env.release(index); defer env.discard(index);
_ = try env.index_get(index); const indexable = try env.expect(try env.pop_local());
defer env.discard(indexable);
const value = try env.get(indexable, index);
errdefer {
if (value) |ref| {
env.discard(ref);
}
}
try env.locals.push_one(value);
}, },
.set_dynamic => { .set_dynamic => {
const value = env.pop(); const value = try env.pop_local();
defer { defer {
if (value) |object| { if (value) |ref| {
env.release(object); env.discard(ref);
} }
} }
const index = try env.expect_object(env.pop()); const index = try env.expect(try env.pop_local());
defer env.release(index); defer env.discard(index);
try env.index_set(index, value); const indexable = try env.expect(try env.pop_local());
defer env.discard(indexable);
try env.set(indexable, index, value);
}, },
.call => |call| _ = try env.call(call), .call => |call| {
const result = call: {
const callable = try env.expect(try env.pop_local());
defer env.discard(callable);
const call_frame = try env.push_frame(callable, call);
defer env.pop_frame();
break: call try env.call_frame(&call_frame);
};
errdefer {
if (result) |ref| {
env.discard(ref);
}
}
try env.locals.push_one(result);
},
.not => { .not => {
const object = try env.expect_object(env.pop()); if (try env.pop_local()) |value| {
defer env.discard(value);
defer env.release(object); try env.locals.push_one(try env.new_boolean(!value.is_truthy()));
} else {
_ = try env.new_boolean(object.is_false()); try env.locals.push_one(try env.new_boolean(true));
}
}, },
.neg => _ = try env.neg(), .neg => {
const value = try env.expect(try env.pop_local());
defer env.discard(value);
try env.locals.push_one(try env.neg(value));
},
.add => { .add => {
const addable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(addable); defer env.discard(rhs);
_ = switch (try env.expect_numeric(addable)) { const lhs = try env.expect(try env.pop_local());
.fixed => |fixed| try env.fixed_add(fixed),
.float => |float| try env.float_add(float), defer env.discard(lhs);
.vector2 => |vector2| try env.vector2_add(vector2),
.vector3 => |vector3| try env.vector3_add(vector3), try env.locals.push_one(try env.add(lhs, rhs));
};
}, },
.sub => { .sub => {
const subtractable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(subtractable); defer env.discard(rhs);
_ = switch (try env.expect_numeric(subtractable)) { const lhs = try env.expect(try env.pop_local());
.fixed => |fixed| try env.fixed_subtract(fixed),
.float => |float| try env.float_subtract(float), defer env.discard(lhs);
.vector2 => |vector2| try env.vector2_subtract(vector2),
.vector3 => |vector3| try env.vector3_subtract(vector3), try env.locals.push_one(try env.sub(lhs, rhs));
};
}, },
.mul => { .mul => {
const multiplicable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(multiplicable); defer env.discard(rhs);
_ = switch (try env.expect_numeric(multiplicable)) { const lhs = try env.expect(try env.pop_local());
.fixed => |fixed| try env.fixed_multiply(fixed),
.float => |float| try env.float_multiply(float), defer env.discard(lhs);
.vector2 => |vector2| try env.vector2_multiply(vector2),
.vector3 => |vector3| try env.vector3_multiply(vector3), try env.locals.push_one(try env.mul(lhs, rhs));
};
}, },
.div => { .div => {
const divisible = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(divisible); defer env.discard(rhs);
_ = switch (try env.expect_numeric(divisible)) { const lhs = try env.expect(try env.pop_local());
.fixed => |fixed| try env.fixed_divide(fixed),
.float => |float| try env.float_divide(float), defer env.discard(lhs);
.vector2 => |vector2| try env.vector2_divide(vector2),
.vector3 => |vector3| try env.vector3_divide(vector3), try env.locals.push_one(try env.div(lhs, rhs));
};
}, },
.eql => { .eql => {
if (env.pop()) |equatable| { if (try env.pop_local()) |rhs| {
defer env.release(equatable); env.discard(rhs);
_ = try env.equals_object(equatable); if (try env.pop_local()) |lhs| {
env.discard(lhs);
try env.locals.push_one(try env.new_boolean(lhs.equals(rhs)));
} else { } else {
_ = try env.equals_nil(); try env.locals.push_one(try env.new_boolean(false));
}
} else {
if (try env.pop_local()) |lhs| {
env.discard(lhs);
try env.locals.push_one(try env.new_boolean(false));
} else {
try env.locals.push_one(try env.new_boolean(true));
}
} }
}, },
.cgt => { .cgt => {
const comparable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(comparable); defer env.discard(rhs);
_ = try env.compare_greater(comparable); const lhs = try env.expect(try env.pop_local());
defer env.discard(lhs);
try env.locals.push_one(try env.new_boolean(try env.compare(lhs, rhs) > 0));
}, },
.clt => { .clt => {
const comparable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(comparable); defer env.discard(rhs);
_ = try env.compare_less(comparable); const lhs = try env.expect(try env.pop_local());
defer env.discard(lhs);
try env.locals.push_one(try env.new_boolean(try env.compare(lhs, rhs) < 0));
}, },
.cge => { .cge => {
const comparable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(comparable); defer env.discard(rhs);
_ = try env.compare_greater_equals(comparable); const lhs = try env.expect(try env.pop_local());
defer env.discard(lhs);
try env.locals.push_one(try env.new_boolean(try env.compare(lhs, rhs) >= 0));
}, },
.cle => { .cle => {
const comparable = try env.expect_object(env.pop()); const rhs = try env.expect(try env.pop_local());
defer env.release(comparable); defer env.discard(rhs);
_ = try env.compare_less_equals(comparable); const lhs = try env.expect(try env.pop_local());
defer env.discard(lhs);
try env.locals.push_one(try env.new_boolean(try env.compare(lhs, rhs) <= 0));
}, },
.jf => |jf| { .jf => |jf| {
if (env.pop()) |condition| { if (try env.pop_local()) |condition| {
defer env.release(condition); defer env.discard(condition);
if (condition.is_false()) { if (!condition.is_truthy()) {
self.cursor = jf; self.cursor = jf;
} }
} else { } else {
@ -866,10 +1008,10 @@ pub fn execute(self: *Self, env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.Runtime
}, },
.jt => |jt| { .jt => |jt| {
if (env.pop()) |condition| { if (try env.pop_local()) |condition| {
defer env.release(condition); defer env.discard(condition);
if (condition.is_true()) { if (condition.is_truthy()) {
self.cursor = jt; self.cursor = jt;
} }
} }
@ -877,31 +1019,37 @@ pub fn execute(self: *Self, env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.Runtime
} }
} }
return env.pop(); return env.pop_local();
} }
fn get_binding(self: *Self, env: *kym.RuntimeEnv, index: usize) kym.RuntimeError!?*kym.RuntimeObj { pub fn free(self: *Self, env: *kym.RuntimeEnv) void {
if (index >= self.bindings.len) { while (self.constants.pop()) |constant| {
return env.raise(error.IllegalState, "invalid binding", .{}); env.discard(constant);
} }
return self.bindings[index]; self.constants.free();
self.opcodes.free();
env.discard(self.name);
if (self.bindings.len != 0) {
for (self.bindings) |binding| {
if (binding) |value| {
env.discard(value);
}
} }
fn get_constant(self: *const Self, env: *kym.RuntimeEnv, index: usize) kym.RuntimeError!*kym.RuntimeObj { env.allocator.deallocate(self.bindings);
if (index >= self.constants.values.len) {
return env.raise(error.IllegalState, "invalid constant", .{});
} }
return self.constants.values[index]; self.bindings = &.{};
} }
pub fn init(env: *kym.RuntimeEnv, name: []const coral.io.Byte, environment: *const tree.Environment) kym.RuntimeError!Self { pub fn make(env: *kym.RuntimeEnv, name: []const coral.io.Byte, environment: *const tree.Environment) kym.RuntimeError!Self {
var chunk = Self{ var chunk = Self{
.name = (try env.new_symbol(name)).pop().?, .name = try env.new_symbol(name),
.opcodes = OpcodeList.init(env.allocator), .opcodes = OpcodeList.make(env.allocator),
.constants = ConstList.init(env.allocator), .constants = ConstList.make(env.allocator),
.lines = LineList.init(env.allocator), .lines = LineList.make(env.allocator),
.bindings = &.{}, .bindings = &.{},
.arity = environment.argument_count, .arity = environment.argument_count,
.cursor = 0, .cursor = 0,
@ -917,81 +1065,38 @@ pub fn init(env: *kym.RuntimeEnv, name: []const coral.io.Byte, environment: *con
return chunk; return chunk;
} }
fn syscall_import(env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.RuntimeObj { fn syscall_import(env: *kym.RuntimeEnv, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
const arg = (try env.arg_get(0)).pop() orelse { return env.import(file.Path.from(&.{try env.unwrap_string(try frame.expect_arg(env, 0))}));
return env.raise(error.BadOperation, "`@import` requires one argument to be a valid import path", .{});
};
defer env.release(arg);
return (try env.import(file.Path.from(&.{try env.expect_string(arg)}))).pop();
} }
fn syscall_print(env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.RuntimeObj { fn syscall_print(env: *kym.RuntimeEnv, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
const string = (try (try env.arg_get(0)).to_string()).pop().?; env.print(try env.unwrap_string(try frame.expect_arg(env, 0)));
defer env.release(string);
env.print(string.is_string().?);
return null; return null;
} }
fn syscall_vec2(env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.RuntimeObj { fn syscall_vec2(env: *kym.RuntimeEnv, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
const x = @as(f32, get_x: { const x = @as(f32, @floatCast(try env.unwrap_float(try frame.expect_arg(env, 0))));
const x = (try env.arg_get(0)).pop() orelse {
return env.raise(error.BadOperation, "a first argument is required to create a vector", .{});
};
defer env.release(x); if (frame.has_arg(env, 1)) |y| {
return env.new_vector2(x, @floatCast(try env.unwrap_float(y)));
break: get_x @floatCast(try env.expect_float(x));
});
if ((try env.arg_get(1)).pop()) |y| {
defer env.release(y);
return (try env.new_vector2(.{
.y = @floatCast(try env.expect_float(y)),
.x = x,
})).pop();
} }
return (try env.new_vector2(kym.Vector2.from_scalar(x))).pop(); return env.new_vector2(x, x);
} }
fn syscall_vec3(env: *kym.RuntimeEnv) kym.RuntimeError!?*kym.RuntimeObj { fn syscall_vec3(env: *kym.RuntimeEnv, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
const x = @as(f32, get_x: { const x = @as(f32, @floatCast(try env.unwrap_float(try frame.expect_arg(env, 0))));
const x = (try env.arg_get(0)).pop() orelse {
return env.raise(error.BadOperation, "a first argument is required to create a vector", .{});
};
defer env.release(x); if (frame.has_arg(env, 1)) |y| {
return env.new_vector3(
break: get_x @floatCast(try env.expect_float(x)); x,
}); @floatCast(try env.unwrap_float(y)),
@floatCast(try env.unwrap_float(try frame.expect_arg(env, 2))),
if ((try env.arg_get(1)).pop()) |y| { );
defer env.release(y);
return (try env.new_vector3(.{
.z = @as(f32, get_z: {
const z = (try env.arg_get(0)).pop() orelse {
return env.raise(error.BadOperation,
"a third argument is required to create a vector if a first and second exist", .{});
};
defer env.release(z);
break: get_z @floatCast(try env.expect_float(z));
}),
.y = @floatCast(try env.expect_float(y)),
.x = x,
})).pop();
} }
return (try env.new_vector3(kym.Vector3.from_scalar(x))).pop(); return env.new_vector3(x, x, x);
} }
pub const typeinfo = &kym.Typeinfo{ pub const typeinfo = &kym.Typeinfo{
@ -1002,16 +1107,16 @@ pub const typeinfo = &kym.Typeinfo{
.to_string = typeinfo_to_string, .to_string = typeinfo_to_string,
}; };
fn typeinfo_call(context: kym.Typeinfo.CallContext) kym.RuntimeError!?*kym.RuntimeObj { fn typeinfo_call(env: *kym.RuntimeEnv, userdata: []coral.io.Byte, frame: *const kym.Frame) kym.RuntimeError!?*kym.RuntimeRef {
return @as(*Self, @ptrCast(@alignCast(context.userdata))).execute(context.env); return @as(*Self, @ptrCast(@alignCast(userdata))).execute(env, frame);
} }
fn typeinfo_destruct(context: kym.Typeinfo.DestructContext) void { fn typeinfo_destruct(env: *kym.RuntimeEnv, userdata: []coral.io.Byte) void {
@as(*Self, @ptrCast(@alignCast(context.userdata))).deinit(context.env); @as(*Self, @ptrCast(@alignCast(userdata))).free(env);
} }
fn typeinfo_to_string(context: kym.Typeinfo.ToStringContext) kym.RuntimeError!*kym.RuntimeObj { fn typeinfo_to_string(env: *kym.RuntimeEnv, userdata: []coral.io.Byte) coral.io.AllocationError!*kym.RuntimeRef {
return (try (try context.env.push(@as(*Self, @ptrCast(@alignCast(context.userdata))).name)).to_string()).pop().?; return env.to_string(@as(*Self, @ptrCast(@alignCast(userdata))).name);
} }
pub fn write(self: *Self, line: tokens.Line, opcode: Opcode) coral.io.AllocationError!void { pub fn write(self: *Self, line: tokens.Line, opcode: Opcode) coral.io.AllocationError!void {

View File

@ -5,41 +5,41 @@ const kym = @import("../kym.zig");
associative: RefTable, associative: RefTable,
contiguous: RefList, contiguous: RefList,
const RefList = coral.list.Stack(?*kym.RuntimeObj); const RefList = coral.list.Stack(?*kym.RuntimeRef);
const RefTable = coral.map.Table(*kym.RuntimeObj, *kym.RuntimeObj, struct { const RefTable = coral.map.Table(*kym.RuntimeRef, *kym.RuntimeRef, struct {
pub const hash = kym.RuntimeObj.hash; pub const hash = kym.RuntimeRef.hash;
pub const equals = kym.RuntimeObj.equals; pub const equals = kym.RuntimeRef.equals;
}); });
const Self = @This(); const Self = @This();
pub fn deinit(self: *Self, env: *kym.RuntimeEnv) void { pub fn free(self: *Self, env: *kym.RuntimeEnv) void {
{ {
var field_iterable = self.associative.as_iterable(); var field_iterable = self.associative.as_iterable();
while (field_iterable.next()) |entry| { while (field_iterable.next()) |entry| {
env.release(entry.key); env.discard(entry.key);
env.release(entry.value); env.discard(entry.value);
} }
} }
self.associative.deinit(); self.associative.free();
while (self.contiguous.pop()) |value| { while (self.contiguous.pop()) |value| {
if (value) |ref| { if (value) |ref| {
env.release(ref); env.discard(ref);
} }
} }
self.contiguous.deinit(); self.contiguous.free();
} }
pub fn init(env: *kym.RuntimeEnv) Self { pub fn make(env: *kym.RuntimeEnv) Self {
return .{ return .{
.associative = RefTable.init(env.allocator, .{}), .associative = RefTable.make(env.allocator, .{}),
.contiguous = RefList.init(env.allocator), .contiguous = RefList.make(env.allocator),
}; };
} }
@ -51,41 +51,41 @@ pub const typeinfo = &kym.Typeinfo{
.set = typeinfo_set, .set = typeinfo_set,
}; };
fn typeinfo_destruct(context: kym.Typeinfo.DestructContext) void { fn typeinfo_destruct(env: *kym.RuntimeEnv, userdata: []coral.io.Byte) void {
@as(*Self, @ptrCast(@alignCast(context.userdata))).deinit(context.env); @as(*Self, @ptrCast(@alignCast(userdata))).free(env);
} }
fn typeinfo_get(context: kym.Typeinfo.GetContext) kym.RuntimeError!?*kym.RuntimeObj { fn typeinfo_get(env: *kym.RuntimeEnv, userdata: []coral.io.Byte, index: *const kym.RuntimeRef) kym.RuntimeError!?*kym.RuntimeRef {
const table = @as(*Self, @ptrCast(@alignCast(context.userdata))); const table = @as(*Self, @ptrCast(@alignCast(userdata)));
const index = (try context.push_index()).pop().?; const acquired_index = index.acquire();
defer context.env.release(index); defer env.discard(acquired_index);
if (index.is_fixed()) |fixed| { if (acquired_index.as_fixed()) |fixed| {
if (fixed < 0) { if (fixed < 0) {
// TODO: Negative indexing. // TODO: Negative indexing.
unreachable; unreachable;
} }
if (fixed < table.contiguous.values.len) { if (fixed < table.contiguous.values.len) {
return table.contiguous.values[@intCast(fixed)]; return (table.contiguous.values[@intCast(fixed)] orelse return null).acquire();
} }
} }
if (table.associative.lookup(index)) |value| { if (table.associative.lookup(acquired_index)) |value| {
return value; return value.acquire();
} }
return null; return null;
} }
fn typeinfo_set(context: kym.Typeinfo.SetContext) kym.RuntimeError!void { fn typeinfo_set(env: *kym.RuntimeEnv, userdata: []coral.io.Byte, index: *const kym.RuntimeRef, value: ?*const kym.RuntimeRef) kym.RuntimeError!void {
const table = @as(*Self, @ptrCast(@alignCast(context.userdata))); const table = @as(*Self, @ptrCast(@alignCast(userdata)));
const index = (try context.push_index()).pop().?; const acquired_index = index.acquire();
errdefer context.env.release(index); errdefer env.discard(acquired_index);
if (index.is_fixed()) |fixed| { if (acquired_index.as_fixed()) |fixed| {
if (fixed < 0) { if (fixed < 0) {
// TODO: Negative indexing. // TODO: Negative indexing.
unreachable; unreachable;
@ -95,34 +95,28 @@ fn typeinfo_set(context: kym.Typeinfo.SetContext) kym.RuntimeError!void {
const maybe_replacing = &table.contiguous.values[@intCast(fixed)]; const maybe_replacing = &table.contiguous.values[@intCast(fixed)];
if (maybe_replacing.*) |replacing| { if (maybe_replacing.*) |replacing| {
context.env.release(replacing); env.discard(replacing);
} }
if ((try context.push_value()).pop()) |value| { maybe_replacing.* = if (value) |ref| ref.acquire() else null;
errdefer context.env.release(value);
maybe_replacing.* = value;
} else {
maybe_replacing.* = null;
}
return; return;
} }
} }
const value = (try context.push_value()).pop() orelse { const acquired_value = (value orelse {
if (table.associative.remove(index)) |removed| { if (table.associative.remove(acquired_index)) |removed| {
context.env.release(removed.key); env.discard(removed.key);
context.env.release(removed.value); env.discard(removed.value);
} }
return; return;
}; }).acquire();
errdefer context.env.release(value); errdefer env.discard(acquired_value);
if (try table.associative.replace(index, value)) |replaced| { if (try table.associative.replace(acquired_index, acquired_value)) |replaced| {
context.env.release(replaced.key); env.discard(replaced.key);
context.env.release(replaced.value); env.discard(replaced.value);
} }
} }

View File

@ -192,17 +192,17 @@ pub const Root = struct {
return coral.io.allocate_one(self.arena.as_allocator(), stmt); return coral.io.allocate_one(self.arena.as_allocator(), stmt);
} }
pub fn deinit(self: *Root) void { pub fn free(self: *Root) void {
self.error_messages.deinit(); self.error_messages.free();
self.arena.deinit(); self.arena.free();
} }
pub fn init(allocator: coral.io.Allocator) coral.io.AllocationError!Root { pub fn make(allocator: coral.io.Allocator) coral.io.AllocationError!Root {
const arena_page_size = 4096; const arena_page_size = 4096;
return .{ return .{
.arena = coral.arena.Stacking.init(allocator, arena_page_size), .arena = coral.arena.Stacking.make(allocator, arena_page_size),
.error_messages = MessageList.init(allocator), .error_messages = MessageList.make(allocator),
.environment = .{}, .environment = .{},
}; };
} }

View File

@ -23,7 +23,7 @@ pub fn run_app(file_access: file.Access) void {
defer ext.SDL_Quit(); defer ext.SDL_Quit();
var script_env = kym.RuntimeEnv.init(heap.allocator, 255, .{ var script_env = kym.RuntimeEnv.make(heap.allocator, .{
.print = app.log_info, .print = app.log_info,
.print_error = app.log_fail, .print_error = app.log_fail,
.import_access = file_access, .import_access = file_access,
@ -31,7 +31,7 @@ pub fn run_app(file_access: file.Access) void {
return app.log_fail("failed to initialize script runtime"); return app.log_fail("failed to initialize script runtime");
}; };
defer script_env.deinit(); defer script_env.free();
var manifest = app.Manifest{}; var manifest = app.Manifest{};