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