mirror of
https://github.com/bolero-MURAKAMI/Sprout
synced 2024-11-12 21:09:01 +00:00
add carry_to_exp2
This commit is contained in:
parent
92c7169221
commit
0805756259
2 changed files with 76 additions and 0 deletions
|
@ -10,6 +10,7 @@
|
|||
|
||||
#include <sprout/config.hpp>
|
||||
#include <sprout/integer/integer_digits.hpp>
|
||||
#include <sprout/integer/carry_to_exp2.hpp>
|
||||
#include <sprout/integer/static_pow.hpp>
|
||||
|
||||
#endif // #ifndef SPROUT_INTEGER_HPP
|
||||
|
|
75
sprout/integer/carry_to_exp2.hpp
Normal file
75
sprout/integer/carry_to_exp2.hpp
Normal file
|
@ -0,0 +1,75 @@
|
|||
/*=============================================================================
|
||||
Copyright (c) 2011-2016 Bolero MURAKAMI
|
||||
https://github.com/bolero-MURAKAMI/Sprout
|
||||
|
||||
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 SPROUT_INTEGER_CARRY_TO_EXP2_HPP
|
||||
#define SPROUT_INTEGER_CARRY_TO_EXP2_HPP
|
||||
|
||||
#include <climits>
|
||||
#include <limits>
|
||||
#include <sprout/config.hpp>
|
||||
#include <sprout/bit/cntl0.hpp>
|
||||
#include <sprout/bit/popcount.hpp>
|
||||
#include <sprout/type_traits/is_sint.hpp>
|
||||
#include <sprout/type_traits/is_uint.hpp>
|
||||
#include <sprout/type_traits/enabler_if.hpp>
|
||||
|
||||
namespace sprout {
|
||||
namespace detail {
|
||||
template<typename IntType>
|
||||
inline SPROUT_CONSTEXPR IntType
|
||||
carry_to_exp2_simpl(IntType x) {
|
||||
return sprout::popcount(x) <= 1 ? x
|
||||
: IntType(1) << (sizeof(IntType) * CHAR_BIT - sprout::cntl0(x))
|
||||
;
|
||||
}
|
||||
template<typename UIntType>
|
||||
inline SPROUT_CONSTEXPR UIntType
|
||||
carry_to_exp2_uimpl(int n) {
|
||||
return !n ? UIntType(0)
|
||||
: UIntType(1) << (sizeof(UIntType) * CHAR_BIT - n)
|
||||
;
|
||||
}
|
||||
} // namespace detail
|
||||
//
|
||||
// carry_to_exp2
|
||||
//
|
||||
// note: for signed integer
|
||||
// x == MIN
|
||||
// --> carry_to_exp2(x) returns x .
|
||||
// x < 0
|
||||
// --> carry_to_exp2(x) returns -carry_to_exp2(-x) .
|
||||
// -(MIN / 2) < x
|
||||
// --> carry_to_exp2(x) returns MIN .
|
||||
//
|
||||
// note: for unsigned integer
|
||||
// MAX / 2 + 1 < x
|
||||
// --> carry_to_exp2(x) returns 0 .
|
||||
//
|
||||
template<
|
||||
typename IntType,
|
||||
typename sprout::enabler_if<sprout::is_sint<IntType>::value>::type = sprout::enabler
|
||||
>
|
||||
inline SPROUT_CONSTEXPR IntType
|
||||
carry_to_exp2(IntType x) {
|
||||
return x == std::numeric_limits<IntType>::min() ? x
|
||||
: x < IntType(0) ? -sprout::detail::carry_to_exp2_simpl(-x)
|
||||
: sprout::detail::carry_to_exp2_simpl(x)
|
||||
;
|
||||
}
|
||||
template<
|
||||
typename UIntType,
|
||||
typename sprout::enabler_if<sprout::is_uint<UIntType>::value>::type = sprout::enabler
|
||||
>
|
||||
inline SPROUT_CONSTEXPR UIntType
|
||||
carry_to_exp2(UIntType x) {
|
||||
return sprout::popcount(x) <= 1 ? x
|
||||
: sprout::detail::carry_to_exp2_uimpl<UIntType>(sprout::cntl0(x))
|
||||
;
|
||||
}
|
||||
} // namespace sprout
|
||||
|
||||
#endif // #ifndef SPROUT_INTEGER_CARRY_TO_EXP2_HPP
|
Loading…
Reference in a new issue