2019-02-05 20:05:53 +08:00
|
|
|
// RUN: %clang_cc1 -std=c++14 -fsyntax-only -verify %s
|
2018-05-11 10:43:08 +08:00
|
|
|
|
|
|
|
using nullptr_t = decltype(nullptr);
|
2010-01-13 01:06:20 +08:00
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct Base {
|
|
|
|
T inner;
|
|
|
|
};
|
|
|
|
|
2018-05-11 10:43:08 +08:00
|
|
|
int z;
|
|
|
|
|
2010-01-13 01:06:20 +08:00
|
|
|
template<typename T>
|
2018-05-11 10:43:08 +08:00
|
|
|
struct X : Base<T> {
|
|
|
|
static int z;
|
|
|
|
|
|
|
|
template<int U>
|
2010-01-13 01:06:20 +08:00
|
|
|
struct Inner {
|
|
|
|
};
|
|
|
|
|
|
|
|
bool f(T other) {
|
2018-05-11 10:43:08 +08:00
|
|
|
// A pair of comparisons; 'inner' is a dependent name so can't be assumed
|
|
|
|
// to be a template.
|
|
|
|
return this->inner < other > ::z;
|
2010-01-13 01:06:20 +08:00
|
|
|
}
|
|
|
|
};
|
2018-05-11 10:43:08 +08:00
|
|
|
|
|
|
|
void use_x(X<int> x) { x.f(0); }
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct Y {
|
|
|
|
static int z;
|
|
|
|
|
|
|
|
template<int U>
|
|
|
|
struct Inner : Y { // expected-note {{declared here}}
|
|
|
|
};
|
|
|
|
|
2020-10-14 10:31:41 +08:00
|
|
|
bool f(T other) {
|
2018-05-11 10:43:08 +08:00
|
|
|
// We can determine that 'inner' does not exist at parse time, so can
|
|
|
|
// perform typo correction in this case.
|
|
|
|
return this->inner<other>::z; // expected-error {{no template named 'inner' in 'Y<T>'; did you mean 'Inner'?}}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Q { constexpr operator int() { return 0; } };
|
2020-10-14 10:31:41 +08:00
|
|
|
void use_y(Y<Q> x) { x.f(Q()); }
|