cg2
C CG2 SOURCE GOUNAND 22/08/25 21:15:02 11434 $ KPREC,INVDIA,ILUM,ILUI, $ LRES,LNMV, $ R,P,Q,Z, $ ITER,INMV,RESID,BRTOL,ICALRS,IDDOT,IMVEC,IMPR,IRET) IMPLICIT INTEGER(I-N) IMPLICIT REAL*8 (A-H,O-Z) C*********************************************************************** C NOM : CG2 C DESCRIPTION : C Résolution d'un système linéaire Ax=b C par la méthode du gradient conjugué préconditionné. C C LANGAGE : FORTRAN 77 + chouhia ESOPE (pour les E/S) C AUTEUR : Stéphane GOUNAND (CEA/DRN/DMT/SEMT/TTMF) C mél : gounand@semt2.smts.cea.fr C REFERENCE (bibtex-like) : C @BOOK{templates, C AUTHOR={R.Barrett, M.Berry, T.F.Chan, J.Demmel, J.Donato, C J.Dongarra, V.Eijkhout, R.Pozo, C.Romine, C H. Van der Vorst}, C TITLE={Templates for the Solution of Linear Systems : C Building Blocks for Iterative Methods}, C PUBLISHER={SIAM}, YEAR={1994}, ADDRESS={Philadelphia,PA} } C -> URL : http://www.netlib.org/templates/Templates.html C*********************************************************************** C APPELES (BLAS 1) : GCOPY, GAXPY (subroutines) C GNRM2, GDOT (functions) C APPELES (Calcul) : GMOMV : produit Matrice Morse - Vecteur C PSOLVE : Preconditionner solve C*********************************************************************** C ENTREES : C ENTREES/SORTIES : C SORTIES : C CODE RETOUR (IRET) : 0 si ok C <0 si problème (non utilisé !) C >0 si l'algorithme n'a pas convergé C*********************************************************************** C VERSION : v1, 09/02/06 C HISTORIQUE : v1, 09/02/06, création C HISTORIQUE : C HISTORIQUE : C*********************************************************************** C Prière de PRENDRE LE TEMPS de compléter les commentaires C en cas de modification de ce sous-programme afin de faciliter C la maintenance ! C*********************************************************************** * * .. Includes .. a commenter si mise au point ok -INC PPARAM -INC CCOPTIO -INC CCREEL -INC SMLREEL POINTEUR LRES.MLREEL -INC SMLENTI POINTEUR LNMV.MLENTI -INC SMMATRIK POINTEUR KMORS.PMORS POINTEUR KISA.IZA POINTEUR INVDIA.IZA POINTEUR ILUM.PMORS POINTEUR ILUI.IZA POINTEUR B.IZA POINTEUR X.IZA * .. Work Vectors for CG POINTEUR R.IZA,P.IZA,Q.IZA,Z.IZA * .. Scalar Arguments .. INTEGER ITER, KPREC, IMPR, IRET REAL*8 RESID * * * .. Variables locales * .. Parameters * .. INTEGER MAXIT * .. * .. External subroutines and functions.. * .. * .. Executable Statements .. * IRET = 0 INMV = 0 IWARN = 0 IGRAN = 0 MGRAN = 5 MAXIT = ITER TOL = RESID IF(IMPR.GT.5) WRITE(IOIMP,*) 'Entrée dans cg2.eso' * * Set parameter tolerances. * RHOTOL = BRTOL OMETOL = BRTOL * RRMAX = 10.D0 RRMAX = 1.D0/(XZPREC**0.25D0) * * Calcul du résidu initial * C r(0)=b C r(0)=b-Ax INMV=INMV+1 * ITER = 0 IF(IMPR.GT.5) THEN WRITE(IOIMP,*) '||B||=',BNRM2 ENDIF IF (ICALRS.EQ.1) BNRM2=RNRM2 IF (BNRM2.LT.XPETIT) BNRM2 = ONE RESID = RNRM2 / BNRM2 IF(IMPR.GT.5) THEN WRITE(IOIMP,*) 'Initial residual=',RESID ENDIF RESMAX = RESID * RRMAX LNMV.LECT(ITER+1)=INMV * IF (RESID.LE.TOL) THEN IRET = 0 GOTO 30 ENDIF * 10 CONTINUE * Gestion du CTRL-C if (ierr.NE.0) return * * Perform (Preconditioned) Conjugate Gradient Iteration. * ITER = ITER +1 IF(IMPR.GT.7) THEN WRITE(IOIMP,*) 'ITER=',ITER ENDIF * * Preconditioner Solve C Mz(i-1)=r(i-1) $ ,Z,R) C rho(i-1)=r(i-1)'z(i-1) * WRITE(IOIMP,*) ' RHO=',RHO IF (ABS(RHO).LT.RHOTOL) THEN RHO=SIGN(XZPREC,RHO) IWARN=IWARN+1 ENDIF * * Compute direction vector P * IF (ITER.GT.1) THEN C beta(i-1)=rho(i-1)/rho(i-2) BETA = RHO / RHO1 C p(i)=z(i-1)+beta(i-1)p(i-1) ELSE C p(1)=z(0) ENDIF * * Compute scalar ALPHA (save A*p to q) * C q(i)=Ap(i) INMV=INMV+1 C alpha(i)=rho(i-1)/(p(i)'q(i)) * WRITE(IOIMP,*) ' OMEGA=',OMEGA IF (ABS(OMEGA).LT.OMETOL) THEN OMEGA=SIGN(XZPREC,OMEGA) IWARN=IWARN+1 ENDIF * * Compute current solution vector x * C x(i)=x(i-1)+alpha(i)p(i) * * Compute residual vector R, find norm, * then check for tolerance * C r(i)=r(i-1)-alpha(i)r(i) * IF(IMPR.GT.7) THEN WRITE(IOIMP,*) 'Résidu=',RESID ENDIF LNMV.LECT(ITER+1)=INMV IF (RESID.GT.RESMAX) THEN IGRAN=IGRAN+1 ELSE IGRAN=0 ENDIF * * Iteration successful * IF (RESID.LE.TOL) THEN IRET = 0 * * Iteration fails * * ELSE IF (ITER.EQ.MAXIT.OR.IGRAN.GT.MGRAN) THEN ELSE IF (INMV.GE.MAXIT.OR.IGRAN.GT.MGRAN) THEN IRET = 1 * * Do some more * ELSE RHO1 = RHO GOTO 10 ENDIF * * Normal termination * 30 CONTINUE IF (IWARN.GT.0.AND.IMPR.GT.0) THEN WRITE(IOIMP,*) 'CG nearly failed ',IWARN $ ,' times.' ENDIF RETURN * * End of CG2 * END
© Cast3M 2003 - Tous droits réservés.
Mentions légales