mirror of
https://codeberg.org/ziglang/zig.git
synced 2026-04-26 13:01:34 +03:00
be9b42d707
This was already possible in practice by just wrapping a packed union into a packed struct. Now it's also possible without doing that.
222 lines
7.1 KiB
Zig
222 lines
7.1 KiB
Zig
const std = @import("std");
|
|
const builtin = @import("builtin");
|
|
const assert = std.debug.assert;
|
|
const expect = std.testing.expect;
|
|
const expectEqual = std.testing.expectEqual;
|
|
|
|
test "flags in packed union" {
|
|
if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_sparc64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_spirv) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_riscv64) return error.SkipZigTest; // TODO
|
|
|
|
try testFlagsInPackedUnion();
|
|
try comptime testFlagsInPackedUnion();
|
|
}
|
|
|
|
fn testFlagsInPackedUnion() !void {
|
|
const FlagBits = packed struct(u8) {
|
|
enable_1: bool = false,
|
|
enable_2: bool = false,
|
|
enable_3: bool = false,
|
|
enable_4: bool = false,
|
|
other_flags: packed union {
|
|
flags: packed struct(u4) {
|
|
enable_1: bool = true,
|
|
enable_2: bool = false,
|
|
enable_3: bool = false,
|
|
enable_4: bool = false,
|
|
},
|
|
bits: u4,
|
|
} = .{ .flags = .{} },
|
|
};
|
|
var test_bits: FlagBits = .{};
|
|
|
|
try expectEqual(false, test_bits.enable_1);
|
|
try expectEqual(true, test_bits.other_flags.flags.enable_1);
|
|
|
|
test_bits.enable_1 = true;
|
|
|
|
try expectEqual(true, test_bits.enable_1);
|
|
try expectEqual(true, test_bits.other_flags.flags.enable_1);
|
|
|
|
test_bits.other_flags.flags.enable_1 = false;
|
|
|
|
try expectEqual(true, test_bits.enable_1);
|
|
try expectEqual(false, test_bits.other_flags.flags.enable_1);
|
|
}
|
|
|
|
test "flags in packed union at offset" {
|
|
if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_sparc64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_spirv) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_riscv64) return error.SkipZigTest; // TODO
|
|
|
|
try testFlagsInPackedUnionAtOffset();
|
|
try comptime testFlagsInPackedUnionAtOffset();
|
|
}
|
|
|
|
fn testFlagsInPackedUnionAtOffset() !void {
|
|
const FlagBits = packed union {
|
|
base_flags: packed struct(u12) {
|
|
a: packed union {
|
|
flags: packed struct(u4) {
|
|
enable_1: bool = true,
|
|
enable_2: bool = false,
|
|
enable_3: bool = false,
|
|
enable_4: bool = false,
|
|
},
|
|
bits: u4,
|
|
},
|
|
pad: u8 = 0,
|
|
},
|
|
adv_flags: packed struct(u12) {
|
|
pad: u8 = 0,
|
|
adv: packed union {
|
|
flags: packed struct(u4) {
|
|
enable_1: bool = true,
|
|
enable_2: bool = false,
|
|
enable_3: bool = false,
|
|
enable_4: bool = false,
|
|
},
|
|
bits: u4,
|
|
},
|
|
},
|
|
};
|
|
var test_bits: FlagBits = .{ .adv_flags = .{ .adv = .{ .flags = .{} } } };
|
|
|
|
try expectEqual(@as(u8, 0), test_bits.adv_flags.pad);
|
|
try expectEqual(true, test_bits.adv_flags.adv.flags.enable_1);
|
|
try expectEqual(false, test_bits.adv_flags.adv.flags.enable_2);
|
|
|
|
test_bits.adv_flags.adv.flags.enable_1 = false;
|
|
test_bits.adv_flags.adv.flags.enable_2 = true;
|
|
try expectEqual(@as(u8, 0), test_bits.adv_flags.pad);
|
|
try expectEqual(false, test_bits.adv_flags.adv.flags.enable_1);
|
|
try expectEqual(true, test_bits.adv_flags.adv.flags.enable_2);
|
|
|
|
test_bits.adv_flags.adv.bits = 12;
|
|
try expectEqual(@as(u8, 0), test_bits.adv_flags.pad);
|
|
try expectEqual(false, test_bits.adv_flags.adv.flags.enable_1);
|
|
try expectEqual(false, test_bits.adv_flags.adv.flags.enable_2);
|
|
}
|
|
|
|
// Originally reported at https://github.com/ziglang/zig/issues/16581
|
|
test "packed union in packed struct" {
|
|
if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_riscv64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_spirv) return error.SkipZigTest;
|
|
|
|
try testPackedUnionInPackedStruct();
|
|
try comptime testPackedUnionInPackedStruct();
|
|
}
|
|
|
|
fn testPackedUnionInPackedStruct() !void {
|
|
const ReadRequest = packed struct { key: i32 };
|
|
const RequestType = enum(u1) {
|
|
read,
|
|
insert,
|
|
};
|
|
const RequestUnion = packed union {
|
|
read: ReadRequest,
|
|
};
|
|
|
|
const Request = packed struct {
|
|
active_type: RequestType,
|
|
request: RequestUnion,
|
|
const Self = @This();
|
|
|
|
fn init(read: ReadRequest) Self {
|
|
return .{
|
|
.active_type = .read,
|
|
.request = RequestUnion{ .read = read },
|
|
};
|
|
}
|
|
};
|
|
|
|
try std.testing.expectEqual(RequestType.read, Request.init(.{ .key = 3 }).active_type);
|
|
}
|
|
|
|
test "packed union initialized with a runtime value" {
|
|
if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; // TODO
|
|
if (builtin.zig_backend == .stage2_sparc64) return error.SkipZigTest; // TODO
|
|
if (builtin.zig_backend == .stage2_riscv64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_spirv) return error.SkipZigTest;
|
|
|
|
const Fields = packed struct {
|
|
timestamp: u50,
|
|
random_bits: u13,
|
|
};
|
|
const ID = packed union {
|
|
value: u63,
|
|
fields: Fields,
|
|
|
|
fn getValue() i64 {
|
|
return 1341;
|
|
}
|
|
};
|
|
|
|
const timestamp: i64 = ID.getValue();
|
|
const id = ID{ .fields = Fields{
|
|
.timestamp = @as(u50, @intCast(timestamp)),
|
|
.random_bits = 420,
|
|
} };
|
|
try std.testing.expect((ID{ .value = id.value }).fields.timestamp == timestamp);
|
|
}
|
|
|
|
test "assigning to non-active field at comptime" {
|
|
comptime {
|
|
const FlagBits = packed union {
|
|
flags: packed struct {},
|
|
bits: packed struct {},
|
|
};
|
|
|
|
var test_bits: FlagBits = .{ .flags = .{} };
|
|
test_bits.bits = .{};
|
|
}
|
|
}
|
|
|
|
test "packed union with explicit backing integer" {
|
|
if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_riscv64) return error.SkipZigTest;
|
|
if (builtin.zig_backend == .stage2_spirv) return error.SkipZigTest;
|
|
|
|
const U = packed union(i32) {
|
|
raw: i32,
|
|
unsigned_halves: packed struct { low: u16, high: u16 },
|
|
|
|
fn check(val: @This()) !void {
|
|
try expect(@as(i32, @bitCast(val)) == -2);
|
|
try expect(@as(u32, @bitCast(val)) == 0xFFFFFFFE);
|
|
try expect(val.raw == -2);
|
|
try expect(val.unsigned_halves.low == 0xFFFE);
|
|
try expect(val.unsigned_halves.high == 0xFFFF);
|
|
}
|
|
};
|
|
try U.check(.{ .raw = -2 });
|
|
try comptime U.check(.{ .raw = -2 });
|
|
}
|
|
|
|
test "packed union equality" {
|
|
const Foo = packed union {
|
|
a: u4,
|
|
b: i4,
|
|
};
|
|
|
|
const S = struct {
|
|
fn doTest(x: Foo, y: Foo) !void {
|
|
try expect(x == y);
|
|
try expect(!(x != y));
|
|
}
|
|
};
|
|
|
|
const x: Foo = .{ .a = 3 };
|
|
const y: Foo = .{ .b = 3 };
|
|
|
|
try S.doTest(x, y);
|
|
comptime try S.doTest(x, y);
|
|
}
|