commit 6c0e2be69e11e4ec8f9b04d3f80b9d4496737b0b Author: Jonathan Wakely Date: Thu Mar 22 13:08:52 2018 +0000 PR libstdc++/85040 fix std::less etc. ambiguities PR libstdc++/85040 * include/bits/stl_function.h (greater::__not_overloaded) (less::__not_overloaded, greater_equal::__not_overloaded) (less_equal::__not_overloaded): Fix ambiguous specializations. * testsuite/20_util/function_objects/comparisons_pointer.cc: Add tests for type with overlaoded operators. diff --git a/libstdc++-v3/include/bits/stl_function.h b/libstdc++-v3/include/bits/stl_function.h index 0affaf7da3a..9e81ad3f20e 100644 --- a/libstdc++-v3/include/bits/stl_function.h +++ b/libstdc++-v3/include/bits/stl_function.h @@ -523,15 +523,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION static_cast(std::forward<_Up>(__u))); } - template - struct __not_overloaded; - - // False if we can call operator>(T,U) - template - struct __not_overloaded<_Tp, _Up, __void_t< - decltype(operator>(std::declval<_Tp>(), std::declval<_Up>()))>> - : false_type { }; - + // True if there is no viable operator> member function. template struct __not_overloaded2 : true_type { }; @@ -541,8 +533,15 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION decltype(std::declval<_Tp>().operator>(std::declval<_Up>()))>> : false_type { }; + // True if there is no overloaded operator> for these operands. + template + struct __not_overloaded : __not_overloaded2<_Tp, _Up> { }; + + // False if we can call operator>(T,U) template - struct __not_overloaded<_Tp, _Up> : __not_overloaded2<_Tp, _Up> { }; + struct __not_overloaded<_Tp, _Up, __void_t< + decltype(operator>(std::declval<_Tp>(), std::declval<_Up>()))>> + : false_type { }; template using __ptr_cmp = __and_<__not_overloaded<_Tp, _Up>, @@ -586,15 +585,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION static_cast(std::forward<_Up>(__u))); } - template - struct __not_overloaded; - - // False if we can call operator<(T,U) - template - struct __not_overloaded<_Tp, _Up, __void_t< - decltype(operator<(std::declval<_Tp>(), std::declval<_Up>()))>> - : false_type { }; - + // True if there is no viable operator< member function. template struct __not_overloaded2 : true_type { }; @@ -604,8 +595,15 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION decltype(std::declval<_Tp>().operator<(std::declval<_Up>()))>> : false_type { }; + // True if there is no overloaded operator< for these operands. + template + struct __not_overloaded : __not_overloaded2<_Tp, _Up> { }; + + // False if we can call operator<(T,U) template - struct __not_overloaded<_Tp, _Up> : __not_overloaded2<_Tp, _Up> { }; + struct __not_overloaded<_Tp, _Up, __void_t< + decltype(operator<(std::declval<_Tp>(), std::declval<_Up>()))>> + : false_type { }; template using __ptr_cmp = __and_<__not_overloaded<_Tp, _Up>, @@ -649,15 +647,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION static_cast(std::forward<_Up>(__u))); } - template - struct __not_overloaded; - - // False if we can call operator>=(T,U) - template - struct __not_overloaded<_Tp, _Up, __void_t< - decltype(operator>=(std::declval<_Tp>(), std::declval<_Up>()))>> - : false_type { }; - + // True if there is no viable operator>= member function. template struct __not_overloaded2 : true_type { }; @@ -667,8 +657,15 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION decltype(std::declval<_Tp>().operator>=(std::declval<_Up>()))>> : false_type { }; + // True if there is no overloaded operator>= for these operands. + template + struct __not_overloaded : __not_overloaded2<_Tp, _Up> { }; + + // False if we can call operator>=(T,U) template - struct __not_overloaded<_Tp, _Up> : __not_overloaded2<_Tp, _Up> { }; + struct __not_overloaded<_Tp, _Up, __void_t< + decltype(operator>=(std::declval<_Tp>(), std::declval<_Up>()))>> + : false_type { }; template using __ptr_cmp = __and_<__not_overloaded<_Tp, _Up>, @@ -712,15 +709,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION static_cast(std::forward<_Up>(__u))); } - template - struct __not_overloaded; - - // False if we can call operator<=(T,U) - template - struct __not_overloaded<_Tp, _Up, __void_t< - decltype(operator<=(std::declval<_Tp>(), std::declval<_Up>()))>> - : false_type { }; - + // True if there is no viable operator<= member function. template struct __not_overloaded2 : true_type { }; @@ -730,8 +719,15 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION decltype(std::declval<_Tp>().operator<=(std::declval<_Up>()))>> : false_type { }; + // True if there is no overloaded operator<= for these operands. + template + struct __not_overloaded : __not_overloaded2<_Tp, _Up> { }; + + // False if we can call operator<=(T,U) template - struct __not_overloaded<_Tp, _Up> : __not_overloaded2<_Tp, _Up> { }; + struct __not_overloaded<_Tp, _Up, __void_t< + decltype(operator<=(std::declval<_Tp>(), std::declval<_Up>()))>> + : false_type { }; template using __ptr_cmp = __and_<__not_overloaded<_Tp, _Up>, diff --git a/libstdc++-v3/testsuite/20_util/function_objects/comparisons_pointer.cc b/libstdc++-v3/testsuite/20_util/function_objects/comparisons_pointer.cc index 474190cdf81..7cec294f8c2 100644 --- a/libstdc++-v3/testsuite/20_util/function_objects/comparisons_pointer.cc +++ b/libstdc++-v3/testsuite/20_util/function_objects/comparisons_pointer.cc @@ -195,6 +195,39 @@ test05() #endif } +struct Overloaded { + bool operator>(int) { return true; } + bool operator<(int) { return false; } + bool operator>=(int) { return true; } + bool operator<=(int) { return false; } +}; +bool operator>(Overloaded, Overloaded) { return false; } +bool operator<(Overloaded, Overloaded) { return false; } +bool operator>=(Overloaded, Overloaded) { return true; } +bool operator<=(Overloaded, Overloaded) { return true; } + +void +test06() +{ +#if __cplusplus >= 201402L + std::greater gt; + std::less lt; + std::greater_equal ge; + std::less_equal le; + + Overloaded o; + VERIFY( !gt(o, o) ); + VERIFY( !lt(o, o) ); + VERIFY( ge(o, o) ); + VERIFY( le(o, o) ); + + VERIFY( gt(o, 1) ); + VERIFY( !lt(o, 1) ); + VERIFY( ge(o, 1) ); + VERIFY( !le(o, 1) ); +#endif +} + int main() { @@ -203,4 +236,5 @@ main() test03(); test04(); test05(); + test06(); }