206 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			206 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_cc1 %s -triple=x86_64-apple-darwin10 -emit-llvm -o %t
 | |
| // RUN: %clang_cc1 %s -triple=x86_64-apple-darwin10 -disable-llvm-optzns -O3 -emit-llvm -o %t.opt
 | |
| // RUN: FileCheck --check-prefix=CHECK %s < %t
 | |
| // RUN: FileCheck --check-prefix=CHECK-OPT %s < %t.opt
 | |
| 
 | |
| namespace {
 | |
|   struct A {
 | |
|     virtual void f() { }
 | |
|   };
 | |
| }
 | |
| 
 | |
| void f() { A b; }
 | |
| 
 | |
| struct B {
 | |
|   B();
 | |
|   virtual void f();
 | |
| };
 | |
| 
 | |
| B::B() { }
 | |
| 
 | |
| struct C : virtual B {
 | |
|   C();
 | |
|   virtual void f() { } 
 | |
| };
 | |
| 
 | |
| C::C() { } 
 | |
| 
 | |
| struct D {
 | |
|   virtual void f();
 | |
| };
 | |
| 
 | |
| void D::f() { }
 | |
| 
 | |
| static struct : D { } e;
 | |
| 
 | |
| // The destructor is the key function.
 | |
| template<typename T>
 | |
| struct E {
 | |
|   virtual ~E();
 | |
| };
 | |
| 
 | |
| template<typename T> E<T>::~E() { }
 | |
| 
 | |
| // Anchor is the key function
 | |
| template<>
 | |
| struct E<char> {
 | |
|   virtual void anchor();
 | |
| };
 | |
| 
 | |
| void E<char>::anchor() { }
 | |
| 
 | |
| template struct E<short>;
 | |
| extern template struct E<int>;
 | |
| 
 | |
| void use_E() {
 | |
|   E<int> ei;
 | |
|   (void)ei;
 | |
|   E<long> el;
 | |
|   (void)el;
 | |
| }
 | |
| 
 | |
| // No key function
 | |
| template<typename T>
 | |
| struct F {
 | |
|   virtual void foo() { }
 | |
| };
 | |
| 
 | |
| // No key function
 | |
| template<>
 | |
| struct F<char> {
 | |
|   virtual void foo() { }
 | |
| };
 | |
| 
 | |
| template struct F<short>;
 | |
| extern template struct F<int>;
 | |
| 
 | |
| void use_F() {
 | |
|   F<char> fc;
 | |
|   fc.foo();
 | |
|   F<int> fi;
 | |
|   fi.foo();
 | |
|   F<long> fl;
 | |
|   (void)fl;
 | |
| }
 | |
| 
 | |
| // B has a key function that is not defined in this translation unit so its vtable
 | |
| // has external linkage.
 | |
| // CHECK-DAG: @_ZTV1B = external unnamed_addr constant
 | |
| 
 | |
| // C has no key function, so its vtable should have weak_odr linkage
 | |
| // and hidden visibility (rdar://problem/7523229).
 | |
| // CHECK-DAG: @_ZTV1C = linkonce_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1C = linkonce_odr constant
 | |
| // CHECK-DAG: @_ZTI1C = linkonce_odr constant
 | |
| // CHECK-DAG: @_ZTT1C = linkonce_odr unnamed_addr constant
 | |
| 
 | |
| // D has a key function that is defined in this translation unit so its vtable is
 | |
| // defined in the translation unit.
 | |
| // CHECK-DAG: @_ZTV1D = unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1D = constant
 | |
| // CHECK-DAG: @_ZTI1D = constant
 | |
| 
 | |
| // E<char> is an explicit specialization with a key function defined
 | |
| // in this translation unit, so its vtable should have external
 | |
| // linkage.
 | |
| // CHECK-DAG: @_ZTV1EIcE = unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1EIcE = constant
 | |
| // CHECK-DAG: @_ZTI1EIcE = constant
 | |
| 
 | |
| // E<short> is an explicit template instantiation with a key function
 | |
| // defined in this translation unit, so its vtable should have
 | |
| // weak_odr linkage.
 | |
| // CHECK-DAG: @_ZTV1EIsE = weak_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1EIsE = weak_odr constant
 | |
| // CHECK-DAG: @_ZTI1EIsE = weak_odr constant
 | |
| 
 | |
| // F<short> is an explicit template instantiation without a key
 | |
| // function, so its vtable should have weak_odr linkage
 | |
| // CHECK-DAG: @_ZTV1FIsE = weak_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1FIsE = weak_odr constant
 | |
| // CHECK-DAG: @_ZTI1FIsE = weak_odr constant
 | |
| 
 | |
| // E<long> is an implicit template instantiation with a key function
 | |
| // defined in this translation unit, so its vtable should have
 | |
| // linkonce_odr linkage.
 | |
| // CHECK-DAG: @_ZTV1EIlE = linkonce_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1EIlE = linkonce_odr constant
 | |
| // CHECK-DAG: @_ZTI1EIlE = linkonce_odr constant
 | |
| 
 | |
| // F<long> is an implicit template instantiation with no key function,
 | |
| // so its vtable should have linkonce_odr linkage.
 | |
| // CHECK-DAG: @_ZTV1FIlE = linkonce_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1FIlE = linkonce_odr constant
 | |
| // CHECK-DAG: @_ZTI1FIlE = linkonce_odr constant
 | |
| 
 | |
| // F<int> is an explicit template instantiation declaration without a
 | |
| // key function, so its vtable should have external linkage.
 | |
| // CHECK-DAG: @_ZTV1FIiE = external unnamed_addr constant
 | |
| // CHECK-OPT-DAG: @_ZTV1FIiE = external unnamed_addr constant
 | |
| 
 | |
| // E<int> is an explicit template instantiation declaration. It has a
 | |
| // key function that is not instantiated, so we should only reference
 | |
| // its vtable, not define it.
 | |
| // CHECK-DAG: @_ZTV1EIiE = external unnamed_addr constant
 | |
| // CHECK-OPT-DAG: @_ZTV1EIiE = external unnamed_addr constant
 | |
| 
 | |
| // The anonymous struct for e has no linkage, so the vtable should have
 | |
| // internal linkage.
 | |
| // CHECK-DAG: @"_ZTV3$_0" = internal unnamed_addr constant
 | |
| // CHECK-DAG: @"_ZTS3$_0" = internal constant
 | |
| // CHECK-DAG: @"_ZTI3$_0" = internal constant
 | |
| 
 | |
| // The A vtable should have internal linkage since it is inside an anonymous 
 | |
| // namespace.
 | |
| // CHECK-DAG: @_ZTVN12_GLOBAL__N_11AE = internal unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTSN12_GLOBAL__N_11AE = internal constant
 | |
| // CHECK-DAG: @_ZTIN12_GLOBAL__N_11AE = internal constant
 | |
| 
 | |
| // F<char> is an explicit specialization without a key function, so
 | |
| // its vtable should have linkonce_odr linkage.
 | |
| // CHECK-DAG: @_ZTV1FIcE = linkonce_odr unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTS1FIcE = linkonce_odr constant
 | |
| // CHECK-DAG: @_ZTI1FIcE = linkonce_odr constant
 | |
| 
 | |
| // CHECK-DAG: @_ZTV1GIiE = linkonce_odr unnamed_addr constant
 | |
| template <typename T>
 | |
| class G {
 | |
| public:
 | |
|   G() {}
 | |
|   virtual void f0();
 | |
|   virtual void f1();
 | |
| };
 | |
| template <>
 | |
| void G<int>::f1() {}
 | |
| template <typename T>
 | |
| void G<T>::f0() {}
 | |
| void G_f0()  { new G<int>(); }
 | |
| 
 | |
| // H<int> has a key function without a body but it's a template instantiation
 | |
| // so its VTable must be emitted.
 | |
| // CHECK-DAG: @_ZTV1HIiE = linkonce_odr unnamed_addr constant
 | |
| template <typename T>
 | |
| class H {
 | |
| public:
 | |
|   virtual ~H();
 | |
| };
 | |
| 
 | |
| void use_H() {
 | |
|   H<int> h;
 | |
| }
 | |
| 
 | |
| // I<int> has an explicit instantiation declaration and needs a VTT and
 | |
| // construction vtables.
 | |
| 
 | |
| // CHECK-DAG: @_ZTV1IIiE = external unnamed_addr constant
 | |
| // CHECK-DAG: @_ZTT1IIiE = external unnamed_addr constant
 | |
| // CHECK-NOT: @_ZTC1IIiE
 | |
| //
 | |
| // CHECK-OPT-DAG: @_ZTV1IIiE = external unnamed_addr constant
 | |
| // CHECK-OPT-DAG: @_ZTT1IIiE = external unnamed_addr constant
 | |
| struct VBase1 { virtual void f(); }; struct VBase2 : virtual VBase1 {};
 | |
| template<typename T>
 | |
| struct I : VBase2 {};
 | |
| extern template struct I<int>;
 | |
| I<int> i;
 |