| Daniel Dunbar | 8fbe78f | 2009-12-15 20:14:24 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -fsyntax-only -verify %s | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 2 |  | 
|  | 3 | struct A {}; | 
|  | 4 | enum B { Dummy }; | 
|  | 5 | namespace C {} | 
| Sebastian Redl | 72b597d | 2009-01-25 19:43:20 +0000 | [diff] [blame] | 6 | struct D : A {}; | 
| Sebastian Redl | a7bc847 | 2009-01-26 19:54:48 +0000 | [diff] [blame] | 7 | struct E : A {}; | 
|  | 8 | struct F : D, E {}; | 
|  | 9 | struct G : virtual D {}; | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 10 |  | 
|  | 11 | int A::*pdi1; | 
|  | 12 | int (::A::*pdi2); | 
|  | 13 | int (A::*pfi)(int); | 
|  | 14 |  | 
| John McCall | 1f476a1 | 2010-02-26 08:45:28 +0000 | [diff] [blame] | 15 | int B::*pbi; // expected-error {{expected a class or namespace}} | 
| Sebastian Redl | 9ed6efd | 2009-01-24 21:16:55 +0000 | [diff] [blame] | 16 | int C::*pci; // expected-error {{'pci' does not point into a class}} | 
|  | 17 | void A::*pdv; // expected-error {{'pdv' declared as a member pointer to void}} | 
| Anders Carlsson | 124f34c | 2009-06-30 00:06:57 +0000 | [diff] [blame] | 18 | int& A::*pdr; // expected-error {{'pdr' declared as a member pointer to a reference}} | 
| Sebastian Redl | c6d52f5 | 2009-01-24 23:29:36 +0000 | [diff] [blame] | 19 |  | 
|  | 20 | void f() { | 
|  | 21 | // This requires tentative parsing. | 
|  | 22 | int (A::*pf)(int, int); | 
| Sebastian Redl | 72b597d | 2009-01-25 19:43:20 +0000 | [diff] [blame] | 23 |  | 
|  | 24 | // Implicit conversion to bool. | 
|  | 25 | bool b = pdi1; | 
|  | 26 | b = pfi; | 
|  | 27 |  | 
|  | 28 | // Conversion from null pointer constant. | 
|  | 29 | pf = 0; | 
|  | 30 | pf = __null; | 
|  | 31 |  | 
|  | 32 | // Conversion to member of derived. | 
|  | 33 | int D::*pdid = pdi1; | 
|  | 34 | pdid = pdi2; | 
| Sebastian Redl | a7bc847 | 2009-01-26 19:54:48 +0000 | [diff] [blame] | 35 |  | 
|  | 36 | // Fail conversion due to ambiguity and virtuality. | 
| John McCall | 85f9055 | 2010-03-10 11:27:22 +0000 | [diff] [blame] | 37 | int F::*pdif = pdi1; // expected-error {{ambiguous conversion from pointer to member of base class 'A' to pointer to member of derived class 'F':}} | 
|  | 38 | int G::*pdig = pdi1; // expected-error {{conversion from pointer to member of class 'A' to pointer to member of class 'G' via virtual base 'D' is not allowed}} | 
| Sebastian Redl | ed8f200 | 2009-01-28 18:33:18 +0000 | [diff] [blame] | 39 |  | 
|  | 40 | // Conversion to member of base. | 
| Douglas Gregor | c68e140 | 2010-04-09 00:35:39 +0000 | [diff] [blame] | 41 | pdi1 = pdid; // expected-error {{assigning to 'int A::*' from incompatible type 'int D::*'}} | 
| Douglas Gregor | b00b10e | 2009-08-24 17:42:35 +0000 | [diff] [blame] | 42 |  | 
|  | 43 | // Comparisons | 
|  | 44 | int (A::*pf2)(int, int); | 
|  | 45 | int (D::*pf3)(int, int) = 0; | 
|  | 46 | bool b1 = (pf == pf2); (void)b1; | 
|  | 47 | bool b2 = (pf != pf2); (void)b2; | 
|  | 48 | bool b3 = (pf == pf3); (void)b3; | 
|  | 49 | bool b4 = (pf != 0); (void)b4; | 
| Sebastian Redl | c6d52f5 | 2009-01-24 23:29:36 +0000 | [diff] [blame] | 50 | } | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 51 |  | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 52 | struct TheBase | 
|  | 53 | { | 
|  | 54 | void d(); | 
|  | 55 | }; | 
|  | 56 |  | 
|  | 57 | struct HasMembers : TheBase | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 58 | { | 
|  | 59 | int i; | 
|  | 60 | void f(); | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 61 |  | 
|  | 62 | void g(); | 
|  | 63 | void g(int); | 
|  | 64 | static void g(double); | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 65 | }; | 
|  | 66 |  | 
|  | 67 | namespace Fake | 
|  | 68 | { | 
|  | 69 | int i; | 
|  | 70 | void f(); | 
|  | 71 | } | 
|  | 72 |  | 
|  | 73 | void g() { | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 74 | HasMembers hm; | 
|  | 75 |  | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 76 | int HasMembers::*pmi = &HasMembers::i; | 
|  | 77 | int *pni = &Fake::i; | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 78 | int *pmii = &hm.i; | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 79 |  | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 80 | void (HasMembers::*pmf)() = &HasMembers::f; | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 81 | void (*pnf)() = &Fake::f; | 
| Sebastian Redl | 9a8dd0d | 2010-01-11 15:56:56 +0000 | [diff] [blame] | 82 | &hm.f; // expected-error {{must explicitly qualify}} expected-warning{{result unused}} | 
| Sebastian Redl | 18f8ff6 | 2009-02-04 21:23:32 +0000 | [diff] [blame] | 83 |  | 
|  | 84 | void (HasMembers::*pmgv)() = &HasMembers::g; | 
|  | 85 | void (HasMembers::*pmgi)(int) = &HasMembers::g; | 
|  | 86 | void (*pmgd)(double) = &HasMembers::g; | 
|  | 87 |  | 
|  | 88 | void (HasMembers::*pmd)() = &HasMembers::d; | 
| Sebastian Redl | 3d3f75a | 2009-02-03 20:19:35 +0000 | [diff] [blame] | 89 | } | 
| Sebastian Redl | 112a9766 | 2009-02-07 00:15:38 +0000 | [diff] [blame] | 90 |  | 
| Douglas Gregor | 2ec748c | 2009-05-14 00:28:11 +0000 | [diff] [blame] | 91 | struct Incomplete; | 
| Sebastian Redl | 9c3b4b1 | 2009-02-07 00:41:42 +0000 | [diff] [blame] | 92 |  | 
| Sebastian Redl | 112a9766 | 2009-02-07 00:15:38 +0000 | [diff] [blame] | 93 | void h() { | 
|  | 94 | HasMembers hm, *phm = &hm; | 
|  | 95 |  | 
|  | 96 | int HasMembers::*pi = &HasMembers::i; | 
|  | 97 | hm.*pi = 0; | 
|  | 98 | int i = phm->*pi; | 
|  | 99 | (void)&(hm.*pi); | 
|  | 100 | (void)&(phm->*pi); | 
| Fariborz Jahanian | 03b4f66 | 2009-10-08 18:00:39 +0000 | [diff] [blame] | 101 | (void)&((&hm)->*pi); | 
| Sebastian Redl | 112a9766 | 2009-02-07 00:15:38 +0000 | [diff] [blame] | 102 |  | 
|  | 103 | void (HasMembers::*pf)() = &HasMembers::f; | 
|  | 104 | (hm.*pf)(); | 
|  | 105 | (phm->*pf)(); | 
| Sebastian Redl | 9c3b4b1 | 2009-02-07 00:41:42 +0000 | [diff] [blame] | 106 |  | 
| John McCall | 85f9055 | 2010-03-10 11:27:22 +0000 | [diff] [blame] | 107 | (void)(hm->*pi); // expected-error {{left hand operand to ->* must be a pointer to class compatible with the right hand operand, but is 'HasMembers'}} | 
|  | 108 | (void)(phm.*pi); // expected-error {{left hand operand to .* must be a class compatible with the right hand operand, but is 'HasMembers *'}} | 
| Sebastian Redl | 9c3b4b1 | 2009-02-07 00:41:42 +0000 | [diff] [blame] | 109 | (void)(i.*pi); // expected-error {{left hand operand to .* must be a class compatible with the right hand operand, but is 'int'}} | 
|  | 110 | int *ptr; | 
|  | 111 | (void)(ptr->*pi); // expected-error {{left hand operand to ->* must be a pointer to class compatible with the right hand operand, but is 'int *'}} | 
|  | 112 |  | 
|  | 113 | int A::*pai = 0; | 
|  | 114 | D d, *pd = &d; | 
|  | 115 | (void)(d.*pai); | 
|  | 116 | (void)(pd->*pai); | 
|  | 117 | F f, *ptrf = &f; | 
| John McCall | 85f9055 | 2010-03-10 11:27:22 +0000 | [diff] [blame] | 118 | (void)(f.*pai); // expected-error {{left hand operand to .* must be a class compatible with the right hand operand, but is 'F'}} | 
|  | 119 | (void)(ptrf->*pai); // expected-error {{left hand operand to ->* must be a pointer to class compatible with the right hand operand, but is 'F *'}} | 
| Sebastian Redl | 9c3b4b1 | 2009-02-07 00:41:42 +0000 | [diff] [blame] | 120 |  | 
| Douglas Gregor | ac1fb65 | 2009-03-24 19:52:54 +0000 | [diff] [blame] | 121 | (void)(hm.*i); // expected-error {{pointer-to-member}} | 
|  | 122 | (void)(phm->*i); // expected-error {{pointer-to-member}} | 
| Sebastian Redl | 9c3b4b1 | 2009-02-07 00:41:42 +0000 | [diff] [blame] | 123 |  | 
|  | 124 | Incomplete *inc; | 
|  | 125 | int Incomplete::*pii = 0; | 
| Douglas Gregor | 2ec748c | 2009-05-14 00:28:11 +0000 | [diff] [blame] | 126 | (void)(inc->*pii); // okay | 
| Sebastian Redl | 112a9766 | 2009-02-07 00:15:38 +0000 | [diff] [blame] | 127 | } | 
|  | 128 |  | 
|  | 129 | struct OverloadsPtrMem | 
|  | 130 | { | 
|  | 131 | int operator ->*(const char *); | 
|  | 132 | }; | 
|  | 133 |  | 
|  | 134 | void i() { | 
|  | 135 | OverloadsPtrMem m; | 
|  | 136 | int foo = m->*"Awesome!"; | 
|  | 137 | } | 
| Sebastian Redl | 9a8dd0d | 2010-01-11 15:56:56 +0000 | [diff] [blame] | 138 |  | 
|  | 139 | namespace pr5985 { | 
|  | 140 | struct c { | 
|  | 141 | void h(); | 
|  | 142 | void f() { | 
|  | 143 | void (c::*p)(); | 
|  | 144 | p = &h; // expected-error {{must explicitly qualify}} | 
|  | 145 | p = &this->h; // expected-error {{must explicitly qualify}} | 
|  | 146 | p = &(*this).h; // expected-error {{must explicitly qualify}} | 
|  | 147 | } | 
|  | 148 | }; | 
|  | 149 | } |