// RUN: %clang_cc1 -fsyntax-only -triple %itanium_abi_triple -verify %s -std=c++11 // RUN: %clang_cc1 -fsyntax-only -triple %ms_abi_triple -verify %s -std=c++11 namespace T1 { class A { virtual int f(); // expected-note{{overridden virtual function is here}} }; class B : A { virtual void f(); // expected-error{{virtual function 'f' has a different return type ('void') than the function it overrides (which has return type 'int')}} }; } namespace T2 { struct a { }; struct b { }; class A { virtual a* f(); // expected-note{{overridden virtual function is here}} }; class B : A { virtual b* f(); // expected-error{{return type of virtual function 'f' is not covariant with the return type of the function it overrides ('T2::b *' is not derived from 'T2::a *')}} }; } namespace T3 { struct a { }; struct b : private a { }; // expected-note{{declared private here}} class A { virtual a* f(); // FIXME: desired-note{{overridden virtual function is here}} }; class B : A { virtual b* f(); // expected-error{{invalid covariant return for virtual function: 'T3::a' is a private base class of 'T3::b'}} }; } namespace T4 { struct a { }; struct a1 : a { }; struct b : a, a1 { }; // expected-warning{{direct base 'T4::a' is inaccessible due to ambiguity:\n struct T4::b -> struct T4::a\n struct T4::b -> struct T4::a1 -> struct T4::a}} class A { virtual a* f(); // expected-note{{overridden virtual function is here}} }; class B : A { virtual b* f(); // expected-error{{return type of virtual function 'f' is not covariant with the return type of the function it overrides (ambiguous conversion from derived class 'T4::b' to base class 'T4::a':\n\ struct T4::b -> struct T4::a\n\ struct T4::b -> struct T4::a1 -> struct T4::a)}} }; } namespace T5 { struct a { }; class A { virtual a* const f(); virtual a* const g(); // expected-note{{overridden virtual function is here}} }; class B : A { virtual a* const f(); virtual a* g(); // expected-error{{return type of virtual function 'g' is not covariant with the return type of the function it overrides ('T5::a *' has different qualifiers than 'T5::a *const')}} }; } namespace T6 { struct a { }; class A { virtual const a* f(); virtual a* g(); // expected-note{{overridden virtual function is here}} }; class B : A { virtual a* f(); virtual const a* g(); // expected-error{{return type of virtual function 'g' is not covariant with the return type of the function it overrides (class type 'const T6::a *' is more qualified than class type 'T6::a *'}} }; } namespace T7 { struct a { }; struct b { }; class A { a* f(); }; class B : A { virtual b* f(); }; } namespace T8 { struct a { }; struct b; // expected-note {{forward declaration of 'T8::b'}} class A { virtual a *f(); }; class B : A { b* f(); // expected-error {{return type of virtual function 'f' is not covariant with the return type of the function it overrides ('T8::b' is incomplete)}} }; } namespace T9 { struct a { }; template<typename T> struct b : a { int a[sizeof(T) ? -1 : -1]; // expected-error {{array with a negative size}} }; class A { virtual a *f(); }; class B : A { virtual b<int> *f(); // expected-note {{in instantiation of template class 'T9::b<int>' requested here}} }; } // PR5656 class X0 { virtual void f0(); }; class X1 : public X0 { void f0() = 0; }; template <typename Base> struct Foo : Base { void f(int) = 0; // expected-error{{not virtual and cannot be declared pure}} }; struct Base1 { virtual void f(int); }; struct Base2 { }; void test() { (void)sizeof(Foo<Base1>); (void)sizeof(Foo<Base2>); // expected-note{{instantiation}} } template<typename Base> struct Foo2 : Base { template<typename T> int f(T); }; void test2() { Foo2<Base1> f1; Foo2<Base2> f2; f1.f(17); f2.f(17); }; struct Foo3 { virtual void f(int) = 0; // expected-note{{unimplemented pure virtual method}} }; template<typename T> struct Bar3 : Foo3 { void f(T); }; void test3() { Bar3<int> b3i; // okay Bar3<float> b3f; // expected-error{{is an abstract class}} } // 5920 namespace PR5920 { class Base {}; template <typename T> class Derived : public Base {}; class Foo { public: virtual Base* Method(); }; class Bar : public Foo { public: virtual Derived<int>* Method(); }; } // Look through template types and typedefs to see whether return types are // pointers or references. namespace PR6110 { class Base {}; class Derived : public Base {}; typedef Base* BaseP; typedef Derived* DerivedP; class X { virtual BaseP f(); }; class X1 : public X { virtual DerivedP f(); }; template <typename T> class Y { virtual T f(); }; template <typename T1, typename T> class Y1 : public Y<T> { virtual T1 f(); }; Y1<Derived*, Base*> y; } // Defer checking for covariance if either return type is dependent. namespace type_dependent_covariance { struct B {}; template <int N> struct TD : public B {}; template <> struct TD<1> {}; template <int N> struct TB {}; struct D : public TB<0> {}; template <int N> struct X { virtual B* f1(); // expected-note{{overridden virtual function is here}} virtual TB<N>* f2(); // expected-note{{overridden virtual function is here}} }; template <int N, int M> struct X1 : X<N> { virtual TD<M>* f1(); // expected-error{{return type of virtual function 'f1' is not covariant with the return type of the function it overrides ('TD<1> *'}} virtual D* f2(); // expected-error{{return type of virtual function 'f2' is not covariant with the return type of the function it overrides ('type_dependent_covariance::D *' is not derived from 'TB<1> *')}} }; X1<0, 0> good; X1<0, 1> bad_derived; // expected-note{{instantiation}} X1<1, 0> bad_base; // expected-note{{instantiation}} } namespace T10 { struct A { }; struct B : A { }; struct C { virtual A&& f(); }; struct D : C { virtual B&& f(); }; }; namespace T11 { struct A { }; struct B : A { }; struct C { virtual A& f(); // expected-note {{overridden virtual function is here}} }; struct D : C { virtual B&& f(); // expected-error {{virtual function 'f' has a different return type ('T11::B &&') than the function it overrides (which has return type 'T11::A &')}} }; }; namespace T12 { struct A { }; struct B : A { }; struct C { virtual A&& f(); // expected-note {{overridden virtual function is here}} }; struct D : C { virtual B& f(); // expected-error {{virtual function 'f' has a different return type ('T12::B &') than the function it overrides (which has return type 'T12::A &&')}} }; }; namespace PR8168 { class A { public: virtual void foo() {} // expected-note{{overridden virtual function is here}} }; class B : public A { public: static void foo() {} // expected-error{{'static' member function 'foo' overrides a virtual function}} }; }