xref: /aosp_15_r20/external/llvm/test/Transforms/DeadArgElim/keepalive.ll (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker; RUN: opt < %s -deadargelim -S | FileCheck %s
2*9880d681SAndroid Build Coastguard Worker
3*9880d681SAndroid Build Coastguard Worker%Ty = type <{ i32, i32 }>
4*9880d681SAndroid Build Coastguard Worker
5*9880d681SAndroid Build Coastguard Worker; Check if the pass doesn't modify anything that doesn't need changing. We feed
6*9880d681SAndroid Build Coastguard Worker; an unused argument to each function to lure it into changing _something_ about
7*9880d681SAndroid Build Coastguard Worker; the function and then changing too much.
8*9880d681SAndroid Build Coastguard Worker
9*9880d681SAndroid Build Coastguard Worker; This checks if the return value attributes are not removed
10*9880d681SAndroid Build Coastguard Worker; CHECK: define internal zeroext i32 @test1() #0
11*9880d681SAndroid Build Coastguard Workerdefine internal zeroext i32 @test1(i32 %DEADARG1) nounwind {
12*9880d681SAndroid Build Coastguard Worker        ret i32 1
13*9880d681SAndroid Build Coastguard Worker}
14*9880d681SAndroid Build Coastguard Worker
15*9880d681SAndroid Build Coastguard Worker; This checks if the struct doesn't get non-packed
16*9880d681SAndroid Build Coastguard Worker; CHECK-LABEL: define internal <{ i32, i32 }> @test2(
17*9880d681SAndroid Build Coastguard Workerdefine internal <{ i32, i32 }> @test2(i32 %DEADARG1) {
18*9880d681SAndroid Build Coastguard Worker        ret <{ i32, i32 }> <{ i32 1, i32 2 }>
19*9880d681SAndroid Build Coastguard Worker}
20*9880d681SAndroid Build Coastguard Worker
21*9880d681SAndroid Build Coastguard Worker; We use this external function to make sure the return values don't become dead
22*9880d681SAndroid Build Coastguard Workerdeclare void @user(i32, <{ i32, i32 }>)
23*9880d681SAndroid Build Coastguard Worker
24*9880d681SAndroid Build Coastguard Workerdefine void @caller() {
25*9880d681SAndroid Build Coastguard Worker        %B = call i32 @test1(i32 1)
26*9880d681SAndroid Build Coastguard Worker        %C = call <{ i32, i32 }> @test2(i32 2)
27*9880d681SAndroid Build Coastguard Worker        call void @user(i32 %B, <{ i32, i32 }> %C)
28*9880d681SAndroid Build Coastguard Worker        ret void
29*9880d681SAndroid Build Coastguard Worker}
30*9880d681SAndroid Build Coastguard Worker
31*9880d681SAndroid Build Coastguard Worker; We can't remove 'this' here, as that would put argmem in ecx instead of
32*9880d681SAndroid Build Coastguard Worker; memory.
33*9880d681SAndroid Build Coastguard Workerdefine internal x86_thiscallcc i32 @unused_this(i32* %this, i32* inalloca %argmem) {
34*9880d681SAndroid Build Coastguard Worker	%v = load i32, i32* %argmem
35*9880d681SAndroid Build Coastguard Worker	ret i32 %v
36*9880d681SAndroid Build Coastguard Worker}
37*9880d681SAndroid Build Coastguard Worker; CHECK-LABEL: define internal x86_thiscallcc i32 @unused_this(i32* %this, i32* inalloca %argmem)
38*9880d681SAndroid Build Coastguard Worker
39*9880d681SAndroid Build Coastguard Workerdefine i32 @caller2() {
40*9880d681SAndroid Build Coastguard Worker	%t = alloca i32
41*9880d681SAndroid Build Coastguard Worker	%m = alloca inalloca i32
42*9880d681SAndroid Build Coastguard Worker	store i32 42, i32* %m
43*9880d681SAndroid Build Coastguard Worker	%v = call x86_thiscallcc i32 @unused_this(i32* %t, i32* inalloca %m)
44*9880d681SAndroid Build Coastguard Worker	ret i32 %v
45*9880d681SAndroid Build Coastguard Worker}
46*9880d681SAndroid Build Coastguard Worker
47*9880d681SAndroid Build Coastguard Worker; CHECK: attributes #0 = { nounwind }
48