aboutsummaryrefslogtreecommitdiff
path: root/src/codegen/c.zig
diff options
context:
space:
mode:
authormlugg <mlugg@mlugg.co.uk>2025-05-26 05:07:13 +0100
committermlugg <mlugg@mlugg.co.uk>2025-06-01 08:24:01 +0100
commitadd2976a9ba76ec661ae5668eb2a8dca2ccfad42 (patch)
tree54ea8377660395007c1ac5188863f5cbe3a3310e /src/codegen/c.zig
parentb48d6ff619de424e664cf11b43e2a03fecbca6ce (diff)
downloadzig-add2976a9ba76ec661ae5668eb2a8dca2ccfad42.tar.gz
zig-add2976a9ba76ec661ae5668eb2a8dca2ccfad42.zip
compiler: implement better shuffle AIR
Runtime `@shuffle` has two cases which backends generally want to handle differently for efficiency: * One runtime vector operand; some result elements may be comptime-known * Two runtime vector operands; some result elements may be undefined The latter case happens if both vectors given to `@shuffle` are runtime-known and they are both used (i.e. the mask refers to them). Otherwise, if the result is not entirely comptime-known, we are in the former case. `Sema` now diffentiates these two cases in the AIR so that backends can easily handle them however they want to. Note that this *doesn't* really involve Sema doing any more work than it would otherwise need to, so there's not really a negative here! Most existing backends have their lowerings for `@shuffle` migrated in this commit. The LLVM backend uses new lowerings suggested by Jacob as ones which it will handle effectively. The x86_64 backend has not yet been migrated; for now there's a panic in there. Jacob will implement that before this is merged anywhere.
Diffstat (limited to 'src/codegen/c.zig')
-rw-r--r--src/codegen/c.zig74
1 files changed, 57 insertions, 17 deletions
diff --git a/src/codegen/c.zig b/src/codegen/c.zig
index 8d947ce56a..c68abc06ce 100644
--- a/src/codegen/c.zig
+++ b/src/codegen/c.zig
@@ -3374,7 +3374,8 @@ fn genBodyInner(f: *Function, body: []const Air.Inst.Index) error{ AnalysisFail,
.error_name => try airErrorName(f, inst),
.splat => try airSplat(f, inst),
.select => try airSelect(f, inst),
- .shuffle => try airShuffle(f, inst),
+ .shuffle_one => try airShuffleOne(f, inst),
+ .shuffle_two => try airShuffleTwo(f, inst),
.reduce => try airReduce(f, inst),
.aggregate_init => try airAggregateInit(f, inst),
.union_init => try airUnionInit(f, inst),
@@ -7163,34 +7164,73 @@ fn airSelect(f: *Function, inst: Air.Inst.Index) !CValue {
return local;
}
-fn airShuffle(f: *Function, inst: Air.Inst.Index) !CValue {
+fn airShuffleOne(f: *Function, inst: Air.Inst.Index) !CValue {
const pt = f.object.dg.pt;
const zcu = pt.zcu;
- const ty_pl = f.air.instructions.items(.data)[@intFromEnum(inst)].ty_pl;
- const extra = f.air.extraData(Air.Shuffle, ty_pl.payload).data;
-
- const mask = Value.fromInterned(extra.mask);
- const lhs = try f.resolveInst(extra.a);
- const rhs = try f.resolveInst(extra.b);
- const inst_ty = f.typeOfIndex(inst);
+ const unwrapped = f.air.unwrapShuffleOne(zcu, inst);
+ const mask = unwrapped.mask;
+ const operand = try f.resolveInst(unwrapped.operand);
+ const inst_ty = unwrapped.result_ty;
const writer = f.object.writer();
const local = try f.allocLocal(inst, inst_ty);
- try reap(f, inst, &.{ extra.a, extra.b }); // local cannot alias operands
- for (0..extra.mask_len) |index| {
+ try reap(f, inst, &.{unwrapped.operand}); // local cannot alias operand
+ for (mask, 0..) |mask_elem, out_idx| {
try f.writeCValue(writer, local, .Other);
try writer.writeByte('[');
- try f.object.dg.renderValue(writer, try pt.intValue(.usize, index), .Other);
+ try f.object.dg.renderValue(writer, try pt.intValue(.usize, out_idx), .Other);
try writer.writeAll("] = ");
+ switch (mask_elem.unwrap()) {
+ .elem => |src_idx| {
+ try f.writeCValue(writer, operand, .Other);
+ try writer.writeByte('[');
+ try f.object.dg.renderValue(writer, try pt.intValue(.usize, src_idx), .Other);
+ try writer.writeByte(']');
+ },
+ .value => |val| try f.object.dg.renderValue(writer, .fromInterned(val), .Other),
+ }
+ try writer.writeAll(";\n");
+ }
- const mask_elem = (try mask.elemValue(pt, index)).toSignedInt(zcu);
- const src_val = try pt.intValue(.usize, @as(u64, @intCast(mask_elem ^ mask_elem >> 63)));
+ return local;
+}
- try f.writeCValue(writer, if (mask_elem >= 0) lhs else rhs, .Other);
+fn airShuffleTwo(f: *Function, inst: Air.Inst.Index) !CValue {
+ const pt = f.object.dg.pt;
+ const zcu = pt.zcu;
+
+ const unwrapped = f.air.unwrapShuffleTwo(zcu, inst);
+ const mask = unwrapped.mask;
+ const operand_a = try f.resolveInst(unwrapped.operand_a);
+ const operand_b = try f.resolveInst(unwrapped.operand_b);
+ const inst_ty = unwrapped.result_ty;
+ const elem_ty = inst_ty.childType(zcu);
+
+ const writer = f.object.writer();
+ const local = try f.allocLocal(inst, inst_ty);
+ try reap(f, inst, &.{ unwrapped.operand_a, unwrapped.operand_b }); // local cannot alias operands
+ for (mask, 0..) |mask_elem, out_idx| {
+ try f.writeCValue(writer, local, .Other);
try writer.writeByte('[');
- try f.object.dg.renderValue(writer, src_val, .Other);
- try writer.writeAll("];\n");
+ try f.object.dg.renderValue(writer, try pt.intValue(.usize, out_idx), .Other);
+ try writer.writeAll("] = ");
+ switch (mask_elem.unwrap()) {
+ .a_elem => |src_idx| {
+ try f.writeCValue(writer, operand_a, .Other);
+ try writer.writeByte('[');
+ try f.object.dg.renderValue(writer, try pt.intValue(.usize, src_idx), .Other);
+ try writer.writeByte(']');
+ },
+ .b_elem => |src_idx| {
+ try f.writeCValue(writer, operand_b, .Other);
+ try writer.writeByte('[');
+ try f.object.dg.renderValue(writer, try pt.intValue(.usize, src_idx), .Other);
+ try writer.writeByte(']');
+ },
+ .undef => try f.object.dg.renderUndefValue(writer, elem_ty, .Other),
+ }
+ try writer.writeAll(";\n");
}
return local;