|  | // RUN: %clang_cc1 %s -fsyntax-only -verify -triple %itanium_abi_triple -Wweak-vtables -Wweak-template-vtables | 
|  | // RUN: %clang_cc1 %s -fsyntax-only -triple %ms_abi_triple -Werror -Wno-weak-vtables -Wno-weak-template-vtables | 
|  |  | 
|  | struct A { // expected-warning {{'A' has no out-of-line virtual method definitions; its vtable will be emitted in every translation unit}} | 
|  | virtual void f() { } | 
|  | }; | 
|  |  | 
|  | template<typename T> struct B { | 
|  | virtual void f() { } | 
|  | }; | 
|  |  | 
|  | namespace { | 
|  | struct C { | 
|  | virtual void f() { } | 
|  | }; | 
|  | } | 
|  |  | 
|  | void f() { | 
|  | struct A { | 
|  | virtual void f() { } | 
|  | }; | 
|  |  | 
|  | A *a; | 
|  | a->f(); | 
|  | } | 
|  |  | 
|  | // Use the vtables | 
|  | void uses(A &a, B<int> &b, C &c) { | 
|  | a.f(); | 
|  | b.f(); | 
|  | c.f(); | 
|  | } | 
|  |  | 
|  | // <rdar://problem/9979458> | 
|  | class Parent { | 
|  | public: | 
|  | Parent() {} | 
|  | virtual ~Parent(); | 
|  | virtual void * getFoo() const = 0; | 
|  | }; | 
|  |  | 
|  | class Derived : public Parent { | 
|  | public: | 
|  | Derived(); | 
|  | void * getFoo() const; | 
|  | }; | 
|  |  | 
|  | class VeryDerived : public Derived { // expected-warning{{'VeryDerived' has no out-of-line virtual method definitions; its vtable will be emitted in every translation unit}} | 
|  | public: | 
|  | void * getFoo() const { return 0; } | 
|  | }; | 
|  |  | 
|  | Parent::~Parent() {} | 
|  |  | 
|  | void uses(Parent &p, Derived &d, VeryDerived &vd) { | 
|  | p.getFoo(); | 
|  | d.getFoo(); | 
|  | vd.getFoo(); | 
|  | } | 
|  |  | 
|  | template<typename T> struct TemplVirt { | 
|  | virtual void f(); | 
|  | }; | 
|  |  | 
|  | template class TemplVirt<float>; // expected-warning{{explicit template instantiation 'TemplVirt<float>' will emit a vtable in every translation unit}} | 
|  |  | 
|  | template<> struct TemplVirt<bool> { | 
|  | virtual void f(); | 
|  | }; | 
|  |  | 
|  | template<> struct TemplVirt<long> { // expected-warning{{'TemplVirt<long>' has no out-of-line virtual method definitions; its vtable will be emitted in every translation unit}} | 
|  | virtual void f() {} | 
|  | }; | 
|  |  | 
|  | void uses(TemplVirt<float>& f, TemplVirt<bool>& b, TemplVirt<long>& l) { | 
|  | f.f(); | 
|  | b.f(); | 
|  | l.f(); | 
|  | } |