p5.cpp revision 651f13cea278ec967336033dd032faef0e9fc2ec
1// RUN: %clang_cc1 -fsyntax-only -verify %s
2
3// C++0x [temp.arg.nontype] p5:
4//   The following conversions are performed on each expression used as
5//   a non-type template-argument. If a non-type template-argument cannot be
6//   converted to the type of the corresponding template-parameter then the
7//   program is ill-formed.
8//     -- for a non-type template-parameter of integral or enumeration type,
9//        integral promotions (4.5) and integral conversions (4.7) are applied.
10namespace integral_parameters {
11  template<short s> struct X0 { };
12  X0<17> x0i;
13  X0<'a'> x0c;
14  template<char c> struct X1 { };
15  X1<100l> x1l;
16}
17
18//     -- for a non-type template-parameter of type pointer to object,
19//        qualification conversions (4.4) and the array-to-pointer conversion
20//        (4.2) are applied; if the template-argument is of type
21//        std::nullptr_t, the null pointer conversion (4.10) is applied.
22namespace pointer_to_object_parameters {
23  // PR6226
24  struct Str {
25    Str(const char *);
26  };
27
28  template<const char *s>
29  struct A {
30    Str get() { return s; }
31  };
32
33  char hello[6] = "Hello";
34  extern const char world[6];
35  const char world[6] = "world";
36  void test() {
37    (void)A<hello>().get();
38    (void)A<world>().get();
39  }
40
41  class X {
42  public:
43    X();
44    X(int, int);
45    operator int() const;
46  };
47
48  template<X const *Ptr> struct A2; // expected-note{{template parameter is declared here}}
49
50  X *X_ptr;
51  X an_X;
52  X array_of_Xs[10];
53  A2<X_ptr> *a12; // expected-error{{must have its address taken}}
54  A2<array_of_Xs> *a13;
55  A2<&an_X> *a13_2;
56  A2<(&an_X)> *a13_3; // expected-warning{{address non-type template argument cannot be surrounded by parentheses}}
57
58  // PR6244
59  struct X1 {} X1v;
60  template <X1*> struct X2 { };
61  template <X1* Value> struct X3 : X2<Value> { };
62  struct X4 : X3<&X1v> { };
63
64  // PR6563
65  int *bar;
66  template <int *> struct zed {}; // expected-note 2{{template parameter is declared here}}
67  void g(zed<bar>*); // expected-error{{must have its address taken}}
68
69  int baz;
70  void g2(zed<baz>*); // expected-error{{must have its address taken}}
71
72  void g3(zed<&baz>*); // okay
73}
74
75//     -- For a non-type template-parameter of type reference to object, no
76//        conversions apply. The type referred to by the reference may be more
77//        cv-qualified than the (otherwise identical) type of the
78//        template-argument. The template-parameter is bound directly to the
79//        template-argument, which shall be an lvalue.
80namespace reference_parameters {
81  template <int& N> struct S0 { }; // expected-note 3 {{template parameter is declared here}}
82  template <const int& N> struct S1 { }; // expected-note 2 {{template parameter is declared here}}
83  template <volatile int& N> struct S2 { }; // expected-note 2 {{template parameter is declared here}}
84  template <const volatile int& N> struct S3 { };
85  int i;
86  extern const int ci;
87  volatile int vi;
88  extern const volatile int cvi;
89  void test() {
90    S0<i> s0;
91    S0<ci> s0c; // expected-error{{reference binding of non-type template parameter of type 'int &' to template argument of type 'const int' ignores qualifiers}}
92    S0<vi> s0v; // expected-error{{reference binding of non-type template parameter of type 'int &' to template argument of type 'volatile int' ignores qualifiers}}
93    S0<cvi> s0cv; // expected-error{{reference binding of non-type template parameter of type 'int &' to template argument of type 'const volatile int' ignores qualifiers}}
94
95    S1<i> s1;
96    S1<ci> s1c;
97    S1<vi> s1v; // expected-error{{reference binding of non-type template parameter of type 'const int &' to template argument of type 'volatile int' ignores qualifiers}}
98    S1<cvi> s1cv; // expected-error{{reference binding of non-type template parameter of type 'const int &' to template argument of type 'const volatile int' ignores qualifiers}}
99
100    S2<i> s2;
101    S2<ci> s2c; // expected-error{{reference binding of non-type template parameter of type 'volatile int &' to template argument of type 'const int' ignores qualifiers}}
102    S2<vi> s2v;
103    S2<cvi> s2cv; // expected-error{{reference binding of non-type template parameter of type 'volatile int &' to template argument of type 'const volatile int' ignores qualifiers}}
104
105    S3<i> s3;
106    S3<ci> s3c;
107    S3<vi> s3v;
108    S3<cvi> s3cv;
109  }
110
111  namespace PR6250 {
112    template <typename T, const T &ref> void inc() {
113      ref++; // expected-error{{read-only variable is not assignable}}
114    }
115
116    template<typename T, const T &ref> void bind() {
117      T &ref2 = ref; // expected-error{{drops qualifiers}}
118    }
119
120    int counter;
121    void test() {
122      inc<int, counter>(); // expected-note{{instantiation of}}
123      bind<int, counter>(); // expected-note{{instantiation of}}
124    }
125  }
126
127  namespace PR6749 {
128    template <int& i> struct foo {}; // expected-note{{template parameter is declared here}}
129    int x, &y = x;
130    foo<y> f; // expected-error{{is not an object}}
131  }
132}
133
134//     -- For a non-type template-parameter of type pointer to function, the
135//        function-to-pointer conversion (4.3) is applied; if the
136//        template-argument is of type std::nullptr_t, the null pointer
137//        conversion (4.10) is applied. If the template-argument represents
138//        a set of overloaded functions (or a pointer to such), the matching
139//        function is selected from the set (13.4).
140namespace pointer_to_function {
141  template<int (*)(int)> struct X0 { }; // expected-note 3{{template parameter is declared here}}
142  int f(int);
143  int f(float);
144  int g(float);
145  int (*funcptr)(int);
146  void x0a(X0<f>);
147  void x0b(X0<&f>);
148  void x0c(X0<g>); // expected-error{{non-type template argument of type 'int (float)' cannot be converted to a value of type 'int (*)(int)'}}
149  void x0d(X0<&g>); // expected-error{{non-type template argument of type 'int (*)(float)' cannot be converted to a value of type 'int (*)(int)'}}
150  void x0e(X0<funcptr>); // expected-error{{must have its address taken}}
151}
152
153//     -- For a non-type template-parameter of type reference to function, no
154//        conversions apply. If the template-argument represents a set of
155//        overloaded functions, the matching function is selected from the set
156//        (13.4).
157namespace reference_to_function {
158  template<int (&)(int)> struct X0 { }; // expected-note 4{{template parameter is declared here}}
159  int f(int);
160  int f(float);
161  int g(float);
162  int (*funcptr)(int);
163  void x0a(X0<f>);
164  void x0b(X0<&f>); // expected-error{{address taken in non-type template argument for template parameter of reference type 'int (&)(int)'}}
165  void x0c(X0<g>); // expected-error{{non-type template parameter of reference type 'int (&)(int)' cannot bind to template argument of type 'int (float)'}}
166  void x0d(X0<&g>); // expected-error{{address taken in non-type template argument for template parameter of reference type 'int (&)(int)'}}
167  void x0e(X0<funcptr>); // expected-error{{non-type template parameter of reference type 'int (&)(int)' cannot bind to template argument of type 'int (*)(int)'}}
168}
169//     -- For a non-type template-parameter of type pointer to member function,
170//        if the template-argument is of type std::nullptr_t, the null member
171//        pointer conversion (4.11) is applied; otherwise, no conversions
172//        apply. If the template-argument represents a set of overloaded member
173//        functions, the matching member function is selected from the set
174//        (13.4).
175namespace pointer_to_member_function {
176  struct X { };
177  struct Y : X {
178    int f(int);
179    int g(int);
180    int g(float);
181    float h(float);
182  };
183
184  template<int (Y::*)(int)> struct X0 {}; // expected-note{{template parameter is declared here}}
185  X0<&Y::f> x0a;
186  X0<&Y::g> x0b;
187  X0<&Y::h> x0c; // expected-error-re{{non-type template argument of type 'float (pointer_to_member_function::Y::*)(float){{( __attribute__\(\(thiscall\)\))?}}' cannot be converted to a value of type 'int (pointer_to_member_function::Y::*)(int){{( __attribute__\(\(thiscall\)\))?}}'}}
188}
189
190//     -- For a non-type template-parameter of type pointer to data member,
191//        qualification conversions (4.4) are applied; if the template-argument
192//        is of type std::nullptr_t, the null member pointer conversion (4.11)
193//        is applied.
194namespace pointer_to_member_data {
195  struct X { int x; };
196  struct Y : X { int y; };
197
198  template<int Y::*> struct X0 {}; // expected-note{{template parameter is declared here}}
199  X0<&Y::y> x0a;
200  X0<&Y::x> x0b;  // expected-error{{non-type template argument of type 'int pointer_to_member_data::X::*' cannot be converted to a value of type 'int pointer_to_member_data::Y::*'}}
201
202  // Test qualification conversions
203  template<const int Y::*> struct X1 {};
204  X1<&Y::y> x1a;
205}
206