; 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) define i1 @gep_sub_1_uge_inbounds(ptr %dst, ptr %lower) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true ; CHECK-NEXT: [[DST_SUB_4:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -4 ; CHECK-NEXT: [[CMP_SUB_4:%.*]] = icmp uge ptr [[DST_SUB_4]], [[LOWER]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_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.sub.1 = getelementptr inbounds i8, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp uge ptr %dst.sub.1, %lower %dst.sub.3 = getelementptr inbounds i8, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp uge ptr %dst.sub.3, %lower %res.1 = xor i1 %cmp.sub.1, %cmp.sub.3 %dst.sub.4 = getelementptr inbounds i8, ptr %dst.add.3, i64 -4 %cmp.sub.4 = icmp uge ptr %dst.sub.4, %lower %res.2 = xor i1 %res.1, %cmp.sub.4 ret i1 %res.2 } define i1 @gep_sub_1_uge_only_inner_inbounds(ptr %dst, ptr %lower) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp uge ptr [[DST_SUB_1]], [[LOWER]] ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[CMP_SUB_3:%.*]] = icmp uge ptr [[DST_SUB_3]], [[LOWER]] ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_SUB_1]], [[CMP_SUB_3]] ; CHECK-NEXT: [[DST_SUB_4:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -4 ; CHECK-NEXT: [[CMP_SUB_4:%.*]] = icmp uge ptr [[DST_SUB_4]], [[LOWER]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_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.sub.1 = getelementptr i8, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp uge ptr %dst.sub.1, %lower %dst.sub.3 = getelementptr i8, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp uge ptr %dst.sub.3, %lower %res.1 = xor i1 %cmp.sub.1, %cmp.sub.3 %dst.sub.4 = getelementptr i8, ptr %dst.add.3, i64 -4 %cmp.sub.4 = icmp uge ptr %dst.sub.4, %lower %res.2 = xor i1 %res.1, %cmp.sub.4 ret i1 %res.2 } define i1 @gep_sub_1_uge_only_outer_inbounds(ptr %dst, ptr %lower) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp uge ptr [[DST_SUB_1]], [[LOWER]] ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[CMP_SUB_3:%.*]] = icmp uge ptr [[DST_SUB_3]], [[LOWER]] ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_SUB_1]], [[CMP_SUB_3]] ; CHECK-NEXT: [[DST_SUB_4:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -4 ; CHECK-NEXT: [[CMP_SUB_4:%.*]] = icmp uge ptr [[DST_SUB_4]], [[LOWER]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_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.sub.1 = getelementptr inbounds i8, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp uge ptr %dst.sub.1, %lower %dst.sub.3 = getelementptr inbounds i8, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp uge ptr %dst.sub.3, %lower %res.1 = xor i1 %cmp.sub.1, %cmp.sub.3 %dst.sub.4 = getelementptr inbounds i8, ptr %dst.add.3, i64 -4 %cmp.sub.4 = icmp uge ptr %dst.sub.4, %lower %res.2 = xor i1 %res.1, %cmp.sub.4 ret i1 %res.2 } define i1 @gep_sub_1_uge_no_inbounds(ptr %dst, ptr %lower) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp uge ptr [[DST_SUB_1]], [[LOWER]] ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[CMP_SUB_3:%.*]] = icmp uge ptr [[DST_SUB_3]], [[LOWER]] ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_SUB_1]], [[CMP_SUB_3]] ; CHECK-NEXT: [[DST_SUB_4:%.*]] = getelementptr i8, ptr [[DST_ADD_3]], i64 -4 ; CHECK-NEXT: [[CMP_SUB_4:%.*]] = icmp uge ptr [[DST_SUB_4]], [[LOWER]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_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.sub.1 = getelementptr i8, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp uge ptr %dst.sub.1, %lower %dst.sub.3 = getelementptr i8, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp uge ptr %dst.sub.3, %lower %res.1 = xor i1 %cmp.sub.1, %cmp.sub.3 %dst.sub.4 = getelementptr i8, ptr %dst.add.3, i64 -4 %cmp.sub.4 = icmp uge ptr %dst.sub.4, %lower %res.2 = xor i1 %res.1, %cmp.sub.4 ret i1 %res.2 } define i1 @gep_sub_1_ult(ptr %dst, ptr %upper) { ; CHECK-LABEL: @gep_sub_1_ult( ; CHECK-NEXT: [[DST_ADD_4:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i64 4 ; CHECK-NEXT: [[PRE:%.*]] = icmp ult ptr [[DST_ADD_4]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[PRE]]) ; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds i8, ptr [[DST]], i64 3 ; CHECK-NEXT: [[DST_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true ; CHECK-NEXT: ret i1 [[RES_1]] ; %dst.add.4 = getelementptr inbounds i8, ptr %dst, i64 4 %pre = icmp ult ptr %dst.add.4, %upper call void @llvm.assume(i1 %pre) %dst.add.3 = getelementptr inbounds i8, ptr %dst, i64 3 %dst.sub.1 = getelementptr inbounds i8, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper %dst.sub.3 = getelementptr inbounds i8, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp ult ptr %dst.sub.3, %upper %res.1 = xor i1 %cmp.sub.1, %cmp.sub.3 ret i1 %res.1 } define i1 @gep_sub_ult_var_idx(ptr %dst, ptr %upper, i8 %idx) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 -1 ; CHECK-NEXT: [[DST_SUB_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 -2 ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true ; CHECK-NEXT: [[DST_SUB_1_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_SUB_1]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1_SUB_1]], [[UPPER]] ; CHECK-NEXT: [[CMP_SUB_1_SUB_1_EQ:%.*]] = icmp eq ptr [[DST_SUB_1_SUB_1]], [[DST_SUB_2]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_1_SUB_1]] ; CHECK-NEXT: [[RES_3:%.*]] = xor i1 [[RES_2]], [[CMP_SUB_1_SUB_1_EQ]] ; CHECK-NEXT: ret i1 [[RES_3]] ; %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.sub.1 = getelementptr inbounds i8, ptr %dst.add.idx, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper %dst.sub.2 = getelementptr inbounds i8, ptr %dst.add.idx, i64 -2 %cmp.sub.2 = icmp ult ptr %dst.sub.2, %upper %res.1 = xor i1 %cmp.sub.1, %cmp.sub.2 %dst.sub.1.sub.1 = getelementptr inbounds i8, ptr %dst.sub.1, i64 -1 %cmp.sub.1.sub.1 = icmp ult ptr %dst.sub.1.sub.1, %upper %cmp.sub.1.sub.1.eq = icmp eq ptr %dst.sub.1.sub.1, %dst.sub.2 %res.2 = xor i1 %res.1, %cmp.sub.1.sub.1 %res.3 = xor i1 %res.2, %cmp.sub.1.sub.1.eq ret i1 %res.3 } define i1 @gep_sub_ult_var_idx_sgt_1(ptr %dst, ptr %upper, i8 %idx) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 -1 ; CHECK-NEXT: [[DST_SUB_2:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 -2 ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 true, true ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_IDX]], i64 -3 ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], true ; CHECK-NEXT: ret i1 [[RES_2]] ; %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.sub.1 = getelementptr inbounds i8, ptr %dst.add.idx, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper %dst.sub.2 = getelementptr inbounds i8, ptr %dst.add.idx, i64 -2 %cmp.sub.2 = icmp ult ptr %dst.sub.2, %upper %res.1 = xor i1 %cmp.sub.1, %cmp.sub.2 %dst.sub.3 = getelementptr inbounds i8, ptr %dst.add.idx, i64 -3 %cmp.sub.3 = icmp ult ptr %dst.sub.3, %upper %res.2 = xor i1 %res.1, %cmp.sub.3 ret i1 %res.2 } define i1 @gep_sub_1_ult_var_idx_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: ret i1 true ; %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.sub.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_1_ult_var_idx_only_inner_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr i8, ptr [[DST_ADD_LEN]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: [[CMP_IDX:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER]] ; CHECK-NEXT: ret i1 [[CMP_IDX]] ; %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.sub.1 = getelementptr i8, ptr %dst.add.len, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_1_ult_var_idx_no_inbounds(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_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_SUB_1:%.*]] = getelementptr i8, ptr [[DST_ADD_LEN]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: [[CMP_IDX:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER]] ; CHECK-NEXT: ret i1 [[CMP_IDX]] ; %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.sub.1 = getelementptr i8, ptr %dst.add.len, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_2_ult_var_idx(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_2_ult_var_idx( ; 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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 -2 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: [[CMP_IDX:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER]] ; CHECK-NEXT: ret i1 [[CMP_IDX]] ; %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.sub.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 -2 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_2_ult_var_idx_inbounds_len_sge_2(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_2_ult_var_idx_inbounds_len_sge_2( ; CHECK-NEXT: [[SGE_2:%.*]] = icmp sge i8 [[LEN:%.*]], 2 ; CHECK-NEXT: call void @llvm.assume(i1 [[SGE_2]]) ; 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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: ret i1 true ; %sge.2 = icmp sge i8 %len, 2 call void @llvm.assume(i1 %sge.2) %len.ext = zext i8 %len to i16 %dst.add.len = getelementptr inbounds i8, ptr %dst, i16 %len.ext %dst.sub.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_2_ult_var_idx_inbounds_len_uge_2(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_2_ult_var_idx_inbounds_len_uge_2( ; CHECK-NEXT: [[UGE_2:%.*]] = icmp uge i8 [[LEN:%.*]], 2 ; CHECK-NEXT: call void @llvm.assume(i1 [[UGE_2]]) ; 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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: ret i1 true ; %uge.2 = icmp uge i8 %len, 2 call void @llvm.assume(i1 %uge.2) %len.ext = zext i8 %len to i16 %dst.add.len = getelementptr inbounds i8, ptr %dst, i16 %len.ext %dst.sub.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 -1 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_ult_var_idx_len_sgt_1(ptr %dst, ptr %upper, i8 %len, i8 %idx) { ; CHECK-LABEL: @gep_sub_ult_var_idx_len_sgt_1( ; CHECK-NEXT: [[SGT_1:%.*]] = icmp sgt i8 [[LEN:%.*]], 1 ; CHECK-NEXT: call void @llvm.assume(i1 [[SGT_1]]) ; 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_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[DST_ADD_LEN]], i64 -2 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ult ptr [[DST_SUB_1]], [[UPPER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_SUB_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: [[CMP_IDX:%.*]] = icmp ult ptr [[DST_ADD_IDX]], [[UPPER]] ; CHECK-NEXT: ret i1 [[CMP_IDX]] ; %sgt.1 = icmp sgt i8 %len, 1 call void @llvm.assume(i1 %sgt.1) %len.ext = zext i8 %len to i16 %dst.add.len = getelementptr inbounds i8, ptr %dst, i16 %len.ext %dst.sub.1 = getelementptr inbounds i8, ptr %dst.add.len, i64 -2 %cmp.sub.1 = icmp ult ptr %dst.sub.1, %upper call void @llvm.assume(i1 %cmp.sub.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 %cmp.idx = icmp ult ptr %dst.add.idx, %upper ret i1 %cmp.idx } define i1 @gep_sub_1_ult_var_idx_lower_bound(ptr %lower, ptr %src, i8 %len) { ; CHECK-LABEL: @gep_sub_1_ult_var_idx_lower_bound( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[SRC_UGE_LOWER:%.*]] = icmp uge ptr [[SRC:%.*]], [[LOWER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[SRC_UGE_LOWER]]) ; CHECK-NEXT: [[LEN_POS:%.*]] = icmp sge i8 [[LEN:%.*]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[LEN_POS]]) ; CHECK-NEXT: [[GEP_LEN:%.*]] = getelementptr inbounds i8, ptr [[SRC]], i8 [[LEN]] ; CHECK-NEXT: [[GEP_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[GEP_LEN]], i8 -1 ; CHECK-NEXT: [[RES:%.*]] = icmp ult ptr [[GEP_SUB_1]], [[LOWER]] ; CHECK-NEXT: ret i1 [[RES]] ; entry: %src.uge.lower = icmp uge ptr %src, %lower call void @llvm.assume(i1 %src.uge.lower) %len.pos = icmp sge i8 %len, 0 call void @llvm.assume(i1 %len.pos) %gep.len = getelementptr inbounds i8, ptr %src, i8 %len %gep.sub.1 = getelementptr inbounds i8, ptr %gep.len, i8 -1 %res = icmp ult ptr %gep.sub.1, %lower ret i1 %res } define i1 @gep_sub_1_ult_var_idx_lower_bound_len_ne_0(ptr %lower, ptr %src, i8 %len) { ; CHECK-LABEL: @gep_sub_1_ult_var_idx_lower_bound_len_ne_0( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[LEN_NE_0:%.*]] = icmp ne i8 [[LEN:%.*]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[LEN_NE_0]]) ; CHECK-NEXT: [[SRC_UGE_LOWER:%.*]] = icmp uge ptr [[SRC:%.*]], [[LOWER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[SRC_UGE_LOWER]]) ; CHECK-NEXT: [[LEN_POS:%.*]] = icmp sge i8 [[LEN]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[LEN_POS]]) ; CHECK-NEXT: [[GEP_LEN:%.*]] = getelementptr inbounds i8, ptr [[SRC]], i8 [[LEN]] ; CHECK-NEXT: [[GEP_SUB_1:%.*]] = getelementptr inbounds i8, ptr [[GEP_LEN]], i8 -1 ; CHECK-NEXT: ret i1 false ; entry: %len.ne.0 = icmp ne i8 %len, 0 call void @llvm.assume(i1 %len.ne.0) %src.uge.lower = icmp uge ptr %src, %lower call void @llvm.assume(i1 %src.uge.lower) %len.pos = icmp sge i8 %len, 0 call void @llvm.assume(i1 %len.pos) %gep.len = getelementptr inbounds i8, ptr %src, i8 %len %gep.sub.1 = getelementptr inbounds i8, ptr %gep.len, i8 -1 %res = icmp ult ptr %gep.sub.1, %lower ret i1 %res } define i1 @gep_sub_2_ult_var_idx_lower_bound_len_ne_0(ptr %lower, ptr %src, i8 %len) { ; CHECK-LABEL: @gep_sub_2_ult_var_idx_lower_bound_len_ne_0( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[LEN_NE_0:%.*]] = icmp ne i8 [[LEN:%.*]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[LEN_NE_0]]) ; CHECK-NEXT: [[SRC_UGE_LOWER:%.*]] = icmp uge ptr [[SRC:%.*]], [[LOWER:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[SRC_UGE_LOWER]]) ; CHECK-NEXT: [[LEN_POS:%.*]] = icmp sge i8 [[LEN]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[LEN_POS]]) ; CHECK-NEXT: [[GEP_LEN:%.*]] = getelementptr inbounds i8, ptr [[SRC]], i8 [[LEN]] ; CHECK-NEXT: [[GEP_SUB_2:%.*]] = getelementptr inbounds i8, ptr [[GEP_LEN]], i8 -2 ; CHECK-NEXT: [[RES:%.*]] = icmp ult ptr [[GEP_SUB_2]], [[LOWER]] ; CHECK-NEXT: ret i1 [[RES]] ; entry: %len.ne.0 = icmp ne i8 %len, 0 call void @llvm.assume(i1 %len.ne.0) %src.uge.lower = icmp uge ptr %src, %lower call void @llvm.assume(i1 %src.uge.lower) %len.pos = icmp sge i8 %len, 0 call void @llvm.assume(i1 %len.pos) %gep.len = getelementptr inbounds i8, ptr %src, i8 %len %gep.sub.2 = getelementptr inbounds i8, ptr %gep.len, i8 -2 %res = icmp ult ptr %gep.sub.2, %lower ret i1 %res } define i1 @gep_i16_sub_1_uge_inbounds(ptr %dst, ptr %lower) { ; CHECK-LABEL: @gep_i16_sub_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_SUB_1:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i64 -1 ; CHECK-NEXT: [[DST_SUB_2:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i64 -2 ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 false, true ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[CMP_SUB_3:%.*]] = icmp ule ptr [[DST_SUB_3]], [[LOWER]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_3]] ; 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.sub.1 = getelementptr inbounds i16, ptr %dst.add.3, i64 -1 %cmp.sub.1 = icmp ule ptr %dst.sub.1, %lower %dst.sub.2 = getelementptr inbounds i16, ptr %dst.add.3, i64 -2 %cmp.sub.2 = icmp ule ptr %dst.sub.2, %dst %res.1 = xor i1 %cmp.sub.1, %cmp.sub.2 %dst.sub.3 = getelementptr inbounds i16, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp ule ptr %dst.sub.3, %lower %res.2 = xor i1 %res.1, %cmp.sub.3 ret i1 %res.2 } define i1 @gep_i16_sub_1_uge_inbounds_var_idx(ptr %dst, i64 %off) { ; CHECK-LABEL: @gep_i16_sub_1_uge_inbounds_var_idx( ; CHECK-NEXT: [[OFF_UGE:%.*]] = icmp sge i64 [[OFF:%.*]], 1 ; CHECK-NEXT: call void @llvm.assume(i1 [[OFF_UGE]]) ; CHECK-NEXT: [[DST_ADD_3:%.*]] = getelementptr inbounds i8, ptr [[DST:%.*]], i64 [[OFF]] ; CHECK-NEXT: [[DST_SUB_1:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i32 -1 ; CHECK-NEXT: [[CMP_SUB_1:%.*]] = icmp ule ptr [[DST_SUB_1]], [[DST]] ; CHECK-NEXT: [[DST_SUB_2:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i64 -2 ; CHECK-NEXT: [[CMP_SUB_2:%.*]] = icmp ule ptr [[DST_SUB_2]], [[DST]] ; CHECK-NEXT: [[RES_1:%.*]] = xor i1 [[CMP_SUB_1]], [[CMP_SUB_2]] ; CHECK-NEXT: [[DST_SUB_3:%.*]] = getelementptr inbounds i16, ptr [[DST_ADD_3]], i64 -3 ; CHECK-NEXT: [[CMP_SUB_3:%.*]] = icmp ule ptr [[DST_SUB_3]], [[DST]] ; CHECK-NEXT: [[RES_2:%.*]] = xor i1 [[RES_1]], [[CMP_SUB_3]] ; CHECK-NEXT: ret i1 [[RES_2]] ; %off.uge = icmp sge i64 %off, 1 call void @llvm.assume(i1 %off.uge) %dst.add.3 = getelementptr inbounds i8, ptr %dst, i64 %off %dst.sub.1 = getelementptr inbounds i16, ptr %dst.add.3, i32 -1 %cmp.sub.1 = icmp ule ptr %dst.sub.1, %dst %dst.sub.2 = getelementptr inbounds i16, ptr %dst.add.3, i64 -2 %cmp.sub.2 = icmp ule ptr %dst.sub.2, %dst %res.1 = xor i1 %cmp.sub.1, %cmp.sub.2 %dst.sub.3 = getelementptr inbounds i16, ptr %dst.add.3, i64 -3 %cmp.sub.3 = icmp ule ptr %dst.sub.3, %dst %res.2 = xor i1 %res.1, %cmp.sub.3 ret i1 %res.2 } define i1 @gep_i32_two_indices_known_lt_and_positive(ptr %a, i8 %idx.1, i8 %idx.2) { ; CHECK-LABEL: @gep_i32_two_indices_known_lt_and_positive( ; CHECK-NEXT: [[LT:%.*]] = icmp ult i8 [[IDX_1:%.*]], [[IDX_2:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[LT]]) ; CHECK-NEXT: [[IDX_1_POS:%.*]] = icmp sge i8 [[IDX_1]], 0 ; CHECK-NEXT: [[IDX_2_POS:%.*]] = icmp sge i8 [[IDX_2]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[IDX_1_POS]]) ; CHECK-NEXT: call void @llvm.assume(i1 [[IDX_2_POS]]) ; CHECK-NEXT: [[GEP_1:%.*]] = getelementptr inbounds i32, ptr [[A:%.*]], i8 [[IDX_1]] ; CHECK-NEXT: [[GEP_2:%.*]] = getelementptr inbounds i32, ptr [[GEP_1]], i8 -3 ; CHECK-NEXT: [[GEP_3:%.*]] = getelementptr inbounds i32, ptr [[A]], i8 [[IDX_2]] ; CHECK-NEXT: [[GEP_4:%.*]] = getelementptr inbounds i32, ptr [[GEP_3]], i8 -3 ; CHECK-NEXT: ret i1 true ; %lt = icmp ult i8 %idx.1, %idx.2 call void @llvm.assume(i1 %lt) %idx.1.pos = icmp sge i8 %idx.1, 0 %idx.2.pos = icmp sge i8 %idx.2, 0 call void @llvm.assume(i1 %idx.1.pos) call void @llvm.assume(i1 %idx.2.pos) %gep.1 = getelementptr inbounds i32, ptr %a, i8 %idx.1 %gep.2 = getelementptr inbounds i32, ptr %gep.1, i8 -3 %gep.3 = getelementptr inbounds i32, ptr %a, i8 %idx.2 %gep.4 = getelementptr inbounds i32, ptr %gep.3, i8 -3 %c = icmp ult ptr %gep.2, %gep.4 ret i1 %c } define i1 @gep_i32_two_indices_known_lt_and_not_known_positive(ptr %a, i8 %idx.1, i8 %idx.2) { ; CHECK-LABEL: @gep_i32_two_indices_known_lt_and_not_known_positive( ; CHECK-NEXT: [[LT:%.*]] = icmp ult i8 [[IDX_1:%.*]], [[IDX_2:%.*]] ; CHECK-NEXT: call void @llvm.assume(i1 [[LT]]) ; CHECK-NEXT: [[GEP_1:%.*]] = getelementptr inbounds i32, ptr [[A:%.*]], i8 [[IDX_1]] ; CHECK-NEXT: [[GEP_2:%.*]] = getelementptr inbounds i32, ptr [[GEP_1]], i8 -3 ; CHECK-NEXT: [[GEP_3:%.*]] = getelementptr inbounds i32, ptr [[A]], i8 [[IDX_2]] ; CHECK-NEXT: [[GEP_4:%.*]] = getelementptr inbounds i32, ptr [[GEP_3]], i8 -3 ; CHECK-NEXT: [[C:%.*]] = icmp ult ptr [[GEP_2]], [[GEP_4]] ; CHECK-NEXT: ret i1 [[C]] ; %lt = icmp ult i8 %idx.1, %idx.2 call void @llvm.assume(i1 %lt) %gep.1 = getelementptr inbounds i32, ptr %a, i8 %idx.1 %gep.2 = getelementptr inbounds i32, ptr %gep.1, i8 -3 %gep.3 = getelementptr inbounds i32, ptr %a, i8 %idx.2 %gep.4 = getelementptr inbounds i32, ptr %gep.3, i8 -3 %c = icmp ult ptr %gep.2, %gep.4 ret i1 %c } define i1 @gep_i32_two_indices_known_positive_but_not_lt(ptr %a, i8 %idx.1, i8 %idx.2) { ; CHECK-LABEL: @gep_i32_two_indices_known_positive_but_not_lt( ; CHECK-NEXT: [[IDX_1_POS:%.*]] = icmp sge i8 [[IDX_1:%.*]], 0 ; CHECK-NEXT: [[IDX_2_POS:%.*]] = icmp sge i8 [[IDX_2:%.*]], 0 ; CHECK-NEXT: call void @llvm.assume(i1 [[IDX_1_POS]]) ; CHECK-NEXT: call void @llvm.assume(i1 [[IDX_2_POS]]) ; CHECK-NEXT: [[GEP_1:%.*]] = getelementptr inbounds i32, ptr [[A:%.*]], i8 [[IDX_1]] ; CHECK-NEXT: [[GEP_2:%.*]] = getelementptr inbounds i32, ptr [[GEP_1]], i8 -3 ; CHECK-NEXT: [[GEP_3:%.*]] = getelementptr inbounds i32, ptr [[A]], i8 [[IDX_2]] ; CHECK-NEXT: [[GEP_4:%.*]] = getelementptr inbounds i32, ptr [[GEP_3]], i8 -3 ; CHECK-NEXT: [[C:%.*]] = icmp ult ptr [[GEP_2]], [[GEP_4]] ; CHECK-NEXT: ret i1 [[C]] ; %idx.1.pos = icmp sge i8 %idx.1, 0 %idx.2.pos = icmp sge i8 %idx.2, 0 call void @llvm.assume(i1 %idx.1.pos) call void @llvm.assume(i1 %idx.2.pos) %gep.1 = getelementptr inbounds i32, ptr %a, i8 %idx.1 %gep.2 = getelementptr inbounds i32, ptr %gep.1, i8 -3 %gep.3 = getelementptr inbounds i32, ptr %a, i8 %idx.2 %gep.4 = getelementptr inbounds i32, ptr %gep.3, i8 -3 %c = icmp ult ptr %gep.2, %gep.4 ret i1 %c }