annotate clang/test/CodeGenCXX/template-instantiation.cpp @ 150:1d019706d866

LLVM10
author anatofuz
date Thu, 13 Feb 2020 15:10:13 +0900
parents
children 2e18cbf3894f
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 // RUN: %clang_cc1 %s -O1 -disable-llvm-passes -triple=x86_64-apple-darwin10 -std=c++11 -emit-llvm -o - | FileCheck %s
anatofuz
parents:
diff changeset
2 // RUN: %clang_cc1 %s -O1 -disable-llvm-passes -triple=x86_64-apple-darwin10 -std=c++11 -emit-llvm -o - | FileCheck %s --check-prefix=CHECK2
anatofuz
parents:
diff changeset
3
anatofuz
parents:
diff changeset
4 // Instantiation order varies on different C++ dialects (IE, between C++98 and C++11).
anatofuz
parents:
diff changeset
5 // CHECK-DAG: @_ZN7PR100011xE = global
anatofuz
parents:
diff changeset
6 // CHECK-DAG: @_ZTVN5test018stdio_sync_filebufIA3_iEE = weak_odr unnamed_addr constant
anatofuz
parents:
diff changeset
7 // CHECK-DAG: @_ZN7PR100011SIiE3arrE = linkonce_odr global [3 x i32]
anatofuz
parents:
diff changeset
8 // CHECK-DAG: @_ZTVN5test018stdio_sync_filebufIA4_iEE = linkonce_odr unnamed_addr constant
anatofuz
parents:
diff changeset
9
anatofuz
parents:
diff changeset
10 // Negative checks go under prefix "CHECK2" to avoid interference with CHECK and CHECK-DAG.
anatofuz
parents:
diff changeset
11 // CHECK2-NOT: @_ZN7PR100014kBarE = external global i32
anatofuz
parents:
diff changeset
12 // CHECK2-NOT: @_ZTVN5test118stdio_sync_filebufIwEE = constant
anatofuz
parents:
diff changeset
13 // CHECK2-NOT: _ZTVN5test315basic_fstreamXXIcEE
anatofuz
parents:
diff changeset
14 // CHECK2-NOT: @_ZTVN5test018stdio_sync_filebufIA1_iEE
anatofuz
parents:
diff changeset
15 // CHECK2-NOT: @_ZTVN5test018stdio_sync_filebufIA2_iEE
anatofuz
parents:
diff changeset
16 // CHECK2-NOT: @_ZN7PR100011SIiE3arr2E = linkonce_odr global [3 x i32]A
anatofuz
parents:
diff changeset
17
anatofuz
parents:
diff changeset
18 // CHECK2-NOT: _ZTVN5test31SIiEE
anatofuz
parents:
diff changeset
19 // CHECK2-NOT: _ZTSN5test31SIiEE
anatofuz
parents:
diff changeset
20
anatofuz
parents:
diff changeset
21 // CHECK-LABEL: define linkonce_odr void @_ZN5test21CIiEC1Ev(%"class.test2::C"* %this) unnamed_addr
anatofuz
parents:
diff changeset
22 // CHECK-LABEL: define linkonce_odr void @_ZN5test21CIiE6foobarIdEEvT_(
anatofuz
parents:
diff changeset
23 // CHECK-LABEL: define available_externally void @_ZN5test21CIiE6zedbarEd(
anatofuz
parents:
diff changeset
24
anatofuz
parents:
diff changeset
25 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g1ENS_1SILi1EEE()
anatofuz
parents:
diff changeset
26 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g1ENS_1SILi2EEE()
anatofuz
parents:
diff changeset
27 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g1ENS_1SILi3EEE()
anatofuz
parents:
diff changeset
28 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g2ENS_1SILi1EEE()
anatofuz
parents:
diff changeset
29 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g2ENS_1SILi2EEE()
anatofuz
parents:
diff changeset
30 // CHECK-LABEL: define linkonce_odr void @_ZN7PR106662g2ENS_1SILi3EEE()
anatofuz
parents:
diff changeset
31 // CHECK: declare void @_ZN7PR106662h1ENS_1SILi1EEE()
anatofuz
parents:
diff changeset
32 // CHECK: declare void @_ZN7PR106662h1ENS_1SILi2EEE()
anatofuz
parents:
diff changeset
33 // CHECK: declare void @_ZN7PR106662h1ENS_1SILi3EEE()
anatofuz
parents:
diff changeset
34 // CHECK: declare void @_ZN7PR106662h2ENS_1SILi1EEE()
anatofuz
parents:
diff changeset
35 // CHECK: declare void @_ZN7PR106662h2ENS_1SILi2EEE()
anatofuz
parents:
diff changeset
36 // CHECK: declare void @_ZN7PR106662h2ENS_1SILi3EEE()
anatofuz
parents:
diff changeset
37
anatofuz
parents:
diff changeset
38 namespace test0 {
anatofuz
parents:
diff changeset
39 struct basic_streambuf {
anatofuz
parents:
diff changeset
40 virtual ~basic_streambuf();
anatofuz
parents:
diff changeset
41 };
anatofuz
parents:
diff changeset
42 template<typename _CharT >
anatofuz
parents:
diff changeset
43 struct stdio_sync_filebuf : public basic_streambuf {
anatofuz
parents:
diff changeset
44 virtual void xsgetn();
anatofuz
parents:
diff changeset
45 };
anatofuz
parents:
diff changeset
46
anatofuz
parents:
diff changeset
47 // This specialization is not a key function, so doesn't cause the vtable to
anatofuz
parents:
diff changeset
48 // be instantiated unless we're instantiating a class definition anyway.
anatofuz
parents:
diff changeset
49 template<> void stdio_sync_filebuf<int[1]>::xsgetn() {
anatofuz
parents:
diff changeset
50 }
anatofuz
parents:
diff changeset
51 template<> void stdio_sync_filebuf<int[2]>::xsgetn() {
anatofuz
parents:
diff changeset
52 }
anatofuz
parents:
diff changeset
53 template<> void stdio_sync_filebuf<int[3]>::xsgetn() {
anatofuz
parents:
diff changeset
54 }
anatofuz
parents:
diff changeset
55 template<> void stdio_sync_filebuf<int[4]>::xsgetn() {
anatofuz
parents:
diff changeset
56 }
anatofuz
parents:
diff changeset
57 extern template class stdio_sync_filebuf<int[2]>;
anatofuz
parents:
diff changeset
58
anatofuz
parents:
diff changeset
59 // These two both cause vtables to be emitted.
anatofuz
parents:
diff changeset
60 template class stdio_sync_filebuf<int[3]>;
anatofuz
parents:
diff changeset
61 stdio_sync_filebuf<int[4]> implicit_instantiation;
anatofuz
parents:
diff changeset
62 }
anatofuz
parents:
diff changeset
63
anatofuz
parents:
diff changeset
64 namespace test1 {
anatofuz
parents:
diff changeset
65 struct basic_streambuf {
anatofuz
parents:
diff changeset
66 virtual ~basic_streambuf();
anatofuz
parents:
diff changeset
67 };
anatofuz
parents:
diff changeset
68 template<typename _CharT >
anatofuz
parents:
diff changeset
69 struct stdio_sync_filebuf : public basic_streambuf {
anatofuz
parents:
diff changeset
70 virtual void xsgetn();
anatofuz
parents:
diff changeset
71 };
anatofuz
parents:
diff changeset
72
anatofuz
parents:
diff changeset
73 // Just a declaration should not force the vtable to be emitted.
anatofuz
parents:
diff changeset
74 template<> void stdio_sync_filebuf<wchar_t>::xsgetn();
anatofuz
parents:
diff changeset
75 }
anatofuz
parents:
diff changeset
76
anatofuz
parents:
diff changeset
77 namespace test2 {
anatofuz
parents:
diff changeset
78 template<typename T1>
anatofuz
parents:
diff changeset
79 class C {
anatofuz
parents:
diff changeset
80 public:
anatofuz
parents:
diff changeset
81 virtual ~C();
anatofuz
parents:
diff changeset
82 void zedbar(double) {
anatofuz
parents:
diff changeset
83 }
anatofuz
parents:
diff changeset
84 template<typename T2>
anatofuz
parents:
diff changeset
85 void foobar(T2 foo) {
anatofuz
parents:
diff changeset
86 }
anatofuz
parents:
diff changeset
87 };
anatofuz
parents:
diff changeset
88 extern template class C<int>;
anatofuz
parents:
diff changeset
89 void g() {
anatofuz
parents:
diff changeset
90 // The extern template declaration should not prevent us from producing
anatofuz
parents:
diff changeset
91 // the implicit constructor (test at the top).
anatofuz
parents:
diff changeset
92 C<int> a;
anatofuz
parents:
diff changeset
93
anatofuz
parents:
diff changeset
94 // or foobar(test at the top).
anatofuz
parents:
diff changeset
95 a.foobar(0.0);
anatofuz
parents:
diff changeset
96
anatofuz
parents:
diff changeset
97 // But it should prevent zebbar
anatofuz
parents:
diff changeset
98 // (test at the top).
anatofuz
parents:
diff changeset
99 a.zedbar(0.0);
anatofuz
parents:
diff changeset
100 }
anatofuz
parents:
diff changeset
101 }
anatofuz
parents:
diff changeset
102
anatofuz
parents:
diff changeset
103 namespace test3 {
anatofuz
parents:
diff changeset
104 template<typename T>
anatofuz
parents:
diff changeset
105 class basic_fstreamXX {
anatofuz
parents:
diff changeset
106 virtual void foo(){}
anatofuz
parents:
diff changeset
107 virtual void is_open() const { }
anatofuz
parents:
diff changeset
108 };
anatofuz
parents:
diff changeset
109
anatofuz
parents:
diff changeset
110 extern template class basic_fstreamXX<char>;
anatofuz
parents:
diff changeset
111 // This template instantiation should not cause us to produce a vtable.
anatofuz
parents:
diff changeset
112 // (test at the top).
anatofuz
parents:
diff changeset
113 template void basic_fstreamXX<char>::is_open() const;
anatofuz
parents:
diff changeset
114 }
anatofuz
parents:
diff changeset
115
anatofuz
parents:
diff changeset
116 namespace test3 {
anatofuz
parents:
diff changeset
117 template <typename T>
anatofuz
parents:
diff changeset
118 struct S {
anatofuz
parents:
diff changeset
119 virtual void m();
anatofuz
parents:
diff changeset
120 };
anatofuz
parents:
diff changeset
121
anatofuz
parents:
diff changeset
122 template<typename T>
anatofuz
parents:
diff changeset
123 void S<T>::m() { }
anatofuz
parents:
diff changeset
124
anatofuz
parents:
diff changeset
125 // Should not cause us to produce vtable because template instantiations
anatofuz
parents:
diff changeset
126 // don't have key functions.
anatofuz
parents:
diff changeset
127 template void S<int>::m();
anatofuz
parents:
diff changeset
128 }
anatofuz
parents:
diff changeset
129
anatofuz
parents:
diff changeset
130 namespace test4 {
anatofuz
parents:
diff changeset
131 template <class T> struct A { static void foo(); };
anatofuz
parents:
diff changeset
132
anatofuz
parents:
diff changeset
133 class B {
anatofuz
parents:
diff changeset
134 template <class T> friend void A<T>::foo();
anatofuz
parents:
diff changeset
135 B();
anatofuz
parents:
diff changeset
136 };
anatofuz
parents:
diff changeset
137
anatofuz
parents:
diff changeset
138 template <class T> void A<T>::foo() {
anatofuz
parents:
diff changeset
139 B b;
anatofuz
parents:
diff changeset
140 }
anatofuz
parents:
diff changeset
141
anatofuz
parents:
diff changeset
142 unsigned test() {
anatofuz
parents:
diff changeset
143 A<int>::foo();
anatofuz
parents:
diff changeset
144 }
anatofuz
parents:
diff changeset
145 }
anatofuz
parents:
diff changeset
146
anatofuz
parents:
diff changeset
147 namespace PR8505 {
anatofuz
parents:
diff changeset
148 // Hits an assertion due to bogus instantiation of class B.
anatofuz
parents:
diff changeset
149 template <int i> class A {
anatofuz
parents:
diff changeset
150 class B* g;
anatofuz
parents:
diff changeset
151 };
anatofuz
parents:
diff changeset
152 class B {
anatofuz
parents:
diff changeset
153 void f () {}
anatofuz
parents:
diff changeset
154 };
anatofuz
parents:
diff changeset
155 // Should not instantiate class B since it is introduced in namespace scope.
anatofuz
parents:
diff changeset
156 // CHECK2-NOT: _ZN6PR85051AILi0EE1B1fEv
anatofuz
parents:
diff changeset
157 template class A<0>;
anatofuz
parents:
diff changeset
158 }
anatofuz
parents:
diff changeset
159
anatofuz
parents:
diff changeset
160 // Ensure that when instantiating initializers for static data members to
anatofuz
parents:
diff changeset
161 // complete their type in an unevaluated context, we *do* emit initializers with
anatofuz
parents:
diff changeset
162 // side-effects, but *don't* emit initializers and variables which are otherwise
anatofuz
parents:
diff changeset
163 // unused in the program.
anatofuz
parents:
diff changeset
164 namespace PR10001 {
anatofuz
parents:
diff changeset
165 template <typename T> struct S {
anatofuz
parents:
diff changeset
166 static const int arr[];
anatofuz
parents:
diff changeset
167 static const int arr2[];
anatofuz
parents:
diff changeset
168 static const int x, y;
anatofuz
parents:
diff changeset
169 static int f();
anatofuz
parents:
diff changeset
170 };
anatofuz
parents:
diff changeset
171
anatofuz
parents:
diff changeset
172 extern int foo();
anatofuz
parents:
diff changeset
173 extern int kBar;
anatofuz
parents:
diff changeset
174
anatofuz
parents:
diff changeset
175 template <typename T> const int S<T>::arr[] = { 1, 2, foo() }; // possible side effects
anatofuz
parents:
diff changeset
176 template <typename T> const int S<T>::arr2[] = { 1, 2, kBar }; // no side effects
anatofuz
parents:
diff changeset
177 template <typename T> const int S<T>::x = sizeof(arr) / sizeof(arr[0]);
anatofuz
parents:
diff changeset
178 template <typename T> const int S<T>::y = sizeof(arr2) / sizeof(arr2[0]);
anatofuz
parents:
diff changeset
179 template <typename T> int S<T>::f() { return x + y; }
anatofuz
parents:
diff changeset
180
anatofuz
parents:
diff changeset
181 int x = S<int>::f();
anatofuz
parents:
diff changeset
182 }
anatofuz
parents:
diff changeset
183
anatofuz
parents:
diff changeset
184 // Ensure that definitions are emitted for all friend functions defined within
anatofuz
parents:
diff changeset
185 // class templates. Order of declaration is extremely important here. Different
anatofuz
parents:
diff changeset
186 // instantiations of the class happen at different points during the deferred
anatofuz
parents:
diff changeset
187 // method body parsing and afterward. Those different points of instantiation
anatofuz
parents:
diff changeset
188 // change the exact form the class template appears to have.
anatofuz
parents:
diff changeset
189 namespace PR10666 {
anatofuz
parents:
diff changeset
190 template <int N> struct S {
anatofuz
parents:
diff changeset
191 void f1() { S<1> s; }
anatofuz
parents:
diff changeset
192 friend void g1(S s) {}
anatofuz
parents:
diff changeset
193 friend void h1(S s);
anatofuz
parents:
diff changeset
194 void f2() { S<2> s; }
anatofuz
parents:
diff changeset
195 friend void g2(S s) {}
anatofuz
parents:
diff changeset
196 friend void h2(S s);
anatofuz
parents:
diff changeset
197 void f3() { S<3> s; }
anatofuz
parents:
diff changeset
198 };
anatofuz
parents:
diff changeset
199 void test(S<1> s1, S<2> s2, S<3> s3) {
anatofuz
parents:
diff changeset
200 g1(s1); g1(s2); g1(s3);
anatofuz
parents:
diff changeset
201 g2(s1); g2(s2); g2(s3);
anatofuz
parents:
diff changeset
202 h1(s1); h1(s2); h1(s3);
anatofuz
parents:
diff changeset
203 h2(s1); h2(s2); h2(s3);
anatofuz
parents:
diff changeset
204 }
anatofuz
parents:
diff changeset
205 }