|  | // RUN: %clang_cc1 -fcxx-exceptions -fexceptions -fsyntax-only -verify %s | 
|  | template<typename T, typename U> | 
|  | struct X0 { | 
|  | void f(T x, U y) { | 
|  | (void)(x + y); // expected-error{{invalid operands}} | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct X1 { }; | 
|  |  | 
|  | template struct X0<int, float>; | 
|  | template struct X0<int*, int>; | 
|  | template struct X0<int X1::*, int>; // expected-note{{instantiation of}} | 
|  |  | 
|  | template<typename T> | 
|  | struct X2 { | 
|  | void f(T); | 
|  |  | 
|  | T g(T x, T y) { | 
|  | /* DeclStmt */; | 
|  | T *xp = &x, &yr = y; // expected-error{{pointer to a reference}} | 
|  | /* NullStmt */; | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct X2<int>; | 
|  | template struct X2<int&>; // expected-note{{instantiation of}} | 
|  |  | 
|  | template<typename T> | 
|  | struct X3 { | 
|  | void f(T) { | 
|  | Label: | 
|  | T x; | 
|  | goto Label; | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct X3<int>; | 
|  |  | 
|  | template <typename T> struct X4 { | 
|  | T f() const { | 
|  | return; // expected-error{{non-void function 'f' should return a value}} | 
|  | } | 
|  |  | 
|  | T g() const { | 
|  | return 1; // expected-error{{void function 'g' should not return a value}} | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct X4<void>; // expected-note{{in instantiation of}} | 
|  | template struct X4<int>; // expected-note{{in instantiation of}} | 
|  |  | 
|  | struct Incomplete; // expected-note 2{{forward declaration}} | 
|  |  | 
|  | template<typename T> struct X5 { | 
|  | T f() { } // expected-error{{incomplete result type}} | 
|  | }; | 
|  | void test_X5(X5<Incomplete> x5); // okay! | 
|  |  | 
|  | template struct X5<Incomplete>; // expected-note{{instantiation}} | 
|  |  | 
|  | template<typename T, typename U, typename V> struct X6 { | 
|  | U f(T t, U u, V v) { | 
|  | // IfStmt | 
|  | if (t > 0) | 
|  | return u; | 
|  | else { | 
|  | if (t < 0) | 
|  | return v; // expected-error{{cannot initialize return object of type}} | 
|  | } | 
|  |  | 
|  | if (T x = t) { | 
|  | t = x; | 
|  | } | 
|  | return v; // expected-error{{cannot initialize return object of type}} | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct ConvertibleToInt { | 
|  | operator int() const; | 
|  | }; | 
|  |  | 
|  | template struct X6<ConvertibleToInt, float, char>; | 
|  | template struct X6<bool, int, int*>; // expected-note{{instantiation}} | 
|  |  | 
|  | template <typename T> struct X7 { | 
|  | void f() { | 
|  | void *v = this; | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct X7<int>; | 
|  |  | 
|  | template<typename T> struct While0 { | 
|  | void f(T t) { | 
|  | while (t) { | 
|  | } | 
|  |  | 
|  | while (T t2 = T()) ; | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct While0<float>; | 
|  |  | 
|  | template<typename T> struct Do0 { | 
|  | void f(T t) { | 
|  | do { | 
|  | } while (t); // expected-error{{not contextually}} | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct NotConvertibleToBool { }; | 
|  | template struct Do0<ConvertibleToInt>; | 
|  | template struct Do0<NotConvertibleToBool>; // expected-note{{instantiation}} | 
|  |  | 
|  | template<typename T> struct For0 { | 
|  | void f(T f, T l) { | 
|  | for (; f != l; ++f) { | 
|  | if (*f) | 
|  | continue; | 
|  | else if (*f == 17) | 
|  | break; | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct For0<int*>; | 
|  |  | 
|  | template<typename T> struct Member0 { | 
|  | void f(T t) { | 
|  | t; | 
|  | t.f; | 
|  | t->f; | 
|  |  | 
|  | T* tp; | 
|  | tp.f; // expected-error{{member reference base type 'T *' is not a structure or union}} | 
|  | tp->f; | 
|  |  | 
|  | this->f; // expected-error{{reference to non-static member function must be called}} | 
|  | this.f; // expected-error{{member reference base type 'Member0<T> *' is not a structure or union}} | 
|  | } | 
|  | }; | 
|  |  | 
|  | template<typename T, typename U> struct Switch0 { | 
|  | U f(T value, U v0, U v1, U v2) { | 
|  | switch (value) { | 
|  | case 0: return v0; | 
|  |  | 
|  | case 1: return v1; | 
|  |  | 
|  | case 2: // fall through | 
|  |  | 
|  | default: | 
|  | return  v2; | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct Switch0<int, float>; | 
|  |  | 
|  | template<typename T, int I1, int I2> struct Switch1 { | 
|  | T f(T x, T y, T z) { | 
|  | switch (x) { | 
|  | case I1: return y; // expected-note{{previous}} | 
|  | case I2: return z; // expected-error{{duplicate}} | 
|  | default: return x; | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct Switch1<int, 1, 2>; | 
|  | template struct Switch1<int, 2, 2>; // expected-note{{instantiation}} | 
|  |  | 
|  | template<typename T> struct IndirectGoto0 { | 
|  | void f(T x) { | 
|  | // FIXME: crummy error message below | 
|  | goto *x; // expected-error{{incompatible}} | 
|  |  | 
|  | prior: | 
|  | T prior_label; | 
|  | prior_label = &&prior; // expected-error{{assigning to 'int'}} | 
|  |  | 
|  | T later_label; | 
|  | later_label = &&later; // expected-error{{assigning to 'int'}} | 
|  |  | 
|  | later: | 
|  | (void)(1+1); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template struct IndirectGoto0<void*>; | 
|  | template struct IndirectGoto0<int>; // expected-note{{instantiation}} | 
|  |  | 
|  | template<typename T> struct TryCatch0 { | 
|  | void f() { | 
|  | try { | 
|  | } catch (T t) { // expected-error{{incomplete type}} \ | 
|  | // expected-error{{abstract class}} | 
|  | } catch (...) { | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct Abstract { | 
|  | virtual void foo() = 0; // expected-note{{pure virtual}} | 
|  | }; | 
|  |  | 
|  | template struct TryCatch0<int>; // okay | 
|  | template struct TryCatch0<Incomplete*>; // expected-note{{instantiation}} | 
|  | template struct TryCatch0<Abstract>; // expected-note{{instantiation}} | 
|  |  | 
|  | // PR4383 | 
|  | template<typename T> struct X; | 
|  | template<typename T> struct Y : public X<T> { | 
|  | Y& x() { return *this; } | 
|  | }; | 
|  |  | 
|  | // Make sure our assertions don't get too uppity. | 
|  | namespace test0 { | 
|  | template <class T> class A { void foo(T array[10]); }; | 
|  | template class A<int>; | 
|  | } | 
|  |  | 
|  | namespace PR7016 { | 
|  | template<typename T> void f() { T x = x; } | 
|  | template void f<int>(); | 
|  | } | 
|  |  | 
|  | namespace PR9880 { | 
|  | struct lua_State; | 
|  | struct no_tag { char a; };			// (A) | 
|  | struct yes_tag { long a; long b; };	// (A) | 
|  |  | 
|  | template <typename T> | 
|  | struct HasIndexMetamethod { | 
|  | template <typename U> | 
|  | static no_tag check(...); | 
|  | template <typename U> | 
|  | static yes_tag check(char[sizeof(&U::luaIndex)]); | 
|  | enum { value = sizeof(check<T>(0)) == sizeof(yes_tag) }; | 
|  | }; | 
|  |  | 
|  | class SomeClass { | 
|  | public: | 
|  | int luaIndex(lua_State* L); | 
|  | }; | 
|  |  | 
|  | int i = HasIndexMetamethod<SomeClass>::value; | 
|  | } |