; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --function-signature --check-attributes --check-globals ; RUN: opt -aa-pipeline=basic-aa -passes=attributor -attributor-manifest-internal -attributor-annotate-decl-cs -S < %s | FileCheck %s --check-prefixes=CHECK,TUNIT ; RUN: opt -aa-pipeline=basic-aa -passes=attributor-cgscc -attributor-manifest-internal -attributor-annotate-decl-cs -S < %s | FileCheck %s --check-prefixes=CHECK,CGSCC ; FIXME: Figure out why we need 16 iterations here. declare void @deref_phi_user(ptr %a); ; TEST 1 ; take mininimum of return values ; ;. ; CHECK: @[[G:[a-zA-Z0-9_$"\\.-]+]] = global i64 0 ;. define ptr @test1(ptr dereferenceable(4) %0, ptr dereferenceable(8) %1, i1 zeroext %2) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test1 ; CHECK-SAME: (ptr nofree noundef nonnull readnone dereferenceable(4) "no-capture-maybe-returned" [[TMP0:%.*]], ptr nofree noundef nonnull readnone dereferenceable(8) "no-capture-maybe-returned" [[TMP1:%.*]], i1 zeroext [[TMP2:%.*]]) local_unnamed_addr #[[ATTR0:[0-9]+]] { ; CHECK-NEXT: [[TMP4:%.*]] = select i1 [[TMP2]], ptr [[TMP0]], ptr [[TMP1]] ; CHECK-NEXT: ret ptr [[TMP4]] ; %4 = select i1 %2, ptr %0, ptr %1 ret ptr %4 } ; TEST 2 define ptr @test2(ptr dereferenceable_or_null(4) %0, ptr dereferenceable(8) %1, i1 zeroext %2) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test2 ; CHECK-SAME: (ptr nofree noundef readnone dereferenceable_or_null(4) "no-capture-maybe-returned" [[TMP0:%.*]], ptr nofree noundef nonnull readnone dereferenceable(8) "no-capture-maybe-returned" [[TMP1:%.*]], i1 zeroext [[TMP2:%.*]]) local_unnamed_addr #[[ATTR0]] { ; CHECK-NEXT: [[TMP4:%.*]] = select i1 [[TMP2]], ptr [[TMP0]], ptr [[TMP1]] ; CHECK-NEXT: ret ptr [[TMP4]] ; %4 = select i1 %2, ptr %0, ptr %1 ret ptr %4 } ; TEST 3 ; GEP inbounds define ptr @test3_1(ptr dereferenceable(8) %0) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test3_1 ; CHECK-SAME: (ptr nofree noundef nonnull readnone dereferenceable(8) "no-capture-maybe-returned" [[TMP0:%.*]]) local_unnamed_addr #[[ATTR0]] { ; CHECK-NEXT: [[RET:%.*]] = getelementptr inbounds i32, ptr [[TMP0]], i64 1 ; CHECK-NEXT: ret ptr [[RET]] ; %ret = getelementptr inbounds i32, ptr %0, i64 1 ret ptr %ret } define ptr @test3_2(ptr dereferenceable_or_null(32) %0) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test3_2 ; CHECK-SAME: (ptr nofree noundef readnone dereferenceable_or_null(32) "no-capture-maybe-returned" [[TMP0:%.*]]) local_unnamed_addr #[[ATTR0]] { ; CHECK-NEXT: [[RET:%.*]] = getelementptr inbounds i32, ptr [[TMP0]], i64 4 ; CHECK-NEXT: ret ptr [[RET]] ; %ret = getelementptr inbounds i32, ptr %0, i64 4 ret ptr %ret } define ptr @test3_3(ptr dereferenceable(8) %0, ptr dereferenceable(16) %1, i1 %2) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test3_3 ; CHECK-SAME: (ptr nofree noundef nonnull readnone dereferenceable(8) "no-capture-maybe-returned" [[TMP0:%.*]], ptr nofree noundef nonnull readnone dereferenceable(16) "no-capture-maybe-returned" [[TMP1:%.*]], i1 [[TMP2:%.*]]) local_unnamed_addr #[[ATTR0]] { ; CHECK-NEXT: [[RET1:%.*]] = getelementptr inbounds i32, ptr [[TMP0]], i64 1 ; CHECK-NEXT: [[RET2:%.*]] = getelementptr inbounds i32, ptr [[TMP1]], i64 2 ; CHECK-NEXT: [[RET:%.*]] = select i1 [[TMP2]], ptr [[RET1]], ptr [[RET2]] ; CHECK-NEXT: ret ptr [[RET]] ; %ret1 = getelementptr inbounds i32, ptr %0, i64 1 %ret2 = getelementptr inbounds i32, ptr %1, i64 2 %ret = select i1 %2, ptr %ret1, ptr %ret2 ret ptr %ret } ; TEST 4 ; Better than known in IR. define dereferenceable(4) ptr @test4(ptr dereferenceable(8) %0) local_unnamed_addr { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(none) ; CHECK-LABEL: define {{[^@]+}}@test4 ; CHECK-SAME: (ptr nofree noundef nonnull readnone returned dereferenceable(8) "no-capture-maybe-returned" [[TMP0:%.*]]) local_unnamed_addr #[[ATTR0]] { ; CHECK-NEXT: ret ptr [[TMP0]] ; ret ptr %0 } ; TEST 5 ; loop in which dereferenceabily "grows" define void @deref_phi_growing(ptr dereferenceable(4000) %a) { ; CHECK-LABEL: define {{[^@]+}}@deref_phi_growing ; CHECK-SAME: (ptr noundef nonnull dereferenceable(4000) [[A:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[FOR_COND:%.*]] ; CHECK: for.cond: ; CHECK-NEXT: [[I_0:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[INC:%.*]], [[FOR_INC:%.*]] ] ; CHECK-NEXT: [[A_ADDR_0:%.*]] = phi ptr [ [[A]], [[ENTRY]] ], [ [[INCDEC_PTR:%.*]], [[FOR_INC]] ] ; CHECK-NEXT: call void @deref_phi_user(ptr nonnull [[A_ADDR_0]]) ; CHECK-NEXT: [[TMP:%.*]] = load i32, ptr [[A_ADDR_0]], align 4 ; CHECK-NEXT: [[CMP:%.*]] = icmp slt i32 [[I_0]], [[TMP]] ; CHECK-NEXT: br i1 [[CMP]], label [[FOR_BODY:%.*]], label [[FOR_COND_CLEANUP:%.*]] ; CHECK: for.cond.cleanup: ; CHECK-NEXT: br label [[FOR_END:%.*]] ; CHECK: for.body: ; CHECK-NEXT: br label [[FOR_INC]] ; CHECK: for.inc: ; CHECK-NEXT: [[INCDEC_PTR]] = getelementptr inbounds i32, ptr [[A_ADDR_0]], i64 -1 ; CHECK-NEXT: [[INC]] = add nuw nsw i32 [[I_0]], 1 ; CHECK-NEXT: br label [[FOR_COND]] ; CHECK: for.end: ; CHECK-NEXT: ret void ; entry: br label %for.cond for.cond: ; preds = %for.inc, %entry %i.0 = phi i32 [ 0, %entry ], [ %inc, %for.inc ] %a.addr.0 = phi ptr [ %a, %entry ], [ %incdec.ptr, %for.inc ] call void @deref_phi_user(ptr %a.addr.0) %tmp = load i32, ptr %a.addr.0, align 4 %cmp = icmp slt i32 %i.0, %tmp br i1 %cmp, label %for.body, label %for.cond.cleanup for.cond.cleanup: ; preds = %for.cond br label %for.end for.body: ; preds = %for.cond br label %for.inc for.inc: ; preds = %for.body %incdec.ptr = getelementptr inbounds i32, ptr %a.addr.0, i64 -1 %inc = add nuw nsw i32 %i.0, 1 br label %for.cond for.end: ; preds = %for.cond.cleanup ret void } ; TEST 6 ; loop in which dereferenceabily "shrinks" define void @deref_phi_shrinking(ptr dereferenceable(4000) %a) { ; CHECK-LABEL: define {{[^@]+}}@deref_phi_shrinking ; CHECK-SAME: (ptr noundef nonnull dereferenceable(4000) [[A:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[FOR_COND:%.*]] ; CHECK: for.cond: ; CHECK-NEXT: [[I_0:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[INC:%.*]], [[FOR_INC:%.*]] ] ; CHECK-NEXT: [[A_ADDR_0:%.*]] = phi ptr [ [[A]], [[ENTRY]] ], [ [[INCDEC_PTR:%.*]], [[FOR_INC]] ] ; CHECK-NEXT: call void @deref_phi_user(ptr nonnull [[A_ADDR_0]]) ; CHECK-NEXT: [[TMP:%.*]] = load i32, ptr [[A_ADDR_0]], align 4 ; CHECK-NEXT: [[CMP:%.*]] = icmp slt i32 [[I_0]], [[TMP]] ; CHECK-NEXT: br i1 [[CMP]], label [[FOR_BODY:%.*]], label [[FOR_COND_CLEANUP:%.*]] ; CHECK: for.cond.cleanup: ; CHECK-NEXT: br label [[FOR_END:%.*]] ; CHECK: for.body: ; CHECK-NEXT: br label [[FOR_INC]] ; CHECK: for.inc: ; CHECK-NEXT: [[INCDEC_PTR]] = getelementptr inbounds i32, ptr [[A_ADDR_0]], i64 1 ; CHECK-NEXT: [[INC]] = add nuw nsw i32 [[I_0]], 1 ; CHECK-NEXT: br label [[FOR_COND]] ; CHECK: for.end: ; CHECK-NEXT: ret void ; entry: br label %for.cond for.cond: ; preds = %for.inc, %entry %i.0 = phi i32 [ 0, %entry ], [ %inc, %for.inc ] %a.addr.0 = phi ptr [ %a, %entry ], [ %incdec.ptr, %for.inc ] call void @deref_phi_user(ptr %a.addr.0) %tmp = load i32, ptr %a.addr.0, align 4 %cmp = icmp slt i32 %i.0, %tmp br i1 %cmp, label %for.body, label %for.cond.cleanup for.cond.cleanup: ; preds = %for.cond br label %for.end for.body: ; preds = %for.cond br label %for.inc for.inc: ; preds = %for.body %incdec.ptr = getelementptr inbounds i32, ptr %a.addr.0, i64 1 %inc = add nuw nsw i32 %i.0, 1 br label %for.cond for.end: ; preds = %for.cond.cleanup ret void } ; TEST 7 ; share known infomation in must-be-executed-context declare ptr @unkown_ptr() willreturn nounwind declare i32 @unkown_f(ptr) willreturn nounwind define ptr @f7_0(ptr %ptr) { ; CHECK: Function Attrs: mustprogress nounwind willreturn ; CHECK-LABEL: define {{[^@]+}}@f7_0 ; CHECK-SAME: (ptr noundef nonnull returned dereferenceable(8) [[PTR:%.*]]) #[[ATTR2:[0-9]+]] { ; CHECK-NEXT: [[T:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull dereferenceable(8) [[PTR]]) #[[ATTR1:[0-9]+]] ; CHECK-NEXT: ret ptr [[PTR]] ; %T = tail call i32 @unkown_f(ptr dereferenceable(8) %ptr) ret ptr %ptr } define void @f7_1(ptr %ptr, i1 %c) { ; CHECK: Function Attrs: mustprogress nounwind willreturn ; CHECK-LABEL: define {{[^@]+}}@f7_1 ; CHECK-SAME: (ptr noundef nonnull align 4 dereferenceable(4) [[PTR:%.*]], i1 noundef [[C:%.*]]) #[[ATTR2]] { ; CHECK-NEXT: [[A:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(4) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[PTR_0:%.*]] = load i32, ptr [[PTR]], align 4 ; CHECK-NEXT: [[B:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(4) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: br i1 [[C]], label [[IF_TRUE:%.*]], label [[IF_FALSE:%.*]] ; CHECK: if.true: ; CHECK-NEXT: [[C:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[D:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[E:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: ret void ; CHECK: if.false: ; CHECK-NEXT: ret void ; %A = tail call i32 @unkown_f(ptr %ptr) %ptr.0 = load i32, ptr %ptr ; deref 4 hold ; FIXME: this should be %B = tail call i32 @unkown_f(ptr nonnull dereferenceable(4) %ptr) %B = tail call i32 @unkown_f(ptr dereferenceable(1) %ptr) br i1%c, label %if.true, label %if.false if.true: %C = tail call i32 @unkown_f(ptr %ptr) %D = tail call i32 @unkown_f(ptr dereferenceable(8) %ptr) %E = tail call i32 @unkown_f(ptr %ptr) ret void if.false: ret void } define void @f7_2(i1 %c) { ; CHECK: Function Attrs: mustprogress nounwind willreturn ; CHECK-LABEL: define {{[^@]+}}@f7_2 ; CHECK-SAME: (i1 noundef [[C:%.*]]) #[[ATTR2]] { ; CHECK-NEXT: [[PTR:%.*]] = tail call nonnull align 4 dereferenceable(4) ptr @unkown_ptr() #[[ATTR1]] ; CHECK-NEXT: [[A:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(4) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[ARG_A_0:%.*]] = load i32, ptr [[PTR]], align 4 ; CHECK-NEXT: [[B:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(4) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: br i1 [[C]], label [[IF_TRUE:%.*]], label [[IF_FALSE:%.*]] ; CHECK: if.true: ; CHECK-NEXT: [[C:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[D:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: [[E:%.*]] = tail call i32 @unkown_f(ptr noundef nonnull align 4 dereferenceable(8) [[PTR]]) #[[ATTR1]] ; CHECK-NEXT: ret void ; CHECK: if.false: ; CHECK-NEXT: ret void ; %ptr = tail call ptr @unkown_ptr() %A = tail call i32 @unkown_f(ptr %ptr) %arg_a.0 = load i32, ptr %ptr ; deref 4 hold %B = tail call i32 @unkown_f(ptr dereferenceable(1) %ptr) br i1%c, label %if.true, label %if.false if.true: %C = tail call i32 @unkown_f(ptr %ptr) %D = tail call i32 @unkown_f(ptr dereferenceable(8) %ptr) %E = tail call i32 @unkown_f(ptr %ptr) ret void if.false: ret void } define ptr @f7_3() { ; CHECK: Function Attrs: mustprogress nounwind willreturn ; CHECK-LABEL: define {{[^@]+}}@f7_3 ; CHECK-SAME: () #[[ATTR2]] { ; CHECK-NEXT: [[PTR:%.*]] = tail call noundef nonnull align 16 dereferenceable(4) ptr @unkown_ptr() #[[ATTR1]] ; CHECK-NEXT: store i32 10, ptr [[PTR]], align 16 ; CHECK-NEXT: ret ptr [[PTR]] ; %ptr = tail call ptr @unkown_ptr() store i32 10, ptr %ptr, align 16 ret ptr %ptr } ; FIXME: This should have a return dereferenceable(8) but we need to make sure it will work in loops as well. define ptr @test_for_minus_index(ptr %p) { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(argmem: write) ; CHECK-LABEL: define {{[^@]+}}@test_for_minus_index ; CHECK-SAME: (ptr nofree nonnull writeonly align 4 "no-capture-maybe-returned" [[P:%.*]]) #[[ATTR3:[0-9]+]] { ; CHECK-NEXT: [[Q:%.*]] = getelementptr inbounds i32, ptr [[P]], i32 -2 ; CHECK-NEXT: store i32 1, ptr [[Q]], align 4 ; CHECK-NEXT: ret ptr [[Q]] ; %q = getelementptr inbounds i32, ptr %p, i32 -2 store i32 1, ptr %q ret ptr %q } define void @deref_or_null_and_nonnull(ptr dereferenceable_or_null(100) %0) { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(argmem: write) ; CHECK-LABEL: define {{[^@]+}}@deref_or_null_and_nonnull ; CHECK-SAME: (ptr nocapture nofree noundef nonnull writeonly align 4 dereferenceable(100) [[TMP0:%.*]]) #[[ATTR3]] { ; CHECK-NEXT: store i32 1, ptr [[TMP0]], align 4 ; CHECK-NEXT: ret void ; store i32 1, ptr %0 ret void } ; TEST 8 ; Use Constant range in deereferenceable ; void g(int *p, long long int *range){ ; int r = *range ; // [10, 99] ; fill_range(p, *range); ; } ; FIXME: %ptr should be dereferenceable(31) define void @test8(ptr %ptr) #0 { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(argmem: write) ; CHECK-LABEL: define {{[^@]+}}@test8 ; CHECK-SAME: (ptr nocapture nofree nonnull writeonly dereferenceable(21) [[PTR:%.*]]) #[[ATTR3]] { ; CHECK-NEXT: br label [[TMP1:%.*]] ; CHECK: 1: ; CHECK-NEXT: [[I_0:%.*]] = phi i32 [ 20, [[TMP0:%.*]] ], [ [[TMP4:%.*]], [[TMP5:%.*]] ] ; CHECK-NEXT: [[TMP2:%.*]] = sext i32 [[I_0]] to i64 ; CHECK-NEXT: [[TMP3:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[TMP2]] ; CHECK-NEXT: store i8 32, ptr [[TMP3]], align 1 ; CHECK-NEXT: [[TMP4]] = add nsw i32 [[I_0]], 1 ; CHECK-NEXT: br label [[TMP5]] ; CHECK: 5: ; CHECK-NEXT: [[TMP6:%.*]] = icmp slt i32 [[TMP4]], 30 ; CHECK-NEXT: br i1 [[TMP6]], label [[TMP1]], label [[TMP7:%.*]] ; CHECK: 7: ; CHECK-NEXT: ret void ; br label %1 1: ; preds = %5, %0 %i.0 = phi i32 [ 20, %0 ], [ %4, %5 ] %2 = sext i32 %i.0 to i64 %3 = getelementptr inbounds i8, ptr %ptr, i64 %2 store i8 32, ptr %3, align 1 %4 = add nsw i32 %i.0, 1 br label %5 5: ; preds = %1 %6 = icmp slt i32 %4, 30 br i1 %6, label %1, label %7 7: ; preds = %5 ret void } ; 8.2 (negative case) define void @test8_neg(i32 %i, ptr %ptr) #0 { ; CHECK: Function Attrs: mustprogress nofree norecurse nosync nounwind willreturn memory(argmem: write) ; CHECK-LABEL: define {{[^@]+}}@test8_neg ; CHECK-SAME: (i32 [[I:%.*]], ptr nocapture nofree nonnull writeonly [[PTR:%.*]]) #[[ATTR3]] { ; CHECK-NEXT: [[TMP1:%.*]] = sext i32 [[I]] to i64 ; CHECK-NEXT: [[TMP2:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[TMP1]] ; CHECK-NEXT: store i8 65, ptr [[TMP2]], align 1 ; CHECK-NEXT: ret void ; %1 = sext i32 %i to i64 %2 = getelementptr inbounds i8, ptr %ptr, i64 %1 store i8 65, ptr %2, align 1 ret void } ; void fill_range(int* p, long long int start){ ; for(long long int i = start;i