Libc++ currently redeclares ::lgamma_r on platforms that provide it. This causes issues when building with modules, and redeclaring functions provided by another library (here the C library) is bad hygiene. Instead, use an asm declaration to call the right function without having to redeclare it.
89 lines
2.6 KiB
C++
89 lines
2.6 KiB
C++
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef _LIBCPP___MATH_GAMMA_H
|
|
#define _LIBCPP___MATH_GAMMA_H
|
|
|
|
#include <__config>
|
|
#include <__type_traits/enable_if.h>
|
|
#include <__type_traits/is_integral.h>
|
|
|
|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
|
# pragma GCC system_header
|
|
#endif
|
|
|
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
|
|
|
namespace __math {
|
|
|
|
// lgamma
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI float lgamma(float __x) _NOEXCEPT { return __builtin_lgammaf(__x); }
|
|
|
|
template <class = int>
|
|
_LIBCPP_HIDE_FROM_ABI double lgamma(double __x) _NOEXCEPT {
|
|
return __builtin_lgamma(__x);
|
|
}
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI long double lgamma(long double __x) _NOEXCEPT { return __builtin_lgammal(__x); }
|
|
|
|
template <class _A1, __enable_if_t<is_integral<_A1>::value, int> = 0>
|
|
inline _LIBCPP_HIDE_FROM_ABI double lgamma(_A1 __x) _NOEXCEPT {
|
|
return __builtin_lgamma((double)__x);
|
|
}
|
|
|
|
// nan
|
|
|
|
// tgamma
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI float tgamma(float __x) _NOEXCEPT { return __builtin_tgammaf(__x); }
|
|
|
|
template <class = int>
|
|
_LIBCPP_HIDE_FROM_ABI double tgamma(double __x) _NOEXCEPT {
|
|
return __builtin_tgamma(__x);
|
|
}
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI long double tgamma(long double __x) _NOEXCEPT { return __builtin_tgammal(__x); }
|
|
|
|
template <class _A1, __enable_if_t<is_integral<_A1>::value, int> = 0>
|
|
inline _LIBCPP_HIDE_FROM_ABI double tgamma(_A1 __x) _NOEXCEPT {
|
|
return __builtin_tgamma((double)__x);
|
|
}
|
|
|
|
// __lgamma_r
|
|
//
|
|
// POSIX systems provide a function named lgamma_r which is a reentrant version of lgamma. Use that
|
|
// whenever possible. However, we avoid re-declaring the actual function since different platforms
|
|
// declare it differently in the first place: instead use `asm` to get the compiler to call the right
|
|
// function.
|
|
|
|
#if defined(_LIBCPP_MSVCRT_LIKE) // reentrant version is not available on Windows
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI double __lgamma_r(double __d) _NOEXCEPT { return __builtin_lgamma(__d); }
|
|
|
|
#else
|
|
|
|
# if defined(_LIBCPP_OBJECT_FORMAT_MACHO)
|
|
double __lgamma_r_shim(double, int*) _NOEXCEPT __asm__("_lgamma_r");
|
|
# else
|
|
double __lgamma_r_shim(double, int*) _NOEXCEPT __asm__("lgamma_r");
|
|
# endif
|
|
|
|
inline _LIBCPP_HIDE_FROM_ABI double __lgamma_r(double __d) _NOEXCEPT {
|
|
int __sign;
|
|
return __math::__lgamma_r_shim(__d, &__sign);
|
|
}
|
|
|
|
#endif
|
|
|
|
} // namespace __math
|
|
|
|
_LIBCPP_END_NAMESPACE_STD
|
|
|
|
#endif // _LIBCPP___MATH_GAMMA_H
|