do not intern Structs, Enums and Unions

This commit is contained in:
Techatrix 2023-01-28 18:02:29 +01:00
parent 9131285db1
commit 55364f2e2b
5 changed files with 240 additions and 284 deletions

View File

@ -23,7 +23,6 @@ allocator: std.mem.Allocator,
ip: InternPool = .{}, ip: InternPool = .{},
document_store: *DocumentStore, document_store: *DocumentStore,
uri: DocumentStore.Uri, uri: DocumentStore.Uri,
decls: std.ArrayListUnmanaged(Decl) = .{},
namespaces: std.MultiArrayList(Namespace) = .{}, namespaces: std.MultiArrayList(Namespace) = .{},
/// Interpreter diagnostic errors /// Interpreter diagnostic errors
@ -69,7 +68,6 @@ pub fn deinit(interpreter: *ComptimeInterpreter) void {
interpreter.namespaces.items(.usingnamespaces)[i].deinit(interpreter.allocator); interpreter.namespaces.items(.usingnamespaces)[i].deinit(interpreter.allocator);
} }
interpreter.namespaces.deinit(interpreter.allocator); interpreter.namespaces.deinit(interpreter.allocator);
interpreter.decls.deinit(interpreter.allocator);
} }
pub const Type = struct { pub const Type = struct {
@ -86,29 +84,18 @@ pub const Value = struct {
ty: Index, ty: Index,
val: Index, val: Index,
}; };
pub const Decl = struct {
name: []const u8,
ty: Index,
val: Index,
alignment: u16,
address_space: std.builtin.AddressSpace,
is_pub: bool,
is_exported: bool,
};
// pub const Comptimeness = enum { @"comptime", runtime }; // pub const Comptimeness = enum { @"comptime", runtime };
pub const NamespaceIndex = InternPool.NamespaceIndex;
pub const Namespace = struct { pub const Namespace = struct {
/// always points to Namespace or Index.none /// always points to Namespace or Index.none
parent: NamespaceIndex, parent: Namespace.Index,
node_idx: Ast.Node.Index, node_idx: Ast.Node.Index,
/// Will be a struct, enum, union, opaque or .none /// Will be a struct, enum, union, opaque or .none
ty: Index, ty: InternPool.Index,
decls: std.StringArrayHashMapUnmanaged(Decl) = .{}, decls: std.StringArrayHashMapUnmanaged(InternPool.DeclIndex) = .{},
usingnamespaces: std.ArrayListUnmanaged(NamespaceIndex) = .{}, usingnamespaces: std.ArrayListUnmanaged(InternPool.DeclIndex) = .{},
pub const Index = InternPool.NamespaceIndex;
// TODO: Actually use this value // TODO: Actually use this value
// comptimeness: Comptimeness, // comptimeness: Comptimeness,
@ -161,15 +148,15 @@ pub const InterpretResult = union(enum) {
pub fn huntItDown( pub fn huntItDown(
interpreter: *ComptimeInterpreter, interpreter: *ComptimeInterpreter,
namespace: NamespaceIndex, namespace: Namespace.Index,
decl_name: []const u8, decl_name: []const u8,
options: InterpretOptions, options: InterpretOptions,
) ?Decl { ) ?InternPool.DeclIndex {
_ = options; _ = options;
var current_namespace = namespace; var current_namespace = namespace;
while (current_namespace != .none) { while (current_namespace != .none) {
const decls: std.StringArrayHashMapUnmanaged(Decl) = interpreter.namespaces.items(.decls)[@enumToInt(current_namespace)]; const decls = interpreter.namespaces.items(.decls)[@enumToInt(current_namespace)];
defer current_namespace = interpreter.namespaces.items(.parent)[@enumToInt(current_namespace)]; defer current_namespace = interpreter.namespaces.items(.parent)[@enumToInt(current_namespace)];
if (decls.get(decl_name)) |decl| { if (decls.get(decl_name)) |decl| {
@ -199,7 +186,7 @@ pub const InterpretError = std.mem.Allocator.Error || std.fmt.ParseIntError || s
pub fn interpret( pub fn interpret(
interpreter: *ComptimeInterpreter, interpreter: *ComptimeInterpreter,
node_idx: Ast.Node.Index, node_idx: Ast.Node.Index,
namespace: NamespaceIndex, namespace: Namespace.Index,
options: InterpretOptions, options: InterpretOptions,
) InterpretError!InterpretResult { ) InterpretError!InterpretResult {
const tree = interpreter.getHandle().tree; const tree = interpreter.getHandle().tree;
@ -229,10 +216,16 @@ pub fn interpret(
.node_idx = node_idx, .node_idx = node_idx,
.ty = undefined, .ty = undefined,
}); });
const container_namespace = @intToEnum(NamespaceIndex, interpreter.namespaces.len - 1); const container_namespace = @intToEnum(Namespace.Index, interpreter.namespaces.len - 1);
var fields = std.ArrayListUnmanaged(InternPool.Struct.Field){}; const struct_index = try interpreter.ip.createStruct(interpreter.allocator, .{
defer fields.deinit(interpreter.allocator); .fields = .{},
.namespace = container_namespace,
.layout = .Auto, // TODO
.backing_int_ty = .none, // TODO
.status = .none, // TODO
});
var struct_info = interpreter.ip.getStruct(struct_index);
var buffer: [2]Ast.Node.Index = undefined; var buffer: [2]Ast.Node.Index = undefined;
@ -259,28 +252,18 @@ pub fn interpret(
); );
continue; continue;
} }
const field_name = try interpreter.ip.get(interpreter.allocator, .{
.bytes = tree.tokenSlice(container_field.ast.main_token), const field_name = tree.tokenSlice(container_field.ast.main_token);
});
const field: InternPool.Struct.Field = .{ try struct_info.fields.put(interpreter.allocator, field_name, .{
.name = field_name,
.ty = init_type_value.val, .ty = init_type_value.val,
.default_value = default_value, .default_value = default_value,
.alignment = 0, // TODO, .alignment = 0, // TODO,
.is_comptime = false, // TODO .is_comptime = false, // TODO
}; });
try fields.append(interpreter.allocator, field);
} }
const struct_type = try interpreter.ip.get(interpreter.allocator, Key{ const struct_type = try interpreter.ip.get(interpreter.allocator, Key{ .struct_type = struct_index }); // TODO potential double free on struct_info.field
.struct_type = .{
.fields = fields.items,
.namespace = namespace,
.layout = .Auto, // TODO
.backing_int_ty = .none, // TODO
},
});
interpreter.namespaces.items(.ty)[@enumToInt(container_namespace)] = struct_type; interpreter.namespaces.items(.ty)[@enumToInt(container_namespace)] = struct_type;
return InterpretResult{ .value = Value{ return InterpretResult{ .value = Value{
@ -317,7 +300,7 @@ pub fn interpret(
if (v.ty != type_type) return InterpretResult{ .nothing = {} }; if (v.ty != type_type) return InterpretResult{ .nothing = {} };
} }
try decls.putNoClobber(interpreter.allocator, name, .{ const decl_index = try interpreter.ip.createDecl(interpreter.allocator, .{
.name = name, .name = name,
.ty = if (type_value) |v| v.val else init_value.?.ty, .ty = if (type_value) |v| v.val else init_value.?.ty,
.val = if (init_value) |init| init.val else .none, .val = if (init_value) |init| init.val else .none,
@ -326,6 +309,7 @@ pub fn interpret(
.is_pub = true, // TODO .is_pub = true, // TODO
.is_exported = false, // TODO .is_exported = false, // TODO
}); });
try decls.putNoClobber(interpreter.allocator, name, decl_index);
// TODO: Am I a dumbo shrimp? (e.g. is this tree shaking correct? works on my machine so like...) // TODO: Am I a dumbo shrimp? (e.g. is this tree shaking correct? works on my machine so like...)
@ -345,7 +329,7 @@ pub fn interpret(
.node_idx = node_idx, .node_idx = node_idx,
.ty = .none, .ty = .none,
}); });
const block_namespace = @intToEnum(NamespaceIndex, interpreter.namespaces.len - 1); const block_namespace = @intToEnum(Namespace.Index, interpreter.namespaces.len - 1);
var buffer: [2]Ast.Node.Index = undefined; var buffer: [2]Ast.Node.Index = undefined;
const statements = ast.blockStatements(tree, node_idx, &buffer).?; const statements = ast.blockStatements(tree, node_idx, &buffer).?;
@ -442,7 +426,8 @@ pub fn interpret(
} }
// Logic to find identifiers in accessible scopes // Logic to find identifiers in accessible scopes
if (interpreter.huntItDown(namespace, identifier, options)) |decl| { if (interpreter.huntItDown(namespace, identifier, options)) |decl_index| {
const decl = interpreter.ip.getDecl(decl_index);
return InterpretResult{ .value = Value{ return InterpretResult{ .value = Value{
.interpreter = interpreter, .interpreter = interpreter,
.node_idx = node_idx, .node_idx = node_idx,
@ -478,7 +463,8 @@ pub fn interpret(
if (irv.val == .none) break :blk true; if (irv.val == .none) break :blk true;
const ty_key = interpreter.ip.indexToKey(irv.val); const ty_key = interpreter.ip.indexToKey(irv.val);
if (interpreter.huntItDown(ty_key.getNamespace(), field_name, options)) |decl| { if (interpreter.huntItDown(ty_key.getNamespace(interpreter.ip), field_name, options)) |decl_index| {
const decl = interpreter.ip.getDecl(decl_index);
return InterpretResult{ .value = Value{ return InterpretResult{ .value = Value{
.interpreter = interpreter, .interpreter = interpreter,
.node_idx = node_idx, .node_idx = node_idx,
@ -492,19 +478,18 @@ pub fn interpret(
_ = error_set_info; _ = error_set_info;
}, },
.union_type => {}, // TODO .union_type => {}, // TODO
.enum_type => |enum_info| { // TODO .enum_type => |enum_index| { // TODO
if (interpreter.ip.contains(Key{ .bytes = field_name })) |field_name_index| { const enum_info = interpreter.ip.getEnum(enum_index);
for (enum_info.fields) |field| { if (enum_info.fields.get(field_name)) |field| {
if (field.name != field_name_index) continue; _ = field;
return InterpretResult{ return InterpretResult{
.value = Value{ .value = Value{
.interpreter = interpreter, .interpreter = interpreter,
.node_idx = data[node_idx].rhs, .node_idx = data[node_idx].rhs,
.ty = irv.val, .ty = irv.val,
.val = .none, // TODO resolve enum value .val = .none, // TODO resolve enum value
}, },
}; };
}
} }
}, },
else => break :blk false, else => break :blk false,
@ -583,17 +568,15 @@ pub fn interpret(
} }; } };
} }
}, },
.struct_type => |struct_info| blk: { .struct_type => |struct_index| blk: {
// if the intern pool does not contain the field name, it is impossible that there is a field with the given name const struct_info = interpreter.ip.getStruct(struct_index);
const field_name_index = interpreter.ip.contains(Key{ .bytes = field_name }) orelse break :blk true; if (struct_info.fields.getIndex(field_name)) |field_index| {
const field = struct_info.fields.values()[field_index];
for (struct_info.fields) |field, i| {
if (field.name != field_name_index) continue;
const val = found_val: { const val = found_val: {
if (irv.val == .none) break :found_val .none; if (irv.val == .none) break :found_val .none;
const val_key = interpreter.ip.indexToKey(irv.val); const val_key = interpreter.ip.indexToKey(irv.val);
if (val_key != .aggregate) break :found_val .none; if (val_key != .aggregate) break :found_val .none;
break :found_val val_key.aggregate[i]; break :found_val val_key.aggregate[field_index];
}; };
return InterpretResult{ .value = Value{ return InterpretResult{ .value = Value{
@ -813,15 +796,17 @@ pub fn interpret(
// TODO: Implement root support // TODO: Implement root support
if (std.mem.eql(u8, import_str[1 .. import_str.len - 1], "root")) { if (std.mem.eql(u8, import_str[1 .. import_str.len - 1], "root")) {
const struct_index = try interpreter.ip.createStruct(interpreter.allocator, .{
.fields = .{},
.namespace = .none,
.layout = .Auto,
.backing_int_ty = .none,
.status = .none,
});
return InterpretResult{ .value = Value{ return InterpretResult{ .value = Value{
.interpreter = interpreter, .interpreter = interpreter,
.node_idx = node_idx, .node_idx = node_idx,
.ty = try interpreter.ip.get(interpreter.allocator, Key{ .struct_type = .{ .ty = try interpreter.ip.get(interpreter.allocator, Key{ .struct_type = struct_index }),
.fields = &.{},
.namespace = .none,
.layout = .Auto,
.backing_int_ty = .none,
} }),
.val = try interpreter.ip.get(interpreter.allocator, Key{ .simple = .undefined_value }), .val = try interpreter.ip.get(interpreter.allocator, Key{ .simple = .undefined_value }),
} }; } };
} }
@ -865,7 +850,7 @@ pub fn interpret(
if (value.ty != type_type) return error.InvalidBuiltin; if (value.ty != type_type) return error.InvalidBuiltin;
if (interpreter.ip.indexToKey(field_name.ty) != .pointer_type) return error.InvalidBuiltin; // Check if it's a []const u8 if (interpreter.ip.indexToKey(field_name.ty) != .pointer_type) return error.InvalidBuiltin; // Check if it's a []const u8
const value_namespace = interpreter.ip.indexToKey(value.val).getNamespace(); const value_namespace = interpreter.ip.indexToKey(value.val).getNamespace(interpreter.ip);
if (value_namespace == .none) return error.InvalidBuiltin; if (value_namespace == .none) return error.InvalidBuiltin;
const name = interpreter.ip.indexToKey(field_name.val).bytes; // TODO add checks const name = interpreter.ip.indexToKey(field_name.val).bytes; // TODO add checks
@ -986,7 +971,8 @@ pub fn interpret(
if (namespace != .none) { if (namespace != .none) {
const decls = &interpreter.namespaces.items(.decls)[@enumToInt(namespace)]; const decls = &interpreter.namespaces.items(.decls)[@enumToInt(namespace)];
try decls.put(interpreter.allocator, name, .{
const decl_index = try interpreter.ip.createDecl(interpreter.allocator, .{
.name = name, .name = name,
.ty = type_type, .ty = type_type,
.val = function_type, .val = function_type,
@ -995,6 +981,7 @@ pub fn interpret(
.is_pub = false, // TODO .is_pub = false, // TODO
.is_exported = false, // TODO .is_exported = false, // TODO
}); });
try decls.putNoClobber(interpreter.allocator, name, decl_index);
} }
return InterpretResult{ .nothing = {} }; return InterpretResult{ .nothing = {} };
@ -1107,7 +1094,7 @@ pub fn interpret(
} }
pub const CallResult = struct { pub const CallResult = struct {
namespace: NamespaceIndex, namespace: Namespace.Index,
result: union(enum) { result: union(enum) {
value: Value, value: Value,
nothing, nothing,
@ -1116,7 +1103,7 @@ pub const CallResult = struct {
pub fn call( pub fn call(
interpreter: *ComptimeInterpreter, interpreter: *ComptimeInterpreter,
namespace: NamespaceIndex, namespace: Namespace.Index,
func_node_idx: Ast.Node.Index, func_node_idx: Ast.Node.Index,
arguments: []const Value, arguments: []const Value,
options: InterpretOptions, options: InterpretOptions,
@ -1136,7 +1123,7 @@ pub fn call(
.node_idx = func_node_idx, .node_idx = func_node_idx,
.ty = .none, .ty = .none,
}); });
const fn_namespace = @intToEnum(NamespaceIndex, interpreter.namespaces.len - 1); const fn_namespace = @intToEnum(Namespace.Index, interpreter.namespaces.len - 1);
const type_type = try interpreter.ip.get(interpreter.allocator, Key{ .simple = .type }); const type_type = try interpreter.ip.get(interpreter.allocator, Key{ .simple = .type });
@ -1157,7 +1144,8 @@ pub fn call(
if (param.name_token) |name_token| { if (param.name_token) |name_token| {
const name = offsets.tokenToSlice(tree, name_token); const name = offsets.tokenToSlice(tree, name_token);
try interpreter.namespaces.items(.decls)[@enumToInt(fn_namespace)].put(interpreter.allocator, name, .{ const decls = &interpreter.namespaces.items(.decls)[@enumToInt(fn_namespace)];
const decl_index = try interpreter.ip.createDecl(interpreter.allocator, .{
.name = name, .name = name,
.ty = tex.val, .ty = tex.val,
.val = arguments[arg_index].val, .val = arguments[arg_index].val,
@ -1166,6 +1154,7 @@ pub fn call(
.is_pub = true, // TODO .is_pub = true, // TODO
.is_exported = false, // TODO .is_exported = false, // TODO
}); });
try decls.putNoClobber(interpreter.allocator, name, decl_index);
arg_index += 1; arg_index += 1;
} }
} }

View File

@ -4,6 +4,11 @@ map: std.AutoArrayHashMapUnmanaged(void, void) = .{},
items: std.MultiArrayList(Item) = .{}, items: std.MultiArrayList(Item) = .{},
extra: std.ArrayListUnmanaged(u8) = .{}, extra: std.ArrayListUnmanaged(u8) = .{},
decls: std.ArrayListUnmanaged(InternPool.Decl) = .{},
structs: std.ArrayListUnmanaged(InternPool.Struct) = .{},
enums: std.ArrayListUnmanaged(InternPool.Enum) = .{},
unions: std.ArrayListUnmanaged(InternPool.Union) = .{},
const InternPool = @This(); const InternPool = @This();
const std = @import("std"); const std = @import("std");
const builtin = @import("builtin"); const builtin = @import("builtin");
@ -36,15 +41,26 @@ pub const Array = packed struct {
sentinel: Index = .none, sentinel: Index = .none,
}; };
pub const FieldStatus = enum {
none,
field_types_wip,
have_field_types,
layout_wip,
have_layout,
fully_resolved_wip,
fully_resolved,
};
pub const StructIndex = enum(u32) { _ };
pub const Struct = struct { pub const Struct = struct {
fields: []const Field, fields: std.StringArrayHashMapUnmanaged(Field),
namespace: NamespaceIndex, namespace: NamespaceIndex,
layout: std.builtin.Type.ContainerLayout = .Auto, layout: std.builtin.Type.ContainerLayout = .Auto,
backing_int_ty: Index, backing_int_ty: Index,
status: FieldStatus,
pub const Field = packed struct { pub const Field = packed struct {
/// guaranteed to be .bytes
name: Index,
ty: Index, ty: Index,
default_value: Index = .none, default_value: Index = .none,
alignment: u16 = 0, alignment: u16 = 0,
@ -66,17 +82,14 @@ pub const ErrorSet = struct {
names: []const Index, names: []const Index,
}; };
pub const EnumIndex = enum(u32) { _ };
pub const Enum = struct { pub const Enum = struct {
tag_type: Index, tag_type: Index,
fields: []const Field, fields: std.StringArrayHashMapUnmanaged(void),
values: std.AutoArrayHashMapUnmanaged(Index, void),
namespace: NamespaceIndex, namespace: NamespaceIndex,
tag_type_infered: bool, tag_type_infered: bool,
pub const Field = packed struct {
/// guaranteed to be .bytes
name: Index,
val: Index,
};
}; };
pub const Function = struct { pub const Function = struct {
@ -94,15 +107,16 @@ pub const Function = struct {
is_var_args: bool = false, is_var_args: bool = false,
}; };
pub const UnionIndex = enum(u32) { _ };
pub const Union = struct { pub const Union = struct {
tag_type: Index, tag_type: Index,
fields: []const Field, fields: std.StringArrayHashMapUnmanaged(Field),
namespace: NamespaceIndex, namespace: NamespaceIndex,
layout: std.builtin.Type.ContainerLayout = .Auto, layout: std.builtin.Type.ContainerLayout = .Auto,
status: FieldStatus,
pub const Field = packed struct { pub const Field = packed struct {
/// guaranteed to be .bytes
name: Index,
ty: Index, ty: Index,
alignment: u16, alignment: u16,
}; };
@ -134,19 +148,34 @@ pub const UnionValue = packed struct {
val: Index, val: Index,
}; };
pub const DeclIndex = enum(u32) { _ };
pub const Decl = struct {
name: []const u8,
ty: Index,
val: Index,
alignment: u16,
address_space: std.builtin.AddressSpace,
is_pub: bool,
is_exported: bool,
};
pub const Key = union(enum) { pub const Key = union(enum) {
simple: Simple, simple: Simple,
int_type: Int, int_type: Int,
pointer_type: Pointer, pointer_type: Pointer,
array_type: Array, array_type: Array,
struct_type: Struct, /// TODO consider *Struct instead of StructIndex
struct_type: StructIndex,
optional_type: Optional, optional_type: Optional,
error_union_type: ErrorUnion, error_union_type: ErrorUnion,
error_set_type: ErrorSet, error_set_type: ErrorSet,
enum_type: Enum, /// TODO consider *Enum instead of EnumIndex
enum_type: EnumIndex,
function_type: Function, function_type: Function,
union_type: Union, /// TODO consider *Union instead of UnionIndex
union_type: UnionIndex,
tuple_type: Tuple, tuple_type: Tuple,
vector_type: Vector, vector_type: Vector,
anyframe_type: AnyFrame, anyframe_type: AnyFrame,
@ -396,7 +425,8 @@ pub const Key = union(enum) {
}, },
.int_type => |int_info| return int_info, .int_type => |int_info| return int_info,
.enum_type => return panicOrElse("TODO", .{ .signedness = .unsigned, .bits = 0 }), .enum_type => return panicOrElse("TODO", .{ .signedness = .unsigned, .bits = 0 }),
.struct_type => |struct_info| { .struct_type => |struct_index| {
const struct_info = ip.getStruct(struct_index);
assert(struct_info.layout == .Packed); assert(struct_info.layout == .Packed);
key = ip.indexToKey(struct_info.backing_int_ty); key = ip.indexToKey(struct_info.backing_int_ty);
}, },
@ -485,11 +515,11 @@ pub const Key = union(enum) {
}; };
} }
pub fn getNamespace(ty: Key) NamespaceIndex { pub fn getNamespace(ty: Key, ip: InternPool) NamespaceIndex {
return switch (ty) { return switch (ty) {
.struct_type => |struct_info| struct_info.namespace, .struct_type => |struct_index| ip.getStruct(struct_index).namespace,
.enum_type => |enum_info| enum_info.namespace, .enum_type => |enum_index| ip.getEnum(enum_index).namespace,
.union_type => |union_info| union_info.namespace, .union_type => |union_index| ip.getUnion(union_index).namespace,
else => .none, else => .none,
}; };
} }
@ -555,10 +585,12 @@ pub const Key = union(enum) {
} }
return null; return null;
}, },
.struct_type => |struct_info| { .struct_type => |struct_index| {
for (struct_info.fields) |field| { const struct_info = ip.getStruct(struct_index);
if (field.is_comptime) continue; var field_it = struct_info.fields.iterator();
if (ip.indexToKey(field.ty).onePossibleValue(ip) != null) continue; while (field_it.next()) |entry| {
if (entry.value_ptr.is_comptime) continue;
if (ip.indexToKey(entry.value_ptr.ty).onePossibleValue(ip) != null) continue;
return null; return null;
} }
return panicOrElse("TODO return empty struct value", null); return panicOrElse("TODO return empty struct value", null);
@ -572,10 +604,11 @@ pub const Key = union(enum) {
}, },
.error_union_type => return null, .error_union_type => return null,
.error_set_type => return null, .error_set_type => return null,
.enum_type => |enum_info| { .enum_type => |enum_index| {
switch (enum_info.fields.len) { const enum_info = ip.getEnum(enum_index);
switch (enum_info.fields.count()) {
0 => return Key{ .simple = .unreachable_value }, 0 => return Key{ .simple = .unreachable_value },
1 => return ip.indexToKey(enum_info.fields[0].val), 1 => return ip.indexToKey(enum_info.values.keys()[0]),
else => return null, else => return null,
} }
}, },
@ -623,11 +656,7 @@ pub const Key = union(enum) {
ip: InternPool, ip: InternPool,
}; };
// TODO implement options pub const FormatOptions = struct {};
pub const FormatOptions = struct {
include_fields: bool = true,
include_declarations: bool = true,
};
fn formatType( fn formatType(
ctx: TypeFormatContext, ctx: TypeFormatContext,
@ -925,27 +954,27 @@ pub const Key = union(enum) {
.bytes => |bytes| try writer.print("\"{}\"", .{std.zig.fmtEscapes(bytes)}), .bytes => |bytes| try writer.print("\"{}\"", .{std.zig.fmtEscapes(bytes)}),
.aggregate => |aggregate| { .aggregate => |aggregate| {
const struct_info = ip.indexToKey(ty).struct_type; const struct_info = ip.getStruct(ip.indexToKey(ty).struct_type);
assert(aggregate.len == struct_info.fields.len); assert(aggregate.len == struct_info.fields.count());
try writer.writeAll(".{"); try writer.writeAll(".{");
var i: u32 = 0; var i: u32 = 0;
while (i < aggregate.len) : (i += 1) { while (i < aggregate.len) : (i += 1) {
if (i != 0) try writer.writeAll(", "); if (i != 0) try writer.writeAll(", ");
const field_name = ip.indexToKey(struct_info.fields[i].name).bytes; const field_name = struct_info.fields.keys()[i];
try writer.print(".{s} = ", .{field_name}); try writer.print(".{s} = ", .{field_name});
try printValue(ip.indexToKey(aggregate[i]), struct_info.fields[i].ty, ip, writer); try printValue(ip.indexToKey(aggregate[i]), struct_info.fields.values()[i].ty, ip, writer);
} }
try writer.writeByte('}'); try writer.writeByte('}');
}, },
.union_value => |union_value| { .union_value => |union_value| {
const union_info = ip.indexToKey(ty).union_type; const union_info = ip.getUnion(ip.indexToKey(ty).union_type);
const name = ip.indexToKey(union_info.fields[union_value.field_index].name).bytes; const name = union_info.fields.keys()[union_value.field_index];
try writer.print(".{{ .{} = {} }}", .{ try writer.print(".{{ .{} = {} }}", .{
std.zig.fmtId(name), std.zig.fmtId(name),
union_value.val.fmtValue(union_info.fields[union_value.field_index].ty, ip), union_value.val.fmtValue(union_info.fields.values()[union_value.field_index].ty, ip),
}); });
}, },
} }
@ -978,6 +1007,7 @@ pub const Item = struct {
/// Two values which have the same type can be equality compared simply /// Two values which have the same type can be equality compared simply
/// by checking if their indexes are equal, provided they are both in /// by checking if their indexes are equal, provided they are both in
/// the same `InternPool`. /// the same `InternPool`.
/// TODO split this into an Optional and non-Optional Index
pub const Index = enum(u32) { pub const Index = enum(u32) {
none = std.math.maxInt(u32), none = std.math.maxInt(u32),
_, _,
@ -1181,6 +1211,21 @@ pub fn deinit(ip: *InternPool, gpa: Allocator) void {
ip.map.deinit(gpa); ip.map.deinit(gpa);
ip.items.deinit(gpa); ip.items.deinit(gpa);
ip.extra.deinit(gpa); ip.extra.deinit(gpa);
for (ip.structs.items) |*item| {
item.fields.deinit(gpa);
}
for (ip.enums.items) |*item| {
item.fields.deinit(gpa);
item.values.deinit(gpa);
}
for (ip.unions.items) |*item| {
item.fields.deinit(gpa);
}
ip.decls.deinit(gpa);
ip.structs.deinit(gpa);
ip.enums.deinit(gpa);
ip.unions.deinit(gpa);
} }
pub fn indexToKey(ip: InternPool, index: Index) Key { pub fn indexToKey(ip: InternPool, index: Index) Key {
@ -1199,17 +1244,18 @@ pub fn indexToKey(ip: InternPool, index: Index) Key {
} }, } },
.type_pointer => .{ .pointer_type = ip.extraData(Pointer, data) }, .type_pointer => .{ .pointer_type = ip.extraData(Pointer, data) },
.type_array => .{ .array_type = ip.extraData(Array, data) }, .type_array => .{ .array_type = ip.extraData(Array, data) },
.type_struct => .{ .struct_type = ip.extraData(Struct, data) },
.type_optional => .{ .optional_type = .{ .payload_type = @intToEnum(Index, data) } }, .type_optional => .{ .optional_type = .{ .payload_type = @intToEnum(Index, data) } },
.type_anyframe => .{ .anyframe_type = .{ .child = @intToEnum(Index, data) } }, .type_anyframe => .{ .anyframe_type = .{ .child = @intToEnum(Index, data) } },
.type_error_union => .{ .error_union_type = ip.extraData(ErrorUnion, data) }, .type_error_union => .{ .error_union_type = ip.extraData(ErrorUnion, data) },
.type_error_set => .{ .error_set_type = ip.extraData(ErrorSet, data) }, .type_error_set => .{ .error_set_type = ip.extraData(ErrorSet, data) },
.type_enum => .{ .enum_type = ip.extraData(Enum, data) },
.type_function => .{ .function_type = ip.extraData(Function, data) }, .type_function => .{ .function_type = ip.extraData(Function, data) },
.type_union => .{ .union_type = ip.extraData(Union, data) },
.type_tuple => .{ .tuple_type = ip.extraData(Tuple, data) }, .type_tuple => .{ .tuple_type = ip.extraData(Tuple, data) },
.type_vector => .{ .vector_type = ip.extraData(Vector, data) }, .type_vector => .{ .vector_type = ip.extraData(Vector, data) },
.type_struct => .{ .struct_type = @intToEnum(StructIndex, data) },
.type_enum => .{ .enum_type = @intToEnum(EnumIndex, data) },
.type_union => .{ .union_type = @intToEnum(UnionIndex, data) },
.int_u32 => .{ .int_u64_value = @intCast(u32, data) }, .int_u32 => .{ .int_u64_value = @intCast(u32, data) },
.int_i32 => .{ .int_i64_value = @bitCast(i32, data) }, .int_i32 => .{ .int_i64_value = @bitCast(i32, data) },
.int_u64 => .{ .int_u64_value = ip.extraData(u64, data) }, .int_u64 => .{ .int_u64_value = ip.extraData(u64, data) },
@ -1248,6 +1294,10 @@ pub fn get(ip: *InternPool, gpa: Allocator, key: Key) Allocator.Error!Index {
.optional_type => |optional_ty| @enumToInt(optional_ty.payload_type), .optional_type => |optional_ty| @enumToInt(optional_ty.payload_type),
.anyframe_type => |anyframe_ty| @enumToInt(anyframe_ty.child), .anyframe_type => |anyframe_ty| @enumToInt(anyframe_ty.child),
.struct_type => |struct_index| @enumToInt(struct_index),
.enum_type => |enum_index| @enumToInt(enum_index),
.union_type => |union_index| @enumToInt(union_index),
.int_u64_value => |int_val| if (tag == .int_u32) @intCast(u32, int_val) else try ip.addExtra(gpa, int_val), .int_u64_value => |int_val| if (tag == .int_u32) @intCast(u32, int_val) else try ip.addExtra(gpa, int_val),
.int_i64_value => |int_val| if (tag == .int_i32) @bitCast(u32, @intCast(u32, int_val)) else try ip.addExtra(gpa, int_val), .int_i64_value => |int_val| if (tag == .int_i32) @bitCast(u32, @intCast(u32, int_val)) else try ip.addExtra(gpa, int_val),
.int_big_value => |big_int_val| try ip.addExtra(gpa, big_int_val.limbs), .int_big_value => |big_int_val| try ip.addExtra(gpa, big_int_val.limbs),
@ -1270,6 +1320,36 @@ pub fn contains(ip: InternPool, key: Key) ?Index {
return @intToEnum(Index, index); return @intToEnum(Index, index);
} }
pub fn getDecl(ip: *InternPool, index: InternPool.DeclIndex) *InternPool.Decl {
return &ip.decls.items[@enumToInt(index)];
}
pub fn getStruct(ip: InternPool, index: InternPool.StructIndex) *InternPool.Struct {
return &ip.structs.items[@enumToInt(index)];
}
pub fn getEnum(ip: InternPool, index: InternPool.EnumIndex) *InternPool.Enum {
return &ip.enums.items[@enumToInt(index)];
}
pub fn getUnion(ip: InternPool, index: InternPool.UnionIndex) *InternPool.Union {
return &ip.unions.items[@enumToInt(index)];
}
pub fn createDecl(ip: *InternPool, gpa: Allocator, decl: InternPool.Decl) error{OutOfMemory}!InternPool.DeclIndex {
try ip.decls.append(gpa, decl);
return @intToEnum(InternPool.DeclIndex, ip.decls.items.len - 1);
}
pub fn createStruct(ip: *InternPool, gpa: Allocator, struct_info: InternPool.Struct) error{OutOfMemory}!InternPool.StructIndex {
try ip.structs.append(gpa, struct_info);
return @intToEnum(InternPool.StructIndex, ip.structs.items.len - 1);
}
pub fn createEnum(ip: *InternPool, gpa: Allocator, enum_info: InternPool.Enum) error{OutOfMemory}!InternPool.EnumIndex {
try ip.enums.append(gpa, enum_info);
return @intToEnum(InternPool.EnumIndex, ip.enums.items.len - 1);
}
pub fn createUnion(ip: *InternPool, gpa: Allocator, union_info: InternPool.Union) error{OutOfMemory}!InternPool.UnionIndex {
try ip.unions.append(gpa, union_info);
return @intToEnum(InternPool.UnionIndex, ip.unions.items.len - 1);
}
fn addExtra(ip: *InternPool, gpa: Allocator, extra: anytype) Allocator.Error!u32 { fn addExtra(ip: *InternPool, gpa: Allocator, extra: anytype) Allocator.Error!u32 {
const T = @TypeOf(extra); const T = @TypeOf(extra);
comptime if (@sizeOf(T) <= 4) { comptime if (@sizeOf(T) <= 4) {
@ -2997,41 +3077,6 @@ test "array type" {
try testExpectFmtType(ip, u32_0_0_array_type, "[3:0]u32"); try testExpectFmtType(ip, u32_0_0_array_type, "[3:0]u32");
} }
test "struct type" {
const gpa = std.testing.allocator;
var ip: InternPool = .{};
defer ip.deinit(gpa);
const i32_type = try ip.get(gpa, .{ .int_type = .{ .signedness = .signed, .bits = 32 } });
const u64_type = try ip.get(gpa, .{ .int_type = .{ .signedness = .unsigned, .bits = 64 } });
const bool_type = try ip.get(gpa, .{ .simple = .bool });
const foo_name = try ip.get(gpa, .{ .bytes = "foo" });
const bar_name = try ip.get(gpa, .{ .bytes = "bar" });
const baz_name = try ip.get(gpa, .{ .bytes = "baz" });
const field1 = Struct.Field{ .name = foo_name, .ty = u64_type };
const field2 = Struct.Field{ .name = bar_name, .ty = i32_type };
const field3 = Struct.Field{ .name = baz_name, .ty = bool_type };
const struct_type_0 = try ip.get(gpa, .{ .struct_type = .{
.fields = &.{ field1, field2, field3 },
.namespace = .none,
.layout = .Auto,
.backing_int_ty = .none,
} });
const struct_type_1 = try ip.get(gpa, .{ .struct_type = .{
.fields = &.{ field1, field2, field3 },
.namespace = .none,
.layout = .Auto,
.backing_int_ty = .none,
} });
try std.testing.expect(struct_type_0 == struct_type_1);
}
test "struct value" { test "struct value" {
const gpa = std.testing.allocator; const gpa = std.testing.allocator;
@ -3041,18 +3086,17 @@ test "struct value" {
const i32_type = try ip.get(gpa, .{ .int_type = .{ .signedness = .signed, .bits = 32 } }); const i32_type = try ip.get(gpa, .{ .int_type = .{ .signedness = .signed, .bits = 32 } });
const bool_type = try ip.get(gpa, .{ .simple = .bool }); const bool_type = try ip.get(gpa, .{ .simple = .bool });
const foo_name = try ip.get(gpa, .{ .bytes = "foo" }); const struct_index = try ip.createStruct(gpa, .{
const bar_name = try ip.get(gpa, .{ .bytes = "bar" }); .fields = .{},
const field1 = Struct.Field{ .name = foo_name, .ty = i32_type };
const field2 = Struct.Field{ .name = bar_name, .ty = bool_type };
const struct_type = try ip.get(gpa, .{ .struct_type = .{
.fields = &.{ field1, field2 },
.namespace = .none, .namespace = .none,
.layout = .Auto, .layout = .Auto,
.backing_int_ty = .none, .backing_int_ty = .none,
} }); .status = .none,
});
const struct_type = try ip.get(gpa, .{ .struct_type = struct_index });
const struct_info = ip.getStruct(struct_index);
try struct_info.fields.put(gpa, "foo", .{ .ty = i32_type });
try struct_info.fields.put(gpa, "bar", .{ .ty = bool_type });
const one_value = try ip.get(gpa, .{ .int_i64_value = 1 }); const one_value = try ip.get(gpa, .{ .int_i64_value = 1 });
const true_value = try ip.get(gpa, .{ .simple = .bool_true }); const true_value = try ip.get(gpa, .{ .simple = .bool_true });
@ -3062,50 +3106,6 @@ test "struct value" {
try ip.testExpectFmtValue(aggregate_value, struct_type, ".{.foo = 1, .bar = true}"); try ip.testExpectFmtValue(aggregate_value, struct_type, ".{.foo = 1, .bar = true}");
} }
test "enum type" {
const gpa = std.testing.allocator;
var ip: InternPool = .{};
defer ip.deinit(gpa);
const zig_name = try ip.get(gpa, .{ .bytes = "zig" });
const cpp_name = try ip.get(gpa, .{ .bytes = "cpp" });
const empty_enum1 = try ip.get(gpa, .{ .enum_type = .{
.tag_type = .none,
.fields = &.{},
.namespace = .none,
.tag_type_infered = true,
} });
const empty_enum2 = try ip.get(gpa, .{ .enum_type = .{
.tag_type = .none,
.fields = &.{},
.namespace = .none,
.tag_type_infered = true,
} });
const field1 = Enum.Field{ .name = zig_name, .val = .none };
const field2 = Enum.Field{ .name = cpp_name, .val = .none };
const enum1 = try ip.get(gpa, .{ .enum_type = .{
.tag_type = .none,
.fields = &.{ field1, field2 },
.namespace = .none,
.tag_type_infered = true,
} });
const enum2 = try ip.get(gpa, .{ .enum_type = .{
.tag_type = .none,
.fields = &.{ field2, field1 },
.namespace = .none,
.tag_type_infered = true,
} });
try std.testing.expect(empty_enum1 == empty_enum2);
try std.testing.expect(empty_enum2 != enum1);
try std.testing.expect(enum1 != enum2);
}
test "function type" { test "function type" {
const gpa = std.testing.allocator; const gpa = std.testing.allocator;
@ -3152,40 +3152,6 @@ test "function type" {
try testExpectFmtType(ip, @"fn() align(4) callconv(.C) type", "fn() align(4) callconv(.C) type"); try testExpectFmtType(ip, @"fn() align(4) callconv(.C) type", "fn() align(4) callconv(.C) type");
} }
test "union type" {
const gpa = std.testing.allocator;
var ip: InternPool = .{};
defer ip.deinit(gpa);
const u32_type = try ip.get(gpa, .{ .int_type = .{ .signedness = .unsigned, .bits = 32 } });
const void_type = try ip.get(gpa, .{ .simple = .void });
const ok_name = try ip.get(gpa, .{ .bytes = "Ok" });
const err_name = try ip.get(gpa, .{ .bytes = "Err" });
var field1 = Union.Field{ .name = ok_name, .ty = u32_type, .alignment = 0 };
var field2 = Union.Field{ .name = err_name, .ty = void_type, .alignment = 0 };
const union_type1 = try ip.get(gpa, .{
.union_type = .{
.tag_type = .none,
.fields = &.{ field1, field2 },
.namespace = .none,
},
});
const union_type2 = try ip.get(gpa, .{
.union_type = .{
.tag_type = .none,
.fields = &.{ field2, field1 },
.namespace = .none,
},
});
try std.testing.expect(union_type1 != union_type2);
}
test "union value" { test "union value" {
const gpa = std.testing.allocator; const gpa = std.testing.allocator;
@ -3198,19 +3164,17 @@ test "union value" {
const int_value = try ip.get(gpa, .{ .int_u64_value = 1 }); const int_value = try ip.get(gpa, .{ .int_u64_value = 1 });
const f16_value = try ip.get(gpa, .{ .float_16_value = 0.25 }); const f16_value = try ip.get(gpa, .{ .float_16_value = 0.25 });
const int_name = try ip.get(gpa, .{ .bytes = "int" }); const union_index = try ip.createUnion(gpa, .{
const float_name = try ip.get(gpa, .{ .bytes = "float" }); .tag_type = .none,
.fields = .{},
var field1 = Union.Field{ .name = int_name, .ty = u32_type, .alignment = 0 }; .namespace = .none,
var field2 = Union.Field{ .name = float_name, .ty = f16_type, .alignment = 0 }; .layout = .Auto,
.status = .none,
const union_type = try ip.get(gpa, .{
.union_type = .{
.tag_type = .none,
.fields = &.{ field1, field2 },
.namespace = .none,
},
}); });
const union_type = try ip.get(gpa, .{ .union_type = union_index });
const union_info = ip.getUnion(union_index);
try union_info.fields.put(gpa, "int", .{ .ty = u32_type, .alignment = 0 });
try union_info.fields.put(gpa, "float", .{ .ty = f16_type, .alignment = 0 });
const union_value1 = try ip.get(gpa, .{ .union_value = .{ const union_value1 = try ip.get(gpa, .{ .union_value = .{
.field_index = 0, .field_index = 0,

View File

@ -24,7 +24,7 @@ pub fn dotCompletions(
.simple => |simple| switch (simple) { .simple => |simple| switch (simple) {
.type => { .type => {
const ty_key = ip.indexToKey(val); const ty_key = ip.indexToKey(val);
const namespace = ty_key.getNamespace(); const namespace = ty_key.getNamespace(ip.*);
if (namespace != .none) { if (namespace != .none) {
// TODO lookup in namespace // TODO lookup in namespace
} }
@ -40,11 +40,12 @@ pub fn dotCompletions(
} }
}, },
.union_type => {}, // TODO .union_type => {}, // TODO
.enum_type => |enum_info| { .enum_type => |enum_index| {
for (enum_info.fields) |field| { const enum_info = ip.getEnum(enum_index);
const field_name = ip.indexToKey(field.name).bytes; var field_it = enum_info.fields.iterator();
while (field_it.next()) |entry| {
try completions.append(arena, .{ try completions.append(arena, .{
.label = field_name, .label = entry.key_ptr.*,
.kind = .Constant, .kind = .Constant,
// include field.val? // include field.val?
}); });
@ -85,14 +86,15 @@ pub fn dotCompletions(
.detail = try std.fmt.allocPrint(arena, "usize ({d})", .{array_info.len}), // TODO how should this be displayed .detail = try std.fmt.allocPrint(arena, "usize ({d})", .{array_info.len}), // TODO how should this be displayed
}); });
}, },
.struct_type => |struct_info| { .struct_type => |struct_index| {
for (struct_info.fields) |field| { const struct_info = ip.getStruct(struct_index);
const field_name = ip.indexToKey(field.name).bytes; var field_it = struct_info.fields.iterator();
while (field_it.next()) |entry| {
try completions.append(arena, types.CompletionItem{ try completions.append(arena, types.CompletionItem{
.label = field_name, .label = entry.key_ptr.*,
.kind = .Field, .kind = .Field,
// TODO include alignment and comptime // TODO include alignment and comptime
.detail = try std.fmt.allocPrint(arena, "{}", .{field.ty.fmtType(ip.*)}), .detail = try std.fmt.allocPrint(arena, "{}", .{entry.value_ptr.ty.fmtType(ip.*)}),
}); });
} }
}, },
@ -103,26 +105,28 @@ pub fn dotCompletions(
.detail = try std.fmt.allocPrint(arena, "{}", .{optional_info.payload_type.fmtType(ip.*)}), .detail = try std.fmt.allocPrint(arena, "{}", .{optional_info.payload_type.fmtType(ip.*)}),
}); });
}, },
.enum_type => |enum_info| { .enum_type => |enum_index| {
for (enum_info.fields) |field| { const enum_info = ip.getEnum(enum_index);
const field_name = ip.indexToKey(field.name).bytes; for (enum_info.fields.keys()) |field_name, i| {
const field_val = enum_info.values.keys()[i];
try completions.append(arena, .{ try completions.append(arena, .{
.label = field_name, .label = field_name,
.kind = .Field, .kind = .Field,
.detail = try std.fmt.allocPrint(arena, "{}", .{field.val.fmtValue(enum_info.tag_type, ip.*)}), .detail = try std.fmt.allocPrint(arena, "{}", .{field_val.fmtValue(enum_info.tag_type, ip.*)}),
}); });
} }
}, },
.union_type => |union_info| { .union_type => |union_index| {
for (union_info.fields) |field| { const union_info = ip.getUnion(union_index);
const field_name = ip.indexToKey(field.name).bytes; var field_it = union_info.fields.iterator();
while (field_it.next()) |entry| {
try completions.append(arena, .{ try completions.append(arena, .{
.label = field_name, .label = entry.key_ptr.*,
.kind = .Field, .kind = .Field,
.detail = if (field.alignment != 0) .detail = if (entry.value_ptr.alignment != 0)
try std.fmt.allocPrint(arena, "align({d}) {}", .{ field.alignment, field.ty.fmtType(ip.*) }) try std.fmt.allocPrint(arena, "align({d}) {}", .{ entry.value_ptr.alignment, entry.value_ptr.ty.fmtType(ip.*) })
else else
try std.fmt.allocPrint(arena, "{}", .{field.ty.fmtType(ip.*)}), try std.fmt.allocPrint(arena, "{}", .{entry.value_ptr.ty.fmtType(ip.*)}),
}); });
} }
}, },

View File

@ -772,7 +772,7 @@ pub fn resolveTypeOfNodeInternal(store: *DocumentStore, arena: *std.heap.ArenaAl
}; };
var interpreter: *ComptimeInterpreter = handle.interpreter.?; var interpreter: *ComptimeInterpreter = handle.interpreter.?;
const root_namespace = @intToEnum(ComptimeInterpreter.NamespaceIndex, 0); const root_namespace = @intToEnum(ComptimeInterpreter.Namespace.Index, 0);
// TODO: Start from current/nearest-current scope // TODO: Start from current/nearest-current scope
const result = interpreter.interpret(node, root_namespace, .{}) catch |err| { const result = interpreter.interpret(node, root_namespace, .{}) catch |err| {

View File

@ -169,16 +169,15 @@ test "ComptimeInterpreter - call return struct" {
try std.testing.expect(result.ty == .simple); try std.testing.expect(result.ty == .simple);
try std.testing.expect(result.ty.simple == .type); try std.testing.expect(result.ty.simple == .type);
const struct_info = result.val.?.struct_type; const struct_info = context.interpreter.ip.getStruct(result.val.?.struct_type);
try std.testing.expectEqual(Index.none, struct_info.backing_int_ty); try std.testing.expectEqual(Index.none, struct_info.backing_int_ty);
try std.testing.expectEqual(std.builtin.Type.ContainerLayout.Auto, struct_info.layout); try std.testing.expectEqual(std.builtin.Type.ContainerLayout.Auto, struct_info.layout);
const field_name = context.interpreter.ip.indexToKey(struct_info.fields[0].name).bytes;
const bool_type = try context.interpreter.ip.get(allocator, .{ .simple = .bool }); const bool_type = try context.interpreter.ip.get(allocator, .{ .simple = .bool });
try std.testing.expectEqual(@as(usize, 1), struct_info.fields.len); try std.testing.expectEqual(@as(usize, 1), struct_info.fields.count());
try std.testing.expectEqualStrings("slay", field_name); try std.testing.expectEqualStrings("slay", struct_info.fields.keys()[0]);
try std.testing.expect(struct_info.fields[0].ty == bool_type); try std.testing.expect(struct_info.fields.values()[0].ty == bool_type);
} }
test "ComptimeInterpreter - call comptime argument" { test "ComptimeInterpreter - call comptime argument" {
@ -284,7 +283,7 @@ const Context = struct {
}; };
} }
const namespace = @intToEnum(ComptimeInterpreter.NamespaceIndex, 0); // root namespace const namespace = @intToEnum(ComptimeInterpreter.Namespace.Index, 0); // root namespace
const result = (try self.interpreter.call(namespace, func_node, args, .{})).result; const result = (try self.interpreter.call(namespace, func_node, args, .{})).result;
try std.testing.expect(result == .value); try std.testing.expect(result == .value);
@ -297,7 +296,7 @@ const Context = struct {
} }
pub fn interpret(self: *Context, node: Ast.Node.Index) !KV { pub fn interpret(self: *Context, node: Ast.Node.Index) !KV {
const namespace = @intToEnum(ComptimeInterpreter.NamespaceIndex, 0); // root namespace const namespace = @intToEnum(ComptimeInterpreter.Namespace.Index, 0); // root namespace
const result = try (try self.interpreter.interpret(node, namespace, .{})).getValue(); const result = try (try self.interpreter.interpret(node, namespace, .{})).getValue();
try std.testing.expect(result.ty != .none); try std.testing.expect(result.ty != .none);