Skip to content

Commit 03f8228

Browse files
committed
[libc++] std::cmp_less and other integer comparison functions could be improved with _LIBCPP_ASSUME
1 parent a6083a1 commit 03f8228

File tree

1 file changed

+13
-12
lines changed
  • libcxx/include/__utility

1 file changed

+13
-12
lines changed

libcxx/include/__utility/cmp.h

Lines changed: 13 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
#ifndef _LIBCPP___UTILITY_CMP_H
1010
#define _LIBCPP___UTILITY_CMP_H
1111

12+
#include "__assert"
1213
#include <__config>
1314
#include <__type_traits/integer_traits.h>
1415
#include <__type_traits/is_signed.h>
@@ -26,16 +27,18 @@ _LIBCPP_BEGIN_NAMESPACE_STD
2627

2728
#if _LIBCPP_STD_VER >= 20
2829

30+
template <__signed_or_unsigned_integer _Tp, __signed_integer _Ip>
31+
inline constexpr bool __comparison_can_promote_to =
32+
__signed_integer<_Tp> ? sizeof(_Tp) <= sizeof(_Ip) : sizeof(_Tp) < sizeof(_Ip);
33+
2934
template <__signed_or_unsigned_integer _Tp, __signed_or_unsigned_integer _Up>
3035
_LIBCPP_HIDE_FROM_ABI constexpr bool cmp_equal(_Tp __t, _Up __u) noexcept {
31-
if constexpr (sizeof(_Tp) < sizeof(int) && sizeof(_Up) < sizeof(int)) {
32-
__builtin_assume(__t < numeric_limits<int>::max() && __u < numeric_limits<int>::max());
36+
if constexpr (is_signed_v<_Tp> == is_signed_v<_Up>)
37+
return __t == __u;
38+
else if constexpr (__comparison_can_promote_to<_Tp, int> && __comparison_can_promote_to<_Up, int>)
3339
return static_cast<int>(__t) == static_cast<int>(__u);
34-
} else if constexpr (sizeof(_Tp) < sizeof(long long) && sizeof(_Up) < sizeof(long long)) {
35-
__builtin_assume(__t < numeric_limits<long long>::max() && __u < numeric_limits<long long>::max());
40+
else if constexpr (__comparison_can_promote_to<_Tp, long long> && __comparison_can_promote_to<_Up, long long>)
3641
return static_cast<long long>(__t) == static_cast<long long>(__u);
37-
} else if constexpr (is_signed_v<_Tp> == is_signed_v<_Up>)
38-
return __t == __u;
3942
else if constexpr (is_signed_v<_Tp>)
4043
return __t < 0 ? false : make_unsigned_t<_Tp>(__t) == __u;
4144
else
@@ -49,14 +52,12 @@ _LIBCPP_HIDE_FROM_ABI constexpr bool cmp_not_equal(_Tp __t, _Up __u) noexcept {
4952

5053
template <__signed_or_unsigned_integer _Tp, __signed_or_unsigned_integer _Up>
5154
_LIBCPP_HIDE_FROM_ABI constexpr bool cmp_less(_Tp __t, _Up __u) noexcept {
52-
if constexpr (sizeof(_Tp) < sizeof(int) && sizeof(_Up) < sizeof(int)) {
53-
__builtin_assume(__t < numeric_limits<int>::max() && __u < numeric_limits<int>::max());
55+
if constexpr (is_signed_v<_Tp> == is_signed_v<_Up>)
56+
return __t < __u;
57+
else if constexpr (__comparison_can_promote_to<_Tp, int> && __comparison_can_promote_to<_Up, int>)
5458
return static_cast<int>(__t) < static_cast<int>(__u);
55-
} else if constexpr (sizeof(_Tp) < sizeof(long long) && sizeof(_Up) < sizeof(long long)) {
56-
__builtin_assume(__t < numeric_limits<long long>::max() && __u < numeric_limits<long long>::max());
59+
else if constexpr (__comparison_can_promote_to<_Tp, long long> && __comparison_can_promote_to<_Up, long long>)
5760
return static_cast<long long>(__t) < static_cast<long long>(__u);
58-
} else if constexpr (is_signed_v<_Tp> == is_signed_v<_Up>)
59-
return __t < __u;
6061
else if constexpr (is_signed_v<_Tp>)
6162
return __t < 0 ? true : make_unsigned_t<_Tp>(__t) < __u;
6263
else

0 commit comments

Comments
 (0)