1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -constraint-elimination -S %s | FileCheck %s
4 ; Tests for using inbounds information from GEPs where the GEP only causes UB in the use blocks.
6 declare void @noundef(i32* noundef) willreturn nounwind
7 declare void @noundef2(i32* noundef)
11 ; %start + %n.ext is guaranteed to not overflow (due to inbounds).
12 ; %start + %idx.ext does not overflow if %idx.ext <= %n.ext.
13 define i1 @inbounds_poison_is_ub_in_use_block_1(i32* %src, i32 %n, i32 %idx) {
14 ; CHECK-LABEL: @inbounds_poison_is_ub_in_use_block_1(
16 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
17 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
18 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
19 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
20 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
21 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
23 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
24 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
25 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
27 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
28 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
31 %n.ext = zext i32 %n to i64
32 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
33 %cmp.idx = icmp ult i32 %idx, %n
34 %idx.ext = zext i32 %idx to i64
35 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
36 br i1 %cmp.idx, label %then, label %else
39 call void @noundef(i32* %upper)
40 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
44 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
48 define i1 @inbounds_poison_is_ub_in_use_block_2(i32* %src, i32 %n, i32 %idx, i1 %c) {
49 ; CHECK-LABEL: @inbounds_poison_is_ub_in_use_block_2(
51 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
52 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
53 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
54 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
55 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
56 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
58 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
59 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
60 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
62 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
63 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
66 %n.ext = zext i32 %n to i64
67 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
68 %cmp.idx = icmp ult i32 %idx, %n
69 %idx.ext = zext i32 %idx to i64
70 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
71 br i1 %cmp.idx, label %then, label %else
74 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
75 call void @noundef(i32* %upper)
79 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
83 declare void @llvm.assume(i1)
85 ; %start + %n.ext is guaranteed to not overflow (due to inbounds).
86 ; %start + %idx.ext does not overflow if %idx.ext <= %n.ext.
87 define i1 @inbounds_poison_is_ub_in_use_block_by_assume(i32* %src, i32 %n, i32 %idx) {
88 ; CHECK-LABEL: @inbounds_poison_is_ub_in_use_block_by_assume(
90 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
91 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
92 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
93 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
94 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
95 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
97 ; CHECK-NEXT: [[CMP_NE:%.*]] = icmp ule i32* null, [[UPPER]]
98 ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_NE]])
99 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
100 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
102 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
103 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
106 %n.ext = zext i32 %n to i64
107 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
108 %cmp.idx = icmp ult i32 %idx, %n
109 %idx.ext = zext i32 %idx to i64
110 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
111 br i1 %cmp.idx, label %then, label %else
114 %cmp.ne = icmp ule i32* null, %upper
115 call void @llvm.assume(i1 %cmp.ne)
116 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
120 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
125 ; %start + %n.ext is guaranteed to not overflow (due to inbounds).
126 ; %start + %idx.ext does not overflow if %idx.ext <= %n.ext.
127 define i1 @inbounds_poison_is_ub_in_in_multiple_use_blocks_1(i32* %src, i32 %n, i32 %idx, i1 %c) {
128 ; CHECK-LABEL: @inbounds_poison_is_ub_in_in_multiple_use_blocks_1(
130 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
131 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
132 ; CHECK-NEXT: br i1 [[C:%.*]], label [[CHECK_BB:%.*]], label [[EXIT:%.*]]
134 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
135 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
136 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
137 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
138 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
140 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
141 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
142 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
144 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
145 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
147 ; CHECK-NEXT: ret i1 false
150 %n.ext = zext i32 %n to i64
151 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
152 br i1 %c, label %check.bb, label %exit
155 call void @noundef(i32* %upper)
156 %cmp.idx = icmp ult i32 %idx, %n
157 %idx.ext = zext i32 %idx to i64
158 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
159 br i1 %cmp.idx, label %then, label %else
162 call void @noundef(i32* %upper)
163 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
167 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
174 define i1 @may_exit_before_ub_is_caused(i32* %src, i32 %n, i32 %idx) {
175 ; CHECK-LABEL: @may_exit_before_ub_is_caused(
177 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
178 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
179 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
180 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
181 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
182 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
184 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
185 ; CHECK-NEXT: call void @use(i1 [[CMP_UPPER_1]])
186 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
187 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
189 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
190 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
193 %n.ext = zext i32 %n to i64
194 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
195 %cmp.idx = icmp ult i32 %idx, %n
196 %idx.ext = zext i32 %idx to i64
197 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
198 br i1 %cmp.idx, label %then, label %else
201 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
202 call void @use(i1 %cmp.upper.1);
203 call void @noundef(i32* %upper)
207 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
211 define i1 @only_UB_in_false_block(i32* %src, i32 %n, i32 %idx) {
212 ; CHECK-LABEL: @only_UB_in_false_block(
214 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
215 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
216 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
217 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
218 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
219 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
221 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
222 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
224 ; CHECK-NEXT: call void @noundef(i32* [[UPPER]])
225 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
226 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
229 %n.ext = zext i32 %n to i64
230 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
231 %cmp.idx = icmp ult i32 %idx, %n
232 %idx.ext = zext i32 %idx to i64
233 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
234 br i1 %cmp.idx, label %then, label %else
237 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
241 call void @noundef(i32* %upper)
242 %cmp.upper.2 = icmp ule i32* %src.idx, %upper
246 define i1 @only_ub_by_assume_in_false_block(i32* %src, i32 %n, i32 %idx) {
247 ; CHECK-LABEL: @only_ub_by_assume_in_false_block(
249 ; CHECK-NEXT: [[N_EXT:%.*]] = zext i32 [[N:%.*]] to i64
250 ; CHECK-NEXT: [[UPPER:%.*]] = getelementptr inbounds i32, i32* [[SRC:%.*]], i64 [[N_EXT]]
251 ; CHECK-NEXT: [[CMP_IDX:%.*]] = icmp ult i32 [[IDX:%.*]], [[N]]
252 ; CHECK-NEXT: [[IDX_EXT:%.*]] = zext i32 [[IDX]] to i64
253 ; CHECK-NEXT: [[SRC_IDX:%.*]] = getelementptr i32, i32* [[SRC]], i64 [[IDX_EXT]]
254 ; CHECK-NEXT: br i1 [[CMP_IDX]], label [[THEN:%.*]], label [[ELSE:%.*]]
256 ; CHECK-NEXT: [[CMP_UPPER_1:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
257 ; CHECK-NEXT: ret i1 [[CMP_UPPER_1]]
259 ; CHECK-NEXT: [[CMP_NE:%.*]] = icmp ule i32* null, [[UPPER]]
260 ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP_NE]])
261 ; CHECK-NEXT: [[CMP_UPPER_2:%.*]] = icmp ule i32* [[SRC_IDX]], [[UPPER]]
262 ; CHECK-NEXT: ret i1 [[CMP_UPPER_2]]
265 %n.ext = zext i32 %n to i64
266 %upper = getelementptr inbounds i32, i32* %src, i64 %n.ext
267 %cmp.idx = icmp ult i32 %idx, %n
268 %idx.ext = zext i32 %idx to i64
269 %src.idx = getelementptr i32, i32* %src, i64 %idx.ext
270 br i1 %cmp.idx, label %then, label %else
273 %cmp.upper.1 = icmp ule i32* %src.idx, %upper
277 %cmp.ne = icmp ule i32* null, %upper
278 call void @llvm.assume(i1 %cmp.ne)
279 %cmp.upper.2 = icmp ule i32* %src.idx, %upper