Problems with GAMMA functions?
Angelo Graziosi
Angelo.Graziosi@roma1.infn.it
Sat Oct 6 21:48:00 GMT 2007
I have found this on Cygwin but, perhaps it is general.
=======================================================
Usually I have the habit to build, on Cygwin, CERNLIB with GFortran (but
also G77 and G95).
CERNLIB has its own tests, between which, 89 mathematical tests.
Before that the set of GAMMA functions were added in GFortran, the build
of CERNLIB completed succeffully all tests. The last I tried was with
GFortran 4.3.0-20070823-127745.
But at least with gfc 4.3.0-20070903-128061, the CERNLIB build gives two
failure:
Test# 14 ( C302 ): *** failed ***
Test# 66 ( E408 ): *** failed ***
These tests use GAMMAs!
I have syntesized the C302 test (gfc_gamma_test_c302m.F) below.
If one build it using the (default) internal gfc GAMMAs, it fails.
If, instead, one links with the EXTERNAL GAMMAs (from a Cernlib build made
with GFortran 4.3.0 .LE. 20070823), it is completed successfully.
In this case, one have to uncomment the line
C external gamma,dgamma
(The Cygwin build of CERNLIB with GFC-4.3.0-20070727 can be found here
http://www.webalice.it/angelo.graziosi/cygwin/cernlib/Cernlib.html).
Interesting is also to compare, in the above cases, the 'fort.10' file
generated by 'gfc_gamma_test_c302m'.
Cheers,
Angelo.
gfc_gamma_test_c302m.F
===========================================================================
C
C gfortran -O0 -funroll-loops -fomit-frame-pointer -fno-second-underscore
C -fno-automatic gfc_gamma_test_c302m.F -o gfc_gamma_test_c302m
C
C
C Uncommenting the 'external' you must build in this way:
C
C gfortran -O0 -funroll-loops -fomit-frame-pointer -fno-second-underscore
C -fno-automatic gfc_gamma_test_c302m.F `cernlib mathlib`
C -o gfc_gamma_test_c302m
C
program main
implicit none
integer LIN,LOUT
COMMON/IOLUNS/LIN,LOUT
integer NTEST,NFAIL,IRC
COMMON/GTSTAT/NTEST,NFAIL,IRC
LOGICAL LTEST1
COMMON /C302LT1/LTEST1
LOUT = 10
NTEST = 0
NFAIL = 0
call c302m
end
C
C ===============================================================
C
SUBROUTINE C302M
C Program to test the MATHLIB routines GAMMA, DGAMMA and
C QGAMMA (C302)
LOGICAL LTEST, LTEST1,LTEST2
COMMON /C302LT1/LTEST1
COMMON/IOLUNS/LIN,LOUT
COMMON/GTSTAT/NTEST,NFAIL,IRC
CALL HEADER('C302',0)
LTEST=.TRUE.
LTEST1=.TRUE.
LTEST2=.TRUE.
CALL C302D
LTEST=LTEST .AND. LTEST1
IRC=ITEST('C302',LTEST)
CALL PAGEND('C302')
RETURN
END
C ---------------------------------------------------------------
SUBROUTINE C302D
IMPLICIT DOUBLE PRECISION (A-H,O-Z)
REAL GAMMA
C
C external gamma,dgamma
C
CHARACTER*6 TFUNC(2)
COMMON/IOLUNS/LIN,LOUT
COMMON/GTSTAT/NTEST,NFAIL,IRC
C
PARAMETER (HALF = 5D-1, PI=3.14159 26535 89793 24D0)
DIMENSION C(0:20)
LOGICAL LTEST1
DIMENSION TOL(2)
COMMON /C302LT1/LTEST1
C
C Set maximum error allowed for test to be considered successful
C
DATA TOL/1D-6, 5D-14/
DATA TFUNC/'GAMMA','DGAMMA'/
LTEST1=.TRUE.
C(0)=1
DO 2 N = 1,20
2 C(N)=(2*N-1)*C(N-1)/2
NF=2
DO 1000 JF=1,NF
ERRMAX= 0.0D0
RERRMAX= 0.0E0
WRITE(LOUT,'(/10X,''Test of C302 '',A)') TFUNC(JF)
WRITE(LOUT,'(/9X,''X '',7X,''Exact'',25X,''Calculated'',
+ 14X,''Rel. Error'')')
DO 1 N = 1,20
X=N+HALF
T=C(ABS(N))*SQRT(PI)
IF(JF.EQ.1) THEN
DR=GAMMA( SNGL(X) )
IF(DR .NE. 0) ER=ABS(SNGL(DR-T)/SNGL(DR) )
WRITE(LOUT,'(1X,F10.1,2E27.9,5X,E10.1)')
+ SNGL(X),SNGL(T),SNGL(DR),SNGL(ER)
ENDIF
IF(JF.EQ.2) THEN
DR=DGAMMA(X)
IF(DR .NE. 0) ER=ABS((DR-T)/DR)
WRITE(LOUT,'(1X,F10.1,2E27.18,5X,E10.1)') X,T,DR,ER
ENDIF
ERRMAX= MAX( ERRMAX,ER )
1 CONTINUE
ETOL=TOL(JF)
WRITE(LOUT,'(/''Largest relative error was'',E10.1)') ERRMAX
LTEST1=LTEST1.AND.(ERRMAX.LE.ETOL)
WRITE(LOUT,'(/''Testing error messages:'')')
IF(JF.EQ.1) DR=GAMMA(-SNGL(HALF))
IF(JF.EQ.2) DR=DGAMMA(-HALF)
1000 CONTINUE
RETURN
END
C
C ================== Aux routines ===============================
C
SUBROUTINE HEADER(CODE,MODE)
C This routine prints a page header with a testing routine name
C message.
CHARACTER*(*) CODE
COMMON/IOLUNS/LIN,LOUT
COMMON/GTSTAT/NTEST,NFAIL,IRC
NTEST=NTEST+1
IF(MODE.EQ.1) PRINT 1000, NTEST,CODE
WRITE(LOUT,1001) CODE
RETURN
1000 FORMAT(' Test#',I3,' ( ',A,' ): started')
1001 FORMAT('1',25X,30('*')/26X,'** Testing Routine ',A,3X,'**'/
+ 26X,30('*'))
END
C ---------------------------------------------------------------
SUBROUTINE PAGEND(CODE)
C This subroutine prints a page end message
CHARACTER*(*) CODE
COMMON/IOLUNS/LIN,LOUT
COMMON/GTSTAT/NTEST,NFAIL,IRC
WRITE(LOUT,1001) CODE
RETURN
1001 FORMAT(/26X,30('*')/26X,'** End of Test of ',A,3X,'**'/
+ 26X,30('*'))
END
FUNCTION ITEST(CODE,TEST)
COMMON/IOLUNS/LIN,LOUT
COMMON/GTSTAT/NTEST,NFAIL,IRC
CHARACTER*(*) CODE
LOGICAL TEST
IF(TEST) THEN
PRINT 1000,NTEST,CODE
IF (LOUT .NE. 6) WRITE(LOUT,1000) NTEST,CODE
ITEST=0
ELSE
PRINT 1001,NTEST,CODE
IF (LOUT .NE. 6) WRITE(LOUT,1001) NTEST,CODE
ITEST=1
ENDIF
NFAIL=NFAIL+ITEST
1000 FORMAT(' Test#',I3,' ( ',A,' ): completed successfully')
1001 FORMAT(' Test#',I3,' ( ',A,' ): *** failed ***')
RETURN
END
===========================================================================
More information about the Fortran
mailing list