; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -passes=constraint-elimination -S %s | FileCheck %s
declare void @llvm.assume(i1 noundef) #0
define i1 @gep_add_1_uge_inbounds(ptr %dst, ptr %lower) {
; CHECK-LABEL: @gep_add_1_uge_inbounds(
; CHECK-NEXT: [[PRE:%.*]] = icmp uge ptr [[DST:%.*]], [[LOWER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 3
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 1
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_1]], i64 1
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 3
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], true
; CHECK-NEXT: ret i1 [[RES_2]]
;
%pre = icmp uge ptr %dst, %lower
call void @llvm.assume(i1 %pre)
%dst.add.3 = getelementptr inbounds i8, ptr %dst, i64 3
%dst.add.1 = getelementptr inbounds i8, ptr %dst, i64 1
%cmp.add.1 = icmp uge ptr %dst.add.1, %lower
%dst.add.2 = getelementptr inbounds i8, ptr %dst.add.1, i64 1
%cmp.add.3 = icmp uge ptr %dst.add.3, %lower
%res.1 = xor i1 %cmp.add.1, %cmp.add.3
%dst.add.4 = getelementptr inbounds i8, ptr %dst.add.3, i64 3
%cmp.add.4 = icmp uge ptr %dst.add.4, %lower
%res.2 = xor i1 %res.1, %cmp.add.4
ret i1 %res.2
}
define i1 @gep_add_1_uge_inbounds_scalable_vector(ptr %dst, ptr %lower) {
; CHECK-LABEL: @gep_add_1_uge_inbounds_scalable_vector(
; CHECK-NEXT: [[PRE:%.*]] = icmp uge ptr [[DST:%.*]], [[LOWER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds <vscale x 4 x i8>, ptr [[DST]], i64 3
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds <vscale x 4 x i8>, ptr [[DST]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp uge ptr [[DST_ADD_1]], [[LOWER]]
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr inbounds <vscale x 4 x i8>, ptr [[DST_ADD_1]], i64 1
; CHECK-NEXT: [[CMP_ADD_3:%.*]] = icmp uge ptr [[DST_ADD_3]], [[LOWER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_ADD_1]], [[CMP_ADD_3]]
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr inbounds <vscale x 4 x i8>, ptr [[DST_ADD_3]], i64 3
; CHECK-NEXT: [[CMP_ADD_4:%.*]] = icmp uge ptr [[DST_ADD_4]], [[LOWER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_ADD_4]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%pre = icmp uge ptr %dst, %lower
call void @llvm.assume(i1 %pre)
%dst.add.3 = getelementptr inbounds <vscale x 4 x i8>, ptr %dst, i64 3
%dst.add.1 = getelementptr inbounds <vscale x 4 x i8>, ptr %dst, i64 1
%cmp.add.1 = icmp uge ptr %dst.add.1, %lower
%dst.add.2 = getelementptr inbounds <vscale x 4 x i8>, ptr %dst.add.1, i64 1
%cmp.add.3 = icmp uge ptr %dst.add.3, %lower
%res.1 = xor i1 %cmp.add.1, %cmp.add.3
%dst.add.4 = getelementptr inbounds <vscale x 4 x i8>, ptr %dst.add.3, i64 3
%cmp.add.4 = icmp uge ptr %dst.add.4, %lower
%res.2 = xor i1 %res.1, %cmp.add.4
ret i1 %res.2
}
define i1 @gep_add_1_uge_only_inner_inbounds(ptr %dst, ptr %lower) {
; CHECK-LABEL: @gep_add_1_uge_only_inner_inbounds(
; CHECK-NEXT: [[PRE:%.*]] = icmp uge ptr [[DST:%.*]], [[LOWER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 3
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 1
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr i8, ptr [[DST_ADD_1]], i64 1
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 3
; CHECK-NEXT: [[CMP_ADD_4:%.*]] = icmp uge ptr [[DST_ADD_4]], [[LOWER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_ADD_4]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%pre = icmp uge ptr %dst, %lower
call void @llvm.assume(i1 %pre)
%dst.add.3 = getelementptr inbounds i8, ptr %dst, i64 3
%dst.add.1 = getelementptr inbounds i8, ptr %dst, i64 1
%cmp.add.1 = icmp uge ptr %dst.add.1, %lower
%dst.add.2 = getelementptr i8, ptr %dst.add.1, i64 1
%cmp.add.3 = icmp uge ptr %dst.add.3, %lower
%res.1 = xor i1 %cmp.add.1, %cmp.add.3
%dst.add.4 = getelementptr i8, ptr %dst.add.3, i64 3
%cmp.add.4 = icmp uge ptr %dst.add.4, %lower
%res.2 = xor i1 %res.1, %cmp.add.4
ret i1 %res.2
}
define i1 @gep_add_1_uge_only_outer_inbounds(ptr %dst, ptr %lower) {
; CHECK-LABEL: @gep_add_1_uge_only_outer_inbounds(
; CHECK-NEXT: [[PRE:%.*]] = icmp uge ptr [[DST:%.*]], [[LOWER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr i8, ptr [[DST]], i64 3
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr i8, ptr [[DST]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp uge ptr [[DST_ADD_1]], [[LOWER]]
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_1]], i64 1
; CHECK-NEXT: [[CMP_ADD_3:%.*]] = icmp uge ptr [[DST_ADD_3]], [[LOWER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_ADD_1]], [[CMP_ADD_3]]
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 3
; CHECK-NEXT: [[CMP_ADD_4:%.*]] = icmp uge ptr [[DST_ADD_4]], [[LOWER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_ADD_4]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%pre = icmp uge ptr %dst, %lower
call void @llvm.assume(i1 %pre)
%dst.add.3 = getelementptr i8, ptr %dst, i64 3
%dst.add.1 = getelementptr i8, ptr %dst, i64 1
%cmp.add.1 = icmp uge ptr %dst.add.1, %lower
%dst.add.2 = getelementptr inbounds i8, ptr %dst.add.1, i64 1
%cmp.add.3 = icmp uge ptr %dst.add.3, %lower
%res.1 = xor i1 %cmp.add.1, %cmp.add.3
%dst.add.4 = getelementptr inbounds i8, ptr %dst.add.3, i64 3
%cmp.add.4 = icmp uge ptr %dst.add.4, %lower
%res.2 = xor i1 %res.1, %cmp.add.4
ret i1 %res.2
}
define i1 @gep_add_1_uge_no_inbounds(ptr %dst, ptr %lower) {
; CHECK-LABEL: @gep_add_1_uge_no_inbounds(
; CHECK-NEXT: [[PRE:%.*]] = icmp uge ptr [[DST:%.*]], [[LOWER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr i8, ptr [[DST]], i64 3
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr i8, ptr [[DST]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp uge ptr [[DST_ADD_1]], [[LOWER]]
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr i8, ptr [[DST_ADD_1]], i64 1
; CHECK-NEXT: [[CMP_ADD_3:%.*]] = icmp uge ptr [[DST_ADD_3]], [[LOWER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_ADD_1]], [[CMP_ADD_3]]
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 3
; CHECK-NEXT: [[CMP_ADD_4:%.*]] = icmp uge ptr [[DST_ADD_4]], [[LOWER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_ADD_4]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%pre = icmp uge ptr %dst, %lower
call void @llvm.assume(i1 %pre)
%dst.add.3 = getelementptr i8, ptr %dst, i64 3
%dst.add.1 = getelementptr i8, ptr %dst, i64 1
%cmp.add.1 = icmp uge ptr %dst.add.1, %lower
%dst.add.2 = getelementptr i8, ptr %dst.add.1, i64 1
%cmp.add.3 = icmp uge ptr %dst.add.3, %lower
%res.1 = xor i1 %cmp.add.1, %cmp.add.3
%dst.add.4 = getelementptr i8, ptr %dst.add.3, i64 3
%cmp.add.4 = icmp uge ptr %dst.add.4, %lower
%res.2 = xor i1 %res.1, %cmp.add.4
ret i1 %res.2
}
define i1 @gep_add_1_ult(ptr %dst, ptr %lower, ptr %upper) {
; CHECK-LABEL: @gep_add_1_ult(
; CHECK-NEXT: [[END:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i64 4
; CHECK-NEXT: [[PRE:%.*]] = icmp ult ptr [[END]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 1
; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_1]], i64 2
; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_1]], i64 3
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true
; CHECK-NEXT: [[DST_ADD_5:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_1]], i64 4
; CHECK-NEXT: [[CMP_ADD_5:%.*]] = icmp ult ptr [[DST_ADD_5]], [[UPPER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_ADD_5]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%end = getelementptr inbounds i8, ptr %dst, i64 4
%pre = icmp ult ptr %end, %upper
call void @llvm.assume(i1 %pre)
%dst.add.1 = getelementptr inbounds i8, ptr %dst, i64 1
%dst.add.3 = getelementptr inbounds i8, ptr %dst.add.1, i64 2
%cmp.add.3 = icmp ult ptr %dst.add.3, %upper
%dst.add.4 = getelementptr inbounds i8, ptr %dst.add.1, i64 3
%cmp.add.4 = icmp ult ptr %dst.add.4, %upper
%res.1 = xor i1 %cmp.add.3, %cmp.add.4
%dst.add.5 = getelementptr inbounds i8, ptr %dst.add.1, i64 4
%cmp.add.5 = icmp ult ptr %dst.add.5, %upper
%res.2 = xor i1 %res.1, %cmp.add.5
ret i1 %res.2
}
define i1 @gep_add_ult_var_idx(ptr %dst, ptr %upper, i8 %idx) {
; CHECK-LABEL: @gep_add_ult_var_idx(
; CHECK-NEXT: [[NOT_ZERO:%.*]] = icmp ne i8 [[IDX:%.*]], 0
; CHECK-NEXT: call void @llvm.assume(i1 [[NOT_ZERO]])
; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i8 [[IDX]] to i16
; CHECK-NEXT: [[DST_ADD_IDX:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i16 [[IDX_EXT]]
; CHECK-NEXT: [[PRE:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 1
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 2
; CHECK-NEXT: [[CMP_ADD_2:%.*]] = icmp ule ptr [[DST_ADD_2]], [[UPPER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, [[CMP_ADD_2]]
; CHECK-NEXT: ret i1 [[RES_1]]
;
%not.zero = icmp ne i8 %idx, 0
call void @llvm.assume(i1 %not.zero)
%idx.ext = zext i8 %idx to i16
%dst.add.idx = getelementptr inbounds i8, ptr %dst, i16 %idx.ext
%pre = icmp ult ptr %dst.add.idx, %upper
call void @llvm.assume(i1 %pre)
%dst.add.1 = getelementptr inbounds i8, ptr %dst.add.idx, i64 1
%cmp.add.1 = icmp ule ptr %dst.add.1, %upper
%dst.add.2 = getelementptr inbounds i8, ptr %dst.add.idx, i64 2
%cmp.add.2 = icmp ule ptr %dst.add.2, %upper
%res.1 = xor i1 %cmp.add.1, %cmp.add.2
ret i1 %res.1
}
define i1 @gep_add_ult_var_idx_sgt_1(ptr %dst, ptr %upper, i8 %idx) {
; CHECK-LABEL: @gep_add_ult_var_idx_sgt_1(
; CHECK-NEXT: [[SGT_1:%.*]] = icmp sgt i8 [[IDX:%.*]], 1
; CHECK-NEXT: call void @llvm.assume(i1 [[SGT_1]])
; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i8 [[IDX]] to i16
; CHECK-NEXT: [[DST_ADD_IDX:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i16 [[IDX_EXT]]
; CHECK-NEXT: [[PRE:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]])
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 1
; CHECK-NEXT: [[DST_ADD_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 2
; CHECK-NEXT: [[CMP_ADD_2:%.*]] = icmp ule ptr [[DST_ADD_2]], [[UPPER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, [[CMP_ADD_2]]
; CHECK-NEXT: ret i1 [[RES_1]]
;
%sgt.1 = icmp sgt i8 %idx, 1
call void @llvm.assume(i1 %sgt.1)
%idx.ext = zext i8 %idx to i16
%dst.add.idx = getelementptr inbounds i8, ptr %dst, i16 %idx.ext
%pre = icmp ult ptr %dst.add.idx, %upper
call void @llvm.assume(i1 %pre)
%dst.add.1 = getelementptr inbounds i8, ptr %dst.add.idx, i64 1
%cmp.add.1 = icmp ule ptr %dst.add.1, %upper
%dst.add.2 = getelementptr inbounds i8, ptr %dst.add.idx, i64 2
%cmp.add.2 = icmp ule ptr %dst.add.2, %upper
%res.1 = xor i1 %cmp.add.1, %cmp.add.2
ret i1 %res.1
}
define i1 @gep_add_1_ult_var_idx_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) {
; CHECK-LABEL: @gep_add_1_ult_var_idx_inbounds(
; CHECK-NEXT: [[NOT_ZERO:%.*]] = icmp ne i8 [[LEN:%.*]], 0
; CHECK-NEXT: call void @llvm.assume(i1 [[NOT_ZERO]])
; CHECK-NEXT: [[LEN_EXT:%.*]] = zext i8 [[LEN]] to i16
; CHECK-NEXT: [[DST_ADD_LEN:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i16 [[LEN_EXT]]
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp ult ptr [[DST_ADD_1]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_ADD_1]])
; CHECK-NEXT: [[CMP_IDX_ULT_LEN:%.*]] = icmp ult i8 [[IDX:%.*]], [[LEN]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_IDX_ULT_LEN]])
; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i8 [[IDX]] to i16
; CHECK-NEXT: [[DST_ADD_IDX:%.*]] = getelementptr inbounds i8, ptr [[DST]], i16 [[IDX_EXT]]
; CHECK-NEXT: [[DST_ADD_IDX_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 1
; CHECK-NEXT: [[DST_ADD_IDX_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 2
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true
; CHECK-NEXT: [[DST_ADD_IDX_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 3
; CHECK-NEXT: [[CMP_IDX_3:%.*]] = icmp ult ptr [[DST_ADD_IDX_3]], [[UPPER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_IDX_3]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%not.zero = icmp ne i8 %len, 0
call void @llvm.assume(i1 %not.zero)
%len.ext = zext i8 %len to i16
%dst.add.len = getelementptr inbounds i8, ptr %dst, i16 %len.ext
%dst.add.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 1
%cmp.add.1 = icmp ult ptr %dst.add.1, %upper
call void @llvm.assume(i1 %cmp.add.1)
%cmp.idx.ult.len = icmp ult i8 %idx, %len
call void @llvm.assume(i1 %cmp.idx.ult.len)
%idx.ext = zext i8 %idx to i16
%dst.add.idx = getelementptr inbounds i8, ptr %dst, i16 %idx.ext
%dst.add.idx.1 = getelementptr inbounds i8, ptr %dst.add.idx, i64 1
%cmp.idx.1 = icmp ult ptr %dst.add.idx.1, %upper
%dst.add.idx.2 = getelementptr inbounds i8, ptr %dst.add.idx, i64 2
%cmp.idx.2 = icmp ult ptr %dst.add.idx.2, %upper
%res.1 = xor i1 %cmp.idx.1, %cmp.idx.2
%dst.add.idx.3 = getelementptr inbounds i8, ptr %dst.add.idx, i64 3
%cmp.idx.3 = icmp ult ptr %dst.add.idx.3, %upper
%res.2 = xor i1 %res.1, %cmp.idx.3
ret i1 %res.2
}
define i1 @gep_add_1_ult_var_idx_only_inner_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) {
; CHECK-LABEL: @gep_add_1_ult_var_idx_only_inner_inbounds(
; CHECK-NEXT: [[NOT_ZERO:%.*]] = icmp ne i8 [[LEN:%.*]], 0
; CHECK-NEXT: call void @llvm.assume(i1 [[NOT_ZERO]])
; CHECK-NEXT: [[LEN_EXT:%.*]] = zext i8 [[LEN]] to i16
; CHECK-NEXT: [[DST_ADD_LEN:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i16 [[LEN_EXT]]
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp ult ptr [[DST_ADD_1]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_ADD_1]])
; CHECK-NEXT: [[CMP_IDX_ULT_LEN:%.*]] = icmp ult i8 [[IDX:%.*]], [[LEN]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_IDX_ULT_LEN]])
; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i8 [[IDX]] to i16
; CHECK-NEXT: [[DST_ADD_IDX:%.*]] = getelementptr inbounds i8, ptr [[DST]], i16 [[IDX_EXT]]
; CHECK-NEXT: [[DST_ADD_IDX_1:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 1
; CHECK-NEXT: [[CMP_IDX_1:%.*]] = icmp ult ptr [[DST_ADD_IDX_1]], [[UPPER]]
; CHECK-NEXT: [[DST_ADD_IDX_2:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 2
; CHECK-NEXT: [[CMP_IDX_2:%.*]] = icmp ult ptr [[DST_ADD_IDX_2]], [[UPPER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_IDX_1]], [[CMP_IDX_2]]
; CHECK-NEXT: [[DST_ADD_IDX_3:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 3
; CHECK-NEXT: [[CMP_IDX_3:%.*]] = icmp ult ptr [[DST_ADD_IDX_3]], [[UPPER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_IDX_3]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%not.zero = icmp ne i8 %len, 0
call void @llvm.assume(i1 %not.zero)
%len.ext = zext i8 %len to i16
%dst.add.len = getelementptr inbounds i8, ptr %dst, i16 %len.ext
%dst.add.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 1
%cmp.add.1 = icmp ult ptr %dst.add.1, %upper
call void @llvm.assume(i1 %cmp.add.1)
%cmp.idx.ult.len = icmp ult i8 %idx, %len
call void @llvm.assume(i1 %cmp.idx.ult.len)
%idx.ext = zext i8 %idx to i16
%dst.add.idx = getelementptr inbounds i8, ptr %dst, i16 %idx.ext
%dst.add.idx.1 = getelementptr i8, ptr %dst.add.idx, i64 1
%cmp.idx.1 = icmp ult ptr %dst.add.idx.1, %upper
%dst.add.idx.2 = getelementptr i8, ptr %dst.add.idx, i64 2
%cmp.idx.2 = icmp ult ptr %dst.add.idx.2, %upper
%res.1 = xor i1 %cmp.idx.1, %cmp.idx.2
%dst.add.idx.3 = getelementptr i8, ptr %dst.add.idx, i64 3
%cmp.idx.3 = icmp ult ptr %dst.add.idx.3, %upper
%res.2 = xor i1 %res.1, %cmp.idx.3
ret i1 %res.2
}
define i1 @gep_add_1_ult_var_idx_no_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) {
; CHECK-LABEL: @gep_add_1_ult_var_idx_no_inbounds(
; CHECK-NEXT: [[NOT_ZERO:%.*]] = icmp ne i8 [[LEN:%.*]], 0
; CHECK-NEXT: call void @llvm.assume(i1 [[NOT_ZERO]])
; CHECK-NEXT: [[LEN_EXT:%.*]] = zext i8 [[LEN]] to i16
; CHECK-NEXT: [[DST_ADD_LEN:%.*]] = getelementptr i8, ptr [[DST:%.*]], i16 [[LEN_EXT]]
; CHECK-NEXT: [[DST_ADD_1:%.*]] = getelementptr i8, ptr [[DST_ADD_LEN]], i64 1
; CHECK-NEXT: [[CMP_ADD_1:%.*]] = icmp ult ptr [[DST_ADD_1]], [[UPPER:%.*]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_ADD_1]])
; CHECK-NEXT: [[CMP_IDX_ULT_LEN:%.*]] = icmp ult i8 [[IDX:%.*]], [[LEN]]
; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_IDX_ULT_LEN]])
; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i8 [[IDX]] to i16
; CHECK-NEXT: [[DST_ADD_IDX:%.*]] = getelementptr i8, ptr [[DST]], i16 [[IDX_EXT]]
; CHECK-NEXT: [[DST_ADD_IDX_1:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 1
; CHECK-NEXT: [[CMP_IDX_1:%.*]] = icmp ult ptr [[DST_ADD_IDX_1]], [[UPPER]]
; CHECK-NEXT: [[DST_ADD_IDX_2:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 2
; CHECK-NEXT: [[CMP_IDX_2:%.*]] = icmp ult ptr [[DST_ADD_IDX_2]], [[UPPER]]
; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_IDX_1]], [[CMP_IDX_2]]
; CHECK-NEXT: [[DST_ADD_IDX_3:%.*]] = getelementptr i8, ptr [[DST_ADD_IDX]], i64 3
; CHECK-NEXT: [[CMP_IDX_3:%.*]] = icmp ult ptr [[DST_ADD_IDX_3]], [[UPPER]]
; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_IDX_3]]
; CHECK-NEXT: ret i1 [[RES_2]]
;
%not.zero = icmp ne i8 %len, 0
call void @llvm.assume(i1 %not.zero)
%len.ext = zext i8 %len to i16
%dst.add.len = getelementptr i8, ptr %dst, i16 %len.ext
%dst.add.1 = getelementptr i8, ptr %dst.add.len, i64 1
%cmp.add.1 = icmp ult ptr %dst.add.1, %upper
call void @llvm.assume(i1 %cmp.add.1)
%cmp.idx.ult.len = icmp ult i8 %idx, %len
call void @llvm.assume(i1 %cmp.idx.ult.len)
%idx.ext = zext i8 %idx to i16
%dst.add.idx = getelementptr i8, ptr %dst, i16 %idx.ext
%dst.add.idx.1 = getelementptr i8, ptr %dst.add.idx, i64 1
%cmp.idx.1 = icmp ult ptr %dst.add.idx.1, %upper
%dst.add.idx.2 = getelementptr i8, ptr %dst.add.idx, i64 2
%cmp.idx.2 = icmp ult ptr %dst.add.idx.2, %upper
%res.1 = xor i1 %cmp.idx.1, %cmp.idx.2
%dst.add.idx.3 = getelementptr i8, ptr %dst.add.idx, i64 3
%cmp.idx.3 = icmp ult ptr %dst.add.idx.3, %upper
%res.2 = xor i1 %res.1, %cmp.idx.3
ret i1 %res.2
}
define i1 @test_chained_no_inbounds(ptr %A, ptr %B) {
; CHECK-LABEL: @test_chained_no_inbounds(
; CHECK-NEXT: entry:
; CHECK-NEXT: [[B_1:%.*]] = getelementptr i8, ptr [[B:%.*]], i64 1
; CHECK-NEXT: [[B_2:%.*]] = getelementptr i8, ptr [[B_1]], i64 1
; CHECK-NEXT: [[C_1:%.*]] = icmp ugt ptr [[A:%.*]], null
; CHECK-NEXT: [[C_2:%.*]] = icmp ugt ptr [[B_1]], [[B_2]]
; CHECK-NEXT: [[OR:%.*]] = or i1 [[C_1]], [[C_2]]
; CHECK-NEXT: br i1 [[OR]], label [[THEN:%.*]], label [[ELSE:%.*]]
; CHECK: then:
; CHECK-NEXT: ret i1 true
; CHECK: else:
; CHECK-NEXT: ret i1 false
;
entry:
%B.1 = getelementptr i8, ptr %B, i64 1
%B.2 = getelementptr i8, ptr %B.1, i64 1
%c.1 = icmp ugt ptr %A, null
%c.2 = icmp ugt ptr %B.1, %B.2
%or = or i1 %c.1, %c.2
br i1 %or, label %then, label %else
then:
ret i1 true
else:
ret i1 false
}