// 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);
}