| 1 | // RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %s |
| 2 | |
| 3 | // The nested-name-specifier of a qualified declarator-id shall not begin with a decltype-specifier. |
| 4 | class foo { |
| 5 | static int i; |
| 6 | void func(); |
| 7 | }; |
| 8 | |
| 9 | int decltype(foo())::i; // expected-error{{'decltype' cannot be used to name a declaration}} |
| 10 | void decltype(foo())::func() { // expected-error{{'decltype' cannot be used to name a declaration}} |
| 11 | } |
| 12 | |
| 13 | |
| 14 | template<typename T> |
| 15 | class tfoo { |
| 16 | static int i; |
| 17 | void func(); |
| 18 | }; |
| 19 | |
| 20 | template<typename T> |
| 21 | int decltype(tfoo<T>())::i; // expected-error{{nested name specifier 'decltype(tfoo<T>())::' for declaration does not refer into a class, class template or class template partial specialization}} |
| 22 | template<typename T> |
| 23 | void decltype(tfoo<T>())::func() { // expected-error{{nested name specifier 'decltype(tfoo<T>())::' for declaration does not refer into a class, class template or class template partial specialization}} |
| 24 | } |
| 25 | |
| 26 | // An init-declarator named with a qualified-id can refer to an element of the |
| 27 | // inline namespace set of the named namespace. |
| 28 | namespace inline_namespaces { |
| 29 | namespace N { |
| 30 | inline namespace M { |
| 31 | void f(); // expected-note {{possible target}} |
| 32 | void g(); |
| 33 | extern int m; // expected-note {{candidate}} |
| 34 | extern int n; |
| 35 | struct S; // expected-note {{candidate}} |
| 36 | struct T; |
| 37 | enum E : int; // expected-note {{candidate}} |
| 38 | enum F : int; |
| 39 | template<typename T> void ft(); // expected-note {{here}} |
| 40 | template<typename T> void gt(); // expected-note {{here}} |
| 41 | template<typename T> extern int mt; // expected-note {{here}} expected-warning {{extension}} |
| 42 | template<typename T> extern int nt; // expected-note {{here}} expected-warning {{extension}} |
| 43 | template<typename T> struct U; // expected-note {{here}} |
| 44 | template<typename T> struct V; // expected-note {{here}} |
| 45 | } |
| 46 | |
| 47 | // When named by unqualified-id, we do *not* look in the inline namespace |
| 48 | // set. |
| 49 | void f() {} // expected-note {{possible target}} |
| 50 | int m; // expected-note {{candidate}} |
| 51 | struct S {}; // expected-note {{candidate}} |
| 52 | enum E : int {}; // expected-note {{candidate}} |
| 53 | |
| 54 | static_assert(&f != &M::f, ""); // expected-error {{reference to overloaded function could not be resolved}} |
| 55 | static_assert(&m != &M::m, ""); // expected-error {{ambiguous}} |
| 56 | typedef S X; // expected-error {{ambiguous}} |
| 57 | typedef E Y; // expected-error {{ambiguous}} |
| 58 | |
| 59 | // When named by (unqualified) template-id, we do look in the inline |
| 60 | // namespace set. See [namespace.def]p8, [temp.explicit]p3, |
| 61 | // [temp.expl.spec]p2. |
| 62 | // |
| 63 | // This is not explicitly specified for partial specializations, but |
| 64 | // that is just a language defect. |
| 65 | template<> void ft<int>() {} |
| 66 | template void ft<char>(); // expected-error {{undefined}} |
| 67 | |
| 68 | template<typename T> int mt<T*>; |
| 69 | template<> int mt<int>; |
| 70 | template int mt<int*>; |
| 71 | template int mt<char>; // expected-error {{undefined}} |
| 72 | |
| 73 | template<typename T> struct U<T*> {}; |
| 74 | template<> struct U<int> {}; |
| 75 | template struct U<int*>; |
| 76 | template struct U<char>; // expected-error {{undefined}} |
| 77 | } |
| 78 | |
| 79 | // When named by qualified-id, we *do* look in the inline namespace set. |
| 80 | void N::g() {} |
| 81 | int N::n; |
| 82 | struct N::T {}; |
| 83 | enum N::F : int {}; |
| 84 | |
| 85 | static_assert(&N::g == &N::M::g, ""); |
| 86 | static_assert(&N::n == &N::M::n, ""); |
| 87 | typedef N::T X; |
| 88 | typedef N::M::T X; |
| 89 | typedef N::F Y; |
| 90 | typedef N::M::F Y; |
| 91 | |
| 92 | template<> void N::gt<int>() {} |
| 93 | template void N::gt<char>(); // expected-error {{undefined}} |
| 94 | |
| 95 | template<typename T> int N::nt<T*>; |
| 96 | template<> int N::nt<int>; |
| 97 | template int N::nt<int*>; |
| 98 | template int N::nt<char>; // expected-error {{undefined}} |
| 99 | |
| 100 | template<typename T> struct N::V<T*> {}; |
| 101 | template<> struct N::V<int> {}; |
| 102 | template struct N::V<int*>; |
| 103 | template struct N::V<char>; // expected-error {{undefined}} |
| 104 | |
| 105 | struct Q {}; |
| 106 | |
| 107 | // Perversely, inline anonymous namespaces can cause an ostensibly |
| 108 | // external-linkage declaration to acquire internal linkage when |
| 109 | // redeclared with a qualified name. |
| 110 | inline namespace { |
| 111 | struct Q {} q; |
| 112 | int f_in_inline(); |
| 113 | extern int v_in_inline; |
| 114 | typedef int t_in_inline; |
| 115 | } |
| 116 | // FIXME: These "extra qualification" warnings are bogus: the qualification |
| 117 | // changes the meaning of the program. |
| 118 | int inline_namespaces::f_in_inline() { // expected-warning {{extra qualification}} |
| 119 | // Finds <anon>::Q, not inline_namespaces::Q |
| 120 | Q x = q; |
| 121 | return 0; |
| 122 | } |
| 123 | int inline_namespaces::v_in_inline = // expected-warning {{extra qualification}} |
| 124 | (Q(q), 0); |
| 125 | } |
| 126 | |