xref: /aosp_15_r20/external/clang/test/CodeGenCXX/rtti-linkage.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -emit-llvm -o - | FileCheck %s -check-prefix=CHECK -check-prefix=CHECK-BOTH
2*67e74705SXin Li // RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -fvisibility hidden -emit-llvm -o - | FileCheck -check-prefix=CHECK-WITH-HIDDEN -check-prefix=CHECK-BOTH %s
3*67e74705SXin Li 
4*67e74705SXin Li #include <typeinfo>
5*67e74705SXin Li 
6*67e74705SXin Li // CHECK-BOTH: _ZTSP1C = internal constant
7*67e74705SXin Li // CHECK-BOTH: _ZTS1C = internal constant
8*67e74705SXin Li // CHECK-BOTH: _ZTI1C = internal constant
9*67e74705SXin Li // CHECK-BOTH: _ZTIP1C = internal constant
10*67e74705SXin Li // CHECK-BOTH: _ZTSPP1C = internal constant
11*67e74705SXin Li // CHECK-BOTH: _ZTIPP1C = internal constant
12*67e74705SXin Li // CHECK-BOTH: _ZTSM1Ci = internal constant
13*67e74705SXin Li // CHECK-BOTH: _ZTIM1Ci = internal constant
14*67e74705SXin Li // CHECK-BOTH: _ZTSPM1Ci = internal constant
15*67e74705SXin Li // CHECK-BOTH: _ZTIPM1Ci = internal constant
16*67e74705SXin Li // CHECK-BOTH: _ZTSM1CS_ = internal constant
17*67e74705SXin Li // CHECK-BOTH: _ZTIM1CS_ = internal constant
18*67e74705SXin Li // CHECK-BOTH: _ZTSM1CPS_ = internal constant
19*67e74705SXin Li // CHECK-BOTH: _ZTIM1CPS_ = internal constant
20*67e74705SXin Li // CHECK-BOTH: _ZTSM1A1C = internal constant
21*67e74705SXin Li // CHECK: _ZTS1A = linkonce_odr constant
22*67e74705SXin Li // CHECK-WITH-HIDDEN: _ZTS1A = linkonce_odr hidden constant
23*67e74705SXin Li // CHECK: _ZTI1A = linkonce_odr constant
24*67e74705SXin Li // CHECK-WITH-HIDDEN: _ZTI1A = linkonce_odr hidden constant
25*67e74705SXin Li // CHECK-BOTH: _ZTIM1A1C = internal constant
26*67e74705SXin Li // CHECK-BOTH: _ZTSM1AP1C = internal constant
27*67e74705SXin Li // CHECK-BOTH: _ZTIM1AP1C = internal constant
28*67e74705SXin Li 
29*67e74705SXin Li // CHECK-WITH-HIDDEN: _ZTSFN12_GLOBAL__N_11DEvE = internal constant
30*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTSPK2T4 = linkonce_odr hidden constant
31*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTS2T4 = linkonce_odr hidden constant
32*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTI2T4 = linkonce_odr hidden constant
33*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTIPK2T4 = linkonce_odr hidden constant
34*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTSZ2t5vE1A = internal constant
35*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTIZ2t5vE1A = internal constant
36*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTSZ2t6vE1A = linkonce_odr hidden constant
37*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTIZ2t6vE1A = linkonce_odr hidden constant
38*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTSPZ2t7vE1A = linkonce_odr hidden constant
39*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTSZ2t7vE1A = linkonce_odr hidden constant
40*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTIZ2t7vE1A = linkonce_odr hidden constant
41*67e74705SXin Li // CHECK-WITH-HIDDEN: @_ZTIPZ2t7vE1A = linkonce_odr hidden constant
42*67e74705SXin Li 
43*67e74705SXin Li // CHECK: _ZTSN12_GLOBAL__N_11DE = internal constant
44*67e74705SXin Li // CHECK: _ZTIN12_GLOBAL__N_11DE = internal constant
45*67e74705SXin Li // CHECK: _ZTSPN12_GLOBAL__N_11DE = internal constant
46*67e74705SXin Li // CHECK: _ZTIPN12_GLOBAL__N_11DE = internal constant
47*67e74705SXin Li // CHECK: _ZTSFN12_GLOBAL__N_11DEvE = internal constant
48*67e74705SXin Li // CHECK: _ZTIFN12_GLOBAL__N_11DEvE = internal constant
49*67e74705SXin Li // CHECK: _ZTSFvN12_GLOBAL__N_11DEE = internal constant
50*67e74705SXin Li // CHECK: _ZTIFvN12_GLOBAL__N_11DEE = internal constant
51*67e74705SXin Li // CHECK: _ZTSPFvvE = linkonce_odr constant
52*67e74705SXin Li // CHECK: _ZTSFvvE = linkonce_odr constant
53*67e74705SXin Li // CHECK: _ZTIFvvE = linkonce_odr constant
54*67e74705SXin Li // CHECK: _ZTIPFvvE = linkonce_odr constant
55*67e74705SXin Li // CHECK: _ZTSN12_GLOBAL__N_11EE = internal constant
56*67e74705SXin Li // CHECK: _ZTIN12_GLOBAL__N_11EE = internal constant
57*67e74705SXin Li // CHECK: _ZTSA10_i = linkonce_odr constant
58*67e74705SXin Li // CHECK: _ZTIA10_i = linkonce_odr constant
59*67e74705SXin Li // CHECK: _ZTI1TILj0EE = linkonce_odr constant
60*67e74705SXin Li // CHECK: _ZTI1TILj1EE = weak_odr constant
61*67e74705SXin Li // CHECK: _ZTI1TILj2EE = external constant
62*67e74705SXin Li // CHECK: _ZTSZ2t5vE1A = internal constant
63*67e74705SXin Li // CHECK: _ZTIZ2t5vE1A = internal constant
64*67e74705SXin Li // CHECK: _ZTS1B = constant
65*67e74705SXin Li // CHECK: _ZTI1B = constant
66*67e74705SXin Li // CHECK: _ZTS1F = linkonce_odr constant
67*67e74705SXin Li // CHECK: _ZTSZ2t6vE1A = linkonce_odr constant
68*67e74705SXin Li // CHECK: _ZTIZ2t6vE1A = linkonce_odr constant
69*67e74705SXin Li // CHECK: _ZTSPZ2t7vE1A = linkonce_odr constant
70*67e74705SXin Li // CHECK: _ZTSZ2t7vE1A = linkonce_odr constant
71*67e74705SXin Li // CHECK: _ZTIZ2t7vE1A = linkonce_odr constant
72*67e74705SXin Li // CHECK: _ZTIPZ2t7vE1A = linkonce_odr constant
73*67e74705SXin Li 
74*67e74705SXin Li // CHECK: _ZTIN12_GLOBAL__N_11DE to
75*67e74705SXin Li 
76*67e74705SXin Li // A has no key function, so its RTTI data should be linkonce_odr.
77*67e74705SXin Li struct A { };
78*67e74705SXin Li 
79*67e74705SXin Li // B has a key function defined in the translation unit, so the RTTI data should
80*67e74705SXin Li // be emitted in this translation unit and have external linkage.
81*67e74705SXin Li struct B : A {
82*67e74705SXin Li   virtual void f();
83*67e74705SXin Li };
f()84*67e74705SXin Li void B::f() { }
85*67e74705SXin Li 
86*67e74705SXin Li // C is an incomplete class type, so any direct or indirect pointer types should have
87*67e74705SXin Li // internal linkage, as should the type info for C itself.
88*67e74705SXin Li struct C;
89*67e74705SXin Li 
t1()90*67e74705SXin Li void t1() {
91*67e74705SXin Li   (void)typeid(C*);
92*67e74705SXin Li   (void)typeid(C**);
93*67e74705SXin Li   (void)typeid(int C::*);
94*67e74705SXin Li   (void)typeid(int C::**);
95*67e74705SXin Li   (void)typeid(C C::*);
96*67e74705SXin Li   (void)typeid(C *C::*);
97*67e74705SXin Li   (void)typeid(C A::*);
98*67e74705SXin Li   (void)typeid(C* A::*);
99*67e74705SXin Li }
100*67e74705SXin Li 
101*67e74705SXin Li namespace {
102*67e74705SXin Li   // D is inside an anonymous namespace, so all type information related to D should have
103*67e74705SXin Li   // internal linkage.
104*67e74705SXin Li   struct D { };
105*67e74705SXin Li 
106*67e74705SXin Li   // E is also inside an anonymous namespace.
107*67e74705SXin Li   enum E { };
108*67e74705SXin Li 
109*67e74705SXin Li };
110*67e74705SXin Li 
111*67e74705SXin Li // F has a key function defined in the translation unit, but it is inline so the RTTI
112*67e74705SXin Li // data should be emitted with linkonce_odr linkage.
113*67e74705SXin Li struct F {
114*67e74705SXin Li   virtual void f();
115*67e74705SXin Li };
116*67e74705SXin Li 
f()117*67e74705SXin Li inline void F::f() { }
118*67e74705SXin Li const D getD();
119*67e74705SXin Li 
t2()120*67e74705SXin Li const std::type_info &t2() {
121*67e74705SXin Li   (void)typeid(const D);
122*67e74705SXin Li   (void)typeid(D *);
123*67e74705SXin Li   (void)typeid(D (*)());
124*67e74705SXin Li   (void)typeid(void (*)(D));
125*67e74705SXin Li   (void)typeid(void (*)(D&));
126*67e74705SXin Li   // The exception specification is not part of the RTTI descriptor, so it should not have
127*67e74705SXin Li   // internal linkage.
128*67e74705SXin Li   (void)typeid(void (*)() throw (D));
129*67e74705SXin Li 
130*67e74705SXin Li   (void)typeid(E);
131*67e74705SXin Li 
132*67e74705SXin Li   return typeid(getD());
133*67e74705SXin Li }
134*67e74705SXin Li 
135*67e74705SXin Li namespace Arrays {
136*67e74705SXin Li   struct A {
137*67e74705SXin Li     static const int a[10];
138*67e74705SXin Li   };
f()139*67e74705SXin Li   const std::type_info &f() {
140*67e74705SXin Li     return typeid(A::a);
141*67e74705SXin Li   }
142*67e74705SXin Li }
143*67e74705SXin Li 
144*67e74705SXin Li template <unsigned N> class T {
anchor()145*67e74705SXin Li   virtual void anchor() {}
146*67e74705SXin Li };
147*67e74705SXin Li template class T<1>;
148*67e74705SXin Li template <> class T<2> { virtual void anchor(); };
t3()149*67e74705SXin Li void t3() {
150*67e74705SXin Li   (void) typeid(T<0>);
151*67e74705SXin Li   (void) typeid(T<1>);
152*67e74705SXin Li   (void) typeid(T<2>);
153*67e74705SXin Li }
154*67e74705SXin Li 
155*67e74705SXin Li // rdar://problem/8778973
156*67e74705SXin Li struct T4 {};
t4(const T4 * ptr)157*67e74705SXin Li void t4(const T4 *ptr) {
158*67e74705SXin Li   const void *value = &typeid(ptr);
159*67e74705SXin Li }
160*67e74705SXin Li 
161*67e74705SXin Li // rdar://16265084
t5()162*67e74705SXin Li void t5() {
163*67e74705SXin Li   struct A {};
164*67e74705SXin Li   const void *value = &typeid(A);
165*67e74705SXin Li }
166*67e74705SXin Li 
t6()167*67e74705SXin Li inline void t6() {
168*67e74705SXin Li   struct A {};
169*67e74705SXin Li   const void *value = &typeid(A);
170*67e74705SXin Li }
t6_helper()171*67e74705SXin Li void t6_helper() {
172*67e74705SXin Li   t6();
173*67e74705SXin Li }
174*67e74705SXin Li 
t7()175*67e74705SXin Li inline void t7() {
176*67e74705SXin Li   struct A {};
177*67e74705SXin Li   const void *value = &typeid(A*);
178*67e74705SXin Li }
t7_helper()179*67e74705SXin Li void t7_helper() {
180*67e74705SXin Li   t7();
181*67e74705SXin Li }
182