OpenLibm/slatec/r1updt.f
Viral B. Shah c977aa998f Add Makefile.extras to build libopenlibm-extras.
Replace amos with slatec
2012-12-31 16:37:05 -05:00

209 lines
5.8 KiB
Fortran

*DECK R1UPDT
SUBROUTINE R1UPDT (M, N, S, LS, U, V, W, SING)
C***BEGIN PROLOGUE R1UPDT
C***SUBSIDIARY
C***PURPOSE Subsidiary to SNSQ and SNSQE
C***LIBRARY SLATEC
C***TYPE SINGLE PRECISION (R1UPDT-S, D1UPDT-D)
C***AUTHOR (UNKNOWN)
C***DESCRIPTION
C
C Given an M by N lower trapezoidal matrix S, an M-vector U,
C and an N-vector V, the problem is to determine an
C orthogonal matrix Q such that
C
C T
C (S + U*V )*Q
C
C is again lower trapezoidal.
C
C This subroutine determines Q as the product of 2*(N - 1)
C transformations
C
C GV(N-1)*...*GV(1)*GW(1)*...*GW(N-1)
C
C where GV(I), GW(I) are Givens rotations in the (I,N) plane
C which eliminate elements in the I-th and N-th planes,
C respectively. Q Itself is not accumulated, rather the
C information to recover the GV, GW rotations is returned.
C
C The subroutine statement is
C
C SUBROUTINE R1UPDT(M,N,S,LS,U,V,W,SING)
C
C where
C
C M is a positive integer input variable set to the number
C of rows of S.
C
C N is a positive integer input variable set to the number
C of columns of S. N must not exceed M.
C
C S is an array of length LS. On input S must contain the lower
C trapezoidal matrix S stored by columns. On output S contains
C the lower trapezoidal matrix produced as described above.
C
C LS is a positive integer input variable not less than
C (N*(2*M-N+1))/2.
C
C U is an input array of length M which must contain the
C vector U.
C
C V is an array of length N. On input V must contain the vector
C V. On output V(I) contains the information necessary to
C recover the Givens rotation GV(I) described above.
C
C W is an output array of length M. W(I) contains information
C necessary to recover the Givens rotation GW(I) described
C above.
C
C SING is a logical output variable. SING is set .TRUE. if any
C of the diagonal elements of the output S are zero. Otherwise
C SING is set .FALSE.
C
C***SEE ALSO SNSQ, SNSQE
C***ROUTINES CALLED R1MACH
C***REVISION HISTORY (YYMMDD)
C 800301 DATE WRITTEN
C 890831 Modified array declarations. (WRB)
C 891214 Prologue converted to Version 4.0 format. (BAB)
C 900326 Removed duplicate information from DESCRIPTION section.
C (WRB)
C 900328 Added TYPE section. (WRB)
C***END PROLOGUE R1UPDT
INTEGER M,N,LS
LOGICAL SING
REAL S(*),U(*),V(*),W(*)
INTEGER I,J,JJ,L,NMJ,NM1
REAL COS,COTAN,GIANT,ONE,P5,P25,SIN,TAN,TAU,TEMP,ZERO
REAL R1MACH
SAVE ONE, P5, P25, ZERO
DATA ONE,P5,P25,ZERO /1.0E0,5.0E-1,2.5E-1,0.0E0/
C***FIRST EXECUTABLE STATEMENT R1UPDT
GIANT = R1MACH(2)
C
C INITIALIZE THE DIAGONAL ELEMENT POINTER.
C
JJ = (N*(2*M - N + 1))/2 - (M - N)
C
C MOVE THE NONTRIVIAL PART OF THE LAST COLUMN OF S INTO W.
C
L = JJ
DO 10 I = N, M
W(I) = S(L)
L = L + 1
10 CONTINUE
C
C ROTATE THE VECTOR V INTO A MULTIPLE OF THE N-TH UNIT VECTOR
C IN SUCH A WAY THAT A SPIKE IS INTRODUCED INTO W.
C
NM1 = N - 1
IF (NM1 .LT. 1) GO TO 70
DO 60 NMJ = 1, NM1
J = N - NMJ
JJ = JJ - (M - J + 1)
W(J) = ZERO
IF (V(J) .EQ. ZERO) GO TO 50
C
C DETERMINE A GIVENS ROTATION WHICH ELIMINATES THE
C J-TH ELEMENT OF V.
C
IF (ABS(V(N)) .GE. ABS(V(J))) GO TO 20
COTAN = V(N)/V(J)
SIN = P5/SQRT(P25+P25*COTAN**2)
COS = SIN*COTAN
TAU = ONE
IF (ABS(COS)*GIANT .GT. ONE) TAU = ONE/COS
GO TO 30
20 CONTINUE
TAN = V(J)/V(N)
COS = P5/SQRT(P25+P25*TAN**2)
SIN = COS*TAN
TAU = SIN
30 CONTINUE
C
C APPLY THE TRANSFORMATION TO V AND STORE THE INFORMATION
C NECESSARY TO RECOVER THE GIVENS ROTATION.
C
V(N) = SIN*V(J) + COS*V(N)
V(J) = TAU
C
C APPLY THE TRANSFORMATION TO S AND EXTEND THE SPIKE IN W.
C
L = JJ
DO 40 I = J, M
TEMP = COS*S(L) - SIN*W(I)
W(I) = SIN*S(L) + COS*W(I)
S(L) = TEMP
L = L + 1
40 CONTINUE
50 CONTINUE
60 CONTINUE
70 CONTINUE
C
C ADD THE SPIKE FROM THE RANK 1 UPDATE TO W.
C
DO 80 I = 1, M
W(I) = W(I) + V(N)*U(I)
80 CONTINUE
C
C ELIMINATE THE SPIKE.
C
SING = .FALSE.
IF (NM1 .LT. 1) GO TO 140
DO 130 J = 1, NM1
IF (W(J) .EQ. ZERO) GO TO 120
C
C DETERMINE A GIVENS ROTATION WHICH ELIMINATES THE
C J-TH ELEMENT OF THE SPIKE.
C
IF (ABS(S(JJ)) .GE. ABS(W(J))) GO TO 90
COTAN = S(JJ)/W(J)
SIN = P5/SQRT(P25+P25*COTAN**2)
COS = SIN*COTAN
TAU = ONE
IF (ABS(COS)*GIANT .GT. ONE) TAU = ONE/COS
GO TO 100
90 CONTINUE
TAN = W(J)/S(JJ)
COS = P5/SQRT(P25+P25*TAN**2)
SIN = COS*TAN
TAU = SIN
100 CONTINUE
C
C APPLY THE TRANSFORMATION TO S AND REDUCE THE SPIKE IN W.
C
L = JJ
DO 110 I = J, M
TEMP = COS*S(L) + SIN*W(I)
W(I) = -SIN*S(L) + COS*W(I)
S(L) = TEMP
L = L + 1
110 CONTINUE
C
C STORE THE INFORMATION NECESSARY TO RECOVER THE
C GIVENS ROTATION.
C
W(J) = TAU
120 CONTINUE
C
C TEST FOR ZERO DIAGONAL ELEMENTS IN THE OUTPUT S.
C
IF (S(JJ) .EQ. ZERO) SING = .TRUE.
JJ = JJ + (M - J + 1)
130 CONTINUE
140 CONTINUE
C
C MOVE W BACK INTO THE LAST COLUMN OF THE OUTPUT S.
C
L = JJ
DO 150 I = N, M
S(L) = W(I)
L = L + 1
150 CONTINUE
IF (S(JJ) .EQ. ZERO) SING = .TRUE.
RETURN
C
C LAST CARD OF SUBROUTINE R1UPDT.
C
END