Clang Project

clang_source_code/test/SemaTemplate/partial-spec-instantiate.cpp
1// RUN: %clang_cc1 -fsyntax-only -verify %s
2// RUN: %clang_cc1 -fsyntax-only -verify -std=c++98 %s
3// RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %s
4
5// PR4607
6template <class T> struct X {};
7
8template <> struct X<char>
9{
10  static char* g();
11};
12
13template <class T> struct X2 {};
14
15template <class U>
16struct X2<U*> {
17  static void f() {
18    X<U>::g();
19  }
20};
21
22void a(char *a, char *b) {X2<char*>::f();}
23
24namespace WonkyAccess {
25  template<typename T>
26  struct X {
27    int m;
28  };
29
30  template<typename U>
31  class Y;
32
33  template<typename U>
34  struct Y<U*> : X<U> { };
35
36  template<>
37  struct Y<float*> : X<float> { };
38
39  int f(Y<int*> y, Y<float*> y2) {
40    return y.m + y2.m;
41  }
42}
43
44// <rdar://problem/9169404>
45namespace rdar9169404 {
46  template<typename T, T N> struct X { };
47  template<bool C> struct X<bool, C> {
48    typedef int type;
49  };
50
51  X<bool, -1>::type value;
52#if __cplusplus >= 201103L
53  // expected-error@-2 {{non-type template argument evaluates to -1, which cannot be narrowed to type 'bool'}}
54#else
55  // expected-no-diagnostics
56#endif
57}
58
59// rdar://problem/39524996
60namespace rdar39524996 {
61  template <typename T, typename U>
62  struct enable_if_not_same
63  {
64    typedef void type;
65  };
66  template <typename T>
67  struct enable_if_not_same<T, T>;
68
69  template <typename T>
70  struct Wrapper {
71    // Assertion triggered on trying to set twice the same partial specialization
72    // enable_if_not_same<int, int>
73    template <class U>
74    Wrapper(const Wrapper<U>& other,
75            typename enable_if_not_same<U, T>::type* = 0) {}
76
77    explicit Wrapper(int i) {}
78  };
79
80  template <class T>
81  struct Container {
82    // It is important that the struct has implicit copy and move constructors.
83    Container() : x() {}
84
85    template <class U>
86    Container(const Container<U>& other) : x(static_cast<T>(other.x)) {}
87
88    // Implicit constructors are member-wise, so the field triggers instantiation
89    // of T constructors and we instantiate all of them for overloading purposes.
90    T x;
91  };
92
93  void takesWrapperInContainer(const Container< Wrapper<int> >& c);
94  void test() {
95    // Type mismatch triggers initialization with conversion which requires
96    // implicit constructors to be instantiated.
97    Container<int> c;
98    takesWrapperInContainer(c);
99  }
100}
101