diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2022-03-18 15:27:08 -0400 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2022-03-18 15:27:08 -0400 |
| commit | d5803441cd0c06d3fb7f11f5cb447c20e0d62d52 (patch) | |
| tree | c478deb5396dd9293bd028f8ddaa4aae44cc03ae | |
| parent | e60c0468aaed9bae19ff40bcc163927e98aa2dd9 (diff) | |
| parent | e99b1b398a1a4c4fb6b5998b7bb5e29ff841c2a5 (diff) | |
| download | zig-d5803441cd0c06d3fb7f11f5cb447c20e0d62d52.tar.gz zig-d5803441cd0c06d3fb7f11f5cb447c20e0d62d52.zip | |
Merge pull request #11220 from schmee/vector-bitreverse
stage2: implement `@bitReverse` for vectors
| -rw-r--r-- | src/Sema.zig | 99 | ||||
| -rw-r--r-- | test/behavior/bitreverse.zig | 76 |
2 files changed, 124 insertions, 51 deletions
diff --git a/src/Sema.zig b/src/Sema.zig index 722c39cfcb..b798405d80 100644 --- a/src/Sema.zig +++ b/src/Sema.zig @@ -13341,28 +13341,14 @@ fn zirTruncate(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai const target = sema.mod.getTarget(); const dest_info = dest_scalar_ty.intInfo(target); - if (dest_info.bits == 0) { - if (is_vector) { - return sema.addConstant( - dest_ty, - try Value.Tag.repeated.create(sema.arena, Value.zero), - ); - } else { - return sema.addConstant(dest_ty, Value.zero); - } + if (try sema.typeHasOnePossibleValue(block, dest_ty_src, dest_ty)) |val| { + return sema.addConstant(dest_ty, val); } if (operand_scalar_ty.zigTypeTag() != .ComptimeInt) { const operand_info = operand_ty.intInfo(target); - if (operand_info.bits == 0) { - if (is_vector) { - return sema.addConstant( - dest_ty, - try Value.Tag.repeated.create(sema.arena, Value.zero), - ); - } else { - return sema.addConstant(dest_ty, Value.zero); - } + if (try sema.typeHasOnePossibleValue(block, operand_src, operand_ty)) |val| { + return sema.addConstant(operand_ty, val); } if (operand_info.signedness != dest_info.signedness) { @@ -13461,15 +13447,9 @@ fn zirBitCount( _ = try checkIntOrVector(sema, block, operand, operand_src); const target = sema.mod.getTarget(); const bits = operand_ty.intInfo(target).bits; - if (bits == 0) { - switch (operand_ty.zigTypeTag()) { - .Vector => return sema.addConstant( - try Type.vector(sema.arena, operand_ty.vectorLen(), Type.comptime_int), - try Value.Tag.repeated.create(sema.arena, Value.zero), - ), - .Int => return Air.Inst.Ref.zero, - else => unreachable, - } + + if (try sema.typeHasOnePossibleValue(block, operand_src, operand_ty)) |val| { + return sema.addConstant(operand_ty, val); } const result_scalar_ty = try Type.smallestUnsignedInt(sema.arena, bits); @@ -13528,10 +13508,12 @@ fn zirByteSwap(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai ); } + if (try sema.typeHasOnePossibleValue(block, operand_src, operand_ty)) |val| { + return sema.addConstant(operand_ty, val); + } + switch (operand_ty.zigTypeTag()) { .Int, .ComptimeInt => { - if (bits == 0) return Air.Inst.Ref.zero; - const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { if (val.isUndef()) return sema.addConstUndef(operand_ty); const result_val = try val.byteSwap(operand_ty, target, sema.arena); @@ -13542,13 +13524,6 @@ fn zirByteSwap(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai return block.addTyOp(.byte_swap, operand_ty, operand); }, .Vector => { - if (bits == 0) { - return sema.addConstant( - operand_ty, - try Value.Tag.repeated.create(sema.arena, Value.zero), - ); - } - const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { if (val.isUndef()) return sema.addConstUndef(operand_ty); @@ -13575,28 +13550,50 @@ fn zirByteSwap(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai fn zirBitReverse(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref { const inst_data = sema.code.instructions.items(.data)[inst].un_node; - const ty_src: LazySrcLoc = .{ .node_offset_builtin_call_arg0 = inst_data.src_node }; const operand_src: LazySrcLoc = .{ .node_offset_builtin_call_arg1 = inst_data.src_node }; const operand = sema.resolveInst(inst_data.operand); const operand_ty = sema.typeOf(operand); - // TODO implement support for vectors - if (operand_ty.zigTypeTag() != .Int) { - return sema.fail(block, ty_src, "expected integer type, found '{}'", .{ - operand_ty, - }); + _ = try sema.checkIntOrVectorAllowComptime(block, operand, operand_src); + + if (try sema.typeHasOnePossibleValue(block, operand_src, operand_ty)) |val| { + return sema.addConstant(operand_ty, val); } + const target = sema.mod.getTarget(); - const bits = operand_ty.intInfo(target).bits; - if (bits == 0) return Air.Inst.Ref.zero; + switch (operand_ty.zigTypeTag()) { + .Int, .ComptimeInt => { + const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { + if (val.isUndef()) return sema.addConstUndef(operand_ty); + const result_val = try val.bitReverse(operand_ty, target, sema.arena); + return sema.addConstant(operand_ty, result_val); + } else operand_src; - const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { - if (val.isUndef()) return sema.addConstUndef(operand_ty); - const result_val = try val.bitReverse(operand_ty, target, sema.arena); - return sema.addConstant(operand_ty, result_val); - } else operand_src; + try sema.requireRuntimeBlock(block, runtime_src); + return block.addTyOp(.bit_reverse, operand_ty, operand); + }, + .Vector => { + const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { + if (val.isUndef()) + return sema.addConstUndef(operand_ty); - try sema.requireRuntimeBlock(block, runtime_src); - return block.addTyOp(.bit_reverse, operand_ty, operand); + const vec_len = operand_ty.vectorLen(); + var elem_buf: Value.ElemValueBuffer = undefined; + const elems = try sema.arena.alloc(Value, vec_len); + for (elems) |*elem, i| { + const elem_val = val.elemValueBuffer(i, &elem_buf); + elem.* = try elem_val.bitReverse(operand_ty, target, sema.arena); + } + return sema.addConstant( + operand_ty, + try Value.Tag.aggregate.create(sema.arena, elems), + ); + } else operand_src; + + try sema.requireRuntimeBlock(block, runtime_src); + return block.addTyOp(.bit_reverse, operand_ty, operand); + }, + else => unreachable, + } } fn zirBitOffsetOf(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref { diff --git a/test/behavior/bitreverse.zig b/test/behavior/bitreverse.zig index 183e9aac55..36a3d3475c 100644 --- a/test/behavior/bitreverse.zig +++ b/test/behavior/bitreverse.zig @@ -87,3 +87,79 @@ fn testBitReverse() !void { var neg32: i32 = -16773785; try expect(@bitReverse(i32, @as(i32, -16773785)) == @bitReverse(i32, neg32)); } + +fn vector8() !void { + var v = @Vector(2, u8){ 0x12, 0x23 }; + var result = @bitReverse(u8, v); + try expect(result[0] == 0x48); + try expect(result[1] == 0xc4); +} + +test "bitReverse vectors u8" { + if (builtin.zig_backend == .stage1) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_wasm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_c) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_x86_64) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest; + + comptime try vector8(); + try vector8(); +} + +fn vector16() !void { + var v = @Vector(2, u16){ 0x1234, 0x2345 }; + var result = @bitReverse(u16, v); + try expect(result[0] == 0x2c48); + try expect(result[1] == 0xa2c4); +} + +test "bitReverse vectors u16" { + if (builtin.zig_backend == .stage1) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_wasm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_c) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_x86_64) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest; + + comptime try vector16(); + try vector16(); +} + +fn vector24() !void { + var v = @Vector(2, u24){ 0x123456, 0x234567 }; + var result = @bitReverse(u24, v); + try expect(result[0] == 0x6a2c48); + try expect(result[1] == 0xe6a2c4); +} + +test "bitReverse vectors u24" { + if (builtin.zig_backend == .stage1) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_wasm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_c) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_x86_64) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest; + + comptime try vector24(); + try vector24(); +} + +fn vector0() !void { + var v = @Vector(2, u0){ 0, 0 }; + var result = @bitReverse(u0, v); + try expect(result[0] == 0); + try expect(result[1] == 0); +} + +test "bitReverse vectors u0" { + if (builtin.zig_backend == .stage1) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_wasm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_c) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_x86_64) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; + if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest; + + comptime try vector0(); + try vector0(); +} |
