aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorMatthew Lugg <mlugg@mlugg.co.uk>2025-02-01 15:47:06 +0000
committerGitHub <noreply@github.com>2025-02-01 15:47:06 +0000
commitc225b780e3944014300555c3b45eaccd7bc7c8ae (patch)
tree4c36c2b86776a5c0116e7212bc31dd1f4bfe3f78 /src
parent58c00a829e8acf438b91e9f7e1729ce79be722fa (diff)
parent149031204c0c804a66571a751b259da95886640d (diff)
downloadzig-c225b780e3944014300555c3b45eaccd7bc7c8ae.tar.gz
zig-c225b780e3944014300555c3b45eaccd7bc7c8ae.zip
Merge pull request #22708 from mlugg/memcpy-alias-zero-bit
Sema: skip aliasing check and runtime operation for `@memcpy` of zero-bit type
Diffstat (limited to 'src')
-rw-r--r--src/Sema.zig24
-rw-r--r--src/Type.zig7
2 files changed, 29 insertions, 2 deletions
diff --git a/src/Sema.zig b/src/Sema.zig
index bc9d5ea55f..93b367b9d8 100644
--- a/src/Sema.zig
+++ b/src/Sema.zig
@@ -24581,8 +24581,12 @@ fn zirSplat(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.I
const len = try sema.usizeCast(block, src, dest_ty.arrayLen(zcu));
- // `len == 0` because `[0:s]T` always has a comptime-known splat.
- if (!dest_ty.hasRuntimeBits(zcu) or len == 0) {
+ if (try sema.typeHasOnePossibleValue(dest_ty)) |val| {
+ return Air.internedToRef(val.toIntern());
+ }
+
+ // We also need this case because `[0:s]T` is not OPV.
+ if (len == 0) {
const empty_aggregate = try pt.intern(.{ .aggregate = .{
.ty = dest_ty.toIntern(),
.storage = .{ .elems = &.{} },
@@ -25924,6 +25928,22 @@ fn zirMemcpy(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!void
}
}
+ zero_bit: {
+ const src_comptime = try src_elem_ty.comptimeOnlySema(pt);
+ const dest_comptime = try dest_elem_ty.comptimeOnlySema(pt);
+ assert(src_comptime == dest_comptime); // IMC
+ if (src_comptime) break :zero_bit;
+
+ const src_has_bits = try src_elem_ty.hasRuntimeBitsIgnoreComptimeSema(pt);
+ const dest_has_bits = try dest_elem_ty.hasRuntimeBitsIgnoreComptimeSema(pt);
+ assert(src_has_bits == dest_has_bits); // IMC
+ if (src_has_bits) break :zero_bit;
+
+ // The element type is zero-bit. We've done all validation (aside from the aliasing check,
+ // which we must skip) so we're done.
+ return;
+ }
+
const runtime_src = rs: {
const dest_ptr_val = try sema.resolveDefinedValue(block, dest_src, dest_ptr) orelse break :rs dest_src;
const src_ptr_val = try sema.resolveDefinedValue(block, src_src, src_ptr) orelse break :rs src_src;
diff --git a/src/Type.zig b/src/Type.zig
index e6fc9c7d6a..403caad668 100644
--- a/src/Type.zig
+++ b/src/Type.zig
@@ -452,6 +452,13 @@ pub fn hasRuntimeBitsIgnoreComptime(ty: Type, zcu: *const Zcu) bool {
return hasRuntimeBitsInner(ty, true, .eager, zcu, {}) catch unreachable;
}
+pub fn hasRuntimeBitsIgnoreComptimeSema(ty: Type, pt: Zcu.PerThread) SemaError!bool {
+ return hasRuntimeBitsInner(ty, true, .sema, pt.zcu, pt.tid) catch |err| switch (err) {
+ error.NeedLazy => unreachable, // this would require a resolve strat of lazy
+ else => |e| return e,
+ };
+}
+
/// true if and only if the type takes up space in memory at runtime.
/// There are two reasons a type will return false:
/// * the type is a comptime-only type. For example, the type `type` itself.