diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2021-10-29 17:48:34 -0700 |
|---|---|---|
| committer | Andrew Kelley <andrew@ziglang.org> | 2021-10-29 17:49:02 -0700 |
| commit | d6067db06267e37dec65202667741bc1b63fe980 (patch) | |
| tree | a693698a60b30a7d0ca2764056858c8b74066cd7 /src/Sema.zig | |
| parent | 5ff01bd820ea08005a422f046ad5bbad663b0dab (diff) | |
| download | zig-d6067db06267e37dec65202667741bc1b63fe980.tar.gz zig-d6067db06267e37dec65202667741bc1b63fe980.zip | |
stage2: implement `@popCount` for non-vectors
Diffstat (limited to 'src/Sema.zig')
| -rw-r--r-- | src/Sema.zig | 26 |
1 files changed, 24 insertions, 2 deletions
diff --git a/src/Sema.zig b/src/Sema.zig index 83b81ef174..0788997576 100644 --- a/src/Sema.zig +++ b/src/Sema.zig @@ -9904,8 +9904,30 @@ fn zirCtz(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Ins fn zirPopCount(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref { const inst_data = sema.code.instructions.items(.data)[inst].un_node; - const src = inst_data.src(); - return sema.fail(block, src, "TODO: Sema.zirPopCount", .{}); + 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 target = sema.mod.getTarget(); + const bits = operand_ty.intInfo(target).bits; + if (bits == 0) return Air.Inst.Ref.zero; + + const result_ty = try Type.smallestUnsignedInt(sema.arena, bits); + + const runtime_src = if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| { + if (val.isUndef()) return sema.addConstUndef(result_ty); + const result_val = try val.popCount(operand_ty, target, sema.arena); + return sema.addConstant(result_ty, result_val); + } else operand_src; + + try sema.requireRuntimeBlock(block, runtime_src); + return block.addTyOp(.popcount, result_ty, operand); } fn zirByteSwap(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref { |
