source: LMDZ6/branches/Amaury_dev/libf/phylmd/stratosphere_mask.F90 @ 5133

Last change on this file since 5133 was 5117, checked in by abarral, 6 months ago

rename modules properly lmdz_*
move some unused files to obsolete/
(lint) uppercase fortran keywords

  • Property svn:keywords set to Id
File size: 7.9 KB
RevLine 
[5099]1
[2772]2! $Id: stratosphere_mask.F90 5117 2024-07-24 14:23:34Z abarral $
[5099]3
[3123]4SUBROUTINE stratosphere_mask(missing_val, pphis, t_seri, pplay, xlat)
[2527]5
6!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
[5099]7
[2527]8! determination of tropopause height and temperature from gridded temperature data
[5099]9
[3119]10! reference: Reichler, T., M. Dameris, and R. Sausen (GRL, 10.1029/2003GL018240, 2003)
[2527]11! modified: 6/28/06 tjr
12! adapted to LMDZ by C. Kleinschmitt (2016-02-15)
[3119]13! committed to LMDz by O. Boucher (2016) with a mistake
14! mistake corrected by O. Boucher (2017-12-11)
[5099]15
[3119]16! input:  temp(nlon,nlat,nlev)  3D-temperature field
17!         ps(nlon,nlat)         2D-surface pressure field
18!         zs(nlon,nlat)         2D-surface height
19!         nlon                  grid points in x
20!         nlat                  grid points in y
21!         pfull(nlon,nlat,nlev) full pressure levels in Pa
22!         plimu                 upper limit for tropopause pressure
23!         pliml                 lower limit for tropopause pressure
24!         gamma                 tropopause criterion, e.g. -0.002 K/m
[5099]25
[3119]26! output: p_tropopause(klon)    tropopause pressure in Pa with missing values
27!         t_tropopause(klon)    tropopause temperature in K with missing values
28!         z_tropopause(klon)    tropopause height in m with missing values
29!         stratomask            stratospheric mask withtout missing values
30!         ifil                  # of undetermined values
[5099]31
[2527]32!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
33
34USE dimphy
[2536]35USE phys_local_var_mod, ONLY: stratomask
[2992]36USE phys_local_var_mod, ONLY: p_tropopause, z_tropopause, t_tropopause
[5112]37USE lmdz_print_control, ONLY: lunout, prt_level
[2527]38
39IMPLICIT NONE
40
[3123]41INCLUDE "YOMCST.h"
42
[2992]43REAL, INTENT(IN)                       :: missing_val ! missing value, also XIOS
[3123]44REAL,DIMENSION(klon),INTENT(IN)        :: pphis   ! Geopotentiel de surface
[2527]45REAL,DIMENSION(klon,klev),INTENT(IN)   :: t_seri  ! Temperature
46REAL,DIMENSION(klon,klev),INTENT(IN)   :: pplay   ! pression pour le mileu de chaque couche (en Pa)
47REAL,DIMENSION(klon),INTENT(IN)        :: xlat    ! latitudes pour chaque point
48
49REAL, PARAMETER                        :: plimu=45000.
50REAL, PARAMETER                        :: pliml=7500.
51REAL, PARAMETER                        :: gamma=-0.002
[5103]52LOGICAL, PARAMETER                     :: dofill=.TRUE.
[2992]53REAL,DIMENSION(klon)                   :: tp
[2527]54REAL,DIMENSION(klev)                   :: t, p
[2992]55INTEGER                                :: i, k, ifil
[2527]56REAL                                   :: ptrp, ttrp, ztrp, psrf, zsrf, pi
57
58pi     = 4.*ATAN(1.)
59
[2992]60!--computing tropopause
[2527]61DO i=1,klon
62  DO k=1,klev
63    t(k)=t_seri(i,klev+1-k)
64    p(k)=pplay(i,klev+1-k)
65  ENDDO
66  psrf=pplay(i,1)
[3123]67  zsrf=pphis(i)/RG           !--altitude de la surface
[5103]68  CALL twmo(missing_val, klev, t, p, psrf, zsrf, plimu, pliml, gamma, ptrp, ttrp, ztrp)
[2527]69  tp(i)=ptrp
[2992]70  p_tropopause(i)=ptrp
71  z_tropopause(i)=ztrp
72  t_tropopause(i)=ttrp
[2527]73ENDDO
74
[2992]75!--filling holes in tp but not in p_tropopause
[2527]76IF (dofill) THEN
77  ifil=0
78  DO i=1,klon
[5082]79  IF (ABS(tp(i)/missing_val-1.0)<0.01) THEN
[2527]80    !set missing values to very simple profile (neighbour averaging too expensive in LMDZ)
81    tp(i)=50000.-20000.*cos(xlat(i)/360.*2.*pi)
82    ifil=ifil+1
83  ENDIF
84  ENDDO
[5099]85
[2527]86ENDIF
[5099]87
[2527]88DO i=1, klon
89DO k=1, klev
[5082]90  IF (pplay(i,k)<tp(i)) THEN
[2536]91    stratomask(i,k)=1.0
[2527]92  ELSE
[2536]93    stratomask(i,k)=0.0
[2527]94  ENDIF
95ENDDO
96ENDDO
97
[5082]98IF (ifil>0 .AND. prt_level >5) THEN
[5116]99  WRITE(lunout,*)'Tropopause: number of undetermined values =', ifil
[2527]100ENDIF
101
[5105]102
[2527]103END SUBROUTINE stratosphere_mask
104
105!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
106! twmo
107!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
108
[5103]109SUBROUTINE twmo(missing_val, level, t, p, ps, zs, plimu, pliml, gamma, ptrp, ttrp, ztrp)
[2527]110
[3119]111! reference: Reichler, T., M. Dameris, and R. Sausen (GRL, 10.1029/2003GL018240, 2003)
112
[5113]113IMPLICIT NONE
[2527]114
115include "YOMCST.h"
116
[5117]117INTEGER,INTENT(IN)              :: level
118REAL,INTENT(IN)                 :: missing_val
119REAL,INTENT(IN),DIMENSION(level):: t, p
120REAL,INTENT(IN)                 :: plimu, pliml, gamma, ps, zs
121REAL,INTENT(OUT)                :: ptrp, ttrp, ztrp
[2527]122
[5117]123REAL,parameter                  :: deltaz = 2000.0
[2527]124
125real     :: faktor
126real     :: pmk, pm, a, b, tm, dtdp, dtdz, dlnp, tdlnp
127real     :: ag, bg, ptph, ttph, a0, b0
128real     :: pm0, tm0, pmk0, dtdz0
129real     :: p2km, asum, aquer
130real     :: pmk2, pm2, a2, b2, tm2, dtdp2, dtdz2
[5117]131INTEGER  :: icount, jj, j
[2527]132
[2992]133ptrp=missing_val
134ttrp=missing_val
135ztrp=missing_val
[2527]136
[3119]137faktor = -RG/RD
[2527]138
[5103]139DO j=level,2,-1
[2527]140
141   ! dt/dz
142   pmk= 0.5 * (p(j-1)**rkappa+p(j)**rkappa)        ! p**k at half level
143   pm = pmk**(1./rkappa)                    ! p at half level
144   a = (t(j-1)-t(j))/(p(j-1)**rkappa-p(j)**rkappa)    ! dT/dp^k
145   b = t(j)-(a*p(j)**rkappa)
146   tm = a * pmk + b                    ! T at half level         
147   dtdp = a * rkappa * pm**(rkappa-1.) ! dT/dp at half level
148   dtdz = faktor*dtdp*pm/tm            ! dT/dz at half level
149
150   ! dt/dz valid?
[5117]151   IF (j==level)     go to 999                 ! no, start level, initialize first
152   IF (dtdz<=gamma)  go to 999                 ! no, dt/dz < -2 K/km
153   IF (dtdz0>gamma) go to 999                 ! no, dt/dz below > -2 K/km
154   IF (pm>plimu)    go to 999                 ! no, pm too low
[2527]155
156   ! dtdz is valid, calculate tropopause pressure ptph
157   ag = (dtdz-dtdz0) / (pmk-pmk0)     
158   bg = dtdz0 - (ag * pmk0)         
159   ptph = exp(log((gamma-bg)/ag)/rkappa)
160
161   ! calculate temperature at this ptph assuming linear gamma
162   ! interpolation
163   ttph = tm0
164   ttph = ttph - (bg * log(pm0)  + ag * (pm0**rkappa) /rkappa) / faktor*t(j)
165   ttph = ttph + (bg * log(ptph) + ag * (ptph**rkappa)/rkappa) / faktor*t(j)
166
[5117]167   IF (ptph<pliml) go to 999
168   IF (ptph>plimu) go to 999
[2527]169
170   ! 2nd test: dtdz above 2 km must not exceed gamma
171   p2km = ptph + deltaz*(pm/tm)*faktor      ! p at ptph + 2km
172   asum = 0.0                               ! dtdz above
173   icount = 0                               ! number of levels above
174
175   ! test until apm < p2km
176   do jj=j,2,-1
177
178       pmk2 = .5 * (p(jj-1)**rkappa+p(jj)**rkappa)    ! p mean ^kappa
179       pm2 = pmk2**(1/rkappa)                      ! p mean
[5116]180       IF(pm2>ptph) go to 110                ! doesn't happen
181       IF(pm2<p2km) go to 888                ! ptropo is valid
[2527]182
183       a2 = (t(jj-1)-t(jj))                     ! a
184       a2 = a2/(p(jj-1)**rkappa-p(jj)**rkappa)
185       b2 = t(jj)-(a2*p(jj)**rkappa)               ! b
186       tm2 = a2 * pmk2 + b2                     ! T mean
187       dtdp2 = a2 * rkappa * (pm2**(rkappa-1))        ! dt/dp
188       dtdz2 = faktor*dtdp2*pm2/tm2
189       asum = asum+dtdz2
190       icount = icount+1
191       aquer = asum/float(icount)               ! dt/dz mean
192   
193       ! discard ptropo ?
[5117]194        IF (aquer<=gamma) go to 999           ! dt/dz above < gamma
[2527]195
196110 continue
197    enddo                   ! test next level
198
199888 continue                    ! ptph is valid
200    ptrp = ptph
201    ttrp = ttph
202
203! now calculate height of tropopause by integrating hypsometric equation
204! from ps to ptrp
205! linearly interpolate in p (results are identical to p^kappa)
206
207jj = LEVEL                                       ! bottom
[5117]208DO while ((P(jj)>PS) .OR. (T(jj)<100))     ! T must be valid too
[2527]209  jj=jj-1
[5103]210END DO
[2527]211
212DLNP = log(PS/P(jj))                             ! from surface pressure
213TM = T(jj)                                       ! take TM of lowest level (better: extrapolate)
214TDLNP = TM*DLNP
215
[5117]216DO while ( (JJ>=2) .AND. (PTRP<P(jj-1)) )
[2527]217  DLNP = log(P(jj)/P(jj-1))
218  TM = 0.5 * (T(jj) + T(jj-1))
219  TDLNP = TDLNP + TM*DLNP
220  JJ=JJ-1
[5103]221END DO
[2527]222
223DLNP = log(P(jj)/PTRP)                           ! up to tropopause pressure
224TM = 0.5 * (T(jj) + TTRP)                        ! use TTRP to get TM of this level
225TDLNP = TDLNP + TM*DLNP
226
[3119]227ZTRP = ZS + TDLNP*RD/RG
[2527]228
[5116]229!!if (ZTRP .lt. 0) THEN
[5103]230!!  PRINT*,'ZTRP=',ZTRP
231!!  PRINT*,'PS=',PS
232!!  PRINT*,'P=',P
233!!  PRINT*,'T=',T
234!!  PRINT*,'ZS=',ZS
[3119]235!!  stop
236!!endif
[2527]237
238return
239
240999 continue                    ! continue search at next higher level
241    tm0 = tm
242    pm0 = pm
243    pmk0 = pmk
244    dtdz0  = dtdz
245    a0 = a
246    b0 = b
247
[5103]248END DO
[2527]249
250! no tropopouse found
[5105]251
[5103]252END SUBROUTINE  twmo
Note: See TracBrowser for help on using the repository browser.