16#ifndef EIGEN_ARCH_GENERIC_PACKET_MATH_FUNCTIONS_H
17#define EIGEN_ARCH_GENERIC_PACKET_MATH_FUNCTIONS_H
20#include "../../InternalHeaderCheck.h"
29struct make_integer<float> {
30 typedef numext::int32_t type;
33struct make_integer<double> {
34 typedef numext::int64_t type;
37struct make_integer<half> {
38 typedef numext::int16_t type;
41struct make_integer<bfloat16> {
42 typedef numext::int16_t type;
84template <
typename Packet,
int N>
86 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
87 const typename unpacket_traits<Packet>::type coeff[]) {
88 EIGEN_STATIC_ASSERT((N > 0), YOU_MADE_A_PROGRAMMING_MISTAKE);
89 return pmadd(ppolevl<Packet, N - 1>::run(x, coeff), x, pset1<Packet>(coeff[N]));
93template <
typename Packet>
94struct ppolevl<Packet, 0> {
95 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
96 const typename unpacket_traits<Packet>::type coeff[]) {
97 EIGEN_UNUSED_VARIABLE(x);
98 return pset1<Packet>(coeff[0]);
154template <
typename Packet,
int N>
156 EIGEN_DEVICE_FUNC
static EIGEN_STRONG_INLINE Packet run(Packet x,
157 const typename unpacket_traits<Packet>::type coef[]) {
158 typedef typename unpacket_traits<Packet>::type Scalar;
159 Packet b0 = pset1<Packet>(coef[0]);
160 Packet b1 = pset1<Packet>(
static_cast<Scalar
>(0.f));
163 for (
int i = 1; i < N; i++) {
166 b0 = psub(pmadd(x, b1, pset1<Packet>(coef[i])), b2);
169 return pmul(pset1<Packet>(
static_cast<Scalar
>(0.5f)), psub(b0, b2));
173template <
typename Packet>
174EIGEN_STRONG_INLINE EIGEN_DEVICE_FUNC Packet pfrexp_generic_get_biased_exponent(
const Packet& a) {
175 typedef typename unpacket_traits<Packet>::type Scalar;
176 typedef typename unpacket_traits<Packet>::integer_packet PacketI;
177 static constexpr int mantissa_bits = numext::numeric_limits<Scalar>::digits - 1;
178 return pcast<PacketI, Packet>(plogical_shift_right<mantissa_bits>(preinterpret<PacketI>(pabs(a))));
183template <
typename Packet>
184EIGEN_STRONG_INLINE EIGEN_DEVICE_FUNC Packet pfrexp_generic(
const Packet& a, Packet& exponent) {
185 typedef typename unpacket_traits<Packet>::type Scalar;
186 typedef typename make_unsigned<typename make_integer<Scalar>::type>::type ScalarUI;
187 static constexpr int TotalBits =
sizeof(Scalar) * CHAR_BIT, MantissaBits = numext::numeric_limits<Scalar>::digits - 1,
188 ExponentBits = TotalBits - MantissaBits - 1;
190 constexpr ScalarUI scalar_sign_mantissa_mask =
191 ~(((ScalarUI(1) << ExponentBits) - ScalarUI(1)) << MantissaBits);
192 const Packet sign_mantissa_mask = pset1frombits<Packet>(
static_cast<ScalarUI
>(scalar_sign_mantissa_mask));
193 const Packet half = pset1<Packet>(Scalar(0.5));
194 const Packet zero = pzero(a);
195 const Packet normal_min = pset1<Packet>((numext::numeric_limits<Scalar>::min)());
198 const Packet is_denormal = pcmp_lt(pabs(a), normal_min);
199 constexpr ScalarUI scalar_normalization_offset = ScalarUI(MantissaBits + 1);
201 const Scalar scalar_normalization_factor = Scalar(ScalarUI(1) <<
int(scalar_normalization_offset));
202 const Packet normalization_factor = pset1<Packet>(scalar_normalization_factor);
203 const Packet normalized_a = pselect(is_denormal, pmul(a, normalization_factor), a);
206 const Scalar scalar_exponent_offset = -Scalar((ScalarUI(1) << (ExponentBits - 1)) - ScalarUI(2));
207 Packet exponent_offset = pset1<Packet>(scalar_exponent_offset);
208 const Packet normalization_offset = pset1<Packet>(-Scalar(scalar_normalization_offset));
209 exponent_offset = pselect(is_denormal, padd(exponent_offset, normalization_offset), exponent_offset);
212 exponent = pfrexp_generic_get_biased_exponent(normalized_a);
215 const Scalar scalar_non_finite_exponent = Scalar((ScalarUI(1) << ExponentBits) - ScalarUI(1));
216 const Packet non_finite_exponent = pset1<Packet>(scalar_non_finite_exponent);
217 const Packet is_zero_or_not_finite = por(pcmp_eq(a, zero), pcmp_eq(exponent, non_finite_exponent));
218 const Packet m = pselect(is_zero_or_not_finite, a, por(pand(normalized_a, sign_mantissa_mask), half));
219 exponent = pselect(is_zero_or_not_finite, zero, padd(exponent, exponent_offset));
225template <
typename Packet>
226EIGEN_STRONG_INLINE EIGEN_DEVICE_FUNC Packet pldexp_generic(
const Packet& a,
const Packet& exponent) {
249 typedef typename unpacket_traits<Packet>::integer_packet PacketI;
250 typedef typename unpacket_traits<Packet>::type Scalar;
251 typedef typename unpacket_traits<PacketI>::type ScalarI;
252 static constexpr int TotalBits =
sizeof(Scalar) * CHAR_BIT, MantissaBits = numext::numeric_limits<Scalar>::digits - 1,
253 ExponentBits = TotalBits - MantissaBits - 1;
255 const Packet max_exponent = pset1<Packet>(Scalar((ScalarI(1) << ExponentBits) + ScalarI(MantissaBits - 1)));
256 const PacketI bias = pset1<PacketI>((ScalarI(1) << (ExponentBits - 1)) - ScalarI(1));
257 const PacketI e = pcast<Packet, PacketI>(pmin(pmax(exponent, pnegate(max_exponent)), max_exponent));
258 PacketI b = parithmetic_shift_right<2>(e);
259 Packet c = preinterpret<Packet>(plogical_shift_left<MantissaBits>(padd(b, bias)));
260 Packet out = pmul(pmul(pmul(a, c), c), c);
261 b = pnmadd(pset1<PacketI>(3), b, e);
262 c = preinterpret<Packet>(plogical_shift_left<MantissaBits>(padd(b, bias)));
276template <
typename Packet>
277EIGEN_STRONG_INLINE EIGEN_DEVICE_FUNC Packet pldexp_fast(
const Packet& a,
const Packet& exponent) {
278 typedef typename unpacket_traits<Packet>::integer_packet PacketI;
279 typedef typename unpacket_traits<Packet>::type Scalar;
280 typedef typename unpacket_traits<PacketI>::type ScalarI;
281 static constexpr int TotalBits =
sizeof(Scalar) * CHAR_BIT, MantissaBits = numext::numeric_limits<Scalar>::digits - 1,
282 ExponentBits = TotalBits - MantissaBits - 1;
284 const Packet bias = pset1<Packet>(Scalar((ScalarI(1) << (ExponentBits - 1)) - ScalarI(1)));
285 const Packet limit = pset1<Packet>(Scalar((ScalarI(1) << ExponentBits) - ScalarI(1)));
287 const PacketI e = pcast<Packet, PacketI>(pmin(pmax(padd(exponent, bias), pzero(limit)), limit));
289 return pmul(a, preinterpret<Packet>(plogical_shift_left<MantissaBits>(e)));
295template <
typename Packet>
296EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet cbrt_halley_iteration_step(
const Packet& x_k,
298 typedef typename unpacket_traits<Packet>::type Scalar;
299 Packet x_k_cb = pmul(x_k, pmul(x_k, x_k));
300 Packet denom = pmadd(pset1<Packet>(Scalar(2)), x_k_cb, y);
301 Packet num = psub(x_k_cb, y);
302 Packet r = pdiv(num, denom);
303 return pnmadd(x_k, r, x_k);
308template <
typename Packet>
309EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet cbrt_decompose(
const Packet& x, Packet& e_div3) {
310 typedef typename unpacket_traits<Packet>::type Scalar;
318 constexpr Scalar kOneThird = Scalar(1) / 3;
319 e_div3 = pceil(pmul(e, pset1<Packet>(kOneThird)));
320 Packet e_mod3 = pnmadd(pset1<Packet>(Scalar(3)), e_div3, e);
323 return pldexp_fast(s, e_mod3);
326template <
typename Packet>
327EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet cbrt_special_cases_and_sign(
const Packet& x,
328 const Packet& abs_root) {
329 typedef typename unpacket_traits<Packet>::type Scalar;
332 const Packet sign_mask = pset1<Packet>(Scalar(-0.0));
333 const Packet x_sign = pand(sign_mask, x);
334 Packet root = por(x_sign, abs_root);
337 const Packet is_not_finite = por(pisinf(x), pisnan(x));
338 const Packet is_zero = pcmp_eq(pzero(x), x);
339 const Packet use_x = por(is_not_finite, is_zero);
340 return pselect(use_x, x, root);
371template <
typename Packet>
372EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pcbrt_float(
const Packet& x) {
373 typedef typename unpacket_traits<Packet>::type Scalar;
374 static_assert(std::is_same<Scalar, float>::value,
"Scalar type must be float");
379 const Packet y = cbrt_decompose(pabs(x), e_div3);
383 constexpr float alpha[] = {5.9220016002655029296875e-01f, -1.3859539031982421875e+00f, 1.4581282138824462890625e+00f,
384 3.408401906490325927734375e-01f};
385 Packet r = ppolevl<Packet, 3>::run(y, alpha);
388 r = cbrt_halley_iteration_step(r, y);
391 r = pldexp_fast(r, e_div3);
393 return cbrt_special_cases_and_sign(x, r);
402template <
typename Packet>
403EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pcbrt_double(
const Packet& x) {
404 typedef typename unpacket_traits<Packet>::type Scalar;
405 static_assert(std::is_same<Scalar, double>::value,
"Scalar type must be double");
410 const Packet y = cbrt_decompose(pabs(x), e_div3);
414 constexpr double alpha[] = {-4.69470621553356115551736138513660989701747894287109375e-01,
415 1.072314636518546304699839311069808900356292724609375e+00,
416 3.81249427609571867048288140722434036433696746826171875e-01};
417 Packet r = ppolevl<Packet, 2>::run(y, alpha);
420 r = cbrt_halley_iteration_step(r, y);
421 r = cbrt_halley_iteration_step(r, y);
424 r = pldexp_fast(r, e_div3);
425 return cbrt_special_cases_and_sign(x, r);
434template <
typename Packet,
bool base2>
435EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog_impl_float(
const Packet _x) {
436 const Packet cst_1 = pset1<Packet>(1.0f);
437 const Packet cst_minus_inf = pset1frombits<Packet>(
static_cast<Eigen::numext::uint32_t
>(0xff800000u));
438 const Packet cst_pos_inf = pset1frombits<Packet>(
static_cast<Eigen::numext::uint32_t
>(0x7f800000u));
440 const Packet cst_cephes_SQRTHF = pset1<Packet>(0.707106781186547524f);
452 Packet mask = pcmp_lt(x, cst_cephes_SQRTHF);
453 Packet tmp = pand(x, mask);
455 e = psub(e, pand(cst_1, mask));
460 constexpr float alpha[] = {0.18256296349849254f, 1.0000000190281063f, 1.0000000190281136f};
461 constexpr float beta[] = {0.049616247954120038f, 0.59923249590823520f, 1.4999999999999927f, 1.0f};
463 Packet p = ppolevl<Packet, 2>::run(x, alpha);
465 Packet q = ppolevl<Packet, 3>::run(x, beta);
470 const Packet cst_log2e = pset1<Packet>(
static_cast<float>(EIGEN_LOG2E));
471 x = pmadd(x, cst_log2e, e);
473 const Packet cst_ln2 = pset1<Packet>(
static_cast<float>(EIGEN_LN2));
474 x = pmadd(e, cst_ln2, x);
477 Packet invalid_mask = pcmp_lt_or_nan(_x, pzero(_x));
478 Packet iszero_mask = pcmp_eq(_x, pzero(_x));
479 Packet pos_inf_mask = pcmp_eq(_x, cst_pos_inf);
484 return pselect(iszero_mask, cst_minus_inf, por(pselect(pos_inf_mask, cst_pos_inf, x), invalid_mask));
487template <
typename Packet>
488EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog_float(
const Packet _x) {
489 return plog_impl_float<Packet,
false>(_x);
492template <
typename Packet>
493EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog2_float(
const Packet _x) {
494 return plog_impl_float<Packet,
true>(_x);
506template <
typename Packet,
bool base2>
507EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog_impl_double(
const Packet _x) {
510 const Packet cst_1 = pset1<Packet>(1.0);
511 const Packet cst_neg_half = pset1<Packet>(-0.5);
512 const Packet cst_minus_inf = pset1frombits<Packet>(
static_cast<uint64_t
>(0xfff0000000000000ull));
513 const Packet cst_pos_inf = pset1frombits<Packet>(
static_cast<uint64_t
>(0x7ff0000000000000ull));
517 const Packet cst_cephes_SQRTHF = pset1<Packet>(0.70710678118654752440E0);
518 const Packet cst_cephes_log_p0 = pset1<Packet>(1.01875663804580931796E-4);
519 const Packet cst_cephes_log_p1 = pset1<Packet>(4.97494994976747001425E-1);
520 const Packet cst_cephes_log_p2 = pset1<Packet>(4.70579119878881725854E0);
521 const Packet cst_cephes_log_p3 = pset1<Packet>(1.44989225341610930846E1);
522 const Packet cst_cephes_log_p4 = pset1<Packet>(1.79368678507819816313E1);
523 const Packet cst_cephes_log_p5 = pset1<Packet>(7.70838733755885391666E0);
525 const Packet cst_cephes_log_q0 = pset1<Packet>(1.0);
526 const Packet cst_cephes_log_q1 = pset1<Packet>(1.12873587189167450590E1);
527 const Packet cst_cephes_log_q2 = pset1<Packet>(4.52279145837532221105E1);
528 const Packet cst_cephes_log_q3 = pset1<Packet>(8.29875266912776603211E1);
529 const Packet cst_cephes_log_q4 = pset1<Packet>(7.11544750618563894466E1);
530 const Packet cst_cephes_log_q5 = pset1<Packet>(2.31251620126765340583E1);
543 Packet mask = pcmp_lt(x, cst_cephes_SQRTHF);
544 Packet tmp = pand(x, mask);
546 e = psub(e, pand(cst_1, mask));
549 Packet x2 = pmul(x, x);
550 Packet x3 = pmul(x2, x);
555 y = pmadd(cst_cephes_log_p0, x, cst_cephes_log_p1);
556 y1 = pmadd(cst_cephes_log_p3, x, cst_cephes_log_p4);
557 y = pmadd(y, x, cst_cephes_log_p2);
558 y1 = pmadd(y1, x, cst_cephes_log_p5);
559 y_ = pmadd(y, x3, y1);
561 y = pmadd(cst_cephes_log_q0, x, cst_cephes_log_q1);
562 y1 = pmadd(cst_cephes_log_q3, x, cst_cephes_log_q4);
563 y = pmadd(y, x, cst_cephes_log_q2);
564 y1 = pmadd(y1, x, cst_cephes_log_q5);
565 y = pmadd(y, x3, y1);
570 y = pmadd(cst_neg_half, x2, y);
575 const Packet cst_log2e = pset1<Packet>(
static_cast<double>(EIGEN_LOG2E));
576 x = pmadd(x, cst_log2e, e);
578 const Packet cst_ln2 = pset1<Packet>(
static_cast<double>(EIGEN_LN2));
579 x = pmadd(e, cst_ln2, x);
582 Packet invalid_mask = pcmp_lt_or_nan(_x, pzero(_x));
583 Packet iszero_mask = pcmp_eq(_x, pzero(_x));
584 Packet pos_inf_mask = pcmp_eq(_x, cst_pos_inf);
589 return pselect(iszero_mask, cst_minus_inf, por(pselect(pos_inf_mask, cst_pos_inf, x), invalid_mask));
592template <
typename Packet>
593EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog_double(
const Packet _x) {
594 return plog_impl_double<Packet,
false>(_x);
597template <
typename Packet>
598EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog2_double(
const Packet _x) {
599 return plog_impl_double<Packet,
true>(_x);
605template <
typename Packet>
606EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet generic_log1p(
const Packet& x) {
607 typedef typename unpacket_traits<Packet>::type ScalarType;
608 const Packet one = pset1<Packet>(ScalarType(1));
609 Packet xp1 = padd(x, one);
610 Packet small_mask = pcmp_eq(xp1, one);
611 Packet log1 = plog(xp1);
612 Packet inf_mask = pcmp_eq(xp1, log1);
613 Packet log_large = pmul(x, pdiv(log1, psub(xp1, one)));
614 return pselect(por(small_mask, inf_mask), x, log_large);
620template <
typename Packet>
621EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet generic_expm1(
const Packet& x) {
622 typedef typename unpacket_traits<Packet>::type ScalarType;
623 const Packet one = pset1<Packet>(ScalarType(1));
624 const Packet neg_one = pset1<Packet>(ScalarType(-1));
626 Packet one_mask = pcmp_eq(u, one);
627 Packet u_minus_one = psub(u, one);
628 Packet neg_one_mask = pcmp_eq(u_minus_one, neg_one);
629 Packet logu = plog(u);
634 Packet pos_inf_mask = pcmp_eq(logu, u);
635 Packet
expm1 = pmul(u_minus_one, pdiv(x, logu));
637 return pselect(one_mask, x, pselect(neg_one_mask, neg_one,
expm1));
644template <
typename Packet>
645EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pexp_float(
const Packet _x) {
646 const Packet cst_zero = pset1<Packet>(0.0f);
647 const Packet cst_one = pset1<Packet>(1.0f);
648 const Packet cst_half = pset1<Packet>(0.5f);
649 const Packet cst_exp_hi = pset1<Packet>(88.723f);
650 const Packet cst_exp_lo = pset1<Packet>(-104.f);
651 const Packet cst_pldexp_threshold = pset1<Packet>(87.0);
653 const Packet cst_cephes_LOG2EF = pset1<Packet>(1.44269504088896341f);
654 const Packet cst_p2 = pset1<Packet>(0.49999988079071044921875f);
655 const Packet cst_p3 = pset1<Packet>(0.16666518151760101318359375f);
656 const Packet cst_p4 = pset1<Packet>(4.166965186595916748046875e-2f);
657 const Packet cst_p5 = pset1<Packet>(8.36894474923610687255859375e-3f);
658 const Packet cst_p6 = pset1<Packet>(1.37449637986719608306884765625e-3f);
661 Packet zero_mask = pcmp_lt(_x, cst_exp_lo);
662 Packet x = pmin(_x, cst_exp_hi);
666 Packet m = pfloor(pmadd(x, cst_cephes_LOG2EF, cst_half));
671 const Packet cst_cephes_exp_C1 = pset1<Packet>(-0.693359375f);
672 const Packet cst_cephes_exp_C2 = pset1<Packet>(2.12194440e-4f);
673 Packet r = pmadd(m, cst_cephes_exp_C1, x);
674 r = pmadd(m, cst_cephes_exp_C2, r);
678 const Packet r2 = pmul(r, r);
679 Packet p_even = pmadd(r2, cst_p6, cst_p4);
680 const Packet p_odd = pmadd(r2, cst_p5, cst_p3);
681 p_even = pmadd(r2, p_even, cst_p2);
682 const Packet p_low = padd(r, cst_one);
683 Packet y = pmadd(r, p_odd, p_even);
684 y = pmadd(r2, y, p_low);
687 const Packet fast_pldexp_unsafe = pcmp_lt(cst_pldexp_threshold, pabs(x));
688 if (!predux_any(fast_pldexp_unsafe)) {
691 return pmax(pldexp_fast(y, m), _x);
693 return pselect(zero_mask, cst_zero, pmax(pldexp(y, m), _x));
696template <
typename Packet>
697EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pexp_double(
const Packet _x) {
699 const Packet cst_zero = pset1<Packet>(0.0);
700 const Packet cst_1 = pset1<Packet>(1.0);
701 const Packet cst_2 = pset1<Packet>(2.0);
702 const Packet cst_half = pset1<Packet>(0.5);
704 const Packet cst_exp_hi = pset1<Packet>(709.784);
705 const Packet cst_exp_lo = pset1<Packet>(-745.519);
706 const Packet cst_pldexp_threshold = pset1<Packet>(708.0);
707 const Packet cst_cephes_LOG2EF = pset1<Packet>(1.4426950408889634073599);
708 const Packet cst_cephes_exp_p0 = pset1<Packet>(1.26177193074810590878e-4);
709 const Packet cst_cephes_exp_p1 = pset1<Packet>(3.02994407707441961300e-2);
710 const Packet cst_cephes_exp_p2 = pset1<Packet>(9.99999999999999999910e-1);
711 const Packet cst_cephes_exp_q0 = pset1<Packet>(3.00198505138664455042e-6);
712 const Packet cst_cephes_exp_q1 = pset1<Packet>(2.52448340349684104192e-3);
713 const Packet cst_cephes_exp_q2 = pset1<Packet>(2.27265548208155028766e-1);
714 const Packet cst_cephes_exp_q3 = pset1<Packet>(2.00000000000000000009e0);
715 const Packet cst_cephes_exp_C1 = pset1<Packet>(0.693145751953125);
716 const Packet cst_cephes_exp_C2 = pset1<Packet>(1.42860682030941723212e-6);
721 Packet zero_mask = pcmp_lt(_x, cst_exp_lo);
722 x = pmin(x, cst_exp_hi);
724 fx = pmadd(cst_cephes_LOG2EF, x, cst_half);
732 tmp = pmul(fx, cst_cephes_exp_C1);
733 Packet z = pmul(fx, cst_cephes_exp_C2);
737 Packet x2 = pmul(x, x);
740 Packet px = cst_cephes_exp_p0;
741 px = pmadd(px, x2, cst_cephes_exp_p1);
742 px = pmadd(px, x2, cst_cephes_exp_p2);
746 Packet qx = cst_cephes_exp_q0;
747 qx = pmadd(qx, x2, cst_cephes_exp_q1);
748 qx = pmadd(qx, x2, cst_cephes_exp_q2);
749 qx = pmadd(qx, x2, cst_cephes_exp_q3);
754 x = pdiv(px, psub(qx, px));
755 x = pmadd(cst_2, x, cst_1);
759 const Packet fast_pldexp_unsafe = pcmp_lt(cst_pldexp_threshold, pabs(_x));
760 if (!predux_any(fast_pldexp_unsafe)) {
763 return pmax(pldexp_fast(x, fx), _x);
765 return pselect(zero_mask, cst_zero, pmax(pldexp(x, fx), _x));
777inline float trig_reduce_huge(
float xf, Eigen::numext::int32_t* quadrant) {
778 using Eigen::numext::int32_t;
779 using Eigen::numext::int64_t;
780 using Eigen::numext::uint32_t;
781 using Eigen::numext::uint64_t;
783 const double pio2_62 = 3.4061215800865545e-19;
784 const uint64_t zero_dot_five = uint64_t(1) << 61;
788 static const uint32_t two_over_pi[] = {
789 0x00000028, 0x000028be, 0x0028be60, 0x28be60db, 0xbe60db93, 0x60db9391, 0xdb939105, 0x9391054a, 0x91054a7f,
790 0x054a7f09, 0x4a7f09d5, 0x7f09d5f4, 0x09d5f47d, 0xd5f47d4d, 0xf47d4d37, 0x7d4d3770, 0x4d377036, 0x377036d8,
791 0x7036d8a5, 0x36d8a566, 0xd8a5664f, 0xa5664f10, 0x664f10e4, 0x4f10e410, 0x10e41000, 0xe4100000};
793 uint32_t xi = numext::bit_cast<uint32_t>(xf);
798 uint32_t e = (xi >> 23) - 118;
800 xi = ((xi & 0x007fffffu) | 0x00800000u) << (e & 0x7);
803 uint32_t twoopi_1 = two_over_pi[i - 1];
804 uint32_t twoopi_2 = two_over_pi[i + 3];
805 uint32_t twoopi_3 = two_over_pi[i + 7];
809 p = uint64_t(xi) * twoopi_3;
810 p = uint64_t(xi) * twoopi_2 + (p >> 32);
811 p = (uint64_t(xi * twoopi_1) << 32) + p;
814 uint64_t q = (p + zero_dot_five) >> 62;
821 return float(
double(int64_t(p)) * pio2_62);
824template <
bool ComputeSine,
typename Packet,
bool ComputeBoth = false>
825EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS
826#if EIGEN_COMP_GNUC_STRICT
827 __attribute__((optimize(
"-fno-unsafe-math-optimizations")))
830 psincos_float(
const Packet& _x) {
831 typedef typename unpacket_traits<Packet>::integer_packet PacketI;
833 const Packet cst_2oPI = pset1<Packet>(0.636619746685028076171875f);
834 const Packet cst_rounding_magic = pset1<Packet>(12582912);
835 const PacketI csti_1 = pset1<PacketI>(1);
836 const Packet cst_sign_mask = pset1frombits<Packet>(
static_cast<Eigen::numext::uint32_t
>(0x80000000u));
841 Packet y = pmul(x, cst_2oPI);
844 Packet y_round = padd(y, cst_rounding_magic);
845 EIGEN_OPTIMIZATION_BARRIER(y_round)
846 PacketI y_int = preinterpret<PacketI>(y_round);
847 y = psub(y_round, cst_rounding_magic);
851#if defined(EIGEN_VECTORIZE_FMA)
854 const float huge_th = ComputeSine ? 117435.992f : 71476.0625f;
855 x = pmadd(y, pset1<Packet>(-1.57079601287841796875f), x);
856 x = pmadd(y, pset1<Packet>(-3.1391647326017846353352069854736328125e-07f), x);
857 x = pmadd(y, pset1<Packet>(-5.390302529957764765544681040410068817436695098876953125e-15f), x);
865 const float huge_th = ComputeSine ? 25966.f : 18838.f;
866 x = pmadd(y, pset1<Packet>(-1.5703125), x);
867 EIGEN_OPTIMIZATION_BARRIER(x)
868 x = pmadd(y, pset1<Packet>(-0.000483989715576171875), x);
869 EIGEN_OPTIMIZATION_BARRIER(x)
870 x = pmadd(y, pset1<Packet>(1.62865035235881805419921875e-07), x);
871 x = pmadd(y, pset1<Packet>(5.5644315544167710640977020375430583953857421875e-11), x);
887 if (predux_any(pcmp_le(pset1<Packet>(huge_th), pabs(_x)))) {
888 const int PacketSize = unpacket_traits<Packet>::size;
889 EIGEN_ALIGN_TO_BOUNDARY(
sizeof(Packet))
float vals[PacketSize];
890 EIGEN_ALIGN_TO_BOUNDARY(
sizeof(Packet))
float x_cpy[PacketSize];
891 EIGEN_ALIGN_TO_BOUNDARY(
sizeof(Packet)) Eigen::numext::int32_t y_int2[PacketSize];
892 pstoreu(vals, pabs(_x));
894 pstoreu(y_int2, y_int);
895 for (
int k = 0; k < PacketSize; ++k) {
897 if (val >= huge_th && (numext::isfinite)(val)) x_cpy[k] = trig_reduce_huge(val, &y_int2[k]);
899 x = ploadu<Packet>(x_cpy);
900 y_int = ploadu<PacketI>(y_int2);
906 Packet sign_bit = ComputeSine ? pxor(_x, preinterpret<Packet>(plogical_shift_left<30>(y_int)))
907 : preinterpret<Packet>(plogical_shift_left<30>(padd(y_int, csti_1)));
908 sign_bit = pand(sign_bit, cst_sign_mask);
912 Packet poly_mask = preinterpret<Packet>(pcmp_eq(pand(y_int, csti_1), pzero(y_int)));
914 Packet x2 = pmul(x, x);
917 Packet y1 = pset1<Packet>(2.4372266125283204019069671630859375e-05f);
918 y1 = pmadd(y1, x2, pset1<Packet>(-0.00138865201734006404876708984375f));
919 y1 = pmadd(y1, x2, pset1<Packet>(0.041666619479656219482421875f));
920 y1 = pmadd(y1, x2, pset1<Packet>(-0.5f));
921 y1 = pmadd(y1, x2, pset1<Packet>(1.f));
931 Packet y2 = pset1<Packet>(-0.0001959234114083702898469196984621021329076029360294342041015625f);
932 y2 = pmadd(y2, x2, pset1<Packet>(0.0083326873655616851693794799871284340042620897293090820312500000f));
933 y2 = pmadd(y2, x2, pset1<Packet>(-0.1666666203982298255503735617821803316473960876464843750000000000f));
935 y2 = pmadd(y2, x, x);
939 Packet peven = peven_mask(x);
940 Packet ysin = pselect(poly_mask, y2, y1);
941 Packet ycos = pselect(poly_mask, y1, y2);
942 Packet sign_bit_sin = pxor(_x, preinterpret<Packet>(plogical_shift_left<30>(y_int)));
943 Packet sign_bit_cos = preinterpret<Packet>(plogical_shift_left<30>(padd(y_int, csti_1)));
944 sign_bit_sin = pand(sign_bit_sin, cst_sign_mask);
945 sign_bit_cos = pand(sign_bit_cos, cst_sign_mask);
946 y = pselect(peven, pxor(ysin, sign_bit_sin), pxor(ycos, sign_bit_cos));
948 y = ComputeSine ? pselect(poly_mask, y2, y1) : pselect(poly_mask, y1, y2);
949 y = pxor(y, sign_bit);
955template <
typename Packet>
956EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet psin_float(
const Packet& x) {
957 return psincos_float<true>(x);
960template <
typename Packet>
961EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pcos_float(
const Packet& x) {
962 return psincos_float<false>(x);
968template <
typename Packet>
969Packet trig_reduce_small_double(
const Packet& x,
const Packet& q) {
971 const Packet cst_pio2_a = pset1<Packet>(-1.570796325802803);
972 const Packet cst_pio2_b = pset1<Packet>(-9.920935184482005e-10);
975 t = pmadd(cst_pio2_a, q, x);
976 t = pmadd(cst_pio2_b, q, t);
983template <
typename Packet>
984Packet trig_reduce_medium_double(
const Packet& x,
const Packet& q_high,
const Packet& q_low) {
986 const Packet cst_pio2_a = pset1<Packet>(-1.570796325802803);
987 const Packet cst_pio2_b = pset1<Packet>(-9.920935184482005e-10);
988 const Packet cst_pio2_c = pset1<Packet>(-6.123234014771656e-17);
989 const Packet cst_pio2_d = pset1<Packet>(1.903488962019325e-25);
992 t = pmadd(cst_pio2_a, q_high, x);
993 t = pmadd(cst_pio2_a, q_low, t);
994 t = pmadd(cst_pio2_b, q_high, t);
995 t = pmadd(cst_pio2_b, q_low, t);
996 t = pmadd(cst_pio2_c, q_high, t);
997 t = pmadd(cst_pio2_c, q_low, t);
998 t = pmadd(cst_pio2_d, padd(q_low, q_high), t);
1002template <
bool ComputeSine,
typename Packet,
bool ComputeBoth = false>
1003EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS
1004#if EIGEN_COMP_GNUC_STRICT
1005 __attribute__((optimize(
"-fno-unsafe-math-optimizations")))
1008 psincos_double(
const Packet& x) {
1009 typedef typename unpacket_traits<Packet>::integer_packet PacketI;
1010 typedef typename unpacket_traits<PacketI>::type ScalarI;
1012 const Packet cst_sign_mask = pset1frombits<Packet>(
static_cast<Eigen::numext::uint64_t
>(0x8000000000000000u));
1015 const double small_th = 15;
1017 const double huge_th = 1e14;
1019 const Packet cst_2oPI = pset1<Packet>(0.63661977236758134307553505349006);
1021 const PacketI cst_one = pset1<PacketI>(ScalarI(1));
1023 const Packet cst_split = pset1<Packet>(1 << 24);
1025 Packet x_abs = pabs(x);
1032 if (EIGEN_PREDICT_FALSE(predux_any(pcmp_le(pset1<Packet>(small_th), x_abs)))) {
1033 Packet q_high = pmul(pfloor(pmul(x_abs, pdiv(cst_2oPI, cst_split))), cst_split);
1034 Packet q_low_noround = psub(pmul(x_abs, cst_2oPI), q_high);
1035 q_int = pcast<Packet, PacketI>(padd(q_low_noround, pset1<Packet>(0.5)));
1036 Packet q_low = pcast<PacketI, Packet>(q_int);
1037 s = trig_reduce_medium_double(x_abs, q_high, q_low);
1039 Packet qval_noround = pmul(x_abs, cst_2oPI);
1040 q_int = pcast<Packet, PacketI>(padd(qval_noround, pset1<Packet>(0.5)));
1041 Packet q = pcast<PacketI, Packet>(q_int);
1042 s = trig_reduce_small_double(x_abs, q);
1046 Packet ss = pmul(s, s);
1056 Packet sc1_num = pmadd(ss, pset1<Packet>(80737373), pset1<Packet>(-13853547000));
1057 Packet sc2_num = pmadd(sc1_num, ss, pset1<Packet>(727718024880));
1058 Packet sc3_num = pmadd(sc2_num, ss, pset1<Packet>(-11275015752000));
1059 Packet sc4_num = pmadd(sc3_num, ss, pset1<Packet>(23594700729600));
1060 Packet sc1_denum = pmadd(ss, pset1<Packet>(147173), pset1<Packet>(39328920));
1061 Packet sc2_denum = pmadd(sc1_denum, ss, pset1<Packet>(5772800880));
1062 Packet sc3_denum = pmadd(sc2_denum, ss, pset1<Packet>(522334612800));
1063 Packet sc4_denum = pmadd(sc3_denum, ss, pset1<Packet>(23594700729600));
1064 Packet scos = pdiv(sc4_num, sc4_denum);
1074 Packet ss1_num = pmadd(ss, pset1<Packet>(4585922449), pset1<Packet>(-1066023933480));
1075 Packet ss2_num = pmadd(ss1_num, ss, pset1<Packet>(83284044283440));
1076 Packet ss3_num = pmadd(ss2_num, ss, pset1<Packet>(-2303682236856000));
1077 Packet ss4_num = pmadd(ss3_num, ss, pset1<Packet>(15605159573203200));
1078 Packet ss1_denum = pmadd(ss, pset1<Packet>(1029037), pset1<Packet>(345207016));
1079 Packet ss2_denum = pmadd(ss1_denum, ss, pset1<Packet>(61570292784));
1080 Packet ss3_denum = pmadd(ss2_denum, ss, pset1<Packet>(6603948711360));
1081 Packet ss4_denum = pmadd(ss3_denum, ss, pset1<Packet>(346781323848960));
1082 Packet ssin = pdiv(pmul(s, ss4_num), pmul(pset1<Packet>(45), ss4_denum));
1084 Packet poly_mask = preinterpret<Packet>(pcmp_eq(pand(q_int, cst_one), pzero(q_int)));
1086 Packet sign_sin = pxor(x, preinterpret<Packet>(plogical_shift_left<62>(q_int)));
1087 Packet sign_cos = preinterpret<Packet>(plogical_shift_left<62>(padd(q_int, cst_one)));
1088 Packet sign_bit, sFinalRes;
1090 Packet peven = peven_mask(x);
1091 sign_bit = pselect((s), sign_sin, sign_cos);
1092 sFinalRes = pselect(pxor(peven, poly_mask), ssin, scos);
1094 sign_bit = ComputeSine ? sign_sin : sign_cos;
1095 sFinalRes = ComputeSine ? pselect(poly_mask, ssin, scos) : pselect(poly_mask, scos, ssin);
1097 sign_bit = pand(sign_bit, cst_sign_mask);
1098 sFinalRes = pxor(sFinalRes, sign_bit);
1103 if (EIGEN_PREDICT_FALSE(predux_any(pcmp_le(pset1<Packet>(huge_th), x_abs)))) {
1104 const int PacketSize = unpacket_traits<Packet>::size;
1105 EIGEN_ALIGN_TO_BOUNDARY(
sizeof(Packet))
double sincos_vals[PacketSize];
1106 EIGEN_ALIGN_TO_BOUNDARY(
sizeof(Packet))
double x_cpy[PacketSize];
1108 pstoreu(sincos_vals, sFinalRes);
1109 for (
int k = 0; k < PacketSize; ++k) {
1110 double val = x_cpy[k];
1111 if (std::abs(val) > huge_th && (numext::isfinite)(val)) {
1113 sincos_vals[k] = k % 2 == 0 ? std::sin(val) : std::
cos(val);
1115 sincos_vals[k] = ComputeSine ? std::sin(val) : std::
cos(val);
1118 sFinalRes = ploadu<Packet>(sincos_vals);
1123template <
typename Packet>
1124EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet psin_double(
const Packet& x) {
1125 return psincos_double<true>(x);
1128template <
typename Packet>
1129EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pcos_double(
const Packet& x) {
1130 return psincos_double<false>(x);
1134template <
typename Packet>
1135EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pacos_float(
const Packet& x_in) {
1136 typedef typename unpacket_traits<Packet>::type Scalar;
1137 static_assert(std::is_same<Scalar, float>::value,
"Scalar type must be float");
1139 const Packet cst_one = pset1<Packet>(Scalar(1));
1140 const Packet cst_pi = pset1<Packet>(Scalar(EIGEN_PI));
1141 const Packet p6 = pset1<Packet>(Scalar(2.36423197202384471893310546875e-3));
1142 const Packet p5 = pset1<Packet>(Scalar(-1.1368644423782825469970703125e-2));
1143 const Packet p4 = pset1<Packet>(Scalar(2.717843465507030487060546875e-2));
1144 const Packet p3 = pset1<Packet>(Scalar(-4.8969544470310211181640625e-2));
1145 const Packet p2 = pset1<Packet>(Scalar(8.8804088532924652099609375e-2));
1146 const Packet p1 = pset1<Packet>(Scalar(-0.214591205120086669921875));
1147 const Packet p0 = pset1<Packet>(Scalar(1.57079637050628662109375));
1152 const Packet neg_mask = psignbit(x_in);
1153 const Packet abs_x = pabs(x_in);
1159 Packet x2 = pmul(x_in, x_in);
1160 Packet p_even = pmadd(p6, x2, p4);
1161 Packet p_odd = pmadd(p5, x2, p3);
1162 p_even = pmadd(p_even, x2, p2);
1163 p_odd = pmadd(p_odd, x2, p1);
1164 p_even = pmadd(p_even, x2, p0);
1165 Packet p = pmadd(p_odd, abs_x, p_even);
1170 Packet denom = psqrt(psub(cst_one, abs_x));
1171 Packet result = pmul(denom, p);
1173 return pselect(neg_mask, psub(cst_pi, result), result);
1177template <
typename Packet>
1178EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pasin_float(
const Packet& x_in) {
1179 typedef typename unpacket_traits<Packet>::type Scalar;
1180 static_assert(std::is_same<Scalar, float>::value,
"Scalar type must be float");
1182 constexpr float kPiOverTwo =
static_cast<float>(EIGEN_PI / 2);
1184 const Packet cst_half = pset1<Packet>(0.5f);
1185 const Packet cst_one = pset1<Packet>(1.0f);
1186 const Packet cst_two = pset1<Packet>(2.0f);
1187 const Packet cst_pi_over_two = pset1<Packet>(kPiOverTwo);
1189 const Packet abs_x = pabs(x_in);
1190 const Packet sign_mask = pandnot(x_in, abs_x);
1191 const Packet invalid_mask = pcmp_lt(cst_one, abs_x);
1198 const Packet x_large = psqrt(pnmadd(cst_half, abs_x, cst_half));
1199 const Packet large_mask = pcmp_lt(cst_half, abs_x);
1200 const Packet x = pselect(large_mask, x_large, abs_x);
1201 const Packet x2 = pmul(x, x);
1205 constexpr float alpha[] = {5.08838854730129241943359375e-2f, 3.95139865577220916748046875e-2f,
1206 7.550220191478729248046875e-2f, 0.16664917767047882080078125f, 1.00000011920928955078125f};
1207 Packet p = ppolevl<Packet, 4>::run(x2, alpha);
1210 const Packet p_large = pnmadd(cst_two, p, cst_pi_over_two);
1211 p = pselect(large_mask, p_large, p);
1213 p = pxor(p, sign_mask);
1215 return por(invalid_mask, p);
1218template <
typename Scalar>
1219struct patan_reduced {
1220 template <
typename Packet>
1221 static EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet run(
const Packet& x);
1225template <
typename Packet>
1226EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet patan_reduced<double>::run(
const Packet& x) {
1227 constexpr double alpha[] = {2.6667153866462208e-05, 3.0917513112462781e-03, 5.2574296781008604e-02,
1228 3.0409318473444424e-01, 7.5365702534987022e-01, 8.2704055405494614e-01,
1229 3.3004361289279920e-01};
1231 constexpr double beta[] = {
1232 2.7311202462436667e-04, 1.0899150928962708e-02, 1.1548932646420353e-01, 4.9716458728465573e-01, 1.0,
1233 9.3705509168587852e-01, 3.3004361289279920e-01};
1235 Packet x2 = pmul(x, x);
1236 Packet p = ppolevl<Packet, 6>::run(x2, alpha);
1237 Packet q = ppolevl<Packet, 6>::run(x2, beta);
1238 return pmul(x, pdiv(p, q));
1243template <
typename Packet>
1244EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet patan_reduced<float>::run(
const Packet& x) {
1245 constexpr float alpha[] = {1.12026982009410858154296875e-01f, 7.296695709228515625e-01f, 8.109951019287109375e-01f};
1247 constexpr float beta[] = {1.00917108356952667236328125e-02f, 2.8318560123443603515625e-01f, 1.0f,
1248 8.109951019287109375e-01f};
1250 Packet x2 = pmul(x, x);
1251 Packet p = ppolevl<Packet, 2>::run(x2, alpha);
1252 Packet q = ppolevl<Packet, 3>::run(x2, beta);
1253 return pmul(x, pdiv(p, q));
1256template <
typename Packet>
1257EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet generic_atan(
const Packet& x_in) {
1258 typedef typename unpacket_traits<Packet>::type Scalar;
1260 constexpr Scalar kPiOverTwo =
static_cast<Scalar
>(EIGEN_PI / 2);
1262 const Packet cst_signmask = pset1<Packet>(Scalar(-0.0));
1263 const Packet cst_one = pset1<Packet>(Scalar(1));
1264 const Packet cst_pi_over_two = pset1<Packet>(kPiOverTwo);
1270 const Packet abs_x = pabs(x_in);
1271 const Packet x_signmask = pand(x_in, cst_signmask);
1272 const Packet large_mask = pcmp_lt(cst_one, abs_x);
1273 const Packet x = pselect(large_mask, preciprocal(abs_x), abs_x);
1274 const Packet p = patan_reduced<Scalar>::run(x);
1276 Packet result = pselect(large_mask, psub(cst_pi_over_two, p), p);
1278 return pxor(result, x_signmask);
1290template <
typename T>
1291EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS T ptanh_float(
const T& a_x) {
1296#ifdef EIGEN_VECTORIZE_FMA
1297 const T plus_clamp = pset1<T>(8.01773357391357422f);
1298 const T minus_clamp = pset1<T>(-8.01773357391357422f);
1300 const T plus_clamp = pset1<T>(7.90738964080810547f);
1301 const T minus_clamp = pset1<T>(-7.90738964080810547f);
1303 const T x = pmax(pmin(a_x, plus_clamp), minus_clamp);
1313 constexpr float alpha[] = {1.394553628e-8f, 2.102733560e-5f, 3.520756727e-3f, 1.340216100e-1f};
1316 constexpr float beta[] = {8.015776984e-7f, 3.326951409e-4f, 2.597254514e-2f, 4.673548340e-1f, 1.0f};
1319 const T x2 = pmul(x, x);
1320 const T x3 = pmul(x2, x);
1322 T p = ppolevl<T, 3>::run(x2, alpha);
1323 T q = ppolevl<T, 4>::run(x2, beta);
1326 p = pmadd(x3, p, x);
1341template <
typename T>
1342EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS T ptanh_double(
const T& a_x) {
1347#ifdef EIGEN_VECTORIZE_FMA
1348 const T plus_clamp = pset1<T>(17.6610191624600077);
1349 const T minus_clamp = pset1<T>(-17.6610191624600077);
1351 const T plus_clamp = pset1<T>(17.714196154005176);
1352 const T minus_clamp = pset1<T>(-17.714196154005176);
1354 const T x = pmax(pmin(a_x, plus_clamp), minus_clamp);
1364 constexpr double alpha[] = {2.6158007860482230e-23, 7.6534862268749319e-19, 3.1309488231386680e-15,
1365 4.2303918148209176e-12, 2.4618379131293676e-09, 6.8644367682497074e-07,
1366 9.3839087674268880e-05, 5.9809711724441161e-03, 1.5184719640284322e-01};
1369 constexpr double beta[] = {6.463747022670968018e-21, 5.782506856739003571e-17,
1370 1.293019623712687916e-13, 1.123643448069621992e-10,
1371 4.492975677839633985e-08, 8.785185266237658698e-06,
1372 8.295161192716231542e-04, 3.437448108450402717e-02,
1373 4.851805297361760360e-01, 1.0};
1376 const T x2 = pmul(x, x);
1377 const T x3 = pmul(x2, x);
1381 T p = ppolevl<T, 8>::run(x2, alpha);
1382 T q = ppolevl<T, 9>::run(x2, beta);
1385 p = pmadd(x3, p, x);
1391template <
typename Packet>
1392EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet patanh_float(
const Packet& x) {
1393 typedef typename unpacket_traits<Packet>::type Scalar;
1394 static_assert(std::is_same<Scalar, float>::value,
"Scalar type must be float");
1398 constexpr float alpha[] = {0.1819281280040740966796875f, 8.2311116158962249755859375e-2f,
1399 0.14672131836414337158203125f, 0.1997792422771453857421875f, 0.3333373963832855224609375f};
1400 const Packet x2 = pmul(x, x);
1401 const Packet x3 = pmul(x, x2);
1402 Packet p = ppolevl<Packet, 4>::run(x2, alpha);
1403 p = pmadd(x3, p, x);
1406 const Packet half = pset1<Packet>(0.5f);
1407 const Packet one = pset1<Packet>(1.0f);
1408 Packet r = pdiv(padd(one, x), psub(one, x));
1409 r = pmul(half, plog(r));
1411 const Packet x_gt_half = pcmp_le(half, pabs(x));
1412 const Packet x_eq_one = pcmp_eq(one, pabs(x));
1413 const Packet x_gt_one = pcmp_lt(one, pabs(x));
1414 const Packet sign_mask = pset1<Packet>(-0.0f);
1415 const Packet x_sign = pand(sign_mask, x);
1416 const Packet inf = pset1<Packet>(std::numeric_limits<float>::infinity());
1417 return por(x_gt_one, pselect(x_eq_one, por(x_sign, inf), pselect(x_gt_half, r, p)));
1420template <
typename Packet>
1421EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet patanh_double(
const Packet& x) {
1422 typedef typename unpacket_traits<Packet>::type Scalar;
1423 static_assert(std::is_same<Scalar, double>::value,
"Scalar type must be double");
1426 constexpr double alpha[] = {3.3071338469301391e-03, -4.7129526768798737e-02, 1.8185306179826699e-01,
1427 -2.5949536095445679e-01, 1.2306328729812676e-01};
1429 constexpr double beta[] = {-3.8679974580640881e-03, 7.6391885763341910e-02, -4.2828141436397615e-01,
1430 9.8733495886883648e-01, -1.0000000000000000e+00, 3.6918986189438030e-01};
1432 const Packet x2 = pmul(x, x);
1433 const Packet x3 = pmul(x, x2);
1434 Packet p = ppolevl<Packet, 4>::run(x2, alpha);
1435 Packet q = ppolevl<Packet, 5>::run(x2, beta);
1436 Packet y_small = pmadd(x3, pdiv(p, q), x);
1439 const Packet half = pset1<Packet>(0.5);
1440 const Packet one = pset1<Packet>(1.0);
1441 Packet y_large = pdiv(padd(one, x), psub(one, x));
1442 y_large = pmul(half, plog(y_large));
1444 const Packet x_gt_half = pcmp_le(half, pabs(x));
1445 const Packet x_eq_one = pcmp_eq(one, pabs(x));
1446 const Packet x_gt_one = pcmp_lt(one, pabs(x));
1447 const Packet sign_mask = pset1<Packet>(-0.0);
1448 const Packet x_sign = pand(sign_mask, x);
1449 const Packet inf = pset1<Packet>(std::numeric_limits<double>::infinity());
1450 return por(x_gt_one, pselect(x_eq_one, por(x_sign, inf), pselect(x_gt_half, y_large, y_small)));
1453template <
typename Packet>
1454EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pdiv_complex(
const Packet& x,
const Packet& y) {
1455 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1459 const RealPacket y_abs = pabs(y.v);
1460 const RealPacket y_abs_flip = pcplxflip(Packet(y_abs)).v;
1461 const RealPacket y_max = pmax(y_abs, y_abs_flip);
1462 const RealPacket y_scaled = pdiv(y.v, y_max);
1464 const RealPacket y_scaled_sq = pmul(y_scaled, y_scaled);
1465 const RealPacket denom = padd(y_scaled_sq, pcplxflip(Packet(y_scaled_sq)).v);
1466 Packet result_scaled = pmul(x, pconj(Packet(y_scaled)));
1468 result_scaled = Packet(pdiv(result_scaled.v, denom));
1470 return Packet(pdiv(result_scaled.v, y_max));
1473template <
typename Packet>
1474EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet plog_complex(
const Packet& x) {
1475 typedef typename unpacket_traits<Packet>::type Scalar;
1476 typedef typename Scalar::value_type RealScalar;
1477 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1479 RealPacket real_mask_rp = peven_mask(x.v);
1480 Packet real_mask(real_mask_rp);
1483 RealPacket x_flip = pcplxflip(x).v;
1484 Packet x_norm = phypot_complex(x);
1485 RealPacket xlogr = plog(x_norm.v);
1488 RealPacket ximg = patan2(x.v, x_flip);
1490 const RealPacket cst_pos_inf = pset1<RealPacket>(NumTraits<RealScalar>::infinity());
1491 RealPacket x_abs = pabs(x.v);
1492 RealPacket is_x_pos_inf = pcmp_eq(x_abs, cst_pos_inf);
1493 RealPacket is_y_pos_inf = pcplxflip(Packet(is_x_pos_inf)).v;
1494 RealPacket is_any_inf = por(is_x_pos_inf, is_y_pos_inf);
1495 RealPacket xreal = pselect(is_any_inf, cst_pos_inf, xlogr);
1497 Packet xres = pselect(real_mask, Packet(xreal), Packet(ximg));
1501template <
typename Packet>
1502EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet pexp_complex(
const Packet& a) {
1503 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1504 typedef typename unpacket_traits<Packet>::type Scalar;
1505 typedef typename Scalar::value_type RealScalar;
1506 const RealPacket even_mask = peven_mask(a.v);
1507 const RealPacket odd_mask = pcplxflip(Packet(even_mask)).v;
1513 RealPacket x = pand(a.v, even_mask);
1514 x = por(x, pcplxflip(Packet(x)).v);
1515 RealPacket expx = pexp(x);
1518 RealPacket y = pand(odd_mask, a.v);
1519 y = por(y, pcplxflip(Packet(y)).v);
1520 RealPacket cisy = psincos_float<false, RealPacket, true>(y);
1521 cisy = pcplxflip(Packet(cisy)).v;
1523 const RealPacket cst_pos_inf = pset1<RealPacket>(NumTraits<RealScalar>::infinity());
1524 const RealPacket cst_neg_inf = pset1<RealPacket>(-NumTraits<RealScalar>::infinity());
1529 RealPacket cisy_sign = por(pandnot(cisy, pabs(cisy)), pset1<RealPacket>(RealScalar(1)));
1530 cisy = pselect(pcmp_eq(x, cst_neg_inf), cisy_sign, cisy);
1534 RealPacket y_sign = por(pandnot(y, pabs(y)), pset1<RealPacket>(RealScalar(1)));
1535 cisy = pselect(pand(pcmp_eq(x, cst_pos_inf), pisnan(cisy)), pand(y_sign, even_mask), cisy);
1536 Packet result = Packet(pmul(expx, cisy));
1539 result = pselect(Packet(pcmp_eq(y, pzero(y))), Packet(por(pand(expx, even_mask), pand(y, odd_mask))), result);
1544template <
typename Packet>
1545EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet psqrt_complex(
const Packet& a) {
1546 typedef typename unpacket_traits<Packet>::type Scalar;
1547 typedef typename Scalar::value_type RealScalar;
1548 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1586 RealPacket a_abs = pabs(a.v);
1587 RealPacket a_abs_flip = pcplxflip(Packet(a_abs)).v;
1588 RealPacket a_max = pmax(a_abs, a_abs_flip);
1589 RealPacket a_min = pmin(a_abs, a_abs_flip);
1590 RealPacket a_min_zero_mask = pcmp_eq(a_min, pzero(a_min));
1591 RealPacket a_max_zero_mask = pcmp_eq(a_max, pzero(a_max));
1592 RealPacket r = pdiv(a_min, a_max);
1593 const RealPacket cst_one = pset1<RealPacket>(RealScalar(1));
1594 RealPacket l = pmul(a_max, psqrt(padd(cst_one, pmul(r, r))));
1596 l = pselect(a_min_zero_mask, a_max, l);
1601 const RealPacket cst_half = pset1<RealPacket>(RealScalar(0.5));
1603 rho.v = psqrt(pmul(cst_half, padd(a_abs, l)));
1608 RealPacket eta = pandnot(pmul(cst_half, pdiv(a.v, pcplxflip(rho).v)), a_max_zero_mask);
1609 RealPacket real_mask = peven_mask(a.v);
1610 Packet positive_real_result;
1612 positive_real_result.v = pselect(real_mask, rho.v, eta);
1616 const RealPacket cst_imag_sign_mask = pset1<Packet>(Scalar(RealScalar(0.0), RealScalar(-0.0))).v;
1617 RealPacket imag_signs = pand(a.v, cst_imag_sign_mask);
1618 Packet negative_real_result;
1620 negative_real_result.v = por(pabs(pcplxflip(positive_real_result).v), imag_signs);
1623 Packet negative_real_mask;
1624 negative_real_mask.v = pcmp_lt(pand(real_mask, a.v), pzero(a.v));
1625 negative_real_mask.v = por(negative_real_mask.v, pcplxflip(negative_real_mask).v);
1626 Packet result = pselect(negative_real_mask, negative_real_result, positive_real_result);
1633 const RealPacket cst_pos_inf = pset1<RealPacket>(NumTraits<RealScalar>::infinity());
1635 is_inf.v = pcmp_eq(a_abs, cst_pos_inf);
1637 is_real_inf.v = pand(is_inf.v, real_mask);
1638 is_real_inf = por(is_real_inf, pcplxflip(is_real_inf));
1640 Packet real_inf_result;
1641 real_inf_result.v = pmul(a_abs, pset1<Packet>(Scalar(RealScalar(1.0), RealScalar(0.0))).v);
1642 real_inf_result.v = pselect(negative_real_mask.v, pcplxflip(real_inf_result).v, real_inf_result.v);
1645 is_imag_inf.v = pandnot(is_inf.v, real_mask);
1646 is_imag_inf = por(is_imag_inf, pcplxflip(is_imag_inf));
1647 Packet imag_inf_result;
1648 imag_inf_result.v = por(pand(cst_pos_inf, real_mask), pandnot(a.v, real_mask));
1650 Packet result_is_nan = pisnan(result);
1651 result = por(result_is_nan, result);
1653 return pselect(is_imag_inf, imag_inf_result, pselect(is_real_inf, real_inf_result, result));
1658template <
typename Packet>
1659EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet phypot_complex(
const Packet& a) {
1660 typedef typename unpacket_traits<Packet>::type Scalar;
1661 typedef typename Scalar::value_type RealScalar;
1662 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1664 const RealPacket cst_zero_rp = pset1<RealPacket>(
static_cast<RealScalar
>(0.0));
1665 const RealPacket cst_minus_one_rp = pset1<RealPacket>(
static_cast<RealScalar
>(-1.0));
1666 const RealPacket cst_two_rp = pset1<RealPacket>(
static_cast<RealScalar
>(2.0));
1667 const RealPacket evenmask = peven_mask(a.v);
1669 RealPacket a_abs = pabs(a.v);
1670 RealPacket a_flip = pcplxflip(Packet(a_abs)).v;
1671 RealPacket a_all = pselect(evenmask, a_abs, a_flip);
1672 RealPacket b_all = pselect(evenmask, a_flip, a_abs);
1674 RealPacket a2 = pmul(a.v, a.v);
1675 RealPacket a2_flip = pcplxflip(Packet(a2)).v;
1676 RealPacket h = psqrt(padd(a2, a2_flip));
1677 RealPacket h_sq = pmul(h, h);
1678 RealPacket a_sq = pselect(evenmask, a2, a2_flip);
1679 RealPacket m_h_sq = pmul(h_sq, cst_minus_one_rp);
1680 RealPacket m_a_sq = pmul(a_sq, cst_minus_one_rp);
1681 RealPacket x = psub(psub(pmadd(h, h, m_h_sq), pmadd(b_all, b_all, psub(a_sq, h_sq))), pmadd(a_all, a_all, m_a_sq));
1682 h = psub(h, pdiv(x, pmul(cst_two_rp, h)));
1685 RealPacket iszero = pcmp_eq(por(a_abs, a_flip), cst_zero_rp);
1687 h = pandnot(h, iszero);
1691template <
typename Packet>
1692struct psign_impl<Packet, std::enable_if_t<!is_scalar<Packet>::value &&
1693 !NumTraits<typename unpacket_traits<Packet>::type>::IsComplex &&
1694 !NumTraits<typename unpacket_traits<Packet>::type>::IsInteger>> {
1695 static EIGEN_DEVICE_FUNC
inline Packet run(
const Packet& a) {
1696 using Scalar =
typename unpacket_traits<Packet>::type;
1697 const Packet cst_one = pset1<Packet>(Scalar(1));
1698 const Packet cst_zero = pzero(a);
1700 const Packet abs_a = pabs(a);
1701 const Packet sign_mask = pandnot(a, abs_a);
1702 const Packet nonzero_mask = pcmp_lt(cst_zero, abs_a);
1704 return pselect(nonzero_mask, por(sign_mask, cst_one), abs_a);
1708template <
typename Packet>
1709struct psign_impl<Packet, std::enable_if_t<!is_scalar<Packet>::value &&
1710 !NumTraits<typename unpacket_traits<Packet>::type>::IsComplex &&
1711 NumTraits<typename unpacket_traits<Packet>::type>::IsSigned &&
1712 NumTraits<typename unpacket_traits<Packet>::type>::IsInteger>> {
1713 static EIGEN_DEVICE_FUNC
inline Packet run(
const Packet& a) {
1714 using Scalar =
typename unpacket_traits<Packet>::type;
1715 const Packet cst_one = pset1<Packet>(Scalar(1));
1716 const Packet cst_minus_one = pset1<Packet>(Scalar(-1));
1717 const Packet cst_zero = pzero(a);
1719 const Packet positive_mask = pcmp_lt(cst_zero, a);
1720 const Packet positive = pand(positive_mask, cst_one);
1721 const Packet negative_mask = pcmp_lt(a, cst_zero);
1722 const Packet negative = pand(negative_mask, cst_minus_one);
1724 return por(positive, negative);
1728template <
typename Packet>
1729struct psign_impl<Packet, std::enable_if_t<!is_scalar<Packet>::value &&
1730 !NumTraits<typename unpacket_traits<Packet>::type>::IsComplex &&
1731 !NumTraits<typename unpacket_traits<Packet>::type>::IsSigned &&
1732 NumTraits<typename unpacket_traits<Packet>::type>::IsInteger>> {
1733 static EIGEN_DEVICE_FUNC
inline Packet run(
const Packet& a) {
1734 using Scalar =
typename unpacket_traits<Packet>::type;
1735 const Packet cst_one = pset1<Packet>(Scalar(1));
1736 const Packet cst_zero = pzero(a);
1738 const Packet zero_mask = pcmp_eq(cst_zero, a);
1739 return pandnot(cst_one, zero_mask);
1744template <
typename Packet>
1745struct psign_impl<Packet, std::enable_if_t<!is_scalar<Packet>::value &&
1746 NumTraits<typename unpacket_traits<Packet>::type>::IsComplex &&
1747 unpacket_traits<Packet>::vectorizable>> {
1748 static EIGEN_DEVICE_FUNC
inline Packet run(
const Packet& a) {
1749 typedef typename unpacket_traits<Packet>::type Scalar;
1750 typedef typename Scalar::value_type RealScalar;
1751 typedef typename unpacket_traits<Packet>::as_real RealPacket;
1758 RealPacket a_abs = pabs(a.v);
1759 RealPacket a_abs_flip = pcplxflip(Packet(a_abs)).v;
1760 RealPacket a_max = pmax(a_abs, a_abs_flip);
1761 RealPacket a_min = pmin(a_abs, a_abs_flip);
1762 RealPacket a_min_zero_mask = pcmp_eq(a_min, pzero(a_min));
1763 RealPacket a_max_zero_mask = pcmp_eq(a_max, pzero(a_max));
1764 RealPacket r = pdiv(a_min, a_max);
1765 const RealPacket cst_one = pset1<RealPacket>(RealScalar(1));
1766 RealPacket l = pmul(a_max, psqrt(padd(cst_one, pmul(r, r))));
1769 l = pselect(a_min_zero_mask, a_max, l);
1771 RealPacket sign_as_real = pandnot(pdiv(a.v, l), a_max_zero_mask);
1773 sign.v = sign_as_real;
1785template <
typename Packet>
1786EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void absolute_split(
const Packet& x, Packet& n, Packet& r) {
1793template <
typename Packet>
1794EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void fast_twosum(
const Packet& x,
const Packet& y, Packet& s_hi, Packet& s_lo) {
1796 const Packet t = psub(s_hi, x);
1800#ifdef EIGEN_VECTORIZE_FMA
1805template <
typename Packet>
1806EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void twoprod(
const Packet& x,
const Packet& y, Packet& p_hi, Packet& p_lo) {
1808 p_lo = pmsub(x, y, p_hi);
1813template <
typename Packet>
1814EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet twoprod_low(
const Packet& x,
const Packet& y,
const Packet& xy) {
1815 return pmsub(x, y, xy);
1825template <
typename Packet>
1826EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void veltkamp_splitting(
const Packet& x, Packet& x_hi, Packet& x_lo) {
1827 typedef typename unpacket_traits<Packet>::type Scalar;
1828 constexpr int shift = (NumTraits<Scalar>::digits() + 1) / 2;
1829 const Scalar shift_scale = Scalar(uint64_t(1) << shift);
1830 const Packet gamma = pmul(pset1<Packet>(shift_scale + Scalar(1)), x);
1831 Packet rho = psub(x, gamma);
1832 x_hi = padd(rho, gamma);
1833 x_lo = psub(x, x_hi);
1840template <
typename Packet>
1841EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void twoprod(
const Packet& x,
const Packet& y, Packet& p_hi, Packet& p_lo) {
1842 Packet x_hi, x_lo, y_hi, y_lo;
1843 veltkamp_splitting(x, x_hi, x_lo);
1844 veltkamp_splitting(y, y_hi, y_lo);
1847 p_lo = pmadd(x_hi, y_hi, pnegate(p_hi));
1848 p_lo = pmadd(x_hi, y_lo, p_lo);
1849 p_lo = pmadd(x_lo, y_hi, p_lo);
1850 p_lo = pmadd(x_lo, y_lo, p_lo);
1855template <
typename Packet>
1856EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet twoprod_low(
const Packet& x,
const Packet& y,
const Packet& xy) {
1857 Packet x_hi, x_lo, y_hi, y_lo;
1858 veltkamp_splitting(x, x_hi, x_lo);
1859 veltkamp_splitting(y, y_hi, y_lo);
1861 Packet p_lo = pmadd(x_hi, y_hi, pnegate(xy));
1862 p_lo = pmadd(x_hi, y_lo, p_lo);
1863 p_lo = pmadd(x_lo, y_hi, p_lo);
1864 p_lo = pmadd(x_lo, y_lo, p_lo);
1876template <
typename Packet>
1877EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void twosum(
const Packet& x_hi,
const Packet& x_lo,
const Packet& y_hi,
1878 const Packet& y_lo, Packet& s_hi, Packet& s_lo) {
1879 const Packet x_greater_mask = pcmp_lt(pabs(y_hi), pabs(x_hi));
1880 Packet r_hi_1, r_lo_1;
1881 fast_twosum(x_hi, y_hi, r_hi_1, r_lo_1);
1882 Packet r_hi_2, r_lo_2;
1883 fast_twosum(y_hi, x_hi, r_hi_2, r_lo_2);
1884 const Packet r_hi = pselect(x_greater_mask, r_hi_1, r_hi_2);
1886 const Packet s1 = padd(padd(y_lo, r_lo_1), x_lo);
1887 const Packet s2 = padd(padd(x_lo, r_lo_2), y_lo);
1888 const Packet s = pselect(x_greater_mask, s1, s2);
1890 fast_twosum(r_hi, s, s_hi, s_lo);
1895template <
typename Packet>
1896EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void fast_twosum(
const Packet& x_hi,
const Packet& x_lo,
const Packet& y_hi,
1897 const Packet& y_lo, Packet& s_hi, Packet& s_lo) {
1899 fast_twosum(x_hi, y_hi, r_hi, r_lo);
1900 const Packet s = padd(padd(y_lo, r_lo), x_lo);
1901 fast_twosum(r_hi, s, s_hi, s_lo);
1907template <
typename Packet>
1908EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void fast_twosum(
const Packet& x,
const Packet& y_hi,
const Packet& y_lo,
1909 Packet& s_hi, Packet& s_lo) {
1911 fast_twosum(x, y_hi, r_hi, r_lo);
1912 const Packet s = padd(y_lo, r_lo);
1913 fast_twosum(r_hi, s, s_hi, s_lo);
1924template <
typename Packet>
1925EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void twoprod(
const Packet& x_hi,
const Packet& x_lo,
const Packet& y,
1926 Packet& p_hi, Packet& p_lo) {
1928 twoprod(x_hi, y, c_hi, c_lo1);
1929 const Packet c_lo2 = pmul(x_lo, y);
1931 fast_twosum(c_hi, c_lo2, t_hi, t_lo1);
1932 const Packet t_lo2 = padd(t_lo1, c_lo1);
1933 fast_twosum(t_hi, t_lo2, p_hi, p_lo);
1942template <
typename Packet>
1943EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void twoprod(
const Packet& x_hi,
const Packet& x_lo,
const Packet& y_hi,
1944 const Packet& y_lo, Packet& p_hi, Packet& p_lo) {
1945 Packet p_hi_hi, p_hi_lo;
1946 twoprod(x_hi, x_lo, y_hi, p_hi_hi, p_hi_lo);
1947 Packet p_lo_hi, p_lo_lo;
1948 twoprod(x_hi, x_lo, y_lo, p_lo_hi, p_lo_lo);
1949 fast_twosum(p_hi_hi, p_hi_lo, p_lo_hi, p_lo_lo, p_hi, p_lo);
1956template <
typename Packet>
1957EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void doubleword_div_fp(
const Packet& x_hi,
const Packet& x_lo,
const Packet& y,
1958 Packet& z_hi, Packet& z_lo) {
1959 const Packet t_hi = pdiv(x_hi, y);
1960 Packet pi_hi, pi_lo;
1961 twoprod(t_hi, y, pi_hi, pi_lo);
1962 const Packet delta_hi = psub(x_hi, pi_hi);
1963 const Packet delta_t = psub(delta_hi, pi_lo);
1964 const Packet delta = padd(delta_t, x_lo);
1965 const Packet t_lo = pdiv(delta, y);
1966 fast_twosum(t_hi, t_lo, z_hi, z_lo);
1970template <
typename Scalar>
1971struct accurate_log2 {
1972 template <
typename Packet>
1973 EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void operator()(
const Packet& x, Packet& log2_x_hi, Packet& log2_x_lo) {
1974 log2_x_hi = plog2(x);
1975 log2_x_lo = pzero(x);
1992struct accurate_log2<float> {
1993 template <
typename Packet>
1994 EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void operator()(
const Packet& z, Packet& log2_x_hi, Packet& log2_x_lo) {
1996 constexpr double kC0 = 1.442695041742110273474963832995854318141937255859375e+00;
1997 constexpr float kC0_hi =
static_cast<float>(kC0);
1998 constexpr float kC0_lo =
static_cast<float>(kC0 -
static_cast<double>(kC0_hi));
1999 const Packet c0_hi = pset1<Packet>(kC0_hi);
2000 const Packet c0_lo = pset1<Packet>(kC0_lo);
2002 constexpr double kC1 = -7.2134751588268664068692714863573201000690460205078125e-01;
2003 constexpr float kC1_hi =
static_cast<float>(kC1);
2004 constexpr float kC1_lo =
static_cast<float>(kC1 -
static_cast<double>(kC1_hi));
2005 const Packet c1_hi = pset1<Packet>(kC1_hi);
2006 const Packet c1_lo = pset1<Packet>(kC1_lo);
2008 constexpr float c[] = {
2009 9.7010828554630279541015625e-02, -1.6896486282348632812500000e-01, 1.7200836539268493652343750e-01,
2010 -1.7892272770404815673828125e-01, 2.0505344867706298828125000e-01, -2.4046677350997924804687500e-01,
2011 2.8857553005218505859375000e-01, -3.6067414283752441406250000e-01, 4.8089790344238281250000000e-01};
2015 const Packet one = pset1<Packet>(1.0f);
2016 const Packet x = psub(z, one);
2017 Packet p = ppolevl<Packet, 8>::run(x, c);
2021 twoprod(x, p, p_hi, p_lo);
2022 fast_twosum(c1_hi, c1_lo, p_hi, p_lo, p_hi, p_lo);
2023 twoprod(p_hi, p_lo, x, p_hi, p_lo);
2024 fast_twosum(c0_hi, c0_lo, p_hi, p_lo, p_hi, p_lo);
2026 twoprod(p_hi, p_lo, x, log2_x_hi, log2_x_lo);
2038struct accurate_log2<double> {
2039 template <
typename Packet>
2040 EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
void operator()(
const Packet& x, Packet& log2_x_hi, Packet& log2_x_lo) {
2062 const Packet q12 = pset1<Packet>(2.87074255468000586e-9);
2063 const Packet q10 = pset1<Packet>(2.38957980901884082e-8);
2064 const Packet q8 = pset1<Packet>(2.31032094540014656e-7);
2065 const Packet q6 = pset1<Packet>(2.27279857398537278e-6);
2066 const Packet q4 = pset1<Packet>(2.31271023278625638e-5);
2067 const Packet q2 = pset1<Packet>(2.47556738444535513e-4);
2068 const Packet q0 = pset1<Packet>(2.88543873228900172e-3);
2069 const Packet C_hi = pset1<Packet>(0.0400377511598501157);
2070 const Packet C_lo = pset1<Packet>(-4.77726582251425391e-19);
2071 const Packet one = pset1<Packet>(1.0);
2073 const Packet cst_2_log2e_hi = pset1<Packet>(2.88539008177792677);
2074 const Packet cst_2_log2e_lo = pset1<Packet>(4.07660016854549667e-17);
2078 twoprod(cst_2_log2e_hi, cst_2_log2e_lo, psub(x, one), t_hi, t_lo);
2081 doubleword_div_fp(t_hi, t_lo, padd(x, one), r_hi, r_lo);
2084 Packet r2_hi, r2_lo;
2085 twoprod(r_hi, r_lo, r_hi, r_lo, r2_hi, r2_lo);
2087 Packet r4_hi, r4_lo;
2088 twoprod(r2_hi, r2_lo, r2_hi, r2_lo, r4_hi, r4_lo);
2092 Packet q_even = pmadd(q12, r4_hi, q8);
2093 Packet q_odd = pmadd(q10, r4_hi, q6);
2094 q_even = pmadd(q_even, r4_hi, q4);
2095 q_odd = pmadd(q_odd, r4_hi, q2);
2096 q_even = pmadd(q_even, r4_hi, q0);
2097 Packet q = pmadd(q_odd, r2_hi, q_even);
2105 twoprod(r2_hi, r2_lo, q, p_hi, p_lo);
2107 Packet p1_hi, p1_lo;
2108 fast_twosum(C_hi, C_lo, p_hi, p_lo, p1_hi, p1_lo);
2110 Packet p2_hi, p2_lo;
2111 twoprod(r2_hi, r2_lo, p1_hi, p1_lo, p2_hi, p2_lo);
2113 Packet p3_hi, p3_lo;
2114 fast_twosum(one, p2_hi, p2_lo, p3_hi, p3_lo);
2117 twoprod(p3_hi, p3_lo, r_hi, r_lo, log2_x_hi, log2_x_lo);
2126template <
typename Packet>
2127EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_pow_impl(
const Packet& x,
const Packet& y) {
2128 typedef typename unpacket_traits<Packet>::type Scalar;
2131 Packet m_x = pfrexp(x, e_x);
2134 constexpr Scalar sqrt_half = Scalar(0.70710678118654752440);
2135 const Packet m_x_scale_mask = pcmp_lt(m_x, pset1<Packet>(sqrt_half));
2136 m_x = pselect(m_x_scale_mask, pmul(pset1<Packet>(Scalar(2)), m_x), m_x);
2137 e_x = pselect(m_x_scale_mask, psub(e_x, pset1<Packet>(Scalar(1))), e_x);
2140 Packet rx_hi, rx_lo;
2141 accurate_log2<Scalar>()(m_x, rx_hi, rx_lo);
2145 Packet f1_hi, f1_lo, f2_hi, f2_lo;
2146 twoprod(e_x, y, f1_hi, f1_lo);
2147 twoprod(rx_hi, rx_lo, y, f2_hi, f2_lo);
2155 fast_twosum(f1_hi, f1_lo, f2_hi, f2_lo, f_hi, f_lo);
2159 absolute_split(f_hi, n_z, r_z);
2160 r_z = padd(r_z, f_lo);
2162 absolute_split(r_z, n_r, r_z);
2163 n_z = padd(n_z, n_r);
2169 const Packet e_r = generic_exp2(r_z);
2173 constexpr Scalar kPldExpThresh = std::numeric_limits<Scalar>::max_exponent - 2;
2174 const Packet pldexp_fast_unsafe = pcmp_lt(pset1<Packet>(kPldExpThresh), pabs(n_z));
2175 if (predux_any(pldexp_fast_unsafe)) {
2176 return pldexp(e_r, n_z);
2178 return pldexp_fast(e_r, n_z);
2182template <
typename Packet>
2183EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS std::enable_if_t<!is_scalar<Packet>::value, Packet> generic_pow(
2184 const Packet& x,
const Packet& y) {
2185 typedef typename unpacket_traits<Packet>::type Scalar;
2187 const Packet cst_inf = pset1<Packet>(NumTraits<Scalar>::infinity());
2188 const Packet cst_zero = pset1<Packet>(Scalar(0));
2189 const Packet cst_one = pset1<Packet>(Scalar(1));
2190 const Packet cst_nan = pset1<Packet>(NumTraits<Scalar>::quiet_NaN());
2192 const Packet x_abs = pabs(x);
2193 Packet
pow = generic_pow_impl(x_abs, y);
2199 const Packet x_is_negative = pcmp_lt(x, cst_zero);
2200 const Packet x_is_zero = pcmp_eq(x, cst_zero);
2201 const Packet x_is_one = pcmp_eq(x, cst_one);
2202 const Packet x_has_signbit = psignbit(x);
2203 const Packet x_abs_gt_one = pcmp_lt(cst_one, x_abs);
2204 const Packet x_abs_is_inf = pcmp_eq(x_abs, cst_inf);
2207 const Packet y_abs = pabs(y);
2208 const Packet y_abs_is_inf = pcmp_eq(y_abs, cst_inf);
2209 const Packet y_is_negative = pcmp_lt(y, cst_zero);
2210 const Packet y_is_zero = pcmp_eq(y, cst_zero);
2211 const Packet y_is_one = pcmp_eq(y, cst_one);
2213 const Packet y_is_int = pandnot(pcmp_eq(pfloor(y), y), y_abs_is_inf);
2214 const Packet y_div_2 = pmul(y, pset1<Packet>(Scalar(0.5)));
2215 const Packet y_is_even = pcmp_eq(pround(y_div_2), y_div_2);
2216 const Packet y_is_odd_int = pandnot(y_is_int, y_is_even);
2218 constexpr Scalar huge_exponent =
2219 (NumTraits<Scalar>::max_exponent() * Scalar(EIGEN_LN2)) / NumTraits<Scalar>::epsilon();
2220 const Packet y_abs_is_huge = pcmp_le(pset1<Packet>(huge_exponent), y_abs);
2224 pow = pselect(pandnot(x_is_negative, y_is_int), cst_nan,
pow);
2235 pow = pselect(x_is_zero, pselect(y_is_negative, cst_inf, cst_zero),
pow);
2239 pow = pselect(pand(x_has_signbit, y_is_odd_int), pnegate(
pow),
pow);
2247 Packet inf_y_val = pselect(por(pand(y_is_negative, x_is_zero), pxor(y_is_negative, x_abs_gt_one)), cst_inf, cst_zero);
2248 inf_y_val = pselect(pcmp_eq(x, pset1<Packet>(Scalar(-1.0))), cst_one, inf_y_val);
2249 pow = pselect(y_abs_is_huge, inf_y_val,
pow);
2259 auto x_pos_inf_value = pselect(y_is_negative, cst_zero, cst_inf);
2260 auto x_neg_inf_value = pselect(y_is_odd_int, pnegate(x_pos_inf_value), x_pos_inf_value);
2261 pow = pselect(x_abs_is_inf, pselect(x_is_negative, x_neg_inf_value, x_pos_inf_value),
pow);
2264 pow = pselect(por(pisnan(x), pisnan(y)), cst_nan,
pow);
2269 pow = pselect(y_is_one, x, pselect(por(x_is_one, y_is_zero), cst_one,
pow));
2274template <
typename Scalar>
2275EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS std::enable_if_t<is_scalar<Scalar>::value, Scalar> generic_pow(
2276 const Scalar& x,
const Scalar& y) {
2277 return numext::pow(x, y);
2280namespace unary_pow {
2282template <typename ScalarExponent, bool IsInteger = NumTraits<ScalarExponent>::IsInteger>
2283struct exponent_helper {
2284 using safe_abs_type = ScalarExponent;
2285 static constexpr ScalarExponent one_half = ScalarExponent(0.5);
2287 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE ScalarExponent safe_abs(
const ScalarExponent&
exp) {
2288 return numext::abs(
exp);
2290 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
bool is_odd(
const ScalarExponent&
exp) {
2291 eigen_assert(((numext::isfinite)(
exp) &&
exp == numext::floor(
exp)) &&
"exp must be an integer");
2292 ScalarExponent exp_div_2 =
exp * one_half;
2293 ScalarExponent floor_exp_div_2 = numext::floor(exp_div_2);
2294 return exp_div_2 != floor_exp_div_2;
2296 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE ScalarExponent floor_div_two(
const ScalarExponent&
exp) {
2297 ScalarExponent exp_div_2 =
exp * one_half;
2298 return numext::floor(exp_div_2);
2302template <
typename ScalarExponent>
2303struct exponent_helper<ScalarExponent, true> {
2306 using safe_abs_type =
typename numext::get_integer_by_size<
sizeof(ScalarExponent)>::unsigned_type;
2307 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE safe_abs_type safe_abs(
const ScalarExponent&
exp) {
2308 ScalarExponent mask = numext::signbit(
exp);
2309 safe_abs_type result = safe_abs_type(
exp ^ mask);
2310 return result + safe_abs_type(ScalarExponent(1) & mask);
2312 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE
bool is_odd(
const safe_abs_type&
exp) {
2313 return exp % safe_abs_type(2) != safe_abs_type(0);
2315 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE safe_abs_type floor_div_two(
const safe_abs_type&
exp) {
2316 return exp >> safe_abs_type(1);
2320template <
typename Packet,
typename ScalarExponent,
2321 bool ReciprocateIfExponentIsNegative =
2322 !NumTraits<typename unpacket_traits<Packet>::type>::IsInteger && NumTraits<ScalarExponent>::IsSigned>
2324 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent& exponent) {
2325 using Scalar =
typename unpacket_traits<Packet>::type;
2326 const Packet cst_pos_one = pset1<Packet>(Scalar(1));
2327 return exponent < 0 ? pdiv(cst_pos_one, x) : x;
2331template <
typename Packet,
typename ScalarExponent>
2332struct reciprocate<Packet, ScalarExponent, false> {
2335 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent&) {
return x; }
2338template <
typename Packet,
typename ScalarExponent>
2339EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet int_pow(
const Packet& x,
const ScalarExponent& exponent) {
2340 using Scalar =
typename unpacket_traits<Packet>::type;
2341 using ExponentHelper = exponent_helper<ScalarExponent>;
2342 using AbsExponentType =
typename ExponentHelper::safe_abs_type;
2343 const Packet cst_pos_one = pset1<Packet>(Scalar(1));
2344 if (exponent == ScalarExponent(0))
return cst_pos_one;
2346 Packet result = reciprocate<Packet, ScalarExponent>::run(x, exponent);
2347 Packet y = cst_pos_one;
2348 AbsExponentType m = ExponentHelper::safe_abs(exponent);
2351 bool odd = ExponentHelper::is_odd(m);
2352 if (odd) y = pmul(y, result);
2353 result = pmul(result, result);
2354 m = ExponentHelper::floor_div_two(m);
2357 return pmul(y, result);
2360template <
typename Packet>
2361EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE std::enable_if_t<!is_scalar<Packet>::value, Packet> gen_pow(
2362 const Packet& x,
const typename unpacket_traits<Packet>::type& exponent) {
2363 const Packet exponent_packet = pset1<Packet>(exponent);
2364 return generic_pow_impl(x, exponent_packet);
2367template <
typename Scalar>
2368EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE std::enable_if_t<is_scalar<Scalar>::value, Scalar> gen_pow(
2369 const Scalar& x,
const Scalar& exponent) {
2370 return numext::pow(x, exponent);
2373template <
typename Packet,
typename ScalarExponent>
2374EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet handle_nonint_nonint_errors(
const Packet& x,
const Packet& powx,
2375 const ScalarExponent& exponent) {
2376 using Scalar =
typename unpacket_traits<Packet>::type;
2379 const Packet cst_pos_zero = pzero(x);
2380 const Packet cst_pos_one = pset1<Packet>(Scalar(1));
2381 const Packet cst_pos_inf = pset1<Packet>(NumTraits<Scalar>::infinity());
2382 const Packet cst_true = ptrue<Packet>(x);
2384 const bool exponent_is_not_fin = !(numext::isfinite)(exponent);
2385 const bool exponent_is_neg = exponent < ScalarExponent(0);
2386 const bool exponent_is_pos = exponent > ScalarExponent(0);
2388 const Packet exp_is_not_fin = exponent_is_not_fin ? cst_true : cst_pos_zero;
2389 const Packet exp_is_neg = exponent_is_neg ? cst_true : cst_pos_zero;
2390 const Packet exp_is_pos = exponent_is_pos ? cst_true : cst_pos_zero;
2391 const Packet exp_is_inf = pand(exp_is_not_fin, por(exp_is_neg, exp_is_pos));
2392 const Packet exp_is_nan = pandnot(exp_is_not_fin, por(exp_is_neg, exp_is_pos));
2394 const Packet x_is_le_zero = pcmp_le(x, cst_pos_zero);
2395 const Packet x_is_ge_zero = pcmp_le(cst_pos_zero, x);
2396 const Packet x_is_zero = pand(x_is_le_zero, x_is_ge_zero);
2398 const Packet abs_x = pabs(x);
2399 const Packet abs_x_is_le_one = pcmp_le(abs_x, cst_pos_one);
2400 const Packet abs_x_is_ge_one = pcmp_le(cst_pos_one, abs_x);
2401 const Packet abs_x_is_inf = pcmp_eq(abs_x, cst_pos_inf);
2402 const Packet abs_x_is_one = pand(abs_x_is_le_one, abs_x_is_ge_one);
2404 Packet pow_is_inf_if_exp_is_neg = por(x_is_zero, pand(abs_x_is_le_one, exp_is_inf));
2405 Packet pow_is_inf_if_exp_is_pos = por(abs_x_is_inf, pand(abs_x_is_ge_one, exp_is_inf));
2406 Packet pow_is_one = pand(abs_x_is_one, por(exp_is_inf, x_is_ge_zero));
2408 Packet result = powx;
2409 result = por(x_is_le_zero, result);
2410 result = pselect(pow_is_inf_if_exp_is_neg, pand(cst_pos_inf, exp_is_neg), result);
2411 result = pselect(pow_is_inf_if_exp_is_pos, pand(cst_pos_inf, exp_is_pos), result);
2412 result = por(exp_is_nan, result);
2413 result = pselect(pow_is_one, cst_pos_one, result);
2417template <
typename Packet,
typename ScalarExponent,
2418 std::enable_if_t<NumTraits<typename unpacket_traits<Packet>::type>::IsSigned,
bool> =
true>
2419EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet handle_negative_exponent(
const Packet& x,
const ScalarExponent& exponent) {
2420 using Scalar =
typename unpacket_traits<Packet>::type;
2426 const Packet cst_pos_one = pset1<Packet>(Scalar(1));
2427 const bool exponent_is_odd = exponent % ScalarExponent(2) != ScalarExponent(0);
2428 const Packet exp_is_odd = exponent_is_odd ? ptrue<Packet>(x) : pzero<Packet>(x);
2430 const Packet abs_x = pabs(x);
2431 const Packet abs_x_is_one = pcmp_eq(abs_x, cst_pos_one);
2433 Packet result = pselect(exp_is_odd, x, abs_x);
2434 result = pselect(abs_x_is_one, result, pzero<Packet>(x));
2438template <
typename Packet,
typename ScalarExponent,
2439 std::enable_if_t<!NumTraits<typename unpacket_traits<Packet>::type>::IsSigned,
bool> =
true>
2440EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet handle_negative_exponent(
const Packet& x,
const ScalarExponent&) {
2441 using Scalar =
typename unpacket_traits<Packet>::type;
2448 const Scalar pos_one = Scalar(1);
2450 const Packet cst_pos_one = pset1<Packet>(pos_one);
2452 const Packet x_is_one = pcmp_eq(x, cst_pos_one);
2454 return pand(x_is_one, x);
2459template <
typename Packet,
typename ScalarExponent,
2460 bool BaseIsIntegerType = NumTraits<typename unpacket_traits<Packet>::type>::IsInteger,
2461 bool ExponentIsIntegerType = NumTraits<ScalarExponent>::IsInteger,
2462 bool ExponentIsSigned = NumTraits<ScalarExponent>::IsSigned>
2463struct unary_pow_impl;
2465template <
typename Packet,
typename ScalarExponent,
bool ExponentIsSigned>
2466struct unary_pow_impl<Packet, ScalarExponent, false, false, ExponentIsSigned> {
2467 typedef typename unpacket_traits<Packet>::type Scalar;
2468 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent& exponent) {
2469 const bool exponent_is_integer = (numext::isfinite)(exponent) && numext::round(exponent) == exponent;
2470 if (exponent_is_integer) {
2474 bool use_repeated_squaring =
2475 (exponent <= ScalarExponent(7) && (!ExponentIsSigned || exponent >= ScalarExponent(-3)));
2476 return use_repeated_squaring ? unary_pow::int_pow(x, exponent) : generic_pow(x, pset1<Packet>(exponent));
2478 Packet result = unary_pow::gen_pow(x, exponent);
2479 result = unary_pow::handle_nonint_nonint_errors(x, result, exponent);
2485template <
typename Packet,
typename ScalarExponent,
bool ExponentIsSigned>
2486struct unary_pow_impl<Packet, ScalarExponent, false, true, ExponentIsSigned> {
2487 typedef typename unpacket_traits<Packet>::type Scalar;
2488 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent& exponent) {
2489 return unary_pow::int_pow(x, exponent);
2493template <
typename Packet,
typename ScalarExponent>
2494struct unary_pow_impl<Packet, ScalarExponent, true, true, true> {
2495 typedef typename unpacket_traits<Packet>::type Scalar;
2496 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent& exponent) {
2497 if (exponent < ScalarExponent(0)) {
2498 return unary_pow::handle_negative_exponent(x, exponent);
2500 return unary_pow::int_pow(x, exponent);
2505template <
typename Packet,
typename ScalarExponent>
2506struct unary_pow_impl<Packet, ScalarExponent, true, true, false> {
2507 typedef typename unpacket_traits<Packet>::type Scalar;
2508 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run(
const Packet& x,
const ScalarExponent& exponent) {
2509 return unary_pow::int_pow(x, exponent);
2524template <
typename Packet>
2525EIGEN_DEFINE_FUNCTION_ALLOWING_MULTIPLE_DEFINITIONS Packet generic_exp2(
const Packet& _x) {
2526 typedef typename unpacket_traits<Packet>::type Scalar;
2527 constexpr int max_exponent = std::numeric_limits<Scalar>::max_exponent;
2528 constexpr int digits = std::numeric_limits<Scalar>::digits;
2529 constexpr Scalar max_cap = Scalar(max_exponent + 1);
2530 constexpr Scalar min_cap = -Scalar(max_exponent + digits - 1);
2531 Packet x = pmax(pmin(_x, pset1<Packet>(max_cap)), pset1<Packet>(min_cap));
2533 twoprod(pset1<Packet>(Scalar(EIGEN_LN2)), x, p_hi, p_lo);
2534 Packet exp2_hi = pexp(p_hi);
2535 Packet exp2_lo = padd(pset1<Packet>(Scalar(1)), p_lo);
2536 return pmul(exp2_hi, exp2_lo);
2539template <
typename Packet>
2540EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_rint(
const Packet& a) {
2541 using Scalar =
typename unpacket_traits<Packet>::type;
2542 using IntType =
typename numext::get_integer_by_size<
sizeof(Scalar)>::signed_type;
2544 const IntType kLimit = IntType(1) << (NumTraits<Scalar>::digits() - 1);
2545 const Packet cst_limit = pset1<Packet>(
static_cast<Scalar
>(kLimit));
2546 Packet abs_a = pabs(a);
2547 Packet sign_a = pandnot(a, abs_a);
2548 Packet rint_a = padd(abs_a, cst_limit);
2550 EIGEN_OPTIMIZATION_BARRIER(rint_a);
2551 rint_a = psub(rint_a, cst_limit);
2552 rint_a = por(rint_a, sign_a);
2554 Packet mask = pcmp_lt(abs_a, cst_limit);
2555 Packet result = pselect(mask, rint_a, a);
2559template <
typename Packet>
2560EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_floor(
const Packet& a) {
2561 using Scalar =
typename unpacket_traits<Packet>::type;
2562 const Packet cst_1 = pset1<Packet>(Scalar(1));
2563 Packet rint_a = generic_rint(a);
2565 Packet mask = pcmp_lt(a, rint_a);
2566 Packet offset = pand(cst_1, mask);
2567 Packet result = psub(rint_a, offset);
2571template <
typename Packet>
2572EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_ceil(
const Packet& a) {
2573 using Scalar =
typename unpacket_traits<Packet>::type;
2574 const Packet cst_1 = pset1<Packet>(Scalar(1));
2575 const Packet sign_mask = pset1<Packet>(
static_cast<Scalar
>(-0.0));
2576 Packet rint_a = generic_rint(a);
2578 Packet mask = pcmp_lt(rint_a, a);
2579 Packet offset = pand(cst_1, mask);
2580 Packet result = padd(rint_a, offset);
2582 result = por(result, pand(sign_mask, a));
2586template <
typename Packet>
2587EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_trunc(
const Packet& a) {
2588 Packet abs_a = pabs(a);
2589 Packet sign_a = pandnot(a, abs_a);
2590 Packet floor_abs_a = generic_floor(abs_a);
2591 Packet result = por(floor_abs_a, sign_a);
2595template <
typename Packet>
2596EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet generic_round(
const Packet& a) {
2597 using Scalar =
typename unpacket_traits<Packet>::type;
2598 const Packet cst_half = pset1<Packet>(Scalar(0.5));
2599 const Packet cst_1 = pset1<Packet>(Scalar(1));
2600 Packet abs_a = pabs(a);
2601 Packet sign_a = pandnot(a, abs_a);
2602 Packet floor_abs_a = generic_floor(abs_a);
2603 Packet diff = psub(abs_a, floor_abs_a);
2604 Packet mask = pcmp_le(cst_half, diff);
2605 Packet offset = pand(cst_1, mask);
2606 Packet result = padd(floor_abs_a, offset);
2607 result = por(result, sign_a);
2611template <
typename Packet>
2612struct nearest_integer_packetop_impl<Packet, false, false> {
2613 using Scalar =
typename unpacket_traits<Packet>::type;
2614 static_assert(packet_traits<Scalar>::HasRound,
"Generic nearest integer functions are disabled for this type.");
2615 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_floor(
const Packet& x) {
return generic_floor(x); }
2616 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_ceil(
const Packet& x) {
return generic_ceil(x); }
2617 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_rint(
const Packet& x) {
return generic_rint(x); }
2618 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_round(
const Packet& x) {
return generic_round(x); }
2619 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_trunc(
const Packet& x) {
return generic_trunc(x); }
2622template <
typename Packet>
2623struct nearest_integer_packetop_impl<Packet, false, true> {
2624 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_floor(
const Packet& x) {
return x; }
2625 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_ceil(
const Packet& x) {
return x; }
2626 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_rint(
const Packet& x) {
return x; }
2627 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_round(
const Packet& x) {
return x; }
2628 static EIGEN_DEVICE_FUNC EIGEN_STRONG_INLINE Packet run_trunc(
const Packet& x) {
return x; }
const GlobalUnaryPowReturnType< Derived, ScalarExponent > pow(const Eigen::ArrayBase< Derived > &x, const ScalarExponent &exponent)
Namespace containing all symbols from the Eigen library.
Definition B01_Experimental.dox:1
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_cos_op< typename Derived::Scalar >, const Derived > cos(const Eigen::ArrayBase< Derived > &x)
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_expm1_op< typename Derived::Scalar >, const Derived > expm1(const Eigen::ArrayBase< Derived > &x)
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_exp_op< typename Derived::Scalar >, const Derived > exp(const Eigen::ArrayBase< Derived > &x)
const Eigen::CwiseUnaryOp< Eigen::internal::scalar_sign_op< typename Derived::Scalar >, const Derived > sign(const Eigen::ArrayBase< Derived > &x)