OpenLibm/slatec/sdcor.f

193 lines
6.3 KiB
FortranFixed
Raw Normal View History

*DECK SDCOR
SUBROUTINE SDCOR (DFDY, EL, FA, H, IERROR, IMPL, IPVT, MATDIM,
8 MITER, ML, MU, N, NDE, NQ, T, USERS, Y, YH, YWT, EVALFA, SAVE1,
8 SAVE2, A, D, JSTATE)
C***BEGIN PROLOGUE SDCOR
C***SUBSIDIARY
C***PURPOSE Subroutine SDCOR computes corrections to the Y array.
C***LIBRARY SLATEC (SDRIVE)
C***TYPE SINGLE PRECISION (SDCOR-S, DDCOR-D, CDCOR-C)
C***AUTHOR Kahaner, D. K., (NIST)
C National Institute of Standards and Technology
C Gaithersburg, MD 20899
C Sutherland, C. D., (LANL)
C Mail Stop D466
C Los Alamos National Laboratory
C Los Alamos, NM 87545
C***DESCRIPTION
C
C In the case of functional iteration, update Y directly from the
C result of the last call to F.
C In the case of the chord method, compute the corrector error and
C solve the linear system with that as right hand side and DFDY as
C coefficient matrix, using the LU decomposition if MITER is 1, 2, 4,
C or 5.
C***ROUTINES CALLED SGBSL, SGESL, SNRM2
C***REVISION HISTORY (YYMMDD)
C 790601 DATE WRITTEN
C 900329 Initial submission to SLATEC.
C***END PROLOGUE SDCOR
INTEGER I, IERROR, IFLAG, IMPL, J, JSTATE, MATDIM, MITER, ML, MU,
8 MW, N, NDE, NQ
REAL A(MATDIM,*), D, DFDY(MATDIM,*), EL(13,12), H,
8 SAVE1(*), SAVE2(*), SNRM2, T, Y(*), YH(N,*), YWT(*)
INTEGER IPVT(*)
LOGICAL EVALFA
C***FIRST EXECUTABLE STATEMENT SDCOR
IF (MITER .EQ. 0) THEN
IF (IERROR .EQ. 1 .OR. IERROR .EQ. 5) THEN
DO 100 I = 1,N
100 SAVE1(I) = (H*SAVE2(I) - YH(I,2) - SAVE1(I))/YWT(I)
ELSE
DO 102 I = 1,N
SAVE1(I) = (H*SAVE2(I) - YH(I,2) - SAVE1(I))/
8 MAX(ABS(Y(I)), YWT(I))
102 CONTINUE
END IF
D = SNRM2(N, SAVE1, 1)/SQRT(REAL(N))
DO 105 I = 1,N
105 SAVE1(I) = H*SAVE2(I) - YH(I,2)
ELSE IF (MITER .EQ. 1 .OR. MITER .EQ. 2) THEN
IF (IMPL .EQ. 0) THEN
DO 130 I = 1,N
130 SAVE2(I) = H*SAVE2(I) - YH(I,2) - SAVE1(I)
ELSE IF (IMPL .EQ. 1) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A, MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 150 I = 1,N
150 SAVE2(I) = H*SAVE2(I)
DO 160 J = 1,N
DO 160 I = 1,N
160 SAVE2(I) = SAVE2(I) - A(I,J)*(YH(J,2) + SAVE1(J))
ELSE IF (IMPL .EQ. 2) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A, MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 180 I = 1,N
180 SAVE2(I) = H*SAVE2(I) - A(I,1)*(YH(I,2) + SAVE1(I))
ELSE IF (IMPL .EQ. 3) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A, MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 140 I = 1,N
140 SAVE2(I) = H*SAVE2(I)
DO 170 J = 1,NDE
DO 170 I = 1,NDE
170 SAVE2(I) = SAVE2(I) - A(I,J)*(YH(J,2) + SAVE1(J))
END IF
CALL SGESL (DFDY, MATDIM, N, IPVT, SAVE2, 0)
IF (IERROR .EQ. 1 .OR. IERROR .EQ. 5) THEN
DO 200 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
200 SAVE2(I) = SAVE2(I)/YWT(I)
ELSE
DO 205 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
205 SAVE2(I) = SAVE2(I)/MAX(ABS(Y(I)), YWT(I))
END IF
D = SNRM2(N, SAVE2, 1)/SQRT(REAL(N))
ELSE IF (MITER .EQ. 4 .OR. MITER .EQ. 5) THEN
IF (IMPL .EQ. 0) THEN
DO 230 I = 1,N
230 SAVE2(I) = H*SAVE2(I) - YH(I,2) - SAVE1(I)
ELSE IF (IMPL .EQ. 1) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A(ML+1,1), MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 250 I = 1,N
250 SAVE2(I) = H*SAVE2(I)
MW = ML + 1 + MU
DO 260 J = 1,N
DO 260 I = MAX(ML+1, MW+1-J), MIN(MW+N-J, MW+ML)
SAVE2(I+J-MW) = SAVE2(I+J-MW)
8 - A(I,J)*(YH(J,2) + SAVE1(J))
260 CONTINUE
ELSE IF (IMPL .EQ. 2) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A, MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 280 I = 1,N
280 SAVE2(I) = H*SAVE2(I) - A(I,1)*(YH(I,2) + SAVE1(I))
ELSE IF (IMPL .EQ. 3) THEN
IF (EVALFA) THEN
CALL FA (N, T, Y, A(ML+1,1), MATDIM, ML, MU, NDE)
IF (N .EQ. 0) THEN
JSTATE = 9
RETURN
END IF
ELSE
EVALFA = .TRUE.
END IF
DO 270 I = 1,N
270 SAVE2(I) = H*SAVE2(I)
MW = ML + 1 + MU
DO 290 J = 1,NDE
DO 290 I = MAX(ML+1, MW+1-J), MIN(MW+NDE-J, MW+ML)
SAVE2(I+J-MW) = SAVE2(I+J-MW)
8 - A(I,J)*(YH(J,2) + SAVE1(J))
290 CONTINUE
END IF
CALL SGBSL (DFDY, MATDIM, N, ML, MU, IPVT, SAVE2, 0)
IF (IERROR .EQ. 1 .OR. IERROR .EQ. 5) THEN
DO 300 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
300 SAVE2(I) = SAVE2(I)/YWT(I)
ELSE
DO 305 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
305 SAVE2(I) = SAVE2(I)/MAX(ABS(Y(I)), YWT(I))
END IF
D = SNRM2(N, SAVE2, 1)/SQRT(REAL(N))
ELSE IF (MITER .EQ. 3) THEN
IFLAG = 2
CALL USERS (Y, YH(1,2), YWT, SAVE1, SAVE2, T, H, EL(1,NQ), IMPL,
8 N, NDE, IFLAG)
IF (N .EQ. 0) THEN
JSTATE = 10
RETURN
END IF
IF (IERROR .EQ. 1 .OR. IERROR .EQ. 5) THEN
DO 320 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
320 SAVE2(I) = SAVE2(I)/YWT(I)
ELSE
DO 325 I = 1,N
SAVE1(I) = SAVE1(I) + SAVE2(I)
325 SAVE2(I) = SAVE2(I)/MAX(ABS(Y(I)), YWT(I))
END IF
D = SNRM2(N, SAVE2, 1)/SQRT(REAL(N))
END IF
RETURN
END