Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Noah LSM for HAFS #419

Closed
wants to merge 4 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,8 @@ if (${CMAKE_Fortran_COMPILER_ID} STREQUAL "GNU")
SET_SOURCE_FILES_PROPERTIES(${CMAKE_CURRENT_SOURCE_DIR}/physics/module_SF_JSFC.F90
${CMAKE_CURRENT_SOURCE_DIR}/physics/module_BL_MYJPBL.F90
PROPERTIES COMPILE_FLAGS "-fdefault-real-8 -fdefault-double-8")
SET_SOURCE_FILES_PROPERTIES(${CMAKE_CURRENT_SOURCE_DIR}/physics/module_sf_noahlsm.F
PROPERTIES COMPILE_FLAGS "-fdefault-real-8 -fdefault-double-8 -ffree-form")
if (PROJECT STREQUAL "CCPP-FV3")
# Set 32-bit floating point precision flags for certain files
# that are executed in the dynamics (fast physics part)
Expand Down
8 changes: 6 additions & 2 deletions physics/GFS_surface_generic.F90
Original file line number Diff line number Diff line change
Expand Up @@ -155,10 +155,14 @@ subroutine GFS_surface_generic_pre_run (im, levs, vfrac, islmsk, isot, ivegsrc,
else
soiltyp(i) = 9
endif
if (ivegsrc == 1) then
if (ivegsrc == 0 .or. ivegsrc == 4) then
vegtype(i) = 24
elseif (ivegsrc == 1) then
vegtype(i) = 15
elseif(ivegsrc == 2) then
elseif (ivegsrc == 2) then
vegtype(i) = 13
elseif (ivegsrc == 3 .or. ivegsrc == 5) then
vegtype(i) = 15
endif
slopetyp(i) = 9
else
Expand Down
4,775 changes: 4,775 additions & 0 deletions physics/module_sf_noahlsm.F

Large diffs are not rendered by default.

14 changes: 13 additions & 1 deletion physics/sfc_drv.f
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,19 @@ subroutine lsm_noah_init(me, isot, ivegsrc, nlunit,
! Initialize CCPP error handling variables
errmsg = ''
errflg = 0


if (ivegsrc > 2) then
errmsg = 'The NOAH LSM expects that the ivegsrc physics '//
& 'namelist parameter is 0, 1, or 2. Exiting...'
errflg = 1
return
end if
if (isot > 1) then
errmsg = 'The NOAH LSM expects that the isot physics '//
& 'namelist parameter is 0, or 1. Exiting...'
errflg = 1
return
end if
!--- initialize soil vegetation
call set_soilveg(me, isot, ivegsrc, nlunit)

Expand Down
13 changes: 12 additions & 1 deletion physics/sfc_drv_ruc.F90
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,18 @@ subroutine lsm_ruc_init (me, isot, ivegsrc, nlunit, &
! Initialize CCPP error handling variables
errmsg = ''
errflg = 0


if (ivegsrc /= 1) then
errmsg = 'The RUC LSM expects that the ivegsrc physics namelist parameter is 1. Exiting...'
errflg = 1
return
end if
if (isot > 1) then
errmsg = 'The RUC LSM expects that the isot physics namelist parameter is 0, or 1. Exiting...'
errflg = 1
return
end if

!--- initialize soil vegetation
call set_soilveg_ruc(me, isot, ivegsrc, nlunit)

Expand Down
236 changes: 236 additions & 0 deletions physics/sfc_noah.F90
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
!> \file sfc_noah.F90
!! This file contains the Noah land surface scheme driver.

!> This module contains the CCPP-compliant Noah land surface scheme driver for
!! the hurricane application.
module sfc_noah

implicit none

private

public :: sfc_noah_init, sfc_noah_run, sfc_noah_finalize

contains

!> \ingroup Noah_LSM_hafs
!! \section arg_table_sfc_noah_init Argument Table
!! \htmlinclude sfc_noah_init.html
!!
subroutine sfc_noah_init(lsm, lsm_noah_hafs, nsoil, ua_phys, fasdas, errmsg, errflg)

use machine, only : kind_phys

implicit none

integer, intent(in) :: lsm, lsm_noah_hafs, nsoil, fasdas
logical, intent(in) :: ua_phys

character(len=*), intent(out) :: errmsg
integer, intent(out) :: errflg

! Initialize CCPP error handling variables
errmsg = ''
errflg = 0

if (lsm/=lsm_noah_hafs) then
write(errmsg,'(*(a))') "Logic error: namelist choice of LSM is different from NOAH HAFS"
errflg = 1
return
end if

if (nsoil < 2) then
write(errmsg,'(*(a))') "The NOAH HAFS scheme expects at least 2 soil layers."
errflg = 1
return
end if

if (ua_phys) then
write(errmsg,'(*(a))') "The NOAH HAFS scheme has not been tested with ua_phys = T"
errflg = 1
return
end if


if (fasdas > 0) then
write(errmsg,'(*(a))') "The NOAH HAFS scheme has not been tested with fasdas > 0"
errflg = 1
return
end if

!GJF: check for rdlai != F?
!GJF: check for usemonalb != T?

end subroutine sfc_noah_init


!! \section arg_table_sfc_noah_finalize Argument Table
!! \htmlinclude sfc_noah_finalize.html
!!
subroutine sfc_noah_finalize(errmsg, errflg)

implicit none

character(len=*), intent(out) :: errmsg
integer, intent(out) :: errflg

! Initialize CCPP error handling variables
errmsg = ''
errflg = 0

end subroutine sfc_noah_finalize


!> \defgroup Noah_LSM_hafs Noah LSM Model for the hurricane application
!! \section arg_table_sfc_noah_run Argument Table
!! \htmlinclude sfc_noah_run.html
!!
!> \section general_noah_hafs_drv GFS sfc_drv General Algorithm
!> @{
subroutine sfc_noah_run (im, flag_lsm, srflag, isurban, rdlai, &
ua_phys, usemonalb, aoasis, fasdas, dt, zlvl, &
nsoil, sthick, lwdn, soldn, solnet, sfcprs, prcp, sfctmp, q1k, &
th1, qs1, dqsdt2, vegtyp, soiltyp, slopetyp, shdfac, shmin, &
shmax, albbrd, snoalb, tbot, z0brd, z0k, emissi, embrd, cmc, t1,&
stc, smc, swc, snowhk, sneqv, chk, cp, rd, sigma, cph2o, cpice, &
lsubf, sheat, eta_kinematic, ec, edir, ett, esnow, etp, ssoil, &
flx1, flx2, flx3, sncovr, runoff1, runoff2, soilm, qsurf, ribb, &
smcwlt, smcref, smcmax, opt_thcnd, errmsg, errflg)

use machine , only : kind_phys
use module_sf_noahlsm, only: sflx, lutype, sltype

implicit none

integer, intent(in) :: im, isurban, nsoil, opt_thcnd, fasdas
logical, intent(in) :: rdlai, ua_phys, usemonalb
!GJF: usemonalb = True if the surface diffused shortwave albedo is EITHER read from input OR
! provided by a previous scheme (like radiation: as is done in GFS_rrtmgp_sw_pre)
real(kind=kind_phys), intent(in) :: aoasis

real(kind=kind_phys), intent(in) :: dt, cp, rd, sigma, cph2o, cpice, lsubf

integer, dimension(im), intent(in) :: vegtyp, soiltyp, slopetyp
logical, dimension(im), intent(in) :: flag_lsm
real(kind=kind_phys), dimension(im), intent(in) :: srflag, zlvl, lwdn, soldn, solnet, &
sfcprs, prcp, sfctmp, q1k, th1, qs1, &
dqsdt2, shmin, shmax, snoalb, tbot
real(kind=kind_phys), dimension(nsoil), intent(in) :: sthick

real(kind=kind_phys), dimension(im), intent(inout) :: shdfac, albbrd, z0brd, z0k, emissi, &
cmc, t1, snowhk, sneqv, chk, flx1, &
flx2, flx3, ribb
real(kind=kind_phys), dimension(im,nsoil), intent(inout) :: stc, smc, swc

!variables that are intent(out) in module_sf_noahlsm, but are inout here due to being set within an IF statement
real(kind=kind_phys), dimension(im), intent(inout) :: embrd, sheat, eta_kinematic, ec, &
edir, ett, esnow, etp, ssoil, sncovr, &
runoff1, runoff2, soilm, qsurf, smcwlt, &
smcref, smcmax

character(len=*), intent(out) :: errmsg
integer, intent(out) :: errflg

!GJF: There is some confusion regarding specific humidities vs mixing ratios in NOAH LSM.
! Looking at module_sf_noahlsm.F, sometimes the comments say mixing ratio and sometimes
! specific humidity. The WRF code (module_sf_noahdrv.F) specifically converts from mixing
! ratio to specific humidity in preparation for calling SFLX, so I am assuming that
! all inputs/outputs into SFLX should be specific humidities, despite some comments in
! module_sf_noahdrv.F describing arguments saying "mixing ratios". This applies to many
! arguments into SFLX (q1k, qs1, dqsdt2, eta_kinematic, qsurf, etc.).

! local Variables
integer :: i, k
logical, parameter :: local = .false. !(not actually used in SFLX) described in module_sf_noahlsm as:
! Flag for local-site simulation (where there is no maps for albedo, veg fraction, and roughness
! true: all LSM parameters (inluding albedo, veg fraction and roughness length) will be defined by three tables

real(kind=kind_phys) :: dummy

!GJF: The following variables are part of the interface to SFLX but not required as diagnostic
! output or otherwise outside of this subroutine (at least as part of a GFS-based suite).
! If any of these variables are needed by other schemes or diagnostics, one needs to add it to
! the host model and CCPP metadata. Alternatively, none of these variables NEED to be allocated
! and one could also just pass in dummy arguments.
!
! The variables descriptions are from module_sf_noahlsm.F:
!
! albedok (output from SFLX): surface albedo including snow effect (unitless fraction)
! =snow-free albedo (alb) when sneqv=0, or
! =fct(msnoalb,alb,vegtyp,shdfac,shdmin) when sneqv>0
! eta (output from SFLX), eta_kinematic is what is passed out instead of eta
! fdown (output from SFLX) : Radiation forcing at the surface (W m-2) = SOLDN*(1-alb)+LWDN
! et (output from SFLX): plant transpiration from a particular root (soil) layer (W m-2)
! drip (output from SFLX): through-fall of precip and/or dew in excess of canopy water-holding capacity (m)
! dew (output from SFLX): dewfall (or frostfall for t<273.15) (m)
! beta (output from SFLX): ratio of actual/potential evap (dimensionless)
! snomlt (output from SFLX): snow melt (m) (water equivalent)
! runoff3 (output from SFLX): numerical trunctation in excess of porosity (smcmax) for a given soil layer at the end of a time step (m s-1).
! rc (output from SFLX): canopy resistance (s m-1)
! pc (output from SFLX): plant coefficient (unitless fraction, 0-1) where pc*etp = actual transp
! rsmin (output from SFLX): minimum canopy resistance (s m-1)
! xlai (output from SFLX): leaf area index (dimensionless)
! rcs (output from SFLX): incoming solar rc factor (dimensionless)
! rct (output from SFLX): air temperature rc factor (dimensionless)
! rcq (output from SFLX): atmos vapor pressure deficit rc factor (dimensionless)
! rcsoil (output from SFLX): soil moisture rc factor (dimensionless)
! soilw (output from SFLX): available soil moisture in root zone (unitless fraction between smcwlt and smcmax)
! smav (output from SFLX): soil moisture availability for each layer, as a fraction between smcwlt and smcmax.
! snotime1 (input/output from SFLX): no documentation in module_sf_noahlsm.F, but described as "initial number of timesteps since last snowfall" in module_sf_noahdrv.F; related to CCPP nondimensional_snow_age for NoahMP? Since inout, need to initialize here?
! smcdry (output from SFLX): dry soil moisture threshold where direct evap frm top layer ends (volumetric)
! smcmax (output from SFLX): porosity, i.e. saturated value of soil moisture (volumetric)
! nroot (output from SFLX): number of root layers, a function of veg type, determined in subroutine redprm.

integer :: nroot
real(kind=kind_phys) :: albedok, eta, fdown, drip, dew, beta, snomlt, &
runoff3, rc, pc, rsmin, xlai, rcs, rct, rcq, &
rcsoil, soilw, snotime1, smcdry
real (kind=kind_phys), dimension(nsoil) :: et, smav
real(kind=kind_phys) :: sfcheadrt, infxsrt, etpnd1 !don't appear to be used unless WRF_HYDRO preprocessor directive is defined and no documentation
real(kind=kind_phys) :: xsda_qfx, hfx_phy, qfx_phy, xqnorm, hcpct_fasdas !only used if fasdas = 1

!variables associated with UA_PHYS (not used for now)
real(kind=kind_phys) :: flx4, fvb, fbur, fgsn

errmsg = ''
errflg = 0

do i=1, im
if (flag_lsm(i)) then
!GJF: Why do LSMs want the dynamics time step instead of the physics time step?
call sflx (i, 1, srflag(i), &
isurban, dt, zlvl(i), nsoil, sthick, & !c
local, & !L
lutype, sltype, & !CL
lwdn(i), soldn(i), solnet(i), sfcprs(i), prcp(i), & !F
sfctmp(i), q1k(i), dummy, dummy, dummy, dummy, & !F
th1(i), qs1(i), dqsdt2(i), & !I
vegtyp(i), soiltyp(i), slopetyp(i), shdfac(i), & !I
shmin(i), shmax(i), & !I
albbrd(i), snoalb(i), tbot(i), z0brd(i), z0k(i), & !S
emissi(i), embrd(i), & !S
cmc(i), t1(i), stc(i,:), smc(i,:), swc(i,:), & !H
snowhk(i), sneqv(i), albedok, chk(i), dummy, & !H
cp, rd, sigma, cph2o, cpice, lsubf, &
eta, sheat(i), eta_kinematic(i), fdown, & !O
ec(i), edir(i), et, ett(i), esnow(i), drip, dew, & !O
beta, etp(i), ssoil(i), flx1(i), flx2(i), flx3(i),& !O
flx4, fvb, fbur, fgsn, ua_phys, & !UA
snomlt, sncovr(i), runoff1(i), runoff2(i),runoff3,& !O
rc, pc, rsmin, xlai, rcs, rct, rcq, rcsoil, & !O
soilw, soilm(i), qsurf(i), smav, & !D
rdlai, usemonalb, snotime1, ribb(i), &
smcwlt(i), smcdry, smcref(i), smcmax(i), nroot, &
sfcheadrt, infxsrt, etpnd1, opt_thcnd, aoasis, &
xsda_qfx, hfx_phy, qfx_phy, xqnorm, fasdas, & !fasdas
hcpct_fasdas, & !fasdas
errflg, errmsg)
if (errflg > 0) return
endif
end do


end subroutine sfc_noah_run
!> @}

end module sfc_noah
Loading