| // RUN: %clang_cc1 -verify -fsyntax-only %s -Wno-c++11-extensions -Wno-c++1y-extensions -DPRECXX11 |
| // RUN: %clang_cc1 -std=c++11 -verify -fsyntax-only -Wno-c++1y-extensions %s |
| // RUN: %clang_cc1 -std=c++1y -verify -fsyntax-only %s |
| |
| class A { |
| template<typename T> const T wrong; // expected-error {{member 'wrong' declared as a template}} |
| template<typename T> const T wrong_init = 5; // expected-error {{member 'wrong_init' declared as a template}} |
| template<typename T, typename T0> static const T right = T(100); |
| template<typename T> static const T right<T,int> = 5; |
| template<typename T> const int right<int,T>; // expected-error {{member 'right' declared as a template}} |
| template<typename T> const float right<float,T> = 5; // expected-error {{member 'right' declared as a template}} |
| template<> static const int right<int,int> = 7; // expected-error {{explicit specialization of 'right' in class scope}} |
| template<> static const float right<float,int>; // expected-error {{explicit specialization of 'right' in class scope}} |
| template static const int right<int,int>; // expected-error {{template specialization requires 'template<>'}} \ |
| // expected-error {{explicit specialization of 'right' in class scope}} |
| }; |
| |
| namespace out_of_line { |
| class B0 { |
| template<typename T, typename T0> static const T right = T(100); |
| template<typename T> static const T right<T,int> = T(5); |
| }; |
| template<> const int B0::right<int,int> = 7; |
| template const int B0::right<int,int>; |
| template<> const int B0::right<int,float>; |
| template const int B0::right<int,float>; |
| |
| class B1 { |
| template<typename T, typename T0> static const T right; |
| template<typename T> static const T right<T,int>; |
| }; |
| template<typename T, typename T0> const T B1::right = T(100); |
| template<typename T> const T B1::right<T,int> = T(5); |
| |
| class B2 { |
| template<typename T, typename T0> static const T right = T(100); // expected-note {{previous definition is here}} |
| template<typename T> static const T right<T,int> = T(5); // expected-note {{previous definition is here}} |
| }; |
| template<typename T, typename T0> const T B2::right = T(100); // expected-error {{redefinition of 'right'}} |
| template<typename T> const T B2::right<T,int> = T(5); // expected-error {{redefinition of 'right'}} |
| |
| class B3 { |
| template<typename T, typename T0> static const T right = T(100); |
| template<typename T> static const T right<T,int> = T(5); |
| }; |
| template<typename T, typename T0> const T B3::right; // expected-error {{forward declaration of variable template cannot have a nested name specifier}} |
| template<typename T> const T B3::right<T,int>; // expected-error {{forward declaration of variable template partial specialization cannot have a nested name specifier}} |
| |
| class B4 { |
| template<typename T, typename T0> static const T right; |
| template<typename T> static const T right<T,int>; |
| template<typename T, typename T0> static const T right_def = T(100); |
| template<typename T> static const T right_def<T,int>; // expected-note {{explicit instantiation refers here}} |
| }; |
| template<typename T, typename T0> const T B4::right; // expected-error {{forward declaration of variable template cannot have a nested name specifier}} |
| template<typename T> const T B4::right<T,int>; // expected-error {{forward declaration of variable template partial specialization cannot have a nested name specifier}} \ |
| // expected-note {{explicit instantiation refers here}} |
| template const int B4::right<int,int>; // expected-error {{explicit instantiation of undefined static data member template 'right' of class}} |
| template const int B4::right_def<int,int>; // expected-error {{explicit instantiation of undefined static data member template 'right_def' of class}} |
| } |
| |
| namespace non_const_init { |
| class A { |
| template<typename T> static T wrong_inst = T(10); // expected-error {{non-const static data member must be initialized out of line}} |
| template<typename T> static T wrong_inst_fixed; |
| }; |
| template int A::wrong_inst<int>; // expected-note {{in instantiation of static data member 'non_const_init::A::wrong_inst<int>' requested here}} |
| template<typename T> T A::wrong_inst_fixed = T(10); |
| template int A::wrong_inst_fixed<int>; |
| |
| class B { |
| template<typename T> static T wrong_inst; |
| template<typename T> static T wrong_inst<T*> = T(100); // expected-error {{non-const static data member must be initialized out of line}} |
| |
| template<typename T> static T wrong_inst_fixed; |
| template<typename T> static T wrong_inst_fixed<T*>; |
| }; |
| template int B::wrong_inst<int*>; // expected-note {{in instantiation of static data member 'non_const_init::B::wrong_inst<int *>' requested here}} |
| template<typename T> T B::wrong_inst_fixed = T(100); |
| template int B::wrong_inst_fixed<int>; |
| |
| class C { |
| template<typename T> static const T right_inst = T(10); |
| template<typename T> static const T right_inst<T*> = T(100); |
| }; |
| template const int C::right_inst<int>; |
| template const int C::right_inst<int*>; |
| |
| namespace pointers { |
| |
| struct C0 { |
| template<typename U> static U Data; |
| template<typename U> static const U Data<U*> = U(); // Okay |
| }; |
| template const int C0::Data<int*>; |
| |
| struct C1a { |
| template<typename U> static U Data; |
| template<typename U> static U* Data<U>; // Okay, with out-of-line definition |
| }; |
| template<typename T> T* C1a::Data<T> = new T(); |
| template int* C1a::Data<int>; |
| |
| struct C1b { |
| template<typename U> static U Data; |
| template<typename U> static const U* Data<U>; // Okay, with out-of-line definition |
| }; |
| template<typename T> const T* C1b::Data<T> = (T*)(0); |
| template const int* C1b::Data<int>; |
| |
| struct C2a { |
| template<typename U> static U Data; |
| template<typename U> static U* Data<U> = new U(); // expected-error {{non-const static data member must be initialized out of line}} |
| }; |
| template int* C2a::Data<int>; // expected-note {{in instantiation of static data member 'non_const_init::pointers::C2a::Data<int>' requested here}} |
| |
| struct C2b { // FIXME: ?!? Should this be an error? pointer-types are automatically non-const? |
| template<typename U> static U Data; |
| template<typename U> static const U* Data<U> = (U*)(0); // expected-error {{non-const static data member must be initialized out of line}} |
| }; |
| template const int* C2b::Data<int>; // expected-note {{in instantiation of static data member 'non_const_init::pointers::C2b::Data<int>' requested here}} |
| } |
| } |
| |
| struct matrix_constants { |
| // TODO: (?) |
| }; |
| |
| namespace in_class_template { |
| |
| template<typename T> |
| class D0 { |
| template<typename U> static U Data; |
| template<typename U> static const U Data<U*> = U(); |
| }; |
| template const int D0<float>::Data<int*>; |
| |
| template<typename T> |
| class D1 { |
| template<typename U> static U Data; |
| template<typename U> static U* Data<U*>; |
| }; |
| template<typename T> |
| template<typename U> U* D1<T>::Data<U*> = (U*)(0); |
| template int* D1<float>::Data<int*>; |
| |
| template<typename T> |
| class D2 { |
| template<typename U> static U Data; |
| template<typename U> static U* Data<U*>; |
| }; |
| template<> |
| template<typename U> U* D2<float>::Data<U*> = (U*)(0) + 1; |
| template int* D1<float>::Data<int*>; |
| |
| template<typename T> |
| struct D3 { |
| template<typename U> static const U Data = U(100); |
| }; |
| template const int D3<float>::Data<int>; |
| #ifndef PRECXX11 |
| static_assert(D3<float>::Data<int> == 100, ""); |
| #endif |
| |
| namespace bug_files { |
| // FIXME: A bug has been filed addressing an issue similar to these. |
| // No error diagnosis should be produced, because an |
| // explicit specialization of a member templates of class |
| // template specialization should not inherit the partial |
| // specializations from the class template specialization. |
| |
| template<typename T> |
| class D0 { |
| template<typename U> static U Data; |
| template<typename U> static const U Data<U*> = U(10); // expected-note {{previous definition is here}} |
| }; |
| template<> |
| template<typename U> U D0<float>::Data<U*> = U(100); // expected-error{{redefinition of 'Data'}} |
| |
| template<typename T> |
| class D1 { |
| template<typename U> static U Data; |
| template<typename U> static U* Data<U*>; // expected-note {{previous definition is here}} |
| }; |
| template<typename T> |
| template<typename U> U* D1<T>::Data<U*> = (U*)(0); |
| template<> |
| template<typename U> U* D1<float>::Data<U*> = (U*)(0) + 1; // expected-error{{redefinition of 'Data'}} |
| } |
| |
| namespace other_bugs { |
| // FIXME: This fails to properly initilize the variable 'k'. |
| |
| template<typename A> struct S { |
| template<typename B> static int V; |
| template<typename B> static int V0; |
| }; |
| template struct S<int>; |
| template<typename A> template<typename B> int S<A>::V0 = 123; |
| template<typename A> template<typename B> int S<A>::V<B> = 123; |
| int k = S<int>::V<void>; |
| } |
| } |
| |
| namespace in_nested_classes { |
| // TODO: |
| } |
| |