Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / CodeGenCXX / cxx1y-init-captures.cpp
blob674ef889776c478b2bec8c7a407cc4eadf540a57
1 // RUN: %clang_cc1 -std=c++1y -triple x86_64-linux-gnu -emit-llvm %s -o - | FileCheck %s
3 struct S {
4 S();
5 S(S &&);
6 ~S();
7 };
9 void f() {
10 (void) [s(S{})] {};
13 // CHECK-LABEL: define{{.*}} void @_Z1fv(
14 // CHECK: call void @_ZN1SC1Ev(
15 // CHECK: call void @"_ZZ1fvEN3$_0D1Ev"(
17 // CHECK-LABEL: define internal void @"_ZZ1fvEN3$_0D1Ev"(
18 // CHECK: @"_ZZ1fvEN3$_0D2Ev"(
20 // D2 at end of file.
22 void g() {
23 [a(1), b(2)] { return a + b; } ();
26 // CHECK-LABEL: define{{.*}} void @_Z1gv(
27 // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 0
28 // CHECK: store i32 1, ptr
29 // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 1
30 // CHECK: store i32 2, ptr
31 // CHECK: call noundef i32 @"_ZZ1gvENK3$_0clEv"(
33 // CHECK-LABEL: define internal noundef i32 @"_ZZ1gvENK3$_0clEv"(
34 // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 0
35 // CHECK: load i32, ptr
36 // CHECK: getelementptr inbounds {{.*}}, i32 0, i32 1
37 // CHECK: load i32, ptr
39 // CHECK: add nsw i32
41 // CHECK-LABEL: define{{.*}} void @_Z18init_capture_dtorsv
42 void init_capture_dtors() {
43 // Ensure that init-captures are not treated as separate full-expressions.
44 struct HasDtor { ~HasDtor() {} };
45 void some_function_call();
46 void other_function_call();
47 // CHECK: call {{.*}}some_function_call
48 // CHECK: call {{.*}}HasDtorD
49 ([x = (HasDtor(), 0)]{}, some_function_call());
50 // CHECK: call {{.*}}other_function_call
51 other_function_call();
54 int h(int a) {
55 // CHECK-LABEL: define{{.*}} i32 @_Z1hi(
56 // CHECK: %[[A_ADDR:.*]] = alloca i32,
57 // CHECK: %[[OUTER:.*]] = alloca
58 // CHECK: store i32 {{.*}}, ptr %[[A_ADDR]],
60 // Initialize init-capture 'b(a)' by reference.
61 // CHECK: getelementptr inbounds {{.*}}, ptr %[[OUTER]], i32 0, i32 0
62 // CHECK: store ptr %[[A_ADDR]], ptr {{.*}},
64 // Initialize init-capture 'c(a)' by copy.
65 // CHECK: getelementptr inbounds {{.*}}, ptr %[[OUTER]], i32 0, i32 1
66 // CHECK: load i32, ptr %[[A_ADDR]],
67 // CHECK: store i32
69 // CHECK: call noundef i32 @"_ZZ1hiENK3$_0clEv"(ptr {{[^,]*}} %[[OUTER]])
70 return [&b(a), c(a)] {
71 // CHECK-LABEL: define internal noundef i32 @"_ZZ1hiENK3$_0clEv"(
72 // CHECK: %[[OUTER_ADDR:.*]] = alloca
73 // CHECK: %[[INNER:.*]] = alloca
74 // CHECK: store {{.*}}, ptr %[[OUTER_ADDR]],
76 // Capture outer 'c' by reference.
77 // CHECK: %[[OUTER:.*]] = load ptr, ptr %[[OUTER_ADDR]]
78 // CHECK: getelementptr inbounds {{.*}}, ptr %[[INNER]], i32 0, i32 0
79 // CHECK-NEXT: getelementptr inbounds {{.*}}, ptr %[[OUTER]], i32 0, i32 1
80 // CHECK-NEXT: store ptr %
82 // Capture outer 'b' by copy.
83 // CHECK: getelementptr inbounds {{.*}}, ptr %[[INNER]], i32 0, i32 1
84 // CHECK-NEXT: getelementptr inbounds {{.*}}, ptr %[[OUTER]], i32 0, i32 0
85 // CHECK-NEXT: load ptr, ptr %
86 // CHECK-NEXT: load i32, ptr %
87 // CHECK-NEXT: store i32
89 // CHECK: call noundef i32 @"_ZZZ1hiENK3$_0clEvENKUlvE_clEv"(ptr {{[^,]*}} %[[INNER]])
90 return [=, &c] {
91 // CHECK-LABEL: define internal void @"_ZZ1fvEN3$_0D2Ev"(
92 // CHECK: call void @_ZN1SD1Ev(
94 // CHECK-LABEL: define internal noundef i32 @"_ZZZ1hiENK3$_0clEvENKUlvE_clEv"(
95 // CHECK: %[[INNER_ADDR:.*]] = alloca
96 // CHECK: store {{.*}}, ptr %[[INNER_ADDR]],
97 // CHECK: %[[INNER:.*]] = load ptr, ptr %[[INNER_ADDR]]
99 // Load capture of 'b'
100 // CHECK: getelementptr inbounds {{.*}}, ptr %[[INNER]], i32 0, i32 1
101 // CHECK: load i32, ptr %
103 // Load capture of 'c'
104 // CHECK: getelementptr inbounds {{.*}}, ptr %[[INNER]], i32 0, i32 0
105 // CHECK: load ptr, ptr %
106 // CHECK: load i32, ptr %
108 // CHECK: add nsw i32
109 return b + c;
110 } ();
111 } ();
114 // Ensure we can emit code for init-captures in global lambdas too.
115 auto global_lambda = [a = 0] () mutable { return ++a; };
116 int get_incremented() { return global_lambda(); }