| // RUN: not %clang_cc1 -std=c++11 %s -fsyntax-only 2>&1 | FileCheck %s |
| // RUN: %clang_cc1 -std=c++11 %s -fsyntax-only -DWARN 2>&1 | FileCheck %s --check-prefix=CHECK-WARN |
| |
| #ifndef WARN |
| |
| // Ensure that the diagnostics we produce for this situation appear in a |
| // deterministic order. This requires ADL to provide lookup results in a |
| // deterministic order. |
| template<typename T, typename> struct Error { typedef typename T::error error; }; |
| struct X { template<typename T> friend typename Error<X, T>::error f(X, T); }; |
| struct Y { template<typename T> friend typename Error<Y, T>::error f(T, Y); }; |
| |
| void g() { |
| f(X(), Y()); |
| } |
| |
| // We don't really care which order these two diagnostics appear (although the |
| // order below is source order, which seems best). The crucial fact is that |
| // there is one single order that is stable across multiple runs of clang. |
| // |
| // CHECK: no type named 'error' in 'X' |
| // CHECK: no type named 'error' in 'Y' |
| // CHECK: no matching function for call to 'f' |
| |
| |
| struct Oper { |
| template<typename T, typename U = typename Error<Oper, T>::error> operator T(); |
| |
| operator int*(); |
| operator float*(); |
| operator X*(); |
| operator Y*(); |
| |
| operator int(*[1])(); |
| operator int(*[2])(); |
| operator int(*[3])(); |
| operator int(*[4])(); |
| operator int(*[5])(); |
| operator int(*[6])(); |
| operator int(*[7])(); |
| operator int(*[8])(); |
| operator float(*[1])(); |
| operator float(*[2])(); |
| operator float(*[3])(); |
| operator float(*[4])(); |
| operator float(*[5])(); |
| operator float(*[6])(); |
| operator float(*[7])(); |
| operator float(*[8])(); |
| }; |
| int *p = Oper() + 0; |
| |
| // CHECK: no type named 'error' in 'Oper' |
| // CHECK: in instantiation of template class 'Error<Oper, int *>' |
| // CHECK: no type named 'error' in 'Oper' |
| // CHECK: in instantiation of template class 'Error<Oper, float *>' |
| // CHECK: no type named 'error' in 'Oper' |
| // CHECK: in instantiation of template class 'Error<Oper, X *>' |
| // CHECK: no type named 'error' in 'Oper' |
| // CHECK: in instantiation of template class 'Error<Oper, Y *>' |
| |
| #endif |
| |
| template<typename T> struct UndefButUsed { |
| static inline int f(); |
| static int g() { return f(); } |
| }; |
| int undef_but_used = UndefButUsed<int>::g() + UndefButUsed<float>::g() + UndefButUsed<char>::g() + UndefButUsed<void>::g(); |
| |
| // CHECK-WARN: inline function 'UndefButUsed<int>::f' is not defined |
| // CHECK-WARN: inline function 'UndefButUsed<float>::f' is not defined |
| // CHECK-WARN: inline function 'UndefButUsed<char>::f' is not defined |
| // CHECK-WARN: inline function 'UndefButUsed<void>::f' is not defined |