xref: /aosp_15_r20/external/clang/test/CodeGenCXX/cxx1y-init-captures.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -std=c++1y -triple x86_64-linux-gnu -emit-llvm %s -o - | FileCheck %s
2*67e74705SXin Li 
3*67e74705SXin Li struct S {
4*67e74705SXin Li   S();
5*67e74705SXin Li   S(S &&);
6*67e74705SXin Li   ~S();
7*67e74705SXin Li };
8*67e74705SXin Li 
f()9*67e74705SXin Li void f() {
10*67e74705SXin Li   (void) [s(S{})] {};
11*67e74705SXin Li }
12*67e74705SXin Li 
13*67e74705SXin Li // CHECK-LABEL: define void @_Z1fv(
14*67e74705SXin Li // CHECK: call void @_ZN1SC1Ev(
15*67e74705SXin Li // CHECK: call void @"_ZZ1fvEN3$_0D1Ev"(
16*67e74705SXin Li 
17*67e74705SXin Li // CHECK-LABEL: define internal void @"_ZZ1fvEN3$_0D1Ev"(
18*67e74705SXin Li // CHECK: @"_ZZ1fvEN3$_0D2Ev"(
19*67e74705SXin Li 
20*67e74705SXin Li // D2 at end of file.
21*67e74705SXin Li 
g()22*67e74705SXin Li void g() {
23*67e74705SXin Li   [a(1), b(2)] { return a + b; } ();
24*67e74705SXin Li }
25*67e74705SXin Li 
26*67e74705SXin Li // CHECK-LABEL: define void @_Z1gv(
27*67e74705SXin Li // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 0
28*67e74705SXin Li // CHECK: store i32 1, i32*
29*67e74705SXin Li // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 1
30*67e74705SXin Li // CHECK: store i32 2, i32*
31*67e74705SXin Li // CHECK: call i32 @"_ZZ1gvENK3$_1clEv"(
32*67e74705SXin Li 
33*67e74705SXin Li // CHECK-LABEL: define internal i32 @"_ZZ1gvENK3$_1clEv"(
34*67e74705SXin Li // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 0
35*67e74705SXin Li // CHECK: load i32, i32*
36*67e74705SXin Li // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 1
37*67e74705SXin Li // CHECK: load i32, i32*
38*67e74705SXin Li 
39*67e74705SXin Li // CHECK: add nsw i32
40*67e74705SXin Li 
h(int a)41*67e74705SXin Li int h(int a) {
42*67e74705SXin Li   // CHECK-LABEL: define i32 @_Z1hi(
43*67e74705SXin Li   // CHECK: %[[A_ADDR:.*]] = alloca i32,
44*67e74705SXin Li   // CHECK: %[[OUTER:.*]] = alloca
45*67e74705SXin Li   // CHECK: store i32 {{.*}}, i32* %[[A_ADDR]],
46*67e74705SXin Li   //
47*67e74705SXin Li   // Initialize init-capture 'b(a)' by reference.
48*67e74705SXin Li   // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[OUTER]], i32 0, i32 0
49*67e74705SXin Li   // CHECK: store i32* %[[A_ADDR]], i32** {{.*}},
50*67e74705SXin Li   //
51*67e74705SXin Li   // Initialize init-capture 'c(a)' by copy.
52*67e74705SXin Li   // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[OUTER]], i32 0, i32 1
53*67e74705SXin Li   // CHECK: load i32, i32* %[[A_ADDR]],
54*67e74705SXin Li   // CHECK: store i32
55*67e74705SXin Li   //
56*67e74705SXin Li   // CHECK: call i32 @"_ZZ1hiENK3$_2clEv"({{.*}}* %[[OUTER]])
57*67e74705SXin Li   return [&b(a), c(a)] {
58*67e74705SXin Li     // CHECK-LABEL: define internal i32 @"_ZZ1hiENK3$_2clEv"(
59*67e74705SXin Li     // CHECK: %[[OUTER_ADDR:.*]] = alloca
60*67e74705SXin Li     // CHECK: %[[INNER:.*]] = alloca
61*67e74705SXin Li     // CHECK: store {{.*}}, {{.*}}** %[[OUTER_ADDR]],
62*67e74705SXin Li     //
63*67e74705SXin Li     // Capture outer 'c' by reference.
64*67e74705SXin Li     // CHECK: %[[OUTER:.*]] = load {{.*}}*, {{.*}}** %[[OUTER_ADDR]]
65*67e74705SXin Li     // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[INNER]], i32 0, i32 0
66*67e74705SXin Li     // CHECK-NEXT: getelementptr inbounds {{.*}}, {{.*}}* %[[OUTER]], i32 0, i32 1
67*67e74705SXin Li     // CHECK-NEXT: store i32* %
68*67e74705SXin Li     //
69*67e74705SXin Li     // Capture outer 'b' by copy.
70*67e74705SXin Li     // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[INNER]], i32 0, i32 1
71*67e74705SXin Li     // CHECK-NEXT: getelementptr inbounds {{.*}}, {{.*}}* %[[OUTER]], i32 0, i32 0
72*67e74705SXin Li     // CHECK-NEXT: load i32*, i32** %
73*67e74705SXin Li     // CHECK-NEXT: load i32, i32* %
74*67e74705SXin Li     // CHECK-NEXT: store i32
75*67e74705SXin Li     //
76*67e74705SXin Li     // CHECK: call i32 @"_ZZZ1hiENK3$_2clEvENKUlvE_clEv"({{.*}}* %[[INNER]])
77*67e74705SXin Li     return [=, &c] {
78*67e74705SXin Li       // CHECK-LABEL: define internal void @"_ZZ1fvEN3$_0D2Ev"(
79*67e74705SXin Li       // CHECK: call void @_ZN1SD1Ev(
80*67e74705SXin Li 
81*67e74705SXin Li       // CHECK-LABEL: define internal i32 @"_ZZZ1hiENK3$_2clEvENKUlvE_clEv"(
82*67e74705SXin Li       // CHECK: %[[INNER_ADDR:.*]] = alloca
83*67e74705SXin Li       // CHECK: store {{.*}}, {{.*}}** %[[INNER_ADDR]],
84*67e74705SXin Li       // CHECK: %[[INNER:.*]] = load {{.*}}*, {{.*}}** %[[INNER_ADDR]]
85*67e74705SXin Li       //
86*67e74705SXin Li       // Load capture of 'b'
87*67e74705SXin Li       // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[INNER]], i32 0, i32 1
88*67e74705SXin Li       // CHECK: load i32, i32* %
89*67e74705SXin Li       //
90*67e74705SXin Li       // Load capture of 'c'
91*67e74705SXin Li       // CHECK: getelementptr inbounds {{.*}}, {{.*}}* %[[INNER]], i32 0, i32 0
92*67e74705SXin Li       // CHECK: load i32*, i32** %
93*67e74705SXin Li       // CHECK: load i32, i32* %
94*67e74705SXin Li       //
95*67e74705SXin Li       // CHECK: add nsw i32
96*67e74705SXin Li       return b + c;
97*67e74705SXin Li     } ();
98*67e74705SXin Li   } ();
99*67e74705SXin Li }
100*67e74705SXin Li 
101*67e74705SXin Li // Ensure we can emit code for init-captures in global lambdas too.
__anon163a35e40502() 102*67e74705SXin Li auto global_lambda = [a = 0] () mutable { return ++a; };
get_incremented()103*67e74705SXin Li int get_incremented() { return global_lambda(); }
104