aboutsummaryrefslogtreecommitdiff
path: root/src/Sema.zig
diff options
context:
space:
mode:
authorRobin Voetter <robin@voetter.nl>2021-10-16 14:47:55 +0200
committerRobin Voetter <robin@voetter.nl>2021-10-17 20:33:04 +0200
commit9336a87452eda87c19cb707484d0b6dfb4140b57 (patch)
tree72d50163e2ad1389d5752de0e841b8f68755f3d5 /src/Sema.zig
parent6a3659c4e005d9730fb824b77b416ef33200dbfe (diff)
downloadzig-9336a87452eda87c19cb707484d0b6dfb4140b57.tar.gz
zig-9336a87452eda87c19cb707484d0b6dfb4140b57.zip
stage2: bitNot
Diffstat (limited to 'src/Sema.zig')
-rw-r--r--src/Sema.zig38
1 files changed, 36 insertions, 2 deletions
diff --git a/src/Sema.zig b/src/Sema.zig
index 229ae054b2..6ad423a8e8 100644
--- a/src/Sema.zig
+++ b/src/Sema.zig
@@ -6627,8 +6627,42 @@ fn zirBitNot(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.
const tracy = trace(@src());
defer tracy.end();
- _ = inst;
- return sema.fail(block, sema.src, "TODO implement zirBitNot", .{});
+ const inst_data = sema.code.instructions.items(.data)[inst].un_node;
+ const src = inst_data.src();
+ const operand_src = src; // TODO put this on the operand, not the '~'
+
+ const operand = sema.resolveInst(inst_data.operand);
+ const operand_type = sema.typeOf(operand);
+ const scalar_type = operand_type.scalarType();
+
+ if (scalar_type.zigTypeTag() != .Int) {
+ return sema.fail(block, src, "unable to perform binary not operation on type '{}'", .{operand_type});
+ }
+
+ if (try sema.resolveMaybeUndefVal(block, operand_src, operand)) |val| {
+ const target = sema.mod.getTarget();
+ if (val.isUndef()) {
+ return sema.addConstUndef(scalar_type);
+ } else if (operand_type.zigTypeTag() == .Vector) {
+ const vec_len = operand_type.arrayLen();
+ var elem_val_buf: Value.ElemValueBuffer = undefined;
+ const elems = try sema.arena.alloc(Value, vec_len);
+ for (elems) |*elem, i| {
+ const elem_val = val.elemValueBuffer(i, &elem_val_buf);
+ elem.* = try elem_val.bitwiseNot(scalar_type, sema.arena, target);
+ }
+ return sema.addConstant(
+ operand_type,
+ try Value.Tag.array.create(sema.arena, elems),
+ );
+ } else {
+ const result_val = try val.bitwiseNot(scalar_type, sema.arena, target);
+ return sema.addConstant(scalar_type, result_val);
+ }
+ }
+
+ try sema.requireRuntimeBlock(block, src);
+ return block.addTyOp(.not, operand_type, operand);
}
fn zirArrayCat(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Inst.Ref {