[clang][modules] Don't prevent translation of FW_Private includes when explicitly...
[llvm-project.git] / llvm / test / Transforms / SimplifyCFG / assume.ll
blobeaebf376a9d72f413bd7ae56a42fbed03b37df7a
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -passes=simplifycfg -simplifycfg-require-and-preserve-domtree=1 -S < %s | FileCheck %s
4 define void @assume_false_to_unreachable1() {
5 ; CHECK-LABEL: @assume_false_to_unreachable1(
6 ; CHECK-NEXT:    unreachable
8   call void @llvm.assume(i1 0)
9   ret void
13 define void @assume_undef_to_unreachable() {
14 ; CHECK-LABEL: @assume_undef_to_unreachable(
15 ; CHECK-NEXT:    unreachable
17   call void @llvm.assume(i1 undef)
18   ret void
22 define i32 @speculate_block_with_assume_basic(i1 %c, i32 %x) {
23 ; CHECK-LABEL: @speculate_block_with_assume_basic(
24 ; CHECK-NEXT:  entry:
25 ; CHECK-NEXT:    [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 1, i32 0
26 ; CHECK-NEXT:    ret i32 [[SPEC_SELECT]]
28 entry:
29   br i1 %c, label %if, label %join
31 if:
32   %cmp = icmp ne i32 %x, 0
33   call void @llvm.assume(i1 %cmp)
34   br label %join
36 join:
37   %phi = phi i32 [ 0, %entry ], [ 1, %if ]
38   ret i32 %phi
41 define i32 @speculate_block_with_assume_extra_instr(i1 %c, i32 %x) {
42 ; CHECK-LABEL: @speculate_block_with_assume_extra_instr(
43 ; CHECK-NEXT:  entry:
44 ; CHECK-NEXT:    [[ADD:%.*]] = add i32 [[X:%.*]], 1
45 ; CHECK-NEXT:    [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 [[ADD]], i32 0
46 ; CHECK-NEXT:    ret i32 [[SPEC_SELECT]]
48 entry:
49   br i1 %c, label %if, label %join
51 if:
52   %add = add i32 %x, 1
53   %cmp = icmp ne i32 %add, 0
54   call void @llvm.assume(i1 %cmp)
55   br label %join
57 join:
58   %phi = phi i32 [ 0, %entry ], [ %add, %if ]
59   ret i32 %phi
62 ; We only allow speculating one instruction. Here %add and %add2 are used by
63 ; the assume, but not ephemeral, because they are also used by %phi.
64 define i32 @speculate_block_with_assume_extra_instrs_too_many(i1 %c, i32 %x) {
65 ; CHECK-LABEL: @speculate_block_with_assume_extra_instrs_too_many(
66 ; CHECK-NEXT:  entry:
67 ; CHECK-NEXT:    br i1 [[C:%.*]], label [[IF:%.*]], label [[JOIN:%.*]]
68 ; CHECK:       if:
69 ; CHECK-NEXT:    [[ADD:%.*]] = add i32 [[X:%.*]], 1
70 ; CHECK-NEXT:    [[ADD2:%.*]] = add i32 [[ADD]], 1
71 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ne i32 [[ADD2]], 0
72 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
73 ; CHECK-NEXT:    br label [[JOIN]]
74 ; CHECK:       join:
75 ; CHECK-NEXT:    [[PHI:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[ADD2]], [[IF]] ]
76 ; CHECK-NEXT:    ret i32 [[PHI]]
78 entry:
79   br i1 %c, label %if, label %join
81 if:
82   %add = add i32 %x, 1
83   %add2 = add i32 %add, 1
84   %cmp = icmp ne i32 %add2, 0
85   call void @llvm.assume(i1 %cmp)
86   br label %join
88 join:
89   %phi = phi i32 [ 0, %entry ], [ %add2, %if ]
90   ret i32 %phi
93 define i32 @speculate_block_with_assume_extra_instrs_okay(i1 %c, i32 %x) {
94 ; CHECK-LABEL: @speculate_block_with_assume_extra_instrs_okay(
95 ; CHECK-NEXT:  entry:
96 ; CHECK-NEXT:    [[ADD:%.*]] = add i32 [[X:%.*]], 1
97 ; CHECK-NEXT:    [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 [[ADD]], i32 0
98 ; CHECK-NEXT:    ret i32 [[SPEC_SELECT]]
100 entry:
101   br i1 %c, label %if, label %join
104   %add = add i32 %x, 1
105   %add2 = add i32 %add, 1
106   %cmp = icmp ne i32 %add2, 0
107   call void @llvm.assume(i1 %cmp)
108   br label %join
110 join:
111   %phi = phi i32 [ 0, %entry ], [ %add, %if ]
112   ret i32 %phi
115 define i32 @speculate_block_with_assume_operand_bundle(i1 %c, ptr %p) {
116 ; CHECK-LABEL: @speculate_block_with_assume_operand_bundle(
117 ; CHECK-NEXT:  entry:
118 ; CHECK-NEXT:    [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 1, i32 0
119 ; CHECK-NEXT:    ret i32 [[SPEC_SELECT]]
121 entry:
122   br i1 %c, label %if, label %join
125   call void @llvm.assume(i1 true) ["nonnull"(ptr %p)]
126   br label %join
128 join:
129   %phi = phi i32 [ 0, %entry ], [ 1, %if ]
130   ret i32 %phi
133 define void @empty_block_with_assume(i1 %c, i32 %x) {
134 ; CHECK-LABEL: @empty_block_with_assume(
135 ; CHECK-NEXT:  entry:
136 ; CHECK-NEXT:    br i1 [[C:%.*]], label [[IF:%.*]], label [[ELSE:%.*]]
137 ; CHECK:       if:
138 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ne i32 [[X:%.*]], 0
139 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
140 ; CHECK-NEXT:    br label [[JOIN:%.*]]
141 ; CHECK:       else:
142 ; CHECK-NEXT:    call void @dummy()
143 ; CHECK-NEXT:    br label [[JOIN]]
144 ; CHECK:       join:
145 ; CHECK-NEXT:    ret void
147 entry:
148   br i1 %c, label %if, label %else
151   %cmp = icmp ne i32 %x, 0
152   call void @llvm.assume(i1 %cmp)
153   br label %join
155 else:
156   call void @dummy()
157   br label %join
159 join:
160   ret void
163 define void @not_empty_block_with_assume(i1 %c) {
164 ; CHECK-LABEL: @not_empty_block_with_assume(
165 ; CHECK-NEXT:  entry:
166 ; CHECK-NEXT:    br i1 [[C:%.*]], label [[IF:%.*]], label [[ELSE:%.*]]
167 ; CHECK:       if:
168 ; CHECK-NEXT:    [[X:%.*]] = call i32 @may_have_side_effect()
169 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ne i32 [[X]], 0
170 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
171 ; CHECK-NEXT:    br label [[JOIN:%.*]]
172 ; CHECK:       else:
173 ; CHECK-NEXT:    call void @dummy()
174 ; CHECK-NEXT:    br label [[JOIN]]
175 ; CHECK:       join:
176 ; CHECK-NEXT:    ret void
178 entry:
179   br i1 %c, label %if, label %else
182   %x = call i32 @may_have_side_effect()
183   %cmp = icmp ne i32 %x, 0
184   call void @llvm.assume(i1 %cmp)
185   br label %join
187 else:
188   call void @dummy()
189   br label %join
191 join:
192   ret void
195 declare void @dummy()
196 declare i32 @may_have_side_effect()
197 declare void @llvm.assume(i1) nounwind