mirror of
https://github.com/ziglang/zig.git
synced 2025-12-14 18:23:12 +00:00
This was previously implemented by analyzing the AIR prior to the ZIR `make_ptr_const` instruction. This solution was highly delicate, and in particular broke down whenever there was a second `alloc` between the `store` and `alloc` instructions, which is especially common in destructure statements. Sema now uses a different strategy to detect whether a `const` is comptime-known. When the `alloc` is created, Sema begins tracking all pointers and stores which refer to that allocation in temporary local state. If any store is not comptime-known or has a higher runtime index than the allocation, the allocation is marked as being runtime-known. When we reach the `make_ptr_const` instruction, if the allocation is not marked as runtime-known, it must be comptime-known. Sema will use the set of `store` instructions to re-initialize the value in comptime memory. We optimize for the common case of a single `store` instruction by not creating a comptime alloc in this case, instead directly plucking the result value from the instruction. Resolves: #16083
141 lines
3.6 KiB
Zig
141 lines
3.6 KiB
Zig
const std = @import("std");
|
|
const assert = std.debug.assert;
|
|
const expect = std.testing.expect;
|
|
|
|
test "simple destructure" {
|
|
const S = struct {
|
|
fn doTheTest() !void {
|
|
var x: u32 = undefined;
|
|
x, const y, var z: u64 = .{ 1, @as(u16, 2), 3 };
|
|
|
|
comptime assert(@TypeOf(y) == u16);
|
|
|
|
try expect(x == 1);
|
|
try expect(y == 2);
|
|
try expect(z == 3);
|
|
}
|
|
};
|
|
|
|
try S.doTheTest();
|
|
try comptime S.doTheTest();
|
|
}
|
|
|
|
test "destructure with comptime syntax" {
|
|
const S = struct {
|
|
fn doTheTest() void {
|
|
comptime var x: f32 = undefined;
|
|
comptime x, const y, var z = .{ 0.5, 123, 456 }; // z is a comptime var
|
|
|
|
comptime assert(@TypeOf(y) == comptime_int);
|
|
comptime assert(@TypeOf(z) == comptime_int);
|
|
comptime assert(x == 0.5);
|
|
comptime assert(y == 123);
|
|
comptime assert(z == 456);
|
|
}
|
|
};
|
|
|
|
S.doTheTest();
|
|
comptime S.doTheTest();
|
|
}
|
|
|
|
test "destructure from labeled block" {
|
|
const S = struct {
|
|
fn doTheTest(rt_true: bool) !void {
|
|
const x: u32, const y: u8, const z: i64 = blk: {
|
|
if (rt_true) break :blk .{ 1, 2, 3 };
|
|
break :blk .{ 4, 5, 6 };
|
|
};
|
|
|
|
try expect(x == 1);
|
|
try expect(y == 2);
|
|
try expect(z == 3);
|
|
}
|
|
};
|
|
|
|
try S.doTheTest(true);
|
|
try comptime S.doTheTest(true);
|
|
}
|
|
|
|
test "destructure tuple value" {
|
|
const tup: struct { f32, u32, i64 } = .{ 10.0, 20, 30 };
|
|
const x, const y, const z = tup;
|
|
|
|
comptime assert(@TypeOf(x) == f32);
|
|
comptime assert(@TypeOf(y) == u32);
|
|
comptime assert(@TypeOf(z) == i64);
|
|
|
|
try expect(x == 10.0);
|
|
try expect(y == 20);
|
|
try expect(z == 30);
|
|
}
|
|
|
|
test "destructure array value" {
|
|
const arr: [3]u32 = .{ 10, 20, 30 };
|
|
const x, const y, const z = arr;
|
|
|
|
comptime assert(@TypeOf(x) == u32);
|
|
comptime assert(@TypeOf(y) == u32);
|
|
comptime assert(@TypeOf(z) == u32);
|
|
|
|
try expect(x == 10);
|
|
try expect(y == 20);
|
|
try expect(z == 30);
|
|
}
|
|
|
|
test "destructure from struct init with named tuple fields" {
|
|
const Tuple = struct { u8, u16, u32 };
|
|
const x, const y, const z = Tuple{
|
|
.@"0" = 100,
|
|
.@"1" = 200,
|
|
.@"2" = 300,
|
|
};
|
|
|
|
comptime assert(@TypeOf(x) == u8);
|
|
comptime assert(@TypeOf(y) == u16);
|
|
comptime assert(@TypeOf(z) == u32);
|
|
|
|
try expect(x == 100);
|
|
try expect(y == 200);
|
|
try expect(z == 300);
|
|
}
|
|
|
|
test "destructure of comptime-known tuple is comptime-known" {
|
|
const x, const y = .{ 1, 2 };
|
|
|
|
comptime assert(@TypeOf(x) == comptime_int);
|
|
comptime assert(x == 1);
|
|
|
|
comptime assert(@TypeOf(y) == comptime_int);
|
|
comptime assert(y == 2);
|
|
}
|
|
|
|
test "destructure of comptime-known tuple where some destinations are runtime-known is comptime-known" {
|
|
var z: u32 = undefined;
|
|
var x: u8, const y, z = .{ 1, 2, 3 };
|
|
|
|
comptime assert(@TypeOf(y) == comptime_int);
|
|
comptime assert(y == 2);
|
|
|
|
try expect(x == 1);
|
|
try expect(z == 3);
|
|
}
|
|
|
|
test "destructure of tuple with comptime fields results in some comptime-known values" {
|
|
var runtime: u32 = 42;
|
|
const a, const b, const c, const d = .{ 123, runtime, 456, runtime };
|
|
|
|
// a, c are comptime-known
|
|
// b, d are runtime-known
|
|
|
|
comptime assert(@TypeOf(a) == comptime_int);
|
|
comptime assert(@TypeOf(b) == u32);
|
|
comptime assert(@TypeOf(c) == comptime_int);
|
|
comptime assert(@TypeOf(d) == u32);
|
|
|
|
comptime assert(a == 123);
|
|
comptime assert(c == 456);
|
|
|
|
try expect(b == 42);
|
|
try expect(d == 42);
|
|
}
|