e_hypotf.c 2.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. /* e_hypotf.c -- float version of e_hypot.c.
  2. * Conversion to float by Ian Lance Taylor, Cygnus Support, ian@cygnus.com.
  3. */
  4. /*
  5. * ====================================================
  6. * Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
  7. *
  8. * Developed at SunPro, a Sun Microsystems, Inc. business.
  9. * Permission to use, copy, modify, and distribute this
  10. * software is freely granted, provided that this notice
  11. * is preserved.
  12. * ====================================================
  13. */
  14. #include <sys/cdefs.h>
  15. __FBSDID("$FreeBSD$");
  16. #include "math.h"
  17. #include "math_private.h"
  18. float
  19. __ieee754_hypotf(float x, float y)
  20. {
  21. float a,b,t1,t2,y1,y2,w;
  22. int32_t j,k,ha,hb;
  23. GET_FLOAT_WORD(ha,x);
  24. ha &= 0x7fffffff;
  25. GET_FLOAT_WORD(hb,y);
  26. hb &= 0x7fffffff;
  27. if(hb > ha) {a=y;b=x;j=ha; ha=hb;hb=j;} else {a=x;b=y;}
  28. a = fabsf(a);
  29. b = fabsf(b);
  30. if((ha-hb)>0xf000000) {return a+b;} /* x/y > 2**30 */
  31. k=0;
  32. if(ha > 0x58800000) { /* a>2**50 */
  33. if(ha >= 0x7f800000) { /* Inf or NaN */
  34. /* Use original arg order iff result is NaN; quieten sNaNs. */
  35. w = fabsf(x+0.0F)-fabsf(y+0.0F);
  36. if(ha == 0x7f800000) w = a;
  37. if(hb == 0x7f800000) w = b;
  38. return w;
  39. }
  40. /* scale a and b by 2**-68 */
  41. ha -= 0x22000000; hb -= 0x22000000; k += 68;
  42. SET_FLOAT_WORD(a,ha);
  43. SET_FLOAT_WORD(b,hb);
  44. }
  45. if(hb < 0x26800000) { /* b < 2**-50 */
  46. if(hb <= 0x007fffff) { /* subnormal b or 0 */
  47. if(hb==0) return a;
  48. SET_FLOAT_WORD(t1,0x7e800000); /* t1=2^126 */
  49. b *= t1;
  50. a *= t1;
  51. k -= 126;
  52. } else { /* scale a and b by 2^68 */
  53. ha += 0x22000000; /* a *= 2^68 */
  54. hb += 0x22000000; /* b *= 2^68 */
  55. k -= 68;
  56. SET_FLOAT_WORD(a,ha);
  57. SET_FLOAT_WORD(b,hb);
  58. }
  59. }
  60. /* medium size a and b */
  61. w = a-b;
  62. if (w>b) {
  63. SET_FLOAT_WORD(t1,ha&0xfffff000);
  64. t2 = a-t1;
  65. w = __ieee754_sqrtf(t1*t1-(b*(-b)-t2*(a+t1)));
  66. } else {
  67. a = a+a;
  68. SET_FLOAT_WORD(y1,hb&0xfffff000);
  69. y2 = b - y1;
  70. SET_FLOAT_WORD(t1,(ha+0x00800000)&0xfffff000);
  71. t2 = a - t1;
  72. w = __ieee754_sqrtf(t1*y1-(w*(-w)-(t1*y2+t2*b)));
  73. }
  74. if(k!=0) {
  75. SET_FLOAT_WORD(t1,0x3f800000+(k<<23));
  76. return t1*w;
  77. } else return w;
  78. }