OpenLibm/slatec/dxset.f

332 lines
13 KiB
FortranFixed
Raw Normal View History

*DECK DXSET
SUBROUTINE DXSET (IRAD, NRADPL, DZERO, NBITS, IERROR)
C***BEGIN PROLOGUE DXSET
C***PURPOSE To provide double-precision floating-point arithmetic
C with an extended exponent range.
C***LIBRARY SLATEC
C***CATEGORY A3D
C***TYPE DOUBLE PRECISION (XSET-S, DXSET-D)
C***KEYWORDS EXTENDED-RANGE DOUBLE-PRECISION ARITHMETIC
C***AUTHOR Lozier, Daniel W., (National Bureau of Standards)
C Smith, John M., (NBS and George Mason University)
C***DESCRIPTION
C
C SUBROUTINE DXSET MUST BE CALLED PRIOR TO CALLING ANY OTHER
C EXTENDED-RANGE SUBROUTINE. IT CALCULATES AND STORES SEVERAL
C MACHINE-DEPENDENT CONSTANTS IN COMMON BLOCKS. THE USER MUST
C SUPPLY FOUR CONSTANTS THAT PERTAIN TO HIS PARTICULAR COMPUTER.
C THE CONSTANTS ARE
C
C IRAD = THE INTERNAL BASE OF DOUBLE-PRECISION
C ARITHMETIC IN THE COMPUTER.
C NRADPL = THE NUMBER OF RADIX PLACES CARRIED IN
C THE DOUBLE-PRECISION REPRESENTATION.
C DZERO = THE SMALLEST OF 1/DMIN, DMAX, DMAXLN WHERE
C DMIN = THE SMALLEST POSITIVE DOUBLE-PRECISION
C NUMBER OR AN UPPER BOUND TO THIS NUMBER,
C DMAX = THE LARGEST DOUBLE-PRECISION NUMBER
C OR A LOWER BOUND TO THIS NUMBER,
C DMAXLN = THE LARGEST DOUBLE-PRECISION NUMBER
C SUCH THAT LOG10(DMAXLN) CAN BE COMPUTED BY THE
C FORTRAN SYSTEM (ON MOST SYSTEMS DMAXLN = DMAX).
C NBITS = THE NUMBER OF BITS (EXCLUSIVE OF SIGN) IN
C AN INTEGER COMPUTER WORD.
C
C ALTERNATIVELY, ANY OR ALL OF THE CONSTANTS CAN BE GIVEN
C THE VALUE 0 (0.0D0 FOR DZERO). IF A CONSTANT IS ZERO, DXSET TRIES
C TO ASSIGN AN APPROPRIATE VALUE BY CALLING I1MACH
C (SEE P.A.FOX, A.D.HALL, N.L.SCHRYER, ALGORITHM 528 FRAMEWORK
C FOR A PORTABLE LIBRARY, ACM TRANSACTIONS ON MATH SOFTWARE,
C V.4, NO.2, JUNE 1978, 177-188).
C
C THIS IS THE SETTING-UP SUBROUTINE FOR A PACKAGE OF SUBROUTINES
C THAT FACILITATE THE USE OF EXTENDED-RANGE ARITHMETIC. EXTENDED-RANGE
C ARITHMETIC ON A PARTICULAR COMPUTER IS DEFINED ON THE SET OF NUMBERS
C OF THE FORM
C
C (X,IX) = X*RADIX**IX
C
C WHERE X IS A DOUBLE-PRECISION NUMBER CALLED THE PRINCIPAL PART,
C IX IS AN INTEGER CALLED THE AUXILIARY INDEX, AND RADIX IS THE
C INTERNAL BASE OF THE DOUBLE-PRECISION ARITHMETIC. OBVIOUSLY,
C EACH REAL NUMBER IS REPRESENTABLE WITHOUT ERROR BY MORE THAN ONE
C EXTENDED-RANGE FORM. CONVERSIONS BETWEEN DIFFERENT FORMS ARE
C ESSENTIAL IN CARRYING OUT ARITHMETIC OPERATIONS. WITH THE CHOICE
C OF RADIX WE HAVE MADE, AND THE SUBROUTINES WE HAVE WRITTEN, THESE
C CONVERSIONS ARE PERFORMED WITHOUT ERROR (AT LEAST ON MOST COMPUTERS).
C (SEE SMITH, J.M., OLVER, F.W.J., AND LOZIER, D.W., EXTENDED-RANGE
C ARITHMETIC AND NORMALIZED LEGENDRE POLYNOMIALS, ACM TRANSACTIONS ON
C MATHEMATICAL SOFTWARE, MARCH 1981).
C
C AN EXTENDED-RANGE NUMBER (X,IX) IS SAID TO BE IN ADJUSTED FORM IF
C X AND IX ARE ZERO OR
C
C RADIX**(-L) .LE. ABS(X) .LT. RADIX**L
C
C IS SATISFIED, WHERE L IS A COMPUTER-DEPENDENT INTEGER DEFINED IN THIS
C SUBROUTINE. TWO EXTENDED-RANGE NUMBERS IN ADJUSTED FORM CAN BE ADDED,
C SUBTRACTED, MULTIPLIED OR DIVIDED (IF THE DIVISOR IS NONZERO) WITHOUT
C CAUSING OVERFLOW OR UNDERFLOW IN THE PRINCIPAL PART OF THE RESULT.
C WITH PROPER USE OF THE EXTENDED-RANGE SUBROUTINES, THE ONLY OVERFLOW
C THAT CAN OCCUR IS INTEGER OVERFLOW IN THE AUXILIARY INDEX. IF THIS
C IS DETECTED, THE SOFTWARE CALLS XERROR (A GENERAL ERROR-HANDLING
C FORTRAN SUBROUTINE PACKAGE).
C
C MULTIPLICATION AND DIVISION IS PERFORMED BY SETTING
C
C (X,IX)*(Y,IY) = (X*Y,IX+IY)
C OR
C (X,IX)/(Y,IY) = (X/Y,IX-IY).
C
C PRE-ADJUSTMENT OF THE OPERANDS IS ESSENTIAL TO AVOID
C OVERFLOW OR UNDERFLOW OF THE PRINCIPAL PART. SUBROUTINE
C DXADJ (SEE BELOW) MAY BE CALLED TO TRANSFORM ANY EXTENDED-
C RANGE NUMBER INTO ADJUSTED FORM.
C
C ADDITION AND SUBTRACTION REQUIRE THE USE OF SUBROUTINE DXADD
C (SEE BELOW). THE INPUT OPERANDS NEED NOT BE IN ADJUSTED FORM.
C HOWEVER, THE RESULT OF ADDITION OR SUBTRACTION IS RETURNED
C IN ADJUSTED FORM. THUS, FOR EXAMPLE, IF (X,IX),(Y,IY),
C (U,IU), AND (V,IV) ARE IN ADJUSTED FORM, THEN
C
C (X,IX)*(Y,IY) + (U,IU)*(V,IV)
C
C CAN BE COMPUTED AND STORED IN ADJUSTED FORM WITH NO EXPLICIT
C CALLS TO DXADJ.
C
C WHEN AN EXTENDED-RANGE NUMBER IS TO BE PRINTED, IT MUST BE
C CONVERTED TO AN EXTENDED-RANGE FORM WITH DECIMAL RADIX. SUBROUTINE
C DXCON IS PROVIDED FOR THIS PURPOSE.
C
C THE SUBROUTINES CONTAINED IN THIS PACKAGE ARE
C
C SUBROUTINE DXADD
C USAGE
C CALL DXADD(X,IX,Y,IY,Z,IZ,IERROR)
C IF (IERROR.NE.0) RETURN
C DESCRIPTION
C FORMS THE EXTENDED-RANGE SUM (Z,IZ) =
C (X,IX) + (Y,IY). (Z,IZ) IS ADJUSTED
C BEFORE RETURNING. THE INPUT OPERANDS
C NEED NOT BE IN ADJUSTED FORM, BUT THEIR
C PRINCIPAL PARTS MUST SATISFY
C RADIX**(-2L).LE.ABS(X).LE.RADIX**(2L),
C RADIX**(-2L).LE.ABS(Y).LE.RADIX**(2L).
C
C SUBROUTINE DXADJ
C USAGE
C CALL DXADJ(X,IX,IERROR)
C IF (IERROR.NE.0) RETURN
C DESCRIPTION
C TRANSFORMS (X,IX) SO THAT
C RADIX**(-L) .LE. ABS(X) .LT. RADIX**L.
C ON MOST COMPUTERS THIS TRANSFORMATION DOES
C NOT CHANGE THE MANTISSA OF X PROVIDED RADIX IS
C THE NUMBER BASE OF DOUBLE-PRECISION ARITHMETIC.
C
C SUBROUTINE DXC210
C USAGE
C CALL DXC210(K,Z,J,IERROR)
C IF (IERROR.NE.0) RETURN
C DESCRIPTION
C GIVEN K THIS SUBROUTINE COMPUTES J AND Z
C SUCH THAT RADIX**K = Z*10**J, WHERE Z IS IN
C THE RANGE 1/10 .LE. Z .LT. 1.
C THE VALUE OF Z WILL BE ACCURATE TO FULL
C DOUBLE-PRECISION PROVIDED THE NUMBER
C OF DECIMAL PLACES IN THE LARGEST
C INTEGER PLUS THE NUMBER OF DECIMAL
C PLACES CARRIED IN DOUBLE-PRECISION DOES NOT
C EXCEED 60. DXC210 IS CALLED BY SUBROUTINE
C DXCON WHEN NECESSARY. THE USER SHOULD
C NEVER NEED TO CALL DXC210 DIRECTLY.
C
C SUBROUTINE DXCON
C USAGE
C CALL DXCON(X,IX,IERROR)
C IF (IERROR.NE.0) RETURN
C DESCRIPTION
C CONVERTS (X,IX) = X*RADIX**IX
C TO DECIMAL FORM IN PREPARATION FOR
C PRINTING, SO THAT (X,IX) = X*10**IX
C WHERE 1/10 .LE. ABS(X) .LT. 1
C IS RETURNED, EXCEPT THAT IF
C (ABS(X),IX) IS BETWEEN RADIX**(-2L)
C AND RADIX**(2L) THEN THE REDUCED
C FORM WITH IX = 0 IS RETURNED.
C
C SUBROUTINE DXRED
C USAGE
C CALL DXRED(X,IX,IERROR)
C IF (IERROR.NE.0) RETURN
C DESCRIPTION
C IF
C RADIX**(-2L) .LE. (ABS(X),IX) .LE. RADIX**(2L)
C THEN DXRED TRANSFORMS (X,IX) SO THAT IX=0.
C IF (X,IX) IS OUTSIDE THE ABOVE RANGE,
C THEN DXRED TAKES NO ACTION.
C THIS SUBROUTINE IS USEFUL IF THE
C RESULTS OF EXTENDED-RANGE CALCULATIONS
C ARE TO BE USED IN SUBSEQUENT ORDINARY
C DOUBLE-PRECISION CALCULATIONS.
C
C***REFERENCES Smith, Olver and Lozier, Extended-Range Arithmetic and
C Normalized Legendre Polynomials, ACM Trans on Math
C Softw, v 7, n 1, March 1981, pp 93--105.
C***ROUTINES CALLED I1MACH, XERMSG
C***COMMON BLOCKS DXBLK1, DXBLK2, DXBLK3
C***REVISION HISTORY (YYMMDD)
C 820712 DATE WRITTEN
C 881020 Revised to meet SLATEC CML recommendations. (DWL and JMS)
C 901019 Revisions to prologue. (DWL and WRB)
C 901106 Changed all specific intrinsics to generic. (WRB)
C Corrected order of sections in prologue and added TYPE
C section. (WRB)
C CALLs to XERROR changed to CALLs to XERMSG. (WRB)
C 920127 Revised PURPOSE section of prologue. (DWL)
C***END PROLOGUE DXSET
INTEGER IRAD, NRADPL, NBITS
DOUBLE PRECISION DZERO, DZEROX
COMMON /DXBLK1/ NBITSF
SAVE /DXBLK1/
DOUBLE PRECISION RADIX, RADIXL, RAD2L, DLG10R
INTEGER L, L2, KMAX
COMMON /DXBLK2/ RADIX, RADIXL, RAD2L, DLG10R, L, L2, KMAX
SAVE /DXBLK2/
INTEGER NLG102, MLG102, LG102
COMMON /DXBLK3/ NLG102, MLG102, LG102(21)
SAVE /DXBLK3/
INTEGER IFLAG
SAVE IFLAG
C
DIMENSION LOG102(20), LGTEMP(20)
SAVE LOG102
C
C LOG102 CONTAINS THE FIRST 60 DIGITS OF LOG10(2) FOR USE IN
C CONVERSION OF EXTENDED-RANGE NUMBERS TO BASE 10 .
DATA LOG102 /301,029,995,663,981,195,213,738,894,724,493,026,768,
* 189,881,462,108,541,310,428/
C
C FOLLOWING CODING PREVENTS DXSET FROM BEING EXECUTED MORE THAN ONCE.
C THIS IS IMPORTANT BECAUSE SOME SUBROUTINES (SUCH AS DXNRMP AND
C DXLEGF) CALL DXSET TO MAKE SURE EXTENDED-RANGE ARITHMETIC HAS
C BEEN INITIALIZED. THE USER MAY WANT TO PRE-EMPT THIS CALL, FOR
C EXAMPLE WHEN I1MACH IS NOT AVAILABLE. SEE CODING BELOW.
DATA IFLAG /0/
C***FIRST EXECUTABLE STATEMENT DXSET
IERROR=0
IF (IFLAG .NE. 0) RETURN
IRADX = IRAD
NRDPLC = NRADPL
DZEROX = DZERO
IMINEX = 0
IMAXEX = 0
NBITSX = NBITS
C FOLLOWING 5 STATEMENTS SHOULD BE DELETED IF I1MACH IS
C NOT AVAILABLE OR NOT CONFIGURED TO RETURN THE CORRECT
C MACHINE-DEPENDENT VALUES.
IF (IRADX .EQ. 0) IRADX = I1MACH (10)
IF (NRDPLC .EQ. 0) NRDPLC = I1MACH (14)
IF (DZEROX .EQ. 0.0D0) IMINEX = I1MACH (15)
IF (DZEROX .EQ. 0.0D0) IMAXEX = I1MACH (16)
IF (NBITSX .EQ. 0) NBITSX = I1MACH (8)
IF (IRADX.EQ.2) GO TO 10
IF (IRADX.EQ.4) GO TO 10
IF (IRADX.EQ.8) GO TO 10
IF (IRADX.EQ.16) GO TO 10
CALL XERMSG ('SLATEC', 'DXSET', 'IMPROPER VALUE OF IRAD', 201, 1)
IERROR=201
RETURN
10 CONTINUE
LOG2R=0
IF (IRADX.EQ.2) LOG2R = 1
IF (IRADX.EQ.4) LOG2R = 2
IF (IRADX.EQ.8) LOG2R = 3
IF (IRADX.EQ.16) LOG2R = 4
NBITSF=LOG2R*NRDPLC
RADIX = IRADX
DLG10R = LOG10(RADIX)
IF (DZEROX .NE. 0.0D0) GO TO 14
LX = MIN ((1-IMINEX)/2, (IMAXEX-1)/2)
GO TO 16
14 LX = 0.5D0*LOG10(DZEROX)/DLG10R
C RADIX**(2*L) SHOULD NOT OVERFLOW, BUT REDUCE L BY 1 FOR FURTHER
C PROTECTION.
LX=LX-1
16 L2 = 2*LX
IF (LX.GE.4) GO TO 20
CALL XERMSG ('SLATEC', 'DXSET', 'IMPROPER VALUE OF DZERO', 202, 1)
IERROR=202
RETURN
20 L = LX
RADIXL = RADIX**L
RAD2L = RADIXL**2
C IT IS NECESSARY TO RESTRICT NBITS (OR NBITSX) TO BE LESS THAN SOME
C UPPER LIMIT BECAUSE OF BINARY-TO-DECIMAL CONVERSION. SUCH CONVERSION
C IS DONE BY DXC210 AND REQUIRES A CONSTANT THAT IS STORED TO SOME FIXED
C PRECISION. THE STORED CONSTANT (LOG102 IN THIS ROUTINE) PROVIDES
C FOR CONVERSIONS ACCURATE TO THE LAST DECIMAL DIGIT WHEN THE INTEGER
C WORD LENGTH DOES NOT EXCEED 63. A LOWER LIMIT OF 15 BITS IS IMPOSED
C BECAUSE THE SOFTWARE IS DESIGNED TO RUN ON COMPUTERS WITH INTEGER WORD
C LENGTH OF AT LEAST 16 BITS.
IF (15.LE.NBITSX .AND. NBITSX.LE.63) GO TO 30
CALL XERMSG ('SLATEC', 'DXSET', 'IMPROPER VALUE OF NBITS', 203, 1)
IERROR=203
RETURN
30 CONTINUE
KMAX = 2**(NBITSX-1) - L2
NB = (NBITSX-1)/2
MLG102 = 2**NB
IF (1.LE.NRDPLC*LOG2R .AND. NRDPLC*LOG2R.LE.120) GO TO 40
CALL XERMSG ('SLATEC', 'DXSET', 'IMPROPER VALUE OF NRADPL', 204,
+ 1)
IERROR=204
RETURN
40 CONTINUE
NLG102 = NRDPLC*LOG2R/NB + 3
NP1 = NLG102 + 1
C
C AFTER COMPLETION OF THE FOLLOWING LOOP, IC CONTAINS
C THE INTEGER PART AND LGTEMP CONTAINS THE FRACTIONAL PART
C OF LOG10(IRADX) IN RADIX 1000.
IC = 0
DO 50 II=1,20
I = 21 - II
IT = LOG2R*LOG102(I) + IC
IC = IT/1000
LGTEMP(I) = MOD(IT,1000)
50 CONTINUE
C
C AFTER COMPLETION OF THE FOLLOWING LOOP, LG102 CONTAINS
C LOG10(IRADX) IN RADIX MLG102. THE RADIX POINT IS
C BETWEEN LG102(1) AND LG102(2).
LG102(1) = IC
DO 80 I=2,NP1
LG102X = 0
DO 70 J=1,NB
IC = 0
DO 60 KK=1,20
K = 21 - KK
IT = 2*LGTEMP(K) + IC
IC = IT/1000
LGTEMP(K) = MOD(IT,1000)
60 CONTINUE
LG102X = 2*LG102X + IC
70 CONTINUE
LG102(I) = LG102X
80 CONTINUE
C
C CHECK SPECIAL CONDITIONS REQUIRED BY SUBROUTINES...
IF (NRDPLC.LT.L) GO TO 90
CALL XERMSG ('SLATEC', 'DXSET', 'NRADPL .GE. L', 205, 1)
IERROR=205
RETURN
90 IF (6*L.LE.KMAX) GO TO 100
CALL XERMSG ('SLATEC', 'DXSET', '6*L .GT. KMAX', 206, 1)
IERROR=206
RETURN
100 CONTINUE
IFLAG = 1
RETURN
END