xref: /minix3/external/bsd/llvm/dist/clang/test/CodeGenCXX/cxx11-thread-local.cpp (revision f4a2713ac843a11c696ec80c0a5e3e5d80b4d338)
1*f4a2713aSLionel Sambuc // RUN: %clang_cc1 -std=c++11 -emit-llvm %s -o - -triple x86_64-linux-gnu | FileCheck %s
2*f4a2713aSLionel Sambuc 
3*f4a2713aSLionel Sambuc int f();
4*f4a2713aSLionel Sambuc int g();
5*f4a2713aSLionel Sambuc 
6*f4a2713aSLionel Sambuc // CHECK: @a = thread_local global i32 0
7*f4a2713aSLionel Sambuc thread_local int a = f();
8*f4a2713aSLionel Sambuc extern thread_local int b;
9*f4a2713aSLionel Sambuc // CHECK: @c = global i32 0
10*f4a2713aSLionel Sambuc int c = b;
11*f4a2713aSLionel Sambuc // CHECK: @_ZL1d = internal thread_local global i32 0
12*f4a2713aSLionel Sambuc static thread_local int d = g();
13*f4a2713aSLionel Sambuc 
14*f4a2713aSLionel Sambuc struct U { static thread_local int m; };
15*f4a2713aSLionel Sambuc // CHECK: @_ZN1U1mE = thread_local global i32 0
16*f4a2713aSLionel Sambuc thread_local int U::m = f();
17*f4a2713aSLionel Sambuc 
18*f4a2713aSLionel Sambuc template<typename T> struct V { static thread_local int m; };
19*f4a2713aSLionel Sambuc template<typename T> thread_local int V<T>::m = g();
20*f4a2713aSLionel Sambuc 
21*f4a2713aSLionel Sambuc // CHECK: @e = global i32 0
22*f4a2713aSLionel Sambuc int e = V<int>::m;
23*f4a2713aSLionel Sambuc 
24*f4a2713aSLionel Sambuc // CHECK: @_ZN1VIiE1mE = weak_odr thread_local global i32 0
25*f4a2713aSLionel Sambuc 
26*f4a2713aSLionel Sambuc // CHECK: @_ZZ1fvE1n = internal thread_local global i32 0
27*f4a2713aSLionel Sambuc 
28*f4a2713aSLionel Sambuc // CHECK: @_ZGVZ1fvE1n = internal thread_local global i8 0
29*f4a2713aSLionel Sambuc 
30*f4a2713aSLionel Sambuc // CHECK: @_ZZ8tls_dtorvE1s = internal thread_local global
31*f4a2713aSLionel Sambuc // CHECK: @_ZGVZ8tls_dtorvE1s = internal thread_local global i8 0
32*f4a2713aSLionel Sambuc 
33*f4a2713aSLionel Sambuc // CHECK: @_ZZ8tls_dtorvE1t = internal thread_local global
34*f4a2713aSLionel Sambuc // CHECK: @_ZGVZ8tls_dtorvE1t = internal thread_local global i8 0
35*f4a2713aSLionel Sambuc 
36*f4a2713aSLionel Sambuc // CHECK: @_ZZ8tls_dtorvE1u = internal thread_local global
37*f4a2713aSLionel Sambuc // CHECK: @_ZGVZ8tls_dtorvE1u = internal thread_local global i8 0
38*f4a2713aSLionel Sambuc // CHECK: @_ZGRZ8tls_dtorvE1u = private thread_local global
39*f4a2713aSLionel Sambuc 
40*f4a2713aSLionel Sambuc // CHECK: @_ZGVN1VIiE1mE = weak_odr thread_local global i64 0
41*f4a2713aSLionel Sambuc 
42*f4a2713aSLionel Sambuc // CHECK: @__tls_guard = internal thread_local global i8 0
43*f4a2713aSLionel Sambuc 
44*f4a2713aSLionel Sambuc // CHECK: @llvm.global_ctors = appending global {{.*}} @[[GLOBAL_INIT:[^ ]*]]
45*f4a2713aSLionel Sambuc 
46*f4a2713aSLionel Sambuc // CHECK: @_ZTH1a = alias void ()* @__tls_init
47*f4a2713aSLionel Sambuc // CHECK: @_ZTHL1d = alias internal void ()* @__tls_init
48*f4a2713aSLionel Sambuc // CHECK: @_ZTHN1U1mE = alias void ()* @__tls_init
49*f4a2713aSLionel Sambuc // CHECK: @_ZTHN1VIiE1mE = alias weak_odr void ()* @__tls_init
50*f4a2713aSLionel Sambuc 
51*f4a2713aSLionel Sambuc 
52*f4a2713aSLionel Sambuc // Individual variable initialization functions:
53*f4a2713aSLionel Sambuc 
54*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[A_INIT:.*]]()
55*f4a2713aSLionel Sambuc // CHECK: call i32 @_Z1fv()
56*f4a2713aSLionel Sambuc // CHECK-NEXT: store i32 {{.*}}, i32* @a, align 4
57*f4a2713aSLionel Sambuc 
58*f4a2713aSLionel Sambuc // CHECK-LABEL: define i32 @_Z1fv()
59*f4a2713aSLionel Sambuc int f() {
60*f4a2713aSLionel Sambuc   // CHECK: %[[GUARD:.*]] = load i8* @_ZGVZ1fvE1n, align 1
61*f4a2713aSLionel Sambuc   // CHECK: %[[NEED_INIT:.*]] = icmp eq i8 %[[GUARD]], 0
62*f4a2713aSLionel Sambuc   // CHECK: br i1 %[[NEED_INIT]]
63*f4a2713aSLionel Sambuc 
64*f4a2713aSLionel Sambuc   // CHECK: %[[CALL:.*]] = call i32 @_Z1gv()
65*f4a2713aSLionel Sambuc   // CHECK: store i32 %[[CALL]], i32* @_ZZ1fvE1n, align 4
66*f4a2713aSLionel Sambuc   // CHECK: store i8 1, i8* @_ZGVZ1fvE1n
67*f4a2713aSLionel Sambuc   // CHECK: br label
68*f4a2713aSLionel Sambuc   static thread_local int n = g();
69*f4a2713aSLionel Sambuc 
70*f4a2713aSLionel Sambuc   // CHECK: load i32* @_ZZ1fvE1n, align 4
71*f4a2713aSLionel Sambuc   return n;
72*f4a2713aSLionel Sambuc }
73*f4a2713aSLionel Sambuc 
74*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[C_INIT:.*]]()
75*f4a2713aSLionel Sambuc // CHECK: call i32* @_ZTW1b()
76*f4a2713aSLionel Sambuc // CHECK-NEXT: load i32* %{{.*}}, align 4
77*f4a2713aSLionel Sambuc // CHECK-NEXT: store i32 %{{.*}}, i32* @c, align 4
78*f4a2713aSLionel Sambuc 
79*f4a2713aSLionel Sambuc // CHECK-LABEL: define weak_odr hidden i32* @_ZTW1b()
80*f4a2713aSLionel Sambuc // CHECK: br i1 icmp ne (void ()* @_ZTH1b, void ()* null),
81*f4a2713aSLionel Sambuc // not null:
82*f4a2713aSLionel Sambuc // CHECK: call void @_ZTH1b()
83*f4a2713aSLionel Sambuc // CHECK: br label
84*f4a2713aSLionel Sambuc // finally:
85*f4a2713aSLionel Sambuc // CHECK: ret i32* @b
86*f4a2713aSLionel Sambuc 
87*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[D_INIT:.*]]()
88*f4a2713aSLionel Sambuc // CHECK: call i32 @_Z1gv()
89*f4a2713aSLionel Sambuc // CHECK-NEXT: store i32 %{{.*}}, i32* @_ZL1d, align 4
90*f4a2713aSLionel Sambuc 
91*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[U_M_INIT:.*]]()
92*f4a2713aSLionel Sambuc // CHECK: call i32 @_Z1fv()
93*f4a2713aSLionel Sambuc // CHECK-NEXT: store i32 %{{.*}}, i32* @_ZN1U1mE, align 4
94*f4a2713aSLionel Sambuc 
95*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[E_INIT:.*]]()
96*f4a2713aSLionel Sambuc // CHECK: call i32* @_ZTWN1VIiE1mE()
97*f4a2713aSLionel Sambuc // CHECK-NEXT: load i32* %{{.*}}, align 4
98*f4a2713aSLionel Sambuc // CHECK-NEXT: store i32 %{{.*}}, i32* @e, align 4
99*f4a2713aSLionel Sambuc 
100*f4a2713aSLionel Sambuc // CHECK-LABEL: define weak_odr hidden i32* @_ZTWN1VIiE1mE()
101*f4a2713aSLionel Sambuc // CHECK: call void @_ZTHN1VIiE1mE()
102*f4a2713aSLionel Sambuc // CHECK: ret i32* @_ZN1VIiE1mE
103*f4a2713aSLionel Sambuc 
104*f4a2713aSLionel Sambuc 
105*f4a2713aSLionel Sambuc struct S { S(); ~S(); };
106*f4a2713aSLionel Sambuc struct T { ~T(); };
107*f4a2713aSLionel Sambuc 
108*f4a2713aSLionel Sambuc // CHECK-LABEL: define void @_Z8tls_dtorv()
109*f4a2713aSLionel Sambuc void tls_dtor() {
110*f4a2713aSLionel Sambuc   // CHECK: load i8* @_ZGVZ8tls_dtorvE1s
111*f4a2713aSLionel Sambuc   // CHECK: call void @_ZN1SC1Ev(%struct.S* @_ZZ8tls_dtorvE1s)
112*f4a2713aSLionel Sambuc   // CHECK: call i32 @__cxa_thread_atexit({{.*}}@_ZN1SD1Ev {{.*}} @_ZZ8tls_dtorvE1s{{.*}} @__dso_handle
113*f4a2713aSLionel Sambuc   // CHECK: store i8 1, i8* @_ZGVZ8tls_dtorvE1s
114*f4a2713aSLionel Sambuc   static thread_local S s;
115*f4a2713aSLionel Sambuc 
116*f4a2713aSLionel Sambuc   // CHECK: load i8* @_ZGVZ8tls_dtorvE1t
117*f4a2713aSLionel Sambuc   // CHECK-NOT: _ZN1T
118*f4a2713aSLionel Sambuc   // CHECK: call i32 @__cxa_thread_atexit({{.*}}@_ZN1TD1Ev {{.*}}@_ZZ8tls_dtorvE1t{{.*}} @__dso_handle
119*f4a2713aSLionel Sambuc   // CHECK: store i8 1, i8* @_ZGVZ8tls_dtorvE1t
120*f4a2713aSLionel Sambuc   static thread_local T t;
121*f4a2713aSLionel Sambuc 
122*f4a2713aSLionel Sambuc   // CHECK: load i8* @_ZGVZ8tls_dtorvE1u
123*f4a2713aSLionel Sambuc   // CHECK: call void @_ZN1SC1Ev(%struct.S* @_ZGRZ8tls_dtorvE1u)
124*f4a2713aSLionel Sambuc   // CHECK: call i32 @__cxa_thread_atexit({{.*}}@_ZN1SD1Ev {{.*}} @_ZGRZ8tls_dtorvE1u{{.*}} @__dso_handle
125*f4a2713aSLionel Sambuc   // CHECK: store i8 1, i8* @_ZGVZ8tls_dtorvE1u
126*f4a2713aSLionel Sambuc   static thread_local const S &u = S();
127*f4a2713aSLionel Sambuc }
128*f4a2713aSLionel Sambuc 
129*f4a2713aSLionel Sambuc // CHECK: declare i32 @__cxa_thread_atexit(void (i8*)*, i8*, i8*)
130*f4a2713aSLionel Sambuc 
131*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @_Z7PR15991v(
132*f4a2713aSLionel Sambuc int PR15991() {
133*f4a2713aSLionel Sambuc   thread_local int n;
134*f4a2713aSLionel Sambuc   auto l = [] { return n; };
135*f4a2713aSLionel Sambuc   return l();
136*f4a2713aSLionel Sambuc }
137*f4a2713aSLionel Sambuc 
138*f4a2713aSLionel Sambuc // CHECK: define {{.*}} @[[V_M_INIT:.*]]()
139*f4a2713aSLionel Sambuc // CHECK: load i8* bitcast (i64* @_ZGVN1VIiE1mE to i8*)
140*f4a2713aSLionel Sambuc // CHECK: %[[V_M_INITIALIZED:.*]] = icmp eq i8 %{{.*}}, 0
141*f4a2713aSLionel Sambuc // CHECK: br i1 %[[V_M_INITIALIZED]],
142*f4a2713aSLionel Sambuc // need init:
143*f4a2713aSLionel Sambuc // CHECK: call i32 @_Z1gv()
144*f4a2713aSLionel Sambuc // CHECK: store i32 %{{.*}}, i32* @_ZN1VIiE1mE, align 4
145*f4a2713aSLionel Sambuc // CHECK: store i64 1, i64* @_ZGVN1VIiE1mE
146*f4a2713aSLionel Sambuc // CHECK: br label
147*f4a2713aSLionel Sambuc 
148*f4a2713aSLionel Sambuc // CHECK: define {{.*}}@[[GLOBAL_INIT:.*]]()
149*f4a2713aSLionel Sambuc // CHECK: call void @[[C_INIT]]()
150*f4a2713aSLionel Sambuc // CHECK: call void @[[E_INIT]]()
151*f4a2713aSLionel Sambuc 
152*f4a2713aSLionel Sambuc 
153*f4a2713aSLionel Sambuc // CHECK: define {{.*}}@__tls_init()
154*f4a2713aSLionel Sambuc // CHECK: load i8* @__tls_guard
155*f4a2713aSLionel Sambuc // CHECK: %[[NEED_TLS_INIT:.*]] = icmp eq i8 %{{.*}}, 0
156*f4a2713aSLionel Sambuc // CHECK: store i8 1, i8* @__tls_guard
157*f4a2713aSLionel Sambuc // CHECK: br i1 %[[NEED_TLS_INIT]],
158*f4a2713aSLionel Sambuc // init:
159*f4a2713aSLionel Sambuc // CHECK: call void @[[A_INIT]]()
160*f4a2713aSLionel Sambuc // CHECK: call void @[[D_INIT]]()
161*f4a2713aSLionel Sambuc // CHECK: call void @[[U_M_INIT]]()
162*f4a2713aSLionel Sambuc // CHECK: call void @[[V_M_INIT]]()
163*f4a2713aSLionel Sambuc 
164*f4a2713aSLionel Sambuc 
165*f4a2713aSLionel Sambuc // CHECK: define weak_odr hidden i32* @_ZTW1a() {
166*f4a2713aSLionel Sambuc // CHECK:   call void @_ZTH1a()
167*f4a2713aSLionel Sambuc // CHECK:   ret i32* @a
168*f4a2713aSLionel Sambuc // CHECK: }
169*f4a2713aSLionel Sambuc 
170*f4a2713aSLionel Sambuc 
171*f4a2713aSLionel Sambuc // CHECK: declare extern_weak void @_ZTH1b()
172*f4a2713aSLionel Sambuc 
173*f4a2713aSLionel Sambuc 
174*f4a2713aSLionel Sambuc // CHECK-LABEL: define internal hidden i32* @_ZTWL1d()
175*f4a2713aSLionel Sambuc // CHECK: call void @_ZTHL1d()
176*f4a2713aSLionel Sambuc // CHECK: ret i32* @_ZL1d
177*f4a2713aSLionel Sambuc 
178*f4a2713aSLionel Sambuc // CHECK-LABEL: define weak_odr hidden i32* @_ZTWN1U1mE()
179*f4a2713aSLionel Sambuc // CHECK: call void @_ZTHN1U1mE()
180*f4a2713aSLionel Sambuc // CHECK: ret i32* @_ZN1U1mE
181