-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
proposal: inline structs #18648
Comments
"but what if you did this?" const S = struct {
a_c: u8,
a_d: u32,
b_e: u8,
b_f: u32,
};
I would want adding a field to |
"but didn't you say you can't create temporaries/immediates?" yes and per-Result Location Semantics and observed in #12064 we see that the code above is semantically equivalent to the following which is totally valid per this proposal.
|
as for the driving use case, this is the first one i thought of https://git.sr.ht/~nektro/magnolia-desktop/tree/1ddabe69/item/src/ttf.zig#L83 |
FYI, I think something like this could be done through comptime meta function that generates the namespaced struct for you. |
While this makes sense, it would make the proposal quite tricky to implement, since In general, I don't like this proposal; I think it's solving a rather esoteric problem in exchange for what's actually a pretty fundamental change to the type system. This introduces a whole new class of type into Zig: types which, in a very real sense, aren't really types! They represent data, but you can't actually store an instance of them, and queries which are normally fine across the board like If optimizing the memory footprint is of concern, then you're presumably storing a large array of these structs. Would a kind of "recursive |
I hadn't thought to try that before for this case, lemme give it a shot and report back.
I think they share a lot of properties with comptime-only types but if it arrives that its too complicated for the benefit, that's fair. |
before and after:
wrote the code to do this at comptime but I dont think i'll be continuing with this after it not being that much of a meaningful difference |
posting the code in case anyone finds this later and really wants it fn InlineStruct(comptime T: type) type {
const info = @typeInfo(T).Struct;
var new_fields: []const std.builtin.Type.StructField = &.{};
for (info.fields) |old_field| {
switch (@typeInfo(old_field.type)) {
.Struct => {
const other_info = @typeInfo(old_field.type).Struct;
for (other_info.fields) |other_field| {
new_fields = new_fields ++ &[_]std.builtin.Type.StructField{.{
.name = std.fmt.comptimePrint("{s}_{s}", .{ old_field.name, other_field.name }),
.type = other_field.type,
.default_value = other_field.default_value,
.is_comptime = other_field.is_comptime,
.alignment = other_field.alignment,
}};
}
},
else => {
new_fields = new_fields ++ &[_]std.builtin.Type.StructField{old_field};
},
}
}
return @Type(@unionInit(std.builtin.Type, "Struct", .{
.layout = .Auto,
.backing_integer = null,
.fields = new_fields,
.decls = &.{},
.is_tuple = false,
}));
}
fn InlineStructMixin(comptime T: type, comptime inline_name: string, comptime I: type) type {
return struct {
pub inline fn inlineSet(this: *T, comptime field_name: std.meta.FieldEnum(I), value: std.meta.FieldType(I, field_name)) void {
const F = std.meta.FieldType(I, field_name);
switch (@typeInfo(F)) {
.Struct => {
inline for (@typeInfo(F).Struct.fields) |field| {
@field(@field(this, inline_name), std.fmt.comptimePrint("{s}_{s}", .{ @tagName(field_name), field.name })) = @field(value, field.name);
}
},
else => {
@field(@field(this, inline_name), @tagName(field_name)) = value;
},
}
}
};
} |
Since Zig's earliest days it has said that structs declared with an auto layout are subject to optimizations such as field reordering in order to enable the most efficient padding for example. As of #14336 this optimization is now in place in ways towards #168. However consider the following code:
and compare that to:
evaluating
@sizeOf
on each of these yields16
and12
respectively. This is due to the compiler not being able to void the assumption thats.a
may be a valid immediate. or the user doing&s.a
. this heuristic could be potentially performed during semantic analysis would be quite expensive and have many other ramifications. however, the writer may know ahead of time that they are only using the struct for semantic namespacing of the fields and are okay with telling the compiler ahead of time that the types won't be used as an immediate and thus the inner fields may be reodered as if the higher struct wasn't there. this is whatinline struct
provides.assuming we edit
S
to useinline struct
for thea
andb
fields,s.a
is compile error&s.a
is a compile errors.a.c
is okays.a.c = foo
is okay@sizeOf(inline struct {..})
is compile error@alignOf(inline struct {..})
is compile errorA
orB
would still show up in@typeInfo(S)
with a layout of.Inline
but cannot be reified through@Type
@offsetOf(S, "a")
is compile error@offsetOf(S, "a.c")
working or not is up to the discretion of the implementeracceptance and subsequent implementation of this proposal would make
assert(@sizeOf(S) == @sizeOf(T));
pass.The text was updated successfully, but these errors were encountered: