
This removes the delayed typo correction functionality from Clang (regular typo correction still remains) due to fragility of the solution. An RFC was posted here: https://discourse.llvm.org/t/rfc-removing-support-for-delayed-typo-correction/86631 and while that RFC was asking for folks to consider stepping up to be maintainers, and we did have a few new contributors show some interest, experiments show that it's likely worth it to remove this functionality entirely and focus efforts on improving regular typo correction. This removal fixes ~20 open issues (quite possibly more), improves compile time performance by roughly .3-.4% (https://llvm-compile-time-tracker.com/?config=Overview&stat=instructions%3Au&remote=AaronBallman&sortBy=date), and does not appear to regress diagnostic behavior in a way we wouldn't find acceptable. Fixes #142457 Fixes #139913 Fixes #138850 Fixes #137867 Fixes #137860 Fixes #107840 Fixes #93308 Fixes #69470 Fixes #59391 Fixes #58172 Fixes #46215 Fixes #45915 Fixes #45891 Fixes #44490 Fixes #36703 Fixes #32903 Fixes #23312 Fixes #69874
210 lines
9.5 KiB
C++
210 lines
9.5 KiB
C++
// RUN: %clang_cc1 -std=c++20 -verify %s
|
|
|
|
// expected-error@+1 {{invalid operands to binary expression ('const char[5]' and 'float')}}
|
|
constexpr bool CausesRecoveryExpr = "test" + 1.0f;
|
|
|
|
template<typename T>
|
|
concept ReferencesCRE = CausesRecoveryExpr;
|
|
|
|
template<typename T> requires CausesRecoveryExpr // #NVC1REQ
|
|
void NoViableCands1(){} // #NVC1
|
|
|
|
template<typename T> requires ReferencesCRE<T> // #NVC2REQ
|
|
void NoViableCands2(){} // #NVC2
|
|
|
|
template<ReferencesCRE T> // #NVC3REQ
|
|
void NoViableCands3(){} // #NVC3
|
|
|
|
void NVCUse() {
|
|
NoViableCands1<int>();
|
|
// expected-error@-1 {{no matching function for call to 'NoViableCands1'}}
|
|
// expected-note@#NVC1{{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#NVC1REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
|
|
NoViableCands2<int>();
|
|
// expected-error@-1 {{no matching function for call to 'NoViableCands2'}}
|
|
// expected-note@#NVC2{{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#NVC2REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
NoViableCands3<int>();
|
|
// expected-error@-1 {{no matching function for call to 'NoViableCands3'}}
|
|
// expected-note@#NVC3{{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#NVC3REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
}
|
|
|
|
template<typename T> requires CausesRecoveryExpr // #OVC1REQ
|
|
void OtherViableCands1(){} // #OVC1
|
|
|
|
template<typename T>
|
|
void OtherViableCands1(){} // #OVC1_ALT
|
|
|
|
template<typename T> requires ReferencesCRE<T> // #OVC2REQ
|
|
void OtherViableCands2(){} // #OVC2
|
|
|
|
template<typename T>
|
|
void OtherViableCands2(){} // #OVC2_ALT
|
|
|
|
template<ReferencesCRE T> // #OVC3REQ
|
|
void OtherViableCands3(){} // #OVC3
|
|
template<typename T>
|
|
void OtherViableCands3(){} // #OVC3_ALT
|
|
|
|
void OVCUse() {
|
|
OtherViableCands1<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands1'}}
|
|
// expected-note@#OVC1_ALT {{candidate function}}
|
|
// expected-note@#OVC1 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OVC1REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
OtherViableCands2<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands2'}}
|
|
// expected-note@#OVC2_ALT {{candidate function}}
|
|
// expected-note@#OVC2 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OVC2REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
OtherViableCands3<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands3'}}
|
|
// expected-note@#OVC3_ALT {{candidate function}}
|
|
// expected-note@#OVC3 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OVC3REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
}
|
|
|
|
template<typename T> requires CausesRecoveryExpr // #OBNVC1REQ
|
|
void OtherBadNoViableCands1(){} // #OBNVC1
|
|
|
|
template<typename T> requires false // #OBNVC1REQ_ALT
|
|
void OtherBadNoViableCands1(){} // #OBNVC1_ALT
|
|
|
|
template<typename T> requires ReferencesCRE<T> // #OBNVC2REQ
|
|
void OtherBadNoViableCands2(){} // #OBNVC2
|
|
|
|
template<typename T> requires false// #OBNVC2REQ_ALT
|
|
void OtherBadNoViableCands2(){} // #OBNVC2_ALT
|
|
|
|
template<ReferencesCRE T> // #OBNVC3REQ
|
|
void OtherBadNoViableCands3(){} // #OBNVC3
|
|
template<typename T> requires false // #OBNVC3REQ_ALT
|
|
void OtherBadNoViableCands3(){} // #OBNVC3_ALT
|
|
|
|
void OBNVCUse() {
|
|
OtherBadNoViableCands1<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherBadNoViableCands1'}}
|
|
// expected-note@#OBNVC1_ALT {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC1REQ_ALT {{because 'false' evaluated to false}}
|
|
// expected-note@#OBNVC1 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC1REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
OtherBadNoViableCands2<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherBadNoViableCands2'}}
|
|
// expected-note@#OBNVC2_ALT {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC2REQ_ALT {{because 'false' evaluated to false}}
|
|
// expected-note@#OBNVC2 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC2REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
OtherBadNoViableCands3<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherBadNoViableCands3'}}
|
|
// expected-note@#OBNVC3_ALT {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC3REQ_ALT {{because 'false' evaluated to false}}
|
|
// expected-note@#OBNVC3 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#OBNVC3REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
}
|
|
|
|
|
|
// Same tests with member functions.
|
|
struct OVC {
|
|
template<typename T> requires CausesRecoveryExpr // #MEMOVC1REQ
|
|
void OtherViableCands1(){} // #MEMOVC1
|
|
|
|
template<typename T>
|
|
void OtherViableCands1(){} // #MEMOVC1_ALT
|
|
|
|
template<typename T> requires ReferencesCRE<T> // #MEMOVC2REQ
|
|
void OtherViableCands2(){} // #MEMOVC2
|
|
|
|
template<typename T>
|
|
void OtherViableCands2(){} // #MEMOVC2_ALT
|
|
|
|
template<ReferencesCRE T> // #MEMOVC3REQ
|
|
void OtherViableCands3(){} // #MEMOVC3
|
|
template<typename T>
|
|
void OtherViableCands3(){} // #MEMOVC3_ALT
|
|
};
|
|
|
|
void MemOVCUse() {
|
|
OVC S;
|
|
S.OtherViableCands1<int>();
|
|
// expected-error@-1 {{no matching member function for call to 'OtherViableCands1'}}
|
|
// expected-note@#MEMOVC1_ALT {{candidate function}}
|
|
// expected-note@#MEMOVC1 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#MEMOVC1REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
S.OtherViableCands2<int>();
|
|
// expected-error@-1 {{no matching member function for call to 'OtherViableCands2'}}
|
|
// expected-note@#MEMOVC2_ALT {{candidate function}}
|
|
// expected-note@#MEMOVC2 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#MEMOVC2REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
S.OtherViableCands3<int>();
|
|
// expected-error@-1 {{no matching member function for call to 'OtherViableCands3'}}
|
|
// expected-note@#MEMOVC3_ALT {{candidate function}}
|
|
// expected-note@#MEMOVC3 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#MEMOVC3REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
}
|
|
|
|
struct StaticOVC {
|
|
template<typename T> requires CausesRecoveryExpr // #SMEMOVC1REQ
|
|
static void OtherViableCands1(){} // #SMEMOVC1
|
|
|
|
template<typename T>
|
|
static void OtherViableCands1(){} // #SMEMOVC1_ALT
|
|
|
|
template<typename T> requires ReferencesCRE<T> // #SMEMOVC2REQ
|
|
static void OtherViableCands2(){} // #SMEMOVC2
|
|
|
|
template<typename T>
|
|
static void OtherViableCands2(){} // #SMEMOVC2_ALT
|
|
|
|
template<ReferencesCRE T> // #SMEMOVC3REQ
|
|
static void OtherViableCands3(){} // #SMEMOVC3
|
|
template<typename T>
|
|
static void OtherViableCands3(){} // #SMEMOVC3_ALT
|
|
};
|
|
|
|
void StaticMemOVCUse() {
|
|
StaticOVC::OtherViableCands1<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands1'}}
|
|
// expected-note@#SMEMOVC1_ALT {{candidate function}}
|
|
// expected-note@#SMEMOVC1 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#SMEMOVC1REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
StaticOVC::OtherViableCands2<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands2'}}
|
|
// expected-note@#SMEMOVC2_ALT {{candidate function}}
|
|
// expected-note@#SMEMOVC2 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#SMEMOVC2REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
StaticOVC::OtherViableCands3<int>();
|
|
// expected-error@-1 {{no matching function for call to 'OtherViableCands3'}}
|
|
// expected-note@#SMEMOVC3_ALT {{candidate function}}
|
|
// expected-note@#SMEMOVC3 {{candidate template ignored: constraints not satisfied}}
|
|
// expected-note@#SMEMOVC3REQ{{because substituted constraint expression is ill-formed: constraint depends on a previously diagnosed expression}}
|
|
}
|
|
|
|
namespace GH58548 {
|
|
|
|
template <class, class> struct formatter; // #primary-template
|
|
template <class, class> struct basic_format_context {};
|
|
|
|
template <typename CharType>
|
|
concept has_format_function =
|
|
format(basic_format_context<CharType, CharType>());
|
|
|
|
template <typename ValueType, typename CharType>
|
|
requires has_format_function<CharType>
|
|
struct formatter<ValueType, CharType> {
|
|
template <typename OutputIt>
|
|
CharType format(basic_format_context<OutputIt, CharType>);
|
|
};
|
|
|
|
template <class Ctx> int handle_replacement_field(Ctx arg) {
|
|
formatter<decltype(arg), int> ctx; // expected-error {{implicit instantiation of undefined template}}
|
|
return 0;
|
|
}
|
|
|
|
int x = handle_replacement_field(0);
|
|
// expected-note@-1 {{template specialization 'GH58548::handle_replacement_field<int>' requested here}}
|
|
// expected-note@#primary-template {{is declared here}}
|
|
|
|
} // GH58548
|