diff options
| author | John Schmidt <john.schmidt.h@gmail.com> | 2022-03-18 13:29:00 +0100 |
|---|---|---|
| committer | John Schmidt <john.schmidt.h@gmail.com> | 2022-03-18 16:28:38 +0100 |
| commit | 870341e32e4fa4c8ef1bc3b0163d6530cab5911e (patch) | |
| tree | 1b3a42c965a89225fbde7540a72e3412a3330002 /src/Sema.zig | |
| parent | f3f5a5d05b7056aceb408b701613242d053019ab (diff) | |
| download | zig-870341e32e4fa4c8ef1bc3b0163d6530cab5911e.tar.gz zig-870341e32e4fa4c8ef1bc3b0163d6530cab5911e.zip | |
stage2: implement `@bitReverse` for vectors
Diffstat (limited to 'src/Sema.zig')
| -rw-r--r-- | src/Sema.zig | 62 |
1 files changed, 46 insertions, 16 deletions
diff --git a/src/Sema.zig b/src/Sema.zig index 722c39cfcb..696fcb0e22 100644 --- a/src/Sema.zig +++ b/src/Sema.zig @@ -13575,28 +13575,58 @@ 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, - }); - } + const scalar_ty = try sema.checkIntOrVectorAllowComptime(block, operand, operand_src); + const target = sema.mod.getTarget(); - const bits = operand_ty.intInfo(target).bits; - if (bits == 0) return Air.Inst.Ref.zero; + const bits = scalar_ty.intInfo(target).bits; + if (bits == 0) { + switch (operand_ty.zigTypeTag()) { + .Vector => return sema.addConstant( + operand_ty, + try Value.Tag.repeated.create(sema.arena, Value.zero), + ), + .Int => return Air.Inst.Ref.zero, + else => unreachable, + } + } - 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; + 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; - try sema.requireRuntimeBlock(block, runtime_src); - return block.addTyOp(.bit_reverse, operand_ty, operand); + 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); + + 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 { |
