Blob Blame Raw


#include "tutil.h"
#include "tmathutil.h"
#include "tconvert.h"
#include <iterator>

using TConsts::epsilon;

TMathException::TMathException(std::string msg)
	: m_msg(toWideString(msg))
{
}

namespace
{

//-------------------------------------------------------------------------

//! maximum order for a polynomial
const int MAX_ORDER = 12;

//! smallest relative error we want
const double RELERROR = 1.0e-14;

//! max power of 10 we wish to search to
const int MAXPOW = 32;

//!max number of iterations
const int MAXIT = 800;

//! a coefficient smaller than SMALL_ENOUGH is considered to be zero (0.0).
const double SMALL_ENOUGH = 1.0e-12;

//-------------------------------------------------------------------------

inline int getEl(int i, int j, int n)
{
	return (j - 1) + (i - 1) * n;
}
//-------------------------------------------------------------------------

//! structure type for representing a polynomial
typedef struct
	{
	int ord;
	double coef[MAX_ORDER];
} poly;

//-------------------------------------------------------------------------

//  compute number of root in (min,max)
double evalpoly(int ord, double *coef, double x);
int numchanges(int np, poly *sseq, double a);
int buildsturm(int ord, poly *sseq);
int modrf(int ord, double *coef, double a, double b, double *val);

//-------------------------------------------------------------------------

void convert(const std::vector<double> &v, poly &p)
{
	assert((int)v.size() <= MAX_ORDER);
	if ((int)v.size() > MAX_ORDER)
		return;

	p.ord = v.size() - 1;

	std::copy(v.begin(), v.end(), p.coef);
}
//-------------------------------------------------------------------------

void convert(const poly &p, std::vector<double> &v)
{
	v.resize(p.ord);
	std::copy(p.coef, p.coef + p.ord, v.begin());
}

//-------------------------------------------------------------------------

/*
  * modp
  *
  *  calculates the modulus of u(x) / v(x) leaving it in r, it
  *  returns 0 if r(x) is a constant.
  *  note: this function assumes the leading coefficient of v 
  *  is 1 or -1
  */
int modp(poly *u, poly *v, poly *r)
{
	int k, j;
	double *nr, *end, *uc;

	nr = r->coef;
	end = &u->coef[u->ord];

	uc = u->coef;
	while (uc <= end)
		*nr++ = *uc++;

	if (v->coef[v->ord] < 0.0) {
		for (k = u->ord - v->ord - 1; k >= 0; k -= 2)
			r->coef[k] = -r->coef[k];

		for (k = u->ord - v->ord; k >= 0; k--)
			for (j = v->ord + k - 1; j >= k; j--)
				r->coef[j] = -r->coef[j] - r->coef[v->ord + k] * v->coef[j - k];
	} else {
		for (k = u->ord - v->ord; k >= 0; k--)
			for (j = v->ord + k - 1; j >= k; j--)
				r->coef[j] -= r->coef[v->ord + k] * v->coef[j - k];
	}

	k = v->ord - 1;
	while (k >= 0 && fabs(r->coef[k]) < SMALL_ENOUGH) {
		r->coef[k] = 0.0;
		k--;
	}

	r->ord = (k < 0) ? 0 : k;

	return (r->ord);
}

//-------------------------------------------------------------------------

/*
  * buildsturm
  *
  *  build up a sturm sequence for a polynomial in sseq, returning
  * the number of polynomials in the sequence
  */
int buildsturm(int ord, poly *sseq)
{
	int i;
	double f, *fp, *fc;
	poly *sp;

	sseq[0].ord = ord;
	sseq[1].ord = ord - 1;

	//  calculate the derivative and normalise the leadingcoefficient.
	f = fabs(sseq[0].coef[ord] * ord);
	fp = sseq[1].coef;
	fc = sseq[0].coef + 1;
	for (i = 1; i <= ord; i++)
		*fp++ = *fc++ * i / f;

	// construct the rest of the Sturm sequence
	for (sp = sseq + 2; modp(sp - 2, sp - 1, sp); sp++) {
		//  reverse the sign and normalise
		f = -fabs(sp->coef[sp->ord]);
		for (fp = &sp->coef[sp->ord]; fp >= sp->coef; fp--)
			*fp /= f;
	}

	sp->coef[0] = -sp->coef[0]; // reverse the sign

	return (sp - sseq);
}

//-------------------------------------------------------------------------

/*
  return the number of distinct real roots of the polynomial
  described in sseq
  */
int numroots(int np, poly *sseq, int &atneg, int &atpos)
{
	int
		atposinf = 0,
		atneginf = 0;

	poly *s;
	double f, lf;

	// change at positive infinity
	lf = sseq[0].coef[sseq[0].ord];

	for (s = sseq + 1; s <= sseq + np; ++s) {
		f = s->coef[s->ord];
		if (0.0 == lf || lf * f < 0.0)
			++atposinf;
		lf = f;
	}

	// change at negative infinity
	if (sseq[0].ord & 1)
		lf = -sseq[0].coef[sseq[0].ord];
	else
		lf = sseq[0].coef[sseq[0].ord];

	for (s = sseq + 1; s <= sseq + np; ++s) {
		if (s->ord & 1)
			f = -s->coef[s->ord];
		else
			f = s->coef[s->ord];

		if (0.0 == lf || lf * f < 0.0)
			++atneginf;
		lf = f;
	}

	atneg = atneginf;
	atpos = atposinf;

	return atneginf - atposinf;
}

//-------------------------------------------------------------------------

/*
  * numchanges
  *
  * return the number of sign changes in the Sturm sequence in
  * sseq at the value a.
  */
int numchanges(int np, poly *sseq, double a)
{
	int changes;
	double f, lf;
	poly *s;

	changes = 0;

	lf = evalpoly(sseq[0].ord, sseq[0].coef, a);

	for (s = sseq + 1; s <= sseq + np; s++) {
		f = evalpoly(s->ord, s->coef, a);
		if (lf == 0.0 || lf * f < 0)
			changes++;
		lf = f;
	}

	return (changes);
}

//-------------------------------------------------------------------------

/*
  * sbisect
  *
  * uses a bisection based on the sturm sequence for the polynomial
  * described in sseq to isolate intervals in which roots occur,
  * the roots are returned in the roots array in order of magnitude.
  */
void sbisect(int np,
			 poly *sseq,
			 double min,
			 double max,
			 int atmin,
			 int atmax,
			 double *roots)
{
	double mid;
	int n1 = 0, n2 = 0, its, atmid;

	if ((atmin - atmax) == 1) {
		//first try a less expensive technique.

		if (modrf(sseq->ord, sseq->coef, min, max, &roots[0]))
			return;

		/*
        * if we get here we have to evaluate the root the hard
        * way by using the Sturm sequence.
      */
		for (its = 0; its < MAXIT; its++) {
			mid = (min + max) / 2;

			atmid = numchanges(np, sseq, mid);

			if (fabs(mid) > RELERROR) {
				if (fabs((max - min) / mid) < RELERROR) {
					roots[0] = mid;
					return;
				}
			} else if (fabs(max - min) < RELERROR) {
				roots[0] = mid;
				return;
			}

			if ((atmin - atmid) == 0)
				min = mid;
			else
				max = mid;
		}

		if (its == MAXIT) {
			/*
      fprintf(stderr, "sbisect: overflow min %f max %f\
      diff %e nroot %d n1 %d n2 %d\n",
      min, max, max - min, nroot, n1, n2);
        */
			roots[0] = mid;
		}

		return;
	}

	/*
    * more than one root in the interval, we have to bisect...
    */
	for (its = 0; its < MAXIT; its++) {
		mid = (min + max) / 2;

		atmid = numchanges(np, sseq, mid);

		n1 = atmin - atmid;
		n2 = atmid - atmax;

		if (n1 != 0 && n2 != 0) {
			sbisect(np, sseq, min, mid, atmin, atmid, roots);
			sbisect(np, sseq, mid, max, atmid, atmax, &roots[n1]);
			break;
		}

		if (n1 == 0)
			min = mid;
		else
			max = mid;
	}

	if (its == MAXIT) {
		/*
    fprintf(stderr, "sbisect: roots too close together\n");
    fprintf(stderr, "sbisect: overflow min %f max %f diff %e\
    nroot %d n1 %d n2 %d\n",
    min, max, max - min, nroot, n1, n2);
      */
		for (n1 = atmax; n1 < atmin; n1++)
			roots[n1 - atmax] = mid;
	}
}

//-------------------------------------------------------------------------

/*
  * evalpoly
  *
  * evaluate polynomial defined in coef returning its value.
    */
double evalpoly(int ord, double *coef, double x)
{
	double *fp, f;

	fp = &coef[ord];
	f = *fp;

	for (fp--; fp >= coef; fp--)
		f = x * f + *fp;

	return (f);
}

//-------------------------------------------------------------------------

/*
    * modrf
    *
    * uses the modified regula-falsi method to evaluate the root
    * in interval [a,b] of the polynomial described in coef. The
    * root is returned is returned in *val. The routine returns zero
    * if it can't converge.
    */
int modrf(int ord, double *coef, double a, double b, double *val)
{
	int its;
	double fa, fb, x, fx, lfx;
	double *fp, *scoef, *ecoef;

	scoef = coef;
	ecoef = &coef[ord];

	fb = fa = *ecoef;
	for (fp = ecoef - 1; fp >= scoef; fp--) {
		fa = a * fa + *fp;
		fb = b * fb + *fp;
	}

	//  if there is no sign difference the method won't work
	if (fa * fb > 0.0)
		return (0);

	if (fabs(fa) < RELERROR) {
		*val = a;
		return (1);
	}

	if (fabs(fb) < RELERROR) {
		*val = b;
		return (1);
	}

	lfx = fa;

	for (its = 0; its < MAXIT; its++) {
		x = (fb * a - fa * b) / (fb - fa);

		fx = *ecoef;
		for (fp = ecoef - 1; fp >= scoef; fp--)
			fx = x * fx + *fp;

		if (fabs(x) > RELERROR) {
			if (fabs(fx / x) < RELERROR) {
				*val = x;
				return (1);
			}
		} else if (fabs(fx) < RELERROR) {
			*val = x;
			return (1);
		}

		if ((fa * fx) < 0) {
			b = x;
			fb = fx;
			if ((lfx * fx) > 0)
				fa /= 2;
		} else {
			a = x;
			fa = fx;
			if ((lfx * fx) > 0)
				fb /= 2;
		}

		lfx = fx;
	}

	// fprintf(stderr, "modrf overflow %f %f %f\n", a, b, fx);

	return (0);
}

//-------------------------------------------------------------------------

/*!
    a x^2 + b x + c
    
      Remark: 
      poly[0] = c 
      poly[1] = b
      poly[2] = a 
    */
int rootForQuadraticEquation(const std::vector<double> &v, std::vector<double> &sol)
{
	double q, delta;

	/*
      if( isAlmostZero(v[2]))
      {
      if ( isAlmostZero(v[1]) )
      return -1;
      
        sol.push_back(-v[0]/v[1]);
        return 1;
        }
      */

	if (isAlmostZero(v[1])) {
		q = -v[0] / v[2];
		if (q < 0)
			return 0;
		else if (isAlmostZero(q)) {
			sol.push_back(0);
			return 1;
		}

		q = sqrt(q);
		sol.push_back(-q);
		sol.push_back(q);

		return 2;
	}

	delta = sq(v[1]) - 4.0 * v[0] * v[2];

	if (delta < 0.0)
		return 0;

	assert(v[2] != 0);

	if (isAlmostZero(delta)) {
		sol.push_back(-v[1] / (v[2] * 2.0));
		return 1;
	}

	q = -0.5 * (v[1] + tsign(v[1]) * sqrt(delta));

	assert(q != 0);

	sol.push_back(v[0] / q);
	sol.push_back(q / v[2]);

	return 2;
}

//-----------------------------------------------------------------------------

/*
    a x^3+b x^2 + c x + d
    
      Remark: 
      poly[0] = d 
      poly[1] = c
      poly[2] = b 
      poly[3] = a 
    */
int rootForCubicEquation(const std::vector<double> &p, std::vector<double> &sol)
{
	/*
    if( isAlmostZero(p[3]) )
    return rootForQuadraticEquation(p,sol);
      */

	if (isAlmostZero(p[0])) {
		int numberOfSol;
		std::vector<double> redPol(3);
		redPol[0] = p[1];
		redPol[1] = p[2];
		redPol[2] = p[3];

		numberOfSol = rootForQuadraticEquation(redPol, sol);

		for (int i = 0; i < numberOfSol; ++i)
			if (0.0 == sol[i])
				return numberOfSol;

		// altrimenti devo contare la soluzione nulla
		++numberOfSol;
		sol.push_back(0);
		return numberOfSol;
	}

	double
		inv_v3 = 1.0 / p[3],
		a = p[2] * inv_v3,
		b = p[1] * inv_v3,
		c = p[0] * inv_v3;

	static const double inv_3 = 1.0 / 3.0;
	static const double inv_9 = 1.0 / 9.0;
	static const double inv_54 = 1.0 / 54.0;

	double
		Q = (sq(a) - 3.0 * b) * inv_9,
		R = (2.0 * sq(a) * a - 9.0 * a * b + 27.0 * c) * inv_54;

	double
		R_2 = sq(R),
		Q_3 = sq(Q) * Q;

	if (R_2 < Q_3) {
		double Q_sqrt = sqrt(Q);
		double theta = acos(R / (Q * Q_sqrt));
		sol.push_back(-2 * Q_sqrt * cos(theta * inv_3) - a * inv_3);
		sol.push_back(-2 * Q_sqrt * cos((theta - TConsts::pi * 2.0) * inv_3) - a * inv_3);
		sol.push_back(-2 * Q_sqrt * cos((theta + TConsts::pi * 2.0) * inv_3) - a * inv_3);
		std::sort(sol.begin(), sol.end());
		return 3;
	}

	double A = -tsign(R) * pow((fabs(R) + sqrt(R_2 - Q_3)), inv_3);
	double B = A != 0 ? Q / A : 0;
	sol.push_back((A + B) - a * inv_3);
	return 1;
}

//-----------------------------------------------------------------------------

int rootForGreaterThanThreeEquation(const std::vector<double> &p, std::vector<double> &sol)
{
	poly sseq[MAX_ORDER];

	convert(p, sseq[0]);

	int np = buildsturm(p.size() - 1, sseq);

	int atmin, atmax;

	int nroot = numroots(np, sseq, atmin, atmax);

	if (nroot == 0)
		return 0;

	double minVal = -1.0;

	UINT i = 0;

	int nchanges = numchanges(np, sseq, minVal);

	for (i = 0; nchanges != atmin && i != (UINT)MAXPOW; ++i) {
		minVal *= 10.0;
		nchanges = numchanges(np, sseq, minVal);
	}

	if (nchanges != atmin) {
		atmin = nchanges;
	}

	double maxVal = 1.0;
	nchanges = numchanges(np, sseq, maxVal);

	for (i = 0; nchanges != atmax && i != (UINT)MAXPOW; ++i) {
		maxVal *= 10.0;
		nchanges = numchanges(np, sseq, maxVal);
	}

	if (nchanges != atmax) {
		atmax = nchanges;
	}

	nroot = atmin - atmax;

	assert(nroot > 0);

	poly outPoly;

	outPoly.ord = nroot;

	sbisect(np, sseq, minVal, maxVal, atmin, atmax, outPoly.coef);

	convert(outPoly, sol);

	return nroot < 0 ? -1 : nroot;
}

//-----------------------------------------------------------------------------

} // end of unnamed namespace

//-----------------------------------------------------------------------------

void tLUDecomposition(double *a, int n, int *indx, double &d)
{
	int i, imax, j, k;
	double big, dum, sum, temp;
	std::vector<double> vv(n);

	d = 1.0;
	for (i = 1; i <= n; i++) {
		big = 0.0;
		for (j = 1; j <= n; j++)
			if ((temp = fabs(a[getEl(i, j, n)])) > big)
				big = temp;

		if (big == 0.0)
			throw TMathException("Singular matrix in routine tLUDecomposition\n");

		vv[i - 1] = 1.0 / big;
	}

	for (j = 1; j <= n; j++) {
		for (i = 1; i < j; i++) {
			sum = a[getEl(i, j, n)];
			for (k = 1; k < i; k++)
				sum -= a[getEl(i, k, n)] * a[getEl(k, j, n)];
			a[getEl(i, j, n)] = sum;
		}
		big = 0.0;
		for (i = j; i <= n; i++) {
			sum = a[getEl(i, j, n)];
			for (k = 1; k < j; k++)
				sum -= a[getEl(i, k, n)] * a[getEl(k, j, n)];
			a[getEl(i, j, n)] = sum;
			if ((dum = vv[i - 1] * fabs(sum)) >= big) {
				big = dum;
				imax = i;
			}
		}

		if (j != imax) {
			for (k = 1; k <= n; k++) {
				dum = a[getEl(imax, k, n)];
				a[getEl(imax, k, n)] = a[getEl(j, k, n)];
				a[getEl(j, k, n)] = dum;
			}
			d = -(d);
			vv[imax - 1] = vv[j - 1];
		}

		indx[j - 1] = imax;

		if (a[getEl(j, j, n)] == 0.0)
			a[getEl(j, j, n)] = TConsts::epsilon;

		if (j != n) {
			dum = 1.0 / (a[getEl(j, j, n)]);
			for (i = j + 1; i <= n; i++)
				a[getEl(i, j, n)] *= dum;
		}
	}
}

//-----------------------------------------------------------------------------

void tbackSubstitution(double *a, int n, int *indx, double *b)
{
	int i, ii = 0, ip, j;
	double sum;

	for (i = 1; i <= n; i++) {
		ip = indx[i - 1];
		sum = b[ip - 1];
		b[ip - 1] = b[i - 1];
		if (ii)
			for (j = ii; j <= i - 1; j++)
				sum -= a[getEl(i, j, n)] * b[j - 1];
		else if (sum)
			ii = i;
		b[i - 1] = sum;
	}
	for (i = n; i >= 1; i--) {
		sum = b[i - 1];
		for (j = i + 1; j <= n; j++)
			sum -= a[getEl(i, j, n)] * b[j - 1];
		b[i - 1] = sum / a[getEl(i, i, n)];
	}
}

//-----------------------------------------------------------------------------

double tdet(double *LUa, int n, double d)
{
	for (int i = 1; i <= n; ++i)
		d *= LUa[getEl(i, i, n)];

	return d;
}

//-----------------------------------------------------------------------------

double tdet(double *a, int n)
{
	double d;
	std::vector<int> indx(n);

	tLUDecomposition(a, n, &indx[0], d);
	for (int i = 1; i <= n; ++i)
		d *= a[getEl(i, i, n)];

	return d;
}

//-----------------------------------------------------------------------------

void tsolveSistem(double *a, int n, double *res)
{
	double d;
	std::vector<int> indx(n);

	tLUDecomposition(a, n, &indx[0], d);
	assert(tdet(a, n, d) != 0);
	/*
  if( isAlmostZero(tdet(a, n, d)) )
    throw TMathException("Singular matrix in routine tLUDecomposition\n");    
  */
	tbackSubstitution(a, n, &indx[0], res);
}

//-----------------------------------------------------------------------------

int rootFinding(const std::vector<double> &in_poly, std::vector<double> &sol)
{
	// per ora risolvo solo i polinomi di grado al piu' pari a 3
	assert((int)in_poly.size() <= MAX_ORDER);
	if (in_poly.empty() || (int)in_poly.size() > MAX_ORDER)
		return -1;

	std::vector<double> p;
	std::copy(in_poly.begin(), in_poly.end(), std::back_inserter(p));

	// eat zero in poly
	while (!p.empty() && isAlmostZero(p.back()))
		p.pop_back();

	sol.clear();

	while (!p.empty() && p.front() == 0) {
		sol.push_back(0.0);
		p.erase(p.begin()); //se i coefficienti bassi sono zero, ci sono soluzioni pari a 0.0. le metto,
	}						//e abbasso il grado del polinomio(piu' veloce)

	switch (p.size()) {
	case 0:
		if (sol.empty())
			return INFINITE_SOLUTIONS;
		CASE 1 : //no solutions
				 CASE 2 : sol.push_back(-p[0] / p[1]);
		CASE 3 : rootForQuadraticEquation(p, sol);
		CASE 4 : rootForCubicEquation(p, sol);
	DEFAULT:
		rootForGreaterThanThreeEquation(p, sol);
	}
	return sol.size();
}

//-----------------------------------------------------------------------------

/*
*/
int numberOfRootsInInterval(int order, const double *polyH, double min, double max)
{
	poly sseq[MAX_ORDER];

	int
		i,
		nchanges_0,
		nchanges_1,
		np;

	if (order > MAX_ORDER)
		return -1;

	while (polyH[order] == 0.0 && order > 0)
		--order;

	// init a sturm's sequence with our polynomious
	for (i = order; i >= 0; --i)
		sseq[0].coef[i] = polyH[i];

	//  build the Sturm sequence
	np = buildsturm(order, sseq);

	//compute number of variation on 0.0
	nchanges_0 = numchanges(np, sseq, min);

	nchanges_1 = numchanges(np, sseq, max);

	return (nchanges_0 - nchanges_1);
}

//-----------------------------------------------------------------------------

double quadraticRoot(double a, double b, double c)
{
	double bb, q;
	// caso lineare
	if (fabs(a) < epsilon) {
		if (fabs(b) >= epsilon)
			return -c / b;
		return 1;
	}
	bb = b * b;
	q = bb - 4.0 * a * c;
	if (q < 0.0)
		return 1;
	q = sqrt(q);
	if (b < 0.0)
		q = -q;
	q = -0.5 * (b + q);
	double root1 = -1;
	double root2 = -1;
	if (fabs(q) >= epsilon)
		root1 = c / q;
	if (fabs(a) >= epsilon)
		root2 = q / a;
	if (0.0 - epsilon <= root1 && root1 <= 1.0 + epsilon)
		return root1;
	if (0.0 - epsilon <= root2 && root2 <= 1.0 + epsilon)
		return root2;
	return 1;
}

//-----------------------------------------------------------------------------

double cubicRoot(double a, double b, double c, double d)
{
	double A, Q, R, QQQ, RR;
	double theta;
	/* Test for a quadratic or linear degeneracy			*/
	if (fabs(a) < epsilon)
		return quadraticRoot(b, c, d);
	/* Normalize							*/
	b /= a;
	c /= a;
	d /= a;
	a = 1.0;
	/* Compute discriminants						*/
	Q = (b * b - 3.0 * c) / 9.0;
	QQQ = Q * Q * Q;
	R = (2.0 * b * b * b - 9.0 * b * c + 27.0 * d) / 54.0;
	RR = R * R;
	/* Three real roots							*/
	if (RR < QQQ) {
		theta = acos(R / sqrt(QQQ));
		double root[3];
		root[0] = root[1] = root[2] = -2.0 * sqrt(Q);
		root[0] *= cos(theta / 3.0);
		root[1] *= cos((theta + 2 * TConsts::pi) / 3.0);
		root[2] *= cos((theta - 2 * TConsts::pi) / 3.0);
		root[0] -= b / 3.0;
		root[1] -= b / 3.0;
		root[2] -= b / 3.0;
		if (0.0 - epsilon < root[0] && root[0] < 1.0 + epsilon)
			return root[0];
		if (0.0 - epsilon < root[1] && root[1] < 1.0 + epsilon)
			return root[1];
		if (0.0 - epsilon < root[2] && root[2] < 1.0 + epsilon)
			return root[2];
		return 1;
	}
	/* One real root							*/
	else {
		double root = 0;
		A = -pow(fabs(R) + sqrt(RR - QQQ), 1.0 / 3.0);
		if (A != 0.0) {
			if (R < 0.0)
				A = -A;
			root = A + Q / A;
		}
		root -= b / 3.0;
		if (0.0 - epsilon < root && root < 1.0 + epsilon)
			return root;
		return 1;
	}
}

//-----------------------------------------------------------------------------
//  End Of File
//-----------------------------------------------------------------------------