You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
250 lines
5.3 KiB
250 lines
5.3 KiB
// 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-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;
|
|
}
|