|  | // RUN: %clang_cc1 -triple x86_64-pc-linux -emit-llvm -o - %s | FileCheck %s | 
|  | // RUN: %clang_cc1 -triple x86_64-apple-darwin -emit-llvm -o - %s | \ | 
|  | // RUN: FileCheck --check-prefix=MACHO %s | 
|  |  | 
|  | // CHECK: @_ZN5test11A1aE ={{.*}} constant i32 10, align 4 | 
|  | // CHECK: @_ZN5test212_GLOBAL__N_11AIiE1xE = internal global i32 0, align 4 | 
|  | // CHECK: @_ZN5test31AIiE1xE = weak_odr global i32 0, comdat, align 4 | 
|  | // CHECK: @_ZGVN5test31AIiE1xE = weak_odr global i64 0, comdat($_ZN5test31AIiE1xE) | 
|  | // MACHO: @_ZGVN5test31AIiE1xE = weak_odr global i64 0 | 
|  | // MACHO-NOT: comdat | 
|  |  | 
|  | // CHECK: _ZN5test51U2k0E ={{.*}} global i32 0 | 
|  | // CHECK: _ZN5test51U2k1E ={{.*}} global i32 0 | 
|  | // CHECK: _ZN5test51U2k2E ={{.*}} constant i32 76 | 
|  | // CHECK-NOT: test51U2k3E | 
|  | // CHECK-NOT: test51U2k4E | 
|  |  | 
|  | // PR5564. | 
|  | namespace test1 { | 
|  | struct A { | 
|  | static const int a = 10; | 
|  | }; | 
|  |  | 
|  | const int A::a; | 
|  |  | 
|  | struct S { | 
|  | static int i; | 
|  | }; | 
|  |  | 
|  | void f() { | 
|  | int a = S::i; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Test that we don't use guards for initializing template static data | 
|  | // members with internal linkage. | 
|  | namespace test2 { | 
|  | int foo(); | 
|  |  | 
|  | namespace { | 
|  | template <class T> struct A { | 
|  | static int x; | 
|  | }; | 
|  |  | 
|  | template <class T> int A<T>::x = foo(); | 
|  | template struct A<int>; | 
|  | } | 
|  |  | 
|  | // CHECK-LABEL: define internal void @__cxx_global_var_init() | 
|  | // CHECK:      [[TMP:%.*]] = call i32 @_ZN5test23fooEv() | 
|  | // CHECK-NEXT: store i32 [[TMP]], i32* @_ZN5test212_GLOBAL__N_11AIiE1xE, align 4 | 
|  | // CHECK-NEXT: ret void | 
|  | } | 
|  |  | 
|  | // Test that we don't use threadsafe statics when initializing | 
|  | // template static data members. | 
|  | namespace test3 { | 
|  | int foo(); | 
|  |  | 
|  | template <class T> struct A { | 
|  | static int x; | 
|  | }; | 
|  |  | 
|  | template <class T> int A<T>::x = foo(); | 
|  | template struct A<int>; | 
|  |  | 
|  | // CHECK-LABEL: define internal void @__cxx_global_var_init.1() {{.*}} comdat($_ZN5test31AIiE1xE) | 
|  | // MACHO-LABEL: define internal void @__cxx_global_var_init.1() | 
|  | // MACHO-NOT: comdat | 
|  | // CHECK:      [[GUARDBYTE:%.*]] = load i8, i8* bitcast (i64* @_ZGVN5test31AIiE1xE to i8*) | 
|  | // CHECK-NEXT: [[UNINITIALIZED:%.*]] = icmp eq i8 [[GUARDBYTE]], 0 | 
|  | // CHECK-NEXT: br i1 [[UNINITIALIZED]] | 
|  | // CHECK:      [[TMP:%.*]] = call i32 @_ZN5test33fooEv() | 
|  | // CHECK-NEXT: store i32 [[TMP]], i32* @_ZN5test31AIiE1xE, align 4 | 
|  | // CHECK-NEXT: store i64 1, i64* @_ZGVN5test31AIiE1xE | 
|  | // CHECK-NEXT: br label | 
|  | // CHECK:      ret void | 
|  | } | 
|  |  | 
|  | // Test that we can fold member lookup expressions which resolve to static data | 
|  | // members. | 
|  | namespace test4 { | 
|  | struct A { | 
|  | static const int n = 76; | 
|  | }; | 
|  |  | 
|  | int f(A *a) { | 
|  | // CHECK-LABEL: define{{.*}} i32 @_ZN5test41fEPNS_1AE | 
|  | // CHECK: ret i32 76 | 
|  | return a->n; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Test that static data members in unions behave properly. | 
|  | namespace test5 { | 
|  | union U { | 
|  | static int k0; | 
|  | static const int k1; | 
|  | static const int k2 = 76; | 
|  | static const int k3; | 
|  | static const int k4 = 81; | 
|  | }; | 
|  | int U::k0; | 
|  | const int U::k1 = (k0 = 9, 42); | 
|  | const int U::k2; | 
|  |  | 
|  | // CHECK: store i32 9, i32* @_ZN5test51U2k0E | 
|  | // CHECK: store i32 {{.*}}, i32* @_ZN5test51U2k1E | 
|  | // CHECK-NOT: store {{.*}} i32* @_ZN5test51U2k2E | 
|  | } |