// Copyright (C) 2023 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QCOMPARE_H #error "Do not include qcomparehelpers.h directly. Use qcompare.h instead." #endif #ifndef QCOMPAREHELPERS_H #define QCOMPAREHELPERS_H #if 0 #pragma qt_no_master_include #pragma qt_sync_skip_header_check #pragma qt_sync_stop_processing #endif #include #include #include #include #ifdef __cpp_lib_three_way_comparison #include #endif #include #include // std::less, std::hash QT_BEGIN_NAMESPACE class QPartialOrdering; namespace QtOrderingPrivate { #ifdef __cpp_lib_three_way_comparison template struct StdOrdering; template struct QtOrdering; #define QT_STD_MAP(x) \ template <> struct StdOrdering< Qt::x##_ordering> : q20::type_identity {};\ template <> struct StdOrdering : q20::type_identity {};\ template <> struct QtOrdering : q20::type_identity< Qt::x##_ordering> {};\ template <> struct QtOrdering< Qt::x##_ordering> : q20::type_identity< Qt::x##_ordering> {};\ /* end */ QT_STD_MAP(partial) QT_STD_MAP(weak) QT_STD_MAP(strong) #undef QT_STD_MAP template <> struct StdOrdering : q20::type_identity {}; template <> struct QtOrdering : q20::type_identity< Qt::partial_ordering> {}; template constexpr auto to_std(In in) noexcept -> typename QtOrderingPrivate::StdOrdering::type { return in; } template constexpr auto to_Qt(In in) noexcept -> typename QtOrderingPrivate::QtOrdering::type { return in; } #endif // __cpp_lib_three_way_comparison } // namespace QtOrderingPrivate /* For all the macros these parameter names are used: * LeftType - the type of the left operand of the comparison * RightType - the type of the right operand of the comparison * Constexpr - must be either constexpr or empty. Defines whether the operator is constexpr or not * Noexcept - a noexcept specifier. By default the relational operators are expected to be noexcept. However, there are some cases when this cannot be achieved (e.g. QDir). The public macros will pass noexcept(true) or noexcept(false) in this parameter, because conditional noexcept is known to cause some issues. However, internally we might want to pass a predicate here for some specific classes (e.g. QList, etc). * Attributes - an optional list of attributes. For example, pass \c QT_ASCII_CAST_WARN when defining comparisons between C-style string and an encoding-aware string type. The macros require two helper functions. For operators to be constexpr, these must be constexpr, too. Additionally, other attributes (like Q__EXPORT, Q_DECL_CONST_FUNCTION, etc) can be applied to them. Aside from that, their declaration should match: bool comparesEqual(LeftType, RightType) noexcept; ReturnType compareThreeWay(LeftType, RightType) noexcept; The ReturnType can be one of Qt::{partial,weak,strong}_ordering. The actual type depends on the macro being used. It makes sense to define the helper functions as hidden friends of the class, so that they could be found via ADL, and don't participate in unintended implicit conversions. */ /* Some systems (e.g. QNX and Integrity (GHS compiler)) have bugs in handling conditional noexcept in lambdas. This macro is needed to overcome such bugs and provide a noexcept check only on platforms that behave normally. It does nothing on the systems that have problems. */ #if !defined(Q_OS_QNX) && !defined(Q_CC_GHS) # define QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, Func) \ constexpr auto f = []() Noexcept {}; \ static_assert(!noexcept(f()) || noexcept(Func(lhs, rhs)), \ "Use *_NON_NOEXCEPT version of the macro, " \ "or make the helper function noexcept") #else # define QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, Func) /* no check */ #endif // Seems that qdoc uses C++20 even when Qt is compiled in C++17 mode. // Or at least it defines __cpp_lib_three_way_comparison. // Let qdoc see only the C++17 operators for now, because that's what our docs // currently describe. #if defined(__cpp_lib_three_way_comparison) && !defined(Q_QDOC) // C++20 - provide operator==() for equality, and operator<=>() for ordering #define QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator==(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, comparesEqual); \ return comparesEqual(lhs, rhs); \ } #define QT_DECLARE_3WAY_HELPER_STRONG(LeftType, RightType, Constexpr, Noexcept, Attributes) \ Attributes \ friend Constexpr std::strong_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, compareThreeWay); \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_3WAY_HELPER_WEAK(LeftType, RightType, Constexpr, Noexcept, Attributes) \ Attributes \ friend Constexpr std::weak_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, compareThreeWay); \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_3WAY_HELPER_PARTIAL(LeftType, RightType, Constexpr, Noexcept, Attributes) \ Attributes \ friend Constexpr std::partial_ordering \ operator<=>(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, compareThreeWay); \ return compareThreeWay(lhs, rhs); \ } #define QT_DECLARE_ORDERING_OPERATORS_HELPER(OrderingType, LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Noexcept, Attributes) \ QT_DECLARE_3WAY_HELPER_ ## OrderingType (LeftType, RightType, Constexpr, Noexcept, Attributes) #ifdef Q_COMPILER_LACKS_THREE_WAY_COMPARE_SYMMETRY // define reversed versions of the operators manually, because buggy MSVC versions do not do it #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator==(RightType const &lhs, LeftType const &rhs) Noexcept \ { return comparesEqual(rhs, lhs); } #define QT_DECLARE_REVERSED_3WAY_HELPER_STRONG(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr std::strong_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) Noexcept \ { \ const auto r = compareThreeWay(rhs, lhs); \ return QtOrderingPrivate::reversed(r); \ } #define QT_DECLARE_REVERSED_3WAY_HELPER_WEAK(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr std::weak_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) Noexcept \ { \ const auto r = compareThreeWay(rhs, lhs); \ return QtOrderingPrivate::reversed(r); \ } #define QT_DECLARE_REVERSED_3WAY_HELPER_PARTIAL(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr std::partial_ordering \ operator<=>(RightType const &lhs, LeftType const &rhs) Noexcept \ { \ const auto r = compareThreeWay(rhs, lhs); \ return QtOrderingPrivate::reversed(r); \ } #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Noexcept, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ QT_DECLARE_REVERSED_3WAY_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, \ Noexcept, Attributes) #else // dummy macros for C++17 compatibility, reversed operators are generated by the compiler #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Noexcept, Attributes) #endif // Q_COMPILER_LACKS_THREE_WAY_COMPARE_SYMMETRY #else // C++17 - provide operator==() and operator!=() for equality, // and all 4 comparison operators for ordering #define QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator==(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, comparesEqual); \ return comparesEqual(lhs, rhs); \ } \ Attributes \ friend Constexpr bool operator!=(LeftType const &lhs, RightType const &rhs) Noexcept \ { return !comparesEqual(lhs, rhs); } // Helpers for reversed comparison, using the existing comparesEqual() function. #define QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator==(RightType const &lhs, LeftType const &rhs) Noexcept \ { return comparesEqual(rhs, lhs); } \ Attributes \ friend Constexpr bool operator!=(RightType const &lhs, LeftType const &rhs) Noexcept \ { return !comparesEqual(rhs, lhs); } #define QT_DECLARE_ORDERING_HELPER_TEMPLATE(OrderingType, LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator<(LeftType const &lhs, RightType const &rhs) Noexcept \ { \ QT_COMPARISON_NOEXCEPT_CHECK(Noexcept, compareThreeWay); \ return is_lt(compareThreeWay(lhs, rhs)); \ } \ Attributes \ friend Constexpr bool operator>(LeftType const &lhs, RightType const &rhs) Noexcept \ { return is_gt(compareThreeWay(lhs, rhs)); } \ Attributes \ friend Constexpr bool operator<=(LeftType const &lhs, RightType const &rhs) Noexcept \ { return is_lteq(compareThreeWay(lhs, rhs)); } \ Attributes \ friend Constexpr bool operator>=(LeftType const &lhs, RightType const &rhs) Noexcept \ { return is_gteq(compareThreeWay(lhs, rhs)); } #define QT_DECLARE_ORDERING_HELPER_PARTIAL(LeftType, RightType, Constexpr, Noexcept, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::partial_ordering, LeftType, RightType, Constexpr, \ Noexcept, Attributes) #define QT_DECLARE_ORDERING_HELPER_WEAK(LeftType, RightType, Constexpr, Noexcept, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::weak_ordering, LeftType, RightType, Constexpr, \ Noexcept, Attributes) #define QT_DECLARE_ORDERING_HELPER_STRONG(LeftType, RightType, Constexpr, Noexcept, Attributes) \ QT_DECLARE_ORDERING_HELPER_TEMPLATE(Qt::strong_ordering, LeftType, RightType, Constexpr, \ Noexcept, Attributes) #define QT_DECLARE_ORDERING_OPERATORS_HELPER(OrderingString, LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, Constexpr, Noexcept, Attributes) \ QT_DECLARE_ORDERING_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, Noexcept, \ Attributes) // Helpers for reversed ordering, using the existing compareThreeWay() function. #define QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(OrderingType, LeftType, RightType, Constexpr, \ Noexcept, Attributes) \ Attributes \ friend Constexpr bool operator<(RightType const &lhs, LeftType const &rhs) Noexcept \ { return is_gt(compareThreeWay(rhs, lhs)); } \ Attributes \ friend Constexpr bool operator>(RightType const &lhs, LeftType const &rhs) Noexcept \ { return is_lt(compareThreeWay(rhs, lhs)); } \ Attributes \ friend Constexpr bool operator<=(RightType const &lhs, LeftType const &rhs) Noexcept \ { return is_gteq(compareThreeWay(rhs, lhs)); } \ Attributes \ friend Constexpr bool operator>=(RightType const &lhs, LeftType const &rhs) Noexcept \ { return is_lteq(compareThreeWay(rhs, lhs)); } #define QT_DECLARE_REVERSED_ORDERING_HELPER_PARTIAL(LeftType, RightType, Constexpr, Noexcept, \ Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::partial_ordering, LeftType, RightType, \ Constexpr, Noexcept, Attributes) #define QT_DECLARE_REVERSED_ORDERING_HELPER_WEAK(LeftType, RightType, Constexpr, Noexcept, \ Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::weak_ordering, LeftType, RightType, \ Constexpr, Noexcept, Attributes) #define QT_DECLARE_REVERSED_ORDERING_HELPER_STRONG(LeftType, RightType, Constexpr, Noexcept, \ Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_TEMPLATE(Qt::strong_ordering, LeftType, RightType, \ Constexpr, Noexcept, Attributes) #define QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(OrderingString, LeftType, RightType, \ Constexpr, Noexcept, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, Constexpr, Noexcept, \ Attributes) \ QT_DECLARE_REVERSED_ORDERING_HELPER_ ## OrderingString (LeftType, RightType, Constexpr, \ Noexcept, Attributes) #endif // __cpp_lib_three_way_comparison /* Public API starts here */ // Equality operators #define QT_DECLARE_EQUALITY_COMPARABLE_1(Type) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(Type, Type, /* non-constexpr */, noexcept(true), \ /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_2(LeftType, RightType) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_3(LeftType, RightType, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) #define Q_DECLARE_EQUALITY_COMPARABLE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_EQUALITY_COMPARABLE, __VA_ARGS__) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_1(Type) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(Type, Type, constexpr, noexcept(true), \ /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, constexpr, noexcept(true), \ /* no attributes */) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, constexpr, noexcept(true), \ Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, constexpr, noexcept(true), \ Attributes) #define Q_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE, __VA_ARGS__) #define QT_DECLARE_EQUALITY_COMPARABLE_NON_NOEXCEPT_1(Type) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(Type, Type, /* non-constexpr */, noexcept(false), \ /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_NON_NOEXCEPT_2(LeftType, RightType) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) #define QT_DECLARE_EQUALITY_COMPARABLE_NON_NOEXCEPT_3(LeftType, RightType, Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) \ QT_DECLARE_EQUALITY_OPERATORS_REVERSED_HELPER(LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) #define Q_DECLARE_EQUALITY_COMPARABLE_NON_NOEXCEPT(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_EQUALITY_COMPARABLE_NON_NOEXCEPT, __VA_ARGS__) // Partial ordering operators #define QT_DECLARE_PARTIALLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, Type, Type, /* non-constexpr */, \ noexcept(true), /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, noexcept(true), \ /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, noexcept(true), Attributes) #define Q_DECLARE_PARTIALLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_PARTIALLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, Type, Type, constexpr, noexcept(true), \ /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, constexpr, noexcept(true), \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, constexpr, \ noexcept(true), Attributes) #define Q_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_PARTIALLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) #define QT_DECLARE_PARTIALLY_ORDERED_NON_NOEXCEPT_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, Type, Type, /* non-constexpr */, \ noexcept(false), /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_NON_NOEXCEPT_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, noexcept(false), \ /* no attributes */) #define QT_DECLARE_PARTIALLY_ORDERED_NON_NOEXCEPT_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(PARTIAL, LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(PARTIAL, LeftType, RightType, \ /* non-constexpr */, noexcept(false), Attributes) #define Q_DECLARE_PARTIALLY_ORDERED_NON_NOEXCEPT(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_PARTIALLY_ORDERED_NON_NOEXCEPT, __VA_ARGS__) // Weak ordering operators #define QT_DECLARE_WEAKLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, Type, Type, /* non-constexpr */, noexcept(true), \ /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) #define Q_DECLARE_WEAKLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_WEAKLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, Type, Type, constexpr, noexcept(true), \ /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, constexpr, noexcept(true), \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, constexpr, \ noexcept(true), Attributes) #define Q_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_WEAKLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) #define QT_DECLARE_WEAKLY_ORDERED_NON_NOEXCEPT_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, Type, Type, /* non-constexpr */, noexcept(false), \ /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_NON_NOEXCEPT_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) #define QT_DECLARE_WEAKLY_ORDERED_NON_NOEXCEPT_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(WEAK, LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) #define Q_DECLARE_WEAKLY_ORDERED_NON_NOEXCEPT(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_WEAKLY_ORDERED_NON_NOEXCEPT, __VA_ARGS__) // Strong ordering operators #define QT_DECLARE_STRONGLY_ORDERED_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, Type, Type, /* non-constexpr */, \ noexcept(true), /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, noexcept(true), \ /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ noexcept(true), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, noexcept(true), Attributes) #define Q_DECLARE_STRONGLY_ORDERED(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_STRONGLY_ORDERED, __VA_ARGS__) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, Type, Type, constexpr, noexcept(true), \ /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, constexpr, \ noexcept(true), /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, constexpr, noexcept(true), \ Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, constexpr, \ noexcept(true), Attributes) #define Q_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE, __VA_ARGS__) #define QT_DECLARE_STRONGLY_ORDERED_NON_NOEXCEPT_1(Type) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, Type, Type, /* non-constexpr */, \ noexcept(false), /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_NON_NOEXCEPT_2(LeftType, RightType) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ noexcept(false), /* no attributes */) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, noexcept(false), \ /* no attributes */) #define QT_DECLARE_STRONGLY_ORDERED_NON_NOEXCEPT_3(LeftType, RightType, Attributes) \ QT_DECLARE_ORDERING_OPERATORS_HELPER(STRONG, LeftType, RightType, /* non-constexpr */, \ noexcept(false), Attributes) \ QT_DECLARE_ORDERING_OPERATORS_REVERSED_HELPER(STRONG, LeftType, RightType, \ /* non-constexpr */, noexcept(false), Attributes) #define Q_DECLARE_STRONGLY_ORDERED_NON_NOEXCEPT(...) \ QT_OVERLOADED_MACRO(QT_DECLARE_STRONGLY_ORDERED_NON_NOEXCEPT, __VA_ARGS__) namespace QtPrivate { template constexpr bool IsIntegralType_v = std::numeric_limits>::is_specialized && std::numeric_limits>::is_integer; template constexpr bool IsFloatType_v = std::is_floating_point_v; #if QFLOAT16_IS_NATIVE template <> inline constexpr bool IsFloatType_v = true; #endif } // namespace QtPrivate namespace QtOrderingPrivate { template constexpr Qt::strong_ordering strongOrderingCompareDefaultImpl(T lhs, U rhs) noexcept { #ifdef __cpp_lib_three_way_comparison return lhs <=> rhs; #else if (lhs == rhs) return Qt::strong_ordering::equivalent; else if (lhs < rhs) return Qt::strong_ordering::less; else return Qt::strong_ordering::greater; #endif // __cpp_lib_three_way_comparison } } // namespace QtOrderingPrivate namespace Qt { template using if_integral = std::enable_if_t, bool>; template using if_floating_point = std::enable_if_t, bool>; template using if_compatible_pointers = std::enable_if_t, std::is_base_of, std::is_base_of>, bool>; template using if_enum = std::enable_if_t, bool>; template = true, if_integral = true> constexpr Qt::strong_ordering compareThreeWay(LeftInt lhs, RightInt rhs) noexcept { static_assert(std::is_signed_v == std::is_signed_v, "Qt::compareThreeWay() does not allow mixed-sign comparison."); #ifdef __cpp_lib_three_way_comparison return lhs <=> rhs; #else if (lhs == rhs) return Qt::strong_ordering::equivalent; else if (lhs < rhs) return Qt::strong_ordering::less; else return Qt::strong_ordering::greater; #endif // __cpp_lib_three_way_comparison } template = true, if_floating_point = true> constexpr Qt::partial_ordering compareThreeWay(LeftFloat lhs, RightFloat rhs) noexcept { QT_WARNING_PUSH QT_WARNING_DISABLE_FLOAT_COMPARE #ifdef __cpp_lib_three_way_comparison return lhs <=> rhs; #else if (lhs < rhs) return Qt::partial_ordering::less; else if (lhs > rhs) return Qt::partial_ordering::greater; else if (lhs == rhs) return Qt::partial_ordering::equivalent; else return Qt::partial_ordering::unordered; #endif // __cpp_lib_three_way_comparison QT_WARNING_POP } template = true, if_floating_point = true> constexpr Qt::partial_ordering compareThreeWay(IntType lhs, FloatType rhs) noexcept { return compareThreeWay(FloatType(lhs), rhs); } template = true, if_integral = true> constexpr Qt::partial_ordering compareThreeWay(FloatType lhs, IntType rhs) noexcept { return compareThreeWay(lhs, FloatType(rhs)); } #if QT_DEPRECATED_SINCE(6, 8) template = true> QT_DEPRECATED_VERSION_X_6_8("Wrap the pointers into Qt::totally_ordered_wrapper and use the respective overload instead.") constexpr Qt::strong_ordering compareThreeWay(const LeftType *lhs, const RightType *rhs) noexcept { #ifdef __cpp_lib_three_way_comparison return std::compare_three_way{}(lhs, rhs); #else if (lhs == rhs) return Qt::strong_ordering::equivalent; else if (std::less<>{}(lhs, rhs)) return Qt::strong_ordering::less; else return Qt::strong_ordering::greater; #endif // __cpp_lib_three_way_comparison } template QT_DEPRECATED_VERSION_X_6_8("Wrap the pointer into Qt::totally_ordered_wrapper and use the respective overload instead.") constexpr Qt::strong_ordering compareThreeWay(const T *lhs, std::nullptr_t rhs) noexcept { return compareThreeWay(lhs, static_cast(rhs)); } template QT_DEPRECATED_VERSION_X_6_8("Wrap the pointer into Qt::totally_ordered_wrapper and use the respective overload instead.") constexpr Qt::strong_ordering compareThreeWay(std::nullptr_t lhs, const T *rhs) noexcept { return compareThreeWay(static_cast(lhs), rhs); } #endif // QT_DEPRECATED_SINCE(6, 8) template = true> constexpr Qt::strong_ordering compareThreeWay(Enum lhs, Enum rhs) noexcept { return compareThreeWay(qToUnderlying(lhs), qToUnderlying(rhs)); } } // namespace Qt namespace QtOrderingPrivate { template constexpr std::tuple qt_tuple_pop_front_impl(const std::tuple &t, std::index_sequence) noexcept { return std::tuple(std::get(t)...); } template constexpr std::tuple qt_tuple_pop_front(const std::tuple &t) noexcept { return qt_tuple_pop_front_impl(t, std::index_sequence_for{}); } template constexpr auto compareThreeWayMulti(const std::tuple &lhs, // ie. not empty const std::tuple &rhs) noexcept { static_assert(sizeof...(LhsTail) == sizeof...(RhsTail), // expanded together below, but provide a nicer error message: "The tuple arguments have to have the same size."); using Qt::compareThreeWay; using R = std::common_type_t< decltype(compareThreeWay(std::declval(), std::declval())), decltype(compareThreeWay(std::declval(), std::declval()))... >; const auto &l = std::get<0>(lhs); const auto &r = std::get<0>(rhs); static_assert(noexcept(compareThreeWay(l, r)), "This function requires all relational operators to be noexcept."); const auto res = compareThreeWay(l, r); if constexpr (sizeof...(LhsTail) > 0) { if (is_eq(res)) return R{compareThreeWayMulti(qt_tuple_pop_front(lhs), qt_tuple_pop_front(rhs))}; } return R{res}; } } //QtOrderingPrivate namespace Qt { // A wrapper class that adapts the wrappee to use the strongly-ordered // function objects for implementing the relational operators. // Mostly useful to avoid UB on pointers (which it currently mandates P to be), // because all the comparison helpers (incl. std::compare_three_way on // std::tuple!) will use the language-level operators. // template class totally_ordered_wrapper { static_assert(std::is_pointer_v

); using T = std::remove_pointer_t

; P ptr; public: totally_ordered_wrapper() noexcept = default; Q_IMPLICIT constexpr totally_ordered_wrapper(std::nullptr_t) // requires std::is_pointer_v

: totally_ordered_wrapper(P{nullptr}) {} explicit constexpr totally_ordered_wrapper(P p) noexcept : ptr(p) {} constexpr P get() const noexcept { return ptr; } constexpr void reset(P p) noexcept { ptr = p; } constexpr P operator->() const noexcept { return get(); } template , bool> = true> constexpr U &operator*() const noexcept { return *get(); } explicit constexpr operator bool() const noexcept { return get(); } private: // TODO: Replace the constraints with std::common_type_t when // a bug in VxWorks is fixed! template using if_compatible_types = std::enable_if_t, std::is_pointer, std::disjunction, std::is_convertible>>, bool>; #define MAKE_RELOP(Ret, op, Op) \ template = true> \ friend constexpr Ret operator op (const totally_ordered_wrapper

&lhs, const totally_ordered_wrapper &rhs) noexcept \ { return std:: Op {}(lhs.ptr, rhs.get()); } \ template = true> \ friend constexpr Ret operator op (const totally_ordered_wrapper

&lhs, const U &rhs) noexcept \ { return std:: Op {}(lhs.ptr, rhs ); } \ template = true> \ friend constexpr Ret operator op (const U &lhs, const totally_ordered_wrapper

&rhs) noexcept \ { return std:: Op {}(lhs, rhs.ptr); } \ friend constexpr Ret operator op (const totally_ordered_wrapper &lhs, std::nullptr_t) noexcept \ { return std:: Op {}(lhs.ptr, P(nullptr)); } \ friend constexpr Ret operator op (std::nullptr_t, const totally_ordered_wrapper &rhs) noexcept \ { return std:: Op {}(P(nullptr), rhs.ptr); } \ /* end */ MAKE_RELOP(bool, ==, equal_to<>) MAKE_RELOP(bool, !=, not_equal_to<>) MAKE_RELOP(bool, < , less<>) MAKE_RELOP(bool, <=, less_equal<>) MAKE_RELOP(bool, > , greater<>) MAKE_RELOP(bool, >=, greater_equal<>) #ifdef __cpp_lib_three_way_comparison MAKE_RELOP(auto, <=>, compare_three_way) #endif #undef MAKE_RELOP friend void qt_ptr_swap(totally_ordered_wrapper &lhs, totally_ordered_wrapper &rhs) noexcept { qt_ptr_swap(lhs.ptr, rhs.ptr); } friend void swap(totally_ordered_wrapper &lhs, totally_ordered_wrapper &rhs) noexcept { qt_ptr_swap(lhs, rhs); } friend size_t qHash(totally_ordered_wrapper key, size_t seed = 0) noexcept { return qHash(key.ptr, seed); } }; template = true> constexpr Qt::strong_ordering compareThreeWay(Qt::totally_ordered_wrapper lhs, Qt::totally_ordered_wrapper rhs) noexcept { return QtOrderingPrivate::strongOrderingCompareDefaultImpl(lhs, rhs); } template = true> constexpr Qt::strong_ordering compareThreeWay(Qt::totally_ordered_wrapper lhs, U *rhs) noexcept { return QtOrderingPrivate::strongOrderingCompareDefaultImpl(lhs, rhs); } template = true> constexpr Qt::strong_ordering compareThreeWay(U *lhs, Qt::totally_ordered_wrapper rhs) noexcept { return QtOrderingPrivate::strongOrderingCompareDefaultImpl(lhs, rhs); } template constexpr Qt::strong_ordering compareThreeWay(Qt::totally_ordered_wrapper lhs, std::nullptr_t rhs) noexcept { return QtOrderingPrivate::strongOrderingCompareDefaultImpl(lhs, rhs); } template constexpr Qt::strong_ordering compareThreeWay(std::nullptr_t lhs, Qt::totally_ordered_wrapper rhs) noexcept { return QtOrderingPrivate::strongOrderingCompareDefaultImpl(lhs, rhs); } } //Qt template class QTypeInfo> : public QTypeInfo

{}; namespace QtOrderingPrivate { namespace CompareThreeWayTester { using Qt::compareThreeWay; // Check if compareThreeWay is implemented for the (LT, RT) argument // pair. template constexpr inline bool hasCompareThreeWay = false; template constexpr inline bool hasCompareThreeWay< LT, RT, std::void_t(), std::declval()))> > = true; // Check if the operation is noexcept. We have two different overloads, // depending on the available compareThreeWay() implementation. // Both are declared, but not implemented. To be used only in unevaluated // context. template , bool> = true> constexpr bool compareThreeWayNoexcept() noexcept { return noexcept(compareThreeWay(std::declval(), std::declval())); } template && hasCompareThreeWay, bool> = true> constexpr bool compareThreeWayNoexcept() noexcept { return noexcept(compareThreeWay(std::declval(), std::declval())); } } // namespace CompareThreeWayTester } // namespace QtOrderingPrivate QT_END_NAMESPACE namespace std { template struct hash> { using argument_type = QT_PREPEND_NAMESPACE(Qt::totally_ordered_wrapper)

; using result_type = size_t; constexpr result_type operator()(argument_type w) const noexcept { return std::hash

{}(w.get()); } }; } #endif // QCOMPAREHELPERS_H