!> @file singleton_mod.f90
!--------------------------------------------------------------------------------------------------!
! This file is part of the PALM model system.
!
! PALM is free software: you can redistribute it and/or modify it under the terms of the GNU General
! Public License as published by the Free Software Foundation, either version 3 of the License, or
! (at your option) any later version.
!
! PALM is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the
! implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
! Public License for more details.
!
! You should have received a copy of the GNU General Public License along with PALM. If not, see
! .
!
! Copyright 1997-2020 Leibniz Universitaet Hannover
!--------------------------------------------------------------------------------------------------!
!
!
! Current revisions:
! -----------------
!
!
! Former revisions:
! -----------------
! $Id: singleton_mod.f90 4591 2020-07-06 15:56:08Z Giersch $
! File re-formatted to follow the PALM coding standard
!
!
! 4182 2019-08-22 15:20:23Z scharf
! Corrected "Former revisions" section
!
! 3761 2019-02-25 15:31:42Z raasch
! Statement added to prevent compiler warning about unused variables
!
! Revision 1.1 2002/05/02 18:56:59 raasch
! Initial revision
!
!
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> Multivariate Fast Fourier Transform
!>
!> Fortran 90 Implementation of Singleton's mixed-radix algorithm, RC Singleton, Stanford Research
!> Institute, Sept. 1968.
!>
!> Adapted from fftn.c, translated from Fortran 66 to C by Mark Olesen and John Beale.
!>
!> Fourier transforms can be computed either in place, using assumed size arguments, or by generic
!> function, using assumed shape arguments.
!>
!> Public:
!>
!> fftkind kind parameter of complex arguments and function results.
!>
!> fft(array, dim, inv, stat) generic transform function
!> COMPLEX(fftkind), DIMENSION(:,...,:), INTENT(IN) :: array
!> INTEGER, DIMENSION(:), INTENT(IN), OPTIONAL:: dim
!> LOGICAL, INTENT(IN), OPTIONAL:: inv
!> INTEGER, INTENT(OUT), OPTIONAL:: stat
!>
!> fftn(array, shape, dim, inv, stat) in place transform subroutine
!> COMPLEX(fftkind), DIMENSION(*), INTENT(INOUT) :: array
!> INTEGER, DIMENSION(:), INTENT(IN) :: shape
!> INTEGER, DIMENSION(:), INTENT(IN), OPTIONAL:: dim
!> LOGICAL, INTENT(IN), OPTIONAL:: inv
!> INTEGER, INTENT(OUT), OPTIONAL:: stat
!>
!>
!> Formal Parameters:
!>
!> array The complex array to be transformed. Array can be of arbitrary rank (i.e. up to seven).
!>
!> shape With subroutine fftn, the shape of the array to be transformed has to be passed
!> separately, since fftradix - the internal transformation routine - will always treat
!> array as one dimensional. The product of elements in shape must be the number of
!> elements in array.
!> Although passing array with assumed shape would have been nicer, I prefered assumed
!> size in order to prevent the compiler from using a copy-in-copy-out mechanism. That
!> would generally be necessary with fftn passing array to fftradix and with fftn being
!> prepared for accepting non consecutive array sections. Using assumed size, it's up to
!> the user to pass an array argument, that can be addressed as continous one dimensional
!> array without copying. Otherwise, transformation will not really be performed in place.
!> On the other hand, since the rank of array and the size of shape needn't match, fftn
!> is appropriate for handling more than seven dimensions. As far as function fft is
!> concerned all this doesn't matter, because the argument will be copied anyway. Thus no
!> extra shape argument is needed for fft.
!>
!> Optional Parameters:
!>
!> dim One dimensional integer array, containing the dimensions to be transformed. Default
!> is (/1,...,N/) with N being the rank of array, i.e. complete transform. dim can
!> restrict transformation to a subset of available dimensions. Its size must not exceed
!> the rank of array or the size of shape respectivly.
!>
!> inv If .true., inverse transformation will be performed. Default is .false., i.e. forward
!> transformation.
!>
!> stat If present, a system dependent nonzero status value will be returned in stat, if
!> allocation of temporary storage failed.
!>
!>
!> Scaling:
!>
!> Transformation results will always be scaled by the square root of the product of sizes of each
!> dimension in dim. (See examples below)
!>
!>
!> Examples:
!>
!> Let A be a L*M*N three dimensional complex array. Then
!>
!> result = fft(A)
!>
!> will produce a three dimensional transform, scaled by sqrt(L*M*N), while
!>
!> call fftn(A, SHAPE(A))
!>
!> will do the same in place.
!>
!> result = fft(A, dim=(/1,3/))
!>
!> will transform with respect to the first and the third dimension, scaled by sqrt(L*N).
!>
!> result = fft(fft(A), inv=.true.)
!>
!> should (approximately) reproduce A.
!> With B having the same shape as A
!>
!> result = fft(fft(A) * CONJG(fft(B)), inv=.true.)
!>
!> will correlate A and B.
!>
!>
!> Remarks:
!>
!> Following changes have been introduced with respect to fftn.c:
!> - Complex arguments and results are of type complex, rather than real an imaginary part
!> separately.
!> - Increment parameter (magnitude of isign) has been dropped, inc is always one, direction of
!> transform is given by inv.
!> - maxf and maxp have been dropped. The amount of temporary storage needed is determined by the
!> fftradix routine. Both fftn and fft can handle any size of array. (Maybe they take a lot of
!> time and memory, but they will do it)
!>
!> Redesigning fftradix in a way, that it handles assumed shape arrays would have been desirable.
!> However, I found it rather hard to do this in an efficient way. Problems were:
!> - To prevent stride multiplications when indexing arrays. At least our compiler was not clever
!> enough to discover that in fact additions would do the job as well. On the other hand, I
!> haven't been clever enough to find an implementation using array operations.
!> - fftradix is rather large and different versions would be necessaray for each possible rank of
!> array.
!> Consequently, in place transformation still needs the argument stored in a consecutive bunch of
!> memory and can't be performed on array sections like A(100:199:-3, 50:1020). Calling fftn with
!> such sections will most probably imply copy-in-copy-out. However, the function fft works with
!> everything it gets and should be convenient to use.
!>
!> Michael Steffens, 09.12.96,
!> Restructured fftradix for better optimization. M. Steffens, 4 June 1997
!--------------------------------------------------------------------------------------------------!
MODULE singleton
USE kinds
IMPLICIT NONE
PRIVATE
PUBLIC :: fft !<
PUBLIC :: fftn !<
REAL(wp), PARAMETER :: cos72 = 0.30901699437494742_wp !<
REAL(wp), PARAMETER :: pi = 3.14159265358979323_wp !<
REAL(wp), PARAMETER :: sin60 = 0.86602540378443865_wp !<
REAL(wp), PARAMETER :: sin72 = 0.95105651629515357_wp !<
INTERFACE fft
MODULE PROCEDURE fft1d
MODULE PROCEDURE fft2d
MODULE PROCEDURE fft3d
MODULE PROCEDURE fft4d
MODULE PROCEDURE fft5d
MODULE PROCEDURE fft6d
MODULE PROCEDURE fft7d
END INTERFACE
CONTAINS
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft1d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION(SIZE(array, 1)) :: ft !<
INTEGER(iwp) :: idum !<
INTEGER(iwp) :: ishape(1) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn( ft, ishape, inv = inv, stat = stat )
!
!-- Next statement to prevent compiler warning about unused variable
IF ( PRESENT( dim ) ) idum = 1
END FUNCTION fft1d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft2d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION(SIZE(array, 1), SIZE(array, 2)) :: ft !<
INTEGER(iwp) :: ishape(2) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn( ft, ishape, dim, inv, stat )
END FUNCTION fft2d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft3d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION(SIZE(array, 1), SIZE(array, 2), SIZE(array, 3)) :: ft !<
INTEGER(iwp) :: ishape(3) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array)
CALL fftn(ft, ishape, dim, inv, stat)
END FUNCTION fft3d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft4d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:,:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION(SIZE(array, 1), SIZE(array, 2), SIZE(array, 3), SIZE(array, 4)) :: ft !<
INTEGER(iwp) :: ishape(4) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn(ft, ishape, dim, inv, stat)
END FUNCTION fft4d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft5d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:,:,:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION(SIZE(array, 1), SIZE(array, 2), SIZE(array, 3), SIZE(array, 4), SIZE(array, 5)) :: ft !<
INTEGER(iwp) :: ishape(5) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn(ft, ishape, dim, inv, stat)
END FUNCTION fft5d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft6d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:,:,:,:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION( SIZE(array, 1), SIZE(array, 2), SIZE(array, 3), SIZE(array, 4), &
SIZE(array, 5), SIZE(array, 6)) :: ft !<
INTEGER(iwp) :: ishape(6) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn(ft, ishape, dim, inv, stat)
END FUNCTION fft6d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing function description.
!--------------------------------------------------------------------------------------------------!
FUNCTION fft7d( array, dim, inv, stat ) RESULT( ft )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(:,:,:,:,:,:,:), INTENT(IN) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Function result
COMPLEX(wp), DIMENSION( SIZE(array, 1), SIZE(array, 2), SIZE(array, 3), SIZE(array, 4), &
SIZE(array, 5), SIZE(array, 6), SIZE(array, 7)) :: ft !<
INTEGER(iwp) :: ishape(7) !<
!
!-- Intrinsics used
INTRINSIC SIZE, SHAPE
ft = array
ishape = SHAPE( array )
CALL fftn(ft, ishape, dim, inv, stat)
END FUNCTION fft7d
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing subroutine description.
!--------------------------------------------------------------------------------------------------!
SUBROUTINE fftn( array, shape, dim, inv, stat )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(*), INTENT(INOUT) :: array !<
INTEGER(iwp), DIMENSION(:), INTENT(IN) :: shape !<
INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: dim !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN), OPTIONAL :: inv !<
!
!-- Local arrays
INTEGER(iwp), DIMENSION(SIZE(shape)) :: d !<
!
!-- Local scalars
LOGICAL :: inverse !<
INTEGER(iwp) :: i, ndim, ntotal !<
REAL(wp) :: scale !<
!
!-- Intrinsics used
INTRINSIC PRESENT, MIN, PRODUCT, SIZE, SQRT
!
!-- Optional parameter settings
IF ( PRESENT( inv ) ) THEN
inverse = inv
ELSE
inverse = .FALSE.
END IF
IF ( PRESENT( dim ) ) THEN
ndim = MIN( SIZE( dim ), SIZE( d ) )
d(1:ndim) = DIM( 1:ndim )
ELSE
ndim = SIZE( d )
d = (/( i, i = 1, SIZE( d ) )/)
END IF
ntotal = PRODUCT( shape )
scale = SQRT( 1.0_wp / PRODUCT( shape( d(1:ndim) ) ) )
DO i = 1, ntotal
array(i) = CMPLX( REAL( array(i) ) * scale, AIMAG( array(i) ) * scale, KIND = wp )
END DO
DO i = 1, ndim
CALL fftradix( array, ntotal, shape( d(i) ), PRODUCT( shape( 1:d(i) ) ), inverse, stat )
IF ( PRESENT( stat ) ) THEN
IF ( stat /= 0 ) RETURN
END IF
END DO
END SUBROUTINE fftn
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing subroutine description.
!--------------------------------------------------------------------------------------------------!
SUBROUTINE fftradix( array, ntotal, npass, nspan, inv, stat )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(*), INTENT(INOUT) :: array !<
INTEGER(iwp), INTENT(IN) :: ntotal, npass, nspan !<
INTEGER(iwp), INTENT(OUT), OPTIONAL :: stat !<
LOGICAL, INTENT(IN) :: inv !<
!
!-- Local arrays
COMPLEX(wp), DIMENSION(:), ALLOCATABLE :: ctmp !<
INTEGER(iwp), DIMENSION(BIT_SIZE(0)) :: factor !<
INTEGER(iwp), DIMENSION(:), ALLOCATABLE :: perm !<
REAL(wp), DIMENSION(:), ALLOCATABLE :: sine, cosine !<
!
!-- Local scalars
INTEGER(iwp) :: maxfactor, nfactor, nsquare, nperm !<
!
!-- Intrinsics used
INTRINSIC MAXVAL, MOD, PRESENT, ISHFT, BIT_SIZE, SIN, COS, CMPLX, REAL, AIMAG
IF ( npass <= 1 ) RETURN
CALL factorize( npass, factor, nfactor, nsquare )
maxfactor = MAXVAL( factor(:nfactor) )
IF ( nfactor - ISHFT( nsquare, 1 ) > 0 ) THEN
nperm = MAX( nfactor + 1, PRODUCT( factor(nsquare+1: nfactor-nsquare) ) - 1 )
ELSE
nperm = nfactor + 1
END IF
IF ( PRESENT( stat ) ) THEN
ALLOCATE( ctmp(maxfactor), sine(maxfactor), cosine(maxfactor), STAT = stat )
IF ( stat /= 0 ) RETURN
CALL transform( array, ntotal, npass, nspan, factor, nfactor, ctmp, sine, cosine, inv )
DEALLOCATE( sine, cosine, STAT = stat )
IF ( stat /= 0 ) RETURN
ALLOCATE( perm(nperm), STAT = stat )
IF ( stat /= 0 ) RETURN
CALL permute( array, ntotal, npass, nspan, factor, nfactor, nsquare, maxfactor, ctmp, perm )
DEALLOCATE( perm, ctmp, STAT = stat )
IF ( stat /= 0 ) RETURN
ELSE
ALLOCATE( ctmp(maxfactor), sine(maxfactor), cosine(maxfactor) )
CALL transform( array, ntotal, npass, nspan, factor, nfactor, ctmp, sine, cosine, inv )
DEALLOCATE( sine, cosine )
ALLOCATE( perm(nperm) )
CALL permute( array, ntotal, npass, nspan, factor, nfactor, nsquare, maxfactor, ctmp, perm )
DEALLOCATE( perm, ctmp )
END IF
CONTAINS
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing subroutine description.
!--------------------------------------------------------------------------------------------------!
SUBROUTINE factorize( npass, factor, nfactor, nsquare )
!
!-- Formal parameters
INTEGER(iwp), INTENT(IN) :: npass !<
INTEGER(iwp), INTENT(OUT) :: nfactor, nsquare !<
INTEGER(iwp), DIMENSION(*), INTENT(OUT) :: factor !<
!
!-- Local scalars
INTEGER(iwp) :: j, jj, k !<
nfactor = 0
k = npass
DO WHILE ( MOD( k, 16 ) == 0 )
nfactor = nfactor + 1
factor(nfactor) = 4
k = k / 16
END DO
j = 3
jj = 9
DO
DO WHILE ( MOD( k, jj ) == 0 )
nfactor = nfactor + 1
factor(nfactor) = j
k = k / jj
END DO
j = j + 2
jj = j * j
IF ( jj > k ) EXIT
END DO
IF ( k <= 4 ) THEN
nsquare = nfactor
factor(nfactor + 1) = k
IF ( k /= 1 ) nfactor = nfactor + 1
ELSE
IF ( k - ISHFT( k / 4, 2 ) == 0 ) THEN
nfactor = nfactor + 1
factor(nfactor) = 2
k = k / 4
END IF
nsquare = nfactor
j = 2
DO
IF ( MOD(k, j) == 0 ) THEN
nfactor = nfactor + 1
factor(nfactor) = j
k = k / j
END IF
j = ISHFT( (j + 1) / 2, 1 ) + 1
IF ( j > k ) EXIT
END DO
END IF
IF ( nsquare > 0 ) THEN
j = nsquare
DO
nfactor = nfactor + 1
factor(nfactor) = factor(j)
j = j - 1
IF ( j == 0 ) EXIT
END DO
END IF
END SUBROUTINE factorize
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing subroutine description.
!--------------------------------------------------------------------------------------------------!
SUBROUTINE transform( array, ntotal, npass, nspan, factor, nfactor, ctmp, sine, cosine, inv )
!-- Compute fourier transform
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(*), INTENT(IN OUT) :: array !<
COMPLEX(wp), DIMENSION(*), INTENT(OUT) :: ctmp !<
INTEGER(iwp), INTENT(IN) :: ntotal, npass, nspan !<
INTEGER(iwp), DIMENSION(*), INTENT(IN) :: factor !<
INTEGER(iwp), INTENT(IN) :: nfactor !<
LOGICAL, INTENT(IN) :: inv !<
REAL(wp), DIMENSION(*), INTENT(OUT) :: sine, cosine !<
!
!-- Local scalars
COMPLEX(wp) :: cc, cj, ck, cjp, cjm, ckp, ckm !<
INTEGER(iwp) :: ii, ispan !<
INTEGER(iwp) :: j, jc, jf, jj !<
INTEGER(iwp) :: k, kk, kspan, k1, k2, k3, k4 !<
INTEGER(iwp) :: nn, nt !<
REAL(wp) :: s60, c72, s72, pi2, radf !<
REAL(wp) :: c1, s1, c2, s2, c3, s3, cd, sd, ak !<
c72 = cos72
IF ( inv ) THEN
s72 = sin72
s60 = sin60
pi2 = pi
ELSE
s72 = - sin72
s60 = - sin60
pi2 = - pi
END IF
nt = ntotal
nn = nt - 1
kspan = nspan
jc = nspan / npass
radf = pi2 * jc
pi2 = pi2 * 2.0_wp !-- Use 2 PI from here on
ii = 0
jf = 0
DO
sd = radf / kspan
cd = SIN( sd )
cd = 2.0_wp * cd * cd
sd = SIN( sd + sd )
kk = 1
ii = ii + 1
SELECT CASE ( factor(ii) )
CASE ( 2 )
!
!-- Transform for factor of 2 (including rotation factor)
kspan = kspan / 2
k1 = kspan + 2
DO
DO
k2 = kk + kspan
ck = array(k2)
array(k2) = array(kk) - ck
array(kk) = array(kk) + ck
kk = k2 + kspan
IF ( kk > nn ) EXIT
END DO
kk = kk - nn
IF ( kk > jc ) EXIT
END DO
IF ( kk > kspan ) RETURN
DO
c1 = 1.0_wp - cd
s1 = sd
DO
DO
DO
k2 = kk + kspan
ck = array(kk) - array(k2)
array(kk) = array(kk) + array(k2)
array(k2) = ck * CMPLX( c1, s1, KIND = wp )
kk = k2 + kspan
IF ( kk >= nt ) EXIT
END DO
k2 = kk - nt
c1 = - c1
kk = k1 - k2
IF ( kk <= k2 ) EXIT
END DO
ak = c1 - (cd * c1 + sd * s1)
s1 = sd * c1 - cd * s1 + s1
c1 = 2.0_wp - ( ak * ak + s1 * s1 )
s1 = s1 * c1
c1 = c1 * ak
kk = kk + jc
IF ( kk >= k2 ) EXIT
END DO
k1 = k1 + 1 + 1
kk = ( k1 - kspan ) / 2 + jc
IF ( kk > jc + jc ) EXIT
END DO
!
!-- Transform for factor of 4
CASE ( 4 )
ispan = kspan
kspan = kspan / 4
DO
c1 = 1.0_wp
s1 = 0.0_wp
DO
DO
k1 = kk + kspan
k2 = k1 + kspan
k3 = k2 + kspan
ckp = array(kk) + array(k2)
ckm = array(kk) - array(k2)
cjp = array(k1) + array(k3)
cjm = array(k1) - array(k3)
array(kk) = ckp + cjp
cjp = ckp - cjp
IF ( inv ) THEN
ckp = ckm + CMPLX( - AIMAG( cjm ), REAL( cjm ), KIND = wp )
ckm = ckm + CMPLX( AIMAG( cjm ), - REAL( cjm ), KIND = wp )
ELSE
ckp = ckm + CMPLX( AIMAG( cjm ), - REAL( cjm ), KIND = wp )
ckm = ckm + CMPLX( - AIMAG( cjm ), REAL( cjm ), KIND = wp )
END IF
!
!-- Avoid useless multiplies
IF ( s1 == 0.0_wp ) THEN
array(k1) = ckp
array(k2) = cjp
array(k3) = ckm
ELSE
array(k1) = ckp * CMPLX( c1, s1, KIND = wp )
array(k2) = cjp * CMPLX( c2, s2, KIND = wp )
array(k3) = ckm * CMPLX( c3, s3, KIND = wp )
END IF
kk = k3 + kspan
IF ( kk > nt ) EXIT
END DO
c2 = c1 - ( cd * c1 + sd * s1 )
s1 = sd * c1 - cd * s1 + s1
c1 = 2.0_wp - ( c2 * c2 + s1 * s1 )
s1 = s1 * c1
c1 = c1 * c2
!
!-- Values of c2, c3, s2, s3 that will get used next time
c2 = c1 * c1 - s1 * s1
s2 = 2.0_wp * c1 * s1
c3 = c2 * c1 - s2 * s1
s3 = c2 * s1 + s2 * c1
kk = kk - nt + jc
IF ( kk > kspan ) EXIT
END DO
kk = kk - kspan + 1
IF ( kk > jc ) EXIT
END DO
IF ( kspan == jc ) RETURN
CASE default
!
!-- Transform for odd factors
k = factor(ii)
ispan = kspan
kspan = kspan / k
SELECT CASE ( k )
!
!-- Transform for factor of 3 (optional code)
CASE ( 3 )
DO
DO
k1 = kk + kspan
k2 = k1 + kspan
ck = array(kk)
cj = array(k1) + array(k2)
array(kk) = ck + cj
ck = ck - CMPLX( 0.5_wp * REAL( cj ), 0.5_wp * AIMAG( cj ), KIND = wp )
cj = CMPLX( ( REAL( array(k1) ) - REAL( array(k2) ) ) * s60, &
( AIMAG( array(k1) ) - AIMAG( array(k2) ) ) * s60, KIND = wp )
array(k1) = ck + CMPLX( - AIMAG( cj ), REAL( cj ), KIND = wp )
array(k2) = ck + CMPLX( AIMAG( cj ), - REAL( cj ), KIND = wp )
kk = k2 + kspan
IF ( kk >= nn ) EXIT
END DO
kk = kk - nn
IF ( kk > kspan ) EXIT
END DO
!
!-- Transform for factor of 5 (optional code)
CASE ( 5 )
c2 = c72 * c72 - s72 * s72
s2 = 2.0_wp * c72 * s72
DO
DO
k1 = kk + kspan
k2 = k1 + kspan
k3 = k2 + kspan
k4 = k3 + kspan
ckp = array(k1) + array(k4)
ckm = array(k1) - array(k4)
cjp = array(k2) + array(k3)
cjm = array(k2) - array(k3)
cc = array(kk)
array(kk) = cc + ckp + cjp
ck = CMPLX( REAL( ckp ) * c72, AIMAG( ckp ) * c72, KIND = wp ) + &
CMPLX( REAL( cjp ) * c2, AIMAG( cjp ) * c2, KIND = wp ) + cc
cj = CMPLX( REAL( ckm ) * s72, AIMAG( ckm ) * s72, KIND = wp) + &
CMPLX( REAL( cjm ) * s2, AIMAG( cjm ) * s2, KIND = wp )
array(k1) = ck + CMPLX( - AIMAG( cj ), REAL( cj ), KIND = wp )
array(k4) = ck + CMPLX( AIMAG( cj ), - REAL( cj ), KIND = wp )
ck = CMPLX( REAL( ckp ) * c2, AIMAG( ckp ) * c2, KIND = wp ) + &
CMPLX( REAL( cjp ) * c72, AIMAG( cjp ) * c72, KIND = wp ) + cc
cj = CMPLX( REAL( ckm ) * s2, AIMAG( ckm ) * s2, KIND = wp ) - &
CMPLX( REAL( cjm ) * s72, AIMAG( cjm ) * s72, KIND = wp )
array(k2) = ck + CMPLX( -AIMAG( cj ), REAL( cj ), KIND = wp )
array(k3) = ck + CMPLX( AIMAG( cj ), - REAL( cj ), KIND = wp )
kk = k4 + kspan
IF ( kk >= nn ) EXIT
END DO
kk = kk - nn
IF ( kk > kspan ) EXIT
END DO
CASE default
IF ( k /= jf ) THEN
jf = k
s1 = pi2 / k
c1 = COS( s1 )
s1 = SIN( s1 )
cosine(jf) = 1.0_wp
sine(jf) = 0.0_wp
j = 1
DO
cosine(j) = cosine(k) * c1 + sine(k) * s1
sine(j) = cosine(k) * s1 - sine(k) * c1
k = k - 1
cosine(k) = cosine(j)
sine(k) = - sine(j)
j = j + 1
IF ( j >= k ) EXIT
END DO
END IF
DO
DO
k1 = kk
k2 = kk + ispan
cc = array(kk)
ck = cc
j = 1
k1 = k1 + kspan
DO
k2 = k2 - kspan
j = j + 1
ctmp(j) = array(k1) + array(k2)
ck = ck + ctmp(j)
j = j + 1
ctmp(j) = array(k1) - array(k2)
k1 = k1 + kspan
IF ( k1 >= k2 ) EXIT
END DO
array(kk) = ck
k1 = kk
k2 = kk + ispan
j = 1
DO
k1 = k1 + kspan
k2 = k2 - kspan
jj = j
ck = cc
cj = ( 0.0_wp, 0.0_wp )
k = 1
DO
k = k + 1
ck = ck + CMPLX( REAL( ctmp(k) ) * cosine(jj), AIMAG( ctmp(k) ) * &
cosine(jj), KIND = wp )
k = k + 1
cj = cj + CMPLX( REAL( ctmp(k) ) * sine(jj), AIMAG( ctmp(k) ) * sine(jj), &
KIND = wp )
jj = jj + j
IF ( jj > jf ) jj = jj - jf
IF ( k >= jf ) EXIT
END DO
k = jf - j
array(k1) = ck + CMPLX( - AIMAG( cj ), REAL( cj ), KIND = wp )
array(k2) = ck + CMPLX( AIMAG( cj ), -REAL( cj ), KIND = wp )
j = j + 1
IF ( j >= k ) EXIT
END DO
kk = kk + ispan
IF ( kk > nn ) EXIT
END DO
kk = kk - nn
IF ( kk > kspan ) EXIT
END DO
END SELECT
!
!-- Multiply by rotation factor (except for factors of 2 and 4)
IF ( ii == nfactor ) RETURN
kk = jc + 1
DO
c2 = 1.0_wp - cd
s1 = sd
DO
c1 = c2
s2 = s1
kk = kk + kspan
DO
DO
array(kk) = CMPLX( c2, s2, KIND = wp ) * array(kk)
kk = kk + ispan
IF ( kk > nt ) EXIT
END DO
ak = s1 * s2
s2 = s1 * c2 + c1 * s2
c2 = c1 * c2 - ak
kk = kk - nt + kspan
IF ( kk > ispan ) EXIT
END DO
c2 = c1 - ( cd * c1 + sd * s1 )
s1 = s1 + sd * c1 - cd * s1
c1 = 2.0_wp - ( c2 * c2 + s1 * s1 )
s1 = s1 * c1
c2 = c2 * c1
kk = kk - ispan + jc
IF ( kk > kspan ) EXIT
END DO
kk = kk - kspan + jc + 1
IF ( kk > jc + jc ) EXIT
END DO
END SELECT
END DO
END SUBROUTINE transform
!--------------------------------------------------------------------------------------------------!
! Description:
! ------------
!> @todo Missing subroutine description.
!--------------------------------------------------------------------------------------------------!
SUBROUTINE permute( array, ntotal, npass, nspan, factor, nfactor, nsquare, maxfactor, ctmp, perm )
!
!-- Formal parameters
COMPLEX(wp), DIMENSION(*), INTENT(IN OUT) :: array !<
COMPLEX(wp), DIMENSION(*), INTENT(OUT) :: ctmp !<
INTEGER(iwp), INTENT(IN) :: ntotal, npass, nspan !<
INTEGER(iwp), INTENT(IN) :: nfactor, nsquare !<
INTEGER(iwp), INTENT(IN) :: maxfactor !<
INTEGER(iwp), DIMENSION(*), INTENT(IN OUT) :: factor !<
INTEGER(iwp), DIMENSION(*), INTENT(OUT) :: perm !<
!
!-- Local scalars
COMPLEX(wp) :: ck !<
INTEGER(iwp) :: ii, ispan !<
INTEGER(iwp) :: j, jc, jj !<
INTEGER(iwp) :: k, kk, kspan, kt, k1, k2, k3 !<
INTEGER(iwp) :: nn, nt !<
!
!-- Permute the results to normal order---done in two stages
!-- Permutation for square factors of n
nt = ntotal
nn = nt - 1
kt = nsquare
kspan = nspan
jc = nspan / npass
perm (1) = nspan
IF ( kt > 0 ) THEN
k = kt + kt + 1
IF ( nfactor < k ) k = k - 1
j = 1
perm(k + 1) = jc
DO
perm(j + 1) = perm(j) / factor(j)
perm(k) = perm(k + 1) * factor(j)
j = j + 1
k = k - 1
IF ( j >= k ) EXIT
END DO
k3 = perm(k + 1)
kspan = perm(2)
kk = jc + 1
k2 = kspan + 1
j = 1
IF ( npass /= ntotal ) THEN
permute_multi: DO
DO
DO
k = kk + jc
DO
!
!-- Swap array(kk) <> array(k2)
ck = array(kk)
array(kk) = array(k2)
array(k2) = ck
kk = kk + 1
k2 = k2 + 1
IF ( kk >= k ) EXIT
END DO
kk = kk + nspan - jc
k2 = k2 + nspan - jc
IF ( kk >= nt ) EXIT
END DO
kk = kk - nt + jc
k2 = k2 - nt + kspan
IF ( k2 >= nspan ) EXIT
END DO
DO
DO
k2 = k2 - perm(j)
j = j + 1
k2 = perm(j + 1) + k2
IF ( k2 <= perm(j) ) EXIT
END DO
j = 1
DO
IF ( kk < k2 ) CYCLE permute_multi
kk = kk + jc
k2 = k2 + kspan
IF ( k2 >= nspan ) EXIT
END DO
IF ( kk >= nspan ) EXIT
END DO
EXIT
END DO permute_multi
ELSE
permute_single: DO
DO
!
!-- Swap array(kk) <> array(k2)
ck = array(kk)
array(kk) = array(k2)
array(k2) = ck
kk = kk + 1
k2 = k2 + kspan
IF ( k2 >= nspan ) EXIT
END DO
DO
DO
k2 = k2 - perm(j)
j = j + 1
k2 = perm(j + 1) + k2
IF ( k2 <= perm(j) ) EXIT
END DO
j = 1
DO
IF ( kk < k2 ) CYCLE permute_single
kk = kk + 1
k2 = k2 + kspan
IF ( k2 >= nspan ) EXIT
END DO
IF ( kk >= nspan ) EXIT
END DO
EXIT
END DO permute_single
END IF
jc = k3
END IF
IF ( ISHFT( kt, 1 ) + 1 >= nfactor ) RETURN
ispan = perm(kt + 1)
!
!-- Permutation for square-free factors of n
j = nfactor - kt
factor( j + 1 ) = 1
DO
factor(j) = factor(j) * factor(j+1)
j = j - 1
IF ( j == kt ) EXIT
END DO
kt = kt + 1
nn = factor( kt ) - 1
j = 0
jj = 0
DO
k = kt + 1
k2 = factor(kt)
kk = factor(k)
j = j + 1
IF ( j > nn ) EXIT !-- Exit infinite loop
jj = jj + kk
DO WHILE ( jj >= k2 )
jj = jj - k2
k2 = kk
k = k + 1
kk = factor(k)
jj = jj + kk
END DO
perm(j) = jj
END DO
!
!-- Determine the permutation cycles of length greater than 1
j = 0
DO
DO
j = j + 1
kk = perm(j)
IF ( kk >= 0 ) EXIT
END DO
IF ( kk /= j ) THEN
DO
k = kk
kk = perm(k)
perm(k) = - kk
IF ( kk == j ) EXIT
END DO
k3 = kk
ELSE
perm(j) = - j
IF ( j == nn ) EXIT !-- Exit infinite loop
END IF
END DO
!
!-- Reorder a and b, following the permutation cycles
DO
j = k3 + 1
nt = nt - ispan
ii = nt - 1 + 1
IF ( nt < 0 ) EXIT !-- Exit infinite loop
DO
DO
j = j - 1
IF ( perm(j) >= 0 ) EXIT
END DO
jj = jc
DO
kspan = jj
IF ( jj > maxfactor ) kspan = maxfactor
jj = jj - kspan
k = perm(j)
kk = jc * k + ii + jj
k1 = kk + kspan
k2 = 0
DO
k2 = k2 + 1
ctmp(k2) = array(k1)
k1 = k1 - 1
IF ( k1 == kk ) EXIT
END DO
DO
k1 = kk + kspan
k2 = k1 - jc * ( k + perm(k) )
k = - perm(k)
DO
array(k1) = array(k2)
k1 = k1 - 1
k2 = k2 - 1
IF ( k1 == kk ) EXIT
END DO
kk = k2
IF ( k == j ) EXIT
END DO
k1 = kk + kspan
k2 = 0
DO
k2 = k2 + 1
array(k1) = ctmp(k2)
k1 = k1 - 1
IF ( k1 == kk ) EXIT
END DO
IF ( jj == 0 ) EXIT
END DO
IF ( j == 1 ) EXIT
END DO
END DO
END SUBROUTINE permute
END SUBROUTINE fftradix
END MODULE singleton