; RUN: opt < %s -aa-pipeline=basic-aa -passes=aa-eval -print-all-alias-modref-info -disable-output 2>&1 | FileCheck %s ; CHECK-LABEL: test_no_lower_bound ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @test_no_lower_bound(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p, i64 4 %b = getelementptr nuw i8, ptr %p, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: test_lower_bound_lt_size ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @test_lower_bound_lt_size(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p %add = getelementptr nuw i8, ptr %p, i64 2 %b = getelementptr nuw i8, ptr %add, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: test_lower_bound_ge_size ; ; CHECK-DAG: NoAlias: i32* %a, i32* %b define void @test_lower_bound_ge_size(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p %add = getelementptr nuw i8, ptr %p, i64 4 %b = getelementptr nuw i8, ptr %add, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: test_not_all_nuw ; ; If part of the addressing is done with non-nuw GEPs, we can't use properties ; implied by the last GEP with the whole offset. In this case, the calculation ; of %add (%p + 4) could wrap the pointer index type, such that %add + %i ; could still alias with %p. ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @test_not_all_nuw(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p %add = getelementptr i8, ptr %p, i64 4 %b = getelementptr nuw i8, ptr %add, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: test_multi_step_not_all_nuw ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @test_multi_step_not_all_nuw(ptr %p, i64 %i, i64 %j, i64 %k) { %a = getelementptr i8, ptr %p %add = getelementptr i8, ptr %p, i64 4 %step1 = getelementptr i8, ptr %add, i64 %i %step2 = getelementptr i8, ptr %step1, i64 %j %b = getelementptr nuw i8, ptr %step2, i64 %k load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: test_multi_step_all_nuw ; ; CHECK-DAG: NoAlias: i32* %a, i32* %b define void @test_multi_step_all_nuw(ptr %p, i64 %i, i64 %j, i64 %k) { %a = getelementptr i8, ptr %p %add = getelementptr nuw i8, ptr %p, i64 4 %step1 = getelementptr nuw i8, ptr %add, i64 %i %step2 = getelementptr nuw i8, ptr %step1, i64 %j %b = getelementptr nuw i8, ptr %step2, i64 %k load i32, ptr %a load i32, ptr %b ret void } %struct = type { i64, [2 x i32], i64 } ; CHECK-LABEL: test_struct_no_nuw ; ; The array access may alias with the struct elements before and after, because ; we cannot prove that (%arr + %i) does not alias with the base pointer %p. ; ; CHECK-DAG: MayAlias: i32* %arrayidx, i64* %st ; CHECK-DAG: NoAlias: i64* %after, i64* %st ; CHECK-DAG: MayAlias: i64* %after, i32* %arrayidx define void @test_struct_no_nuw(ptr %st, i64 %i) { %arr = getelementptr i8, ptr %st, i64 8 %arrayidx = getelementptr [2 x i32], ptr %arr, i64 0, i64 %i %after = getelementptr i8, ptr %st, i64 16 load i64, ptr %st load i32, ptr %arrayidx load i64, ptr %after ret void } ; CHECK-LABEL: test_struct_nuw ; ; We can prove that the array access does not alias with struct element before, ; because we can prove that (%arr + %i) does not wrap the pointer index ; type (add nuw). The array access may still alias with the struct element ; after, as the add nuw property does not preclude this. ; ; CHECK-DAG: NoAlias: i32* %arrayidx, i64* %st ; CHECK-DAG: NoAlias: i64* %after, i64* %st ; CHECK-DAG: MayAlias: i64* %after, i32* %arrayidx define void @test_struct_nuw(ptr %st, i64 %i) { %arr = getelementptr nuw i8, ptr %st, i64 8 %arrayidx = getelementptr nuw [2 x i32], ptr %arr, i64 0, i64 %i %after = getelementptr nuw i8, ptr %st, i64 16 load i64, ptr %st load i32, ptr %arrayidx load i64, ptr %after ret void } ; CHECK-LABEL: constant_offset_overflow ; ; If subtraction of constant offsets could overflow in an unsigned sense, we ; cannot prove the lower bound between the GEPs and so they may still alias. ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @constant_offset_overflow(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p, i64 -8 %add = getelementptr nuw i8, ptr %p, i64 4 %b = getelementptr nuw i8, ptr %add, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: equal_var_idx_noalias ; ; If GEPs have equal variable indices, we can prove NoAlias when the Scale of ; the RHS GEP is greater, as in this scenario the constant lower bound holds. ; ; CHECK-DAG: NoAlias: i32* %a, i32* %b define void @equal_var_idx_noalias(ptr %p, i64 %i) { %a = getelementptr i8, ptr %p, i64 %i %add = getelementptr nuw i8, ptr %p, i64 4 %b = getelementptr nuw i16, ptr %add, i64 %i load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: equal_var_idx_alias ; ; If GEPs have equal variable indices, we cannot prove NoAlias when the Scale of ; the RHS GEP is ult Scale of the LHS GEP. ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @equal_var_idx_alias(ptr %p, i64 %i) { %a = getelementptr i32, ptr %p, i64 %i %add = getelementptr nuw i8, ptr %p, i64 4 %b = getelementptr nuw i16, ptr %add, i64 %i load i32, ptr %b load i32, ptr %a ret void } ; CHECK-LABEL: both_var_idx ; ; If the RHS GEP has unmatched variable indices, we cannot prove a constant ; lower bound between GEPs. ; ; CHECK-DAG: MayAlias: i32* %a, i32* %b define void @both_var_idx(ptr %p, i64 %i, i64 %j) { %a = getelementptr i8, ptr %p, i64 %i %add = getelementptr nuw i8, ptr %p, i64 4 %b = getelementptr nuw i8, ptr %add, i64 %j load i32, ptr %a load i32, ptr %b ret void } ; CHECK-LABEL: add_no_nuw ; CHECK: MayAlias: i8* %gep, i8* %p define i8 @add_no_nuw(ptr %p, i64 %n) { store i8 3, ptr %p %add = add i64 %n, 1 %gep = getelementptr nuw i8, ptr %p, i64 %add %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: add_nuw ; CHECK: NoAlias: i8* %gep, i8* %p define i8 @add_nuw(ptr %p, i64 %n) { store i8 3, ptr %p %add = add nuw i64 %n, 1 %gep = getelementptr nuw i8, ptr %p, i64 %add %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: add_no_nuw ; CHECK: MayAlias: i8* %gep, i16* %p define i8 @add_no_nuw_scale(ptr %p, i64 %n) { store i16 3, ptr %p %add = add i64 %n, 1 %gep = getelementptr nuw i16, ptr %p, i64 %add %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: add_nuw ; CHECK: NoAlias: i8* %gep, i16* %p define i8 @add_nuw_scale(ptr %p, i64 %n) { store i16 3, ptr %p %add = add nuw i64 %n, 1 %gep = getelementptr nuw i16, ptr %p, i64 %add %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: sub_nuw ; CHECK: MayAlias: i8* %gep, i8* %p define i8 @sub_nuw(ptr %p, i64 %n) { store i8 3, ptr %p %add = sub nuw i64 %n, 1 %gep = getelementptr nuw i8, ptr %p, i64 %add %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: mul_no_nuw ; CHECK: MayAlias: i8* %gep, i16* %p define i8 @mul_no_nuw(ptr %p, i64 %n) { store i16 3, ptr %p %add = add nuw i64 %n, 1 %mul = mul i64 %add, 2 %gep = getelementptr nuw i8, ptr %p, i64 %mul %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: mul_nuw ; CHECK: NoAlias: i8* %gep, i16* %p define i8 @mul_nuw(ptr %p, i64 %n) { store i16 3, ptr %p %add = add nuw i64 %n, 1 %mul = mul nuw i64 %add, 2 %gep = getelementptr nuw i8, ptr %p, i64 %mul %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: shl_no_nuw ; CHECK: MayAlias: i8* %gep, i16* %p define i8 @shl_no_nuw(ptr %p, i64 %n) { store i16 3, ptr %p %add = add nuw i64 %n, 1 %shl = shl i64 %add, 1 %gep = getelementptr nuw i8, ptr %p, i64 %shl %val = load i8, ptr %gep ret i8 %val } ; CHECK-LABEL: shl_nuw ; CHECK: NoAlias: i8* %gep, i16* %p define i8 @shl_nuw(ptr %p, i64 %n) { store i16 3, ptr %p %add = add nuw i64 %n, 1 %shl = shl nuw i64 %add, 1 %gep = getelementptr nuw i8, ptr %p, i64 %shl %val = load i8, ptr %gep ret i8 %val }