// Copyright 2019 Hans Dembinski // // Distributed under the Boost Software License, Version 1.0. // (See accompanying file LICENSE_1_0.txt // or copy at http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_HISTOGRAM_DETAIL_SAFE_COMPARISON_HPP #define BOOST_HISTOGRAM_DETAIL_SAFE_COMPARISON_HPP #include #include #include namespace boost { namespace histogram { namespace detail { template auto make_unsigned(const T& t) noexcept { static_assert(std::is_integral::value, ""); return static_cast>(t); } template using number_category = mp11::mp_if, mp11::mp_if, type, type>, type>; // version of std::equal_to<> which handles signed and unsigned integers correctly struct safe_equal { template bool operator()(const T& t, const U& u) const noexcept { return impl(number_category{}, number_category{}, t, u); } template bool impl(C1, C2, const T& t, const U& u) const noexcept { return t == u; } template bool impl(type, type, const T& t, const U& u) const noexcept { return t >= 0 && make_unsigned(t) == u; } template bool impl(type, type, const T& t, const U& u) const noexcept { return impl(type{}, type{}, u, t); } }; // version of std::less<> which handles signed and unsigned integers correctly struct safe_less { template bool operator()(const T& t, const U& u) const noexcept { return impl(number_category{}, number_category{}, t, u); } template bool impl(C1, C2, const T& t, const U& u) const noexcept { return t < u; } template bool impl(type, type, const T& t, const U& u) const noexcept { return t < 0 || make_unsigned(t) < u; } template bool impl(type, type, const T& t, const U& u) const noexcept { return 0 < u && t < make_unsigned(u); } }; // version of std::greater<> which handles signed and unsigned integers correctly struct safe_greater { template bool operator()(const T& t, const U& u) const noexcept { return safe_less()(u, t); } }; } // namespace detail } // namespace histogram } // namespace boost #endif