1 // RUN: %clang_cc1 -std=c++1y -fms-compatibility -fno-spell-checking -fsyntax-only -verify %s 2 3 4 template <class T> 5 class A { 6 public: 7 void f(T a) { }// expected-note 2{{must qualify identifier to find this declaration in dependent base class}} 8 void g();// expected-note 2{{must qualify identifier to find this declaration in dependent base class}} 9 }; 10 11 template <class T> 12 class B : public A<T> { 13 public: 14 void z(T a) 15 { 16 f(a); // expected-warning 2{{use of identifier 'f' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 17 g(); // expected-warning 2{{use of identifier 'g' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 18 } 19 }; 20 21 template class B<int>; // expected-note {{requested here}} 22 template class B<char>; // expected-note {{requested here}} 23 24 void test() 25 { 26 B<int> b; 27 b.z(3); 28 } 29 30 struct A2 { 31 template<class T> void f(T) { 32 XX; //expected-error {{use of undeclared identifier 'XX'}} 33 A2::XX; //expected-error {{no member named 'XX' in 'A2'}} 34 } 35 }; 36 template void A2::f(int); 37 38 template<class T0> 39 struct A3 { 40 template<class T1> void f(T1) { 41 XX; //expected-error {{use of undeclared identifier 'XX'}} 42 } 43 }; 44 template void A3<int>::f(int); 45 46 template<class T0> 47 struct A4 { 48 void f(char) { 49 XX; //expected-error {{use of undeclared identifier 'XX'}} 50 } 51 }; 52 template class A4<int>; 53 54 55 namespace lookup_dependent_bases_id_expr { 56 57 template<class T> class A { 58 public: 59 int var; 60 }; 61 62 63 template<class T> 64 class B : public A<T> { 65 public: 66 void f() { 67 var = 3; // expected-warning {{use of undeclared identifier 'var'; unqualified lookup into dependent bases of class template 'B' is a Microsoft extension}} 68 } 69 }; 70 71 template class B<int>; 72 73 } 74 75 76 77 namespace lookup_dependent_base_class_static_function { 78 79 template <class T> 80 class A { 81 public: 82 static void static_func();// expected-note {{must qualify identifier to find this declaration in dependent base class}} 83 void func();// expected-note {{must qualify identifier to find this declaration in dependent base class}} 84 }; 85 86 87 template <class T> 88 class B : public A<T> { 89 public: 90 static void z2(){ 91 static_func(); // expected-warning {{use of identifier 'static_func' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 92 func(); // expected-warning {{use of identifier 'func' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} expected-error {{call to non-static member function without an object argument}} 93 } 94 }; 95 template class B<int>; // expected-note {{requested here}} 96 97 } 98 99 100 101 namespace lookup_dependent_base_class_default_argument { 102 103 template<class T> 104 class A { 105 public: 106 static int f1(); // expected-note {{must qualify identifier to find this declaration in dependent base class}} 107 int f2(); // expected-note {{must qualify identifier to find this declaration in dependent base class}} 108 }; 109 110 template<class T> 111 class B : public A<T> { 112 public: 113 void g1(int p = f1());// expected-warning {{use of identifier 'f1' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 114 void g2(int p = f2());// expected-warning {{use of identifier 'f2' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} expected-error {{call to non-static member function without an object argument}} 115 }; 116 117 void foo() 118 { 119 B<int> b; 120 b.g1(); // expected-note {{required here}} 121 b.g2(); // expected-note {{required here}} 122 } 123 124 } 125 126 127 namespace lookup_dependent_base_class_friend { 128 129 template <class T> 130 class B { 131 public: 132 static void g(); // expected-note {{must qualify identifier to find this declaration in dependent base class}} 133 }; 134 135 template <class T> 136 class A : public B<T> { 137 public: 138 friend void foo(A<T> p){ 139 g(); // expected-warning {{use of identifier 'g' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 140 } 141 }; 142 143 int main2() 144 { 145 A<int> a; 146 foo(a); // expected-note {{requested here}} 147 } 148 149 } 150 151 152 namespace lookup_dependent_base_no_typo_correction { 153 154 class C { 155 public: 156 int m_hWnd; 157 }; 158 159 template <class T> 160 class A : public T { 161 public: 162 void f(int hWnd) { 163 m_hWnd = 1; // expected-warning {{use of undeclared identifier 'm_hWnd'; unqualified lookup into dependent bases of class template 'A' is a Microsoft extension}} 164 } 165 }; 166 167 template class A<C>; 168 169 } 170 171 namespace PR12701 { 172 173 class A {}; 174 class B {}; 175 176 template <class T> 177 class Base { 178 public: 179 bool base_fun(void* p) { return false; } // expected-note {{must qualify identifier to find this declaration in dependent base class}} 180 operator T*() const { return 0; } 181 }; 182 183 template <class T> 184 class Container : public Base<T> { 185 public: 186 template <typename S> 187 bool operator=(const Container<S>& rhs) { 188 return base_fun(rhs); // expected-warning {{use of identifier 'base_fun' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 189 } 190 }; 191 192 void f() { 193 Container<A> text_provider; 194 Container<B> text_provider2; 195 text_provider2 = text_provider; // expected-note {{in instantiation of function template specialization}} 196 } 197 198 } // namespace PR12701 199 200 namespace PR16014 { 201 202 struct A { 203 int a; 204 static int sa; 205 }; 206 template <typename T> struct B : T { 207 int foo() { return a; } // expected-warning {{lookup into dependent bases}} 208 int *bar() { return &a; } // expected-warning {{lookup into dependent bases}} 209 int baz() { return T::a; } 210 int T::*qux() { return &T::a; } 211 static int T::*stuff() { return &T::a; } 212 static int stuff1() { return T::sa; } 213 static int *stuff2() { return &T::sa; } 214 static int stuff3() { return sa; } // expected-warning {{lookup into dependent bases}} 215 static int *stuff4() { return &sa; } // expected-warning {{lookup into dependent bases}} 216 }; 217 218 template <typename T> struct C : T { 219 int foo() { return b; } // expected-error {{no member named 'b' in 'PR16014::C<PR16014::A>'}} expected-warning {{lookup into dependent bases}} 220 int *bar() { return &b; } // expected-error {{no member named 'b' in 'PR16014::C<PR16014::A>'}} expected-warning {{lookup into dependent bases}} 221 int baz() { return T::b; } // expected-error {{no member named 'b' in 'PR16014::A'}} 222 int T::*qux() { return &T::b; } // expected-error {{no member named 'b' in 'PR16014::A'}} 223 int T::*fuz() { return &U::a; } // expected-error {{use of undeclared identifier 'U'}} \ 224 // expected-warning {{unqualified lookup into dependent bases of class template 'C'}} 225 }; 226 227 template struct B<A>; 228 template struct C<A>; // expected-note-re 1+ {{in instantiation of member function 'PR16014::C<PR16014::A>::{{.*}}' requested here}} 229 230 template <typename T> struct D : T { 231 struct Inner { 232 int foo() { 233 // FIXME: MSVC can find this in D's base T! Even worse, if ::sa exists, 234 // clang will use it instead. 235 return sa; // expected-error {{use of undeclared identifier 'sa'}} 236 } 237 }; 238 }; 239 template struct D<A>; 240 241 } 242 243 namespace PR19233 { 244 template <class T> 245 struct A : T { 246 void foo() { 247 ::undef(); // expected-error {{no member named 'undef' in the global namespace}} 248 } 249 void bar() { 250 ::UndefClass::undef(); // expected-error {{no member named 'UndefClass' in the global namespace}} 251 } 252 void baz() { 253 B::qux(); // expected-error {{use of undeclared identifier 'B'}} \ 254 // expected-warning {{unqualified lookup into dependent bases of class template 'A'}} 255 } 256 }; 257 258 struct B { void qux(); }; 259 struct C : B { }; 260 template struct A<C>; // No error! B is a base of A<C>, and qux is available. 261 262 struct D { }; 263 template struct A<D>; // expected-note {{in instantiation of member function 'PR19233::A<PR19233::D>::baz' requested here}} 264 265 } 266 267 namespace nonmethod_missing_this { 268 template <typename T> struct Base { int y = 42; }; 269 template <typename T> struct Derived : Base<T> { 270 int x = y; // expected-warning {{lookup into dependent bases}} 271 auto foo(int j) -> decltype(y * j) { // expected-warning {{lookup into dependent bases}} 272 return y * j; // expected-warning {{lookup into dependent bases}} 273 } 274 int bar() { 275 return [&] { return y; }(); // expected-warning {{lookup into dependent bases}} 276 } 277 }; 278 template struct Derived<int>; 279 } 280 281 namespace typedef_in_base { 282 template <typename T> struct A { typedef T NameFromBase; }; 283 template <typename T> struct B : A<T> { 284 NameFromBase m; // expected-warning {{found via unqualified lookup into dependent bases}} 285 }; 286 static_assert(sizeof(B<int>) == 4, ""); 287 } 288 289 namespace struct_in_base { 290 template <typename T> struct A { struct NameFromBase {}; }; 291 template <typename T> struct B : A<T> { 292 NameFromBase m; // expected-warning {{found via unqualified lookup into dependent bases}} 293 }; 294 static_assert(sizeof(B<int>) == 1, ""); 295 } 296 297 namespace enum_in_base { 298 template <typename T> struct A { enum NameFromBase { X }; }; 299 template <typename T> struct B : A<T> { 300 NameFromBase m; // expected-warning {{found via unqualified lookup into dependent bases}} 301 }; 302 static_assert(sizeof(B<int>) == sizeof(A<int>::NameFromBase), ""); 303 } 304 305 namespace two_types_in_base { 306 template <typename T> struct A { typedef T NameFromBase; }; // expected-note {{member found by ambiguous name lookup}} 307 template <typename T> struct B { struct NameFromBase { T m; }; }; // expected-note {{member found by ambiguous name lookup}} 308 template <typename T> struct C : A<T>, B<T> { 309 NameFromBase m; // expected-error {{member 'NameFromBase' found in multiple base classes of different types}} expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 310 }; 311 static_assert(sizeof(C<int>) != 0, ""); // expected-note {{in instantiation of template class 'two_types_in_base::C<int>' requested here}} 312 } 313 314 namespace type_and_decl_in_base { 315 template <typename T> struct A { typedef T NameFromBase; }; 316 template <typename T> struct B { static const T NameFromBase = 42; }; 317 template <typename T> struct C : A<T>, B<T> { 318 NameFromBase m; // expected-error {{unknown type name 'NameFromBase'}} 319 }; 320 } 321 322 namespace classify_type_from_base { 323 template <typename T> struct A { struct NameFromBase {}; }; 324 template <typename T> struct B : A<T> { 325 A<NameFromBase> m; // expected-warning {{found via unqualified lookup into dependent bases}} 326 }; 327 } 328 329 namespace classify_nontype_from_base { 330 // MSVC does not do lookup of non-type declarations from dependent template base 331 // classes. The extra lookup only applies to types. 332 template <typename T> struct A { void NameFromBase() {} }; 333 template <void (*F)()> struct B { }; 334 template <typename T> struct C : A<T> { 335 B<C::NameFromBase> a; // correct 336 B<NameFromBase> b; // expected-error {{use of undeclared identifier 'NameFromBase'}} 337 }; 338 } 339 340 namespace template_in_base { 341 template <typename T> struct A { 342 template <typename U> struct NameFromBase { U x; }; 343 }; 344 template <typename T> struct B : A<T> { 345 // Correct form. 346 typename B::template NameFromBase<T> m; 347 }; 348 template <typename T> struct C : A<T> { 349 // Incorrect form. 350 NameFromBase<T> m; // expected-error {{unknown type name 'NameFromBase'}} 351 //expected-error@-1 {{expected member name or ';' after declaration specifiers}} 352 }; 353 } 354 355 namespace type_in_inner_class_in_base { 356 template <typename T> 357 struct A { 358 struct B { typedef T NameFromBase; }; 359 }; 360 template <typename T> 361 struct C : A<T>::B { NameFromBase m; }; // expected-error {{unknown type name 'NameFromBase'}} 362 } 363 364 namespace type_in_inner_template_class_in_base { 365 template <typename T> 366 struct A { 367 template <typename U> struct B { typedef U InnerType; }; 368 }; 369 template <typename T> 370 struct C : A<T>::template B<T> { 371 NameFromBase m; // expected-error {{unknown type name 'NameFromBase'}} 372 }; 373 } 374 375 namespace have_nondependent_base { 376 template <typename T> 377 struct A { 378 // Nothing, lookup should fail. 379 }; 380 template <typename T> 381 struct B : A<T> { NameFromBase m; }; // expected-error {{unknown type name 'NameFromBase'}} 382 struct C : A<int> { NameFromBase m; }; // expected-error {{unknown type name 'NameFromBase'}} 383 } 384 385 namespace type_in_base_of_dependent_base { 386 struct A { typedef int NameFromBase; }; 387 template <typename T> 388 struct B : A {}; 389 template <typename T> 390 struct C : B<T> { NameFromBase m; }; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 391 } 392 393 namespace type_in_second_dependent_base { 394 template <typename T> 395 struct A {}; 396 template<typename T> 397 struct B { typedef T NameFromBase; }; 398 template <typename T> 399 struct D : A<T>, B<T> { NameFromBase m; }; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 400 } 401 402 namespace type_in_second_non_dependent_base { 403 struct A {}; 404 struct B { typedef int NameFromBase; }; 405 template<typename T> 406 struct C : A, B {}; 407 template <typename T> 408 struct D : C<T> { NameFromBase m; }; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 409 } 410 411 namespace type_in_virtual_base_of_dependent_base { 412 template <typename T> 413 struct A { typedef T NameFromBase; }; 414 template <typename T> 415 struct B : virtual A<T> {}; 416 template <typename T> 417 struct C : B<T>, virtual A<T> { NameFromBase m; }; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 418 C<int> c; 419 } 420 421 namespace type_in_base_of_multiple_dependent_bases { 422 template <typename T> 423 struct A { typedef T NameFromBase; }; 424 template <typename T> 425 struct B : public A<T> {}; 426 template <typename T> 427 struct C : B<T>, public A<T> { NameFromBase m; }; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} expected-warning {{direct base 'A<int>' is inaccessible due to ambiguity:}} 428 C<int> c; // expected-note {{in instantiation of template class 'type_in_base_of_multiple_dependent_bases::C<int>' requested here}} 429 } 430 431 namespace type_in_dependent_base_of_non_dependent_type { 432 template<typename T> struct A { typedef int NameFromBase; }; 433 template<typename T> struct B : A<T> { 434 struct C; 435 template<typename TT> 436 struct D : C { 437 NameFromBase m; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 438 }; 439 struct E : C { 440 NameFromBase m; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 441 }; 442 }; 443 template<typename T> struct B<T>::C : B { 444 NameFromBase m; // expected-warning {{use of identifier 'NameFromBase' found via unqualified lookup into dependent bases of class templates is a Microsoft extension}} 445 }; 446 template<typename T> struct F : B<T>::C { 447 NameFromBase m; // expected-error {{unknown type name 'NameFromBase'}} 448 }; 449 } 450 451 namespace lookup_in_function_contexts { 452 template <typename T> struct A { typedef T NameFromBase; }; 453 template <typename T> 454 struct B : A<T> { 455 // expected-warning@+1 {{lookup into dependent bases}} 456 static auto lateSpecifiedFunc() -> decltype(NameFromBase()) { 457 return {}; 458 } 459 460 static void memberFunc() { 461 NameFromBase x; // expected-warning {{lookup into dependent bases}} 462 } 463 464 static void funcLocalClass() { 465 struct X { 466 NameFromBase x; // expected-warning {{lookup into dependent bases}} 467 } y; 468 } 469 470 void localClassMethod() { 471 struct X { 472 void bar() { 473 NameFromBase m; // expected-warning {{lookup into dependent bases}} 474 } 475 } x; 476 x.bar(); 477 } 478 479 static void funcLambda() { 480 auto l = []() { 481 NameFromBase x; // expected-warning {{lookup into dependent bases}} 482 }; 483 l(); 484 } 485 486 static constexpr int constexprFunc() { 487 NameFromBase x = {}; // expected-warning {{lookup into dependent bases}} 488 return sizeof(x); 489 } 490 491 static auto autoFunc() { 492 NameFromBase x; // expected-warning {{lookup into dependent bases}} 493 return x; 494 } 495 }; 496 497 // Force us to parse the methods. 498 template struct B<int>; 499 } 500 501 namespace function_template_deduction { 502 // Overloaded function templates. 503 template <int N> int f() { return N; } 504 template <typename T> int f() { return sizeof(T); } 505 506 // Dependent base class with type. 507 template <typename T> 508 struct A { typedef T NameFromBase; }; 509 template <typename T> 510 struct B : A<T> { 511 // expected-warning@+1 {{found via unqualified lookup into dependent bases}} 512 int x = f<NameFromBase>(); 513 }; 514 515 // Dependent base class with enum. 516 template <typename T> struct C { enum { NameFromBase = 4 }; }; 517 template <typename T> struct D : C<T> { 518 // expected-warning@+1 {{use of undeclared identifier 'NameFromBase'; unqualified lookup into dependent bases}} 519 int x = f<NameFromBase>(); 520 }; 521 } 522 523 namespace function_template_undef_impl { 524 template<class T> 525 void f() { 526 Undef::staticMethod(); // expected-error {{use of undeclared identifier 'Undef'}} 527 UndefVar.method(); // expected-error {{use of undeclared identifier 'UndefVar'}} 528 } 529 } 530 531 namespace PR20716 { 532 template <template <typename T> class A> 533 struct B : A<int> 534 { 535 XXX x; // expected-error {{unknown type name}} 536 }; 537 538 template <typename T> 539 struct C {}; 540 541 template <typename T> 542 using D = C<T>; 543 544 template <typename T> 545 struct E : D<T> 546 { 547 XXX x; // expected-error {{unknown type name}} 548 }; 549 } 550 551 namespace PR23810 { 552 void f(int); 553 struct Base { 554 void f(); // expected-note{{must qualify identifier to find this declaration in dependent base class}} 555 }; 556 template <typename T> struct Template : T { 557 void member() { 558 f(); // expected-warning {{found via unqualified lookup into dependent bases}} 559 } 560 }; 561 void test() { 562 Template<Base> x; 563 x.member(); // expected-note{{requested here}} 564 }; 565 } 566 567 namespace PR23823 { 568 // Don't delay lookup in SFINAE context. 569 template <typename T> decltype(g(T())) check(); // expected-note{{candidate template ignored: substitution failure [with T = int]: use of undeclared identifier 'g'}} 570 decltype(check<int>()) x; // expected-error{{no matching function for call to 'check'}} 571 572 void h(); 573 template <typename T> decltype(h(T())) check2(); // expected-note{{candidate template ignored: substitution failure [with T = int]: no matching function for call to 'h'}} 574 decltype(check2<int>()) y; // expected-error{{no matching function for call to 'check2'}} 575 } 576