111 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			111 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| // 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
 | |
| }
 |