| // RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %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}} |
| } |
| |
| struct A { |
| bool operator==(Z&); // expected-note 2{{candidate function}} |
| }; |
| |
| A make_A(); |
| |
| bool operator==(A&, Z&); // expected-note 3{{candidate function}} |
| |
| void h(A a, const A ac, Z z) { |
| make_A() == z; // expected-warning{{equality comparison result unused}} |
| a == z; // expected-error{{use of overloaded operator '==' is ambiguous}} |
| ac == z; // expected-error{{invalid operands to binary expression ('const A' and 'Z')}} |
| } |
| |
| struct B { |
| bool operator==(const B&) const; |
| |
| void test(Z z) { |
| make_A() == z; // expected-warning{{equality comparison result unused}} |
| } |
| }; |
| |
| // we shouldn't see warnings about self-comparison, |
| // this is a member function, we dunno what it'll do |
| bool i(B b) |
| { |
| return b == b; |
| } |
| |
| enum Enum1 { }; |
| enum Enum2 { }; |
| |
| struct E1 { |
| E1(Enum1) { } |
| }; |
| |
| struct E2 { |
| E2(Enum2); |
| }; |
| |
| // C++ [over.match.oper]p3 - enum restriction. |
| float& operator==(E1, E2); // expected-note{{candidate function}} |
| |
| void enum_test(Enum1 enum1, Enum2 enum2, E1 e1, E2 e2, Enum1 next_enum1) { |
| float &f1 = (e1 == e2); |
| float &f2 = (enum1 == e2); |
| float &f3 = (e1 == enum2); |
| float &f4 = (enum1 == next_enum1); // expected-error{{non-const lvalue reference to type 'float' cannot bind to a temporary of type 'bool'}} |
| } |
| |
| // PR5244 - Argument-dependent lookup would include the two operators below, |
| // which would break later assumptions and lead to a crash. |
| class pr5244_foo |
| { |
| pr5244_foo(int); |
| pr5244_foo(char); |
| }; |
| |
| bool operator==(const pr5244_foo& s1, const pr5244_foo& s2); // expected-note{{candidate function}} |
| bool operator==(char c, const pr5244_foo& s); // expected-note{{candidate function}} |
| |
| enum pr5244_bar |
| { |
| pr5244_BAR |
| }; |
| |
| class pr5244_baz |
| { |
| public: |
| pr5244_bar quux; |
| }; |
| |
| void pr5244_barbaz() |
| { |
| pr5244_baz quuux; |
| (void)(pr5244_BAR == quuux.quux); |
| } |
| |
| |
| |
| 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*(); |
| long& operator*() const volatile; |
| }; |
| |
| void test_smartptr(SmartPtr ptr, const SmartPtr cptr, |
| const volatile SmartPtr cvptr) { |
| int &ir = *ptr; |
| long &lr = *cptr; |
| long &lr2 = *cvptr; |
| } |
| |
| |
| 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); // expected-warning {{expression result unused}} |
| } |
| |
| struct Callable { |
| int& operator()(int, double = 2.71828); // expected-note{{candidate function}} |
| float& operator()(int, double, long, ...); // expected-note{{candidate function}} |
| |
| double& operator()(float); // expected-note{{candidate function}} |
| }; |
| |
| struct Callable2 { |
| int& operator()(int i = 0); |
| double& operator()(...) const; |
| }; |
| |
| struct DerivesCallable : public Callable { |
| }; |
| |
| void test_callable(Callable c, Callable2 c2, const Callable2& c2c, |
| DerivesCallable dc) { |
| int &ir = c(1); |
| float &fr = c(1, 3.14159, 17, 42); |
| |
| c(); // expected-error{{no matching function for call to object of type 'Callable'}} |
| |
| double &dr = c(1.0f); |
| |
| int &ir2 = c2(); |
| int &ir3 = c2(1); |
| double &fr2 = c2c(); |
| |
| int &ir4 = dc(17); |
| double &fr3 = dc(3.14159f); |
| } |
| |
| typedef float FLOAT; |
| typedef int& INTREF; |
| typedef INTREF Func1(FLOAT, double); |
| typedef float& Func2(int, double); |
| |
| struct ConvertToFunc { |
| operator Func1*(); // expected-note 2{{conversion candidate of type 'INTREF (*)(FLOAT, double)'}} |
| operator Func2&(); // expected-note 2{{conversion candidate of type 'float &(&)(int, double)'}} |
| void operator()(); |
| }; |
| |
| struct ConvertToFuncDerived : ConvertToFunc { }; |
| |
| void test_funcptr_call(ConvertToFunc ctf, ConvertToFuncDerived ctfd) { |
| int &i1 = ctf(1.0f, 2.0); |
| float &f1 = ctf((short int)1, 1.0f); |
| ctf((long int)17, 2.0); // expected-error{{call to object of type 'ConvertToFunc' is ambiguous}} |
| ctf(); |
| |
| int &i2 = ctfd(1.0f, 2.0); |
| float &f2 = ctfd((short int)1, 1.0f); |
| ctfd((long int)17, 2.0); // expected-error{{call to object of type 'ConvertToFuncDerived' is ambiguous}} |
| ctfd(); |
| } |
| |
| 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->'}} |
| } |
| |
| struct CopyConBase { |
| }; |
| |
| struct CopyCon : public CopyConBase { |
| CopyCon(const CopyConBase &Base); |
| |
| CopyCon(const CopyConBase *Base) { |
| *this = *Base; |
| } |
| }; |
| |
| namespace N { |
| struct X { }; |
| } |
| |
| namespace M { |
| N::X operator+(N::X, N::X); |
| } |
| |
| namespace M { |
| void test_X(N::X x) { |
| (void)(x + x); |
| } |
| } |
| |
| struct AA { bool operator!=(AA&); }; |
| struct BB : AA {}; |
| bool x(BB y, BB z) { return y != z; } |
| |
| |
| struct AX { |
| AX& operator ->(); // expected-note {{declared here}} |
| int b; |
| }; |
| |
| void m() { |
| AX a; |
| a->b = 0; // expected-error {{circular pointer delegation detected}} |
| } |
| |
| struct CircA { |
| struct CircB& operator->(); // expected-note {{declared here}} |
| int val; |
| }; |
| struct CircB { |
| struct CircC& operator->(); // expected-note {{declared here}} |
| }; |
| struct CircC { |
| struct CircA& operator->(); // expected-note {{declared here}} |
| }; |
| |
| void circ() { |
| CircA a; |
| a->val = 0; // expected-error {{circular pointer delegation detected}} |
| } |
| |
| // PR5360: Arrays should lead to built-in candidates for subscript. |
| typedef enum { |
| LastReg = 23, |
| } Register; |
| class RegAlloc { |
| int getPriority(Register r) { |
| return usepri[r]; |
| } |
| int usepri[LastReg + 1]; |
| }; |
| |
| // PR5546: Don't generate incorrect and ambiguous overloads for multi-level |
| // arrays. |
| namespace pr5546 |
| { |
| enum { X }; |
| extern const char *const sMoveCommands[][2][2]; |
| const char* a() { return sMoveCommands[X][0][0]; } |
| const char* b() { return (*(sMoveCommands+X))[0][0]; } |
| } |
| |
| // PR5512 and its discussion |
| namespace pr5512 { |
| struct Y { |
| operator short(); |
| operator float(); |
| }; |
| void g_test(Y y) { |
| short s = 0; |
| // DR507, this should be ambiguous, but we special-case assignment |
| s = y; |
| // Note: DR507, this is ambiguous as specified |
| //s += y; |
| } |
| |
| struct S {}; |
| void operator +=(int&, S); |
| void f(S s) { |
| int i = 0; |
| i += s; |
| } |
| |
| struct A {operator int();}; |
| int a; |
| void b(A x) { |
| a += x; |
| } |
| } |
| |
| // PR5900 |
| namespace pr5900 { |
| struct NotAnArray {}; |
| void test0() { |
| NotAnArray x; |
| x[0] = 0; // expected-error {{does not provide a subscript operator}} |
| } |
| |
| struct NonConstArray { |
| int operator[](unsigned); // expected-note {{candidate}} |
| }; |
| int test1() { |
| const NonConstArray x = NonConstArray(); |
| return x[0]; // expected-error {{no viable overloaded operator[] for type}} |
| } |
| |
| // Not really part of this PR, but implemented at the same time. |
| struct NotAFunction {}; |
| void test2() { |
| NotAFunction x; |
| x(); // expected-error {{does not provide a call operator}} |
| } |
| } |
| |
| // Operator lookup through using declarations. |
| namespace N { |
| struct X2 { }; |
| } |
| |
| namespace N2 { |
| namespace M { |
| namespace Inner { |
| template<typename T> |
| N::X2 &operator<<(N::X2&, const T&); |
| } |
| using Inner::operator<<; |
| } |
| } |
| |
| void test_lookup_through_using() { |
| using namespace N2::M; |
| N::X2 x; |
| x << 17; |
| } |
| |
| namespace rdar9136502 { |
| struct X { |
| int i(); // expected-note{{possible target for call}} |
| int i(int); // expected-note{{possible target for call}} |
| }; |
| |
| struct Y { |
| Y &operator<<(int); |
| }; |
| |
| void f(X x, Y y) { |
| y << x |
| .i; // expected-error{{reference to non-static member function must be called; did you mean to call it with no arguments?}} |
| } |
| } |
| |
| namespace rdar9222009 { |
| class StringRef { |
| inline bool operator==(StringRef LHS, StringRef RHS) { // expected-error{{overloaded 'operator==' must be a binary operator (has 3 parameters)}} |
| return !(LHS == RHS); // expected-error{{invalid operands to binary expression ('rdar9222009::StringRef' and 'rdar9222009::StringRef')}} |
| } |
| }; |
| |
| } |
| |
| namespace PR11784 { |
| struct A { A& operator=(void (*x)()); }; |
| void f(); |
| void f(int); |
| void g() { A x; x = f; } |
| } |
| |
| namespace test10 { |
| struct A { |
| void operator[](float (*fn)(int)); // expected-note 2 {{not viable: no overload of 'bar' matching 'float (*)(int)'}} |
| }; |
| |
| float foo(int); |
| float foo(float); |
| |
| template <class T> T bar(T); |
| template <class T, class U> T bar(U); |
| |
| void test(A &a) { |
| a[&foo]; |
| a[foo]; |
| |
| a[&bar<int>]; // expected-error {{no viable overloaded operator[]}} |
| a[bar<int>]; // expected-error {{no viable overloaded operator[]}} |
| |
| // If these fail, it's because we're not letting the overload |
| // resolution for operator| resolve the overload of 'bar'. |
| a[&bar<float>]; |
| a[bar<float>]; |
| } |
| } |
| |
| struct InvalidOperatorEquals { |
| InvalidOperatorEquals operator=() = delete; // expected-error {{overloaded 'operator=' must be a binary operator}} |
| }; |
| |
| namespace PR7681 { |
| template <typename PT1, typename PT2> class PointerUnion; |
| void foo(PointerUnion<int*, float*> &Result) { |
| Result = 1; // expected-error {{no viable overloaded '='}} // expected-note {{type 'PointerUnion<int *, float *>' is incomplete}} |
| } |
| } |
| |
| namespace PR14995 { |
| struct B {}; |
| template<typename ...T> void operator++(B, T...) {} |
| |
| void f() { |
| B b; |
| b++; // ok |
| ++b; // ok |
| } |
| |
| template<typename... T> |
| struct C { |
| void operator-- (T...) {} |
| }; |
| |
| void g() { |
| C<int> postfix; |
| C<> prefix; |
| postfix--; // ok |
| --prefix; // ok |
| } |
| |
| struct D {}; |
| template<typename T> void operator++(D, T) {} |
| |
| void h() { |
| D d; |
| d++; // ok |
| ++d; // expected-error{{cannot increment value of type 'PR14995::D'}} |
| } |
| |
| template<typename...T> struct E { |
| void operator++(T...) {} // expected-error{{parameter of overloaded post-increment operator must have type 'int' (not 'char')}} |
| }; |
| |
| E<char> e; // expected-note {{in instantiation of template class 'PR14995::E<char>' requested here}} |
| |
| struct F { |
| template<typename... T> |
| int operator++ (T...) {} |
| }; |
| |
| int k1 = F().operator++(0, 0); |
| int k2 = F().operator++('0'); |
| // expected-error@-5 {{overloaded 'operator++' must be a unary or binary operator}} |
| // expected-note@-3 {{in instantiation of function template specialization 'PR14995::F::operator++<int, int>' requested here}} |
| // expected-error@-4 {{no matching member function for call to 'operator++'}} |
| // expected-note@-8 {{candidate template ignored: substitution failure}} |
| // expected-error@-9 {{parameter of overloaded post-increment operator must have type 'int' (not 'char')}} |
| // expected-note@-6 {{in instantiation of function template specialization 'PR14995::F::operator++<char>' requested here}} |
| // expected-error@-7 {{no matching member function for call to 'operator++'}} |
| // expected-note@-12 {{candidate template ignored: substitution failure}} |
| } // namespace PR14995 |
| |
| namespace ConversionVersusTemplateOrdering { |
| struct A { |
| operator short() = delete; |
| template <typename T> operator T(); |
| } a; |
| struct B { |
| template <typename T> operator T(); |
| operator short() = delete; |
| } b; |
| int x = a; |
| int y = b; |
| } |
| |
| namespace NoADLForMemberOnlyOperators { |
| template<typename T> struct A { typename T::error e; }; // expected-error {{type 'char' cannot be used prior to '::'}} |
| template<typename T> struct B { int n; }; |
| |
| void f(B<A<void> > b1, B<A<int> > b2, B<A<char> > b3) { |
| b1 = b1; // ok, does not instantiate A<void>. |
| (void)b1->n; // expected-error {{is not a pointer}} |
| b2[3]; // expected-error {{does not provide a subscript}} |
| b3 / 0; // expected-note {{in instantiation of}} expected-error {{invalid operands to}} |
| } |
| } |
| |
| |
| namespace PR27027 { |
| template <class T> void operator+(T, T) = delete; // expected-note 4 {{candidate}} |
| template <class T> void operator+(T) = delete; // expected-note 4 {{candidate}} |
| |
| struct A {} a_global; |
| void f() { |
| A a; |
| +a; // expected-error {{overload resolution selected deleted operator '+'}} |
| a + a; // expected-error {{overload resolution selected deleted operator '+'}} |
| bool operator+(A); |
| extern bool operator+(A, A); |
| +a; // OK |
| a + a; |
| } |
| bool test_global_1 = +a_global; // expected-error {{overload resolution selected deleted operator '+'}} |
| bool test_global_2 = a_global + a_global; // expected-error {{overload resolution selected deleted operator '+'}} |
| } |
| |
| namespace LateADLInNonDependentExpressions { |
| struct A {}; |
| struct B : A {}; |
| int &operator+(A, A); |
| int &operator!(A); |
| int &operator+=(A, A); |
| int &operator<<(A, A); |
| int &operator++(A); |
| int &operator++(A, int); |
| int &operator->*(A, A); |
| |
| template<typename T> void f() { |
| // An instantiation-dependent value of type B. |
| // These are all non-dependent operator calls of type int&. |
| #define idB ((void()), B()) |
| int &a = idB + idB, |
| &b = !idB, |
| &c = idB += idB, |
| &d = idB << idB, |
| &e = ++idB, |
| &f = idB++, |
| &g = idB ->* idB; |
| } |
| |
| // These should not be found by ADL in the template instantiation. |
| float &operator+(B, B); |
| float &operator!(B); |
| float &operator+=(B, B); |
| float &operator<<(B, B); |
| float &operator++(B); |
| float &operator++(B, int); |
| float &operator->*(B, B); |
| template void f<int>(); |
| } |