reference, declarationdefinition
definition → references, declarations, derived classes, virtual overrides
reference to multiple definitions → definitions
unreferenced
    1
    2
    3
    4
    5
    6
    7
    8
    9
   10
   11
   12
   13
   14
   15
   16
   17
   18
   19
   20
   21
   22
   23
   24
   25
   26
   27
   28
   29
   30
   31
   32
   33
   34
   35
   36
   37
   38
   39
   40
   41
   42
   43
   44
   45
   46
   47
   48
   49
   50
   51
   52
   53
   54
   55
   56
   57
   58
   59
   60
   61
   62
   63
   64
   65
   66
   67
   68
   69
   70
   71
   72
   73
   74
   75
// RUN: %clang_cc1 -std=c++11 -verify %s -Wno-tautological-compare

struct A { operator decltype(nullptr)(); };
struct B { operator int A::*(); };
void f(A a, B b, int A::*pi) {
  (void)(a == a);
  (void)(a != a);
  (void)(a < a); // expected-error {{invalid operands}}
  (void)(a > a); // expected-error {{invalid operands}}
  (void)(a <= a); // expected-error {{invalid operands}}
  (void)(a >= a); // expected-error {{invalid operands}}

  (void)(a == b);
  (void)(a != b);
  (void)(a < b); // expected-error {{invalid operands}}
  (void)(a > b); // expected-error {{invalid operands}}
  (void)(a <= b); // expected-error {{invalid operands}}
  (void)(a >= b); // expected-error {{invalid operands}}

  (void)(b == a);
  (void)(b != a);
  (void)(b < a); // expected-error {{invalid operands}}
  (void)(b > a); // expected-error {{invalid operands}}
  (void)(b <= a); // expected-error {{invalid operands}}
  (void)(b >= a); // expected-error {{invalid operands}}

  (void)(a == pi);
  (void)(a != pi);
  (void)(a < pi); // expected-error {{invalid operands}}
  (void)(a > pi); // expected-error {{invalid operands}}
  (void)(a <= pi); // expected-error {{invalid operands}}
  (void)(a >= pi); // expected-error {{invalid operands}}

  (void)(pi == a);
  (void)(pi != a);
  (void)(pi < a); // expected-error {{invalid operands}}
  (void)(pi > a); // expected-error {{invalid operands}}
  (void)(pi <= a); // expected-error {{invalid operands}}
  (void)(pi >= a); // expected-error {{invalid operands}}

  (void)(b == pi);
  (void)(b != pi);
  (void)(b < pi); // expected-error {{invalid operands}}
  (void)(b > pi); // expected-error {{invalid operands}}
  (void)(b <= pi); // expected-error {{invalid operands}}
  (void)(b >= pi); // expected-error {{invalid operands}}

  (void)(pi == b);
  (void)(pi != b);
  (void)(pi < b); // expected-error {{invalid operands}}
  (void)(pi > b); // expected-error {{invalid operands}}
  (void)(pi <= b); // expected-error {{invalid operands}}
  (void)(pi >= b); // expected-error {{invalid operands}}

  (void)(b == b);
  (void)(b != b);
  (void)(b < b); // expected-error {{invalid operands}}
  (void)(b > b); // expected-error {{invalid operands}}
  (void)(b <= b); // expected-error {{invalid operands}}
  (void)(b >= b); // expected-error {{invalid operands}}

  (void)(pi == pi);
  (void)(pi != pi);
  (void)(pi < pi); // expected-error {{invalid operands}}
  (void)(pi > pi); // expected-error {{invalid operands}}
  (void)(pi <= pi); // expected-error {{invalid operands}}
  (void)(pi >= pi); // expected-error {{invalid operands}}
}

// FIXME: This is wrong: type T = 'const volatile int * const A::* const B::*'
// would work here, and there exists a builtin candidate for that type.
struct C { operator const int *A::*B::*(); };
void g(C c, volatile int *A::*B::*p) {
  (void)(c == p); // expected-error {{invalid operands}}
}