2009-09-03 07:58:38 +08:00
|
|
|
// RUN: clang-cc -fsyntax-only -verify %s
|
|
|
|
template<typename T>
|
|
|
|
void call_f0(T x) {
|
|
|
|
x.Base::f0();
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Base {
|
|
|
|
void f0();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct X0 : Base {
|
|
|
|
typedef Base CrazyBase;
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_f0(X0 x0) {
|
|
|
|
call_f0(x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TheBase, typename T>
|
|
|
|
void call_f0_through_typedef(T x) {
|
|
|
|
typedef TheBase Base2;
|
|
|
|
x.Base2::f0();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_f0_through_typedef(X0 x0) {
|
|
|
|
call_f0_through_typedef<Base>(x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename TheBase, typename T>
|
|
|
|
void call_f0_through_typedef2(T x) {
|
|
|
|
typedef TheBase CrazyBase; // expected-note{{current scope}}
|
2009-09-04 05:38:09 +08:00
|
|
|
x.CrazyBase::f0(); // expected-error{{ambiguous}} \
|
|
|
|
// expected-error 2{{no member named}}
|
2009-09-03 07:58:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct OtherBase { };
|
|
|
|
|
|
|
|
struct X1 : Base, OtherBase {
|
|
|
|
typedef OtherBase CrazyBase; // expected-note{{object type}}
|
|
|
|
};
|
|
|
|
|
|
|
|
void test_f0_through_typedef2(X0 x0, X1 x1) {
|
|
|
|
call_f0_through_typedef2<Base>(x0);
|
2009-09-04 05:38:09 +08:00
|
|
|
call_f0_through_typedef2<OtherBase>(x1); // expected-note{{instantiation}}
|
|
|
|
call_f0_through_typedef2<Base>(x1); // expected-note{{instantiation}}
|
2009-09-03 07:58:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-04 06:13:48 +08:00
|
|
|
struct X2 {
|
|
|
|
operator int() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T, typename U>
|
|
|
|
T convert(const U& value) {
|
|
|
|
return value.operator T(); // expected-error{{operator long}}
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_convert(X2 x2) {
|
|
|
|
convert<int>(x2);
|
|
|
|
convert<long>(x2); // expected-note{{instantiation}}
|
|
|
|
}
|
2009-09-05 01:36:40 +08:00
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void destruct(T* ptr) {
|
|
|
|
ptr->~T();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void destruct_intptr(int *ip) {
|
|
|
|
ip->~T();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_destruct(X2 *x2p, int *ip) {
|
|
|
|
destruct(x2p);
|
|
|
|
destruct(ip);
|
|
|
|
destruct_intptr<int>(ip);
|
|
|
|
}
|