Nemanja Ivanovic | db64e7e | 2019-02-05 12:05:53 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -std=c++14 -fsyntax-only -verify %s |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 2 | |
| 3 | template<typename T> class A; |
| 4 | |
| 5 | extern "C++" { |
| 6 | template<typename T> class B; |
| 7 | } |
| 8 | |
| 9 | namespace N { |
| 10 | template<typename T> class C; |
| 11 | } |
| 12 | |
Alex Lorenz | 560ae56 | 2016-11-02 15:46:34 +0000 | [diff] [blame] | 13 | extern "C" { // expected-note {{extern "C" language linkage specification begins here}} |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 14 | template<typename T> class D; // expected-error{{templates must have C++ linkage}} |
| 15 | } |
| 16 | |
Alex Lorenz | 560ae56 | 2016-11-02 15:46:34 +0000 | [diff] [blame] | 17 | extern "C" { // expected-note 2 {{extern "C" language linkage specification begins here}} |
Benjamin Kramer | 35e6fee | 2014-02-02 16:35:43 +0000 | [diff] [blame] | 18 | class PR17968 { |
| 19 | template<typename T> class D; // expected-error{{templates must have C++ linkage}} |
| 20 | template<typename T> void f(); // expected-error{{templates must have C++ linkage}} |
| 21 | }; |
| 22 | } |
| 23 | |
Douglas Gregor | f9aa526 | 2009-05-03 17:18:57 +0000 | [diff] [blame] | 24 | template<class U> class A; // expected-note{{previous template declaration is here}} |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 25 | |
| 26 | template<int N> class A; // expected-error{{template parameter has a different kind in template redeclaration}} |
| 27 | |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 28 | template<int N> class NonTypeTemplateParm; |
| 29 | |
| 30 | typedef int INT; |
| 31 | |
Chris Lattner | 810d330 | 2009-02-19 23:45:49 +0000 | [diff] [blame] | 32 | template<INT M> class NonTypeTemplateParm; // expected-note{{previous non-type template parameter with type 'INT' (aka 'int') is here}} |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 33 | |
| 34 | template<long> class NonTypeTemplateParm; // expected-error{{template non-type parameter has a different type 'long' in template redeclaration}} |
| 35 | |
| 36 | template<template<typename T> class X> class TemplateTemplateParm; |
| 37 | |
| 38 | template<template<class> class Y> class TemplateTemplateParm; // expected-note{{previous template declaration is here}} \ |
| 39 | // expected-note{{previous template template parameter is here}} |
| 40 | |
| 41 | template<typename> class TemplateTemplateParm; // expected-error{{template parameter has a different kind in template redeclaration}} |
| 42 | |
| 43 | template<template<typename T, int> class X> class TemplateTemplateParm; // expected-error{{too many template parameters in template template parameter redeclaration}} |
| 44 | |
Douglas Gregor | 71a5718 | 2009-06-22 23:20:33 +0000 | [diff] [blame] | 45 | template<typename T> |
| 46 | struct test {}; // expected-note{{previous definition}} |
| 47 | |
| 48 | template<typename T> |
| 49 | struct test : T {}; // expected-error{{redefinition}} |
| 50 | |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 51 | class X { |
| 52 | public: |
| 53 | template<typename T> class C; |
| 54 | }; |
| 55 | |
| 56 | void f() { |
Douglas Gregor | 866ad5d | 2009-11-05 20:02:41 +0000 | [diff] [blame] | 57 | template<typename T> class X; // expected-error{{expression}} |
Douglas Gregor | cd72ba9 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 58 | } |
Douglas Gregor | 4ebb7f3 | 2009-11-05 20:54:04 +0000 | [diff] [blame] | 59 | |
Tim Northover | 36bb6d5 | 2017-12-09 12:09:54 +0000 | [diff] [blame] | 60 | template<typename T> class X1 var; // expected-error {{variable has incomplete type 'class X1'}} \ |
Larisse Voufo | 39a1e50 | 2013-08-06 01:03:05 +0000 | [diff] [blame] | 61 | // expected-note {{forward declaration of 'X1'}} |
Douglas Gregor | ce40e2e | 2010-04-12 16:00:01 +0000 | [diff] [blame] | 62 | |
| 63 | namespace M { |
| 64 | } |
| 65 | |
| 66 | template<typename T> class M::C3 { }; // expected-error{{out-of-line definition of 'C3' does not match any declaration in namespace 'M'}} |
Nick Lewycky | 6147891 | 2010-11-08 23:29:42 +0000 | [diff] [blame] | 67 | |
| 68 | namespace PR8001 { |
| 69 | template<typename T1> |
| 70 | struct Foo { |
| 71 | template<typename T2> class Bar; |
| 72 | typedef Bar<T1> Baz; |
| 73 | |
| 74 | template<typename T2> |
| 75 | struct Bar { |
| 76 | Bar() {} |
| 77 | }; |
| 78 | }; |
| 79 | |
| 80 | void pr8001() { |
| 81 | Foo<int>::Baz x; |
| 82 | Foo<int>::Bar<int> y(x); |
| 83 | } |
| 84 | } |
Douglas Gregor | fbf8752 | 2011-10-21 15:47:52 +0000 | [diff] [blame] | 85 | |
Douglas Gregor | 4109afa | 2011-11-07 17:43:18 +0000 | [diff] [blame] | 86 | namespace rdar9676205 { |
| 87 | template <unsigned, class _Tp> class tuple_element; |
| 88 | |
| 89 | template <class _T1, class _T2> class pair; |
| 90 | |
| 91 | template <class _T1, class _T2> |
| 92 | class tuple_element<0, pair<_T1, _T2> > |
| 93 | { |
| 94 | template <class _Tp> |
| 95 | struct X |
| 96 | { |
| 97 | template <class _Up, bool = X<_Up>::value> |
| 98 | struct Y |
| 99 | : public X<_Up>, |
| 100 | public Y<_Up> |
| 101 | { }; |
| 102 | }; |
| 103 | }; |
| 104 | } |
Richard Smith | 6483d22 | 2012-04-21 01:27:54 +0000 | [diff] [blame] | 105 | |
| 106 | namespace redecl { |
| 107 | int A; // expected-note {{here}} |
| 108 | template<typename T> struct A; // expected-error {{different kind of symbol}} |
| 109 | |
Richard Smith | 9f2a7b2 | 2012-04-21 01:51:32 +0000 | [diff] [blame] | 110 | int B; // expected-note {{here}} |
Richard Smith | 6483d22 | 2012-04-21 01:27:54 +0000 | [diff] [blame] | 111 | template<typename T> struct B { // expected-error {{different kind of symbol}} |
| 112 | }; |
| 113 | |
| 114 | template<typename T> struct F; |
| 115 | template<typename T> struct K; |
| 116 | |
Richard Smith | 9f2a7b2 | 2012-04-21 01:51:32 +0000 | [diff] [blame] | 117 | int G, H; // expected-note {{here}} |
Richard Smith | 6483d22 | 2012-04-21 01:27:54 +0000 | [diff] [blame] | 118 | |
| 119 | struct S { |
| 120 | int C; // expected-note {{here}} |
| 121 | template<typename T> struct C; // expected-error {{different kind of symbol}} |
| 122 | |
Richard Smith | 9f2a7b2 | 2012-04-21 01:51:32 +0000 | [diff] [blame] | 123 | int D; // expected-note {{here}} |
Richard Smith | 6483d22 | 2012-04-21 01:27:54 +0000 | [diff] [blame] | 124 | template<typename T> struct D { // expected-error {{different kind of symbol}} |
| 125 | }; |
| 126 | |
| 127 | int E; |
| 128 | template<typename T> friend struct E { // expected-error {{cannot define a type in a friend}} |
| 129 | }; |
| 130 | |
| 131 | int F; |
| 132 | template<typename T> friend struct F; // ok, redecl::F |
| 133 | |
| 134 | template<typename T> struct G; // ok |
| 135 | |
| 136 | template<typename T> friend struct H; // expected-error {{different kind of symbol}} |
| 137 | |
| 138 | int I, J, K; |
| 139 | |
| 140 | struct U { |
| 141 | template<typename T> struct I; // ok |
| 142 | template<typename T> struct J { // ok |
| 143 | }; |
| 144 | template<typename T> friend struct K; // ok, redecl::K |
| 145 | }; |
| 146 | }; |
| 147 | } |
David Majnemer | 8e1a913 | 2015-01-15 07:04:38 +0000 | [diff] [blame] | 148 | |
| 149 | extern "C" template <typename T> // expected-error{{templates must have C++ linkage}} |
Alex Lorenz | 560ae56 | 2016-11-02 15:46:34 +0000 | [diff] [blame] | 150 | void DontCrashOnThis() { // expected-note@-1 {{extern "C" language linkage specification begins here}} |
David Majnemer | 8e1a913 | 2015-01-15 07:04:38 +0000 | [diff] [blame] | 151 | T &pT = T(); |
| 152 | pT; |
| 153 | } |
Richard Smith | ac9f5a9 | 2015-12-19 03:12:14 +0000 | [diff] [blame] | 154 | |
| 155 | namespace abstract_dependent_class { |
| 156 | template<typename T> struct A { |
| 157 | virtual A<T> *clone() = 0; // expected-note {{pure virtual}} |
| 158 | }; |
| 159 | template<typename T> A<T> *A<T>::clone() { return new A<T>; } // expected-error {{abstract class type 'A<T>'}} |
| 160 | } |