#ifndef SPROUT_MATH_POW_HPP #define SPROUT_MATH_POW_HPP #include #include #include #include #include #include #if SPROUT_USE_BUILTIN_CMATH_FUNCTION # include #endif namespace sprout { namespace math { namespace detail { template struct pow_promoted {}; template struct pow_promoted< X, Y, typename std::enable_if< std::is_arithmetic::value && std::is_arithmetic::value && (std::is_same::value || std::is_same::value) >::type > { public: typedef long double type; }; template struct pow_promoted< X, Y, typename std::enable_if< std::is_arithmetic::value && std::is_arithmetic::value && !(std::is_same::value || std::is_same::value) >::type > { public: typedef double type; }; } // namespace detail // // pow_promoted // template struct pow_promoted : public sprout::math::detail::pow_promoted< typename std::remove_cv::type, typename std::remove_cv::type > {}; namespace detail { template< typename FloatType, typename sprout::enabler_if::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR FloatType pow(FloatType x, FloatType y) { return sprout::math::detail::exp(y * sprout::math::detail::log(x)); } template< typename ArithmeticType1, typename ArithmeticType2, typename sprout::enabler_if< std::is_arithmetic::value && std::is_arithmetic::value >::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::math::pow_promoted::type pow(ArithmeticType1 x, ArithmeticType2 y) { typedef typename sprout::math::pow_promoted::type type; return sprout::math::detail::pow(static_cast(x), static_cast(y)); } } // namespace detail # if SPROUT_USE_BUILTIN_CMATH_FUNCTION using std::pow; # else using sprout::math::detail::pow; # endif } // namespace math using sprout::math::pow; } // namespace sprout #endif // #ifndef SPROUT_MATH_POW_HPP