aboutsummaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authorAndrew Kelley <andrew@ziglang.org>2022-12-19 15:47:31 -0500
committerGitHub <noreply@github.com>2022-12-19 15:47:31 -0500
commit0fb53bd245e258f69654119e5a1913d6d42dc181 (patch)
treeb2fe08ac32aa43c42cf05d29985c326d20b57994 /lib
parent3542dbf0ea5bc1ddb1c5e1c856745dc07e6c0a18 (diff)
parent0768115b01f01ab1c75da3e42ffcdc99078eaad2 (diff)
downloadzig-0fb53bd245e258f69654119e5a1913d6d42dc181.tar.gz
zig-0fb53bd245e258f69654119e5a1913d6d42dc181.zip
Merge pull request #14000 from jacobly0/zero-bit-fields
codegen: fix taking the address of a field in a zero-bit struct
Diffstat (limited to 'lib')
-rw-r--r--lib/std/mem.zig17
1 files changed, 12 insertions, 5 deletions
diff --git a/lib/std/mem.zig b/lib/std/mem.zig
index 17281390a8..a392335a70 100644
--- a/lib/std/mem.zig
+++ b/lib/std/mem.zig
@@ -3291,7 +3291,7 @@ pub fn nativeToBig(comptime T: type, x: T) T {
/// - The delta required to align the pointer is not a multiple of the pointee's
/// type.
pub fn alignPointerOffset(ptr: anytype, align_to: usize) ?usize {
- assert(align_to != 0 and @popCount(align_to) == 1);
+ assert(isValidAlign(align_to));
const T = @TypeOf(ptr);
const info = @typeInfo(T);
@@ -3751,6 +3751,7 @@ pub fn alignForwardLog2(addr: usize, log2_alignment: u8) usize {
/// The alignment must be a power of 2 and greater than 0.
/// Asserts that rounding up the address does not cause integer overflow.
pub fn alignForwardGeneric(comptime T: type, addr: T, alignment: T) T {
+ assert(isValidAlignGeneric(T, alignment));
return alignBackwardGeneric(T, addr + (alignment - 1), alignment);
}
@@ -3846,7 +3847,7 @@ test "alignForward" {
/// Round an address down to the previous (or current) aligned address.
/// Unlike `alignBackward`, `alignment` can be any positive number, not just a power of 2.
pub fn alignBackwardAnyAlign(i: usize, alignment: usize) usize {
- if (@popCount(alignment) == 1)
+ if (isValidAlign(alignment))
return alignBackward(i, alignment);
assert(alignment != 0);
return i - @mod(i, alignment);
@@ -3861,7 +3862,7 @@ pub fn alignBackward(addr: usize, alignment: usize) usize {
/// Round an address down to the previous (or current) aligned address.
/// The alignment must be a power of 2 and greater than 0.
pub fn alignBackwardGeneric(comptime T: type, addr: T, alignment: T) T {
- assert(@popCount(alignment) == 1);
+ assert(isValidAlignGeneric(T, alignment));
// 000010000 // example alignment
// 000001111 // subtract 1
// 111110000 // binary not
@@ -3871,11 +3872,17 @@ pub fn alignBackwardGeneric(comptime T: type, addr: T, alignment: T) T {
/// Returns whether `alignment` is a valid alignment, meaning it is
/// a positive power of 2.
pub fn isValidAlign(alignment: usize) bool {
- return @popCount(alignment) == 1;
+ return isValidAlignGeneric(usize, alignment);
+}
+
+/// Returns whether `alignment` is a valid alignment, meaning it is
+/// a positive power of 2.
+pub fn isValidAlignGeneric(comptime T: type, alignment: T) bool {
+ return alignment > 0 and std.math.isPowerOfTwo(alignment);
}
pub fn isAlignedAnyAlign(i: usize, alignment: usize) bool {
- if (@popCount(alignment) == 1)
+ if (isValidAlign(alignment))
return isAligned(i, alignment);
assert(alignment != 0);
return 0 == @mod(i, alignment);