#include "MGCLStdAfx.h"
/********************************************************************/
/* Copyright (c) 2017 System fugen G.K. and Yuzi Mizuno          */
/* All rights reserved.                                             */
/********************************************************************/

#include "cskernel/b1bslv.h"
#include "cskernel/b1bfac.h"
#include "cskernel/b2vb.h"

//The original codes of this program comes from the FORTRAN code SPLINT of
//"A Practical Guide to Splines" by Carl de Boor.


//  BLGINT PRODUCES THE B-SPLINE COEFF.S RCOEF OF THE SPLINE OF ORDER
//   K  WITH KNOTS  T(I), I=1,..., N + K , WHICH TAKES ON THE VALUE 
//   P(I,NCD) AT     TAU(I), I=1,..., N . 
//
// ******  I N P U T  ****** 
//  TAU(N)..ARRAY OF LENGTH  N , CONTAINING DATA POINT ABSCISSAE. 
//    A S S U M P T I O N . . .  TAU  IS STRICTLY INCREASING 
//  P(IP,NCD)...CORRESPONDING ARRAY OF LENGTH  N , CONTAINING DATA POINT ORDINATES 
//  T(N+K)...KNOT SEQUENCE, OF LENGTH  N+K 
//  N.....NUMBER OF DATA POINTS AND DIMENSION OF SPLINE SPACE  S(K,T) 
//  K.....ORDER OF SPLINE 
//  NCD...SPACE DIMENSION OF POINTS P 
//  IP....ROW DIMENSION OF THE VARIABLE P 
//  IRC...ROW DIMENSION OF THE VARIABLE RCOEF 
//
// ******  O U T P U T  ****** 
//  Q.....ARRAY OF SIZE  (2*K-1)*N , CONTAINING THE TRIANGULAR FACTORIZ- 
//       ATION OF THE COEFFICIENT MATRIX OF THE LINEAR SYSTEM FOR THE
//	   B-COEFFICIENTS OF THE SPLINE INTERPOLANT. 
//           THE B-COEFFS FOR THE INTERPOLANT OF AN ADDITIONAL DATA SET 
//        (TAU(I),HTAU(I)), I=1,...,N  WITH THE SAME DATA ABSCISSAE CAN 
//        BE OBTAINED WITHOUT GOING THROUGH ALL THE CALCULATIONS IN THIS 
//          ROUTINE, SIMPLY BY LOADING  HTAU  INTO  RCOEF AND THEN
//		  EXECUTING THE CALL B1BSLV ( Q, 2*K-1, N, K-1, K-1, RCOEF ) 
//  RCOEF(IRC,NCD)..THE B-COEFFICIENTS OF THE INTERPOLANT, OF LENGTH  N 
//  Function's return value.....AN INTEGER INDICATING SUCCESS (= 1)  OR FAILURE (= 2) 
//       THE LINEAR SYSTEM TO BE SOLVED IS (THEORETICALLY) INVERTIBLE IF 
//         AND ONLY IF 
//              T(I) .LT. TAU(I) .LT. TAU(I+K),    ALL I. 
//       VIOLATION OF THIS CONDITION IS CERTAIN TO LEAD TO  IFLAG = 2.  
//
// ******  M E T H O D  ****** 
//  THE I-TH EQUATION OF THE LINEAR SYSTEM  A*RCOEF = B  FOR THE B-CO- 
//    EFFS OF THE INTERPOLANT ENFORCES INTERPOLATION AT  TAU(I), I=1,...,N. 
//    HENCE,  B(I) = P(I), ALL I, AND     A  IS A BAND MATRIX WITH  2K-1 
//   BANDS (IF IT IS INVERTIBLE). 
//    THE MATRIX  A  IS GENERATED ROW BY ROW AND STORED, DIAGONAL BY DI- 
//AGONAL, IN THE  R O W S  OF THE ARRAY  Q, WITH THE MAIN DIAGONAL GO- 
//   ING INTO ROW  K .  SEE COMMENTS IN THE PROGRAM BELOW. 
//     THE BANDED SYSTEM IS THEN SOLVED BY A CALL TO  B1BFAC (WHICH CON- 
//      STRUCTS THE TRIANGULAR FACTORIZATION FOR  A  AND STORES IT AGAIN IN 
//   Q ), FOLLOWED BY A CALL TO  B1BSLV (WHICH THEN OBTAINS THE SOLUTION 
//     RCOEF  BY SUBSTITUTION). 
//    B1BFAC  DOES NO PIVOTING, SINCE THE TOTAL POSITIVITY OF THE MATRIX 
//     A  MAKES THIS UNNECESSARY. 
//
int blgint_(const double *tau, const double *p, const double *t, int k, int n, int ncd, int ip, int irc, 
	double *q, double *rcoef){

    // Local variables 
    int left, lenq, iflag;
    double taui;
    int kpkm2, i, j;
    int ipk, jj, km1, np1, npk, kpkm1;

    double riatt_buf[20];
	double* riatt=riatt_buf;
	if(k>20)
		riatt=(double*)malloc(sizeof(double)*k);

    // Parameter adjustments 
    --tau;
    p -= ip+1;
    rcoef -= irc+1;
    --t;
    --q;

    // Function Body 
    np1 = n+1;
    npk = n+k;
    km1 = k-1;
	kpkm1=k+km1;
    kpkm2=km1<<1;
    left=k;
//                ZERO OUT ALL ENTRIES OF Q 
    lenq = n*(kpkm1);
    for (i=1; i<=lenq; ++i)
		q[i] = 0.;

//  *** LOOP OVER I TO CONSTRUCT THE N INTERPOLATON EQUATIONS 
    for(i=1; i<=n; ++i){
		taui = tau[i];
		ipk = i+k;
		if(ipk>np1)
			ipk=np1;
//      ** Find LEFT in the closed interval (I,I+K-1) such that 
//            T(LEFT) .LE. TAU(I) .LT. T(LEFT+1) 
//         Matrix is singular if this is not possible. 
		if(left<i)
		    left=i;
		if(taui<t[left])
		    goto L8700;

L15:
		if (taui<t[left+1])
		    goto L16;
		++left;
		if (left<ipk)
		    goto L15;
		--left;
		if(taui>t[left+1])
		    goto L8700;
L16:
//      *** The I-th equation enforces interpolation at TAUI, hence 
//        A(I,J)=B(J,K,T) (TAUI), all J.  Only the K entries with J= 
//        LEFT-K+1,...,LEFT actually might be nonzero. These K numbers
//        are retutned, in RCOEF  (used for temp.storage here), by the
//        following. 
		b2vb_(k, npk, &t[1], taui, left, riatt);
//        We therefore want RCOEF(J)=B(LEFT-K+J)(TAUI) to go into 
//        A(I,LEFT-K+J), i.e., into Q(I-(LEFT+J)+2*K,(LEFT+J)-K) since
//        A(I+J,J) is to go into Q(I+K,J), all I,J,if we consider Q
//        as a two-dimension array, with 2*K-1 rows (see comments in 
//        B1BFAC). In the present program, we treat Q as an equivalent
//        one-dimendional array. We therefore want RCOEF(J) to go into
//        ENTRY     I-(LEFT+J)+2*K+((LEFT+J)-K-1)*(2*K-1) 
//                   = I-LEFT+1+(LEFT-K)*(2*K-1)+ (2*K-2)*J 
//        of Q. 
		jj = i-left+1+(left-k)*kpkm1;
		for(j=1; j<=k; ++j){
			jj+=kpkm2;
		    q[jj]=riatt[j-1];
		}
    }

//        *** Obtain factoriazation of A, stored again in Q. 
    if(b1bfac_(&q[1], kpkm1, n, km1, km1))
		goto L8700;
//    ***Solve A*RCOEF = P      by back substitution. 
    for (j = 1; j <= ncd; ++j) {
		for (i = 1; i <= n; ++i)
		    rcoef[i+j*irc] = p[i+j*ip];
		b1bslv_(&q[1], kpkm1, n, km1, km1, &rcoef[j*irc+1]);
    }
	iflag=1;
    goto L8888;
// CASE OF ERROR RETURN 
L8700:
    iflag = 2;
L8888:
	if(k>20) free(riatt);
    return iflag;
}
