aboutsummaryrefslogtreecommitdiff
path: root/src/Sema.zig
diff options
context:
space:
mode:
authorVeikka Tuominen <git@vexu.eu>2022-08-05 17:47:01 +0300
committerVeikka Tuominen <git@vexu.eu>2022-08-05 22:13:58 +0300
commit19d5ffc710faa23cd07a6dff23d4afc43c0c7f63 (patch)
tree515867fdb18082dbe7388a14f18ee6210adb1bc8 /src/Sema.zig
parent9116e26c1ffb49cf68bebf3af2a019af08474d12 (diff)
downloadzig-19d5ffc710faa23cd07a6dff23d4afc43c0c7f63.tar.gz
zig-19d5ffc710faa23cd07a6dff23d4afc43c0c7f63.zip
Sema: add safety check for non-power-of-two shift amounts
Diffstat (limited to 'src/Sema.zig')
-rw-r--r--src/Sema.zig128
1 files changed, 88 insertions, 40 deletions
diff --git a/src/Sema.zig b/src/Sema.zig
index e04190973b..3b672ecd42 100644
--- a/src/Sema.zig
+++ b/src/Sema.zig
@@ -10227,34 +10227,57 @@ fn zirShl(
} else rhs;
try sema.requireRuntimeBlock(block, src, runtime_src);
- if (block.wantSafety() and air_tag == .shl_exact) {
- const op_ov_tuple_ty = try sema.overflowArithmeticTupleType(lhs_ty);
- const op_ov = try block.addInst(.{
- .tag = .shl_with_overflow,
- .data = .{ .ty_pl = .{
- .ty = try sema.addType(op_ov_tuple_ty),
- .payload = try sema.addExtra(Air.Bin{
- .lhs = lhs,
- .rhs = rhs,
- }),
- } },
- });
- const ov_bit = try sema.tupleFieldValByIndex(block, src, op_ov, 1, op_ov_tuple_ty);
- const any_ov_bit = if (lhs_ty.zigTypeTag() == .Vector)
- try block.addInst(.{
- .tag = if (block.float_mode == .Optimized) .reduce_optimized else .reduce,
- .data = .{ .reduce = .{
- .operand = ov_bit,
- .operation = .Or,
+ if (block.wantSafety()) {
+ const bit_count = scalar_ty.intInfo(target).bits;
+ if (!std.math.isPowerOfTwo(bit_count)) {
+ const bit_count_val = try Value.Tag.int_u64.create(sema.arena, bit_count);
+
+ const ok = if (rhs_ty.zigTypeTag() == .Vector) ok: {
+ const bit_count_inst = try sema.addConstant(rhs_ty, try Value.Tag.repeated.create(sema.arena, bit_count_val));
+ const lt = try block.addCmpVector(rhs, bit_count_inst, .lt, try sema.addType(rhs_ty));
+ break :ok try block.addInst(.{
+ .tag = .reduce,
+ .data = .{ .reduce = .{
+ .operand = lt,
+ .operation = .And,
+ } },
+ });
+ } else ok: {
+ const bit_count_inst = try sema.addConstant(rhs_ty, bit_count_val);
+ break :ok try block.addBinOp(.cmp_lt, rhs, bit_count_inst);
+ };
+ try sema.addSafetyCheck(block, ok, .shift_rhs_too_big);
+ }
+
+ if (air_tag == .shl_exact) {
+ const op_ov_tuple_ty = try sema.overflowArithmeticTupleType(lhs_ty);
+ const op_ov = try block.addInst(.{
+ .tag = .shl_with_overflow,
+ .data = .{ .ty_pl = .{
+ .ty = try sema.addType(op_ov_tuple_ty),
+ .payload = try sema.addExtra(Air.Bin{
+ .lhs = lhs,
+ .rhs = rhs,
+ }),
} },
- })
- else
- ov_bit;
- const zero_ov = try sema.addConstant(Type.@"u1", Value.zero);
- const no_ov = try block.addBinOp(.cmp_eq, any_ov_bit, zero_ov);
+ });
+ const ov_bit = try sema.tupleFieldValByIndex(block, src, op_ov, 1, op_ov_tuple_ty);
+ const any_ov_bit = if (lhs_ty.zigTypeTag() == .Vector)
+ try block.addInst(.{
+ .tag = if (block.float_mode == .Optimized) .reduce_optimized else .reduce,
+ .data = .{ .reduce = .{
+ .operand = ov_bit,
+ .operation = .Or,
+ } },
+ })
+ else
+ ov_bit;
+ const zero_ov = try sema.addConstant(Type.@"u1", Value.zero);
+ const no_ov = try block.addBinOp(.cmp_eq, any_ov_bit, zero_ov);
- try sema.addSafetyCheck(block, no_ov, .shl_overflow);
- return sema.tupleFieldValByIndex(block, src, op_ov, 0, op_ov_tuple_ty);
+ try sema.addSafetyCheck(block, no_ov, .shl_overflow);
+ return sema.tupleFieldValByIndex(block, src, op_ov, 0, op_ov_tuple_ty);
+ }
}
return block.addBinOp(air_tag, lhs, new_rhs);
}
@@ -10333,20 +10356,43 @@ fn zirShr(
try sema.requireRuntimeBlock(block, src, runtime_src);
const result = try block.addBinOp(air_tag, lhs, rhs);
- if (block.wantSafety() and air_tag == .shr_exact) {
- const back = try block.addBinOp(.shl, result, rhs);
-
- const ok = if (rhs_ty.zigTypeTag() == .Vector) ok: {
- const eql = try block.addCmpVector(lhs, back, .eq, try sema.addType(rhs_ty));
- break :ok try block.addInst(.{
- .tag = if (block.float_mode == .Optimized) .reduce_optimized else .reduce,
- .data = .{ .reduce = .{
- .operand = eql,
- .operation = .And,
- } },
- });
- } else try block.addBinOp(.cmp_eq, lhs, back);
- try sema.addSafetyCheck(block, ok, .shr_overflow);
+ if (block.wantSafety()) {
+ const bit_count = scalar_ty.intInfo(target).bits;
+ if (!std.math.isPowerOfTwo(bit_count)) {
+ const bit_count_val = try Value.Tag.int_u64.create(sema.arena, bit_count);
+
+ const ok = if (rhs_ty.zigTypeTag() == .Vector) ok: {
+ const bit_count_inst = try sema.addConstant(rhs_ty, try Value.Tag.repeated.create(sema.arena, bit_count_val));
+ const lt = try block.addCmpVector(rhs, bit_count_inst, .lt, try sema.addType(rhs_ty));
+ break :ok try block.addInst(.{
+ .tag = .reduce,
+ .data = .{ .reduce = .{
+ .operand = lt,
+ .operation = .And,
+ } },
+ });
+ } else ok: {
+ const bit_count_inst = try sema.addConstant(rhs_ty, bit_count_val);
+ break :ok try block.addBinOp(.cmp_lt, rhs, bit_count_inst);
+ };
+ try sema.addSafetyCheck(block, ok, .shift_rhs_too_big);
+ }
+
+ if (air_tag == .shr_exact) {
+ const back = try block.addBinOp(.shl, result, rhs);
+
+ const ok = if (rhs_ty.zigTypeTag() == .Vector) ok: {
+ const eql = try block.addCmpVector(lhs, back, .eq, try sema.addType(rhs_ty));
+ break :ok try block.addInst(.{
+ .tag = if (block.float_mode == .Optimized) .reduce_optimized else .reduce,
+ .data = .{ .reduce = .{
+ .operand = eql,
+ .operation = .And,
+ } },
+ });
+ } else try block.addBinOp(.cmp_eq, lhs, back);
+ try sema.addSafetyCheck(block, ok, .shr_overflow);
+ }
}
return result;
}
@@ -19972,6 +20018,7 @@ pub const PanicId = enum {
inactive_union_field,
integer_part_out_of_bounds,
corrupt_switch,
+ shift_rhs_too_big,
};
fn addSafetyCheck(
@@ -20268,6 +20315,7 @@ fn safetyPanic(
.inactive_union_field => "access of inactive union field",
.integer_part_out_of_bounds => "integer part of floating point value out of bounds",
.corrupt_switch => "switch on corrupt value",
+ .shift_rhs_too_big => "shift amount is greater than the type size",
};
const msg_inst = msg_inst: {