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
|
||
|
}
|