| // 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; |
| 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-warning{{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; |
| } |