rts-real32.c

     
   1  //! @file rts-real32.c
   2  //! @author (see below)
   3  //
   4  //! @section Copyright
   5  //
   6  // This file is part of VIF - vintage FORTRAN compiler.
   7  // Copyright 2020-2025 J. Marcel van der Veer <algol68g@xs4all.nl>.
   8  //
   9  //! @section License
  10  //
  11  // This program is free software; you can redistribute it and/or modify it 
  12  // under the terms of the GNU General Public License as published by the 
  13  // Free Software Foundation; either version 3 of the License, or 
  14  // (at your option) any later version.
  15  //
  16  // This program is distributed in the hope that it will be useful, but 
  17  // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY 
  18  // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 
  19  // more details. You should have received a copy of the GNU General Public 
  20  // License along with this program. If not, see <http://www.gnu.org/licenses/>.
  21  
  22  //! @section Synopsis
  23  //!
  24  //! Runtime support for REAL*32 and COMPLEX*64.
  25  
  26  // The code is based on the HPA Library, available from:
  27  //   <http://download-mirror.savannah.gnu.org/releases/hpalib/>
  28  //
  29  //   Copyright (C) 2000 Daniel A. Atkinson <DanAtk@aol.com>
  30  //   Copyright (C) 2004 Ivano Primi <ivprimi@libero.it>  
  31  //   Copyright (C) 2022 Marcel van der Veer <algol68g@xs4all.nl> - VIF version.
  32  //
  33  // HPA code is adapted to work with VIF to implement REAL*32 and COMPLEX*64.
  34  // HPA was choosen since it stores a 256-bit float as a 256-bit struct, which
  35  // is convenient for FORTRAN.
  36  //
  37  // The HPA Library is free software; you can redistribute it and/or
  38  // modify it under the terms of the GNU Lesser General Public
  39  // License as published by the Free Software Foundation; either
  40  // version 2.1 of the License, or (at your option) any later version.
  41  //
  42  // The HPA Library is distributed in the hope that it will be useful,
  43  // but WITHOUT ANY WARRANTY; without even the implied warranty of
  44  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  45  // Lesser General Public License for more details.
  46  //
  47  // You should have received a copy of the GNU Lesser General Public
  48  // License along with the HPA Library; if not, write to the Free
  49  // Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
  50  // 02110-1301 USA.
  51  
  52  // The functions forming the HPA library are all implemented in a portable 
  53  // fashion in the C language. The IEEE 754 standard for floating point 
  54  // hardware and software is assumed in the PC/Unix version of this library. 
  55  // A REAL*32 number is represented as a combination of the following elements:
  56  //
  57  //   sign bit(s): 0 -> positive, 1 -> negative ;
  58  //   exponent(e): 15-bit biased integer (bias=16383) ;
  59  //   mantissa(m): 15 words of 16 bit length *including* leading 1. 
  60  //
  61  // The range of representable numbers is then given by
  62  //
  63  //   2^16384 > x > 2^[-16383] => 1.19*10^4932 > x > 1.68*10^-[4932]
  64  // 
  65  // Special values of the exponent are:
  66  //
  67  //  all ones -> infinity (floating point overflow)
  68  //  all zeros -> number = zero. 
  69  //
  70  // Underflow in operations is handled by a flush to zero. Thus, a number 
  71  // with the exponent zero and nonzero mantissa is invalid (not-a-number). 
  72  // From the point of view of the HPA library, a complex number is a 
  73  // structure formed by two REAL*32 numbers.
  74  //
  75  // HPA cannot extend precision beyond the preset, hardcoded precision.
  76  // Hence some math routines will not achieve full precision.
  77  
  78  #include <vif.h>
  79  #include <rts-real32.h>
  80  
  81  static const char *errmsg[] = {
  82    "No error",
  83    "Division by zero",
  84    "Out of domain",
  85    "Bad exponent",
  86    "Floating point overflow",
  87    "Invalid error code"
  88  };
  89  
  90  #define SEXP(z) ((unt_2 *) &(z.value))
  91  
  92  void _pi32 (real_32 *x)
  93  {
  94    *x = X_PI;
  95  }
  96  
  97  int_4 xsigerr (int_4 errcond, int_4 errcode, const char *where)
  98  {
  99  // errcode must come from the evaluation of an error condition.
 100  // errcode, which should describe the type of the error, 
 101  // should always be one between XEDIV, XEDOM, XEBADEXP and XFPOFLOW.  
 102    if (errcond == 0) {
 103      errcode = 0;
 104    }
 105    if (errcode < 0 || errcode > XNERR) {
 106      errcode = XEINV;
 107    }
 108    if (errcode != 0) {
 109      RTE (where, errmsg[errcode]);
 110    }
 111    return errcode;
 112  }
 113  
 114  // Elementary stuff.
 115  
 116  inline real_32 xneg (real_32 s)
 117  {
 118    unt_2 *p = SEXP (s);
 119    *p ^= X_SIGN_MASK;
 120    return s;
 121  }
 122  
 123  inline real_32 xabs (real_32 s)
 124  {
 125    unt_2 *p = SEXP (s);
 126    *p &= X_EXPO_MASK;
 127    return s;
 128  }
 129  
 130  inline int_4 xgetexp (const real_32 * ps)
 131  {
 132    unt_2 *q = (unt_2 *) &(ps->value);
 133    return (*q & X_EXPO_MASK) - X_EXPO_BIAS;
 134  }
 135  
 136  inline int_4 xgetsgn (const real_32 * ps)
 137  {
 138    unt_2 *q = (unt_2 *) &(ps->value);
 139    return (*q & X_SIGN_MASK);
 140  }
 141  
 142  int_4 xreal_cmp (const real_32 * pa, const real_32 * pb)
 143  {
 144    unt_2 e, k, *p, *q, p0, q0;
 145    int_4 m;
 146    p = (unt_2 *) &(pa->value);
 147    q = (unt_2 *) &(pb->value);
 148    for (m = 1; m <= FLT256_LEN && p[m] == 0; m++);
 149    if (m > FLT256_LEN && (*p & X_EXPO_MASK) < X_EXPO_MASK) {
 150  // *pa is actually zero 
 151      p0 = 0;
 152    } else {
 153      p0 = *p;
 154    }
 155    for (m = 1; m <= FLT256_LEN && q[m] == 0; m++);
 156    if (m > FLT256_LEN && (*q & X_EXPO_MASK) < X_EXPO_MASK) {
 157  // *pb is actually zero 
 158      q0 = 0;
 159    } else {
 160      q0 = *q;
 161    }
 162    e = p0 & X_SIGN_MASK;
 163    k = q0 & X_SIGN_MASK;
 164    if (e && !k) {
 165      return -1;
 166    } else if (!e && k) {
 167      return 1;
 168    } else {                        // *pa and *pb have the same sign 
 169      m = (e) ? -1 : 1;
 170      e = p0 & X_EXPO_MASK;
 171      k = q0 & X_EXPO_MASK;
 172      if (e > k) {
 173        return m;
 174      } else if (e < k) {
 175        return -m;
 176      } else {
 177        for (e = 0; *(++p) == *(++q) && e < FLT256_LEN; ++e);
 178        if (e < FLT256_LEN) {
 179          return (*p > *q ? m : -m);
 180        } else {
 181          return 0;
 182        }
 183      }
 184    }
 185  }
 186  
 187  real_32 xreal_2 (real_32 s, int_4 m)
 188  {
 189    unt_2 *p = SEXP (s);
 190    int_4 e;
 191    for (e = 1; e <= FLT256_LEN && p[e] == 0; e++);
 192    if (e <= FLT256_LEN) {
 193      e = *p & X_EXPO_MASK;            // biased exponent 
 194      if (e + m < 0) {
 195        return X_0;
 196      } else if ((xsigerr (e + m >= X_EXPO_MASK, XFPOFLOW, NO_TEXT))) {
 197        return ((s.value[0] & X_SIGN_MASK) ? X_MINUS_INF : X_PLUS_INF);
 198      } else {
 199        *p += m;
 200        return s;
 201      }
 202    } else {                        // s is zero or +-Inf 
 203      return s;
 204    }
 205  }
 206  
 207  real_32 xsfmod (real_32 s, int_4 *p)
 208  {
 209    unt_2 *pa = SEXP (s);
 210    unt_2 *pb = pa + 1;
 211    int_2 e, k;
 212    e = (*pa & X_EXPO_MASK) - X_EXPO_BIAS;
 213    if ((xsigerr (e >= 15, XFPOFLOW, NO_TEXT))) {
 214      *p = -1;
 215      return s;
 216    } else if (e < 0) {
 217      *p = 0;
 218      return s;
 219    }
 220    *p = *pb >> (15 - e);
 221    xlshift (++e, pb, FLT256_LEN);
 222    *pa -= e;
 223    for (e = 0; *pb == 0 && e < xMax_p; ++pb, e += 16);
 224    if (e == xMax_p) {
 225      return X_0;
 226    }
 227    for (k = 0; !((*pb << k) & X_SIGN_MASK); ++k);
 228    if ((k += e)) {
 229      xlshift (k, pa + 1, FLT256_LEN);
 230      *pa -= k;
 231    }
 232    return s;
 233  }
 234  
 235  void xlshift (int_4 n, unt_2 *pm, int_4 m)
 236  {
 237    unt_2 *pc = pm + m - 1;
 238    if (n < 16 * m) {
 239      unt_2 *pa = pm + n / 16;
 240      m = n % 16;
 241      n = 16 - m;
 242      while (pa < pc) {
 243        *pm = (*pa++) << m;
 244        *pm++ |= *pa >> n;
 245      }
 246      *pm++ = *pa << m;
 247    }
 248    while (pm <= pc) {
 249      *pm++ = 0;
 250    }
 251  }
 252  
 253  void xrshift (int_4 n, unt_2 *pm, int_4 m)
 254  {
 255    unt_2 *pc = pm + m - 1;
 256    if (n < 16 * m) {
 257      unt_2 *pa = pc - n / 16;
 258      m = n % 16;
 259      n = 16 - m;
 260      while (pa > pm) {
 261        *pc = (*pa--) >> m;
 262        *pc-- |= *pa << n;
 263      }
 264      *pc-- = *pa >> m;
 265    }
 266    while (pc >= pm) {
 267      *pc-- = 0;
 268    }
 269  }
 270  
 271  real_32 _xI (real_32 x) 
 272  {
 273  // Intrinsic function so arguments are not pointers.
 274    return x;
 275  }
 276  
 277  real_32 _xnint (real_32 x)
 278  {
 279  // Intrinsic function so arguments are not pointers.
 280    if (xgetsgn (&x) == 0) {
 281      return xfloor (xadd (x, X_1_OVER_2, FALSE));
 282    } else {
 283      return xneg (xfloor (xadd (X_1_OVER_2, x, TRUE)));
 284    }
 285  }
 286  
 287  real_32 _xint (real_32 x) 
 288  {
 289  // Intrinsic function so arguments are not pointers.
 290    if (xgetsgn (&x) == 0) {
 291      return xtrunc (x);
 292    } else {
 293      return xneg (xtrunc (x));
 294    }
 295  }
 296  
 297  real_32 _xmax (real_32 a, real_32 b) 
 298  {
 299  // Intrinsic function so arguments are not pointers.
 300    if (xge (a, b)) {
 301      return a;
 302    } else {
 303      return b;
 304    }
 305  }
 306  
 307  real_32 _xmin (real_32 a, real_32 b) 
 308  {
 309  // Intrinsic function so arguments are not pointers.
 310    if (xle (a, b)) {
 311      return a;
 312    } else {
 313      return b;
 314    }
 315  }
 316  
 317  real_32 _xmod (real_32 a, real_32 b)
 318  {
 319  // Intrinsic function so arguments are not pointers.
 320    real_32 q = xdiv (a, b);
 321    if (xsgn (&q) >= 0) {
 322      q = xfloor (q);
 323    } else {
 324      q = xneg (xfloor (xneg (q)));
 325    }
 326    return xadd (a, xmul (b, q), 1);
 327  }
 328  
 329  real_32 xadd (real_32 s, real_32 t, int_4 f)
 330  {
 331    REAL32 pe;
 332    unt_2 h, u;
 333    unt_2 *pc, *pf = pe;
 334    unt_4 n = 0;
 335    unt_2 *pa = SEXP (s);
 336    unt_2 *pb = SEXP (t);
 337    int_2 e = *pa & X_EXPO_MASK;
 338    int_2 k = *pb & X_EXPO_MASK;
 339    if (f != 0) {
 340      *pb ^= X_SIGN_MASK;
 341    }
 342    u = (*pb ^ *pa) & X_SIGN_MASK;
 343    f = 0;
 344    if (e > k) {
 345      if ((k = e - k) >= xMax_p) {
 346        return s;
 347      }
 348      xrshift (k, pb + 1, FLT256_LEN);
 349    } else if (e < k) {
 350      if ((e = k - e) >= xMax_p) {
 351        return t;
 352      }
 353      xrshift (e, pa + 1, FLT256_LEN);
 354      e = k;
 355      pc = pa;
 356      pa = pb;
 357      pb = pc;
 358    } else if (u != 0) {
 359      for (pc = pa, pf = pb; *(++pc) == *(++pf) && f < FLT256_LEN; ++f);
 360      if (f >= FLT256_LEN) {
 361        return X_0;
 362      }
 363      if (*pc < *pf) {
 364        pc = pa;
 365        pa = pb;
 366        pb = pc;
 367      }
 368      pf = pe + f;
 369    }
 370    h = *pa & X_SIGN_MASK;
 371    if (u != 0) {
 372      for (pc = pb + FLT256_LEN; pc > pb; --pc) {
 373        *pc = ~(*pc);
 374      }
 375      n = 1L;
 376    }
 377    for (pc = pe + FLT256_LEN, pa += FLT256_LEN, pb += FLT256_LEN; pc > pf;) {
 378      n += *pa;
 379      pa--;
 380      n += *pb;
 381      pb--;
 382      *pc = n;
 383      pc--;
 384      n >>= 16;
 385    }
 386    if (u != 0) {
 387      for (; *(++pc) == 0; ++f);
 388      for (k = 0; !((*pc << k) & X_SIGN_MASK); ++k);
 389      if ((k += 16 * f)) {
 390        if ((e -= k) <= 0) {
 391          return X_0;
 392        }
 393        xlshift (k, pe + 1, FLT256_LEN);
 394      }
 395    } else {
 396      if (n != 0) {
 397        ++e;
 398        if ((xsigerr (e == (short) X_EXPO_MASK, XFPOFLOW, NO_TEXT))) {
 399          return (!h ? X_PLUS_INF : X_MINUS_INF);
 400        }
 401        ++pf;
 402        xrshift (1, pf, FLT256_LEN);
 403        *pf |= X_SIGN_MASK;
 404      }
 405    }
 406    *pe = e;
 407    *pe |= h;
 408    return *(real_32 *) pe;
 409  }
 410  
 411  real_32 xmul (real_32 s, real_32 t)
 412  {
 413    unt_2 pe[FLT256_LEN + 2], *q0, *q1, h;
 414    unt_2 *pa, *pb, *pc;
 415    unt_4 m, n, p;
 416    int_2 e;
 417    int_2 k;
 418    q0 = SEXP (s);
 419    q1 = SEXP (t);
 420    e = (*q0 & X_EXPO_MASK) - X_EXPO_BIAS;
 421    k = (*q1 & X_EXPO_MASK) + 1;
 422    if ((xsigerr (e > (short) X_EXPO_MASK - k, XFPOFLOW, NO_TEXT))) {
 423      return (((s.value[0] & X_SIGN_MASK) ^ (t.value[0] & X_SIGN_MASK)) ? X_MINUS_INF : X_PLUS_INF);
 424    }
 425    if ((e += k) <= 0) {
 426      return X_0;
 427    }
 428    h = (*q0 ^ *q1) & X_SIGN_MASK;
 429    for (++q1, k = FLT256_LEN, p = n = 0L, pc = pe + FLT256_LEN + 1; k > 0; --k) {
 430      for (pa = q0 + k, pb = q1; pa > q0;) {
 431        m = *pa--;
 432        m *= *pb++;
 433        n += (m & 0xffffL);
 434        p += (m >> 16);
 435      }
 436      *pc-- = n;
 437      n = p + (n >> 16);
 438      p = 0L;
 439    }
 440    *pc = n;
 441    if (!(*pc & X_SIGN_MASK)) {
 442      --e;
 443      if (e <= 0) {
 444        return X_0;
 445      }
 446      xlshift (1, pc, FLT256_LEN + 1);
 447    }
 448    if ((xsigerr (e == (short) X_EXPO_MASK, XFPOFLOW, NO_TEXT))) {
 449      return (!h ? X_PLUS_INF : X_MINUS_INF);
 450    }
 451    *pe = e;
 452    *pe |= h;
 453    return *(real_32 *) pe;
 454  }
 455  
 456  real_32 xdiv (real_32 s, real_32 t)
 457  {
 458    unt_2 *pd = SEXP (s), *pc = SEXP (t);
 459  // Next makes xdiv robust at extreme exponents - MvdV.
 460    if ((*pd & X_EXPO_MASK) == (*pc & X_EXPO_MASK)) {
 461      *pd &= ~X_EXPO_MASK; 
 462      *pc &= ~X_EXPO_MASK; 
 463    }
 464  // HPA implementation.
 465    unt_2 e = *pc;
 466    *pc = X_EXPO_BIAS;
 467    if ((xsigerr (xreal_cmp (&t, &X_0) == 0, XEDIV, "xdiv()"))) {
 468      return X_0;
 469    } else {
 470      real_32 a = dbltox (1 / xtodbl (t));
 471      *pc = e;
 472      pc = SEXP (a);
 473      *pc += X_EXPO_BIAS - (e & X_EXPO_MASK);
 474      *pc |= e & X_SIGN_MASK;
 475      for (unt_2 i = 0; i < xItt_div; ++i) {
 476        a = xmul (a, xadd (X_2, xmul (a, t), 1));
 477      }
 478      return xmul (s, a);
 479    }
 480  }
 481  
 482  
 483  real_32 xevtch (real_32 z, real_32 * a, int_4 m)
 484  {
 485    real_32 *p, f, t, tp, w;
 486    w = xreal_2 (z, 1);
 487    t = X_0;
 488    tp = X_0;
 489    for (p = a + m; p > a;) {
 490      f = xadd (*p--, xadd (xmul (w, t), tp, 1), 0);
 491      tp = t;
 492      t = f;
 493    }
 494    return xadd (*p, xadd (xmul (z, t), tp, 1), 0);
 495  }
 496  
 497  real_32 xexp2 (real_32 x)
 498  {
 499    real_32 s, d, f;
 500    unt_2 *pf = SEXP (x);
 501    int_4 m, k;
 502    if (xreal_cmp (&x, &xE2min) < 0) {
 503      return X_0;
 504    } else if ((xsigerr (xreal_cmp (&x, &xE2max) > 0, XFPOFLOW, NO_TEXT))) {
 505      return X_PLUS_INF;
 506    } else {
 507      m = (*pf & X_SIGN_MASK) ? 1 : 0;
 508      x = xsfmod (x, &k);
 509      if (m != 0) {
 510        k *= -1;
 511      }
 512  // -X_EXPO_BIAS <= k <= +X_EXPO_BIAS 
 513      x = xmul (x, X_LN_2);
 514      if (xgetexp (&x) > -X_EXPO_BIAS) {
 515        x = xreal_2 (x, -1);
 516        s = xmul (x, x);
 517        f = X_0;
 518        for (d = inttox (m = xMS_exp); m > 1; m -= 2, d = inttox (m)) {
 519          f = xdiv (s, xadd (d, f, 0));
 520        }
 521        f = xdiv (x, xadd (d, f, 0));
 522        f = xdiv (xadd (d, f, 0), xadd (d, f, 1));
 523      } else {
 524        f = X_1;
 525      }
 526      pf = SEXP (f);
 527      if (-k > *pf) {
 528        return X_0;
 529      } else {
 530        *pf += k;
 531        if ((xsigerr (*pf >= X_EXPO_MASK, XFPOFLOW, NO_TEXT))) {
 532          return X_PLUS_INF;
 533        } else {
 534          return f;
 535        }
 536      }
 537    }
 538  }
 539  
 540  real_32 xexp (real_32 z)
 541  {
 542    return xexp2 (xmul (z, X_LOG2_E));
 543  }
 544  
 545  real_32 xexp10 (real_32 z)
 546  {
 547    return xexp2 (xmul (z, X_LOG2_10));
 548  }
 549  
 550  real_32 xfmod (real_32 s, real_32 t, real_32 * q)
 551  {
 552    if ((xsigerr (xreal_cmp (&t, &X_0) == 0, XEDIV, "xfmod()"))) {
 553      return X_0;
 554    } else {
 555      unt_2 *p, mask = 0xffff;
 556      int_2 e, i;
 557      int_4 u;
 558      *q = xdiv (s, t);
 559      p = (unt_2 *) &(q->value);
 560      u = (*p & X_SIGN_MASK) ? 0 : 1;
 561      e = (*p &= X_EXPO_MASK);         // biased exponent of *q 
 562      e = e < X_EXPO_BIAS ? 0 : e - X_EXPO_BIAS + 1;
 563      for (i = 1; e / 16 > 0; i++, e -= 16);
 564      if (i <= FLT256_LEN) {
 565  // e = 0, ..., 15 
 566        mask <<= 16 - e;
 567        p[i] &= mask;
 568        for (i++; i <= FLT256_LEN; p[i] = 0, i++);
 569      }
 570  // Now *q == abs(quotient of (s/t)) 
 571      return xadd (s, xmul (t, *q), u);
 572    }
 573  }
 574  
 575  real_32 xfrexp (real_32 s, int_4 *p)
 576  {
 577    unt_2 *ps = SEXP (s), u;
 578    *p = (*ps & X_EXPO_MASK) - X_EXPO_BIAS + 1;
 579    u = *ps & X_SIGN_MASK;
 580    *ps = X_EXPO_BIAS - 1;
 581    *ps |= u;
 582    return s;
 583  }
 584  
 585  static void nullify (int_4 skip, unt_2 *p, int_4 k)
 586  {
 587  // After skipping the first 'skip' bytes of the vector 'p',
 588  // this function nullifies all the remaining ones. 'k' is
 589  // the number of words forming the vector p.
 590  // Warning: 'skip' must be positive !  
 591    int_4 i;
 592    unt_2 mask = 0xffff;
 593    for (i = 0; skip / 16 > 0; i++, skip -= 16);
 594    if (i < k) {
 595  // skip = 0, ..., 15 
 596      mask <<= 16 - skip;
 597      p[i] &= mask;
 598      for (i++; i < k; p[i] = 0, i++);
 599    }
 600  }
 601  
 602  static void canonic_form (real_32 * px)
 603  {
 604    unt_2 *p, u;
 605    int_2 e, i, j, skip;
 606    p = (unt_2 *) &(px->value);
 607    e = (*p & X_EXPO_MASK);            // biased exponent of x 
 608    u = (*p & X_SIGN_MASK);            // sign of x            
 609    if (e < X_EXPO_BIAS - 1) {
 610      return;
 611    } else {
 612      unt_2 mask = 0xffff;
 613  // e >= X_EXPO_BIAS - 1 
 614      for (i = 1, skip = e + 1 - X_EXPO_BIAS; skip / 16 > 0; i++, skip -= 16);
 615      if (i <= FLT256_LEN) {
 616  // skip = 0, ..., 15 
 617        mask >>= skip;
 618        if ((p[i] & mask) != mask) {
 619          return;
 620        } else {
 621          for (j = i + 1; j <= FLT256_LEN && p[j] == 0xffff; j++);
 622          if (j > FLT256_LEN) {
 623            p[i] -= mask;
 624            for (j = i + 1; j <= FLT256_LEN; p[j] = 0, j++);
 625            if (!(p[1] & 0x8000)) {
 626              p[1] = 0x8000;
 627              *p = ++e;
 628              *p |= u;
 629            } else if (u != 0) {
 630              *px = xadd (*px, X_1, 1);
 631            } else {
 632              *px = xadd (*px, X_1, 0);
 633            }
 634          }
 635        }
 636      }
 637    }
 638  }
 639  
 640  real_32 xfrac (real_32 x)
 641  {
 642  // xfrac(x) returns the fractional part of the number x.
 643  // xfrac(x) has the same sign as x.  
 644    unt_2 u, *p;
 645    int_2 e;
 646    int_4 n;
 647    canonic_form (&x);
 648    p = SEXP (x);
 649    e = (*p & X_EXPO_MASK);            // biased exponent of x 
 650    if (e < X_EXPO_BIAS) {
 651      return x;                   // The integer part of x is zero 
 652    } else {
 653      u = *p & X_SIGN_MASK;            // sign of x 
 654      n = e - X_EXPO_BIAS + 1;
 655      xlshift (n, p + 1, FLT256_LEN);
 656      e = X_EXPO_BIAS - 1;
 657  // Now I have to take in account the rule 
 658  // of the leading one.                    
 659      while (e > 0 && !(p[1] & X_SIGN_MASK)) {
 660        xlshift (1, p + 1, FLT256_LEN);
 661        e -= 1;
 662      }
 663  // Now p+1 points to the fractionary part of x, 
 664  // u is its sign, e is its biased exponent.     
 665      p[0] = e;
 666      p[0] |= u;
 667      return *(real_32 *) p;
 668    }
 669  }
 670  
 671  real_32 xtrunc (real_32 x)
 672  {
 673  // xtrunc(x) returns the integer part of the number x.
 674  // xtrunc(x) has the same sign as x.  
 675    unt_2 *p;
 676    int_2 e;
 677    canonic_form (&x);
 678    p = SEXP (x);
 679    e = (*p & X_EXPO_MASK);            // biased exponent of x 
 680    if (e < X_EXPO_BIAS) {
 681      return X_0;               // The integer part of x is zero 
 682    } else {
 683      nullify (e - X_EXPO_BIAS + 1, p + 1, FLT256_LEN);
 684      return *(real_32 *) p;
 685    }
 686  }
 687  
 688  real_32 xround (real_32 x)
 689  {
 690    return xtrunc (xadd (x, X_RND_CORR, x.value[0] & X_SIGN_MASK));
 691  }
 692  
 693  real_32 xceil (real_32 x)
 694  {
 695    unt_2 *ps = SEXP (x);
 696    if ((*ps & X_SIGN_MASK)) {
 697      return xtrunc (x);
 698    } else {
 699      real_32 y = xfrac (x);
 700  // y has the same sign as x (see above). 
 701      return (xreal_cmp (&y, &X_0) > 0 ? xadd (xtrunc (x), X_1, 0) : x);
 702    }
 703  }
 704  
 705  real_32 xfloor (real_32 x)
 706  {
 707    unt_2 *ps = SEXP (x);
 708    if ((*ps & X_SIGN_MASK)) {
 709      real_32 y = xfrac (x);
 710  // y has the same sign as x (see above). 
 711      return (xreal_cmp (&y, &X_0) < 0 ? xadd (xtrunc (x), X_1, 1) : x);
 712    } else {
 713      return xtrunc (x);
 714    }
 715  }
 716  
 717  static void xadd_correction (real_32 * px, int_4 k)
 718  {
 719    int_2 e = (px->value[0] & X_EXPO_MASK) - X_EXPO_BIAS;
 720  //   e = (e > 0 ? e : 0);   
 721    *px = xadd (*px, xreal_2 (X_FIX_CORR, e), k);
 722  }
 723  
 724  real_32 xfix (real_32 x)
 725  {
 726    unt_2 *p;
 727    int_2 e;
 728    xadd_correction (&x, x.value[0] & X_SIGN_MASK);
 729    p = SEXP (x);
 730    e = (*p & X_EXPO_MASK);            // biased exponent of x 
 731    if (e < X_EXPO_BIAS) {
 732      return X_0;               // The integer part of x is zero 
 733    } else {
 734      nullify (e - X_EXPO_BIAS + 1, p + 1, FLT256_LEN);
 735      return *(real_32 *) p;
 736    }
 737  }
 738  
 739  real_32 xtanh (real_32 z)
 740  {
 741    real_32 s, d, f;
 742    int_4 m, k;
 743    if ((k = xgetexp (&z)) > xK_tanh) {
 744      if (xgetsgn (&z)) {
 745        return xneg (X_1);
 746      } else {
 747        return X_1;
 748      }
 749    }
 750    if (k < xK_lin) {
 751      return z;
 752    }
 753    ++k;
 754    if (k > 0) {
 755      z = xreal_2 (z, -k);
 756    }
 757    s = xmul (z, z);
 758    f = X_0;
 759    for (d = inttox (m = xMS_hyp); m > 1;) {
 760      f = xdiv (s, xadd (d, f, 0));
 761      d = inttox (m -= 2);
 762    }
 763    f = xdiv (z, xadd (d, f, 0));
 764    for (; k > 0; --k) {
 765      f = xdiv (xreal_2 (f, 1), xadd (d, xmul (f, f), 0));
 766    }
 767    return f;
 768  }
 769  
 770  real_32 xsinh (real_32 z)
 771  {
 772    int_4 k;
 773    if ((k = xgetexp (&z)) < xK_lin) {
 774      return z;
 775    } else if (k < 0) {
 776      z = xtanh (xreal_2 (z, -1));
 777      return xdiv (xreal_2 (z, 1), xadd (X_1, xmul (z, z), 1));
 778    } else {
 779      z = xexp (z);
 780      return xreal_2 (xadd (z, xdiv (X_1, z), 1), -1);
 781    }
 782  }
 783  
 784  real_32 xcosh (real_32 z)
 785  {
 786    if (xgetexp (&z) < xK_lin) {
 787      return X_1;
 788    }
 789    z = xexp (z);
 790    return xreal_2 (xadd (z, xdiv (X_1, z), 0), -1);
 791  }
 792  
 793  real_32 xatanh (real_32 x)
 794  {
 795    real_32 y = x;
 796    y.value[0] &= X_EXPO_MASK;           // Now y == abs(x) 
 797    if ((xsigerr (xreal_cmp (&y, &X_1) >= 0, XEDOM, "xatanh"))) {
 798      return ((x.value[0] & X_SIGN_MASK) ? X_MINUS_INF : X_PLUS_INF);
 799    } else {
 800      y = xdiv (xadd (X_1, x, 0), xadd (X_1, x, 1));
 801      return xreal_2 (xlog (y), -1);
 802    }
 803  }
 804  
 805  real_32 xasinh (real_32 x)
 806  {
 807    real_32 y = xmul (x, x);
 808    y = xsqrt (xadd (X_1, y, 0));
 809    if ((x.value[0] & X_SIGN_MASK)) {
 810      return xneg (xlog (xadd (y, x, 1)));
 811    } else {
 812      return xlog (xadd (x, y, 0));
 813    }
 814  }
 815  
 816  real_32 xacosh (real_32 x)
 817  {
 818    if ((xsigerr (xreal_cmp (&x, &X_1) < 0, XEDOM, "xacosh()"))) {
 819      return X_0;
 820    } else {
 821      real_32 y = xmul (x, x);
 822      y = xsqrt (xadd (y, X_1, 1));
 823      return xlog (xadd (x, y, 0));
 824    }
 825  }
 826  
 827  real_32 xatan (real_32 z)
 828  {
 829    real_32 s, f;
 830    int_4 k, m;
 831    if ((k = xgetexp (&z)) < xK_lin) {
 832      return z;
 833    }
 834    if (k >= 0) {
 835  // k>=0 is equivalent to abs(z) >= 1.0 
 836      z = xdiv (X_1, z);
 837      m = 1;
 838    } else {
 839      m = 0;
 840    }
 841    f = dbltox (atan (xtodbl (z)));
 842    s = xadd (X_1, xmul (z, z), 0);
 843    for (k = 0; k < xItt_div; ++k) {
 844      f = xadd (f, xdiv (xadd (z, xtan (f), 1), s), 0);
 845    }
 846    if (m != 0) {
 847      if (xgetsgn (&f)) {
 848        return xadd (xneg (X_PI_OVER_2), f, 1);
 849      } else {
 850        return xadd (X_PI_OVER_2, f, 1);
 851      }
 852    } else {
 853      return f;
 854    }
 855  }
 856  
 857  real_32 xasin (real_32 z)
 858  {
 859    real_32 u = z;
 860    u.value[0] &= X_EXPO_MASK;
 861    if ((xsigerr (xreal_cmp (&u, &X_1) > 0, XEDOM, "xasin()"))) {
 862      return ((xgetsgn (&z)) ? xneg (X_PI_OVER_2) : X_PI_OVER_2);
 863    } else {
 864      if (xgetexp (&z) < xK_lin) {
 865        return z;
 866      }
 867      u = xsqrt (xadd (X_1, xmul (z, z), 1));
 868      if (xgetexp (&u) == -X_EXPO_BIAS) {
 869        return ((xgetsgn (&z)) ? xneg (X_PI_OVER_2) : X_PI_OVER_2);
 870      }
 871      return xatan (xdiv (z, u));
 872    }
 873  }
 874  
 875  real_32 xacos (real_32 z)
 876  {
 877    real_32 u = z;
 878    u.value[0] &= X_EXPO_MASK;
 879    if ((xsigerr (xreal_cmp (&u, &X_1) > 0, XEDOM, "xacos()"))) {
 880      return ((xgetsgn (&z)) ? X_PI : X_0);
 881    } else {
 882      if (xgetexp (&z) == -X_EXPO_BIAS) {
 883        return X_PI_OVER_2;
 884      }
 885      u = xsqrt (xadd (X_1, xmul (z, z), 1));
 886      u = xatan (xdiv (u, z));
 887      if (xgetsgn (&z)) {
 888        return xadd (X_PI, u, 0);
 889      } else {
 890        return u;
 891      }
 892    }
 893  }
 894  // Kindly added by A.Haumer 2010-04.09 
 895  
 896  real_32 xatan2 (real_32 y, real_32 x)
 897  {
 898    int_4 rs, is;
 899    rs = xsgn (&x);
 900    is = xsgn (&y);
 901    if (rs > 0) {
 902      return xatan (xdiv (y, x));
 903    } else if (rs < 0) {
 904      x.value[0] ^= X_SIGN_MASK;
 905      y.value[0] ^= X_SIGN_MASK;
 906      if (is >= 0) {
 907        return xadd (X_PI, xatan (xdiv (y, x)), 0);
 908      } else {
 909        return xadd (xatan (xdiv (y, x)), X_PI, 1);
 910      }
 911    } else {                      // x is zero ! 
 912      if (!xsigerr (is == 0, XEDOM, "xatan2()")) {
 913        return (is > 0 ? X_PI_OVER_2 : xneg (X_PI_OVER_2));
 914      } else {
 915        return X_0;             // Dummy value :) 
 916      }
 917    }
 918  }
 919  
 920  real_32 xlog (real_32 z)
 921  {
 922    real_32 f, h;
 923    int_4 k, m;
 924    if ((xsigerr ((xgetsgn (&z)) || xgetexp (&z) == -X_EXPO_BIAS, XEDOM, "xlog()"))) {
 925      return X_MINUS_INF;
 926    } else if (xreal_cmp (&z, &X_1) == 0) {
 927      return X_0;
 928    } else {
 929      z = xfrexp (z, &m);
 930      z = xmul (z, X_SQRT_2);
 931      z = xdiv (xadd (z, X_1, 1), xadd (z, X_1, 0));
 932      h = xreal_2 (z, 1);
 933      z = xmul (z, z);
 934      for (f = h, k = 1; xgetexp (&h) > -xMax_p;) {
 935        h = xmul (h, z);
 936        f = xadd (f, xdiv (h, inttox (k += 2)), 0);
 937      }
 938      return xadd (f, xmul (X_LN_2, dbltox (m - .5)), 0);
 939    }
 940  }
 941  
 942  real_32 xlog2 (real_32 z)
 943  {
 944    real_32 f, h;
 945    int_4 k, m;
 946    if ((xsigerr ((xgetsgn (&z)) || xgetexp (&z) == -X_EXPO_BIAS, XEDOM, "xlog2()"))) {
 947      return X_MINUS_INF;
 948    } else if (xreal_cmp (&z, &X_1) == 0) {
 949      return X_0;
 950    } else {
 951      z = xfrexp (z, &m);
 952      z = xmul (z, X_SQRT_2);
 953      z = xdiv (xadd (z, X_1, 1), xadd (z, X_1, 0));
 954      h = xreal_2 (z, 1);
 955      z = xmul (z, z);
 956      for (f = h, k = 1; xgetexp (&h) > -xMax_p;) {
 957        h = xmul (h, z);
 958        f = xadd (f, xdiv (h, inttox (k += 2)), 0);
 959      }
 960      return xadd (xmul (f, X_LOG2_E), dbltox (m - .5), 0);
 961    }
 962  }
 963  
 964  real_32 xlog10 (real_32 z)
 965  {
 966    real_32 w = xlog (z);
 967    if (xreal_cmp (&w, &X_MINUS_INF) <= 0) {
 968      return X_MINUS_INF;
 969    } else {
 970      return xmul (w, X_LOG10_E);
 971    }
 972  }
 973  
 974  int_4 xeq (real_32 x1, real_32 x2)
 975  {
 976    return (xreal_cmp (&x1, &x2) == 0);
 977  }
 978  
 979  int_4 xneq (real_32 x1, real_32 x2)
 980  {
 981    return (xreal_cmp (&x1, &x2) != 0);
 982  }
 983  
 984  int_4 xgt (real_32 x1, real_32 x2)
 985  {
 986    return (xreal_cmp (&x1, &x2) > 0);
 987  }
 988  
 989  int_4 xge (real_32 x1, real_32 x2)
 990  {
 991    return (xreal_cmp (&x1, &x2) >= 0);
 992  }
 993  
 994  int_4 xlt (real_32 x1, real_32 x2)
 995  {
 996    return (xreal_cmp (&x1, &x2) < 0);
 997  }
 998  
 999  int_4 xle (real_32 x1, real_32 x2)
1000  {
1001    return (xreal_cmp (&x1, &x2) <= 0);
1002  }
1003  // xis_nan (&x) returns 1 if and only if x is not a valid number 
1004  
1005  int_4 xis_nan (const real_32 * u)
1006  {
1007    unt_2 *p = (unt_2 *) &(u->value);
1008    if (*p != 0) {
1009      return 0;
1010    } else {
1011      int_4 i;
1012      for (i = 1; i <= FLT256_LEN && p[i] == 0x0; i++);
1013      return (i <= FLT256_LEN ? 1 : 0);
1014    }
1015  }
1016  
1017  int_4 xis0 (const real_32 * u)
1018  {
1019    unt_2 *p = (unt_2 *) &(u->value);
1020    int_4 m;
1021    for (m = 1; m <= FLT256_LEN && p[m] == 0; m++);
1022    return (m > FLT256_LEN && (*p & X_EXPO_MASK) < X_EXPO_MASK ? 1 : 0);
1023  }
1024  
1025  int_4 xnot0 (const real_32 * u)
1026  {
1027    unt_2 *p = (unt_2 *) &(u->value);
1028    int_4 m;
1029    for (m = 1; m <= FLT256_LEN && p[m] == 0; m++);
1030    return (m > FLT256_LEN && (*p & X_EXPO_MASK) < X_EXPO_MASK ? 0 : 1);
1031  }
1032  
1033  int_4 xlt1 (const real_32 u)
1034  {
1035    return (xlt (xabs (u), X_1));
1036  }
1037  
1038  int_4 xsgn (const real_32 * u)
1039  {
1040    unt_2 *p = (unt_2 *) &(u->value);
1041    int_4 m;
1042    for (m = 1; m <= FLT256_LEN && p[m] == 0; m++);
1043    if ((m > FLT256_LEN && (*p & X_EXPO_MASK) < X_EXPO_MASK) || !*p) {
1044      return 0;
1045    } else {
1046      return ((*p & X_SIGN_MASK) ? -1 : 1);
1047    }
1048  }
1049  
1050  int_4 xis_pinf (const real_32 * u)
1051  {
1052    return (*u->value == X_EXPO_MASK ? 1 : 0);
1053  }
1054  
1055  int_4 xis_minf (const real_32 * u)
1056  {
1057    return (*u->value == (X_EXPO_MASK | X_SIGN_MASK) ? 1 : 0);
1058  }
1059  
1060  int_4 xisordnumb (const real_32 * u)
1061  {
1062    int_4 isNaN, isfinite;
1063    unt_2 *p = (unt_2 *) &(u->value);
1064    if (*p != 0) {
1065      isNaN = 0;
1066    } else {
1067      int_4 i;
1068      for (i = 1; i <= FLT256_LEN && p[i] == 0x0; i++);
1069      isNaN = i <= FLT256_LEN;
1070    }
1071    isfinite = (*p & X_EXPO_MASK) < X_EXPO_MASK;
1072    return (!isNaN && (isfinite) ? 1 : 0);
1073  }
1074  
1075  real_32 xpwr (real_32 s, int_4 n)
1076  {
1077    real_32 t;
1078    unsigned k, m;
1079    t = X_1;
1080    if (n < 0) {
1081      m = -n;
1082      if ((xsigerr (xreal_cmp (&s, &X_0) == 0, XEBADEXP, "xpwr()"))) {
1083        return X_0;
1084      }
1085      s = xdiv (X_1, s);
1086    } else {
1087      m = n;
1088    }
1089    if (m != 0) {
1090      k = 1;
1091      while (1) {
1092        if (k & m) {
1093          t = xmul (s, t);
1094        }
1095        if ((k <<= 1) <= m) {
1096          s = xmul (s, s);
1097        } else {
1098          break;
1099        }
1100      }
1101    } else {
1102      xsigerr (xreal_cmp (&s, &X_0) == 0, XEBADEXP, "xpwr()");
1103    }
1104    return t;
1105  }
1106  
1107  real_32 xpow (real_32 x, real_32 y)
1108  {
1109    if (xsigerr ((xgetsgn (&x)) || xgetexp (&x) == -X_EXPO_BIAS, XEDOM, "xpow()")) {
1110      return X_0;
1111    } else {
1112      return xexp2 (xmul (xlog2 (x), y));
1113    }
1114  }
1115  
1116  real_32 xsqrt (real_32 z)
1117  {
1118    real_32 s, h;
1119    int_2 m, e;
1120    unt_2 *pc;
1121    if ((xsigerr ((xgetsgn (&z)), XEDOM, "xsqrt()"))) {
1122      return X_0;
1123    } else {
1124      pc = SEXP (z);
1125      if (*pc == 0) {
1126        return X_0;
1127      }
1128      e = *pc - X_EXPO_BIAS;
1129      *pc = X_EXPO_BIAS + (e % 2);
1130      e /= 2;
1131      s = dbltox (sqrt (xtodbl (z)));
1132      for (m = 0; m < xItt_div; ++m) {
1133        h = xdiv (xadd (z, xmul (s, s), 1), xreal_2 (s, 1));
1134        s = xadd (s, h, 0);
1135      }
1136      pc = SEXP (s);
1137      *pc += e;
1138      return s;
1139    }
1140  }
1141  
1142  static int_4 xodd (real_32 x)
1143  {
1144    unt_2 *p = SEXP (x);
1145    int_2 e, i;
1146    e = (*p & X_EXPO_MASK) - X_EXPO_BIAS;    // exponent of x 
1147    if (e < 0) {
1148      return 0;
1149    } else {
1150      for (i = 1; e / 16 > 0; i++, e -= 16);
1151  // Now e = 0, ..., 15 
1152      return (i <= FLT256_LEN ? p[i] & 0x8000 >> e : 0);
1153    }
1154  }
1155  
1156  real_32 xtan (real_32 z)
1157  {
1158    int_4 k, m;
1159    z = rred (z, 't', &k);
1160    if ((xsigerr (xreal_cmp (&z, &X_PI_OVER_2) >= 0, XEDOM, "xtan()"))) {
1161      return (!k ? X_PLUS_INF : X_MINUS_INF);
1162    } else {
1163      if (xreal_cmp (&z, &X_PI_OVER_4) == 1) {
1164        m = 1;
1165        z = xadd (X_PI_OVER_2, z, 1);
1166      } else {
1167        m = 0;
1168      }
1169      if (k != 0) {
1170        z = xneg (c_tan (z));
1171      } else {
1172        z = c_tan (z);
1173      }
1174      if (m != 0) {
1175        return xdiv (X_1, z);
1176      } else {
1177        return z;
1178      }
1179    }
1180  }
1181  
1182  real_32 xcos (real_32 z)
1183  {
1184    int_4 k;
1185    z = rred (z, 'c', &k);
1186    if (xgetexp (&z) < xK_lin) {
1187      if (k != 0) {
1188        return xneg (X_1);
1189      } else {
1190        return X_1;
1191      }
1192    }
1193    z = c_tan (xreal_2 (z, -1));
1194    z = xmul (z, z);
1195    z = xdiv (xadd (X_1, z, 1), xadd (X_1, z, 0));
1196    if (k != 0) {
1197      return xneg (z);
1198    } else {
1199      return z;
1200    }
1201  }
1202  
1203  real_32 xsin (real_32 z)
1204  {
1205    int_4 k;
1206    z = rred (z, 's', &k);
1207    if (xgetexp (&z) >= xK_lin) {
1208      z = c_tan (xreal_2 (z, -1));
1209      z = xdiv (xreal_2 (z, 1), xadd (X_1, xmul (z, z), 0));
1210    }
1211    if (k != 0) {
1212      return xneg (z);
1213    } else {
1214      return z;
1215    }
1216  }
1217  
1218  static real_32 c_tan (real_32 z)
1219  {
1220    real_32 s, f, d;
1221    int_4 m;
1222    unt_2 k;
1223    if (xgetexp (&z) < xK_lin)
1224      return z;
1225    s = xneg (xmul (z, z));
1226    for (k = 1; k <= FLT256_LEN && s.value[k] == 0; k++);
1227    if ((xsigerr (s.value[0] == 0xffff && k > FLT256_LEN, XFPOFLOW, NO_TEXT))) {
1228      return X_0;
1229    } else {
1230      f = X_0;
1231      for (d = inttox (m = xMS_trg); m > 1;) {
1232        f = xdiv (s, xadd (d, f, 0));
1233        d = inttox (m -= 2);
1234      }
1235      return xdiv (z, xadd (d, f, 0));
1236    }
1237  }
1238  
1239  static real_32 rred (real_32 z, int_4 kf, int_4 *ps)
1240  {
1241    real_32 is, q;
1242    if (xgetsgn (&z)) {
1243      z = xneg (z);
1244      is = X_1;
1245    } else {
1246      is = X_0;
1247    }
1248    z = xfmod (z, X_PI, &q);
1249    if (kf == 't') {
1250      q = is;
1251    } else if (kf == 's') {
1252      q = xadd (q, is, 0);
1253    }
1254    if (xreal_cmp (&z, &X_PI_OVER_2) == 1) {
1255      z = xadd (X_PI, z, 1);
1256      if (kf == 'c' || kf == 't') {
1257        q = xadd (q, X_1, 0);
1258      }
1259    }
1260    *ps = (xodd (q)) ? 1 : 0;
1261    return z;
1262  }
1263  
1264  // COMPLEX*64
1265  
1266  complex_64 cxadd (complex_64 z1, complex_64 z2, int_4 k)
1267  {
1268    complex_64 w;
1269    w.re = xadd (z1.re, z2.re, k);
1270    w.im = xadd (z1.im, z2.im, k);
1271    return w;
1272  }
1273  
1274  complex_64 cxsum (complex_64 z1, complex_64 z2)
1275  {
1276    complex_64 w;
1277    w.re = xadd (z1.re, z2.re, 0);
1278    w.im = xadd (z1.im, z2.im, 0);
1279    return w;
1280  }
1281  
1282  complex_64 cxsub (complex_64 z1, complex_64 z2)
1283  {
1284    complex_64 w;
1285    w.re = xadd (z1.re, z2.re, 1);
1286    w.im = xadd (z1.im, z2.im, 1);
1287    return w;
1288  }
1289  
1290  complex_64 cxmul (complex_64 z1, complex_64 z2)
1291  {
1292    complex_64 w;
1293    w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 1);
1294    w.im = xadd (xmul (z1.im, z2.re), xmul (z1.re, z2.im), 0);
1295    return w;
1296  }
1297  
1298  complex_64 cxrmul (real_32 c, complex_64 z)
1299  {
1300    complex_64 w;
1301    w.re = xmul (c, z.re);
1302    w.im = xmul (c, z.im);
1303    return w;
1304  }
1305  
1306  complex_64 cxdrot (complex_64 z)
1307  {
1308    complex_64 y;
1309    y.re = z.im;
1310    y.im = z.re;
1311  // Multiplication by +i 
1312    y.re.value[0] ^= X_SIGN_MASK;
1313    return y;
1314  }
1315  
1316  complex_64 cxrrot (complex_64 z)
1317  {
1318    complex_64 y;
1319    y.re = z.im;
1320    y.im = z.re;
1321  // Multiplication by -i 
1322    y.im.value[0] ^= X_SIGN_MASK;
1323    return y;
1324  }
1325  
1326  complex_64 cxdiv (complex_64 z1, complex_64 z2)
1327  {
1328    int_4 tv = cxrec (z2, &z2);
1329    if (!xsigerr (!tv, XEDIV, "cxdiv()")) {
1330      complex_64 w;
1331      w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 1);
1332      w.im = xadd (xmul (z1.im, z2.re), xmul (z1.re, z2.im), 0);
1333      return w;
1334    } else {
1335      return X_0_0I;
1336    }
1337  }
1338  
1339  complex_64 cxinv (complex_64 z)
1340  {
1341    int_4 tv = cxrec (z, &z);
1342    if (!xsigerr (!tv, XEDOM, "cxinv()")) {
1343      return z;
1344    } else {
1345      return X_0_0I;
1346    }
1347  }
1348  
1349  complex_64 cxsqr (complex_64 z)
1350  {
1351    complex_64 w;
1352    w.re = xadd (xmul (z.re, z.re), xmul (z.im, z.im), 1);
1353    w.im = xmul (X_2, xmul (z.im, z.re));
1354    return w;
1355  }
1356  
1357  complex_64 cxsqrt (complex_64 z)
1358  {
1359    complex_64 w;
1360    if (xsgn (&(z.re)) == 0 && xsgn (&(z.im)) == 0) {
1361      w = X_0_0I;
1362    } else if (xis0 (&(z.im))) {
1363      real_32 s = xsqrt (xabs (z.re));
1364      if (xsgn (&(z.re)) == -1) {
1365        w = (complex_64) {X_0, s};
1366      } else {
1367        w = (complex_64) {s, X_0};
1368      }
1369    } else {
1370      real_32 mod = xsqrt (cxabs (z)), arg = xreal_2 (cxarg (z), -1);
1371      w.re = xmul (mod, xcos (arg));
1372      w.im = xmul (mod, xsin (arg));
1373    }   
1374    return w;
1375  }
1376  
1377  complex_64 cxreset (real_32 re, real_32 im)
1378  {
1379    complex_64 y;
1380    y.re = re;
1381    y.im = im;
1382    return y;
1383  }
1384  
1385  complex_64 cxconv (real_32 x)
1386  {
1387    complex_64 y;
1388    y.re = x;
1389    y.im = X_0;
1390    return y;
1391  }
1392  
1393  real_32 cxre (complex_64 z)
1394  {
1395    return z.re;
1396  }
1397  
1398  real_32 cxim (complex_64 z)
1399  {
1400    return z.im;
1401  }
1402  
1403  complex_64 cxswap (complex_64 z)
1404  {
1405    complex_64 y;
1406    y.re = z.im;
1407    y.im = z.re;
1408    return y;
1409  }
1410  
1411  complex_64 cxneg (complex_64 z)
1412  {
1413    z.re.value[0] ^= X_SIGN_MASK;
1414    z.im.value[0] ^= X_SIGN_MASK;
1415    return z;
1416  }
1417  
1418  complex_64 cxconj (complex_64 z)
1419  {
1420    return (z.im.value[0] ^= X_SIGN_MASK, z);
1421  }
1422  
1423  #define XBOUND  FLT256_LEN * 16 + 8
1424  
1425  real_32 cxabs (complex_64 z)
1426  {
1427    if (xreal_cmp (&(z.re), &X_0) == 0 && xreal_cmp (&(z.im), &X_0) == 0) {
1428      return X_0;
1429    } else {
1430      int_4 ea = (z.re.value[0] &= X_EXPO_MASK) - X_EXPO_BIAS;
1431      int_4 eb = (z.im.value[0] &= X_EXPO_MASK) - X_EXPO_BIAS;
1432      if (ea > eb + XBOUND) {
1433        return z.re;
1434      } else if (eb > ea + XBOUND) {
1435        return z.im;
1436      } else {
1437        z.re.value[0] -= eb;
1438        z.im.value[0] = X_EXPO_BIAS;
1439        real_32 x = xsqrt (xadd (xmul (z.re, z.re), xmul (z.im, z.im), 0));
1440        x.value[0] += eb;
1441        return x;
1442      }
1443    }
1444  }
1445  
1446  real_32 cxarg (complex_64 z)
1447  {
1448    int_4 rs = xsgn (&(z.re)), is = xsgn (&(z.im));
1449    if (rs > 0) {
1450      return xatan (xdiv (z.im, z.re));
1451    } else if (rs < 0) {
1452      z.re.value[0] ^= X_SIGN_MASK;
1453      z.im.value[0] ^= X_SIGN_MASK;
1454      if (is >= 0) {
1455        return xadd (X_PI, xatan (xdiv (z.im, z.re)), 0);
1456      } else {
1457        return xadd (xatan (xdiv (z.im, z.re)), X_PI, 1);
1458      }
1459    } else {                      // z.re is zero ! 
1460      if (!xsigerr (is == 0, XEDOM, "cxarg()")) {
1461        return (is > 0 ? X_PI_OVER_2 : xneg (X_PI_OVER_2));
1462      } else {
1463        return xneg (X_PI_OVER_2);       // Dummy value :)
1464      } 
1465    }
1466  }
1467  
1468  int_4 cxrec (complex_64 z, complex_64 * w)
1469  {
1470    if (xreal_cmp (&(z.re), &X_0) == 0 && xreal_cmp (&(z.im), &X_0) == 0) {
1471      return 0;
1472    } else {
1473      int_4 sa = z.re.value[0] & X_SIGN_MASK;
1474      int_4 sb = z.im.value[0] & X_SIGN_MASK;
1475      int_4 ea = (z.re.value[0] &= X_EXPO_MASK) - X_EXPO_BIAS;
1476      int_4 eb = (z.im.value[0] &= X_EXPO_MASK) - X_EXPO_BIAS;
1477      real_32 x;
1478      if (ea > eb + XBOUND) {
1479        x = z.re;
1480      } else if (eb > ea + XBOUND) {
1481        x = z.im;
1482      } else {
1483        z.re.value[0] -= eb;
1484        z.im.value[0] = X_EXPO_BIAS;
1485        x = xsqrt (xadd (xmul (z.re, z.re), xmul (z.im, z.im), 0));
1486        x.value[0] += eb;
1487        z.re.value[0] += eb;
1488        z.im.value[0] += eb;
1489      }
1490      w->re = xdiv (xdiv (z.re, x), x);
1491      w->im = xdiv (xdiv (z.im, x), x);
1492      w->re.value[0] |= sa;
1493      w->im.value[0] |= X_SIGN_MASK ^ sb;
1494      return 1;
1495    }
1496  }
1497  
1498  complex_64 cxexp (complex_64 z)
1499  {
1500    complex_64 w;
1501    w.re = xmul (xexp (z.re), xcos (z.im));
1502    w.im = xmul (xexp (z.re), xsin (z.im));
1503    return w;
1504  }
1505  
1506  complex_64 cxlog (complex_64 z)
1507  {
1508    real_32 mod;
1509    complex_64 w;
1510    mod = cxabs (z);
1511    if (xsigerr (xsgn (&mod) <= 0, XEDOM, "cxlog()")) {
1512      return X_0_0I;
1513    } else {
1514      w.re = xlog (mod);
1515      w.im = cxarg (z);
1516      return w;
1517    }
1518  }
1519  
1520  complex_64 cxlog10 (complex_64 z)
1521  {
1522    real_32 mod;
1523    complex_64 w;
1524    mod = cxabs (z);
1525    if (xsigerr (xsgn (&mod) <= 0, XEDOM, "cxlog10()")) {
1526      return X_0_0I;
1527    } else {
1528      w.re = xlog10 (mod);
1529      w.im = xmul (cxarg (z), X_LOG10_E);
1530      return w;
1531    }
1532  }
1533  
1534  complex_64 cxlog2 (complex_64 z)
1535  {
1536    real_32 mod;
1537    complex_64 w;
1538    mod = cxabs (z);
1539    if (xsigerr (xsgn (&mod) <= 0, XEDOM, "cxlog2()")) {
1540      return X_0_0I;
1541    } else {
1542      w.re = xlog2 (mod);
1543      w.im = xmul (cxarg (z), X_LOG2_E);
1544      return w;
1545    }
1546  }
1547  
1548  complex_64 cxlog_sqrt (complex_64 z)
1549  {
1550    real_32 mod = cxabs (z);
1551    if (xsigerr (xsgn (&mod) <= 0, XEDOM, "cxlog_sqrt()")) {
1552      return X_0_0I;
1553    } else {
1554      complex_64 w;
1555      w.re = xreal_2 (xlog (mod), -1);
1556      w.im = xreal_2 (cxarg (z), -1);
1557      return w;
1558    }
1559  }
1560  
1561  complex_64 cxsinh (complex_64 z)
1562  {
1563    complex_64 w = cxsub (cxexp (z), cxexp (cxneg (z)));
1564    w.re = xreal_2 (w.re, -1);
1565    w.im = xreal_2 (w.im, -1);
1566    return w;
1567  }
1568  
1569  complex_64 cxcosh (complex_64 z)
1570  {
1571    complex_64 w = cxsum (cxexp (z), cxexp (cxneg (z)));
1572    w.re = xreal_2 (w.re, -1);
1573    w.im = xreal_2 (w.im, -1);
1574    return w;
1575  }
1576  
1577  complex_64 cxtanh (complex_64 z)
1578  {
1579    if (xsigerr (xreal_cmp (&(z.re), &xEmax) > 0, XFPOFLOW, NO_TEXT)) {
1580      return X_1_0I;
1581    } else if (xsigerr (xreal_cmp (&(z.re), &xEmin) < 0, XFPOFLOW, NO_TEXT)) {
1582      return cxneg (X_1_0I);
1583    } else {
1584      complex_64 w;
1585      if (xsigerr (!cxrec (cxcosh (z), &w), XEDOM, "cxtanh()")) {
1586        return X_0_0I;
1587      } else {
1588        return cxmul (cxsinh (z), w);
1589      }
1590    }
1591  }
1592  
1593  complex_64 cxasinh (complex_64 z)
1594  {
1595  // This way cxasinh() works fine also with real numbers very near to -oo.                                       
1596    complex_64 w = cxsqrt (cxsum (X_1_0I, cxsqr (z)));
1597    real_32 ls = cxabs (cxsum (z, w));
1598    real_32 rs = xmul (X_VSV, cxabs (z));
1599    if (xreal_cmp (&ls, &rs) < 0) {
1600      return cxneg (cxlog (cxsub (w, z)));
1601    } else {
1602      return cxlog (cxsum (z, w));
1603    }
1604  }
1605  
1606  complex_64 cxacosh (complex_64 z)
1607  {
1608    complex_64 w = cxsqrt (cxsub (cxsqr (z), X_1_0I));
1609    real_32 ls = cxabs (cxsum (z, w));
1610    real_32 rs = xmul (X_VSV, cxabs (z));
1611    if (xreal_cmp (&ls, &rs) < 0) {
1612      return cxneg (cxlog (cxsub (z, w)));
1613    } else {
1614      return cxlog (cxsum (z, w));
1615    }
1616  }
1617  
1618  complex_64 cxatanh (complex_64 z)
1619  {
1620    real_32 t = xadd (xabs (z.re), X_1, 1);
1621    int_4 errcond = xsgn (&(z.im)) == 0 && xsgn (&t) == 0;
1622    if (xsigerr (errcond, XEDOM, "cxatanh()")) {
1623      return X_0_0I;
1624    } else {
1625      complex_64 w = cxdiv (cxsum (X_1_0I, z), cxsub (X_1_0I, z));
1626      w = cxlog_sqrt (w);
1627      return w;
1628    }
1629  }
1630  
1631  complex_64 cxgdiv (complex_64 z1, complex_64 z2)
1632  {
1633    z1.re = xround (z1.re);
1634    z1.im = xround (z1.im);
1635    z2.re = xround (z2.re);
1636    z2.im = xround (z2.im);
1637    real_32 mod2 = xadd (xmul (z2.re, z2.re), xmul (z2.im, z2.im), 0);
1638    if (xsigerr (xreal_cmp (&mod2, &X_PLUS_INF) >= 0, XFPOFLOW, NO_TEXT) || xsigerr (xsgn (&mod2) <= 0, XEDIV, "cxgdiv()")) {
1639      return X_0_0I;
1640    } else {
1641      complex_64 w;
1642      w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 0);
1643      w.im = xadd (xmul (z2.re, z1.im), xmul (z2.im, z1.re), 1);
1644      w.re = xround (xdiv (w.re, mod2));
1645      w.im = xround (xdiv (w.im, mod2));
1646      return w;
1647    }
1648  }
1649  
1650  complex_64 cxidiv (complex_64 z1, complex_64 z2)
1651  {
1652    z1.re = xround (z1.re);
1653    z1.im = xround (z1.im);
1654    z2.re = xround (z2.re);
1655    z2.im = xround (z2.im);
1656    real_32 mod2 = xadd (xmul (z2.re, z2.re), xmul (z2.im, z2.im), 0);
1657    if (xsigerr (xreal_cmp (&mod2, &X_PLUS_INF) >= 0, XFPOFLOW, NO_TEXT) || xsigerr (xsgn (&mod2) <= 0, XEDIV, "cxidiv()")) {
1658      return X_0_0I;
1659    } else {
1660      complex_64 w;
1661      w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 0);
1662      w.im = xadd (xmul (z2.re, z1.im), xmul (z2.im, z1.re), 1);
1663      w.re = xfix (xdiv (w.re, mod2));
1664      w.im = xfix (xdiv (w.im, mod2));
1665      return w;
1666    }
1667  }
1668  
1669  complex_64 cxgmod (complex_64 z1, complex_64 z2)
1670  {
1671    z1.re = xround (z1.re);
1672    z1.im = xround (z1.im);
1673    z2.re = xround (z2.re);
1674    z2.im = xround (z2.im);
1675    real_32 mod2 = xadd (xmul (z2.re, z2.re), xmul (z2.im, z2.im), 0);
1676    if (xsigerr (xreal_cmp (&mod2, &X_PLUS_INF) >= 0, XFPOFLOW, NO_TEXT) || xsigerr (xsgn (&mod2) <= 0, XEDIV, "cxgmod()")) {
1677      return X_0_0I;
1678    } else {
1679      complex_64 w, z;
1680      w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 0);
1681      w.im = xadd (xmul (z2.re, z1.im), xmul (z2.im, z1.re), 1);
1682      w.re = xround (xdiv (w.re, mod2));
1683      w.im = xround (xdiv (w.im, mod2));
1684      z.re = xadd (xmul (w.re, z2.re), xmul (w.im, z2.im), 1);
1685      z.im = xadd (xmul (w.im, z2.re), xmul (w.re, z2.im), 0);
1686      w.re = xround (xadd (z1.re, z.re, 1));
1687      w.im = xround (xadd (z1.im, z.im, 1));
1688      return w;
1689    }
1690  }
1691  
1692  complex_64 cxmod (complex_64 z1, complex_64 z2)
1693  {
1694    z1.re = xround (z1.re);
1695    z1.im = xround (z1.im);
1696    z2.re = xround (z2.re);
1697    z2.im = xround (z2.im);
1698    real_32 mod2 = xadd (xmul (z2.re, z2.re), xmul (z2.im, z2.im), 0);
1699    if (xsigerr (xreal_cmp (&mod2, &X_PLUS_INF) >= 0, XFPOFLOW, NO_TEXT) || xsigerr (xsgn (&mod2) <= 0, XEDIV, "cxmod()")) {
1700      return X_0_0I;
1701    } else {
1702      complex_64 w, z;
1703      w.re = xadd (xmul (z1.re, z2.re), xmul (z1.im, z2.im), 0);
1704      w.im = xadd (xmul (z2.re, z1.im), xmul (z2.im, z1.re), 1);
1705      w.re = xfix (xdiv (w.re, mod2));
1706      w.im = xfix (xdiv (w.im, mod2));
1707      z.re = xadd (xmul (w.re, z2.re), xmul (w.im, z2.im), 1);
1708      z.im = xadd (xmul (w.im, z2.re), xmul (w.re, z2.im), 0);
1709      w.re = xround (xadd (z1.re, z.re, 1));
1710      w.im = xround (xadd (z1.im, z.im, 1));
1711      return w;
1712    }
1713  }
1714  
1715  static void unit_root (int_4 i, int_4 n, real_32 * a, real_32 * b)
1716  {
1717  // We assume n != 0 
1718    i %= n;
1719    *a = xdiv (xmul (xreal_2 (inttox (i), 1), X_PI), inttox (n));
1720    *b = xsin (*a);
1721    *a = xcos (*a);
1722    if (xgetexp (b) < -80) {
1723      *b = X_0;
1724    }
1725    if (xgetexp (a) < -80) {
1726      *a = X_0;
1727    }
1728  }
1729  
1730  complex_64 cxpwr (complex_64 z, int_4 n)
1731  {
1732    real_32 mod = cxabs (z);
1733    if (xsgn (&mod) <= 0) {
1734      (void) xsigerr (n <= 0, XEBADEXP, "cxpwr()");
1735      return X_0_0I;
1736    } else {
1737      real_32 arg = xmul (inttox (n), cxarg (z));
1738      mod = xpwr (mod, n);
1739      complex_64 w;
1740      w.re = xmul (mod, xcos (arg));
1741      w.im = xmul (mod, xsin (arg));
1742      return w;
1743    }
1744  }
1745  
1746  complex_64 cxroot (complex_64 z, int_4 i, int_4 n)
1747  {
1748    if (xsigerr (n == 0, XEBADEXP, "cxroot()")) {
1749      return X_0_0I;
1750    } else {
1751      real_32 mod = cxabs (z);
1752      if (xsgn (&mod) <= 0) {
1753        (void) xsigerr (n < 0, XEBADEXP, "cxroot()");
1754        return X_0_0I;
1755      } else {                    // mod > 0 
1756        real_32 arg = xdiv (cxarg (z), inttox (n));
1757        real_32 e = xdiv (X_1, inttox (n));      // 1/n 
1758  // x^e = exp(e*log(x)) for any x > 0 
1759        mod = xexp (xmul (e, xlog (mod)));
1760        complex_64 w, zz;
1761        w.re = xmul (mod, xcos (arg));
1762        w.im = xmul (mod, xsin (arg));
1763        real_32 a, b;
1764        unit_root (i, n, &a, &b);
1765        zz.re = xadd (xmul (w.re, a), xmul (w.im, b), 1);
1766        zz.im = xadd (xmul (w.im, a), xmul (w.re, b), 0);
1767        return zz;
1768      }
1769    }
1770  }
1771  
1772  complex_64 cxpow (complex_64 z1, complex_64 z2)
1773  {
1774    real_32 mod = cxabs (z1);
1775    if (xsgn (&mod) <= 0) {
1776      (void) xsigerr (xsgn (&z2.re) <= 0, XEBADEXP, "cxpow()");
1777      return X_0_0I;
1778    } else {
1779      real_32 arg = cxarg (z1);
1780      real_32 a = xadd (xmul (z2.re, xlog (mod)), xmul (z2.im, arg), 1);
1781      real_32 b = xadd (xmul (z2.re, arg), xmul (z2.im, xlog (mod)), 0);
1782      complex_64 w;
1783      w.re = xmul (xexp (a), xcos (b));
1784      w.im = xmul (xexp (a), xsin (b));
1785      return w;
1786    }
1787  }
1788  
1789  int_4 cxis0 (const complex_64 * z)
1790  {
1791    return (xis0 (&z->re) && xis0 (&z->im));
1792  }
1793  
1794  int_4 cxnot0 (const complex_64 * z)
1795  {
1796    return (xnot0 (&z->re) || xnot0 (&z->im));
1797  }
1798  
1799  int_4 cxeq (complex_64 z1, complex_64 z2)
1800  {
1801    return (xreal_cmp (&z1.re, &z2.re) == 0 && xreal_cmp (&z1.im, &z2.im) == 0);
1802  }
1803  
1804  int_4 cxneq (complex_64 z1, complex_64 z2)
1805  {
1806    return (xreal_cmp (&z1.re, &z2.re) != 0 || xreal_cmp (&z1.im, &z2.im) != 0);
1807  }
1808  
1809  complex_64 cxsin (complex_64 z)
1810  {
1811    complex_64 w1, w2;
1812    w1 = cxdrot (z);              // Now w1= i*z,  where i={0,1} 
1813    w2 = cxrrot (z);              // Now w2= -i*z, where i={0,1} 
1814    w1 = cxsub (cxexp (w1), cxexp (w2));
1815    w2.re = xreal_2 (w1.im, -1);
1816    w1.re.value[0] ^= X_SIGN_MASK;
1817    w2.im = xreal_2 (w1.re, -1);     // Now w2= (exp(i*z)-exp(-i*z))/2i 
1818    return w2;
1819  }
1820  
1821  complex_64 cxcos (complex_64 z)
1822  {
1823    complex_64 w1, w2;
1824    w1 = cxdrot (z);              // Now w1=  i*z,  where i={0,1} 
1825    w2 = cxrrot (z);              // Now w2= -i*z, where i={0,1} 
1826    w1 = cxsum (cxexp (w1), cxexp (w2));
1827    w2.re = xreal_2 (w1.re, -1);
1828    w2.im = xreal_2 (w1.im, -1);
1829    return w2;
1830  }
1831  
1832  complex_64 cxtan (complex_64 z)
1833  {
1834    if (xsigerr (xreal_cmp (&(z.im), &xEmax) > 0, XFPOFLOW, NO_TEXT)) {
1835      return X_0_1I;
1836    } else if (xsigerr (xreal_cmp (&(z.im), &xEmin) < 0, XFPOFLOW, NO_TEXT)) {
1837      return cxneg (X_0_1I);
1838    } else {
1839      complex_64 w;
1840      if (xsigerr (!cxrec (cxcos (z), &w), XEDOM, "cxtan()")) {
1841        return X_0_0I;
1842      } else {
1843        return cxmul (cxsin (z), w);
1844      }
1845    }
1846  }
1847  
1848  complex_64 cxasin (complex_64 z)
1849  {
1850    complex_64 w = cxsqrt (cxsub (X_1_0I, cxsqr (z)));
1851    real_32 ls = cxabs (cxsum (cxdrot (z), w));
1852    real_32 rs = xmul (X_VSV, cxabs (z));
1853    if (xreal_cmp (&ls, &rs) < 0) {
1854      w = cxdrot (cxlog (cxsub (w, cxdrot (z))));
1855    } else {
1856      w = cxrrot (cxlog (cxsum (cxdrot (z), w)));
1857    }
1858    return w;
1859  }
1860  
1861  complex_64 cxacos (complex_64 z)
1862  {
1863    complex_64 w = cxsqrt (cxsub (X_1_0I, cxsqr (z)));
1864    real_32 ls = cxabs (cxsum (z, cxdrot (w)));
1865    real_32 rs = xmul (X_VSV, cxabs (z));
1866    if (xreal_cmp (&ls, &rs) < 0) {
1867      w = cxdrot (cxlog (cxsub (z, cxdrot (w))));
1868    } else {
1869      w = cxrrot (cxlog (cxsum (z, cxdrot (w))));
1870    }
1871    return w;
1872  }
1873  
1874  complex_64 cxatan (complex_64 z)
1875  {
1876    real_32 mod = xadd (xabs (z.im), X_1, 1);
1877    int_4 errcond = xsgn (&(z.re)) == 0 && xsgn (&mod) == 0;
1878    if (xsigerr (errcond, XEDOM, "cxatan()")) {
1879      return X_0_0I;
1880    } else {
1881  // This way, cxatan() works fine also with complex numbers very far from the origin.
1882      complex_64 w;
1883      mod = cxabs (z);
1884      if (xreal_cmp (&mod, &X_VGV) > 0) {
1885        w = cxsqrt (cxsum (X_1_0I, cxsqr (z)));
1886        w = cxdiv (cxsum (X_1_0I, cxdrot (z)), w);
1887        w = cxrrot (cxlog (w));
1888      } else {
1889        w = cxdiv (cxsum (X_1_0I, cxdrot (z)), cxsub (X_1_0I, cxdrot (z)));
1890        w = cxrrot (cxlog_sqrt (w));
1891      }
1892      return w;
1893    }
1894  }
1895  
1896  complex_64 cxdbl(complex_16 z)
1897  {
1898    complex_64 zz;
1899    zz.re = dbltox (creal (z));
1900    zz.im = dbltox (cimag (z));
1901    return zz;
1902  }
1903  
1904  complex_64 cxquad(complex_32 z)
1905  {
1906    complex_64 zz;
1907    zz.re = dbltox (crealq (z));
1908    zz.im = dbltox (cimagq (z));
1909    return zz;
1910  }
1911  
1912  complex_64 _cquadtop (complex_64 *zz, complex_32 z)
1913  {
1914    zz->re = dbltox (crealq (z));
1915    zz->im = dbltox (cimagq (z));
1916    return *zz;
1917  }
1918  
1919  complex_64 cxreal32(real_32 z)
1920  {
1921    complex_64 zz;
1922    zz.re = z;
1923    zz.im = X_0;
1924    return zz;
1925  }
1926  
1927  complex_64 _coctotop(complex_64 *zz, real_32 z)
1928  {
1929    zz->re = z;
1930    zz->im = X_0;
1931    return *zz;
1932  }
1933  
1934  // VIF additions (REAL*32)
1935  
1936  real_32 xtenup (int_4 n)
1937  {
1938    if (n == 0) {
1939      return X_1;
1940    } else if (n == 1) {
1941      return X_10;
1942    } else if (n == -1) {
1943      return X_1_OVER_10;
1944    }
1945    real_32 s = X_10, t;
1946    unsigned k, m;
1947    t = X_1;
1948    if (n < 0) {
1949      m = -n;
1950      if ((xsigerr (xreal_cmp (&s, &X_0) == 0, XEBADEXP, "xpwr()"))) {
1951        return X_0;
1952      }
1953      s = xdiv (X_1, s);
1954    } else {
1955      m = n;
1956    }
1957    if (m != 0) {
1958      k = 1;
1959      while (1) {
1960        if (k & m) {
1961          t = xmul (s, t);
1962        }
1963        if ((k <<= 1) <= m) {
1964          s = xmul (s, s);
1965        } else {
1966          break;
1967        }
1968      }
1969    } else {
1970      xsigerr (xreal_cmp (&s, &X_0) == 0, XEBADEXP, "xpwr()");
1971    }
1972    return t;
1973  }
1974  
1975  real_32 xcotan (real_32 x)
1976  {
1977  // Intrinsic function so arguments are not pointers.
1978    return xdiv (X_1, xtan (x));
1979  }
1980  
1981  real_32 xacotan (real_32 x)
1982  {
1983  // Intrinsic function so arguments are not pointers.
1984    return xatan (xdiv (X_1, x));
1985  }
1986  
1987  real_32 _xsgn (real_32 a, real_32 b)
1988  {
1989  // Intrinsic function so arguments are not pointers.
1990    real_32 x = (xgetsgn (&a) == 0 ? a : xneg (a));
1991    return (xgetsgn (&b) == 0 ? x : xneg (x));
1992  }
1993  
1994  real_32 _zabs_64 (real_32 re, real_32 im)
1995  {
1996  // Intrinsic function so arguments are not pointers.
1997    return cxabs (CMPLXX (re, im));
1998  }
1999  
2000  // Conversion.
2001  
2002  real_32 inttox (int_4 n)
2003  {
2004    REAL32 pe;
2005    unt_2 *pc, e;
2006    unt_4 k, h;
2007    bzero (pe, sizeof (pe));
2008    k = ABS (n);
2009    pc = (unt_2 *) &k;
2010    if (n == 0) {
2011      return *(real_32 *) pe;
2012    }
2013  
2014    #if __BYTE_ORDER == __LITTLE_ENDIAN
2015      pe[1] = *(pc + 1);
2016      pe[2] = *pc;
2017    #else
2018      pe[1] = *pc;
2019      pe[2] = *(pc + 1);
2020    #endif
2021  
2022    for (e = 0, h = 1; h <= k && e < ((8 * sizeof (unt_4)) - 1); h <<= 1, ++e);
2023    if (h <= k) {
2024      e += 1;
2025    }
2026    *pe = X_EXPO_BIAS + e - 1;
2027    if (n < 0) {
2028      *pe |= X_SIGN_MASK;
2029    }
2030    xlshift ((8 * sizeof (unt_4)) - e, pe + 1, FLT256_LEN);
2031    return *(real_32 *) pe;
2032  }
2033  
2034  real_4 xtoflt (real_32 s)
2035  {
2036  // An extended floating point_4 number is represented as a combination of the
2037  // following elements:
2038  //
2039  //   sign bit(s): 0 -> positive, 1 -> negative ;
2040  //   exponent(e): 15-bit biased integer (bias=16383) ;
2041  //   mantissa(m): 7 (or more) words of 16 bit length with the
2042  //                leading 1 explicitly represented.
2043  //
2044  //   Thus  f = (-1)^s*2^[e-16383] *m ,  with 1 <= m < 2 .
2045  //
2046  // This format supports a dynamic range of:
2047  //
2048  //   2^16384 > f > 2^[-16383]  or
2049  //   1.19*10^4932 > f > 1.68*10^-[4932].
2050  //
2051  // Special values of the exponent are:
2052  //
2053  //   all ones -> infinity (floating point_4 overflow)
2054  //   all zeros -> number = zero.
2055  //
2056  // Underflow in operations is handled by a flush to zero. Thus, a number with
2057  // the exponent zero and nonzero mantissa is invalid (not-a-number).
2058  //
2059  //
2060    union {unt_2 pe[2]; real_4 f;} v;
2061    unt_2 *pc, u;
2062    int_2 i, e;
2063    pc = SEXP (s);
2064    u = *pc & X_SIGN_MASK;
2065    e = (*pc & X_EXPO_MASK) - xF_bias;
2066  //
2067  // u is the sign of the number s.
2068  // e == (exponent of s) + 127 
2069  //
2070    if (e >= xF_max) {
2071      return (!u ? FLT_MAX : -FLT_MAX);
2072    }
2073    if (e <= 0) {
2074      return 0.;
2075    }
2076    for (i = 0; i < 2; v.pe[i] = *++pc, i++);
2077  // In the IEEE 754 Standard the leading 1 is not represented.                    
2078    v.pe[0] &= X_EXPO_MASK;
2079  // Now in pe[0],pe[1] we have 31 bits of mantissa. 
2080  // But only the first 23 ones must be put in the   
2081  // final real_4 number.                             
2082    xrshift (xF_lex - 1, v.pe, 2);
2083  // We have just loaded the mantissa and now we 
2084  // are going to load exponent and sign.        
2085    v.pe[0] |= (e << (16 - xF_lex));
2086    v.pe[0] |= u;
2087  #if __BYTE_ORDER == __LITTLE_ENDIAN
2088    u = v.pe[0];
2089    v.pe[0] = v.pe[1];
2090    v.pe[1] = u;
2091  #endif
2092    return v.f;
2093  }
2094  
2095  real_32 flttox (real_4 y)
2096  {
2097    REAL32 pe; 
2098    unt_2 *pc, u;
2099    int_2 i, e;
2100    if (y < FLT_MIN && y > -FLT_MIN) {
2101      return X_0;
2102    }
2103    bzero (pe, sizeof (pe));
2104    pc = (unt_2 *) &y;
2105  #if __BYTE_ORDER == __LITTLE_ENDIAN
2106    pc += 1;
2107  #endif
2108    u = *pc & X_SIGN_MASK;
2109    e = xF_bias + ((*pc & X_EXPO_MASK) >> (16 - xF_lex));
2110  // Now u is the sign of y and e is the
2111  // biased exponent (exponent + bias).  
2112  #if __BYTE_ORDER == __LITTLE_ENDIAN
2113    for (i = 1; i < 3; pe[i] = *pc--, i++);
2114  #else
2115    for (i = 1; i < 3; pe[i] = *pc++, i++);
2116  #endif
2117    pc = pe + 1;
2118    xlshift (xF_lex - 1, pc, 2);
2119    *pc |= X_SIGN_MASK;
2120  // We have just put in pe[1],pe[2] the whole 
2121  // mantissa of y with a leading 1.           
2122  // Now we have only to put exponent and sign 
2123  // in pe[0].                                 
2124    *pe = e;
2125    *pe |= u;
2126    return *(real_32 *) pe;
2127  }
2128  
2129  real_8 xtodbl (real_32 s)
2130  {
2131    union {unt_2 pe[4]; real_8 d;} v;
2132    unt_2 *pc, u;
2133    int_2 i, e;
2134    pc = SEXP (s);
2135    u = *pc & X_SIGN_MASK;
2136    e = (*pc & X_EXPO_MASK) - xD_bias;
2137    if (e >= xD_max) {
2138      return (!u ? DBL_MAX : -DBL_MAX);
2139    }
2140    if (e <= 0) {
2141      return 0.;
2142    }
2143    for (i = 0; i < 4; v.pe[i] = *++pc, i++);
2144    v.pe[0] &= X_EXPO_MASK;
2145    xrshift (xD_lex - 1, v.pe, 4);
2146    v.pe[0] |= (e << (16 - xD_lex));
2147    v.pe[0] |= u;
2148  #if __BYTE_ORDER == __LITTLE_ENDIAN
2149    u = v.pe[3];
2150    v.pe[3] = v.pe[0];
2151    v.pe[0] = u;
2152    u = v.pe[2];
2153    v.pe[2] = v.pe[1];
2154    v.pe[1] = u;
2155  #endif
2156    return v.d;
2157  }
2158  
2159  real_32 dbltox (real_8 y)
2160  {
2161    REAL32 pe;
2162    unt_2 *pc, u;
2163    int_2 i, e;
2164    if (y < DBL_MIN && y > -DBL_MIN) {
2165      return X_0;
2166    }
2167    bzero (pe, sizeof (pe));
2168    pc = (unt_2 *) &y;
2169  #if __BYTE_ORDER == __LITTLE_ENDIAN
2170    pc += 3;
2171  #endif
2172    u = *pc & X_SIGN_MASK;
2173    e = xD_bias + ((*pc & X_EXPO_MASK) >> (16 - xD_lex));
2174  #if __BYTE_ORDER == __LITTLE_ENDIAN
2175    for (i = 1; i < 5; pe[i] = *pc--, i++);
2176  #else
2177    for (i = 1; i < 5; pe[i] = *pc++, i++);
2178  #endif
2179    pc = pe + 1;
2180    xlshift (xD_lex - 1, pc, 4);
2181    *pc |= X_SIGN_MASK;
2182    *pe = e;
2183    *pe |= u;
2184    return *(real_32 *) pe;
2185  }
2186  
2187  real_32 _quadtox (real_16 x)
2188  {
2189  // Intrinsic function so arguments are not pointers.
2190    REAL32 z;
2191    unt_2 *y;
2192    int_4 i;
2193    if (x == 0.0q) {
2194      return X_0;
2195    }
2196    int_4 sinf = isinfq (x);
2197    if (sinf == 1) {
2198      return X_PLUS_INF;
2199    } else if (sinf == -1) {
2200      return X_MINUS_INF;
2201    }
2202    if (isnanq (x)) {
2203      return X_NAN;
2204    }
2205    bzero (z, sizeof (z));
2206    y = (unt_2 *) &x;
2207    for (i = 0; i <= 7; i++) {
2208  #if __BYTE_ORDER == __LITTLE_ENDIAN
2209      z[i] = y[7 - i];
2210  #else
2211      z[i] = y[i];
2212  #endif
2213    }
2214  // real_16 skips the default first bit, HPA lib does not.
2215    unt_2 cy = 0x8000;
2216    for (i = 1; i < 8; i++) {
2217      if (z[i] & 0x1) {
2218        z[i] = (z[i] >> 1) | cy;
2219        cy = 0x8000;
2220      } else {
2221        z[i] = (z[i] >> 1) | cy;
2222        cy = 0x0;
2223      }
2224    }
2225    z[8] |= cy;
2226    return *(real_32 *) z;
2227  }
2228  
2229  real_32 _quadtop (real_32 *z, real_16 x)
2230  {
2231    *z = _quadtox (x);
2232    return *z;
2233  }
2234  
2235  real_16 _xtoquad (real_32 x)
2236  {
2237  // Intrinsic function so arguments are not pointers.
2238    REAL16 y;
2239    REAL32 z;
2240    int_4 i;
2241    memcpy (z, x.value, sizeof (real_32));
2242  // Catch NaN, +-Inf is handled correctly.
2243    if (xis_nan (&x)) {
2244      z[0] = 0x7fff;
2245      z[1] = 0xffff;
2246    }
2247  // real_16 skips the default first bit, HPA lib does not.
2248    unt_2 cy = (z[8] & 0x8000 ? 0x1 : 0x0);
2249    for (i = 7; i > 0; i--) {
2250      if (z[i] & 0x8000) {
2251        z[i] = (z[i] << 1) | cy;
2252        cy = 0x1;
2253      } else {
2254        z[i] = (z[i] << 1) | cy;
2255        cy = 0x0;
2256      }
2257    }
2258    for (i = 0; i < 8; i++) {
2259  #if __BYTE_ORDER == __LITTLE_ENDIAN
2260      y[i] = z[7 - i];
2261  #else
2262      y[i] = z[i];
2263  #endif
2264    }
2265  // Avoid 'dereferencing type-punned pointer will break strict-aliasing rules'
2266    real_16 u;
2267    memcpy (&u, &y[0], sizeof (real_16));
2268    return u;
2269  }
2270  
2271  real_32 strtox (char *s, char **end)
2272  {
2273  // This routine replaces the HPA solution - MvdV.
2274  // Algol 68 Genie employs the same algorithm.
2275  //
2276  // Initialize.
2277  #define N (FLT256_DIG + FLT256_GUARD)
2278    errno = 0;
2279    real_32 y[N];
2280    if (end != NO_REF_TEXT && (*end) != NO_TEXT) {
2281      (*end) = &(s[0]);
2282    }
2283    while (isspace (s[0])) {
2284      s++;
2285    }
2286    real_32 W;
2287    if (s[0] == '-') {
2288      W = X_MINUS_1;
2289      s++;
2290    } else if (s[0] == '+') {
2291      W = X_1;
2292      s++;
2293    } else {
2294      W = X_1;
2295    }
2296  // Scan mantissa digits and put them into "y".
2297    while (s[0] == '0') {
2298      s++;
2299    }
2300    int dot = -1, pos = 0, pow = 0;
2301    while (pow < N && (isdigit (s[pos]) || s[pos] == '.')) {
2302      if (s[pos] == '.') {
2303        dot = pos;
2304      } else {
2305        switch (s[pos]) {
2306          case '0': y[pow] = X_0; break;
2307          case '1': y[pow] = W; break;
2308          case '2': y[pow] = xmul (X_2, W); break;
2309          case '3': y[pow] = xmul (X_3, W); break;
2310          case '4': y[pow] = xmul (X_4, W); break;
2311          case '5': y[pow] = xmul (X_5, W); break;
2312          case '6': y[pow] = xmul (X_6, W); break;
2313          case '7': y[pow] = xmul (X_7, W); break;
2314          case '8': y[pow] = xmul (X_8, W); break;
2315          case '9': y[pow] = xmul (X_9, W); break;
2316        }
2317        W = xdiv (W, X_10);
2318        pow++;
2319      }
2320      pos++;
2321    }
2322  // Skip trailing digits.
2323    while (isdigit (s[pos])) {
2324      pos++;
2325    }
2326    if (end != NO_REF_TEXT && (*end) != NO_TEXT) {
2327      (*end) = &(s[pos]);
2328    }
2329  // Sum from low to high to preserve precision.
2330    real_32 sum = X_0;
2331    for (int k = pow - 1; k >= 0; k--) {
2332      sum = xsum (sum, y[k]);
2333    }
2334  // Optional exponent.
2335    int expo = 0;
2336    while (s[pos] == ' ') {
2337      pos++;
2338    }
2339    if (tolower (s[pos]) == 'e' || tolower (s[pos]) == 'q' || tolower (s[pos]) == 'x') {
2340      pos++;
2341      if (isdigit (s[pos]) || s[pos] == '-' || s[pos] == '+') {
2342        expo = (int) strtol (&(s[pos]), end, 10);
2343      }
2344    }
2345  // Standardize.
2346    if (dot >= 0) {
2347      expo += dot - 1;
2348    } else {
2349      expo += pow - 1;
2350    }
2351    while (xnot0 (&sum) && xlt1 (sum)) {
2352      sum = xmul (sum, X_10);
2353      expo--;
2354    }
2355    if (errno == 0) {
2356       return xmul (sum, xtenup (expo));
2357    } else {
2358      return X_0;
2359    }
2360  #undef N
2361  }
2362  
2363  real_32 atox (char *q)
2364  {
2365    return strtox (q, NO_REF_TEXT);
2366  }
2367  
2368  int_4 _xint4 (real_32 x_)
2369  {
2370    static real_16 y_;
2371    y_ = _xtoquad (x_);
2372    return (int_4) _qnint (y_);
2373  }
2374  
2375  int_4 _xnint4 (real_32 x_)
2376  {
2377    static real_16 y_;
2378    if (xgt (x_, X_0)) {
2379      y_ = _xtoquad (xsum (x_, X_1_OVER_2));
2380    }
2381    else {
2382      y_ = _xtoquad (xsub (x_, X_1_OVER_2));
2383    }
2384    return (int_4) (y_);
2385  }
2386  
2387  int_8 _xint8 (real_32 x_)
2388  {
2389    static real_16 y_;
2390    y_ = _xtoquad (x_);
2391    return (int_8) (y_);
2392  }
2393  
2394  int_8 _xnint8 (real_32 x_)
2395  {
2396    static real_16 y_;
2397    if (xgt (x_, X_0)) {
2398      y_ = _xtoquad (xsum (x_, X_1_OVER_2));
2399    }
2400    else {
2401      y_ = _xtoquad (xsub (x_, X_1_OVER_2));
2402    }
2403    return (int_8) (y_);
2404  }
2405  
2406  void _xdump (real_32 *z)
2407  {
2408    printf ("{{");
2409    for (int i = 0; i <= FLT256_LEN; i++) {
2410      printf("0x%04x", (z->value)[i]);
2411      if (i < FLT256_LEN) {
2412        printf(", ");
2413      }
2414    }
2415    printf ("}}\n");
2416    fflush (stdout);
2417  }
2418  
2419  real_32 _x1mach (int_4 *i)
2420  {
2421    switch (*i) {
2422  //    d1mach(1) = b**(emin-1), the smallest positive magnitude. 
2423    case 1: return FLT256_MIN;
2424  //    d1mach(2) = b**emax*(1 - b**(-t)), the largest magnitude. 
2425    case 2: return FLT256_MAX;
2426  //    d1mach(3) = b**(-t), the smallest relative spacing. 
2427    case 3: return FLT256_EPSILON_HALF;
2428  //    d1mach(4) = b**(1-t), the largest relative spacing. 
2429    case 4: return FLT256_EPSILON;
2430  //    d1mach(5) = log10(b) 
2431    case 5: return X_LOG10_2;
2432  //
2433    default: return X_0;
2434    }
2435  }
2436  
     


© 2002-2025 J.M. van der Veer (jmvdveer@xs4all.nl)