xref: /titanic_41/usr/src/lib/libm/common/complex/catanl.c (revision a9d3dcd5820128b4f34bf38f447e47aa95c004e8)
15b2ba9d3SPiotr Jasiukajtis /*
25b2ba9d3SPiotr Jasiukajtis  * CDDL HEADER START
35b2ba9d3SPiotr Jasiukajtis  *
45b2ba9d3SPiotr Jasiukajtis  * The contents of this file are subject to the terms of the
55b2ba9d3SPiotr Jasiukajtis  * Common Development and Distribution License (the "License").
65b2ba9d3SPiotr Jasiukajtis  * You may not use this file except in compliance with the License.
75b2ba9d3SPiotr Jasiukajtis  *
85b2ba9d3SPiotr Jasiukajtis  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
95b2ba9d3SPiotr Jasiukajtis  * or http://www.opensolaris.org/os/licensing.
105b2ba9d3SPiotr Jasiukajtis  * See the License for the specific language governing permissions
115b2ba9d3SPiotr Jasiukajtis  * and limitations under the License.
125b2ba9d3SPiotr Jasiukajtis  *
135b2ba9d3SPiotr Jasiukajtis  * When distributing Covered Code, include this CDDL HEADER in each
145b2ba9d3SPiotr Jasiukajtis  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
155b2ba9d3SPiotr Jasiukajtis  * If applicable, add the following below this CDDL HEADER, with the
165b2ba9d3SPiotr Jasiukajtis  * fields enclosed by brackets "[]" replaced with your own identifying
175b2ba9d3SPiotr Jasiukajtis  * information: Portions Copyright [yyyy] [name of copyright owner]
185b2ba9d3SPiotr Jasiukajtis  *
195b2ba9d3SPiotr Jasiukajtis  * CDDL HEADER END
205b2ba9d3SPiotr Jasiukajtis  */
215b2ba9d3SPiotr Jasiukajtis 
225b2ba9d3SPiotr Jasiukajtis /*
235b2ba9d3SPiotr Jasiukajtis  * Copyright 2011 Nexenta Systems, Inc.  All rights reserved.
245b2ba9d3SPiotr Jasiukajtis  */
255b2ba9d3SPiotr Jasiukajtis /*
265b2ba9d3SPiotr Jasiukajtis  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
275b2ba9d3SPiotr Jasiukajtis  * Use is subject to license terms.
285b2ba9d3SPiotr Jasiukajtis  */
295b2ba9d3SPiotr Jasiukajtis 
30*a9d3dcd5SRichard Lowe #pragma weak __catanl = catanl
315b2ba9d3SPiotr Jasiukajtis 
325b2ba9d3SPiotr Jasiukajtis /* INDENT OFF */
335b2ba9d3SPiotr Jasiukajtis /*
345b2ba9d3SPiotr Jasiukajtis  * ldcomplex catanl(ldcomplex z);
355b2ba9d3SPiotr Jasiukajtis  *
365b2ba9d3SPiotr Jasiukajtis  * Atan(z) return A + Bi where,
375b2ba9d3SPiotr Jasiukajtis  *            1
385b2ba9d3SPiotr Jasiukajtis  *	A = --- * atan2(2x, 1-x*x-y*y)
395b2ba9d3SPiotr Jasiukajtis  *            2
405b2ba9d3SPiotr Jasiukajtis  *
415b2ba9d3SPiotr Jasiukajtis  *            1      [ x*x + (y+1)*(y+1) ]   1               4y
425b2ba9d3SPiotr Jasiukajtis  *       B = --- log [ ----------------- ] = - log (1+ -----------------)
435b2ba9d3SPiotr Jasiukajtis  *            4      [ x*x + (y-1)*(y-1) ]   4         x*x + (y-1)*(y-1)
445b2ba9d3SPiotr Jasiukajtis  *
455b2ba9d3SPiotr Jasiukajtis  *                 2    16  3                         y
465b2ba9d3SPiotr Jasiukajtis  *         = t - 2t   + -- t  - ..., where t = -----------------
475b2ba9d3SPiotr Jasiukajtis  *                      3                      x*x + (y-1)*(y-1)
485b2ba9d3SPiotr Jasiukajtis  * Proof:
495b2ba9d3SPiotr Jasiukajtis  * Let w = atan(z=x+yi) = A + B i. Then tan(w) = z.
505b2ba9d3SPiotr Jasiukajtis  * Since sin(w) = (exp(iw)-exp(-iw))/(2i), cos(w)=(exp(iw)+exp(-iw))/(2),
515b2ba9d3SPiotr Jasiukajtis  * Let p = exp(iw), then z = tan(w) = ((p-1/p)/(p+1/p))/i, or
525b2ba9d3SPiotr Jasiukajtis  * iz = (p*p-1)/(p*p+1), or, after simplification,
535b2ba9d3SPiotr Jasiukajtis  *	p*p = (1+iz)/(1-iz)			            ... (1)
545b2ba9d3SPiotr Jasiukajtis  * LHS of (1) = exp(2iw) = exp(2i(A+Bi)) = exp(-2B)*exp(2iA)
555b2ba9d3SPiotr Jasiukajtis  *            = exp(-2B)*(cos(2A)+i*sin(2A))	            ... (2)
565b2ba9d3SPiotr Jasiukajtis  *              1-y+ix   (1-y+ix)*(1+y+ix)   1-x*x-y*y + 2xi
575b2ba9d3SPiotr Jasiukajtis  * RHS of (1) = ------ = ----------------- = --------------- ... (3)
585b2ba9d3SPiotr Jasiukajtis  *              1+y-ix    (1+y)**2 + x**2    (1+y)**2 + x**2
595b2ba9d3SPiotr Jasiukajtis  *
605b2ba9d3SPiotr Jasiukajtis  * Comparing the real and imaginary parts of (2) and (3), we have:
615b2ba9d3SPiotr Jasiukajtis  * 	cos(2A) : 1-x*x-y*y = sin(2A) : 2x
625b2ba9d3SPiotr Jasiukajtis  * and hence
635b2ba9d3SPiotr Jasiukajtis  *	tan(2A) = 2x/(1-x*x-y*y), or
645b2ba9d3SPiotr Jasiukajtis  *	A = 0.5 * atan2(2x, 1-x*x-y*y)	                    ... (4)
655b2ba9d3SPiotr Jasiukajtis  *
665b2ba9d3SPiotr Jasiukajtis  * For the imaginary part B, Note that |p*p| = exp(-2B), and
675b2ba9d3SPiotr Jasiukajtis  *	|1+iz|   |i-z|   hypot(x,(y-1))
685b2ba9d3SPiotr Jasiukajtis  *       |----| = |---| = --------------
695b2ba9d3SPiotr Jasiukajtis  *	|1-iz|   |i+z|   hypot(x,(y+1))
705b2ba9d3SPiotr Jasiukajtis  * Thus
715b2ba9d3SPiotr Jasiukajtis  *                 x*x + (y+1)*(y+1)
725b2ba9d3SPiotr Jasiukajtis  *	exp(4B) = -----------------, or
735b2ba9d3SPiotr Jasiukajtis  *                 x*x + (y-1)*(y-1)
745b2ba9d3SPiotr Jasiukajtis  *
755b2ba9d3SPiotr Jasiukajtis  *            1     [x^2+(y+1)^2]   1             4y
765b2ba9d3SPiotr Jasiukajtis  *       B =  - log [-----------] = - log(1+ -------------)  ... (5)
775b2ba9d3SPiotr Jasiukajtis  *            4     [x^2+(y-1)^2]   4         x^2+(y-1)^2
785b2ba9d3SPiotr Jasiukajtis  *
795b2ba9d3SPiotr Jasiukajtis  * QED.
805b2ba9d3SPiotr Jasiukajtis  *
815b2ba9d3SPiotr Jasiukajtis  * Note that: if catan( x, y) = ( u, v), then
825b2ba9d3SPiotr Jasiukajtis  *               catan(-x, y) = (-u, v)
835b2ba9d3SPiotr Jasiukajtis  *               catan( x,-y) = ( u,-v)
845b2ba9d3SPiotr Jasiukajtis  *
855b2ba9d3SPiotr Jasiukajtis  * Also,   catan(x,y) = -i*catanh(-y,x), or
865b2ba9d3SPiotr Jasiukajtis  *        catanh(x,y) =  i*catan(-y,x)
875b2ba9d3SPiotr Jasiukajtis  * So, if catanh(y,x) = (v,u), then catan(x,y) = -i*(-v,u) = (u,v), i.e.,
885b2ba9d3SPiotr Jasiukajtis  *         catan(x,y) = (u,v)
895b2ba9d3SPiotr Jasiukajtis  *
905b2ba9d3SPiotr Jasiukajtis  * EXCEPTION CASES (conform to ISO/IEC 9899:1999(E)):
915b2ba9d3SPiotr Jasiukajtis  *    catan( 0  , 0   ) =  (0    ,  0   )
925b2ba9d3SPiotr Jasiukajtis  *    catan( NaN, 0   ) =  (NaN  ,  0   )
935b2ba9d3SPiotr Jasiukajtis  *    catan( 0  , 1   ) =  (0    ,  +inf) with divide-by-zero
945b2ba9d3SPiotr Jasiukajtis  *    catan( inf, y   ) =  (pi/2 ,  0   ) for finite +y
955b2ba9d3SPiotr Jasiukajtis  *    catan( NaN, y   ) =  (NaN  ,  NaN ) with invalid for finite y != 0
965b2ba9d3SPiotr Jasiukajtis  *    catan( x  , inf ) =  (pi/2 ,  0   ) for finite +x
975b2ba9d3SPiotr Jasiukajtis  *    catan( inf, inf ) =  (pi/2 ,  0   )
985b2ba9d3SPiotr Jasiukajtis  *    catan( NaN, inf ) =  (NaN  ,  0   )
995b2ba9d3SPiotr Jasiukajtis  *    catan( x  , NaN ) =  (NaN  ,  NaN ) with invalid for finite x
1005b2ba9d3SPiotr Jasiukajtis  *    catan( inf, NaN ) =  (pi/2 ,  +-0 )
1015b2ba9d3SPiotr Jasiukajtis  */
1025b2ba9d3SPiotr Jasiukajtis /* INDENT ON */
1035b2ba9d3SPiotr Jasiukajtis 
1045b2ba9d3SPiotr Jasiukajtis #include "libm.h"	/* atan2l/atanl/fabsl/isinfl/iszerol/log1pl/logl */
1055b2ba9d3SPiotr Jasiukajtis #include "complex_wrapper.h"
1065b2ba9d3SPiotr Jasiukajtis #include "longdouble.h"
1075b2ba9d3SPiotr Jasiukajtis 
1085b2ba9d3SPiotr Jasiukajtis /* INDENT OFF */
1095b2ba9d3SPiotr Jasiukajtis static const long double
1105b2ba9d3SPiotr Jasiukajtis zero = 0.0L,
1115b2ba9d3SPiotr Jasiukajtis one = 1.0L,
1125b2ba9d3SPiotr Jasiukajtis two = 2.0L,
1135b2ba9d3SPiotr Jasiukajtis half = 0.5L,
1145b2ba9d3SPiotr Jasiukajtis ln2 = 6.931471805599453094172321214581765680755e-0001L,
1155b2ba9d3SPiotr Jasiukajtis pi_2 = 1.570796326794896619231321691639751442098584699687552910487472L,
1165b2ba9d3SPiotr Jasiukajtis #if defined(__x86)
1175b2ba9d3SPiotr Jasiukajtis E = 2.910383045673370361328125000000000000000e-11L,	/* 2**-35 */
1185b2ba9d3SPiotr Jasiukajtis Einv = 3.435973836800000000000000000000000000000e+10L;	/* 2**+35 */
1195b2ba9d3SPiotr Jasiukajtis #else
1205b2ba9d3SPiotr Jasiukajtis E = 8.673617379884035472059622406959533691406e-19L,	/* 2**-60 */
1215b2ba9d3SPiotr Jasiukajtis Einv = 1.152921504606846976000000000000000000000e18L;	/* 2**+60 */
1225b2ba9d3SPiotr Jasiukajtis #endif
1235b2ba9d3SPiotr Jasiukajtis /* INDENT ON */
1245b2ba9d3SPiotr Jasiukajtis 
1255b2ba9d3SPiotr Jasiukajtis ldcomplex
catanl(ldcomplex z)1265b2ba9d3SPiotr Jasiukajtis catanl(ldcomplex z) {
1275b2ba9d3SPiotr Jasiukajtis 	ldcomplex ans;
1285b2ba9d3SPiotr Jasiukajtis 	long double x, y, t1, ax, ay, t;
1295b2ba9d3SPiotr Jasiukajtis 	int hx, hy, ix, iy;
1305b2ba9d3SPiotr Jasiukajtis 
1315b2ba9d3SPiotr Jasiukajtis 	x = LD_RE(z);
1325b2ba9d3SPiotr Jasiukajtis 	y = LD_IM(z);
1335b2ba9d3SPiotr Jasiukajtis 	ax = fabsl(x);
1345b2ba9d3SPiotr Jasiukajtis 	ay = fabsl(y);
1355b2ba9d3SPiotr Jasiukajtis 	hx = HI_XWORD(x);
1365b2ba9d3SPiotr Jasiukajtis 	hy = HI_XWORD(y);
1375b2ba9d3SPiotr Jasiukajtis 	ix = hx & 0x7fffffff;
1385b2ba9d3SPiotr Jasiukajtis 	iy = hy & 0x7fffffff;
1395b2ba9d3SPiotr Jasiukajtis 
1405b2ba9d3SPiotr Jasiukajtis 	/* x is inf or NaN */
1415b2ba9d3SPiotr Jasiukajtis 	if (ix >= 0x7fff0000) {
1425b2ba9d3SPiotr Jasiukajtis 		if (isinfl(x)) {
1435b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = pi_2;
1445b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = zero;
1455b2ba9d3SPiotr Jasiukajtis 		} else {
1465b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = x + x;
1475b2ba9d3SPiotr Jasiukajtis 			if (iszerol(y) || (isinfl(y)))
1485b2ba9d3SPiotr Jasiukajtis 				LD_IM(ans) = zero;
1495b2ba9d3SPiotr Jasiukajtis 			else
1505b2ba9d3SPiotr Jasiukajtis 				LD_IM(ans) = (fabsl(y) - ay) / (fabsl(y) - ay);
1515b2ba9d3SPiotr Jasiukajtis 		}
1525b2ba9d3SPiotr Jasiukajtis 	} else if (iy >= 0x7fff0000) {
1535b2ba9d3SPiotr Jasiukajtis 		/* y is inf or NaN */
1545b2ba9d3SPiotr Jasiukajtis 		if (isinfl(y)) {
1555b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = pi_2;
1565b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = zero;
1575b2ba9d3SPiotr Jasiukajtis 		} else {
1585b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = (fabsl(x) - ax) / (fabsl(x) - ax);
1595b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = y;
1605b2ba9d3SPiotr Jasiukajtis 		}
1615b2ba9d3SPiotr Jasiukajtis 	} else if (iszerol(x)) {
1625b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
1635b2ba9d3SPiotr Jasiukajtis 		/*
1645b2ba9d3SPiotr Jasiukajtis 		 * x = 0
1655b2ba9d3SPiotr Jasiukajtis 		 *      1                            1
1665b2ba9d3SPiotr Jasiukajtis 		 * A = --- * atan2(2x, 1-x*x-y*y) = --- atan2(0,1-|y|)
1675b2ba9d3SPiotr Jasiukajtis 		 *      2                            2
1685b2ba9d3SPiotr Jasiukajtis 		 *
1695b2ba9d3SPiotr Jasiukajtis 		 *     1     [ (y+1)*(y+1) ]   1          2      1         2y
1705b2ba9d3SPiotr Jasiukajtis 		 * B = - log [ ----------- ] = - log (1+ ---) or - log(1+ ----)
1715b2ba9d3SPiotr Jasiukajtis 		 *     4     [ (y-1)*(y-1) ]   2         y-1     2         1-y
1725b2ba9d3SPiotr Jasiukajtis 		 */
1735b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
1745b2ba9d3SPiotr Jasiukajtis 		t = one - ay;
1755b2ba9d3SPiotr Jasiukajtis 		if (ay == one) {
1765b2ba9d3SPiotr Jasiukajtis 			/* y=1: catan(0,1)=(0,+inf) with 1/0 signal */
1775b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = ay / ax;
1785b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = zero;
1795b2ba9d3SPiotr Jasiukajtis 		} else if (ay > one) {	/* y>1 */
1805b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = half * log1pl(two / (-t));
1815b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = pi_2;
1825b2ba9d3SPiotr Jasiukajtis 		} else {		/* y<1 */
1835b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = half * log1pl((ay + ay) / t);
1845b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = zero;
1855b2ba9d3SPiotr Jasiukajtis 		}
1865b2ba9d3SPiotr Jasiukajtis 	} else if (ay < E * (one + ax)) {
1875b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
1885b2ba9d3SPiotr Jasiukajtis 		/*
1895b2ba9d3SPiotr Jasiukajtis 		 * Tiny y (relative to 1+|x|)
1905b2ba9d3SPiotr Jasiukajtis 		 *     |y| < E*(1+|x|)
1915b2ba9d3SPiotr Jasiukajtis 		 * where E=2**-29, -35, -60 for double, extended, quad precision
1925b2ba9d3SPiotr Jasiukajtis 		 *
1935b2ba9d3SPiotr Jasiukajtis 		 *     1                         [x<=1:   atan(x)
1945b2ba9d3SPiotr Jasiukajtis 		 * A = - * atan2(2x,1-x*x-y*y) ~ [      1                 1+x
1955b2ba9d3SPiotr Jasiukajtis 		 *     2                         [x>=1: - atan2(2,(1-x)*(-----))
1965b2ba9d3SPiotr Jasiukajtis 		 *                                      2                  x
1975b2ba9d3SPiotr Jasiukajtis 		 *
1985b2ba9d3SPiotr Jasiukajtis 		 *                               y/x
1995b2ba9d3SPiotr Jasiukajtis 		 * B ~ t*(1-2t), where t = ----------------- is tiny
2005b2ba9d3SPiotr Jasiukajtis 		 *                         x + (y-1)*(y-1)/x
2015b2ba9d3SPiotr Jasiukajtis 		 *
2025b2ba9d3SPiotr Jasiukajtis 		 *                           y
2035b2ba9d3SPiotr Jasiukajtis 		 * (when x < 2**-60, t = ----------- )
2045b2ba9d3SPiotr Jasiukajtis 		 *                       (y-1)*(y-1)
2055b2ba9d3SPiotr Jasiukajtis 		 */
2065b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
2075b2ba9d3SPiotr Jasiukajtis 		if (ay == zero)
2085b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = ay;
2095b2ba9d3SPiotr Jasiukajtis 		else {
2105b2ba9d3SPiotr Jasiukajtis 			t1 = ay - one;
2115b2ba9d3SPiotr Jasiukajtis 			if (ix < 0x3fc30000)
2125b2ba9d3SPiotr Jasiukajtis 				t = ay / (t1 * t1);
2135b2ba9d3SPiotr Jasiukajtis 			else if (ix > 0x403b0000)
2145b2ba9d3SPiotr Jasiukajtis 				t = (ay / ax) / ax;
2155b2ba9d3SPiotr Jasiukajtis 			else
2165b2ba9d3SPiotr Jasiukajtis 				t = ay / (ax * ax + t1 * t1);
2175b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = t * (one - two * t);
2185b2ba9d3SPiotr Jasiukajtis 		}
2195b2ba9d3SPiotr Jasiukajtis 		if (ix < 0x3fff0000)
2205b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = atanl(ax);
2215b2ba9d3SPiotr Jasiukajtis 		else
2225b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = half * atan2l(two, (one - ax) * (one +
2235b2ba9d3SPiotr Jasiukajtis 				one / ax));
2245b2ba9d3SPiotr Jasiukajtis 
2255b2ba9d3SPiotr Jasiukajtis 	} else if (ay > Einv * (one + ax)) {
2265b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
2275b2ba9d3SPiotr Jasiukajtis 		/*
2285b2ba9d3SPiotr Jasiukajtis 		 * Huge y relative to 1+|x|
2295b2ba9d3SPiotr Jasiukajtis 		 *     |y| > Einv*(1+|x|), where Einv~2**(prec/2+3),
2305b2ba9d3SPiotr Jasiukajtis 		 *      1
2315b2ba9d3SPiotr Jasiukajtis 		 * A ~ --- * atan2(2x, -y*y) ~ pi/2
2325b2ba9d3SPiotr Jasiukajtis 		 *      2
2335b2ba9d3SPiotr Jasiukajtis 		 *                               y
2345b2ba9d3SPiotr Jasiukajtis 		 * B ~ t*(1-2t), where t = --------------- is tiny
2355b2ba9d3SPiotr Jasiukajtis 		 *                          (y-1)*(y-1)
2365b2ba9d3SPiotr Jasiukajtis 		 */
2375b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
2385b2ba9d3SPiotr Jasiukajtis 		LD_RE(ans) = pi_2;
2395b2ba9d3SPiotr Jasiukajtis 		t = (ay / (ay - one)) / (ay - one);
2405b2ba9d3SPiotr Jasiukajtis 		LD_IM(ans) = t * (one - (t + t));
2415b2ba9d3SPiotr Jasiukajtis 	} else if (ay == one) {
2425b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
2435b2ba9d3SPiotr Jasiukajtis 		/*
2445b2ba9d3SPiotr Jasiukajtis 		 * y=1
2455b2ba9d3SPiotr Jasiukajtis 		 *     1                      1
2465b2ba9d3SPiotr Jasiukajtis 		 * A = - * atan2(2x, -x*x) = --- atan2(2,-x)
2475b2ba9d3SPiotr Jasiukajtis 		 *     2                      2
2485b2ba9d3SPiotr Jasiukajtis 		 *
2495b2ba9d3SPiotr Jasiukajtis 		 *     1     [ x*x+4]   1          4     [ 0.5(log2-logx) if
2505b2ba9d3SPiotr Jasiukajtis 		 * B = - log [ -----] = - log (1+ ---) = [ |x|<E, else 0.25*
2515b2ba9d3SPiotr Jasiukajtis 		 *     4     [  x*x ]   4         x*x    [ log1p((2/x)*(2/x))
2525b2ba9d3SPiotr Jasiukajtis 		 */
2535b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
2545b2ba9d3SPiotr Jasiukajtis 		LD_RE(ans) = half * atan2l(two, -ax);
2555b2ba9d3SPiotr Jasiukajtis 		if (ax < E)
2565b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = half * (ln2 - logl(ax));
2575b2ba9d3SPiotr Jasiukajtis 		else {
2585b2ba9d3SPiotr Jasiukajtis 			t = two / ax;
2595b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = 0.25L * log1pl(t * t);
2605b2ba9d3SPiotr Jasiukajtis 		}
2615b2ba9d3SPiotr Jasiukajtis 	} else if (ax > Einv * Einv) {
2625b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
2635b2ba9d3SPiotr Jasiukajtis 		/*
2645b2ba9d3SPiotr Jasiukajtis 		 * Huge x:
2655b2ba9d3SPiotr Jasiukajtis 		 * when |x| > 1/E^2,
2665b2ba9d3SPiotr Jasiukajtis 		 *      1                           pi
2675b2ba9d3SPiotr Jasiukajtis 		 * A ~ --- * atan2(2x, -x*x-y*y) ~ ---
2685b2ba9d3SPiotr Jasiukajtis 		 *      2                           2
2695b2ba9d3SPiotr Jasiukajtis 		 *                               y                 y/x
2705b2ba9d3SPiotr Jasiukajtis 		 * B ~ t*(1-2t), where t = --------------- = (-------------- )/x
2715b2ba9d3SPiotr Jasiukajtis 		 *                         x*x+(y-1)*(y-1)     1+((y-1)/x)^2
2725b2ba9d3SPiotr Jasiukajtis 		 */
2735b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
2745b2ba9d3SPiotr Jasiukajtis 		LD_RE(ans) = pi_2;
2755b2ba9d3SPiotr Jasiukajtis 		t = ((ay / ax) / (one + ((ay - one) / ax) * ((ay - one) /
2765b2ba9d3SPiotr Jasiukajtis 			ax))) / ax;
2775b2ba9d3SPiotr Jasiukajtis 		LD_IM(ans) = t * (one - (t + t));
2785b2ba9d3SPiotr Jasiukajtis 	} else if (ax < E * E * E * E) {
2795b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
2805b2ba9d3SPiotr Jasiukajtis 		/*
2815b2ba9d3SPiotr Jasiukajtis 		 * Tiny x:
2825b2ba9d3SPiotr Jasiukajtis 		 * when |x| < E^4,  (note that y != 1)
2835b2ba9d3SPiotr Jasiukajtis 		 *      1                            1
2845b2ba9d3SPiotr Jasiukajtis 		 * A = --- * atan2(2x, 1-x*x-y*y) ~ --- * atan2(2x,1-y*y)
2855b2ba9d3SPiotr Jasiukajtis 		 *      2                            2
2865b2ba9d3SPiotr Jasiukajtis 		 *
2875b2ba9d3SPiotr Jasiukajtis 		 *     1     [ (y+1)*(y+1) ]   1          2      1         2y
2885b2ba9d3SPiotr Jasiukajtis 		 * B = - log [ ----------- ] = - log (1+ ---) or - log(1+ ----)
2895b2ba9d3SPiotr Jasiukajtis 		 *     4     [ (y-1)*(y-1) ]   2         y-1     2         1-y
2905b2ba9d3SPiotr Jasiukajtis 		 */
2915b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
2925b2ba9d3SPiotr Jasiukajtis 		LD_RE(ans) = half * atan2l(ax + ax, (one - ay) * (one + ay));
2935b2ba9d3SPiotr Jasiukajtis 		if (ay > one)	/* y>1 */
2945b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = half * log1pl(two / (ay - one));
2955b2ba9d3SPiotr Jasiukajtis 		else		/* y<1 */
2965b2ba9d3SPiotr Jasiukajtis 			LD_IM(ans) = half * log1pl((ay + ay) / (one - ay));
2975b2ba9d3SPiotr Jasiukajtis 	} else {
2985b2ba9d3SPiotr Jasiukajtis 		/* INDENT OFF */
2995b2ba9d3SPiotr Jasiukajtis 		/*
3005b2ba9d3SPiotr Jasiukajtis 		 * normal x,y
3015b2ba9d3SPiotr Jasiukajtis 		 *      1
3025b2ba9d3SPiotr Jasiukajtis 		 * A = --- * atan2(2x, 1-x*x-y*y)
3035b2ba9d3SPiotr Jasiukajtis 		 *      2
3045b2ba9d3SPiotr Jasiukajtis 		 *
3055b2ba9d3SPiotr Jasiukajtis 		 *     1     [ x*x+(y+1)*(y+1) ]   1               4y
3065b2ba9d3SPiotr Jasiukajtis 		 * B = - log [ --------------- ] = - log (1+ -----------------)
3075b2ba9d3SPiotr Jasiukajtis 		 *     4     [ x*x+(y-1)*(y-1) ]   4         x*x + (y-1)*(y-1)
3085b2ba9d3SPiotr Jasiukajtis 		 */
3095b2ba9d3SPiotr Jasiukajtis 		/* INDENT ON */
3105b2ba9d3SPiotr Jasiukajtis 		t = one - ay;
3115b2ba9d3SPiotr Jasiukajtis 		if (iy >= 0x3ffe0000 && iy < 0x40000000) {
3125b2ba9d3SPiotr Jasiukajtis 			/* y close to 1 */
3135b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = half * (atan2l((ax + ax), (t * (one +
3145b2ba9d3SPiotr Jasiukajtis 				ay) - ax * ax)));
3155b2ba9d3SPiotr Jasiukajtis 		} else if (ix >= 0x3ffe0000 && ix < 0x40000000) {
3165b2ba9d3SPiotr Jasiukajtis 			/* x close to 1 */
3175b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = half * atan2l((ax + ax), ((one - ax) *
3185b2ba9d3SPiotr Jasiukajtis 				(one + ax) - ay * ay));
3195b2ba9d3SPiotr Jasiukajtis 		} else
3205b2ba9d3SPiotr Jasiukajtis 			LD_RE(ans) = half * atan2l((ax + ax), ((one - ax *
3215b2ba9d3SPiotr Jasiukajtis 				ax) - ay * ay));
3225b2ba9d3SPiotr Jasiukajtis 		LD_IM(ans) = 0.25L * log1pl((4.0L * ay) / (ax * ax + t * t));
3235b2ba9d3SPiotr Jasiukajtis 	}
3245b2ba9d3SPiotr Jasiukajtis 	if (hx < 0)
3255b2ba9d3SPiotr Jasiukajtis 		LD_RE(ans) = -LD_RE(ans);
3265b2ba9d3SPiotr Jasiukajtis 	if (hy < 0)
3275b2ba9d3SPiotr Jasiukajtis 		LD_IM(ans) = -LD_IM(ans);
3285b2ba9d3SPiotr Jasiukajtis 	return (ans);
3295b2ba9d3SPiotr Jasiukajtis }
330