forked from fukipa/afid-channel
-
Notifications
You must be signed in to change notification settings - Fork 0
/
fft_common.F90
193 lines (139 loc) · 5.48 KB
/
fft_common.F90
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
!=======================================================================
! This is part of the 2DECOMP&FFT library
!
! 2DECOMP&FFT is a software framework for general-purpose 2D (pencil)
! decomposition. It also implements a highly scalable distributed
! three-dimensional Fast Fourier Transform (FFT).
!
! Copyright (C) 2009-2012 Ning Li, the Numerical Algorithms Group (NAG)
!
!=======================================================================
! This file contains common code shared by all FFT engines
integer, parameter, public :: DECOMP_2D_FFT_FORWARD = -1
integer, parameter, public :: DECOMP_2D_FFT_BACKWARD = 1
! Physical space data can be stored in either X-pencil or Z-pencil
integer, parameter, public :: PHYSICAL_IN_X = 1
integer, parameter, public :: PHYSICAL_IN_Z = 3
integer, save :: format ! input X-pencil or Z-pencil
! The libary can only be initialised once
logical, save :: initialised = .false.
! Global size of the FFT
integer, save :: nx_fft, ny_fft, nz_fft
! 2D processor grid
integer, save, dimension(2) :: dims
! Decomposition objects
TYPE(DECOMP_INFO), save, public :: ph ! physical space
TYPE(DECOMP_INFO), save, public :: sp ! spectral space
! Workspace to store the intermediate Y-pencil data
! *** TODO: investigate how to use only one workspace array
complex(mytype), allocatable, dimension(:,:,:) :: wk2_c2c, wk2_r2c
complex(mytype), allocatable, dimension(:,:,:) :: wk13
public :: decomp_2d_fft_init, decomp_2d_fft_finalize, decomp_2d_fft_get_size
public :: ffty,fftz
! Declare generic interfaces to handle different inputs
interface decomp_2d_fft_init
module procedure fft_init_noarg
module procedure fft_init_arg
module procedure fft_init_general
end interface
interface ffty
module procedure r2c_1m_y
module procedure c2c_1m_y
end interface
interface fftz
module procedure c2r_1m_z
module procedure c2c_1m_z
end interface
contains
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
! Initialise the FFT module
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
subroutine fft_init_noarg
implicit none
call fft_init_arg(PHYSICAL_IN_X) ! default input is X-pencil data
return
end subroutine fft_init_noarg
subroutine fft_init_arg(pencil) ! allow to handle Z-pencil input
implicit none
integer, intent(IN) :: pencil
call fft_init_general(pencil, nx_global, ny_global, nz_global)
return
end subroutine fft_init_arg
! Initialise the FFT library to perform arbitrary size transforms
subroutine fft_init_general(pencil, nx, ny, nz)
implicit none
integer, intent(IN) :: pencil
integer, intent(IN) :: nx, ny, nz
logical, dimension(2) :: dummy_periods
integer, dimension(2) :: dummy_coords
integer :: status, errorcode, ierror
if (initialised) then
errorcode = 4
call decomp_2d_abort(errorcode, &
'FFT library should only be initialised once')
end if
format = pencil
nx_fft = nx
ny_fft = ny
nz_fft = nz
! determine the processor grid in use
call MPI_CART_GET(DECOMP_2D_COMM_CART_X, 2, &
dims, dummy_periods, dummy_coords, ierror)
! for c2r/r2c interface:
! if in physical space, a real array is of size: nx*ny*nz
! in spectral space, the complex array is of size:
! (nx/2+1)*ny*nz, if PHYSICAL_IN_X
! or nx*ny*(nz/2+1), if PHYSICAL_IN_Z
call decomp_info_init(nx, ny, nz, ph)
if (format==PHYSICAL_IN_X) then
call decomp_info_init(nx, ny/2+1, nz, sp)
else if (format==PHYSICAL_IN_Z) then
call decomp_info_init(nx, ny, nz/2+1, sp)
end if
allocate(wk2_c2c(ph%ysz(1),ph%ysz(2),ph%ysz(3)), STAT=status)
allocate(wk2_r2c(sp%ysz(1),sp%ysz(2),sp%ysz(3)), STAT=status)
if (format==PHYSICAL_IN_X) then
allocate(wk13(sp%xsz(1),sp%xsz(2),sp%xsz(3)), STAT=status)
else if (format==PHYSICAL_IN_Z) then
allocate(wk13(sp%zsz(1),sp%zsz(2),sp%zsz(3)), STAT=status)
end if
if (status /= 0) then
errorcode = 3
call decomp_2d_abort(errorcode, &
'Out of memory when initialising FFT')
end if
call init_fft_engine
initialised = .true.
return
end subroutine fft_init_general
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
! Final clean up
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
subroutine decomp_2d_fft_finalize
implicit none
call decomp_info_finalize(ph)
call decomp_info_finalize(sp)
deallocate(wk2_c2c, wk2_r2c, wk13)
call finalize_fft_engine
initialised = .false.
return
end subroutine decomp_2d_fft_finalize
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
! Return the size, starting/ending index of the distributed array
! whose global size is (nx/2+1)*ny*nz, for defining data structures
! in r2c and c2r interfaces
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
subroutine decomp_2d_fft_get_size(istart, iend, isize)
implicit none
integer, dimension(3), intent(OUT) :: istart, iend, isize
if (format==PHYSICAL_IN_X) then
istart = sp%zst
iend = sp%zen
isize = sp%zsz
else if (format==PHYSICAL_IN_Z) then
istart = sp%xst
iend = sp%xen
isize = sp%xsz
end if
return
end subroutine decomp_2d_fft_get_size