-
Notifications
You must be signed in to change notification settings - Fork 69
/
Copy pathforcing.f90
246 lines (209 loc) · 6.52 KB
/
forcing.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
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
!!
!! Copyright (C) 2009-2017 Johns Hopkins University
!!
!! This file is part of lesgo.
!!
!! lesgo 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.
!!
!! lesgo 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 lesgo. If not, see <http://www.gnu.org/licenses/>.
!!
!*******************************************************************************
module forcing
!*******************************************************************************
!
! Provides subroutines and functions for computing forcing terms on the
! velocity field. Provides driver routine for IBM forces
! (forcing_induced), driver routine for RNS, turbine, etc. forcing
! (forcing_applied), and for the projection step. Also included are
! routines for enforcing a uniform inflow and the fringe region
! treatment.
!
#ifdef PPHIT
use hit_inflow, only : inflow_HIT
#endif
implicit none
save
private
public :: forcing_random, forcing_applied, forcing_induced, project
contains
!*******************************************************************************
subroutine forcing_random()
!*******************************************************************************
!
! This subroutine generates a random body force that is helpful to
! trigger transition at low Re DNS. The forces are applied to RHS in
! evaluation of u* (not at wall) so that mass conservation is preserved.
!
use types, only : rprec
use param, only : nx,ny,nz,rms_random_force
use sim_param, only : RHSy, RHSz
real(rprec) :: dummy_rand
integer :: jx,jy,jz
! Note: the "default" rms of a unif variable is 0.289
call init_random_seed
do jz = 2, nz-1 ! don't force too close to the wall
do jy = 1, ny
do jx = 1, nx
call random_number(dummy_rand)
RHSy(jx,jy,jz) = RHSy(jx,jy,jz) + &
(rms_random_force/.289_rprec)*(dummy_rand-.5_rprec)
call random_number(dummy_rand)
RHSz(jx,jy,jz) = RHSz(jx,jy,jz) + &
(rms_random_force/.289_rprec)*(dummy_rand-.5_rprec)
end do
end do
end do
end subroutine forcing_random
!*******************************************************************************
subroutine forcing_applied()
!*******************************************************************************
!
! This subroutine acts as a driver for applying pointwise body forces
! into the domain. Subroutines contained here should modify f{x,y,z}a
! which are explicitly applied forces. These forces are applied to RHS
! in the evaluation of u* so that mass conservation is preserved.
!
use types, only : rprec
#ifdef PPTURBINES
use sim_param, only : fxa, fya, fza
use turbines, only:turbines_forcing
#endif
#ifdef PPATM
use sim_param, only : fxa, fya, fza ! The body force components
use atm_lesgo_interface, only : atm_lesgo_forcing
#endif
implicit none
#ifdef PPTURBINES
! Reset applied force arrays
fxa = 0._rprec
fya = 0._rprec
fza = 0._rprec
call turbines_forcing ()
#endif
#ifdef PPATM
fxa = 0._rprec
fya = 0._rprec
fza = 0._rprec
call atm_lesgo_forcing ()
#endif
end subroutine forcing_applied
!*******************************************************************************
subroutine forcing_induced()
!*******************************************************************************
!
! These forces are designated as induced forces such that they are
! chosen to obtain a desired velocity at time
! step m+1. If this is not the case, care should be taken so that the forces
! here are divergence free in order to preserve mass conservation. For
! non-induced forces such as explicitly applied forces they should be
! placed in forcing_applied.
!
use types, only : rprec
#ifdef PPLVLSET
use level_set, only : level_set_forcing
use sim_param, only : fx, fy, fz
#endif
implicit none
#ifdef PPLVLSET
! Initialize
fx = 0._rprec
fy = 0._rprec
fz = 0._rprec
! Compute the level set IBM forces
call level_set_forcing ()
#endif
end subroutine forcing_induced
!*******************************************************************************
subroutine project ()
!*******************************************************************************
!
! provides u, v, w at 1:nz
!
use param
use sim_param
use messages
use inflow, only : apply_inflow
#ifdef PPMPI
use mpi_defs, only : mpi_sync_real_array, MPI_SYNC_DOWNUP
#endif
implicit none
integer :: jx, jy, jz
integer :: jz_min
real(rprec) :: RHS, tconst
! Caching
tconst = tadv1 * dt
do jz = 1, nz - 1
do jy = 1, ny
do jx = 1, nx
#ifdef PPLVLSET
RHS = -tadv1 * dpdx(jx, jy, jz)
u(jx, jy, jz) = (u(jx, jy, jz) + dt * (RHS + fx(jx, jy, jz)))
RHS = -tadv1 * dpdy(jx, jy, jz)
v(jx, jy, jz) = (v(jx, jy, jz) + dt * (RHS + fy(jx, jy, jz)))
#else
RHS = -tadv1 * dpdx(jx, jy, jz)
u(jx, jy, jz) = (u(jx, jy, jz) + dt * (RHS ))
RHS = -tadv1 * dpdy(jx, jy, jz)
v(jx, jy, jz) = (v(jx, jy, jz) + dt * (RHS ))
#endif
end do
end do
end do
if (coord == 0) then
jz_min = 2
else
jz_min = 1
end if
do jz = jz_min, nz - 1
do jy = 1, ny
do jx = 1, nx
#ifdef PPLVLSET
RHS = -tadv1 * dpdz(jx, jy, jz)
w(jx, jy, jz) = (w(jx, jy, jz) + dt * (RHS + fz(jx, jy, jz)))
#else
RHS = -tadv1 * dpdz(jx, jy, jz)
w(jx, jy, jz) = (w(jx, jy, jz) + dt * (RHS ))
#endif
end do
end do
end do
call apply_inflow()
!--left this stuff last, so BCs are still enforced, no matter what
! inflow_cond does
#ifdef PPMPI
! Exchange ghost node information (since coords overlap)
call mpi_sync_real_array( u, 0, MPI_SYNC_DOWNUP )
call mpi_sync_real_array( v, 0, MPI_SYNC_DOWNUP )
call mpi_sync_real_array( w, 0, MPI_SYNC_DOWNUP )
#endif
!--enfore bc at top
#ifdef PPMPI
if (coord == nproc-1) then
#endif
! Note: for ubc_mom > 0, u and v and nz will be written to output as BOGUS
if (ubc_mom == 0) then ! no-stress top
u(:,:,nz) = u(:,:,nz-1)
v(:,:,nz) = v(:,:,nz-1)
endif
! no permeability
w(:, :, nz)=0._rprec
#ifdef PPMPI
endif
#endif
if (coord == 0) then
! No modulation of u and v since if a stress free condition (lbc_mom=0) is
! applied, it is applied through the momentum equation.
! no permeability
w(:, :, 1)=0._rprec
end if
end subroutine project
end module forcing