Path: blob/main/contrib/llvm-project/libcxx/include/__math/hypot.h
35233 views
//===----------------------------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#ifndef _LIBCPP___MATH_HYPOT_H9#define _LIBCPP___MATH_HYPOT_H1011#include <__algorithm/max.h>12#include <__config>13#include <__math/abs.h>14#include <__math/exponential_functions.h>15#include <__math/roots.h>16#include <__type_traits/enable_if.h>17#include <__type_traits/is_arithmetic.h>18#include <__type_traits/is_same.h>19#include <__type_traits/promote.h>20#include <__utility/pair.h>21#include <limits>2223#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)24# pragma GCC system_header25#endif2627_LIBCPP_PUSH_MACROS28#include <__undef_macros>2930_LIBCPP_BEGIN_NAMESPACE_STD3132namespace __math {3334inline _LIBCPP_HIDE_FROM_ABI float hypot(float __x, float __y) _NOEXCEPT { return __builtin_hypotf(__x, __y); }3536template <class = int>37_LIBCPP_HIDE_FROM_ABI double hypot(double __x, double __y) _NOEXCEPT {38return __builtin_hypot(__x, __y);39}4041inline _LIBCPP_HIDE_FROM_ABI long double hypot(long double __x, long double __y) _NOEXCEPT {42return __builtin_hypotl(__x, __y);43}4445template <class _A1, class _A2, __enable_if_t<is_arithmetic<_A1>::value && is_arithmetic<_A2>::value, int> = 0>46inline _LIBCPP_HIDE_FROM_ABI typename __promote<_A1, _A2>::type hypot(_A1 __x, _A2 __y) _NOEXCEPT {47using __result_type = typename __promote<_A1, _A2>::type;48static_assert(!(_IsSame<_A1, __result_type>::value && _IsSame<_A2, __result_type>::value), "");49return __math::hypot((__result_type)__x, (__result_type)__y);50}5152#if _LIBCPP_STD_VER >= 1753// Computes the three-dimensional hypotenuse: `std::hypot(x,y,z)`.54// The naive implementation might over-/underflow which is why this implementation is more involved:55// If the square of an argument might run into issues, we scale the arguments appropriately.56// See https://github.com/llvm/llvm-project/issues/92782 for a detailed discussion and summary.57template <class _Real>58_LIBCPP_HIDE_FROM_ABI _Real __hypot(_Real __x, _Real __y, _Real __z) {59// Factors needed to determine if over-/underflow might happen60constexpr int __exp = std::numeric_limits<_Real>::max_exponent / 2;61const _Real __overflow_threshold = __math::ldexp(_Real(1), __exp);62const _Real __overflow_scale = __math::ldexp(_Real(1), -(__exp + 20));6364// Scale arguments depending on their size65const _Real __max_abs = std::max(__math::fabs(__x), std::max(__math::fabs(__y), __math::fabs(__z)));66_Real __scale;67if (__max_abs > __overflow_threshold) { // x*x + y*y + z*z might overflow68__scale = __overflow_scale;69} else if (__max_abs < 1 / __overflow_threshold) { // x*x + y*y + z*z might underflow70__scale = 1 / __overflow_scale;71} else {72__scale = 1;73}74__x *= __scale;75__y *= __scale;76__z *= __scale;7778// Compute hypot of scaled arguments and undo scaling79return __math::sqrt(__x * __x + __y * __y + __z * __z) / __scale;80}8182inline _LIBCPP_HIDE_FROM_ABI float hypot(float __x, float __y, float __z) { return __math::__hypot(__x, __y, __z); }8384inline _LIBCPP_HIDE_FROM_ABI double hypot(double __x, double __y, double __z) { return __math::__hypot(__x, __y, __z); }8586inline _LIBCPP_HIDE_FROM_ABI long double hypot(long double __x, long double __y, long double __z) {87return __math::__hypot(__x, __y, __z);88}8990template <class _A1,91class _A2,92class _A3,93std::enable_if_t< is_arithmetic_v<_A1> && is_arithmetic_v<_A2> && is_arithmetic_v<_A3>, int> = 0 >94_LIBCPP_HIDE_FROM_ABI typename __promote<_A1, _A2, _A3>::type hypot(_A1 __x, _A2 __y, _A3 __z) _NOEXCEPT {95using __result_type = typename __promote<_A1, _A2, _A3>::type;96static_assert(!(97std::is_same_v<_A1, __result_type> && std::is_same_v<_A2, __result_type> && std::is_same_v<_A3, __result_type>));98return __math::__hypot(99static_cast<__result_type>(__x), static_cast<__result_type>(__y), static_cast<__result_type>(__z));100}101#endif102103} // namespace __math104105_LIBCPP_END_NAMESPACE_STD106_LIBCPP_POP_MACROS107108#endif // _LIBCPP___MATH_HYPOT_H109110111