* E04GDF Example Program Text * Mark 15 Revised. NAG Copyright 1991. * .. Parameters .. INTEGER N, M, NT, LV, LJ, LIW, LW PARAMETER (N=3,M=15,NT=3,LV=N,LJ=M,LIW=1, + LW=7*N+M*N+2*M+N*N) INTEGER NIN, NOUT PARAMETER (NIN=5,NOUT=6) * .. Arrays in Common .. DOUBLE PRECISION T(M,NT), Y(M) * .. Local Scalars .. DOUBLE PRECISION ETA, FSUMSQ, STEPMX, XTOL INTEGER I, IFAIL, IPRINT, J, MAXCAL, NF, NITER * .. Local Arrays .. DOUBLE PRECISION FJAC(LJ,N), FVEC(M), G(N), S(N), V(LV,N), W(LW), + X(N) INTEGER IW(LIW) * .. External Functions .. DOUBLE PRECISION X02AJF EXTERNAL X02AJF * .. External Subroutines .. EXTERNAL E04GDF, E04YAF, LSQFUN, LSQGRD, LSQMON * .. Intrinsic Functions .. INTRINSIC SQRT * .. Common blocks .. COMMON Y, T * .. Executable Statements .. WRITE (NOUT,*) 'E04GDF Example Program Results' * Skip heading in data file READ (NIN,*) * Observations of TJ (J = 1, 2, 3) are held in T(I, J) * (I = 1, 2, . . . , 15) DO 20 I = 1, M READ (NIN,*) Y(I), (T(I,J),J=1,NT) 20 CONTINUE * Check LSQFUN by calling E04YAF at an arbitrary point. Since * E04YAF only checks the derivatives calculated when IFLAG = 2, * a separate program should be run before using E04YAF or * E04GDF to check that LSQFUN gives the same values for the * elements of FJACC when IFLAG is set to 1 as when IFLAG is * set to 2. X(1) = 0.19D0 X(2) = -1.34D0 X(3) = 0.88D0 IFAIL = 0 * CALL E04YAF(M,N,LSQFUN,X,FVEC,FJAC,LJ,IW,LIW,W,LW,IFAIL) * * Continue setting parameters for E04GDF * * Set IPRINT to 1 to obtain output from LSQMON at each iteration * IPRINT = -1 MAXCAL = 50*N ETA = 0.9D0 XTOL = 10.0D0*SQRT(X02AJF()) * We estimate that the minimum will be within 10 units of the * starting point STEPMX = 10.0D0 * Set up the starting point X(1) = 0.5D0 X(2) = 1.0D0 X(3) = 1.5D0 IFAIL = 1 * CALL E04GDF(M,N,LSQFUN,LSQMON,IPRINT,MAXCAL,ETA,XTOL,STEPMX,X, + FSUMSQ,FVEC,FJAC,LJ,S,V,LV,NITER,NF,IW,LIW,W,LW,IFAIL) * IF (IFAIL.NE.0) THEN WRITE (NOUT,*) WRITE (NOUT,99999) 'Error exit type', IFAIL, + ' - see routine document' END IF IF (IFAIL.NE.1) THEN WRITE (NOUT,*) WRITE (NOUT,99998) 'On exit, the sum of squares is', FSUMSQ WRITE (NOUT,99998) 'at the point', (X(J),J=1,N) CALL LSQGRD(M,N,FVEC,FJAC,LJ,G) WRITE (NOUT,99997) 'The corresponding gradient is', + (G(J),J=1,N) WRITE (NOUT,*) ' (machine dependent)' WRITE (NOUT,*) 'and the residuals are' DO 40 I = 1, M WRITE (NOUT,99996) FVEC(I) 40 CONTINUE END IF STOP * 99999 FORMAT (1X,A,I3,A) 99998 FORMAT (1X,A,3F12.4) 99997 FORMAT (1X,A,1P,3E12.3) 99996 FORMAT (1X,1P,E9.1) END * SUBROUTINE LSQFUN(IFLAG,M,N,XC,FVECC,FJACC,LJC,IW,LIW,W,LW) * Routine to evaluate the residuals and their 1st derivatives. * A COMMON variable could be updated here to count the * number of calls of LSQFUN with IFLAG set to 1 (since NF * in LSQMON only counts calls with IFLAG set to 2) * .. Parameters .. INTEGER MDEC, NT PARAMETER (MDEC=15,NT=3) * .. Scalar Arguments .. INTEGER IFLAG, LIW, LJC, LW, M, N * .. Array Arguments .. DOUBLE PRECISION FJACC(LJC,N), FVECC(M), W(LW), XC(N) INTEGER IW(LIW) * .. Arrays in Common .. DOUBLE PRECISION T(MDEC,NT), Y(MDEC) * .. Local Scalars .. DOUBLE PRECISION DENOM, DUMMY INTEGER I * .. Common blocks .. COMMON Y, T * .. Executable Statements .. DO 20 I = 1, M DENOM = XC(2)*T(I,2) + XC(3)*T(I,3) IF (IFLAG.EQ.2) FVECC(I) = XC(1) + T(I,1)/DENOM - Y(I) FJACC(I,1) = 1.0D0 DUMMY = -1.0D0/(DENOM*DENOM) FJACC(I,2) = T(I,1)*T(I,2)*DUMMY FJACC(I,3) = T(I,1)*T(I,3)*DUMMY 20 CONTINUE RETURN END * SUBROUTINE LSQMON(M,N,XC,FVECC,FJACC,LJC,S,IGRADE,NITER,NF,IW,LIW, + W,LW) * Monitoring routine * .. Parameters .. INTEGER NDEC PARAMETER (NDEC=3) INTEGER NOUT PARAMETER (NOUT=6) * .. Scalar Arguments .. INTEGER IGRADE, LIW, LJC, LW, M, N, NF, NITER * .. Array Arguments .. DOUBLE PRECISION FJACC(LJC,N), FVECC(M), S(N), W(LW), XC(N) INTEGER IW(LIW) * .. Local Scalars .. DOUBLE PRECISION FSUMSQ, GTG INTEGER J * .. Local Arrays .. DOUBLE PRECISION G(NDEC) * .. External Functions .. DOUBLE PRECISION F06EAF EXTERNAL F06EAF * .. External Subroutines .. EXTERNAL LSQGRD * .. Executable Statements .. FSUMSQ = F06EAF(M,FVECC,1,FVECC,1) CALL LSQGRD(M,N,FVECC,FJACC,LJC,G) GTG = F06EAF(N,G,1,G,1) * A COMMON variable giving the number of calls of * LSQFUN with IFLAG set to 1 could be printed here WRITE (NOUT,*) WRITE (NOUT,*) + ' Itns F evals SUMSQ GTG grade' WRITE (NOUT,99999) NITER, NF, FSUMSQ, GTG, IGRADE WRITE (NOUT,*) WRITE (NOUT,*) + ' X G Singular values' DO 20 J = 1, N WRITE (NOUT,99998) XC(J), G(J), S(J) 20 CONTINUE RETURN * 99999 FORMAT (1X,I4,6X,I5,6X,1P,E13.5,6X,1P,E9.1,6X,I3) 99998 FORMAT (1X,1P,E13.5,10X,1P,E9.1,10X,1P,E9.1) END * SUBROUTINE LSQGRD(M,N,FVECC,FJACC,LJC,G) * Routine to evaluate gradient of the sum of squares * .. Scalar Arguments .. INTEGER LJC, M, N * .. Array Arguments .. DOUBLE PRECISION FJACC(LJC,N), FVECC(M), G(N) * .. Local Scalars .. DOUBLE PRECISION SUM INTEGER I, J * .. Executable Statements .. DO 40 J = 1, N SUM = 0.0D0 DO 20 I = 1, M SUM = SUM + FJACC(I,J)*FVECC(I) 20 CONTINUE G(J) = SUM + SUM 40 CONTINUE RETURN END