LAPACK 3.12.1
LAPACK: Linear Algebra PACKage
Loading...
Searching...
No Matches

◆ dsytrs_3()

subroutine dsytrs_3 ( character uplo,
integer n,
integer nrhs,
double precision, dimension( lda, * ) a,
integer lda,
double precision, dimension( * ) e,
integer, dimension( * ) ipiv,
double precision, dimension( ldb, * ) b,
integer ldb,
integer info )

DSYTRS_3

Download DSYTRS_3 + dependencies [TGZ] [ZIP] [TXT]

Purpose:
!> DSYTRS_3 solves a system of linear equations A * X = B with a real
!> symmetric matrix A using the factorization computed
!> by DSYTRF_RK or DSYTRF_BK:
!>
!>    A = P*U*D*(U**T)*(P**T) or A = P*L*D*(L**T)*(P**T),
!>
!> where U (or L) is unit upper (or lower) triangular matrix,
!> U**T (or L**T) is the transpose of U (or L), P is a permutation
!> matrix, P**T is the transpose of P, and D is symmetric and block
!> diagonal with 1-by-1 and 2-by-2 diagonal blocks.
!>
!> This algorithm is using Level 3 BLAS.
!> 
Parameters
[in]UPLO
!>          UPLO is CHARACTER*1
!>          Specifies whether the details of the factorization are
!>          stored as an upper or lower triangular matrix:
!>          = 'U':  Upper triangular, form is A = P*U*D*(U**T)*(P**T);
!>          = 'L':  Lower triangular, form is A = P*L*D*(L**T)*(P**T).
!> 
[in]N
!>          N is INTEGER
!>          The order of the matrix A.  N >= 0.
!> 
[in]NRHS
!>          NRHS is INTEGER
!>          The number of right hand sides, i.e., the number of columns
!>          of the matrix B.  NRHS >= 0.
!> 
[in]A
!>          A is DOUBLE PRECISION array, dimension (LDA,N)
!>          Diagonal of the block diagonal matrix D and factors U or L
!>          as computed by DSYTRF_RK and DSYTRF_BK:
!>            a) ONLY diagonal elements of the symmetric block diagonal
!>               matrix D on the diagonal of A, i.e. D(k,k) = A(k,k);
!>               (superdiagonal (or subdiagonal) elements of D
!>                should be provided on entry in array E), and
!>            b) If UPLO = 'U': factor U in the superdiagonal part of A.
!>               If UPLO = 'L': factor L in the subdiagonal part of A.
!> 
[in]LDA
!>          LDA is INTEGER
!>          The leading dimension of the array A.  LDA >= max(1,N).
!> 
[in]E
!>          E is DOUBLE PRECISION array, dimension (N)
!>          On entry, contains the superdiagonal (or subdiagonal)
!>          elements of the symmetric block diagonal matrix D
!>          with 1-by-1 or 2-by-2 diagonal blocks, where
!>          If UPLO = 'U': E(i) = D(i-1,i),i=2:N, E(1) not referenced;
!>          If UPLO = 'L': E(i) = D(i+1,i),i=1:N-1, E(N) not referenced.
!>
!>          NOTE: For 1-by-1 diagonal block D(k), where
!>          1 <= k <= N, the element E(k) is not referenced in both
!>          UPLO = 'U' or UPLO = 'L' cases.
!> 
[in]IPIV
!>          IPIV is INTEGER array, dimension (N)
!>          Details of the interchanges and the block structure of D
!>          as determined by DSYTRF_RK or DSYTRF_BK.
!> 
[in,out]B
!>          B is DOUBLE PRECISION array, dimension (LDB,NRHS)
!>          On entry, the right hand side matrix B.
!>          On exit, the solution matrix X.
!> 
[in]LDB
!>          LDB is INTEGER
!>          The leading dimension of the array B.  LDB >= max(1,N).
!> 
[out]INFO
!>          INFO is INTEGER
!>          = 0:  successful exit
!>          < 0:  if INFO = -i, the i-th argument had an illegal value
!> 
Author
Univ. of Tennessee
Univ. of California Berkeley
Univ. of Colorado Denver
NAG Ltd.
Contributors:
!>
!>  June 2017,  Igor Kozachenko,
!>                  Computer Science Division,
!>                  University of California, Berkeley
!>
!>  September 2007, Sven Hammarling, Nicholas J. Higham, Craig Lucas,
!>                  School of Mathematics,
!>                  University of Manchester
!>
!> 

Definition at line 161 of file dsytrs_3.f.

163*
164* -- LAPACK computational routine --
165* -- LAPACK is a software package provided by Univ. of Tennessee, --
166* -- Univ. of California Berkeley, Univ. of Colorado Denver and NAG Ltd..--
167*
168* .. Scalar Arguments ..
169 CHARACTER UPLO
170 INTEGER INFO, LDA, LDB, N, NRHS
171* ..
172* .. Array Arguments ..
173 INTEGER IPIV( * )
174 DOUBLE PRECISION A( LDA, * ), B( LDB, * ), E( * )
175* ..
176*
177* =====================================================================
178*
179* .. Parameters ..
180 DOUBLE PRECISION ONE
181 parameter( one = 1.0d+0 )
182* ..
183* .. Local Scalars ..
184 LOGICAL UPPER
185 INTEGER I, J, K, KP
186 DOUBLE PRECISION AK, AKM1, AKM1K, BK, BKM1, DENOM
187* ..
188* .. External Functions ..
189 LOGICAL LSAME
190 EXTERNAL lsame
191* ..
192* .. External Subroutines ..
193 EXTERNAL dscal, dswap, dtrsm, xerbla
194* ..
195* .. Intrinsic Functions ..
196 INTRINSIC abs, max
197* ..
198* .. Executable Statements ..
199*
200 info = 0
201 upper = lsame( uplo, 'U' )
202 IF( .NOT.upper .AND. .NOT.lsame( uplo, 'L' ) ) THEN
203 info = -1
204 ELSE IF( n.LT.0 ) THEN
205 info = -2
206 ELSE IF( nrhs.LT.0 ) THEN
207 info = -3
208 ELSE IF( lda.LT.max( 1, n ) ) THEN
209 info = -5
210 ELSE IF( ldb.LT.max( 1, n ) ) THEN
211 info = -9
212 END IF
213 IF( info.NE.0 ) THEN
214 CALL xerbla( 'DSYTRS_3', -info )
215 RETURN
216 END IF
217*
218* Quick return if possible
219*
220 IF( n.EQ.0 .OR. nrhs.EQ.0 )
221 $ RETURN
222*
223 IF( upper ) THEN
224*
225* Begin Upper
226*
227* Solve A*X = B, where A = U*D*U**T.
228*
229* P**T * B
230*
231* Interchange rows K and IPIV(K) of matrix B in the same order
232* that the formation order of IPIV(I) vector for Upper case.
233*
234* (We can do the simple loop over IPIV with decrement -1,
235* since the ABS value of IPIV( I ) represents the row index
236* of the interchange with row i in both 1x1 and 2x2 pivot cases)
237*
238 DO k = n, 1, -1
239 kp = abs( ipiv( k ) )
240 IF( kp.NE.k ) THEN
241 CALL dswap( nrhs, b( k, 1 ), ldb, b( kp, 1 ), ldb )
242 END IF
243 END DO
244*
245* Compute (U \P**T * B) -> B [ (U \P**T * B) ]
246*
247 CALL dtrsm( 'L', 'U', 'N', 'U', n, nrhs, one, a, lda, b,
248 $ ldb )
249*
250* Compute D \ B -> B [ D \ (U \P**T * B) ]
251*
252 i = n
253 DO WHILE ( i.GE.1 )
254 IF( ipiv( i ).GT.0 ) THEN
255 CALL dscal( nrhs, one / a( i, i ), b( i, 1 ), ldb )
256 ELSE IF ( i.GT.1 ) THEN
257 akm1k = e( i )
258 akm1 = a( i-1, i-1 ) / akm1k
259 ak = a( i, i ) / akm1k
260 denom = akm1*ak - one
261 DO j = 1, nrhs
262 bkm1 = b( i-1, j ) / akm1k
263 bk = b( i, j ) / akm1k
264 b( i-1, j ) = ( ak*bkm1-bk ) / denom
265 b( i, j ) = ( akm1*bk-bkm1 ) / denom
266 END DO
267 i = i - 1
268 END IF
269 i = i - 1
270 END DO
271*
272* Compute (U**T \ B) -> B [ U**T \ (D \ (U \P**T * B) ) ]
273*
274 CALL dtrsm( 'L', 'U', 'T', 'U', n, nrhs, one, a, lda, b,
275 $ ldb )
276*
277* P * B [ P * (U**T \ (D \ (U \P**T * B) )) ]
278*
279* Interchange rows K and IPIV(K) of matrix B in reverse order
280* from the formation order of IPIV(I) vector for Upper case.
281*
282* (We can do the simple loop over IPIV with increment 1,
283* since the ABS value of IPIV(I) represents the row index
284* of the interchange with row i in both 1x1 and 2x2 pivot cases)
285*
286 DO k = 1, n
287 kp = abs( ipiv( k ) )
288 IF( kp.NE.k ) THEN
289 CALL dswap( nrhs, b( k, 1 ), ldb, b( kp, 1 ), ldb )
290 END IF
291 END DO
292*
293 ELSE
294*
295* Begin Lower
296*
297* Solve A*X = B, where A = L*D*L**T.
298*
299* P**T * B
300* Interchange rows K and IPIV(K) of matrix B in the same order
301* that the formation order of IPIV(I) vector for Lower case.
302*
303* (We can do the simple loop over IPIV with increment 1,
304* since the ABS value of IPIV(I) represents the row index
305* of the interchange with row i in both 1x1 and 2x2 pivot cases)
306*
307 DO k = 1, n
308 kp = abs( ipiv( k ) )
309 IF( kp.NE.k ) THEN
310 CALL dswap( nrhs, b( k, 1 ), ldb, b( kp, 1 ), ldb )
311 END IF
312 END DO
313*
314* Compute (L \P**T * B) -> B [ (L \P**T * B) ]
315*
316 CALL dtrsm( 'L', 'L', 'N', 'U', n, nrhs, one, a, lda, b,
317 $ ldb )
318*
319* Compute D \ B -> B [ D \ (L \P**T * B) ]
320*
321 i = 1
322 DO WHILE ( i.LE.n )
323 IF( ipiv( i ).GT.0 ) THEN
324 CALL dscal( nrhs, one / a( i, i ), b( i, 1 ), ldb )
325 ELSE IF( i.LT.n ) THEN
326 akm1k = e( i )
327 akm1 = a( i, i ) / akm1k
328 ak = a( i+1, i+1 ) / akm1k
329 denom = akm1*ak - one
330 DO j = 1, nrhs
331 bkm1 = b( i, j ) / akm1k
332 bk = b( i+1, j ) / akm1k
333 b( i, j ) = ( ak*bkm1-bk ) / denom
334 b( i+1, j ) = ( akm1*bk-bkm1 ) / denom
335 END DO
336 i = i + 1
337 END IF
338 i = i + 1
339 END DO
340*
341* Compute (L**T \ B) -> B [ L**T \ (D \ (L \P**T * B) ) ]
342*
343 CALL dtrsm('L', 'L', 'T', 'U', n, nrhs, one, a, lda, b,
344 $ ldb )
345*
346* P * B [ P * (L**T \ (D \ (L \P**T * B) )) ]
347*
348* Interchange rows K and IPIV(K) of matrix B in reverse order
349* from the formation order of IPIV(I) vector for Lower case.
350*
351* (We can do the simple loop over IPIV with decrement -1,
352* since the ABS value of IPIV(I) represents the row index
353* of the interchange with row i in both 1x1 and 2x2 pivot cases)
354*
355 DO k = n, 1, -1
356 kp = abs( ipiv( k ) )
357 IF( kp.NE.k ) THEN
358 CALL dswap( nrhs, b( k, 1 ), ldb, b( kp, 1 ), ldb )
359 END IF
360 END DO
361*
362* END Lower
363*
364 END IF
365*
366 RETURN
367*
368* End of DSYTRS_3
369*
subroutine xerbla(srname, info)
Definition cblat2.f:3285
logical function lsame(ca, cb)
LSAME
Definition lsame.f:48
subroutine dscal(n, da, dx, incx)
DSCAL
Definition dscal.f:79
subroutine dswap(n, dx, incx, dy, incy)
DSWAP
Definition dswap.f:82
subroutine dtrsm(side, uplo, transa, diag, m, n, alpha, a, lda, b, ldb)
DTRSM
Definition dtrsm.f:181
Here is the call graph for this function:
Here is the caller graph for this function: