184 lines
5.3 KiB
C++
184 lines
5.3 KiB
C++
|
// RUN: %clang_cc1 -std=c++14 -fsyntax-only -verify %s
|
||
|
|
||
|
template<typename T> class A {};
|
||
|
|
||
|
extern "C++" {
|
||
|
template<typename T> class B {};
|
||
|
template<typename T> class A<T *>;
|
||
|
template<> class A<int[1]>;
|
||
|
template class A<int[2]>;
|
||
|
template<typename T> class B<T *>;
|
||
|
template<> class B<int[1]>;
|
||
|
template class B<int[2]>;
|
||
|
}
|
||
|
|
||
|
namespace N {
|
||
|
template<typename T> class C;
|
||
|
}
|
||
|
|
||
|
extern "C" { // expected-note 3 {{extern "C" language linkage specification begins here}}
|
||
|
template<typename T> class D; // expected-error{{templates must have C++ linkage}}
|
||
|
template<typename T> class A<T **>; // expected-error{{templates must have C++ linkage}}
|
||
|
template<> class A<int[3]>; // expected-error{{templates must have C++ linkage}}
|
||
|
template class A<int[4]>; // OK (surprisingly) FIXME: Should we warn on this?
|
||
|
}
|
||
|
|
||
|
extern "C" { // expected-note 2 {{extern "C" language linkage specification begins here}}
|
||
|
class PR17968 {
|
||
|
template<typename T> class D; // expected-error{{templates must have C++ linkage}}
|
||
|
template<typename T> void f(); // expected-error{{templates must have C++ linkage}}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
template<class U> class A; // expected-note{{previous template declaration is here}}
|
||
|
|
||
|
template<int N> class A; // expected-error{{template parameter has a different kind in template redeclaration}}
|
||
|
|
||
|
template<int N> class NonTypeTemplateParm;
|
||
|
|
||
|
typedef int INT;
|
||
|
|
||
|
template<INT M> class NonTypeTemplateParm; // expected-note{{previous non-type template parameter with type 'INT' (aka 'int') is here}}
|
||
|
|
||
|
template<long> class NonTypeTemplateParm; // expected-error{{template non-type parameter has a different type 'long' in template redeclaration}}
|
||
|
|
||
|
template<template<typename T> class X> class TemplateTemplateParm;
|
||
|
|
||
|
template<template<class> class Y> class TemplateTemplateParm; // expected-note{{previous template declaration is here}} \
|
||
|
// expected-note{{previous template template parameter is here}}
|
||
|
|
||
|
template<typename> class TemplateTemplateParm; // expected-error{{template parameter has a different kind in template redeclaration}}
|
||
|
|
||
|
template<template<typename T, int> class X> class TemplateTemplateParm; // expected-error{{too many template parameters in template template parameter redeclaration}}
|
||
|
|
||
|
template<typename T>
|
||
|
struct test {}; // expected-note{{previous definition}}
|
||
|
|
||
|
template<typename T>
|
||
|
struct test : T {}; // expected-error{{redefinition}}
|
||
|
|
||
|
class X {
|
||
|
public:
|
||
|
template<typename T> class C;
|
||
|
};
|
||
|
|
||
|
void f() {
|
||
|
template<typename T> class X; // expected-error{{expression}}
|
||
|
}
|
||
|
|
||
|
template<typename T> class X1 var; // expected-error {{variable has incomplete type 'class X1'}} \
|
||
|
// expected-note {{forward declaration of 'X1'}}
|
||
|
|
||
|
namespace M {
|
||
|
}
|
||
|
|
||
|
template<typename T> class M::C3 { }; // expected-error{{out-of-line definition of 'C3' does not match any declaration in namespace 'M'}}
|
||
|
|
||
|
namespace PR8001 {
|
||
|
template<typename T1>
|
||
|
struct Foo {
|
||
|
template<typename T2> class Bar;
|
||
|
typedef Bar<T1> Baz;
|
||
|
|
||
|
template<typename T2>
|
||
|
struct Bar {
|
||
|
Bar() {}
|
||
|
};
|
||
|
};
|
||
|
|
||
|
void pr8001() {
|
||
|
Foo<int>::Baz x;
|
||
|
Foo<int>::Bar<int> y(x);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
namespace rdar9676205 {
|
||
|
template <unsigned, class _Tp> class tuple_element;
|
||
|
|
||
|
template <class _T1, class _T2> class pair;
|
||
|
|
||
|
template <class _T1, class _T2>
|
||
|
class tuple_element<0, pair<_T1, _T2> >
|
||
|
{
|
||
|
template <class _Tp>
|
||
|
struct X
|
||
|
{
|
||
|
template <class _Up, bool = X<_Up>::value>
|
||
|
struct Y
|
||
|
: public X<_Up>,
|
||
|
public Y<_Up>
|
||
|
{ };
|
||
|
};
|
||
|
};
|
||
|
}
|
||
|
|
||
|
namespace redecl {
|
||
|
int A; // expected-note {{here}}
|
||
|
template<typename T> struct A; // expected-error {{different kind of symbol}}
|
||
|
|
||
|
int B; // expected-note {{here}}
|
||
|
template<typename T> struct B { // expected-error {{different kind of symbol}}
|
||
|
};
|
||
|
|
||
|
template<typename T> struct F;
|
||
|
template<typename T> struct K;
|
||
|
|
||
|
int G, H; // expected-note {{here}}
|
||
|
|
||
|
struct S {
|
||
|
int C; // expected-note {{here}}
|
||
|
template<typename T> struct C; // expected-error {{different kind of symbol}}
|
||
|
|
||
|
int D; // expected-note {{here}}
|
||
|
template<typename T> struct D { // expected-error {{different kind of symbol}}
|
||
|
};
|
||
|
|
||
|
int E;
|
||
|
template<typename T> friend struct E { // expected-error {{cannot define a type in a friend}}
|
||
|
};
|
||
|
|
||
|
int F;
|
||
|
template<typename T> friend struct F; // ok, redecl::F
|
||
|
|
||
|
template<typename T> struct G; // ok
|
||
|
|
||
|
template<typename T> friend struct H; // expected-error {{different kind of symbol}}
|
||
|
|
||
|
int I, J, K;
|
||
|
|
||
|
struct U {
|
||
|
template<typename T> struct I; // ok
|
||
|
template<typename T> struct J { // ok
|
||
|
};
|
||
|
template<typename T> friend struct K; // ok, redecl::K
|
||
|
};
|
||
|
};
|
||
|
}
|
||
|
|
||
|
extern "C" template <typename T> // expected-error{{templates must have C++ linkage}}
|
||
|
void DontCrashOnThis() { // expected-note@-1 {{extern "C" language linkage specification begins here}}
|
||
|
T &pT = T();
|
||
|
pT;
|
||
|
}
|
||
|
|
||
|
namespace abstract_dependent_class {
|
||
|
template<typename T> struct A {
|
||
|
virtual A<T> *clone() = 0; // expected-note {{pure virtual}}
|
||
|
};
|
||
|
template<typename T> A<T> *A<T>::clone() { return new A<T>; } // expected-error {{abstract class type 'A<T>'}}
|
||
|
}
|
||
|
|
||
|
namespace qualified_out_of_line {
|
||
|
struct rbnode {};
|
||
|
template<typename T, typename U> struct pair {};
|
||
|
template<typename K, typename V> struct rbtree {
|
||
|
using base = rbnode;
|
||
|
pair<base, base> f();
|
||
|
};
|
||
|
template<typename K, typename V>
|
||
|
pair<typename rbtree<K, V>::base, typename rbtree<K, V>::base>
|
||
|
rbtree<K, V>::f() {
|
||
|
return {};
|
||
|
}
|
||
|
}
|