forked from kthyng/tracpy
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathloop_pos.f95
231 lines (204 loc) · 10.8 KB
/
loop_pos.f95
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
subroutine pos(ia,ja,ka,ib,jb,kb,x0,y0,z0,x1,y1,z1,ds,dse,dsw,dsn,dss,dsu,dsd,dsmin,dsc,&
&ff,imt,jmt,km,rr,rb,uflux,vflux,wflux,do3d,doturb,upr)
!====================================================================
! calculate the new positions of the trajectory with pos_orgn()
!
! Input:
!
! ia,ja,ka : original position in grid space indices
! x0,y0,z0 : original non-dimensional position in the i,j,k-direction
! of particle (fractions of a grid box side in the
! corresponding direction)
! ds : crossing time to reach the grid box wall (units=s/m3)
! dse,dsw : crossing times for drifter to reach the east and west grid box wall
! dsn,dss : crossing times for drifter to reach the north and south grid box wall
! dsu,dsd : crossing times for drifter to reach the up and down grid box wall
! dsmin : time step based on the interpolation step between model output times.
! : sets a limit on the time step that a drifter can go.
! dsc : Not sure what this is right now
! ff : time direction. ff=1 forward, ff=-1 backward
! imt,jmt,km : grid index sizing constants in (x,y,z), are for
! horizontal and vertical rho grid [scalar]
! rr : time interpolation constant between 0 and 1. Controls how much
! : of earlier time step is used in interpolation (for original time
! at the beginning of the loop?).
! rb : time interpolation constant between 0 and 1. Controls how much
! : of earlier time step is used in interpolation (for next time
! at the end of the loop?).
! uflux : u velocity (zonal) flux field, two time steps [ixjxkxt]
! vflux : v velocity (meridional) flux field, two time steps [ixjxkxt]
! wflux : w velocity (vertical) flux field, two time steps [kxt]
! do3d : Flag to set whether to use 3d velocities or not
! doturb : Flag to set whether or not to use turb/diff and which kind if so
! upr : parameterized turbulent velocities u', v', and w'
! optional because only used if using turb flag for diffusion
! size [6,2]. The 2nd dimension is for two time steps.
! The 1st dimension is: [u'_ia,u'_ia-1,v'_ja,v'_ja-1,w'_ka,w'_ka-1]
!
! Output:
!
! ib,jb,kb : new position in grid space indices
! x1,y1,z1 : updated non-dimensional position in the i,j,k-direction
! of particle (fractions of a grid box side in the
! corresponding direction)
!
! Other parameters used in function:
! rbg : rbg=1-rg for time interpolation between time steps. Controls how much
! : of later time step is used in interpolation.
! uu : time-interpolated flux at ia/ja/ka (depending on ijk)
! nsm=1,nsp=2 : Time index. nsm picks out the earlier bounding time step and
! nsp picks out the later bounding time step for interpolation.
! iam : generic index for grid index -1 for whichever direction, ijk.
! Is only used in the i direction for whatever reason.
!====================================================================
implicit none
integer, intent(in) :: ia, ja, ka, imt, jmt, km,ff
integer, intent(in) :: do3d, doturb
real*8, intent(in) :: x0, y0, z0,ds,dse,dsw,dss,dsn,dsd,dsu,dsmin,dsc,rb, rr
real(kind=8), intent(in), dimension(imt-1,jmt,km,2) :: uflux
real(kind=8), intent(in), dimension(imt,jmt-1,km,2) :: vflux
real(kind=8), intent(in), dimension(0:km,2) :: wflux
real*8, optional, intent(in), dimension(6,2) :: upr
integer, intent(out) :: ib, jb, kb
real*8, intent(out) :: x1, y1, z1
integer :: nsm=1,nsp=2,iam
real(kind=8) :: uu, rbg
real(kind=8), parameter :: UNDEF=1.d20
iam=ia-1
rbg =1.d0-rb
! === calculate the new positions ===
! === of the trajectory ===
! scrivi=.false.
if(ds==dse) then ! eastward grid-cell exit
! scrivi=.false. ! flag for when to write to file I think
uu=(rbg*uflux(ia,ja,ka,nsp)+rb*uflux(ia ,ja,ka,nsm))*ff
! if the drifter is exiting east and the east transport is positive,
! bump the east index up by one to keep it greater than x
! and change the x1 value to be the value at the west side of the new grid cell
if(uu.gt.0.d0) then
ib=ia+1
! KMT: This seems to be like a periodic bc. I don't know what this is for
! or why it is here, or what should replace it. The way I have the grid set up,
! the x direction grid for u stops at imt-1 also.
! What should the boundary condition be if the drifter moves outside the boundary?
! Is this where it should be enforced?
! if(ib.gt.imt) ib=ib-imt ! imt is a grid parameter
endif
x1=dble(ia)
if(doturb==1) then
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr)
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr)
else
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb)
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb)
endif
else if(ds==dsw) then ! westward grid-cell exit
! scrivi=.false.
uu=(rbg*uflux(iam,ja,ka,nsp)+rb*uflux(iam,ja,ka,nsm))*ff
if(uu.lt.0.d0) then
ib=iam
endif
x1=dble(iam)
if(doturb==1) then
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! meridional position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! vertical position
else
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! meridional position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! vertical position
endif
! scrivi=.true.
else if(ds==dsn) then ! northward grid-cell exit
! scrivi=.false.
uu=(rbg*vflux(ia,ja,ka,nsp)+rb*vflux(ia,ja,ka,nsm))*ff
if(uu.gt.0.d0) then
jb=ja+1
endif
y1=dble(ja)
if(doturb==1) then
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! zonal position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! vertical position
else
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! zonal position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! vertical position
endif
else if(ds==dss) then ! southward grid-cell exit
! scrivi=.false.
uu=(rbg*vflux(ia,ja-1,ka,nsp)+rb*vflux(ia,ja-1,ka,nsm))*ff
if(uu.lt.0.d0) then
jb=ja-1
! #ifndef ifs
! if(jb==0) stop 34578
! #endif
endif
y1=dble(ja-1)
if(doturb==1) then
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! zonal position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! vertical position
else
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! zonal position
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! vertical position
endif
else if(ds==dsu) then ! upward grid-cell exit
! scrivi=.false.
call vertvel(rr,ia,ja,ka,imt,jmt,km,ff,uflux,vflux,do3d,wflux)
! #ifdef full_wflux
! uu=wflux(ia,ja,ka,nsm)
! #else
uu=rbg*wflux(ka,nsp)+rb*wflux(ka,nsm)
! #endif
if(uu.gt.0.d0) then
kb=ka+1
endif
z1=dble(ka)
if(kb==km+1) then ! prevent "evaporation" and put particle from the surface
kb=km
z1=dble(km)-0.5d0 ! to the middle of the surface layer
endif
if(doturb==1) then
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! zonal position
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! meridional position
else
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! zonal position
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! meridional position
endif
else if(ds==dsd) then ! downward grid-cell exit
! scrivi=.false.
call vertvel(rr,ia,ja,ka,imt,jmt,km,ff,uflux,vflux,do3d,wflux)
! #ifdef full_wflux
! if(wflux(ia,ja,ka-1,nsm).lt.0.d0) kb=ka-1
! #else
if(rbg*wflux(ka-1,nsp)+rb*wflux(ka-1,nsm).lt.0.d0) kb=ka-1
! #endif
z1=dble(ka-1)
if(doturb==1) then
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! zonal position
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! meridional position
else
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! zonal position
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! meridional position
endif
else if( ds==dsc .or. ds==dsmin) then
! shortest time is the time-steping
! scrivi=.true.
! If there is no spatial solution,
! which should correspond to a convergence zone
if(dse==UNDEF .and. dsw==UNDEF .and. dsn==UNDEF .and. &
dss==UNDEF .and. dsu==UNDEF .and. dsd==UNDEF ) then
! move if atmosphere, freeze if ocean
ib=ia ; jb=ja ; kb=ka
! If there is at least one spatial solution
! but the shortest cross time is the time step
endif
! else
if(doturb==1) then
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! zonal crossing
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! merid. crossing
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb,upr) ! vert. crossing
else
call pos_orgn(1,ia,ja,ka,x0,x1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! zonal crossing
call pos_orgn(2,ia,ja,ka,y0,y1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! merid. crossing
call pos_orgn(3,ia,ja,ka,z0,z1,ds,rr,uflux,vflux,wflux,ff,imt,jmt,km,do3d,doturb) ! vert. crossing
endif
! endif
endif
end subroutine pos