source: lmdz_wrf/trunk/WRFV3/phys/module_cam_constituents.F @ 1577

Last change on this file since 1577 was 1, checked in by lfita, 10 years ago
  • -- --- Opening of the WRF+LMDZ coupling repository --- -- -

WRF: version v3.3
LMDZ: version v1818

More details in:

File size: 14.2 KB
Line 
1#define WRF_PORT
2
3module constituents
4
5!----------------------------------------------------------------------------------------------
6!
7! Purpose: Contains data and functions for manipulating advected and non-advected constituents.
8!
9! Revision history:
10!             B.A. Boville    Original version
11! June 2003   P. Rasch        Add wet/dry m.r. specifier
12! 2004-08-28  B. Eaton        Add query function to allow turning off the default CAM output of
13!                             constituents so that chemistry module can make the outfld calls.
14!                             Allow cnst_get_ind to return without aborting when constituent not
15!                             found.
16! 2006-10-31  B. Eaton        Remove 'non-advected' constituent functionality.
17!----------------------------------------------------------------------------------------------
18  use shr_kind_mod, only: r8 => shr_kind_r8
19  use physconst,    only: r_universal
20
21#ifndef WRF_PORT
22  use spmd_utils,   only: masterproc
23  use abortutils,   only: endrun
24  use cam_logfile,  only: iulog
25#else
26  use module_cam_support,   only: masterproc,endrun,iulog,pcnst
27#endif
28
29  implicit none
30  private
31  save
32!
33! Public interfaces
34!
35  public cnst_add             ! add a constituent to the list of advected constituents
36  public cnst_num_avail       ! returns the number of available slots in the constituent array
37  public cnst_get_ind         ! get the index of a constituent
38  public cnst_get_type_byind  ! get the type of a constituent
39  public cnst_get_type_byname ! get the type of a constituent
40  public cnst_read_iv         ! query whether constituent initial values are read from initial file
41  public cnst_chk_dim         ! check that number of constituents added equals dimensions (pcnst)
42  public cnst_cam_outfld      ! Returns true if default CAM output was specified in the cnst_add calls.
43
44! Public data
45#ifndef WRF_PORT
46  integer, parameter, public :: pcnst  = PCNST      ! number of advected constituents (including water vapor)
47#endif
48
49  character(len=16), public :: cnst_name(pcnst)     ! constituent names
50  character(len=128),public :: cnst_longname(pcnst) ! long name of constituents
51
52! Namelist variables
53  logical, public :: readtrace = .true.             ! true => obtain initial tracer data from IC file
54
55!
56! Constants for each tracer
57  real(r8),    public :: cnst_cp  (pcnst)          ! specific heat at constant pressure (J/kg/K)
58  real(r8),    public :: cnst_cv  (pcnst)          ! specific heat at constant volume (J/kg/K)
59  real(r8),    public :: cnst_mw  (pcnst)          ! molecular weight (kg/kmole)
60  character*3, public :: cnst_type(pcnst)          ! wet or dry mixing ratio
61  real(r8),    public :: cnst_rgas(pcnst)          ! gas constant ()
62  real(r8),    public :: qmin     (pcnst)          ! minimum permitted constituent concentration (kg/kg)
63  real(r8),    public :: qmincg   (pcnst)          ! for backward compatibility only
64  logical,     public :: cnst_fixed_ubc(pcnst) = .false.  ! upper bndy condition = fixed ?
65
66!++bee - temporary... These names should be declared in the module that makes the addfld and outfld calls.
67! Lists of tracer names and diagnostics
68   character(len=16), public :: apcnst    (pcnst)   ! constituents after physics  (FV core only)
69   character(len=16), public :: bpcnst    (pcnst)   ! constituents before physics (FV core only)
70   character(len=16), public :: hadvnam   (pcnst)   ! names of horizontal advection tendencies
71   character(len=16), public :: vadvnam   (pcnst)   ! names of vertical advection tendencies
72   character(len=16), public :: dcconnam  (pcnst)   ! names of convection tendencies
73   character(len=16), public :: fixcnam   (pcnst)   ! names of species slt fixer tendencies
74   character(len=16), public :: tendnam   (pcnst)   ! names of total tendencies of species
75   character(len=16), public :: ptendnam  (pcnst)   ! names of total physics tendencies of species
76   character(len=16), public :: dmetendnam(pcnst)   ! names of dme adjusted tracers (FV)
77   character(len=16), public :: sflxnam   (pcnst)   ! names of surface fluxes of species
78   character(len=16), public :: tottnam   (pcnst)   ! names for horz + vert + fixer tendencies
79
80! Private data
81
82  integer :: padv = 0                      ! index pointer to last advected tracer
83  logical :: read_init_vals(pcnst)         ! true => read initial values from initial file
84  logical :: cam_outfld_(pcnst)            ! true  => default CAM output of constituents in kg/kg
85                                           ! false => chemistry is responsible for making outfld
86                                           !          calls for constituents
87
88!==============================================================================================
89CONTAINS
90!==============================================================================================
91
92  subroutine cnst_add (name, mwc, cpc, qminc, &
93                       ind, longname, readiv, mixtype, cam_outfld, fixed_ubc)
94!-----------------------------------------------------------------------
95!
96! Purpose: Register a constituent to be advected by the large scale winds and transported by
97!          subgrid scale processes.
98!
99!---------------------------------------------------------------------------------
100!
101    character(len=*), intent(in) :: &
102       name      ! constituent name used as variable name in history file output (8 char max)
103    real(r8),intent(in)    :: mwc    ! constituent molecular weight (kg/kmol)
104    real(r8),intent(in)    :: cpc    ! constituent specific heat at constant pressure (J/kg/K)
105    real(r8),intent(in)    :: qminc  ! minimum value of mass mixing ratio (kg/kg)
106                                     ! normally 0., except water 1.E-12, for radiation.
107    integer, intent(out)   :: ind    ! global constituent index (in q array)
108
109    character(len=*), intent(in), optional :: &
110       longname    ! value for long_name attribute in netcdf output (128 char max, defaults to name)
111    logical,          intent(in), optional :: &
112       readiv      ! true => read initial values from initial file (default: true)
113    character(len=*), intent(in), optional :: &
114       mixtype     ! mixing ratio type (dry, wet)
115    logical,          intent(in), optional :: &
116       cam_outfld  ! true => default CAM output of constituent in kg/kg
117    logical,          intent(in), optional :: &
118       fixed_ubc ! true => const has a fixed upper bndy condition
119
120!-----------------------------------------------------------------------
121
122! set tracer index and check validity, advected tracer
123    padv = padv+1
124    ind  = padv
125    if (padv > pcnst) then
126       write(iulog,*) 'CNST_ADD: advected tracer index greater than pcnst = ', pcnst
127       call endrun
128    end if
129
130! set tracer name and constants
131    cnst_name(ind) = name
132    if ( present(longname) )then
133       cnst_longname(ind) = longname
134    else
135       cnst_longname(ind) = name
136    end if
137
138! set whether to read initial values from initial file
139    if ( present(readiv) ) then
140       read_init_vals(ind) = readiv
141    else
142       read_init_vals(ind) = readtrace
143    end if
144
145! set constituent mixing ratio type
146    if ( present(mixtype) )then
147       cnst_type(ind) = mixtype
148    else
149       cnst_type(ind) = 'wet'
150    end if
151
152! set outfld type
153! (false: the module declaring the constituent is responsible for outfld calls)
154    if ( present(cam_outfld) ) then
155       cam_outfld_(ind) = cam_outfld
156    else
157       cam_outfld_(ind) = .true.
158    end if
159
160! set upper boundary condition type
161    if ( present(fixed_ubc) ) then
162       cnst_fixed_ubc(ind) = fixed_ubc
163    else
164       cnst_fixed_ubc(ind) = .false.
165    end if
166
167    cnst_cp  (ind) = cpc
168    cnst_mw  (ind) = mwc
169    qmin     (ind) = qminc
170    qmincg   (ind) = qminc
171    if (ind == 1) qmincg = 0._r8  ! This crap is replicate what was there before ****
172
173    cnst_rgas(ind) = r_universal * mwc
174    cnst_cv  (ind) = cpc - cnst_rgas(ind)
175
176    return
177  end subroutine cnst_add
178
179!==============================================================================
180
181  function cnst_num_avail()
182
183     ! return number of available slots in the constituent array
184
185     integer cnst_num_avail
186
187     cnst_num_avail = pcnst - padv
188
189  end function cnst_num_avail
190
191!==============================================================================
192
193  subroutine cnst_get_ind (name, ind, abort)
194!-----------------------------------------------------------------------
195!
196! Purpose: Get the index of a constituent
197!
198! Author:  B.A. Boville
199!
200!-----------------------------Arguments---------------------------------
201!
202    character(len=*),  intent(in)  :: name  ! constituent name
203    integer,           intent(out) :: ind   ! global constituent index (in q array)
204    logical, optional, intent(in)  :: abort ! optional flag controlling abort
205
206!---------------------------Local workspace-----------------------------
207    integer :: m                                   ! tracer index
208    logical :: abort_on_error
209!-----------------------------------------------------------------------
210
211! Find tracer name in list
212    do m = 1, pcnst
213       if (name == cnst_name(m)) then
214          ind  = m
215          return
216       end if
217    end do
218
219! Unrecognized name
220    abort_on_error = .true.
221    if ( present(abort) ) abort_on_error = abort
222
223    if ( abort_on_error ) then
224       write(iulog,*) 'CNST_GET_IND, name:', name,  ' not found in list:', cnst_name(:)
225#ifdef WRF_PORT
226       call wrf_message(iulog)
227#endif
228       call endrun('CNST_GET_IND: name not found')
229    end if
230
231! error return
232    ind = -1
233
234  end subroutine cnst_get_ind
235
236!==============================================================================================
237
238  character*3 function cnst_get_type_byind (ind)
239!-----------------------------------------------------------------------
240!
241! Purpose: Get the type of a constituent
242!
243! Method:
244! <Describe the algorithm(s) used in the routine.>
245! <Also include any applicable external references.>
246!
247! Author:  P. J. Rasch
248!
249!-----------------------------Arguments---------------------------------
250!
251    integer, intent(in)   :: ind    ! global constituent index (in q array)
252
253!---------------------------Local workspace-----------------------------
254    integer :: m                                   ! tracer index
255
256!-----------------------------------------------------------------------
257
258    if (ind.le.pcnst) then
259       cnst_get_type_byind = cnst_type(ind)
260    else
261       ! Unrecognized name
262       write(iulog,*) 'CNST_GET_TYPE_BYIND, ind:', ind
263       call endrun
264    endif
265
266
267  end function cnst_get_type_byind
268
269!==============================================================================================
270
271  character*3 function cnst_get_type_byname (name)
272!-----------------------------------------------------------------------
273!
274! Purpose: Get the type of a constituent
275!
276! Method:
277! <Describe the algorithm(s) used in the routine.>
278! <Also include any applicable external references.>
279!
280! Author:  P. J. Rasch
281!
282!-----------------------------Arguments---------------------------------
283!
284    character(len=*), intent(in) :: name ! constituent name
285
286!---------------------------Local workspace-----------------------------
287    integer :: m                                   ! tracer index
288
289!-----------------------------------------------------------------------
290
291    do m = 1, pcnst
292       if (name == cnst_name(m)) then
293          cnst_get_type_byname = cnst_type(m)
294          return
295       end if
296    end do
297
298! Unrecognized name
299    write(iulog,*) 'CNST_GET_TYPE_BYNAME, name:', name,  ' not found in list:', cnst_name(:)
300    call endrun
301
302  end function cnst_get_type_byname
303
304!==============================================================================
305  function cnst_read_iv(m)
306!-----------------------------------------------------------------------
307!
308! Purpose: Query whether constituent initial values are read from initial file.
309!
310! Author:  B. Eaton
311!
312!-----------------------------Arguments---------------------------------
313!
314    integer, intent(in) :: m    ! constituent index
315
316    logical :: cnst_read_iv     ! true => read initial values from inital file
317!-----------------------------------------------------------------------
318
319    cnst_read_iv = read_init_vals(m)
320 end function cnst_read_iv
321
322!==============================================================================
323  subroutine cnst_chk_dim
324!-----------------------------------------------------------------------
325!
326! Purpose: Check that the number of registered constituents of each type is the
327!          same as the dimension
328!
329! Method:
330! <Describe the algorithm(s) used in the routine.>
331! <Also include any applicable external references.>
332!
333! Author:  B.A. Boville
334!
335    integer i,m
336!-----------------------------------------------------------------------
337!
338    if (padv /= pcnst) then
339       write(iulog,*)'CNST_CHK_DIM: number of advected tracer ',padv, ' not equal to pcnst = ',pcnst
340       call endrun ()
341    endif
342
343    if (masterproc) then
344       write(iulog,*) 'Advected constituent list:'
345       do i = 1, pcnst
346          write(iulog,'(i4,2x,a8,2x,a128,2x,a3)') i, cnst_name(i), cnst_longname(i), cnst_type(i)
347       end do
348    end if
349
350    ! Set names of advected tracer diagnostics
351    do m=1,pcnst
352       apcnst    (m)  = trim(cnst_name(m))//'AP'
353       bpcnst    (m)  = trim(cnst_name(m))//'BP'
354       hadvnam   (m)  = 'HA'//cnst_name(m)
355       vadvnam   (m)  = 'VA'//cnst_name(m)
356       fixcnam   (m)  = 'DF'//cnst_name(m)
357       tendnam   (m)  = 'TE'//cnst_name(m)
358       ptendnam  (m)  = 'PTE'//cnst_name(m)
359       dmetendnam(m)  = 'DME'//cnst_name(m)
360       tottnam   (m)  = 'TA'//cnst_name(m)
361       sflxnam(m)     = 'SF'//cnst_name(m)
362    end do
363
364
365  end subroutine cnst_chk_dim
366
367!==============================================================================
368
369function cnst_cam_outfld(m)
370!-----------------------------------------------------------------------
371!
372! Purpose:
373! Query whether default CAM outfld calls should be made.
374!
375!-----------------------------------------------------------------------
376   integer, intent(in) :: m                ! constituent index
377   logical             :: cnst_cam_outfld  ! true => use default CAM outfld calls
378!-----------------------------------------------------------------------
379
380   cnst_cam_outfld = cam_outfld_(m)
381
382end function cnst_cam_outfld
383
384!==============================================================================
385
386end module constituents
Note: See TracBrowser for help on using the repository browser.