xref: /aosp_15_r20/external/clang/test/CodeGenCXX/static-init.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 %s -triple=x86_64-pc-linuxs -emit-llvm -o - | FileCheck %s
2*67e74705SXin Li 
3*67e74705SXin Li // CHECK: @_ZZ1hvE1i = internal global i32 0, align 4
4*67e74705SXin Li // CHECK: @base_req = global [4 x i8] c"foo\00", align 1
5*67e74705SXin Li // CHECK: @base_req_uchar = global [4 x i8] c"bar\00", align 1
6*67e74705SXin Li 
7*67e74705SXin Li // CHECK: @_ZZN5test31BC1EvE1u = internal global { i8, [3 x i8] } { i8 97, [3 x i8] undef }, align 4
8*67e74705SXin Li 
9*67e74705SXin Li // CHECK: @_ZZ2h2vE1i = linkonce_odr global i32 0, comdat, align 4
10*67e74705SXin Li // CHECK: @_ZGVZ2h2vE1i = linkonce_odr global i64 0, comdat, align 8{{$}}
11*67e74705SXin Li // CHECK: @_ZZN5test1L6getvarEiE3var = internal constant [4 x i32] [i32 1, i32 0, i32 2, i32 4], align 16
12*67e74705SXin Li // CHECK: @_ZZN5test414useStaticLocalEvE3obj = linkonce_odr global %"struct.test4::HasVTable" zeroinitializer, comdat, align 8
13*67e74705SXin Li 
14*67e74705SXin Li struct A {
15*67e74705SXin Li   A();
16*67e74705SXin Li   ~A();
17*67e74705SXin Li };
18*67e74705SXin Li 
f()19*67e74705SXin Li void f() {
20*67e74705SXin Li   // CHECK: load atomic i8, i8* bitcast (i64* @_ZGVZ1fvE1a to i8*) acquire, align 8
21*67e74705SXin Li   // CHECK: call i32 @__cxa_guard_acquire
22*67e74705SXin Li   // CHECK: call void @_ZN1AC1Ev
23*67e74705SXin Li   // CHECK: call i32 @__cxa_atexit(void (i8*)* bitcast (void (%struct.A*)* @_ZN1AD1Ev to void (i8*)*), i8* getelementptr inbounds (%struct.A, %struct.A* @_ZZ1fvE1a, i32 0, i32 0), i8* @__dso_handle)
24*67e74705SXin Li   // CHECK: call void @__cxa_guard_release
25*67e74705SXin Li   static A a;
26*67e74705SXin Li }
27*67e74705SXin Li 
g()28*67e74705SXin Li void g() {
29*67e74705SXin Li   // CHECK: call i8* @_Znwm(i64 1)
30*67e74705SXin Li   // CHECK: call void @_ZN1AC1Ev(
31*67e74705SXin Li   static A& a = *new A;
32*67e74705SXin Li }
33*67e74705SXin Li 
34*67e74705SXin Li int a();
h()35*67e74705SXin Li void h() {
36*67e74705SXin Li   static const int i = a();
37*67e74705SXin Li }
38*67e74705SXin Li 
39*67e74705SXin Li // CHECK: define linkonce_odr void @_Z2h2v() {{.*}} comdat {
h2()40*67e74705SXin Li inline void h2() {
41*67e74705SXin Li   static int i = a();
42*67e74705SXin Li }
43*67e74705SXin Li 
h3()44*67e74705SXin Li void h3() {
45*67e74705SXin Li   h2();
46*67e74705SXin Li }
47*67e74705SXin Li 
48*67e74705SXin Li // PR6980: this shouldn't crash
49*67e74705SXin Li namespace test0 {
50*67e74705SXin Li   struct A { A(); };
51*67e74705SXin Li   __attribute__((noreturn)) int throw_exception();
52*67e74705SXin Li 
test()53*67e74705SXin Li   void test() {
54*67e74705SXin Li     throw_exception();
55*67e74705SXin Li     static A r;
56*67e74705SXin Li   }
57*67e74705SXin Li }
58*67e74705SXin Li 
59*67e74705SXin Li namespace test1 {
60*67e74705SXin Li   // CHECK-LABEL: define internal i32 @_ZN5test1L6getvarEi(
getvar(int index)61*67e74705SXin Li   static inline int getvar(int index) {
62*67e74705SXin Li     static const int var[] = { 1, 0, 2, 4 };
63*67e74705SXin Li     return var[index];
64*67e74705SXin Li   }
65*67e74705SXin Li 
test()66*67e74705SXin Li   void test() { (void) getvar(2); }
67*67e74705SXin Li }
68*67e74705SXin Li 
69*67e74705SXin Li // Make sure we emit the initializer correctly for the following:
70*67e74705SXin Li char base_req[] = { "foo" };
71*67e74705SXin Li unsigned char base_req_uchar[] = { "bar" };
72*67e74705SXin Li 
73*67e74705SXin Li namespace union_static_local {
74*67e74705SXin Li   // CHECK-LABEL: define internal void @_ZZN18union_static_local4testEvEN1c4mainEv
75*67e74705SXin Li   // CHECK: call void @_ZN18union_static_local1fEPNS_1xE(%"union.union_static_local::x"* bitcast ({ [2 x i8*] }* @_ZZN18union_static_local4testEvE3foo to %"union.union_static_local::x"*))
76*67e74705SXin Li   union x { long double y; const char *x[2]; };
77*67e74705SXin Li   void f(union x*);
test()78*67e74705SXin Li   void test() {
79*67e74705SXin Li     static union x foo = { .x = { "a", "b" } };
80*67e74705SXin Li     struct c {
81*67e74705SXin Li       static void main() {
82*67e74705SXin Li         f(&foo);
83*67e74705SXin Li       }
84*67e74705SXin Li     };
85*67e74705SXin Li     c::main();
86*67e74705SXin Li   }
87*67e74705SXin Li }
88*67e74705SXin Li 
89*67e74705SXin Li // rdar://problem/11091093
90*67e74705SXin Li //   Static variables should be consistent across constructor
91*67e74705SXin Li //   or destructor variants.
92*67e74705SXin Li namespace test2 {
93*67e74705SXin Li   struct A {
94*67e74705SXin Li     A();
95*67e74705SXin Li     ~A();
96*67e74705SXin Li   };
97*67e74705SXin Li 
98*67e74705SXin Li   struct B : virtual A {
99*67e74705SXin Li     B();
100*67e74705SXin Li     ~B();
101*67e74705SXin Li   };
102*67e74705SXin Li 
103*67e74705SXin Li   // If we ever implement this as a delegate ctor call, just change
104*67e74705SXin Li   // this to take variadic arguments or something.
105*67e74705SXin Li   extern int foo();
B()106*67e74705SXin Li   B::B() {
107*67e74705SXin Li     static int x = foo();
108*67e74705SXin Li   }
109*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test21BC2Ev
110*67e74705SXin Li   // CHECK:   load atomic i8, i8* bitcast (i64* @_ZGVZN5test21BC1EvE1x to i8*) acquire,
111*67e74705SXin Li   // CHECK:   call i32 @__cxa_guard_acquire(i64* @_ZGVZN5test21BC1EvE1x)
112*67e74705SXin Li   // CHECK:   [[T0:%.*]] = call i32 @_ZN5test23fooEv()
113*67e74705SXin Li   // CHECK:   store i32 [[T0]], i32* @_ZZN5test21BC1EvE1x,
114*67e74705SXin Li   // CHECK:   call void @__cxa_guard_release(i64* @_ZGVZN5test21BC1EvE1x)
115*67e74705SXin Li 
116*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test21BC1Ev
117*67e74705SXin Li   // CHECK:   load atomic i8, i8* bitcast (i64* @_ZGVZN5test21BC1EvE1x to i8*) acquire,
118*67e74705SXin Li   // CHECK:   call i32 @__cxa_guard_acquire(i64* @_ZGVZN5test21BC1EvE1x)
119*67e74705SXin Li   // CHECK:   [[T0:%.*]] = call i32 @_ZN5test23fooEv()
120*67e74705SXin Li   // CHECK:   store i32 [[T0]], i32* @_ZZN5test21BC1EvE1x,
121*67e74705SXin Li   // CHECK:   call void @__cxa_guard_release(i64* @_ZGVZN5test21BC1EvE1x)
122*67e74705SXin Li 
123*67e74705SXin Li   // This is just for completeness, because we actually emit this
124*67e74705SXin Li   // using a delegate dtor call.
~B()125*67e74705SXin Li   B::~B() {
126*67e74705SXin Li     static int y = foo();
127*67e74705SXin Li   }
128*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test21BD2Ev(
129*67e74705SXin Li   // CHECK:   load atomic i8, i8* bitcast (i64* @_ZGVZN5test21BD1EvE1y to i8*) acquire,
130*67e74705SXin Li   // CHECK:   call i32 @__cxa_guard_acquire(i64* @_ZGVZN5test21BD1EvE1y)
131*67e74705SXin Li   // CHECK:   [[T0:%.*]] = call i32 @_ZN5test23fooEv()
132*67e74705SXin Li   // CHECK:   store i32 [[T0]], i32* @_ZZN5test21BD1EvE1y,
133*67e74705SXin Li   // CHECK:   call void @__cxa_guard_release(i64* @_ZGVZN5test21BD1EvE1y)
134*67e74705SXin Li 
135*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test21BD1Ev(
136*67e74705SXin Li   // CHECK:   call void @_ZN5test21BD2Ev(
137*67e74705SXin Li }
138*67e74705SXin Li 
139*67e74705SXin Li // This shouldn't error out.
140*67e74705SXin Li namespace test3 {
141*67e74705SXin Li   struct A {
142*67e74705SXin Li     A();
143*67e74705SXin Li     ~A();
144*67e74705SXin Li   };
145*67e74705SXin Li 
146*67e74705SXin Li   struct B : virtual A {
147*67e74705SXin Li     B();
148*67e74705SXin Li     ~B();
149*67e74705SXin Li   };
150*67e74705SXin Li 
B()151*67e74705SXin Li   B::B() {
152*67e74705SXin Li     union U { char x; int i; };
153*67e74705SXin Li     static U u = { 'a' };
154*67e74705SXin Li   }
155*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test31BC2Ev(
156*67e74705SXin Li   // CHECK-LABEL: define void @_ZN5test31BC1Ev(
157*67e74705SXin Li }
158*67e74705SXin Li 
159*67e74705SXin Li // We forgot to set the comdat when replacing the global with a different type.
160*67e74705SXin Li namespace test4 {
161*67e74705SXin Li struct HasVTable {
162*67e74705SXin Li   virtual void f();
163*67e74705SXin Li };
useStaticLocal()164*67e74705SXin Li inline HasVTable &useStaticLocal() {
165*67e74705SXin Li   static HasVTable obj;
166*67e74705SXin Li   return obj;
167*67e74705SXin Li }
useit()168*67e74705SXin Li void useit() {
169*67e74705SXin Li   useStaticLocal();
170*67e74705SXin Li }
171*67e74705SXin Li // CHECK: define linkonce_odr dereferenceable(8) %"struct.test4::HasVTable"* @_ZN5test414useStaticLocalEv()
172*67e74705SXin Li // CHECK: ret %"struct.test4::HasVTable"* @_ZZN5test414useStaticLocalEvE3obj
173*67e74705SXin Li }
174