2008-11-07 07:29:22 +08:00
|
|
|
// RUN: clang -fsyntax-only -verify %s
|
|
|
|
class X { };
|
|
|
|
|
|
|
|
X operator+(X, X);
|
|
|
|
|
|
|
|
void f(X x) {
|
|
|
|
x = x + x;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Y;
|
|
|
|
struct Z;
|
|
|
|
|
|
|
|
struct Y {
|
|
|
|
Y(const Z&);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Z {
|
|
|
|
Z(const Y&);
|
|
|
|
};
|
|
|
|
|
|
|
|
Y operator+(Y, Y);
|
|
|
|
bool operator-(Y, Y); // expected-note{{candidate function}}
|
|
|
|
bool operator-(Z, Z); // expected-note{{candidate function}}
|
|
|
|
|
|
|
|
void g(Y y, Z z) {
|
|
|
|
y = y + z;
|
|
|
|
bool b = y - z; // expected-error{{use of overloaded operator '-' is ambiguous; candidates are:}}
|
|
|
|
}
|
|
|
|
|
2008-11-19 07:14:02 +08:00
|
|
|
struct A {
|
|
|
|
bool operator==(Z&); // expected-note{{candidate function}}
|
|
|
|
};
|
|
|
|
|
|
|
|
A make_A();
|
2008-11-07 07:29:22 +08:00
|
|
|
|
2008-11-19 07:14:02 +08:00
|
|
|
bool operator==(A&, Z&); // expected-note{{candidate function}}
|
|
|
|
|
|
|
|
void h(A a, const A ac, Z z) {
|
|
|
|
make_A() == z;
|
|
|
|
a == z; // expected-error{{use of overloaded operator '==' is ambiguous; candidates are:}}
|
|
|
|
ac == z; // expected-error{{invalid operands to binary expression ('struct A const' and 'struct Z')}}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct B {
|
|
|
|
bool operator==(const B&) const;
|
|
|
|
|
|
|
|
void test(Z z) {
|
|
|
|
make_A() == z;
|
|
|
|
}
|
|
|
|
};
|
2008-11-19 11:25:36 +08:00
|
|
|
|
|
|
|
enum Enum1 { };
|
|
|
|
enum Enum2 { };
|
|
|
|
|
|
|
|
struct E1 {
|
|
|
|
E1(Enum1) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct E2 {
|
|
|
|
E2(Enum2);
|
|
|
|
};
|
|
|
|
|
|
|
|
// C++ [over.match.oper]p3 - enum restriction.
|
|
|
|
float& operator==(E1, E2);
|
|
|
|
|
|
|
|
void enum_test(Enum1 enum1, Enum2 enum2, E1 e1, E2 e2) {
|
|
|
|
float &f1 = (e1 == e2);
|
|
|
|
float &f2 = (enum1 == e2);
|
|
|
|
float &f3 = (e1 == enum2);
|
|
|
|
float &f4 = (enum1 == enum2); // expected-error{{non-const reference to type 'float' cannot be initialized with a temporary of type '_Bool'}}
|
|
|
|
}
|
2008-11-19 23:42:04 +08:00
|
|
|
|
|
|
|
|
|
|
|
struct PostInc {
|
|
|
|
PostInc operator++(int);
|
|
|
|
PostInc& operator++();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PostDec {
|
|
|
|
PostDec operator--(int);
|
|
|
|
PostDec& operator--();
|
|
|
|
};
|
|
|
|
|
|
|
|
void incdec_test(PostInc pi, PostDec pd) {
|
|
|
|
const PostInc& pi1 = pi++;
|
|
|
|
const PostDec& pd1 = pd--;
|
|
|
|
PostInc &pi2 = ++pi;
|
|
|
|
PostDec &pd2 = --pd;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SmartPtr {
|
|
|
|
int& operator*();
|
2008-11-21 23:36:28 +08:00
|
|
|
long& operator*() const volatile;
|
2008-11-19 23:42:04 +08:00
|
|
|
};
|
|
|
|
|
2008-11-21 23:36:28 +08:00
|
|
|
void test_smartptr(SmartPtr ptr, const SmartPtr cptr,
|
|
|
|
const volatile SmartPtr cvptr) {
|
2008-11-19 23:42:04 +08:00
|
|
|
int &ir = *ptr;
|
2008-11-21 23:36:28 +08:00
|
|
|
long &lr = *cptr;
|
|
|
|
long &lr2 = *cvptr;
|
2008-11-19 23:42:04 +08:00
|
|
|
}
|
2008-11-20 01:17:41 +08:00
|
|
|
|
|
|
|
|
|
|
|
struct ArrayLike {
|
|
|
|
int& operator[](int);
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_arraylike(ArrayLike a) {
|
|
|
|
int& ir = a[17];
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SmartRef {
|
|
|
|
int* operator&();
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_smartref(SmartRef r) {
|
|
|
|
int* ip = &r;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool& operator,(X, Y);
|
|
|
|
|
|
|
|
void test_comma(X x, Y y) {
|
|
|
|
bool& b1 = (x, y);
|
|
|
|
X& xr = (x, x);
|
|
|
|
}
|
2008-11-20 05:05:33 +08:00
|
|
|
|
|
|
|
|
|
|
|
struct Callable {
|
|
|
|
int& operator()(int, double = 2.71828); // expected-note{{candidate function}}
|
|
|
|
float& operator()(int, double, long, ...); // expected-note{{candidate function}}
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_callable(Callable c) {
|
|
|
|
int &ir = c(1);
|
|
|
|
float &fr = c(1, 3.14159, 17, 42);
|
|
|
|
|
|
|
|
c(); // expected-error{{no matching function for call to object of type 'struct Callable'; candidates are:}}
|
|
|
|
}
|
2008-11-20 06:57:39 +08:00
|
|
|
|
2008-11-21 10:54:28 +08:00
|
|
|
typedef float FLOAT;
|
|
|
|
typedef int& INTREF;
|
|
|
|
typedef INTREF Func1(FLOAT, double);
|
2008-11-20 06:57:39 +08:00
|
|
|
typedef float& Func2(int, double);
|
|
|
|
|
|
|
|
struct ConvertToFunc {
|
2008-11-21 10:54:28 +08:00
|
|
|
operator Func1*(); // expected-note{{conversion candidate of type 'INTREF (*)(FLOAT, double)'}}
|
2008-11-20 06:57:39 +08:00
|
|
|
operator Func2&(); // expected-note{{conversion candidate of type 'float &(&)(int, double)'}}
|
2008-11-20 06:59:19 +08:00
|
|
|
void operator()();
|
2008-11-20 06:57:39 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void test_funcptr_call(ConvertToFunc ctf) {
|
|
|
|
int &i1 = ctf(1.0f, 2.0);
|
|
|
|
float &f2 = ctf((short int)1, 1.0f);
|
|
|
|
ctf((long int)17, 2.0); // expected-error{{error: call to object of type 'struct ConvertToFunc' is ambiguous; candidates are:}}
|
2008-11-20 06:59:19 +08:00
|
|
|
ctf();
|
2008-11-20 06:57:39 +08:00
|
|
|
}
|
2008-11-21 00:27:02 +08:00
|
|
|
|
|
|
|
struct HasMember {
|
|
|
|
int m;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Arrow1 {
|
|
|
|
HasMember* operator->();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Arrow2 {
|
|
|
|
Arrow1 operator->(); // expected-note{{candidate function}}
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_arrow(Arrow1 a1, Arrow2 a2, const Arrow2 a3) {
|
|
|
|
int &i1 = a1->m;
|
|
|
|
int &i2 = a2->m;
|
|
|
|
a3->m; // expected-error{{no viable overloaded 'operator->'; candidates are}}
|
|
|
|
}
|