Use of C99 extra long double math functions after r236148
    Stephen Montgomery-Smith 
    stephen at missouri.edu
       
    Sun Aug 12 23:06:37 UTC 2012
    
    
  
I went on a long road trip yesterday, so I didn't get any code written, 
but I did have a lot of thoughts about clog and casinh.
First, the naive formula (here z=x+I*y)
clog(z) = cpack(log(hypot(x,y)),atan2(x,y))
is going to work in a lot more edge cases then one might expect.  This 
is because hypot and atan2, especially atan2, already do a rather good 
job getting the edge cases right.  I am thinking in particular of when x 
or y are 0 or -0, or one of them is infinity or -infinity.
So writing this code will be quite a bit easier than I was expecting. 
It looks like I will just have to worry about when both x and y are 
infinite, or when one or both of them are nan.
Next, concerning casinh:
On 07/17/2012 06:13 AM, Bruce Evans wrote:
> I translated this to pari.  There was a sign error for the log() term
> (assuming that pari asinh() has the same branch cuts as C99):
I couldn't spot which sign error Bruce had changed.  However I expect it 
has something to do with what happens when x=0 and fabs(y)>1.  This is 
the reasonable choice of the branch cut.  What I think the value should 
be is
casinh(z) = cpack(
signum(x)*sqrt(fabs(y)+sqrt(y^2-1)),
signum(y)*PI)
where the value of signum(x) depends on whether x is 0 or -0.
(I might add that I checked against the Mathematica ArcSinh function, 
and this does NOT follow the above rule.  But the document Steve pointed 
me to says that
casinh(conj(z)) = conj(casinh(z))
which means that we cannot follow the Mathematica conventions.)
> The most obvious immediate difficulty in translating the above into C is
> that y*y and z*z may overflow when the result shouldn't.
This will be a lot easier than I originally expected.  When we are in 
conditions when overflow might occur, we can simply make the approximations
sqrt(y*y-1) = y
csqrt(z*z+1) = signum(x)*z
because in floating point arithmetic, these will not be approximations, 
but true exactly.  And I am thinking that the test I will use for when 
to use these approximations will be (y==y+1) and (z==z+1) respectively. 
  (I would use (z*z==z*z+1) but that test has the overflow problem.)
Finally, I want to tell you guys that the reason I used the code:
if (x>0)
    return clog(z+csqrt(z*z+1));
  else
    return -clog(-z+csqrt(z*z+1));
is this.  Both formulas are mathematically exactly the same.  This is 
true even if one takes into account the branch cuts for csqrt and clog. 
  The difference between the two formulas is numerical errors.  For 
example, if x<0 and z has very large magnitude, then csqrt(z*z+1) will 
be very close to -z.  In fact in floating point arithmetic, if the 
magnitude of z is sufficiently large, they will be the same.
However, as I am typing this, I realize that the code should really be
if (w!=z+1) {
   w = z*z+1;
   if (signum(creal(w))==1)
     return clog(z+csqrt(w));
   else
     return -clog(-z+csqrt(w));
} else /* if (z==z+1) */ {
   if (x>0)
     return clog(2*z);
   else
     return -clog(-2*z);
}
where the signum function is defined so that signum(0)==1 and signum(-0)=-1.
Next: cacosh and cacos.  I had presumed the formula
cacosh(z) = I*cacos(z)
which can be true depending on how one defines the branch cuts.  But 
this formula won't satisfy the C99 standard which mandates
cacosh(conj(z)) = conj(cacosh(z))
That is why in earlier posts I thought there was a mistake in the online 
documentation, and the range of outputs of cacosh should satisfy 
imaginary part in [0,pi] rather than [-pi,pi].
Anyway, I am just posting this update so that you see I am thinking 
about it.  I might get the project done in days, but it might also be 
months.  It depends upon how much other stuff I have going on.
    
    
More information about the freebsd-numerics
mailing list