forked from ESCOMP/CARMA_base
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathvertical.F90
138 lines (113 loc) · 4.95 KB
/
vertical.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
! Include shortname defintions, so that the F77 code does not have to be modified to
! reference the CARMA structure.
#include "carma_globaer.h"
!! This routine drives the vertical transport calculations.
!!
!! NOTE: Since this is only for sedimentation and brownian diffusion of a column within
!! a parent model, the advection of air density, gases and potential temperature have
!! been removed. Also, the divergence corrections (divcor) for 1D transport are not
!! applied, since these columns exist within a parent model that is responsible for the
!! advection.
!!
!! @author Eric Jensen
!! version Mar-1995
subroutine vertical(carma, cstate, rc)
! types
use carma_precision_mod
use carma_enums_mod
use carma_constants_mod
use carma_types_mod
use carmastate_mod
use carma_mod
implicit none
type(carma_type), intent(in) :: carma !! the carma object
type(carmastate_type), intent(inout) :: cstate !! the carma state object
integer, intent(inout) :: rc !! return code, negative indicates failure
! Declare local variables
integer :: ielem
integer :: ibin
integer :: ig
real(kind=f) :: vertadvu(NZP1)
real(kind=f) :: vertadvd(NZP1)
real(kind=f) :: vertdifu(NZP1)
real(kind=f) :: vertdifd(NZP1)
real(kind=f) :: vtrans(NZP1)
real(kind=f) :: old_pc(NZ, NBIN, NELEM)
rc = RC_OK
! Before doing advection, make sure that there are no negative values for
! the concentration element.
call fixcorecol(carma, cstate, rc)
if (rc < RC_OK) return
old_pc(:,:,:) = pc(:,:,:)
sedimentationflux(:,:) = 0._f
do ielem = 1,NELEM ! Loop over particle elements
ig = igelem(ielem) ! particle group
! Should this group participate in sedimentation?
if (grp_do_vtran(ig)) then
! Are there enough particles in the column to bother?
if (maxval(pconmax(:,ig)) .gt. FEW_PC) then
do ibin = 1,NBIN ! Loop over particle mass bins
vtrans(:) = -vf(:,ibin,ig)
! If dry deposition is enabled for this group, then set
! the deposition velocity at the surface.
if (grp_do_drydep(ig)) then
if (igridv .eq. I_CART) then
vtrans(1) = -vd(ibin, ig)
else
vtrans(NZP1) = -vd(ibin, ig)
end if
end if
! Calculate particle transport rates due to vertical advection
! and vertical diffusion, and solve for concentrations at end of time step.
call vertadv(carma, cstate, vtrans, pc(:,ibin,ielem), itbnd_pc, ibbnd_pc, &
pc_topbnd(ibin,ielem), pc_botbnd(ibin,ielem), vertadvu, vertadvd, rc)
if (rc < RC_OK) return
call vertdif(carma, cstate, ig, ibin, itbnd_pc, ibbnd_pc, vertdifu, vertdifd, rc)
if (rc < RC_OK) return
! There are 2 different solvers, versol with uses a PPM scheme and versub
! which using an explicit substepping approach.
if (do_explised) then
call versub(carma, cstate, pconmax(:,ig)*zmet(:), pc(:,ibin,ielem), itbnd_pc, ibbnd_pc, &
ftoppart(ibin,ielem), fbotpart(ibin,ielem), &
pc_topbnd(ibin,ielem), pc_botbnd(ibin,ielem), &
vertadvu, vertadvd, vertdifu, vertdifd, rc)
if (rc < RC_OK) return
else
call versol(carma, cstate, pc(:,ibin,ielem), itbnd_pc, ibbnd_pc, &
ftoppart(ibin,ielem), fbotpart(ibin,ielem), &
pc_topbnd(ibin,ielem), pc_botbnd(ibin,ielem), &
vertadvu, vertadvd, vertdifu, vertdifd, rc)
if (rc < RC_OK) return
end if
end do
endif
endif
enddo ! ielem
! Advection can cause errors in tracer/tracer relationship that can cause
! negative values for the concentration element. Use a mass conserving
! fixer to make sure there are no negative values.
call fixcorecol(carma, cstate, rc)
if (rc < RC_OK) return
! Now the the column has been fixed up, look to see how much mass
! has been lost to the surface.
do ielem = 1,NELEM ! Loop over particle elements
ig = igelem(ielem) ! particle group
! Should this group participate in sedimentation?
if (grp_do_vtran(ig)) then
do ibin = 1,NBIN ! Loop over particle mass bins
! A clunky way to get the mass flux to the surface and to conserve mass
! is to determine the total before and after. Anything lost went to the
! surface.
!
! NOTE: This only works if you assume nothing is lost out the top. It would be
! better to figure out how to get this directly from versol.
pc_surf(ibin,ielem) = pc_surf(ibin, ielem) + sum(old_pc(:,ibin,ielem) * dz(:) ) - &
sum(pc(:,ibin,ielem) * dz(:) )
sedimentationflux(ibin,ielem) = ( sum(old_pc(:,ibin,ielem) * dz(:) ) - &
sum(pc(:,ibin,ielem) * dz(:) ) ) / dtime
enddo ! ibin
end if
end do
! Return to caller with new particle concentrations.
return
end