rts-math.c

     
   1  //! @file rts-math.c
   2  //! @author J. Marcel van der Veer
   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 math operations.
  25  
  26  #include <vif.h>
  27  
  28  // INTEGER*4
  29  
  30  int_4 _up_int_4 (int_4 m, int_4 n)
  31  {
  32  // Special cases.
  33    if (m == 2 && n >= 0) {
  34      return 1 << n;
  35    } else if (m == 1) {
  36      return m;
  37    } else if (m == -1) {
  38      return (n % 2 == 0 ? 1 : -1);
  39    } else if (m == 0) {
  40      return (n == 0 ? 1 : 0);
  41    } else if (n < 0) {
  42      return 0;
  43    }
  44  // General case.
  45    unsigned bit = 1;
  46    int_4 M = m, P = 1;
  47    do {
  48      if (n & bit) {
  49        P *= M;
  50      }
  51      bit <<= 1;
  52      if ((int_4) bit <= n) {
  53        M *= M;
  54      }
  55    } while ((int_4) bit <= n);
  56    return P;
  57  }
  58  
  59  // INTEGER*8
  60  
  61  int_8 _up_int_8 (int_8 m, int_4 n)
  62  {
  63  // Special cases.
  64    if (m == 2 && n >= 0) {
  65      int_8 s = 1;
  66      s <<= n;
  67      return s;
  68    } else if (m == 1) {
  69      return m;
  70    } else if (m == -1) {
  71      return (n % 2 == 0 ? 1 : -1);
  72    } else if (m == 0) {
  73      return (n == 0 ? 1 : 0);
  74    } else if (n < 0) {
  75      return 0;
  76    }
  77  // General case.
  78    unsigned bit = 1;
  79    int_8 M = m, P = 1;
  80    do {
  81      if (n & bit) {
  82        P *= M;
  83      }
  84      bit <<= 1;
  85      if ((int_4) bit <= n) {
  86        M *= M;
  87      }
  88    } while ((int_4) bit <= n);
  89    return P;
  90  }
  91  
  92  // REAL*4
  93  
  94  real_4 _aint (real_4 x)
  95  {
  96    return (x >= 0 ? floorf (x) : - floorf (- x));
  97  }
  98  
  99  real_4 _anint (real_4 x)
 100  {
 101    return (x >= 0 ? floorf (0.5 + x) : - floorf (0.5 - x));
 102  }
 103  
 104  real_4 _up_real_4 (real_4 x, int_4 n)
 105  {
 106  // Only positive n.
 107    if (n < 0) {
 108      return 1 / _up_real_4 (x, -n);
 109    }
 110  // Special cases.
 111    if (x == 0 || x == 1) {
 112      return x;
 113    } else if (x == -1) {
 114      return (n % 2 == 0 ? 1 : -1);
 115    }
 116  // General case.
 117    unsigned bit = 1;
 118    real_4 M = x, P = 1;
 119    do {
 120      if (n & bit) {
 121        P *= M;
 122      }
 123      bit <<= 1;
 124      if ((int_4) bit <= n) {
 125        M *= M;
 126      }
 127    } while ((int_4) bit <= n);
 128    return P;
 129  }
 130  
 131  real_4 cotanf (real_4 x)
 132  {
 133    return 1.0 / tanf (x);
 134  }
 135  
 136  // REAL*8
 137  
 138  real_8 _dint (real_8 x)
 139  {
 140    return (x >= 0 ? floor (x) : - floor (- x));
 141  }
 142  
 143  real_8 _dnint (real_8 x)
 144  {
 145    return (x >= 0 ? floor (0.5 + x) : - floor (0.5 - x));
 146  }
 147  
 148  real_8 _up_real_8 (real_8 x, int_4 n)
 149  {
 150  // Only positive n.
 151    if (n < 0) {
 152      return 1 / _up_real_8 (x, -n);
 153    }
 154  // Special cases.
 155    if (x == 0 || x == 1) {
 156      return x;
 157    } else if (x == -1) {
 158      return (n % 2 == 0 ? 1 : -1);
 159    }
 160  // General case.
 161    unsigned bit = 1;
 162    real_8 M = x, P = 1;
 163    do {
 164      if (n & bit) {
 165        P *= M;
 166      }
 167      bit <<= 1;
 168      if ((int_4) bit <= n) {
 169        M *= M;
 170      }
 171    } while ((int_4) bit <= n);
 172    return P;
 173  }
 174  
 175  real_8 cotan (real_8 x)
 176  {
 177    return 1.0 / tan (x);
 178  }
 179  
 180  real_8 _vif_inverfc (real_8 y)
 181  {
 182  #define N_c_inverfc 34
 183  
 184    static const real_8 c_inverfc[N_c_inverfc] = {
 185      0.91646139826896400000,
 186      0.48882664027310800000,
 187      0.23172920032340500000,
 188      0.12461045461371200000,
 189      -0.07288467655856750000,
 190      0.26999930867002900000,
 191      0.15068904736022300000,
 192      0.11606502534161400000,
 193      0.49999930343979000000,
 194      3.97886080735226000000,
 195      0.00112648096188977922,
 196      1.05739299623423047e-4,
 197      0.00351287146129100025,
 198      7.71708358954120939e-4,
 199      0.00685649426074558612,
 200      0.00339721910367775861,
 201      0.01127491693325048700,
 202      0.01185981170477711040,
 203      0.01429619886978980180,
 204      0.03464942077890999220,
 205      0.00220995927012179067,
 206      0.07434243572417848610,
 207      0.10587217794159548800,
 208      0.01472979383314851210,
 209      0.31684763852013594400,
 210      0.71365763586873036400,
 211      1.05375024970847138000,
 212      1.21448730779995237000,
 213      1.16374581931560831000,
 214      0.95646497474479900600,
 215      0.68626594827409781600,
 216      0.43439749233143011500,
 217      0.24404451059319093500,
 218      0.12078223763524522200
 219    };
 220  
 221    if (y < 0 || y > 2) {
 222      errno = ERANGE;
 223      return 0;
 224    }
 225    if (y == 0) {
 226      return DBL_MAX;
 227    } else if (y == 1) {
 228      return 0;
 229    } else if (y == 2) {
 230      return -DBL_MAX;
 231    } else {
 232  // Next is based on code that originally contained following statement:
 233  //   Copyright (c) 1996 Takuya Ooura. You may use, copy, modify this 
 234  //   code for any purpose and without fee.
 235      real_8 s, t, u, v, x, z;
 236      z = (y <= 1 ? y : 2 - y);
 237      v = c_inverfc[0] - ln (z);
 238      u = sqrt (v);
 239      s = (ln (u) + c_inverfc[1]) / v;
 240      t = 1 / (u + c_inverfc[2]);
 241      x = u *(1 - s *(s *c_inverfc[3] + 0.5)) - ((((c_inverfc[4] *t + c_inverfc[5]) *t + c_inverfc[6]) *t + c_inverfc[7]) *t + c_inverfc[8]) *t;
 242      t = c_inverfc[9] / (x + c_inverfc[9]);
 243      u = t - 0.5;
 244      s = (((((((((c_inverfc[10] *u + c_inverfc[11]) *u - c_inverfc[12]) *u - c_inverfc[13]) *u + c_inverfc[14]) *u + c_inverfc[15]) *u - c_inverfc[16]) *u - c_inverfc[17]) *u + c_inverfc[18]) *u + c_inverfc[19]) *u + c_inverfc[20];
 245      s = ((((((((((((s *u - c_inverfc[21]) *u - c_inverfc[22]) *u + c_inverfc[23]) *u + c_inverfc[24]) *u + c_inverfc[25]) *u + c_inverfc[26]) *u + c_inverfc[27]) *u + c_inverfc[28]) *u + c_inverfc[29]) *u + c_inverfc[30]) *u + c_inverfc[31]) *u + c_inverfc[32]) *t - z *exp (x *x - c_inverfc[33]);
 246      x += s *(x *s + 1);
 247      return (y <= 1 ? x : -x);
 248    }
 249  }
 250  
 251  real_8 _vif_inverf (real_8 y)
 252  {
 253    return _vif_inverfc (1 - y);
 254  }
 255  
 256  void _merfi (real_8 *p, real_8 *x, int_4 *err)
 257  {
 258    (*x) = _vif_inverf (*p);
 259    (*err) = errno;
 260  }
 261  
 262  // REAL*16
 263  
 264  real_16 _qint (real_16 x)
 265  {
 266    return (x >= 0 ? floorq (x) : - floorq (- x));
 267  }
 268  
 269  real_16 _qext (real_8 x)
 270  {
 271    RECORD str;
 272    _srecordf (str, "%.*le", DBL_DIG, x);
 273    return _strtoquad (str, NO_REF_TEXT);
 274  }
 275  
 276  real_16 _qmod (real_16 x, real_16 y)
 277  {
 278    real_16 q;
 279    if ((q = x / y) >= 0) {
 280      q = floorq (q);
 281    } else {
 282      q = -floorq (-q);
 283    }
 284    return (x - y *q);
 285  }
 286  
 287  real_16 _up_real_16 (real_16 x, int_4 n)
 288  {
 289  // Only positive n.
 290    if (n < 0) {
 291      return 1.0q / _up_real_16 (x, -n);
 292    }
 293  // Special cases.
 294    if (x == 0.0q || x == 1.0q) {
 295      return x;
 296    } else if (x == -1.0q) {
 297      return (n % 2 == 0 ? 1.0q : -1.0q);
 298    }
 299  // General case.
 300    unsigned bit = 1;
 301    real_16 M = x, P = 1.0q;
 302    do {
 303      if (n & bit) {
 304        P *= M;
 305      }
 306      bit <<= 1;
 307      if ((int_4) bit <= n) {
 308        M *= M;
 309      }
 310    } while ((int_4) bit <= n);
 311    return P;
 312  }
 313  
 314  real_16 cotanq (real_16 x)
 315  {
 316    return 1.0q / tanq (x);
 317  }
 318  
 319  // COMPLEX*8
 320  
 321  complex_8 _cmplxd (complex_16 z)
 322  {
 323    return CMPLXF ((real_4) creal (z), (real_4) cimag (z));
 324  }
 325  
 326  complex_8 _up_complex_8 (complex_8 x, int_4 n)
 327  {
 328  // Only positive n.
 329    if (n < 0) {
 330      return 1 / _up_complex (x, -n);
 331    }
 332  // General case.
 333    unsigned bit = 1;
 334    complex_8 M = x, P = 1;
 335    do {
 336      if (n & bit) {
 337        P *= M;
 338      }
 339      bit <<= 1;
 340      if ((int_4) bit <= n) {
 341        M *= M;
 342      }
 343    } while ((int_4) bit <= n);
 344    return P;
 345  }
 346  
 347  // COMPLEX*16
 348  
 349  complex_16 _dcmplxq (complex_32 z)
 350  {
 351    return CMPLX (crealq (z), cimagq (z));
 352  }
 353  
 354  complex_16 _up_complex (complex_16 x, int_4 n)
 355  {
 356  // Only positive n.
 357    if (n < 0) {
 358      return 1 / _up_complex (x, -n);
 359    }
 360  // General case.
 361    unsigned bit = 1;
 362    complex_16 M = x, P = 1;
 363    do {
 364      if (n & bit) {
 365        P *= M;
 366      }
 367      bit <<= 1;
 368      if ((int_4) bit <= n) {
 369        M *= M;
 370      }
 371    } while ((int_4) bit <= n);
 372    return P;
 373  }
 374  
 375  // COMPLEX*32
 376  
 377  complex_32 _qcmplxd (complex_16 z)
 378  {
 379    return CMPLXQ (_qext (creal (z)), _qext (cimag (z)));
 380  }
 381  
 382  complex_32 _up_complex_32 (complex_32 x, int_4 n)
 383  {
 384  // Only positive n.
 385    if (n < 0) {
 386      return 1.0q / _up_complex_32 (x, -n);
 387    }
 388  // General case.
 389    unsigned bit = 1;
 390    complex_32 M = x, P = 1.0q;
 391    do {
 392      if (n & bit) {
 393        P *= M;
 394      }
 395      bit <<= 1;
 396      if ((int_4) bit <= n) {
 397        M *= M;
 398      }
 399    } while ((int_4) bit <= n);
 400    return P;
 401  }
 402  
 403  real_4 _zabs_8 (real_4 re, real_4 im)
 404  {
 405    return cabsf (CMPLXF (re, im));
 406  }
 407  
 408  real_8 _zabs_16 (real_8 re, real_8 im)
 409  {
 410    return cabs (CMPLX (re, im));
 411  }
 412  
 413  real_16 _zabs_32 (real_16 re, real_16 im)
 414  {
 415    return cabsq (CMPLXQ (re, im));
 416  }
 417  
 418  void _qhex (real_16 *u)
 419  {
 420    unt_2 *p = (unt_2 *) u;
 421    fprintf (stdout, "{");
 422    for (int_4 i = 0; i <= FLT128_LEN; i++) {
 423      fprintf (stdout, "0x%04x", *(p++));
 424      if (i < FLT128_LEN) {
 425        fprintf (stdout, ", ");
 426      }
 427    }
 428    fprintf (stdout, "}\n");
 429  }
 430  
 431  void _xhex (real_32 *u)
 432  {
 433    fprintf (stdout, "{");
 434    for (int_4 i = 0; i <= FLT256_LEN; i++) {
 435      fprintf (stdout, "0x%04x", u->value[i]);
 436      if (i < FLT256_LEN) {
 437        fprintf (stdout, ", ");
 438      }
 439    }
 440    fprintf (stdout, "}\n");
 441  }
 442  
 443  //
 444  
 445  void _pi4 (real_4 * x)
 446  {
 447    *x = (real_4) M_PI;
 448  }
 449  
 450  void _pi8 (real_8 * x)
 451  {
 452    *x = M_PI;
 453  }
 454  
 455  void _pi16 (real_16 * x)
 456  {
 457    *x = M_PIq;
 458  }
     


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