[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / test / Transforms / TailCallElim / tre-multiple-exits.ll
blob8f69087dd879dbfb079290f01989f4b5e5b40005
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt < %s -tailcallelim -verify-dom-info -S | FileCheck %s
4 ; This test checks that TRE would be done for only one recursive call.
5 ; The test_multiple_exits function has three recursive calls.
6 ; First recursive call could not be eliminated because there is
7 ; escaped pointer to local variable. Second recursive call could
8 ; be eliminated. Thrid recursive call could not be eliminated since
9 ; this is not last call. Thus, test checks that TRE would be done
10 ; for only second recursive call.
12 ; IR for that test was generated from the following C++ source:
14 ; void capture_arg (int*);
15 ; void test_multiple_exits (int param);
16 ;   if (param >= 0 && param < 10) {
17 ;     int temp;
18 ;     capture_arg(&temp);
19 ;     // TRE could not be done because pointer to local
20 ;     // variable "temp" is escaped.
21 ;     test_multiple_exits(param + 1);
22 ;   } else if (param >=10 && param < 20) {
23 ;     // TRE should be done.
24 ;     test_multiple_exits(param + 1);
25 ;   } else if (param >= 20 && param < 22) {
26 ;     // TRE could not be done since recursive
27 ;     // call is not last call.
28 ;     test_multiple_exits(param + 1);
29 ;     func();
30 ;   }
32 ;   return;
33 ; }
35 ; Function Attrs: noinline optnone uwtable
36 declare void @_Z11capture_argPi(i32* %param) #0
38 ; Function Attrs: noinline optnone uwtable
39 declare void @_Z4funcv() #0
41 ; Function Attrs: noinline nounwind uwtable
42 define dso_local void @_Z19test_multiple_exitsi(i32 %param) local_unnamed_addr #2 {
43 ; CHECK-LABEL: @_Z19test_multiple_exitsi(
44 ; CHECK-NEXT:  entry:
45 ; CHECK-NEXT:    [[TEMP:%.*]] = alloca i32, align 4
46 ; CHECK-NEXT:    br label [[TAILRECURSE:%.*]]
47 ; CHECK:       tailrecurse:
48 ; CHECK-NEXT:    [[PARAM_TR:%.*]] = phi i32 [ [[PARAM:%.*]], [[ENTRY:%.*]] ], [ [[ADD6:%.*]], [[IF_THEN5:%.*]] ]
49 ; CHECK-NEXT:    [[TMP0:%.*]] = icmp ult i32 [[PARAM_TR]], 10
50 ; CHECK-NEXT:    br i1 [[TMP0]], label [[IF_THEN:%.*]], label [[IF_ELSE:%.*]]
51 ; CHECK:       if.then:
52 ; CHECK-NEXT:    [[TMP1:%.*]] = bitcast i32* [[TEMP]] to i8*
53 ; CHECK-NEXT:    call void @llvm.lifetime.start.p0i8(i64 4, i8* nonnull [[TMP1]]) #1
54 ; CHECK-NEXT:    call void @_Z11capture_argPi(i32* nonnull [[TEMP]])
55 ; CHECK-NEXT:    [[ADD:%.*]] = add nuw nsw i32 [[PARAM_TR]], 1
56 ; CHECK-NEXT:    call void @_Z19test_multiple_exitsi(i32 [[ADD]])
57 ; CHECK-NEXT:    call void @llvm.lifetime.end.p0i8(i64 4, i8* nonnull [[TMP1]]) #1
58 ; CHECK-NEXT:    br label [[IF_END14:%.*]]
59 ; CHECK:       if.else:
60 ; CHECK-NEXT:    [[PARAM_OFF:%.*]] = add i32 [[PARAM_TR]], -10
61 ; CHECK-NEXT:    [[TMP2:%.*]] = icmp ult i32 [[PARAM_OFF]], 10
62 ; CHECK-NEXT:    br i1 [[TMP2]], label [[IF_THEN5]], label [[IF_ELSE7:%.*]]
63 ; CHECK:       if.then5:
64 ; CHECK-NEXT:    [[ADD6]] = add nuw nsw i32 [[PARAM_TR]], 1
65 ; CHECK-NEXT:    br label [[TAILRECURSE]]
66 ; CHECK:       if.else7:
67 ; CHECK-NEXT:    [[TMP3:%.*]] = and i32 [[PARAM_TR]], -2
68 ; CHECK-NEXT:    [[TMP4:%.*]] = icmp eq i32 [[TMP3]], 20
69 ; CHECK-NEXT:    br i1 [[TMP4]], label [[IF_THEN11:%.*]], label [[IF_END14]]
70 ; CHECK:       if.then11:
71 ; CHECK-NEXT:    [[ADD12:%.*]] = add nsw i32 [[PARAM_TR]], 1
72 ; CHECK-NEXT:    tail call void @_Z19test_multiple_exitsi(i32 [[ADD12]])
73 ; CHECK-NEXT:    tail call void @_Z4funcv()
74 ; CHECK-NEXT:    ret void
75 ; CHECK:       if.end14:
76 ; CHECK-NEXT:    ret void
78 entry:
79   %temp = alloca i32, align 4
80   %0 = icmp ult i32 %param, 10
81   br i1 %0, label %if.then, label %if.else
83 if.then:                                          ; preds = %entry
84   %1 = bitcast i32* %temp to i8*
85   call void @llvm.lifetime.start.p0i8(i64 4, i8* nonnull %1) #2
86   call void @_Z11capture_argPi(i32* nonnull %temp)
87   %add = add nuw nsw i32 %param, 1
88   call void @_Z19test_multiple_exitsi(i32 %add)
89   call void @llvm.lifetime.end.p0i8(i64 4, i8* nonnull %1) #2
90   br label %if.end14
92 if.else:                                          ; preds = %entry
93   %param.off = add i32 %param, -10
94   %2 = icmp ult i32 %param.off, 10
95   br i1 %2, label %if.then5, label %if.else7
97 if.then5:                                         ; preds = %if.else
98   %add6 = add nuw nsw i32 %param, 1
99   call void @_Z19test_multiple_exitsi(i32 %add6)
100   br label %if.end14
102 if.else7:                                         ; preds = %if.else
103   %3 = and i32 %param, -2
104   %4 = icmp eq i32 %3, 20
105   br i1 %4, label %if.then11, label %if.end14
107 if.then11:                                        ; preds = %if.else7
108   %add12 = add nsw i32 %param, 1
109   call void @_Z19test_multiple_exitsi(i32 %add12)
110   call void @_Z4funcv()
111   br label %if.end14
113 if.end14:                                         ; preds = %if.then5, %if.then11, %if.else7, %if.then
114   ret void
117 ; Function Attrs: argmemonly nounwind willreturn
118 declare void @llvm.lifetime.start.p0i8(i64 immarg, i8* nocapture) #2
120 ; Function Attrs: argmemonly nounwind willreturn
121 declare void @llvm.lifetime.end.p0i8(i64 immarg, i8* nocapture) #2
123 attributes #0 = { nofree noinline norecurse nounwind uwtable }
124 attributes #1 = { nounwind uwtable }
125 attributes #2 = { argmemonly nounwind willreturn }