LAPACK 3.3.0

dla_porcond.f

Go to the documentation of this file.
00001       DOUBLE PRECISION FUNCTION DLA_PORCOND( UPLO, N, A, LDA, AF, LDAF,
00002      $                                       CMODE, C, INFO, WORK,
00003      $                                       IWORK )
00004 *
00005 *     -- LAPACK routine (version 3.2.2)                                 --
00006 *     -- Contributed by James Demmel, Deaglan Halligan, Yozo Hida and --
00007 *     -- Jason Riedy of Univ. of California Berkeley.                 --
00008 *     -- June 2010                                                    --
00009 *
00010 *     -- LAPACK is a software package provided by Univ. of Tennessee, --
00011 *     -- Univ. of California Berkeley and NAG Ltd.                    --
00012 *
00013       IMPLICIT NONE
00014 *     ..
00015 *     .. Scalar Arguments ..
00016       CHARACTER          UPLO
00017       INTEGER            N, LDA, LDAF, INFO, CMODE
00018       DOUBLE PRECISION   A( LDA, * ), AF( LDAF, * ), WORK( * ),
00019      $                   C( * )
00020 *     ..
00021 *     .. Array Arguments ..
00022       INTEGER            IWORK( * )
00023 *     ..
00024 *
00025 *  Purpose
00026 *  =======
00027 *
00028 *     DLA_PORCOND Estimates the Skeel condition number of  op(A) * op2(C)
00029 *     where op2 is determined by CMODE as follows
00030 *     CMODE =  1    op2(C) = C
00031 *     CMODE =  0    op2(C) = I
00032 *     CMODE = -1    op2(C) = inv(C)
00033 *     The Skeel condition number  cond(A) = norminf( |inv(A)||A| )
00034 *     is computed by computing scaling factors R such that
00035 *     diag(R)*A*op2(C) is row equilibrated and computing the standard
00036 *     infinity-norm condition number.
00037 *
00038 *  Arguments
00039 *  ==========
00040 *
00041 *     UPLO    (input) CHARACTER*1
00042 *       = 'U':  Upper triangle of A is stored;
00043 *       = 'L':  Lower triangle of A is stored.
00044 *
00045 *     N       (input) INTEGER
00046 *     The number of linear equations, i.e., the order of the
00047 *     matrix A.  N >= 0.
00048 *
00049 *     A       (input) DOUBLE PRECISION array, dimension (LDA,N)
00050 *     On entry, the N-by-N matrix A.
00051 *
00052 *     LDA     (input) INTEGER
00053 *     The leading dimension of the array A.  LDA >= max(1,N).
00054 *
00055 *     AF      (input) DOUBLE PRECISION array, dimension (LDAF,N)
00056 *     The triangular factor U or L from the Cholesky factorization
00057 *     A = U**T*U or A = L*L**T, as computed by DPOTRF.
00058 *
00059 *     LDAF    (input) INTEGER
00060 *     The leading dimension of the array AF.  LDAF >= max(1,N).
00061 *
00062 *     CMODE   (input) INTEGER
00063 *     Determines op2(C) in the formula op(A) * op2(C) as follows:
00064 *     CMODE =  1    op2(C) = C
00065 *     CMODE =  0    op2(C) = I
00066 *     CMODE = -1    op2(C) = inv(C)
00067 *
00068 *     C       (input) DOUBLE PRECISION array, dimension (N)
00069 *     The vector C in the formula op(A) * op2(C).
00070 *
00071 *     INFO    (output) INTEGER
00072 *       = 0:  Successful exit.
00073 *     i > 0:  The ith argument is invalid.
00074 *
00075 *     WORK    (input) DOUBLE PRECISION array, dimension (3*N).
00076 *     Workspace.
00077 *
00078 *     IWORK   (input) INTEGER array, dimension (N).
00079 *     Workspace.
00080 *
00081 *  =====================================================================
00082 *
00083 *     .. Local Scalars ..
00084       INTEGER            KASE, I, J
00085       DOUBLE PRECISION   AINVNM, TMP
00086       LOGICAL            UP
00087 *     ..
00088 *     .. Array Arguments ..
00089       INTEGER            ISAVE( 3 )
00090 *     ..
00091 *     .. External Functions ..
00092       LOGICAL            LSAME
00093       INTEGER            IDAMAX
00094       EXTERNAL           LSAME, IDAMAX
00095 *     ..
00096 *     .. External Subroutines ..
00097       EXTERNAL           DLACN2, DPOTRS, XERBLA
00098 *     ..
00099 *     .. Intrinsic Functions ..
00100       INTRINSIC          ABS, MAX
00101 *     ..
00102 *     .. Executable Statements ..
00103 *
00104       DLA_PORCOND = 0.0D+0
00105 *
00106       INFO = 0
00107       IF( N.LT.0 ) THEN
00108          INFO = -2
00109       END IF
00110       IF( INFO.NE.0 ) THEN
00111          CALL XERBLA( 'DLA_PORCOND', -INFO )
00112          RETURN
00113       END IF
00114 
00115       IF( N.EQ.0 ) THEN
00116          DLA_PORCOND = 1.0D+0
00117          RETURN
00118       END IF
00119       UP = .FALSE.
00120       IF ( LSAME( UPLO, 'U' ) ) UP = .TRUE.
00121 *
00122 *     Compute the equilibration matrix R such that
00123 *     inv(R)*A*C has unit 1-norm.
00124 *
00125       IF ( UP ) THEN
00126          DO I = 1, N
00127             TMP = 0.0D+0
00128             IF ( CMODE .EQ. 1 ) THEN
00129                DO J = 1, I
00130                   TMP = TMP + ABS( A( J, I ) * C( J ) )
00131                END DO
00132                DO J = I+1, N
00133                   TMP = TMP + ABS( A( I, J ) * C( J ) )
00134                END DO
00135             ELSE IF ( CMODE .EQ. 0 ) THEN
00136                DO J = 1, I
00137                   TMP = TMP + ABS( A( J, I ) )
00138                END DO
00139                DO J = I+1, N
00140                   TMP = TMP + ABS( A( I, J ) )
00141                END DO
00142             ELSE
00143                DO J = 1, I
00144                   TMP = TMP + ABS( A( J ,I ) / C( J ) )
00145                END DO
00146                DO J = I+1, N
00147                   TMP = TMP + ABS( A( I, J ) / C( J ) )
00148                END DO
00149             END IF
00150             WORK( 2*N+I ) = TMP
00151          END DO
00152       ELSE
00153          DO I = 1, N
00154             TMP = 0.0D+0
00155             IF ( CMODE .EQ. 1 ) THEN
00156                DO J = 1, I
00157                   TMP = TMP + ABS( A( I, J ) * C( J ) )
00158                END DO
00159                DO J = I+1, N
00160                   TMP = TMP + ABS( A( J, I ) * C( J ) )
00161                END DO
00162             ELSE IF ( CMODE .EQ. 0 ) THEN
00163                DO J = 1, I
00164                   TMP = TMP + ABS( A( I, J ) )
00165                END DO
00166                DO J = I+1, N
00167                   TMP = TMP + ABS( A( J, I ) )
00168                END DO
00169             ELSE
00170                DO J = 1, I
00171                   TMP = TMP + ABS( A( I, J ) / C( J ) )
00172                END DO
00173                DO J = I+1, N
00174                   TMP = TMP + ABS( A( J, I ) / C( J ) )
00175                END DO
00176             END IF
00177             WORK( 2*N+I ) = TMP
00178          END DO
00179       ENDIF
00180 *
00181 *     Estimate the norm of inv(op(A)).
00182 *
00183       AINVNM = 0.0D+0
00184 
00185       KASE = 0
00186    10 CONTINUE
00187       CALL DLACN2( N, WORK( N+1 ), WORK, IWORK, AINVNM, KASE, ISAVE )
00188       IF( KASE.NE.0 ) THEN
00189          IF( KASE.EQ.2 ) THEN
00190 *
00191 *           Multiply by R.
00192 *
00193             DO I = 1, N
00194                WORK( I ) = WORK( I ) * WORK( 2*N+I )
00195             END DO
00196 
00197             IF (UP) THEN
00198                CALL DPOTRS( 'Upper', N, 1, AF, LDAF, WORK, N, INFO )
00199             ELSE
00200                CALL DPOTRS( 'Lower', N, 1, AF, LDAF, WORK, N, INFO )
00201             ENDIF
00202 *
00203 *           Multiply by inv(C).
00204 *
00205             IF ( CMODE .EQ. 1 ) THEN
00206                DO I = 1, N
00207                   WORK( I ) = WORK( I ) / C( I )
00208                END DO
00209             ELSE IF ( CMODE .EQ. -1 ) THEN
00210                DO I = 1, N
00211                   WORK( I ) = WORK( I ) * C( I )
00212                END DO
00213             END IF
00214          ELSE
00215 *
00216 *           Multiply by inv(C').
00217 *
00218             IF ( CMODE .EQ. 1 ) THEN
00219                DO I = 1, N
00220                   WORK( I ) = WORK( I ) / C( I )
00221                END DO
00222             ELSE IF ( CMODE .EQ. -1 ) THEN
00223                DO I = 1, N
00224                   WORK( I ) = WORK( I ) * C( I )
00225                END DO
00226             END IF
00227 
00228             IF ( UP ) THEN
00229                CALL DPOTRS( 'Upper', N, 1, AF, LDAF, WORK, N, INFO )
00230             ELSE
00231                CALL DPOTRS( 'Lower', N, 1, AF, LDAF, WORK, N, INFO )
00232             ENDIF
00233 *
00234 *           Multiply by R.
00235 *
00236             DO I = 1, N
00237                WORK( I ) = WORK( I ) * WORK( 2*N+I )
00238             END DO
00239          END IF
00240          GO TO 10
00241       END IF
00242 *
00243 *     Compute the estimate of the reciprocal condition number.
00244 *
00245       IF( AINVNM .NE. 0.0D+0 )
00246      $   DLA_PORCOND = ( 1.0D+0 / AINVNM )
00247 *
00248       RETURN
00249 *
00250       END
 All Files Functions