1*9880d681SAndroid Build Coastguard Worker; This testcase ensures that CFL AA answers queries soundly when callee tries
2*9880d681SAndroid Build Coastguard Worker; to return the multi-level reference of one of its parameters
3*9880d681SAndroid Build Coastguard Worker
4*9880d681SAndroid Build Coastguard Worker; RUN: opt < %s -disable-basicaa -cfl-steens-aa -aa-eval -print-all-alias-modref-info -disable-output 2>&1 | FileCheck %s
5*9880d681SAndroid Build Coastguard Worker; RUN: opt < %s -aa-pipeline=cfl-steens-aa -passes=aa-eval -print-all-alias-modref-info -disable-output 2>&1 | FileCheck %s
6*9880d681SAndroid Build Coastguard Worker
7*9880d681SAndroid Build Coastguard Workerdeclare noalias i8* @malloc(i64)
8*9880d681SAndroid Build Coastguard Worker
9*9880d681SAndroid Build Coastguard Workerdefine i32*** @return_ref_arg_multilevel_callee(i32* %arg1) {
10*9880d681SAndroid Build Coastguard Worker	%ptr = call noalias i8* @malloc(i64 8)
11*9880d681SAndroid Build Coastguard Worker	%ptr_cast = bitcast i8* %ptr to i32***
12*9880d681SAndroid Build Coastguard Worker  %ptr2 = call noalias i8* @malloc(i64 8)
13*9880d681SAndroid Build Coastguard Worker  %ptr_cast2 = bitcast i8* %ptr2 to i32**
14*9880d681SAndroid Build Coastguard Worker	store i32* %arg1, i32** %ptr_cast2
15*9880d681SAndroid Build Coastguard Worker  store i32** %ptr_cast2, i32*** %ptr_cast
16*9880d681SAndroid Build Coastguard Worker	ret i32*** %ptr_cast
17*9880d681SAndroid Build Coastguard Worker}
18*9880d681SAndroid Build Coastguard Worker; CHECK-LABEL: Function: test_return_ref_arg_multilevel
19*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32* %a, i32*** %b
20*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32** %p, i32*** %b
21*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32* %a, i32** %lb
22*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32** %lb, i32*** %pp
23*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32** %lb, i32*** %b
24*9880d681SAndroid Build Coastguard Worker; CHECK: MayAlias: i32* %a, i32* %lb_deref
25*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32* %lb_deref, i32** %lpp
26*9880d681SAndroid Build Coastguard Worker; CHECK: MayAlias: i32* %lb_deref, i32* %lpp_deref
27*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32* %lpp_deref, i32** %lpp
28*9880d681SAndroid Build Coastguard Worker; CHECK: MayAlias: i32* %lb_deref, i32* %lp
29*9880d681SAndroid Build Coastguard Worker; CHECK: NoAlias: i32* %lp, i32** %lpp
30*9880d681SAndroid Build Coastguard Worker; CHECK: MayAlias: i32* %lp, i32* %lpp_deref
31*9880d681SAndroid Build Coastguard Worker
32*9880d681SAndroid Build Coastguard Worker; We could've proven the following facts if the analysis were inclusion-based:
33*9880d681SAndroid Build Coastguard Worker; NoAlias: i32*** %b, i32*** %pp
34*9880d681SAndroid Build Coastguard Worker; NoAlias: i32** %lb, i32** %p
35*9880d681SAndroid Build Coastguard Workerdefine void @test_return_ref_arg_multilevel() {
36*9880d681SAndroid Build Coastguard Worker  %a = alloca i32, align 4
37*9880d681SAndroid Build Coastguard Worker  %p = alloca i32*, align 8
38*9880d681SAndroid Build Coastguard Worker  %pp = alloca i32**, align 8
39*9880d681SAndroid Build Coastguard Worker
40*9880d681SAndroid Build Coastguard Worker  store i32* %a, i32** %p
41*9880d681SAndroid Build Coastguard Worker  store i32** %p, i32*** %pp
42*9880d681SAndroid Build Coastguard Worker  %b = call i32*** @return_ref_arg_multilevel_callee(i32* %a)
43*9880d681SAndroid Build Coastguard Worker
44*9880d681SAndroid Build Coastguard Worker  %lb = load i32**, i32*** %b
45*9880d681SAndroid Build Coastguard Worker  %lb_deref = load i32*, i32** %lb
46*9880d681SAndroid Build Coastguard Worker  %lpp = load i32**, i32*** %pp
47*9880d681SAndroid Build Coastguard Worker  %lpp_deref = load i32*, i32** %lpp
48*9880d681SAndroid Build Coastguard Worker  %lp = load i32*, i32** %p
49*9880d681SAndroid Build Coastguard Worker
50*9880d681SAndroid Build Coastguard Worker  ret void
51*9880d681SAndroid Build Coastguard Worker}