/* This software was developed by Bruce Hendrickson and Robert Leland   *
 * at Sandia National Laboratories under US Department of Energy        *
 * contract DE-AC04-76DP00789 and is copyrighted by Sandia Corporation. */

#include <stdio.h>
#include <math.h>
#include "defs.h"

/* Solve the shifted, symmetric tridiagonal linear system (T - lambda*I)v = b
   where b is a multiple of e1 (the first column of the identity matrix).
   T is constructed with diagonal alpha[1], alpha[2] ... alpha[j] and
   off diagonal beta[1], beta[2] ... beta[j-1]. The algorithm is based on
   p. 156 of Golub and Van Loan, 2nd ed. Modified this to (1) incorporate
   shift by lambda, (2) use work vectors rather than overwriting so
   won't have to copy input as this is called repeatedly in the bisection
   loop for the extended eigenproblem, and (3) to squawk and die if a
   pivot below machine precision is encountered and (4) exploit the fact
   that all elements of the rhs except the first are 0. */
void      tri_solve(alpha, beta, j, lambda, v, b, d, e)
double   *alpha;		/* vector of Lanczos scalars */
double   *beta;			/* vector of Lanczos scalars */
int       j;			/* system size */
double    lambda;		/* diagonal shift */
double   *v;			/* solution vector */
double    b;			/* scalar multiple of e1 specifying the rhs */
double   *d;			/* work vec. for diagonal of Cholesky factor */
double   *e;			/* work vec. for off diagonal of Cholesky factor */
{
    extern int DEBUG_EVECS;	/* debug flag for eigen computation */
    int       i;		/* loop index */
    double    tol;		/* cutoff for numerical singularity */
    double    diff;		/* an element of the residual vector */
    double    res;		/* the norm of the residual vector */
    void      bail();		/* our exit routine */

    /* set cut-off for "zero" pivot */
    tol = 1.0e-15;

    /* Cholesky factorization of shifted symmetric tridiagonal */
    d[1] = alpha[1] - lambda;
    if (fabs(d[1]) < tol) {
	bail("ERROR: Zero pivot in tri_solve().",1);
    }
    if (j==1) {
	v[1] = b/d[1];
	return;
	/* It's a scalar equation, so we're done. */
    }
    for (i = 2; i <= j; i++) {
	e[i - 1] = beta[i - 1] / d[i - 1];
	d[i] = alpha[i] - lambda - d[i - 1] * e[i - 1] * e[i - 1];
	if (fabs(d[i]) < tol) {
	    bail("ERROR: Zero pivot in tri_solve().",1);
	}
    }

    /* Substitution solution using Cholesky factors */
    v[1] = b;
    for (i = 2; i <= j; i++) {
	v[i] = -e[i - 1] * v[i - 1];
    }
    v[j] = v[j] / d[j];
    for (i = j - 1; i >= 1; i--) {
	v[i] = v[i] / d[i] - e[i] * v[i + 1];
    }

    /* Check residual */
    if (DEBUG_EVECS > 1) {
	diff = b - ((alpha[1] - lambda) * v[1] + beta[1] * v[2]);
	res = diff * diff;
	for (i = 2; i <= j - 1; i++) {
	    diff = beta[i - 1] * v[i - 1] + (alpha[i] - lambda) * v[i] + beta[i] * v[i + 1];
	    res += diff * diff;
	}
	diff = beta[j - 1] * v[j - 1] + (alpha[j] - lambda) * v[j];
	res += diff * diff;
	res = sqrt(res);
	if (res > 1.0e-13) {
	    printf("Tridiagonal solve residual %g\n", res);
	}
    }
}


syntax highlighted by Code2HTML, v. 0.9.1