| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ; RUN: opt < %s -passes=instcombine -S | FileCheck %s |
| |
| ; Passing select(cond, null, v) as nonnull should be optimized to passing v |
| |
| define nonnull ptr @pr48975(ptr %.0) { |
| ; CHECK-LABEL: @pr48975( |
| ; CHECK-NEXT: ret ptr [[DOT4:%.*]] |
| ; |
| %.1 = load ptr, ptr %.0, align 8 |
| %.2 = icmp eq ptr %.1, null |
| %.4 = select i1 %.2, ptr null, ptr %.0 |
| ret ptr %.4 |
| } |
| |
| define nonnull ptr @nonnull_ret(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_ret( |
| ; CHECK-NEXT: ret ptr [[RES:%.*]] |
| ; |
| %res = select i1 %cond, ptr %p, ptr null |
| ret ptr %res |
| } |
| |
| define nonnull ptr @nonnull_ret2(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_ret2( |
| ; CHECK-NEXT: ret ptr [[RES:%.*]] |
| ; |
| %res = select i1 %cond, ptr null, ptr %p |
| ret ptr %res |
| } |
| |
| define nonnull noundef ptr @nonnull_noundef_ret(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_noundef_ret( |
| ; CHECK-NEXT: ret ptr [[RES:%.*]] |
| ; |
| %res = select i1 %cond, ptr %p, ptr null |
| ret ptr %res |
| } |
| |
| define nonnull noundef ptr @nonnull_noundef_ret2(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_noundef_ret2( |
| ; CHECK-NEXT: ret ptr [[RES:%.*]] |
| ; |
| %res = select i1 %cond, ptr null, ptr %p |
| ret ptr %res |
| } |
| |
| |
| define void @nonnull_call(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_call( |
| ; CHECK-NEXT: call void @f(ptr nonnull [[RES:%.*]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %res = select i1 %cond, ptr %p, ptr null |
| call void @f(ptr nonnull %res) |
| ret void |
| } |
| |
| define void @nonnull_call2(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_call2( |
| ; CHECK-NEXT: call void @f(ptr nonnull [[RES:%.*]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %res = select i1 %cond, ptr null, ptr %p |
| call void @f(ptr nonnull %res) |
| ret void |
| } |
| |
| define void @nonnull_noundef_call(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_noundef_call( |
| ; CHECK-NEXT: call void @f(ptr noundef nonnull [[RES:%.*]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %res = select i1 %cond, ptr %p, ptr null |
| call void @f(ptr nonnull noundef %res) |
| ret void |
| } |
| |
| define void @nonnull_noundef_call2(i1 %cond, ptr %p) { |
| ; CHECK-LABEL: @nonnull_noundef_call2( |
| ; CHECK-NEXT: call void @f(ptr noundef nonnull [[RES:%.*]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %res = select i1 %cond, ptr null, ptr %p |
| call void @f(ptr nonnull noundef %res) |
| ret void |
| } |
| |
| define void @nonnull_call_gep(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_call_gep( |
| ; CHECK-NEXT: [[PTR:%.*]] = select i1 [[COND:%.*]], ptr null, ptr [[P:%.*]] |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr [[PTR]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: call void @f(ptr nonnull [[GEP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr i8, ptr %ptr, i64 %off |
| call void @f(ptr nonnull %gep) |
| ret void |
| } |
| |
| define void @nonnull_call_gep_multiuse(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_call_gep_multiuse( |
| ; CHECK-NEXT: [[PTR:%.*]] = select i1 [[COND:%.*]], ptr null, ptr [[P:%.*]] |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: call void @f(ptr nonnull [[GEP]]) |
| ; CHECK-NEXT: call void @f(ptr [[GEP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr inbounds i8, ptr %ptr, i64 %off |
| call void @f(ptr nonnull %gep) |
| call void @f(ptr %gep) |
| ret void |
| } |
| |
| define void @all_nonnull_call_gep_multiuse(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @all_nonnull_call_gep_multiuse( |
| ; CHECK-NEXT: [[PTR:%.*]] = select i1 [[COND:%.*]], ptr null, ptr [[P:%.*]] |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: call void @f(ptr nonnull [[GEP]]) |
| ; CHECK-NEXT: call void @f(ptr nonnull [[GEP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr inbounds i8, ptr %ptr, i64 %off |
| call void @f(ptr nonnull %gep) |
| call void @f(ptr nonnull %gep) |
| ret void |
| } |
| |
| define void @nonnull_call_gep_inbounds(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_call_gep_inbounds( |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr inbounds i8, ptr [[PTR:%.*]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: call void @f(ptr nonnull [[GEP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr inbounds i8, ptr %ptr, i64 %off |
| call void @f(ptr nonnull %gep) |
| ret void |
| } |
| |
| define void @nonnull_dereferenceable_call_gep(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_dereferenceable_call_gep( |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr [[PTR:%.*]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: call void @f(ptr dereferenceable(1) [[GEP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr i8, ptr %ptr, i64 %off |
| call void @f(ptr dereferenceable(1) %gep) |
| ret void |
| } |
| |
| define nonnull ptr @nonnull_ret_gep(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_ret_gep( |
| ; CHECK-NEXT: [[PTR:%.*]] = select i1 [[COND:%.*]], ptr null, ptr [[P:%.*]] |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr [[PTR]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: ret ptr [[GEP]] |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr i8, ptr %ptr, i64 %off |
| ret ptr %gep |
| } |
| |
| define nonnull ptr @nonnull_ret_gep_inbounds(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_ret_gep_inbounds( |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr inbounds i8, ptr [[PTR:%.*]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: ret ptr [[GEP]] |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr inbounds i8, ptr %ptr, i64 %off |
| ret ptr %gep |
| } |
| |
| define dereferenceable(1) ptr @nonnull_dereferenceable_ret_gep(i1 %cond, ptr %p, i64 %off) { |
| ; CHECK-LABEL: @nonnull_dereferenceable_ret_gep( |
| ; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr [[PTR:%.*]], i64 [[OFF:%.*]] |
| ; CHECK-NEXT: ret ptr [[GEP]] |
| ; |
| %ptr = select i1 %cond, ptr null, ptr %p |
| %gep = getelementptr i8, ptr %ptr, i64 %off |
| ret ptr %gep |
| } |
| |
| declare void @f(ptr) |