| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306 | /* * src/dd_real.cc * * This work was supported by the Director, Office of Science, Division * of Mathematical, Information, and Computational Sciences of the * U.S. Department of Energy under contract number DE-AC03-76SF00098. * * Copyright (c) 2000-2007 * * Contains implementation of non-inlined functions of double-double * package.  Inlined functions are found in dd_inline.h (in include directory). */#include <cstdlib>#include <cstdio>#include <cmath>#include <cstring>#include <iostream>#include <iomanip>#include <string>#include "config.h"#include <qd/dd_real.h>#include "util.h"#include <qd/bits.h>#ifndef QD_INLINE#include <qd/dd_inline.h>#endifusing std::cout;using std::cerr;using std::endl;using std::ostream;using std::istream;using std::ios_base;using std::string;using std::setw;/* This routine is called whenever a fatal error occurs. */void dd_real::error(const char *msg) {   if (msg) { cerr << "ERROR " << msg << endl; }}/* Computes the square root of the double-double number dd.   NOTE: dd must be a non-negative number.                   */QD_API dd_real sqrt(const dd_real &a) {  /* Strategy:  Use Karp's trick:  if x is an approximation     to sqrt(a), then        sqrt(a) = a*x + [a - (a*x)^2] * x / 2   (approx)     The approximation is accurate to twice the accuracy of x.     Also, the multiplication (a*x) and [-]*x can be done with     only half the precision.  */  if (a.is_zero())    return 0.0;  if (a.is_negative()) {    dd_real::error("(dd_real::sqrt): Negative argument.");    return dd_real::_nan;  }  double x = 1.0 / std::sqrt(a.x[0]);  double ax = a.x[0] * x;  return dd_real::add(ax, (a - dd_real::sqr(ax)).x[0] * (x * 0.5));}/* Computes the square root of a double in double-double precision.    NOTE: d must not be negative.                                   */dd_real dd_real::sqrt(double d) {  return ::sqrt(dd_real(d));}/* Computes the n-th root of the double-double number a.   NOTE: n must be a positive integer.     NOTE: If n is even, then a must not be negative.       */dd_real nroot(const dd_real &a, int n) {  /* Strategy:  Use Newton iteration for the function          f(x) = x^(-n) - a     to find its root a^{-1/n}.  The iteration is thus          x' = x + x * (1 - a * x^n) / n     which converges quadratically.  We can then find     a^{1/n} by taking the reciprocal.  */  if (n <= 0) {    dd_real::error("(dd_real::nroot): N must be positive.");    return dd_real::_nan;  }  if (n%2 == 0 && a.is_negative()) {    dd_real::error("(dd_real::nroot): Negative argument.");    return dd_real::_nan;  }  if (n == 1) {    return a;  }   if (n == 2) {    return sqrt(a);  }  if (a.is_zero())    return 0.0;  /* Note  a^{-1/n} = exp(-log(a)/n) */  dd_real r = abs(a);  dd_real x = std::exp(-std::log(r.x[0]) / n);  /* Perform Newton's iteration. */  x += x * (1.0 - r * npwr(x, n)) / static_cast<double>(n);  if (a.x[0] < 0.0)    x = -x;  return 1.0/x;}/* Computes the n-th power of a double-double number.    NOTE:  0^0 causes an error.                         */dd_real npwr(const dd_real &a, int n) {    if (n == 0) {    if (a.is_zero()) {      dd_real::error("(dd_real::npwr): Invalid argument.");      return dd_real::_nan;    }    return 1.0;  }  dd_real r = a;  dd_real s = 1.0;  int N = std::abs(n);  if (N > 1) {    /* Use binary exponentiation */    while (N > 0) {      if (N % 2 == 1) {        s *= r;      }      N /= 2;      if (N > 0)        r = sqr(r);    }  } else {    s = r;  }  /* Compute the reciprocal if n is negative. */  if (n < 0)    return (1.0 / s);    return s;}dd_real pow(const dd_real &a, int n) {  return npwr(a, n);}dd_real pow(const dd_real &a, const dd_real &b) {  return exp(b * log(a));}static const int n_inv_fact = 15;static const double inv_fact[n_inv_fact][2] = {  { 1.66666666666666657e-01,  9.25185853854297066e-18},  { 4.16666666666666644e-02,  2.31296463463574266e-18},  { 8.33333333333333322e-03,  1.15648231731787138e-19},  { 1.38888888888888894e-03, -5.30054395437357706e-20},  { 1.98412698412698413e-04,  1.72095582934207053e-22},  { 2.48015873015873016e-05,  2.15119478667758816e-23},  { 2.75573192239858925e-06, -1.85839327404647208e-22},  { 2.75573192239858883e-07,  2.37677146222502973e-23},  { 2.50521083854417202e-08, -1.44881407093591197e-24},  { 2.08767569878681002e-09, -1.20734505911325997e-25},  { 1.60590438368216133e-10,  1.25852945887520981e-26},  { 1.14707455977297245e-11,  2.06555127528307454e-28},  { 7.64716373181981641e-13,  7.03872877733453001e-30},  { 4.77947733238738525e-14,  4.39920548583408126e-31},  { 2.81145725434552060e-15,  1.65088427308614326e-31}};/* Exponential.  Computes exp(x) in double-double precision. */dd_real exp(const dd_real &a) {  /* Strategy:  We first reduce the size of x by noting that               exp(kr + m * log(2)) = 2^m * exp(r)^k     where m and k are integers.  By choosing m appropriately     we can make |kr| <= log(2) / 2 = 0.347.  Then exp(r) is      evaluated using the familiar Taylor series.  Reducing the      argument substantially speeds up the convergence.       */    const double k = 512.0;  const double inv_k = 1.0 / k;  if (a.x[0] <= -709.0)    return 0.0;  if (a.x[0] >=  709.0)    return dd_real::_inf;  if (a.is_zero())    return 1.0;  if (a.is_one())    return dd_real::_e;  double m = std::floor(a.x[0] / dd_real::_log2.x[0] + 0.5);  dd_real r = mul_pwr2(a - dd_real::_log2 * m, inv_k);  dd_real s, t, p;  p = sqr(r);  s = r + mul_pwr2(p, 0.5);  p *= r;  t = p * dd_real(inv_fact[0][0], inv_fact[0][1]);  int i = 0;  do {    s += t;    p *= r;    ++i;    t = p * dd_real(inv_fact[i][0], inv_fact[i][1]);  } while (std::abs(to_double(t)) > inv_k * dd_real::_eps && i < 5);  s += t;  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s = mul_pwr2(s, 2.0) + sqr(s);  s += 1.0;  return ldexp(s, static_cast<int>(m));}/* Logarithm.  Computes log(x) in double-double precision.   This is a natural logarithm (i.e., base e).            */dd_real log(const dd_real &a) {  /* Strategy.  The Taylor series for log converges much more     slowly than that of exp, due to the lack of the factorial     term in the denominator.  Hence this routine instead tries     to determine the root of the function         f(x) = exp(x) - a     using Newton iteration.  The iteration is given by         x' = x - f(x)/f'(x)             = x - (1 - a * exp(-x))            = x + a * exp(-x) - 1.                Only one iteration is needed, since Newton's iteration     approximately doubles the number of digits per iteration. */  if (a.is_one()) {    return 0.0;  }  if (a.x[0] <= 0.0) {    dd_real::error("(dd_real::log): Non-positive argument.");    return dd_real::_nan;  }  dd_real x = std::log(a.x[0]);   /* Initial approximation */  x = x + a * exp(-x) - 1.0;  return x;}dd_real log10(const dd_real &a) {  return log(a) / dd_real::_log10;}static const dd_real _pi16 = dd_real(1.963495408493620697e-01,                                     7.654042494670957545e-18);/* Table of sin(k * pi/16) and cos(k * pi/16). */static const double sin_table [4][2] = {  {1.950903220161282758e-01, -7.991079068461731263e-18},  {3.826834323650897818e-01, -1.005077269646158761e-17},  {5.555702330196021776e-01,  4.709410940561676821e-17},  {7.071067811865475727e-01, -4.833646656726456726e-17}};static const double cos_table [4][2] = {  {9.807852804032304306e-01, 1.854693999782500573e-17},  {9.238795325112867385e-01, 1.764504708433667706e-17},  {8.314696123025452357e-01, 1.407385698472802389e-18},  {7.071067811865475727e-01, -4.833646656726456726e-17}};/* Computes sin(a) using Taylor series.   Assumes |a| <= pi/32.                           */static dd_real sin_taylor(const dd_real &a) {  const double thresh = 0.5 * std::abs(to_double(a)) * dd_real::_eps;  dd_real r, s, t, x;  if (a.is_zero()) {    return 0.0;  }  int i = 0;  x = -sqr(a);  s = a;  r = a;  do {    r *= x;    t = r * dd_real(inv_fact[i][0], inv_fact[i][1]);    s += t;    i += 2;  } while (i < n_inv_fact && std::abs(to_double(t)) > thresh);  return s;}static dd_real cos_taylor(const dd_real &a) {  const double thresh = 0.5 * dd_real::_eps;  dd_real r, s, t, x;  if (a.is_zero()) {    return 1.0;  }  x = -sqr(a);  r = x;  s = 1.0 + mul_pwr2(r, 0.5);  int i = 1;  do {    r *= x;    t = r * dd_real(inv_fact[i][0], inv_fact[i][1]);    s += t;    i += 2;  } while (i < n_inv_fact && std::abs(to_double(t)) > thresh);  return s;}static void sincos_taylor(const dd_real &a,                           dd_real &sin_a, dd_real &cos_a) {  if (a.is_zero()) {    sin_a = 0.0;    cos_a = 1.0;    return;  }  sin_a = sin_taylor(a);  cos_a = sqrt(1.0 - sqr(sin_a));}dd_real sin(const dd_real &a) {    /* Strategy.  To compute sin(x), we choose integers a, b so that       x = s + a * (pi/2) + b * (pi/16)     and |s| <= pi/32.  Using the fact that        sin(pi/16) = 0.5 * sqrt(2 - sqrt(2 + sqrt(2)))     we can compute sin(x) from sin(s), cos(s).  This greatly      increases the convergence of the sine Taylor series. */  if (a.is_zero()) {    return 0.0;  }  // approximately reduce modulo 2*pi  dd_real z = nint(a / dd_real::_2pi);  dd_real r = a - dd_real::_2pi * z;  // approximately reduce modulo pi/2 and then modulo pi/16.  dd_real t;  double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5);  t = r - dd_real::_pi2 * q;  int j = static_cast<int>(q);  q = std::floor(t.x[0] / _pi16.x[0] + 0.5);  t -= _pi16 * q;  int k = static_cast<int>(q);  int abs_k = std::abs(k);  if (j < -2 || j > 2) {    dd_real::error("(dd_real::sin): Cannot reduce modulo pi/2.");    return dd_real::_nan;  }  if (abs_k > 4) {    dd_real::error("(dd_real::sin): Cannot reduce modulo pi/16.");    return dd_real::_nan;  }  if (k == 0) {    switch (j) {      case 0:        return sin_taylor(t);      case 1:        return cos_taylor(t);      case -1:        return -cos_taylor(t);      default:        return -sin_taylor(t);    }  }  dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]);  dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]);  dd_real sin_t, cos_t;  sincos_taylor(t, sin_t, cos_t);  if (j == 0) {    if (k > 0) {      r = u * sin_t + v * cos_t;    } else {      r = u * sin_t - v * cos_t;    }  } else if (j == 1) {    if (k > 0) {      r = u * cos_t - v * sin_t;    } else {      r = u * cos_t + v * sin_t;    }  } else if (j == -1) {    if (k > 0) {      r = v * sin_t - u * cos_t;    } else if (k < 0) {      r = -u * cos_t - v * sin_t;    }  } else {    if (k > 0) {      r = -u * sin_t - v * cos_t;    } else {      r = v * cos_t - u * sin_t;    }  }  return r;}dd_real cos(const dd_real &a) {  if (a.is_zero()) {    return 1.0;  }  // approximately reduce modulo 2*pi  dd_real z = nint(a / dd_real::_2pi);  dd_real r = a - z * dd_real::_2pi;  // approximately reduce modulo pi/2 and then modulo pi/16  dd_real t;  double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5);  t = r - dd_real::_pi2 * q;  int j = static_cast<int>(q);  q = std::floor(t.x[0] / _pi16.x[0] + 0.5);  t -= _pi16 * q;  int k = static_cast<int>(q);  int abs_k = std::abs(k);  if (j < -2 || j > 2) {    dd_real::error("(dd_real::cos): Cannot reduce modulo pi/2.");    return dd_real::_nan;  }  if (abs_k > 4) {    dd_real::error("(dd_real::cos): Cannot reduce modulo pi/16.");    return dd_real::_nan;  }  if (k == 0) {    switch (j) {      case 0:        return cos_taylor(t);      case 1:        return -sin_taylor(t);      case -1:        return sin_taylor(t);      default:        return -cos_taylor(t);    }  }  dd_real sin_t, cos_t;  sincos_taylor(t, sin_t, cos_t);  dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]);  dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]);  if (j == 0) {    if (k > 0) {      r = u * cos_t - v * sin_t;    } else {      r = u * cos_t + v * sin_t;    }  } else if (j == 1) {    if (k > 0) {      r = - u * sin_t - v * cos_t;    } else {      r = v * cos_t - u * sin_t;    }  } else if (j == -1) {    if (k > 0) {      r = u * sin_t + v * cos_t;    } else {      r = u * sin_t - v * cos_t;    }  } else {    if (k > 0) {      r = v * sin_t - u * cos_t;    } else {      r = - u * cos_t - v * sin_t;    }  }  return r;}void sincos(const dd_real &a, dd_real &sin_a, dd_real &cos_a) {  if (a.is_zero()) {    sin_a = 0.0;    cos_a = 1.0;    return;  }  // approximately reduce modulo 2*pi  dd_real z = nint(a / dd_real::_2pi);  dd_real r = a - dd_real::_2pi * z;  // approximately reduce module pi/2 and pi/16  dd_real t;  double q = std::floor(r.x[0] / dd_real::_pi2.x[0] + 0.5);  t = r - dd_real::_pi2 * q;  int j = static_cast<int>(q);  int abs_j = std::abs(j);  q = std::floor(t.x[0] / _pi16.x[0] + 0.5);  t -= _pi16 * q;  int k = static_cast<int>(q);  int abs_k = std::abs(k);  if (abs_j > 2) {    dd_real::error("(dd_real::sincos): Cannot reduce modulo pi/2.");    cos_a = sin_a = dd_real::_nan;    return;  }  if (abs_k > 4) {    dd_real::error("(dd_real::sincos): Cannot reduce modulo pi/16.");    cos_a = sin_a = dd_real::_nan;    return;  }  dd_real sin_t, cos_t;  dd_real s, c;  sincos_taylor(t, sin_t, cos_t);  if (abs_k == 0) {    s = sin_t;    c = cos_t;  } else {    dd_real u(cos_table[abs_k-1][0], cos_table[abs_k-1][1]);    dd_real v(sin_table[abs_k-1][0], sin_table[abs_k-1][1]);    if (k > 0) {      s = u * sin_t + v * cos_t;      c = u * cos_t - v * sin_t;    } else {      s = u * sin_t - v * cos_t;      c = u * cos_t + v * sin_t;    }  }  if (abs_j == 0) {    sin_a = s;    cos_a = c;  } else if (j == 1) {    sin_a = c;    cos_a = -s;  } else if (j == -1) {    sin_a = -c;    cos_a = s;  } else {    sin_a = -s;    cos_a = -c;  }  }dd_real atan(const dd_real &a) {  return atan2(a, dd_real(1.0));}dd_real atan2(const dd_real &y, const dd_real &x) {  /* Strategy: Instead of using Taylor series to compute      arctan, we instead use Newton's iteration to solve     the equation        sin(z) = y/r    or    cos(z) = x/r     where r = sqrt(x^2 + y^2).     The iteration is given by        z' = z + (y - sin(z)) / cos(z)          (for equation 1)        z' = z - (x - cos(z)) / sin(z)          (for equation 2)     Here, x and y are normalized so that x^2 + y^2 = 1.     If |x| > |y|, then first iteration is used since the      denominator is larger.  Otherwise, the second is used.  */  if (x.is_zero()) {        if (y.is_zero()) {      /* Both x and y is zero. */      dd_real::error("(dd_real::atan2): Both arguments zero.");      return dd_real::_nan;    }    return (y.is_positive()) ? dd_real::_pi2 : -dd_real::_pi2;  } else if (y.is_zero()) {    return (x.is_positive()) ? dd_real(0.0) : dd_real::_pi;  }  if (x == y) {    return (y.is_positive()) ? dd_real::_pi4 : -dd_real::_3pi4;  }  if (x == -y) {    return (y.is_positive()) ? dd_real::_3pi4 : -dd_real::_pi4;  }  dd_real r = sqrt(sqr(x) + sqr(y));  dd_real xx = x / r;  dd_real yy = y / r;  /* Compute double precision approximation to atan. */  dd_real z = std::atan2(to_double(y), to_double(x));  dd_real sin_z, cos_z;  if (std::abs(xx.x[0]) > std::abs(yy.x[0])) {    /* Use Newton iteration 1.  z' = z + (y - sin(z)) / cos(z)  */    sincos(z, sin_z, cos_z);    z += (yy - sin_z) / cos_z;  } else {    /* Use Newton iteration 2.  z' = z - (x - cos(z)) / sin(z)  */    sincos(z, sin_z, cos_z);    z -= (xx - cos_z) / sin_z;  }  return z;}dd_real tan(const dd_real &a) {  dd_real s, c;  sincos(a, s, c);  return s/c;}dd_real asin(const dd_real &a) {  dd_real abs_a = abs(a);  if (abs_a > 1.0) {    dd_real::error("(dd_real::asin): Argument out of domain.");    return dd_real::_nan;  }  if (abs_a.is_one()) {    return (a.is_positive()) ? dd_real::_pi2 : -dd_real::_pi2;  }  return atan2(a, sqrt(1.0 - sqr(a)));}dd_real acos(const dd_real &a) {  dd_real abs_a = abs(a);  if (abs_a > 1.0) {    dd_real::error("(dd_real::acos): Argument out of domain.");    return dd_real::_nan;  }  if (abs_a.is_one()) {    return (a.is_positive()) ? dd_real(0.0) : dd_real::_pi;  }  return atan2(sqrt(1.0 - sqr(a)), a);} dd_real sinh(const dd_real &a) {  if (a.is_zero()) {    return 0.0;  }  if (abs(a) > 0.05) {    dd_real ea = exp(a);    return mul_pwr2(ea - inv(ea), 0.5);  }  /* since a is small, using the above formula gives     a lot of cancellation.  So use Taylor series.   */  dd_real s = a;  dd_real t = a;  dd_real r = sqr(t);  double m = 1.0;  double thresh = std::abs((to_double(a)) * dd_real::_eps);  do {    m += 2.0;    t *= r;    t /= (m-1) * m;    s += t;  } while (abs(t) > thresh);  return s;}dd_real cosh(const dd_real &a) {  if (a.is_zero()) {    return 1.0;  }  dd_real ea = exp(a);  return mul_pwr2(ea + inv(ea), 0.5);}dd_real tanh(const dd_real &a) {  if (a.is_zero()) {    return 0.0;  }  if (std::abs(to_double(a)) > 0.05) {    dd_real ea = exp(a);    dd_real inv_ea = inv(ea);    return (ea - inv_ea) / (ea + inv_ea);  } else {    dd_real s, c;    s = sinh(a);    c = sqrt(1.0 + sqr(s));    return s / c;  }}void sincosh(const dd_real &a, dd_real &s, dd_real &c) {  if (std::abs(to_double(a)) <= 0.05) {    s = sinh(a);    c = sqrt(1.0 + sqr(s));  } else {    dd_real ea = exp(a);    dd_real inv_ea = inv(ea);    s = mul_pwr2(ea - inv_ea, 0.5);    c = mul_pwr2(ea + inv_ea, 0.5);  }}dd_real asinh(const dd_real &a) {  return log(a + sqrt(sqr(a) + 1.0));}dd_real acosh(const dd_real &a) {  if (a < 1.0) {    dd_real::error("(dd_real::acosh): Argument out of domain.");    return dd_real::_nan;  }  return log(a + sqrt(sqr(a) - 1.0));}dd_real atanh(const dd_real &a) {  if (abs(a) >= 1.0) {    dd_real::error("(dd_real::atanh): Argument out of domain.");    return dd_real::_nan;  }  return mul_pwr2(log((1.0 + a) / (1.0 - a)), 0.5);}QD_API dd_real fmod(const dd_real &a, const dd_real &b) {  dd_real n = aint(a / b);  return (a - b * n);}QD_API dd_real ddrand() {  static const double m_const = 4.6566128730773926e-10;  /* = 2^{-31} */  double m = m_const;  dd_real r = 0.0;  double d;  /* Strategy:  Generate 31 bits at a time, using lrand48      random number generator.  Shift the bits, and reapeat     4 times. */  for (int i = 0; i < 4; i++, m *= m_const) {//    d = lrand48() * m;    d = std::rand() * m;    r += d;  }  return r;}/* polyeval(c, n, x)   Evaluates the given n-th degree polynomial at x.   The polynomial is given by the array of (n+1) coefficients. */dd_real polyeval(const dd_real *c, int n, const dd_real &x) {  /* Just use Horner's method of polynomial evaluation. */  dd_real r = c[n];    for (int i = n-1; i >= 0; i--) {    r *= x;    r += c[i];  }  return r;}/* polyroot(c, n, x0)   Given an n-th degree polynomial, finds a root close to    the given guess x0.  Note that this uses simple Newton   iteration scheme, and does not work for multiple roots.  */QD_API dd_real polyroot(const dd_real *c, int n,     const dd_real &x0, int max_iter, double thresh) {  dd_real x = x0;  dd_real f;  dd_real *d = new dd_real[n];  bool conv = false;  int i;  double max_c = std::abs(to_double(c[0]));  double v;  if (thresh == 0.0) thresh = dd_real::_eps;  /* Compute the coefficients of the derivatives. */  for (i = 1; i <= n; i++) {    v = std::abs(to_double(c[i]));    if (v > max_c) max_c = v;    d[i-1] = c[i] * static_cast<double>(i);  }  thresh *= max_c;  /* Newton iteration. */  for (i = 0; i < max_iter; i++) {    f = polyeval(c, n, x);    if (abs(f) < thresh) {      conv = true;      break;    }    x -= (f / polyeval(d, n-1, x));  }  delete [] d;  if (!conv) {    dd_real::error("(dd_real::polyroot): Failed to converge.");    return dd_real::_nan;  }  return x;}/* Constructor.  Reads a double-double number from the string s   and constructs a double-double number.                         */dd_real::dd_real(const char *s) {  if (dd_real::read(s, *this)) {    dd_real::error("(dd_real::dd_real): INPUT ERROR.");    *this = dd_real::_nan;  }}dd_real &dd_real::operator=(const char *s) {  if (dd_real::read(s, *this)) {    dd_real::error("(dd_real::operator=): INPUT ERROR.");    *this = dd_real::_nan;  }  return *this;}/* Outputs the double-double number dd. */ostream &operator<<(ostream &os, const dd_real &dd) {  bool showpos = (os.flags() & ios_base::showpos) != 0;  bool uppercase =  (os.flags() & ios_base::uppercase) != 0;  return os << dd.to_string(os.precision(), os.width(), os.flags(),       showpos, uppercase, os.fill());}/* Reads in the double-double number a. */istream &operator>>(istream &s, dd_real &a) {  char str[255];  s >> str;  a = dd_real(str);  return s;}void dd_real::to_digits(char *s, int &expn, int precision) const {  int D = precision + 1;  /* number of digits to compute */  dd_real r = abs(*this);  int e;  /* exponent */  int i, d;  if (x[0] == 0.0) {    /* this == 0.0 */    expn = 0;    for (i = 0; i < precision; i++) s[i] = '0';    return;  }  /* First determine the (approximate) exponent. */  e = to_int(std::floor(std::log10(std::abs(x[0]))));  if (e < -300) {    r *= dd_real(10.0) ^ 300;    r /= dd_real(10.0) ^ (e + 300);  } else if (e > 300) {    r = ldexp(r, -53);    r /= dd_real(10.0) ^ e;    r = ldexp(r, 53);  } else {    r /= dd_real(10.0) ^ e;  }  /* Fix exponent if we are off by one */  if (r >= 10.0) {    r /= 10.0;    e++;  } else if (r < 1.0) {    r *= 10.0;    e--;  }  if (r >= 10.0 || r < 1.0) {    dd_real::error("(dd_real::to_digits): can't compute exponent.");    return;  }  /* Extract the digits */  for (i = 0; i < D; i++) {    d = static_cast<int>(r.x[0]);    r -= d;    r *= 10.0;    s[i] = static_cast<char>(d + '0');  }  /* Fix out of range digits. */  for (i = D-1; i > 0; i--) {    if (s[i] < '0') {      s[i-1]--;      s[i] += 10;    } else if (s[i] > '9') {      s[i-1]++;      s[i] -= 10;    }  }  if (s[0] <= '0') {    dd_real::error("(dd_real::to_digits): non-positive leading digit.");    return;  }  /* Round, handle carry */  if (s[D-1] >= '5') {    s[D-2]++;    i = D-2;    while (i > 0 && s[i] > '9') {      s[i] -= 10;      s[--i]++;    }  }  /* If first digit is 10, shift everything. */  if (s[0] > '9') {     e++;     for (i = precision; i >= 2; i--) s[i] = s[i-1];     s[0] = '1';    s[1] = '0';  }  s[precision] = 0;  expn = e;}/* Writes the double-double number into the character array s of length len.   The integer d specifies how many significant digits to write.   The string s must be able to hold at least (d+8) characters.     showpos indicates whether to use the + sign, and uppercase indicates   whether the E or e is to be used for the exponent. */void dd_real::write(char *s, int len, int precision,     bool showpos, bool uppercase) const {  string str = to_string(precision, 0, ios_base::scientific, showpos, uppercase);  std::strncpy(s, str.c_str(), len-1);  s[len-1] = 0;}void round_string(char *s, int precision, int *offset){	/*	 Input string must be all digits or errors will occur.	 */	int i;	int D = precision ;	/* Round, handle carry */	  if (D>0 && s[D] >= '5') {	    s[D-1]++;	    i = D-1;	    while (i > 0 && s[i] > '9') {	      s[i] -= 10;	      s[--i]++;	    }	  }	  /* If first digit is 10, shift everything. */	  if (s[0] > '9') {	    // e++; // don't modify exponent here	    for (i = precision; i >= 1; i--) s[i+1] = s[i];	    s[0] = '1';	    s[1] = '0';	    (*offset)++ ; // now offset needs to be increased by one	    precision++ ;	  }	  s[precision] = 0; // add terminator for array}string dd_real::to_string(int precision, int width, ios_base::fmtflags fmt,     bool showpos, bool uppercase, char fill) const {  string s;  bool fixed = (fmt & ios_base::fixed) != 0;  bool sgn = true;  int i, e = 0;  if (isnan()) {    s = uppercase ? "NAN" : "nan";    sgn = false;  } else {    if (*this < 0.0)      s += '-';    else if (showpos)      s += '+';    else      sgn = false;    if (isinf()) {      s += uppercase ? "INF" : "inf";    } else if (*this == 0.0) {      /* Zero case */      s += '0';      if (precision > 0) {        s += '.';        s.append(precision, '0');      }    } else {      /* Non-zero case */      int off = (fixed ? (1 + to_int(floor(log10(abs(*this))))) : 1);      int d = precision + off;      int d_with_extra = d;      if(fixed)    	  d_with_extra = std::max(60, d); // longer than the max accuracy for DD      // highly special case - fixed mode, precision is zero, abs(*this) < 1.0      // without this trap a number like 0.9 printed fixed with 0 precision prints as 0      // should be rounded to 1.      if(fixed && (precision == 0) && (abs(*this) < 1.0)){    	  if(abs(*this) >= 0.5)    		  s += '1';    	  else    		  s += '0';    	  return s;      }      // handle near zero to working precision (but not exactly zero)      if (fixed && d <= 0) {        s += '0';        if (precision > 0) {          s += '.';          s.append(precision, '0');        }      } else { // default        char *t; //  = new char[d+1];        int j;        if(fixed){        	t = new char[d_with_extra+1];        	to_digits(t, e, d_with_extra);        }        else{        	t = new char[d+1];        	to_digits(t, e, d);        }        off = e + 1;        if (fixed) {          // fix the string if it's been computed incorrectly          // round here in the decimal string if required          round_string(t, d, &off);          if (off > 0) {            for (i = 0; i < off; i++) s += t[i];            if (precision > 0) {              s += '.';              for (j = 0; j < precision; j++, i++) s += t[i];            }          } else {            s += "0.";            if (off < 0) s.append(-off, '0');            for (i = 0; i < d; i++) s += t[i];          }        } else {          s += t[0];          if (precision > 0) s += '.';          for (i = 1; i <= precision; i++)            s += t[i];        }		delete [] t;      }    }    // trap for improper offset with large values    // without this trap, output of values of the for 10^j - 1 fail for j > 28    // and are output with the point in the wrong place, leading to a dramatically off value    if(fixed && (precision > 0)){    	// make sure that the value isn't dramatically larger    	double from_string = atof(s.c_str());    	// if this ratio is large, then we've got problems    	if( fabs( from_string / this->x[0] ) > 3.0 ){    		int point_position;    		char temp;    		// loop on the string, find the point, move it up one    		// don't act on the first character    		for(i=1; i < s.length(); i++){    			if(s[i] == '.'){    				s[i] = s[i-1] ;    				s[i-1] = '.' ;    				break;    			}    		}        	from_string = atof(s.c_str());        	// if this ratio is large, then the string has not been fixed        	if( fabs( from_string / this->x[0] ) > 3.0 ){        		dd_real::error("Re-rounding unsuccessful in large number fixed point trap.") ;        	}    	}    }    if (!fixed && !isinf()) {      /* Fill in exponent part */      s += uppercase ? 'E' : 'e';      append_expn(s, e);    }  }  /* Fill in the blanks */  int len = s.length();  if (len < width) {    int delta = width - len;    if (fmt & ios_base::internal) {      if (sgn)        s.insert(static_cast<string::size_type>(1), delta, fill);      else        s.insert(static_cast<string::size_type>(0), delta, fill);    } else if (fmt & ios_base::left) {      s.append(delta, fill);    } else {      s.insert(static_cast<string::size_type>(0), delta, fill);    }  }  return s;}/* Reads in a double-double number from the string s. */int dd_real::read(const char *s, dd_real &a) {  const char *p = s;  char ch;  int sign = 0;  int point = -1;  int nd = 0;  int e = 0;  bool done = false;  dd_real r = 0.0;  int nread;    /* Skip any leading spaces */  while (*p == ' ')    p++;  while (!done && (ch = *p) != '\0') {    if (ch >= '0' && ch <= '9') {      int d = ch - '0';      r *= 10.0;      r += static_cast<double>(d);      nd++;    } else {      switch (ch) {      case '.':        if (point >= 0)          return -1;                point = nd;        break;      case '-':      case '+':        if (sign != 0 || nd > 0)          return -1;        sign = (ch == '-') ? -1 : 1;        break;      case 'E':      case 'e':        nread = std::sscanf(p+1, "%d", &e);        done = true;        if (nread != 1)          return -1;        break;      default:        return -1;      }    }        p++;  }  if (point >= 0) {    e -= (nd - point);  }  if (e != 0) {    r *= (dd_real(10.0) ^ e);  }  a = (sign == -1) ? -r : r;  return 0;}/* Debugging routines */void dd_real::dump(const string &name, std::ostream &os) const {  std::ios_base::fmtflags old_flags = os.flags();  std::streamsize old_prec = os.precision(19);  os << std::scientific;  if (name.length() > 0) os << name << " = ";  os << "[ " << setw(27) << x[0] << ", " << setw(27) << x[1] << " ]" << endl;  os.precision(old_prec);  os.flags(old_flags);}void dd_real::dump_bits(const string &name, std::ostream &os) const {  string::size_type len = name.length();  if (len > 0) {    os << name << " = ";    len +=3;  }  os << "[ ";  len += 2;  print_double_info(os, x[0]);  os << endl;  for (string::size_type i = 0; i < len; i++) os << ' ';  print_double_info(os, x[1]);  os << " ]" << endl;}dd_real dd_real::debug_rand() {   if (std::rand() % 2 == 0)    return ddrand();  int expn = 0;  dd_real a = 0.0;  double d;  for (int i = 0; i < 2; i++) {    d = std::ldexp(static_cast<double>(std::rand()) / RAND_MAX, -expn);    a += d;    expn = expn + 54 + std::rand() % 200;  }  return a;}
 |