00001 SUBROUTINE DGEQPF( M, N, A, LDA, JPVT, TAU, WORK, INFO )
00002
00003
00004
00005
00006
00007
00008
00009 INTEGER INFO, LDA, M, N
00010
00011
00012 INTEGER JPVT( * )
00013 DOUBLE PRECISION A( LDA, * ), TAU( * ), WORK( * )
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087 DOUBLE PRECISION ZERO, ONE
00088 PARAMETER ( ZERO = 0.0D+0, ONE = 1.0D+0 )
00089
00090
00091 INTEGER I, ITEMP, J, MA, MN, PVT
00092 DOUBLE PRECISION AII, TEMP, TEMP2, TOL3Z
00093
00094
00095 EXTERNAL DGEQR2, DLARF, DLARFG, DORM2R, DSWAP, XERBLA
00096
00097
00098 INTRINSIC ABS, MAX, MIN, SQRT
00099
00100
00101 INTEGER IDAMAX
00102 DOUBLE PRECISION DLAMCH, DNRM2
00103 EXTERNAL IDAMAX, DLAMCH, DNRM2
00104
00105
00106
00107
00108
00109 INFO = 0
00110 IF( M.LT.0 ) THEN
00111 INFO = -1
00112 ELSE IF( N.LT.0 ) THEN
00113 INFO = -2
00114 ELSE IF( LDA.LT.MAX( 1, M ) ) THEN
00115 INFO = -4
00116 END IF
00117 IF( INFO.NE.0 ) THEN
00118 CALL XERBLA( 'DGEQPF', -INFO )
00119 RETURN
00120 END IF
00121
00122 MN = MIN( M, N )
00123 TOL3Z = SQRT(DLAMCH('Epsilon'))
00124
00125
00126
00127 ITEMP = 1
00128 DO 10 I = 1, N
00129 IF( JPVT( I ).NE.0 ) THEN
00130 IF( I.NE.ITEMP ) THEN
00131 CALL DSWAP( M, A( 1, I ), 1, A( 1, ITEMP ), 1 )
00132 JPVT( I ) = JPVT( ITEMP )
00133 JPVT( ITEMP ) = I
00134 ELSE
00135 JPVT( I ) = I
00136 END IF
00137 ITEMP = ITEMP + 1
00138 ELSE
00139 JPVT( I ) = I
00140 END IF
00141 10 CONTINUE
00142 ITEMP = ITEMP - 1
00143
00144
00145
00146 IF( ITEMP.GT.0 ) THEN
00147 MA = MIN( ITEMP, M )
00148 CALL DGEQR2( M, MA, A, LDA, TAU, WORK, INFO )
00149 IF( MA.LT.N ) THEN
00150 CALL DORM2R( 'Left', 'Transpose', M, N-MA, MA, A, LDA, TAU,
00151 $ A( 1, MA+1 ), LDA, WORK, INFO )
00152 END IF
00153 END IF
00154
00155 IF( ITEMP.LT.MN ) THEN
00156
00157
00158
00159
00160 DO 20 I = ITEMP + 1, N
00161 WORK( I ) = DNRM2( M-ITEMP, A( ITEMP+1, I ), 1 )
00162 WORK( N+I ) = WORK( I )
00163 20 CONTINUE
00164
00165
00166
00167 DO 40 I = ITEMP + 1, MN
00168
00169
00170
00171 PVT = ( I-1 ) + IDAMAX( N-I+1, WORK( I ), 1 )
00172
00173 IF( PVT.NE.I ) THEN
00174 CALL DSWAP( M, A( 1, PVT ), 1, A( 1, I ), 1 )
00175 ITEMP = JPVT( PVT )
00176 JPVT( PVT ) = JPVT( I )
00177 JPVT( I ) = ITEMP
00178 WORK( PVT ) = WORK( I )
00179 WORK( N+PVT ) = WORK( N+I )
00180 END IF
00181
00182
00183
00184 IF( I.LT.M ) THEN
00185 CALL DLARFG( M-I+1, A( I, I ), A( I+1, I ), 1, TAU( I ) )
00186 ELSE
00187 CALL DLARFG( 1, A( M, M ), A( M, M ), 1, TAU( M ) )
00188 END IF
00189
00190 IF( I.LT.N ) THEN
00191
00192
00193
00194 AII = A( I, I )
00195 A( I, I ) = ONE
00196 CALL DLARF( 'LEFT', M-I+1, N-I, A( I, I ), 1, TAU( I ),
00197 $ A( I, I+1 ), LDA, WORK( 2*N+1 ) )
00198 A( I, I ) = AII
00199 END IF
00200
00201
00202
00203 DO 30 J = I + 1, N
00204 IF( WORK( J ).NE.ZERO ) THEN
00205
00206
00207
00208
00209 TEMP = ABS( A( I, J ) ) / WORK( J )
00210 TEMP = MAX( ZERO, ( ONE+TEMP )*( ONE-TEMP ) )
00211 TEMP2 = TEMP*( WORK( J ) / WORK( N+J ) )**2
00212 IF( TEMP2 .LE. TOL3Z ) THEN
00213 IF( M-I.GT.0 ) THEN
00214 WORK( J ) = DNRM2( M-I, A( I+1, J ), 1 )
00215 WORK( N+J ) = WORK( J )
00216 ELSE
00217 WORK( J ) = ZERO
00218 WORK( N+J ) = ZERO
00219 END IF
00220 ELSE
00221 WORK( J ) = WORK( J )*SQRT( TEMP )
00222 END IF
00223 END IF
00224 30 CONTINUE
00225
00226 40 CONTINUE
00227 END IF
00228 RETURN
00229
00230
00231
00232 END