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}