
Previously, Itanium ABI guard variables were set after initialization was complete for non-block declared variables with static and thread storage duration. That resulted in initialization of such variables being restarted in cases where the variable was referenced while it was still under construction. Per C++20 [class.cdtor]p2, such references are permitted (though the value obtained by such an access is unspecified). The late initialization resulted in recursive reinitialization loops for cases like this: template<typename T> struct ct { struct mc { mc() { ct<T>::smf(); } void mf() const {} }; thread_local static mc tlsdm; static void smf() { tlsdm.mf(); } }; template<typename T> thread_local typename ct<T>::mc ct<T>::tlsdm; int main() { ct<int>::smf(); } With this change, guard variables are set before initialization is started so as to avoid such reinitialization loops. Fixes https://github.com/llvm/llvm-project/issues/57828 Reviewed By: rjmccall Differential Revision: https://reviews.llvm.org/D135919
49 lines
1.8 KiB
C++
49 lines
1.8 KiB
C++
// RUN: %clang_cc1 -std=c++1y -triple x86_64-linux-gnu -emit-llvm -o - %s | FileCheck %s
|
|
|
|
// Check that we keep the 'extern' when we instantiate the definition of this
|
|
// variable template specialization.
|
|
template<typename T> extern const int extern_redecl;
|
|
template<typename T> const int extern_redecl = 5;
|
|
template const int extern_redecl<int>;
|
|
|
|
// CHECK: @_Z13extern_redeclIiE = weak_odr constant
|
|
|
|
template<typename T> struct Outer {
|
|
template<typename U> struct Inner {
|
|
template<typename V> static int arr[];
|
|
};
|
|
};
|
|
Outer<char[100]> outer_int;
|
|
int init_arr();
|
|
template<typename T> template<typename U> template<typename V> int Outer<T>::Inner<U>::arr[sizeof(T) + sizeof(U) + sizeof(V)] = { init_arr() };
|
|
int *p = Outer<char[100]>::Inner<char[20]>::arr<char[3]>;
|
|
|
|
namespace PR35456 {
|
|
// CHECK: @_ZN7PR354561nILi0EEE = linkonce_odr global i32 0
|
|
template<int> int n;
|
|
int *p = &n<0>;
|
|
}
|
|
|
|
// CHECK: @_ZN5OuterIA100_cE5InnerIA20_cE3arrIA3_cEE = linkonce_odr global [123 x i32] zeroinitializer
|
|
// CHECK: @_ZGVN5OuterIA100_cE5InnerIA20_cE3arrIA3_cEE = linkonce_odr global
|
|
|
|
// CHECK: @_ZTHN7PR4211112_GLOBAL__N_11nILi0EEE = internal alias {{.*}} @[[PR42111_CTOR:.*]]
|
|
|
|
// CHECK: call {{.*}}@_Z8init_arrv
|
|
|
|
// Ensure that we use guarded initialization for an instantiated thread_local
|
|
// variable with internal linkage.
|
|
namespace PR42111 {
|
|
int f();
|
|
namespace { template <int = 0> thread_local int n = f(); }
|
|
// CHECK: define {{.*}}@[[PR42111_CTOR]](
|
|
// CHECK: load {{.*}} @_ZGVN7PR4211112_GLOBAL__N_11nILi0EEE
|
|
// CHECK: icmp eq i8 {{.*}}, 0
|
|
// CHECK: br i1
|
|
// CHECK: store i8 1, ptr @_ZGVN7PR4211112_GLOBAL__N_11nILi0EEE
|
|
// CHECK: call noundef i32 @_ZN7PR421111fEv(
|
|
// CHECK: [[N_ADDR:%.+]] = call align 4 ptr @llvm.threadlocal.address.p0(ptr align 4 @_ZN7PR4211112_GLOBAL__N_11nILi0EEE)
|
|
// CHECK: store i32 {{.*}}, ptr [[N_ADDR]]
|
|
int g() { return n<> + n<>; }
|
|
}
|