forked from OSchip/llvm-project
243 lines
10 KiB
C++
243 lines
10 KiB
C++
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -DIMPORT=1 -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-IMPORT,CHECK-NO-NS,CHECK-IMPORT-NO-NS --implicit-check-not=unused
|
||
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -DIMPORT=1 -DNS -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-IMPORT,CHECK-NS,CHECK-IMPORT-NS --implicit-check-not=unused
|
||
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -DIMPORT=2 -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-NO-NS --implicit-check-not=unused
|
||
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -DIMPORT=2 -DNS -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-NS --implicit-check-not=unused
|
||
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-NO-NS --implicit-check-not=unused
|
||
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++17 -emit-llvm -DNS -fmodules %s -o - | FileCheck %s --check-prefixes=CHECK,CHECK-NS --implicit-check-not=unused
|
||
|
|
||
|
// Check that we behave sensibly when importing a header containing strong and
|
||
|
// weak, ordered and unordered global initializers.
|
||
|
//
|
||
|
// Our behavior is as follows:
|
||
|
//
|
||
|
// -- for variables with one or more specific points of initialization
|
||
|
// (non-template variables, whether or not they are inline or thread_local),
|
||
|
// emit them if (and only if) a header containing a point of initialization
|
||
|
// is transitively #included / imported.
|
||
|
//
|
||
|
// -- for variables with unordered initialization (any kind of templated
|
||
|
// variable -- excluding explicit specializations), emit them if any part
|
||
|
// of any module that triggers an instantiation is imported.
|
||
|
//
|
||
|
// The intent is to:
|
||
|
//
|
||
|
// 1) preserve order of initialization guarantees
|
||
|
// 2) preserve the behavior of globals with ctors in headers, and specifically
|
||
|
// of std::ios_base::Init (do not run the iostreams initializer nor force
|
||
|
// linking in the iostreams portion of the static library unless <iostream>
|
||
|
// is included)
|
||
|
// 3) behave conservatively-correctly with regard to unordered initializers: we
|
||
|
// might run them in cases where a traditional compilation would not, but
|
||
|
// will never fail to run them in cases where a traditional compilation
|
||
|
// would do so
|
||
|
//
|
||
|
// Perfect handling of unordered initializers would require tracking all
|
||
|
// submodules containing points of instantiation, which is very hard when those
|
||
|
// points of instantiation are within definitions that we skip because we
|
||
|
// already have a (non-visible) definition for the entity:
|
||
|
//
|
||
|
// // a.h
|
||
|
// template<typename> int v = f();
|
||
|
// inline int get() { return v<int>; }
|
||
|
//
|
||
|
// // b.h
|
||
|
// template<typename> int v = f();
|
||
|
// inline int get() { return v<int>; }
|
||
|
//
|
||
|
// If a.h and b.h are built as a module, we will only have a point of
|
||
|
// instantiation for v<int> in one of the two headers, because we will only
|
||
|
// parse one of the two get() functions.
|
||
|
|
||
|
#pragma clang module build m
|
||
|
module m {
|
||
|
module a {
|
||
|
header "foo.h" { size 123 mtime 456789 }
|
||
|
}
|
||
|
module b {}
|
||
|
}
|
||
|
|
||
|
#pragma clang module contents
|
||
|
#pragma clang module begin m.a
|
||
|
inline int non_trivial() { return 3; }
|
||
|
|
||
|
#ifdef NS
|
||
|
namespace ns {
|
||
|
#endif
|
||
|
|
||
|
int a = non_trivial();
|
||
|
inline int b = non_trivial();
|
||
|
thread_local int c = non_trivial();
|
||
|
inline thread_local int d = non_trivial();
|
||
|
|
||
|
template<typename U> int e = non_trivial();
|
||
|
template<typename U> inline int f = non_trivial();
|
||
|
template<typename U> thread_local int g = non_trivial();
|
||
|
template<typename U> inline thread_local int h = non_trivial();
|
||
|
|
||
|
inline int unused = 123; // should not be emitted
|
||
|
|
||
|
template<typename T> struct X {
|
||
|
static int a;
|
||
|
static inline int b = non_trivial();
|
||
|
static thread_local int c;
|
||
|
static inline thread_local int d = non_trivial();
|
||
|
|
||
|
template<typename U> static int e;
|
||
|
template<typename U> static inline int f = non_trivial();
|
||
|
template<typename U> static thread_local int g;
|
||
|
template<typename U> static inline thread_local int h = non_trivial();
|
||
|
|
||
|
static inline int unused = 123; // should not be emitted
|
||
|
};
|
||
|
|
||
|
template<typename T> int X<T>::a = non_trivial();
|
||
|
template<typename T> thread_local int X<T>::c = non_trivial();
|
||
|
template<typename T> template<typename U> int X<T>::e = non_trivial();
|
||
|
template<typename T> template<typename U> thread_local int X<T>::g = non_trivial();
|
||
|
|
||
|
inline void use(bool b, ...) {
|
||
|
if (b) return;
|
||
|
use(true, e<int>, f<int>, g<int>, h<int>,
|
||
|
X<int>::a, X<int>::b, X<int>::c, X<int>::d,
|
||
|
X<int>::e<int>, X<int>::f<int>, X<int>::g<int>, X<int>::h<int>);
|
||
|
}
|
||
|
|
||
|
#ifdef NS
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#pragma clang module end
|
||
|
#pragma clang module endbuild
|
||
|
|
||
|
#if IMPORT == 1
|
||
|
// Import the module and the m.a submodule; runs the ordered initializers and
|
||
|
// the unordered initializers.
|
||
|
#pragma clang module import m.a
|
||
|
#elif IMPORT == 2
|
||
|
// Import the module but not the m.a submodule; runs only the unordered
|
||
|
// initializers.
|
||
|
#pragma clang module import m.b
|
||
|
#else
|
||
|
// Load the module but do not import any submodules; runs only the unordered
|
||
|
// initializers. FIXME: Should this skip all of them?
|
||
|
#pragma clang module load m
|
||
|
#endif
|
||
|
|
||
|
// CHECK-IMPORT-NO-NS-DAG: @[[A:a]] = global i32 0, align 4
|
||
|
// CHECK-IMPORT-NO-NS-DAG: @[[B:b]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-IMPORT-NO-NS-DAG: @[[C:c]] = thread_local global i32 0, align 4
|
||
|
// CHECK-IMPORT-NO-NS-DAG: @[[D:d]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-NO-NS-DAG: @[[E:_Z1eIiE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-NO-NS-DAG: @[[F:_Z1fIiE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-NO-NS-DAG: @[[G:_Z1gIiE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-NO-NS-DAG: @[[H:_Z1hIiE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
|
||
|
// CHECK-IMPORT-NS-DAG: @[[A:_ZN2ns1aE]] = global i32 0, align 4
|
||
|
// CHECK-IMPORT-NS-DAG: @[[B:_ZN2ns1bE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-IMPORT-NS-DAG: @[[BG:_ZGVN2ns1bE]] = linkonce_odr global i64 0, comdat($[[B]]), align 8
|
||
|
// CHECK-IMPORT-NS-DAG: @[[C:_ZN2ns1cE]] = thread_local global i32 0, align 4
|
||
|
// CHECK-IMPORT-NS-DAG: @[[D:_ZN2ns1dE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-IMPORT-NS-DAG: @[[DG:_ZGVN2ns1dE]] = linkonce_odr thread_local global i64 0, comdat($[[D]]), align 8
|
||
|
// CHECK-NS-DAG: @[[E:_ZN2ns1eIiEE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-NS-DAG: @[[F:_ZN2ns1fIiEE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-NS-DAG: @[[G:_ZN2ns1gIiEE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-NS-DAG: @[[H:_ZN2ns1hIiEE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
|
||
|
// CHECK-DAG: @[[XA:_ZN(2ns)?1XIiE1aE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XB:_ZN(2ns)?1XIiE1bE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XC:_ZN(2ns)?1XIiE1cE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XD:_ZN(2ns)?1XIiE1dE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XE:_ZN(2ns)?1XIiE1eIiEE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XF:_ZN(2ns)?1XIiE1fIiEE]] = linkonce_odr global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XG:_ZN(2ns)?1XIiE1gIiEE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
// CHECK-DAG: @[[XH:_ZN(2ns)?1XIiE1hIiEE]] = linkonce_odr thread_local global i32 0, comdat, align 4
|
||
|
|
||
|
// It's OK if the order of the first 6 of these changes.
|
||
|
// CHECK: @llvm.global_ctors = appending global
|
||
|
// CHECK-SAME: @[[E_INIT:[^,]*]], {{[^@]*}} @[[E]]
|
||
|
// CHECK-SAME: @[[F_INIT:[^,]*]], {{[^@]*}} @[[F]]
|
||
|
// CHECK-SAME: @[[XA_INIT:[^,]*]], {{[^@]*}} @[[XA]]
|
||
|
// CHECK-SAME: @[[XE_INIT:[^,]*]], {{[^@]*}} @[[XE]]
|
||
|
// CHECK-SAME: @[[XF_INIT:[^,]*]], {{[^@]*}} @[[XF]]
|
||
|
// CHECK-SAME: @[[XB_INIT:[^,]*]], {{[^@]*}} @[[XB]]
|
||
|
// CHECK-IMPORT-SAME: @[[TU_INIT:[^,]*]], i8* null }]
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[E_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[E]],
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[F_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[F]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[G_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[G]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[H_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[H]],
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[XA_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XA]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[XC_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XC]],
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[XE_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XE]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[XG_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XG]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[XH_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XH]],
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[XF_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XF]],
|
||
|
|
||
|
// CHECK: define {{.*}} @[[XD_INIT:__cxx_global.*]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XD]],
|
||
|
|
||
|
// FIXME: Should this use __cxa_guard_acquire?
|
||
|
// CHECK: define {{.*}} @[[XB_INIT]]()
|
||
|
// CHECK: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK: store {{.*}}, i32* @[[XB]],
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @[[A_INIT:__cxx_global.*]]()
|
||
|
// CHECK-IMPORT: call i32 @_Z11non_trivialv(
|
||
|
// CHECK-IMPORT: store {{.*}}, i32* @[[A]],
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @[[B_INIT:__cxx_global.*]]()
|
||
|
// CHECK-IMPORT: call i32 @__cxa_guard_acquire(i64* @_ZGV
|
||
|
// CHECK-IMPORT: store {{.*}}, i32* @[[B]],
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @[[C_INIT:__cxx_global.*]]()
|
||
|
// CHECK-IMPORT: call i32 @_Z11non_trivialv(
|
||
|
// CHECK-IMPORT: store {{.*}}, i32* @[[C]],
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @[[D_INIT:__cxx_global.*]]()
|
||
|
// CHECK-IMPORT: load {{.*}} (i64* @_ZGV
|
||
|
// CHECK-IMPORT: store {{.*}}, i32* @[[D]],
|
||
|
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @[[TU_INIT]]()
|
||
|
// CHECK-IMPORT: call void @[[A_INIT]]()
|
||
|
// CHECK-IMPORT: call void @[[B_INIT]]()
|
||
|
|
||
|
// CHECK-IMPORT: define {{.*}} @__tls_init()
|
||
|
// CHECK-IMPORT: call void @[[C_INIT]]()
|
||
|
// CHECK-IMPORT: call void @[[D_INIT]]()
|