@@ -2391,6 +2391,16 @@ fn failWithComptimeErrorRetTrace(
2391
2391
return sema.failWithOwnedErrorMsg(block, msg);
2392
2392
}
2393
2393
2394
+ fn failWithInvalidPtrArithmetic(sema: *Sema, block: *Block, src: LazySrcLoc, arithmetic: []const u8, supports: []const u8) CompileError {
2395
+ const msg = msg: {
2396
+ const msg = try sema.errMsg(block, src, "invalid {s} arithmetic operator", .{arithmetic});
2397
+ errdefer msg.destroy(sema.gpa);
2398
+ try sema.errNote(block, src, msg, "{s} arithmetic only supports {s}", .{ arithmetic, supports });
2399
+ break :msg msg;
2400
+ };
2401
+ return sema.failWithOwnedErrorMsg(block, msg);
2402
+ }
2403
+
2394
2404
/// We don't return a pointer to the new error note because the pointer
2395
2405
/// becomes invalid when you add another one.
2396
2406
fn errNote(
@@ -15146,7 +15156,7 @@ fn zirDiv(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Ins
15146
15156
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
15147
15157
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
15148
15158
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
15149
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
15159
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
15150
15160
15151
15161
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
15152
15162
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -15311,7 +15321,7 @@ fn zirDivExact(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai
15311
15321
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
15312
15322
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
15313
15323
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
15314
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
15324
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
15315
15325
15316
15326
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
15317
15327
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -15476,7 +15486,7 @@ fn zirDivFloor(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai
15476
15486
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
15477
15487
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
15478
15488
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
15479
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
15489
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
15480
15490
15481
15491
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
15482
15492
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -15586,7 +15596,7 @@ fn zirDivTrunc(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Ai
15586
15596
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
15587
15597
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
15588
15598
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
15589
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
15599
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
15590
15600
15591
15601
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
15592
15602
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -15827,7 +15837,7 @@ fn zirModRem(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.
15827
15837
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
15828
15838
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
15829
15839
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
15830
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
15840
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
15831
15841
15832
15842
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
15833
15843
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -16011,7 +16021,7 @@ fn zirMod(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Ins
16011
16021
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
16012
16022
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
16013
16023
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
16014
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
16024
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
16015
16025
16016
16026
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
16017
16027
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -16106,7 +16116,7 @@ fn zirRem(sema: *Sema, block: *Block, inst: Zir.Inst.Index) CompileError!Air.Ins
16106
16116
const lhs_zig_ty_tag = try lhs_ty.zigTypeTagOrPoison(mod);
16107
16117
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
16108
16118
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
16109
- try sema.checkInvalidPtrArithmetic (block, src, lhs_ty);
16119
+ try sema.checkInvalidPtrIntArithmetic (block, src, lhs_ty);
16110
16120
16111
16121
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
16112
16122
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -16445,17 +16455,58 @@ fn analyzeArithmetic(
16445
16455
const rhs_zig_ty_tag = try rhs_ty.zigTypeTagOrPoison(mod);
16446
16456
try sema.checkVectorizableBinaryOperands(block, src, lhs_ty, rhs_ty, lhs_src, rhs_src);
16447
16457
16448
- if (lhs_zig_ty_tag == .Pointer) switch (lhs_ty.ptrSize(mod)) {
16449
- .One, .Slice => {},
16450
- .Many, .C => {
16451
- const air_tag: Air.Inst.Tag = switch (zir_tag) {
16452
- .add => .ptr_add,
16453
- .sub => .ptr_sub,
16454
- else => return sema.fail(block, src, "invalid pointer arithmetic operator", .{}),
16455
- };
16456
- return sema.analyzePtrArithmetic(block, src, lhs, rhs, air_tag, lhs_src, rhs_src);
16457
- },
16458
- };
16458
+ if (lhs_zig_ty_tag == .Pointer) {
16459
+ if (rhs_zig_ty_tag == .Pointer) {
16460
+ if (lhs_ty.ptrSize(mod) != .Slice and rhs_ty.ptrSize(mod) != .Slice) {
16461
+ if (zir_tag != .sub) {
16462
+ return sema.failWithInvalidPtrArithmetic(block, src, "pointer-pointer", "subtraction");
16463
+ }
16464
+
16465
+ const runtime_src = runtime_src: {
16466
+ if (try sema.resolveValue(lhs)) |lhs_value| {
16467
+ if (try sema.resolveValue(rhs)) |rhs_value| {
16468
+ const lhs_ptr = switch (mod.intern_pool.indexToKey(lhs_value.toIntern())) {
16469
+ .undef => return sema.failWithUseOfUndef(block, lhs_src),
16470
+ .ptr => |ptr| ptr,
16471
+ else => unreachable,
16472
+ };
16473
+ const rhs_ptr = switch (mod.intern_pool.indexToKey(rhs_value.toIntern())) {
16474
+ .undef => return sema.failWithUseOfUndef(block, rhs_src),
16475
+ .ptr => |ptr| ptr,
16476
+ else => unreachable,
16477
+ };
16478
+ // Make sure the pointers point to the same data.
16479
+ if (!lhs_ptr.base_addr.eql(rhs_ptr.base_addr)) break :runtime_src src;
16480
+ const address = std.math.sub(u64, lhs_ptr.byte_offset, rhs_ptr.byte_offset) catch
16481
+ return sema.fail(block, src, "operation results in overflow", .{});
16482
+ return try mod.intRef(Type.usize, address);
16483
+ } else {
16484
+ break :runtime_src lhs_src;
16485
+ }
16486
+ } else {
16487
+ break :runtime_src rhs_src;
16488
+ }
16489
+ };
16490
+
16491
+ try sema.requireRuntimeBlock(block, src, runtime_src);
16492
+ const lhs_int = try block.addUnOp(.int_from_ptr, lhs);
16493
+ const rhs_int = try block.addUnOp(.int_from_ptr, rhs);
16494
+ return try block.addBinOp(.sub_wrap, lhs_int, rhs_int);
16495
+ }
16496
+ } else {
16497
+ switch (lhs_ty.ptrSize(mod)) {
16498
+ .One, .Slice => {},
16499
+ .Many, .C => {
16500
+ const air_tag: Air.Inst.Tag = switch (zir_tag) {
16501
+ .add => .ptr_add,
16502
+ .sub => .ptr_sub,
16503
+ else => return sema.failWithInvalidPtrArithmetic(block, src, "pointer-integer", "addition and subtraction"),
16504
+ };
16505
+ return sema.analyzePtrArithmetic(block, src, lhs, rhs, air_tag, lhs_src, rhs_src);
16506
+ },
16507
+ }
16508
+ }
16509
+ }
16459
16510
16460
16511
const instructions = &[_]Air.Inst.Ref{ lhs, rhs };
16461
16512
const resolved_type = try sema.resolvePeerTypes(block, src, instructions, .{
@@ -23715,7 +23766,7 @@ fn checkIntType(sema: *Sema, block: *Block, src: LazySrcLoc, ty: Type) CompileEr
23715
23766
}
23716
23767
}
23717
23768
23718
- fn checkInvalidPtrArithmetic (
23769
+ fn checkInvalidPtrIntArithmetic (
23719
23770
sema: *Sema,
23720
23771
block: *Block,
23721
23772
src: LazySrcLoc,
@@ -23725,12 +23776,7 @@ fn checkInvalidPtrArithmetic(
23725
23776
switch (try ty.zigTypeTagOrPoison(mod)) {
23726
23777
.Pointer => switch (ty.ptrSize(mod)) {
23727
23778
.One, .Slice => return,
23728
- .Many, .C => return sema.fail(
23729
- block,
23730
- src,
23731
- "invalid pointer arithmetic operator",
23732
- .{},
23733
- ),
23779
+ .Many, .C => return sema.failWithInvalidPtrArithmetic(block, src, "pointer-integer", "addition and subtraction"),
23734
23780
},
23735
23781
else => return,
23736
23782
}
0 commit comments