// RUN: %clang_cc1 -fsyntax-only -verify %s // expected-no-diagnostics @protocol P1 @end @interface A <P1> @end @interface B : A @end @interface C : B @end template<typename T> struct ConvertsTo { operator T() const; }; // conversion of C* to B* is better than conversion of C* to A*. int &f0(A*); float &f0(B*); void test_f0(C *c) { float &fr1 = f0(c); } // conversion of B* to A* is better than conversion of C* to A* void f1(A*); struct ConvertsToBoth { private: operator C*() const; public: operator B*() const; }; void test_f1(ConvertsTo<B*> toB, ConvertsTo<C*> toC, ConvertsToBoth toBoth) { f1(toB); f1(toC); f1(toBoth); }; // A conversion to an a non-id object pointer type is better than a // conversion to 'id'. int &f2(A*); float &f2(id); void test_f2(B *b) { int &ir = f2(b); } // A conversion to an a non-Class object pointer type is better than a // conversion to 'Class'. int &f3(A*); float &f3(Class); void test_f3(B *b) { int &ir = f3(b); } // When both conversions convert to 'id' or 'Class', pick the most // specific type to convert from. void f4(id); void test_f4(ConvertsTo<B*> toB, ConvertsTo<C*> toC, ConvertsToBoth toBoth) { f4(toB); f4(toC); f4(toBoth); } void f5(id<P1>); void test_f5(ConvertsTo<B*> toB, ConvertsTo<C*> toC, ConvertsToBoth toBoth) { f5(toB); f5(toC); f5(toBoth); } // A conversion to an a non-id object pointer type is better than a // conversion to qualified 'id'. int &f6(A*); float &f6(id<P1>); void test_f6(B *b) { int &ir = f6(b); }