!Functions for single complex !Extract the unit triangular matrix from the return values !of xSKTRF, xSKTF2 SUBROUTINE extract_unit_tri_c(UPLO, A, B) IMPLICIT NONE CHARACTER(LEN=1) :: UPLO COMPLEX(KIND(1.0E0)) :: A(:,:), B(:,:) INTEGER :: J,N N=SIZE(A,1) B=0D0 IF( UPLO == 'U' ) THEN DO J=1, N B(J,J)=1D0 IF( J > 2 ) B(1:J-2, J-1)=A(1:J-2, J) END DO ELSE DO J=1, N B(J,J)=1D0 IF( J 0 ) THEN IF( UPLO == 'U') THEN DO J=2, N T(J-1,J) = A(KD,J) T(J,J-1) = -A(KD,J) END DO ELSE DO J=2, N T(J,J-1) = A(2,J-1) T(J-1,J) = -A(2,J-1) END DO END IF END IF END SUBROUTINE extract_trid_band_c !make a random skew-symmetric matrix SUBROUTINE make_skew_mat_c(A, PROB) IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: A (:,:) REAL(KIND(1.0E0)), OPTIONAL :: PROB REAL(KIND(1.0E0)) :: P INTEGER :: N, I, J P = 2.0 IF( PRESENT(PROB) ) P = PROB N=SIZE(A,2) A=0D0 DO I=1, N DO J=1, I-1 IF(RAND(0) <= P) THEN A(I,J)=CMPLX(RAND(0)*2-1,RAND(0)*2-1) A(J,I)=-A(I,J) END IF END DO END DO END SUBROUTINE make_skew_mat_c SUBROUTINE make_skew_mat_banded_c(Au, Ad, A, PROB) IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: Au(:,:), Ad(:,:), A(:,:) REAL(KIND(1.0E0)), OPTIONAL :: PROB REAL(KIND(1.0E0)) :: P INTEGER :: N, I, J, KD P = 2.0 IF( PRESENT(PROB) ) P = PROB N=SIZE(Au,2) KD=SIZE(Au,1) !Assume without further checks that Ad, A has the same sizes Au=0D0 Ad=0D0 A=0D0 DO I=1,N DO J=1, I-1 IF( I-J < KD ) THEN IF(RAND(0) <= P) THEN A(I,J)=CMPLX(RAND(0)*2-1, RAND(0)*2-1) A(J,I)=-A(I,J) Au(KD+J-I,I) = A(J,I) Ad(1+I-J,J) = A(I,J) END IF END IF END DO END DO END SUBROUTINE make_skew_mat_banded_c !compute the norm of a matrix REAL(KIND(1.0E0)) FUNCTION one_norm_c(A) IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: A (:,:) INTEGER :: N REAL(KIND(1.0E0)) :: DUMMY, CLANGE EXTERNAL CLANGE N = SIZE(A,2) one_norm_c = CLANGE( "1", N, N, A, N, DUMMY) RETURN END FUNCTION one_norm_c !compute the relative residual of the difference of matrices REAL(KIND(1.0E0)) FUNCTION resid_c(A,B) USE matrix_tools, ONLY: one_norm IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: A (:,:), B(:,:) INTEGER :: N REAL(KIND(1.0E0)) :: normdiff, normb, SLAMCH EXTERNAL SLAMCH N = SIZE(A,2) normb = one_norm(B) normdiff = one_norm(A-B) IF( N > 0 .AND. normb > 0 ) THEN resid_c = normdiff/(N*normb*SLAMCH("E")) RETURN ELSE resid_c = one_norm(A-B) RETURN END IF END FUNCTION resid_c !print a matrix on the screen SUBROUTINE print_mat_c(A) IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: A (:,:) INTEGER :: N,M,I,J N=SIZE(A,2) M=SIZE(A,1) DO I=1, M DO J=1, N WRITE (*,'(F6.3)',Advance='NO') A(I,J) WRITE (*,'(A)',Advance='NO') " " END DO WRITE (*,*) END DO WRITE (*,*) END SUBROUTINE print_mat_c SUBROUTINE rowcol_invperm_c(UPLO, A, IPIV) IMPLICIT NONE CHARACTER(LEN=1) :: UPLO COMPLEX(KIND(1.0E0)) :: A(:,:) INTEGER :: IPIV(:) INTEGER :: N, I, N1, N2, DN COMPLEX(KIND(1.0E0)), ALLOCATABLE :: TEMP(:) N=SIZE(A,2) ALLOCATE(TEMP(N)) IF( UPLO == 'U' ) THEN N1 = 1 N2 = N DN = 1 ELSE N1 = N N2 = 1 DN = -1 END IF !First switch columns DO I=N1, N2, DN IF( IPIV(I) /= I ) THEN TEMP=A(1:N,I) A(1:N,I)=A(1:N,IPIV(I)) A(1:N,IPIV(I))=TEMP END IF END DO !then switch rows DO I=N1, N2, DN IF( IPIV(I) /= I ) THEN TEMP=A(I, 1:N) A(I,1:N)=A(IPIV(I),1:N) A(IPIV(I),1:N)=TEMP END IF END DO END SUBROUTINE rowcol_invperm_c SUBROUTINE reference_pfaff_c(A, PFAFF) USE F95_PFAPACK IMPLICIT NONE COMPLEX(KIND(1.0E0)) :: A (:,:), PFAFF INTEGER :: N, ISTAT, I INTEGER, ALLOCATABLE :: IPIV(:) COMPLEX(KIND(1.0E0)), ALLOCATABLE :: Ac (:,:) N = SIZE(A,2) IF( MOD(N,2) == 1 ) THEN PFAFF = 0.0D0 ELSE IF( N == 0) THEN PFAFF = 1.0D0 ELSE IF( N == 2) THEN PFAFF = A(1,2) ELSE IF( N == 4 ) THEN PFAFF = A(1,2)*A(3,4) - A(1,3)*A(2,4) + A(2,3)*A(1,4) ELSE !Use the (full) LTL decomposition which has been tested before ALLOCATE(IPIV(N), Ac(N,N), STAT=ISTAT) IF( ISTAT /=0 ) THEN WRITE (*,*) "Ran out of memory in test!" STOP END IF Ac=A CALL SKTRF(Ac, IPIV, UPLO='U') PFAFF = (1.0D0, 0.0D0) DO I=1, N IF( IPIV(I) /= I ) PFAFF = -PFAFF IF( MOD(I,2) == 1) then PFAFF = PFAFF*Ac(I,I+1) end IF END DO END IF END SUBROUTINE reference_pfaff_c !compute the relative (or absolute) error of two Pfaffians REAL(KIND(1.0E0)) FUNCTION resid_pfaff_c(N, A,B) IMPLICIT NONE INTEGER :: N COMPLEX(KIND(1.0E0)) :: A, B REAL(KIND(1.0E0)) :: SLAMCH EXTERNAL SLAMCH !Take either the relative or absolute error, depending on what's smaller IF( ABS(B) > 0.0E0) THEN resid_pfaff_c = MIN( ABS(A-B)/(MAX(N,1)*ABS(B)*SLAMCH("E")), & ABS(A-B)/(MAX(N,1)*SLAMCH("E")) ) ELSE resid_pfaff_c = ABS(A-B)/(MAX(N,1)*SLAMCH("E")) END IF END FUNCTION resid_pfaff_c REAL(KIND(1.0E0)) FUNCTION resid_pfaff10_c(N, A,B) IMPLICIT NONE INTEGER :: N COMPLEX(KIND(1.0E0)) :: A(2), B REAL(KIND(1.0E0)) :: SLAMCH EXTERNAL SLAMCH !Take either the relative or absolute error, depending on what's smaller IF( ABS(B) > 0.0E0 ) THEN resid_pfaff10_c = & MIN( ABS(A(1)*10**A(2)-B)/(MAX(N,1)*ABS(B)*SLAMCH("E")), & ABS(A(1)*10**A(2)-B)/(MAX(N,1)*SLAMCH("E")) ) ELSE resid_pfaff10_c = ABS(A(1)*10**A(2)-B)/(MAX(N,1)*SLAMCH("E")) END IF END FUNCTION resid_pfaff10_c