From cae1a64c9e51bbf12fa0563344eedd470a247fe6 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Mon, 17 Jul 2017 16:18:34 -0600 Subject: [PATCH 1/8] Refactor setting of icefrac and fracice in SetFracIce It looks safe to make this change because (1) fracice is only used in code with origflag == 1, and (2) icefrac is overwritten in Infiltration before it is used (other than for the setting of fracice) --- src/biogeophys/SoilHydrologyMod.F90 | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index e82b9f76a8..460ca71073 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -132,6 +132,7 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & ! !LOCAL VARIABLES: integer :: j, fc, c real(r8) :: vol_ice(bounds%begc:bounds%endc,1:nlevsoi) !partial volume of ice lens in layer + real(r8) :: icefrac_orig ! original formulation for icefrac character(len=*), parameter :: subname = 'SetFracIce' !----------------------------------------------------------------------- @@ -157,13 +158,12 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & ! fractional impermeability vol_ice(c,j) = min(watsat(c,j), h2osoi_ice(c,j)/(dz(c,j)*denice)) - if (origflag == 1) then - icefrac(c,j) = min(1._r8,h2osoi_ice(c,j)/(h2osoi_ice(c,j)+h2osoi_liq(c,j))) - else - icefrac(c,j) = min(1._r8,vol_ice(c,j)/watsat(c,j)) - endif + icefrac(c,j) = min(1._r8,vol_ice(c,j)/watsat(c,j)) - fracice(c,j) = max(0._r8,exp(-3._r8*(1._r8-icefrac(c,j)))- exp(-3._r8))/(1.0_r8-exp(-3._r8)) + ! fracice is only used in code with origflag == 1. For this calculation, we use + ! the version of icefrac that was used in this original hydrology code. + icefrac_orig = min(1._r8,h2osoi_ice(c,j)/(h2osoi_ice(c,j)+h2osoi_liq(c,j))) + fracice(c,j) = max(0._r8,exp(-3._r8*(1._r8-icefrac_orig))- exp(-3._r8))/(1.0_r8-exp(-3._r8)) end do end do From 0ae83d5aa0cf7848caf9815f09b89e6ea98341fa Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Mon, 17 Jul 2017 16:22:16 -0600 Subject: [PATCH 2/8] Move / remove redundant initial settings in Infiltration --- src/biogeophys/HydrologyNoDrainageMod.F90 | 4 +-- src/biogeophys/SoilHydrologyMod.F90 | 34 ++++++++--------------- 2 files changed, 14 insertions(+), 24 deletions(-) diff --git a/src/biogeophys/HydrologyNoDrainageMod.F90 b/src/biogeophys/HydrologyNoDrainageMod.F90 index d6af54d91a..27340057e8 100644 --- a/src/biogeophys/HydrologyNoDrainageMod.F90 +++ b/src/biogeophys/HydrologyNoDrainageMod.F90 @@ -60,7 +60,7 @@ subroutine HydrologyNoDrainage(bounds, & use clm_time_manager , only : get_step_size, get_nstep use SnowHydrologyMod , only : SnowCompaction, CombineSnowLayers, DivideSnowLayers, SnowCapping use SnowHydrologyMod , only : SnowWater, BuildSnowFilter - use SoilHydrologyMod , only : CLMVICMap, SetFracIce + use SoilHydrologyMod , only : CLMVICMap, SetSoilWaterFractions use SoilHydrologyMod , only : SetQflxTopSoil, Infiltration, TotalSurfaceRunoff use SoilHydrologyMod , only : UpdateUrbanPonding use SoilHydrologyMod , only : WaterTable, PerchedWaterTable @@ -177,7 +177,7 @@ subroutine HydrologyNoDrainage(bounds, & soilhydrology_inst, waterstate_inst) end if - call SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & + call SetSoilWaterFractions(bounds, num_hydrologyc, filter_hydrologyc, & soilhydrology_inst, soilstate_inst, waterstate_inst) call surf_runoff_sat_inst%SaturatedSurfaceRunoff(& diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index 460ca71073..4267d845ff 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -32,7 +32,7 @@ module SoilHydrologyMod ! ! !PUBLIC MEMBER FUNCTIONS: public :: SoilHydReadNML ! Read in the Soil hydrology namelist - public :: SetFracIce ! Set diagnostic variables related to the fraction of ice in each layer + public :: SetSoilWaterFractions ! Set diagnostic variables related to the fraction of water and ice in each layer public :: SetQflxTopSoil ! Set the flux of water into the soil from the top public :: Infiltration ! Calculate infiltration into surface soil layer public :: TotalSurfaceRunoff ! Calculate total surface runoff @@ -112,11 +112,11 @@ subroutine soilHydReadNML( NLFilename ) end subroutine soilhydReadNML !----------------------------------------------------------------------- - subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & + subroutine SetSoilWaterFractions(bounds, num_hydrologyc, filter_hydrologyc, & soilhydrology_inst, soilstate_inst, waterstate_inst) ! ! !DESCRIPTION: - ! Set diagnostic variables related to the fraction of ice in each layer + ! Set diagnostic variables related to the fraction of water and ice in each layer ! ! !USES: use clm_varcon, only : denice @@ -126,7 +126,7 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & integer , intent(in) :: num_hydrologyc ! number of column soil points in column filter integer , intent(in) :: filter_hydrologyc(:) ! column filter for soil points type(soilhydrology_type) , intent(inout) :: soilhydrology_inst - type(soilstate_type) , intent(in) :: soilstate_inst + type(soilstate_type) , intent(inout) :: soilstate_inst type(waterstate_type) , intent(in) :: waterstate_inst ! ! !LOCAL VARIABLES: @@ -134,13 +134,14 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & real(r8) :: vol_ice(bounds%begc:bounds%endc,1:nlevsoi) !partial volume of ice lens in layer real(r8) :: icefrac_orig ! original formulation for icefrac - character(len=*), parameter :: subname = 'SetFracIce' + character(len=*), parameter :: subname = 'SetSoilWaterFractions' !----------------------------------------------------------------------- associate( & dz => col%dz , & ! Input: [real(r8) (:,:) ] layer depth (m) watsat => soilstate_inst%watsat_col , & ! Input: [real(r8) (:,:) ] volumetric soil water at saturation (porosity) + eff_porosity => soilstate_inst%eff_porosity_col , & ! Output: [real(r8) (:,:) ] effective porosity = porosity - vol_ice h2osoi_liq => waterstate_inst%h2osoi_liq_col , & ! Input: [real(r8) (:,:) ] liquid water (kg/m2) h2osoi_ice => waterstate_inst%h2osoi_ice_col , & ! Input: [real(r8) (:,:) ] ice lens (kg/m2) @@ -156,8 +157,8 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & ! Porosity of soil, partial volume of ice and liquid, fraction of ice in each layer, ! fractional impermeability - vol_ice(c,j) = min(watsat(c,j), h2osoi_ice(c,j)/(dz(c,j)*denice)) + eff_porosity(c,j) = max(0.01_r8,watsat(c,j)-vol_ice(c,j)) icefrac(c,j) = min(1._r8,vol_ice(c,j)/watsat(c,j)) ! fracice is only used in code with origflag == 1. For this calculation, we use @@ -169,7 +170,7 @@ subroutine SetFracIce(bounds, num_hydrologyc, filter_hydrologyc, & end associate - end subroutine SetFracIce + end subroutine SetSoilWaterFractions !----------------------------------------------------------------------- subroutine SetQflxTopSoil(bounds, num_hydrologyc, filter_hydrologyc, & @@ -228,19 +229,18 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f integer , intent(in) :: num_urbanc ! number of column urban points in column filter integer , intent(in) :: filter_urbanc(:) ! column filter for urban points type(energyflux_type) , intent(in) :: energyflux_inst - type(soilhydrology_type) , intent(inout) :: soilhydrology_inst - type(soilstate_type) , intent(inout) :: soilstate_inst + type(soilhydrology_type) , intent(in) :: soilhydrology_inst + type(soilstate_type) , intent(in) :: soilstate_inst type(surf_runoff_sat_type), intent(in) :: surf_runoff_sat_inst type(temperature_type) , intent(in) :: temperature_inst type(waterstate_type) , intent(inout) :: waterstate_inst type(waterflux_type) , intent(inout) :: waterflux_inst ! ! !LOCAL VARIABLES: - integer :: c,j,l,fc ! indices + integer :: c,l,fc ! indices real(r8) :: dtime ! land model time step (sec) real(r8) :: s1,su,v ! variable to calculate qinmax real(r8) :: qinmax ! maximum infiltration capacity (mm/s) - real(r8) :: vol_ice(bounds%begc:bounds%endc,1:nlevsoi) ! partial volume of ice lens in layer real(r8) :: alpha_evap(bounds%begc:bounds%endc) ! fraction of total evap from h2osfc real(r8) :: qflx_evap(bounds%begc:bounds%endc) ! local evaporation array real(r8) :: qflx_h2osfc_drain(bounds%begc:bounds%endc) ! bottom drainage from h2osfc @@ -295,7 +295,6 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f watsat => soilstate_inst%watsat_col , & ! Input: [real(r8) (:,:) ] volumetric soil water at saturation (porosity) bsw => soilstate_inst%bsw_col , & ! Input: [real(r8) (:,:) ] Clapp and Hornberger "b" hksat => soilstate_inst%hksat_col , & ! Input: [real(r8) (:,:) ] hydraulic conductivity at saturation (mm H2O /s) - eff_porosity => soilstate_inst%eff_porosity_col , & ! Output: [real(r8) (:,:) ] effective porosity = porosity - vol_ice qflx_sat_surf => surf_runoff_sat_inst%qflx_sat_surf_col, & ! Input: [real(r8) (:) ] surface runoff due to saturated surface (mm H2O /s) fsat => surf_runoff_sat_inst%fsat_col , & ! Input: [real(r8) (:) ] fractional area with water table at surface @@ -309,21 +308,12 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f top_max_moist => soilhydrology_inst%top_max_moist_col, & ! Input: [real(r8) (:) ] maximum soil moisture in top VIC layers top_ice => soilhydrology_inst%top_ice_col , & ! Input: [real(r8) (:) ] ice len in top VIC layers h2osfcflag => soilhydrology_inst%h2osfcflag , & ! Input: logical - icefrac => soilhydrology_inst%icefrac_col & ! Output: [real(r8) (:,:) ] fraction of ice + icefrac => soilhydrology_inst%icefrac_col & ! Input: [real(r8) (:,:) ] fraction of ice ) dtime = get_step_size() ! Infiltration into surface soil layer (minus the evaporation) - do j = 1,nlevsoi - do fc = 1, num_hydrologyc - c = filter_hydrologyc(fc) - ! Porosity of soil, partial volume of ice and liquid - vol_ice(c,j) = min(watsat(c,j), h2osoi_ice(c,j)/(dz(c,j)*denice)) - eff_porosity(c,j) = max(0.01_r8,watsat(c,j)-vol_ice(c,j)) - icefrac(c,j) = min(1._r8,vol_ice(c,j)/watsat(c,j)) - end do - end do do fc = 1, num_hydrologyc c = filter_hydrologyc(fc) From b5fb1f33ab522c4aa2d9d0d20224a9ec563cf444 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Thu, 20 Jul 2017 13:27:35 -0600 Subject: [PATCH 3/8] Add citation for vic method --- src/biogeophys/SurfRunoffSatMod.F90 | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/biogeophys/SurfRunoffSatMod.F90 b/src/biogeophys/SurfRunoffSatMod.F90 index 203bf82c04..ce342aabdb 100644 --- a/src/biogeophys/SurfRunoffSatMod.F90 +++ b/src/biogeophys/SurfRunoffSatMod.F90 @@ -331,6 +331,9 @@ subroutine ComputeFsatVic(bounds, num_hydrologyc, filter_hydrologyc, & ! !DESCRIPTION: ! Compute fsat using the VIC-based parameterization ! + ! Citation: Wood et al. 1992, "A land-surface hydrology parameterization with subgrid + ! variability for general circulation models", JGR 97(D3), 2717-2728. + ! ! !ARGUMENTS: type(bounds_type), intent(in) :: bounds integer, intent(in) :: num_hydrologyc ! number of column soil points in column filter From 93a099c5ba733728d90d43da541d3900fe9887c6 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Fri, 21 Jul 2017 12:22:45 -0600 Subject: [PATCH 4/8] Move calculation of qflx_in_soil and qflx_in_h2osfc out of Infiltration Point is: I want to start making Infiltration be just about calculation infiltration --- src/biogeophys/HydrologyNoDrainageMod.F90 | 5 +- src/biogeophys/SoilHydrologyMod.F90 | 107 +++++++++++++--------- src/biogeophys/WaterfluxType.F90 | 4 + 3 files changed, 71 insertions(+), 45 deletions(-) diff --git a/src/biogeophys/HydrologyNoDrainageMod.F90 b/src/biogeophys/HydrologyNoDrainageMod.F90 index 27340057e8..7df30532fe 100644 --- a/src/biogeophys/HydrologyNoDrainageMod.F90 +++ b/src/biogeophys/HydrologyNoDrainageMod.F90 @@ -61,7 +61,7 @@ subroutine HydrologyNoDrainage(bounds, & use SnowHydrologyMod , only : SnowCompaction, CombineSnowLayers, DivideSnowLayers, SnowCapping use SnowHydrologyMod , only : SnowWater, BuildSnowFilter use SoilHydrologyMod , only : CLMVICMap, SetSoilWaterFractions - use SoilHydrologyMod , only : SetQflxTopSoil, Infiltration, TotalSurfaceRunoff + use SoilHydrologyMod , only : SetQflxInputs, Infiltration, TotalSurfaceRunoff use SoilHydrologyMod , only : UpdateUrbanPonding use SoilHydrologyMod , only : WaterTable, PerchedWaterTable use SoilHydrologyMod , only : ThetaBasedWaterTable, RenewCondensation @@ -184,7 +184,8 @@ subroutine HydrologyNoDrainage(bounds, & bounds, num_hydrologyc, filter_hydrologyc, col, & soilhydrology_inst, soilstate_inst, waterflux_inst) - call SetQflxTopSoil(bounds, num_hydrologyc, filter_hydrologyc, waterflux_inst) + call SetQflxInputs(bounds, num_hydrologyc, filter_hydrologyc, & + waterflux_inst, surf_runoff_sat_inst, waterstate_inst) call Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, filter_urbanc,& energyflux_inst, soilhydrology_inst, soilstate_inst, surf_runoff_sat_inst, & diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index 4267d845ff..a2381a78b0 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -14,6 +14,7 @@ module SoilHydrologyMod use clm_varpar , only : nlevsoi, nlevgrnd, nlayer, nlayert use column_varcon , only : icol_roof, icol_sunwall, icol_shadewall use column_varcon , only : icol_road_imperv + use landunit_varcon , only : istsoil, istcrop use clm_time_manager , only : get_step_size use EnergyFluxType , only : energyflux_type use SoilHydrologyType , only : soilhydrology_type @@ -33,7 +34,7 @@ module SoilHydrologyMod ! !PUBLIC MEMBER FUNCTIONS: public :: SoilHydReadNML ! Read in the Soil hydrology namelist public :: SetSoilWaterFractions ! Set diagnostic variables related to the fraction of water and ice in each layer - public :: SetQflxTopSoil ! Set the flux of water into the soil from the top + public :: SetQflxInputs ! Set the flux of water into the soil from the top public :: Infiltration ! Calculate infiltration into surface soil layer public :: TotalSurfaceRunoff ! Calculate total surface runoff public :: UpdateUrbanPonding ! Update the state variable representing ponding on urban surfaces @@ -173,39 +174,81 @@ subroutine SetSoilWaterFractions(bounds, num_hydrologyc, filter_hydrologyc, & end subroutine SetSoilWaterFractions !----------------------------------------------------------------------- - subroutine SetQflxTopSoil(bounds, num_hydrologyc, filter_hydrologyc, & - waterflux_inst) + subroutine SetQflxInputs(bounds, num_hydrologyc, filter_hydrologyc, & + waterflux_inst, surf_runoff_sat_inst, waterstate_inst) ! ! !DESCRIPTION: - ! Set the flux of water into the soil from the top + ! Set various input fluxes of water ! ! !ARGUMENTS: - type(bounds_type) , intent(in) :: bounds - integer , intent(in) :: num_hydrologyc ! number of column soil points in column filter - integer , intent(in) :: filter_hydrologyc(:) ! column filter for soil points - type(waterflux_type) , intent(inout) :: waterflux_inst + type(bounds_type) , intent(in) :: bounds + integer , intent(in) :: num_hydrologyc ! number of column soil points in column filter + integer , intent(in) :: filter_hydrologyc(:) ! column filter for soil points + type(waterflux_type) , intent(inout) :: waterflux_inst + type(surf_runoff_sat_type) , intent(in) :: surf_runoff_sat_inst + type(waterstate_type) , intent(in) :: waterstate_inst ! ! !LOCAL VARIABLES: integer :: fc, c + real(r8) :: qflx_evap(bounds%begc:bounds%endc) ! local evaporation array + real(r8) :: fsno ! copy of frac_sno - character(len=*), parameter :: subname = 'SetQflxTopSoil' + character(len=*), parameter :: subname = 'SetQflxInputs' !----------------------------------------------------------------------- associate( & - qflx_top_soil => waterflux_inst%qflx_top_soil_col , & ! Output: [real(r8) (:)] net water input into soil from top (mm/s) - qflx_rain_plus_snomelt => waterflux_inst%qflx_rain_plus_snomelt_col , & ! Input: [real(r8) (:)] rain plus snow melt falling on the soil (mm/s) - qflx_snow_h2osfc => waterflux_inst%qflx_snow_h2osfc_col , & ! Input: [real(r8) (:)] snow falling on surface water (mm/s) - qflx_floodc => waterflux_inst%qflx_floodc_col & ! Input: [real(r8) (:)] column flux of flood water from RTM - ) + snl => col%snl , & ! Input: [integer (:) ] minus number of snow layers + + qflx_top_soil => waterflux_inst%qflx_top_soil_col , & ! Output: [real(r8) (:)] net water input into soil from top (mm/s) + qflx_in_soil => waterflux_inst%qflx_in_soil_col , & ! Output: [real(r8) (:)] surface input to soil (mm/s) + qflx_top_soil_to_h2osfc => waterflux_inst%qflx_top_soil_to_h2osfc_col , & ! Output: [real(r8) (:)] portion of qflx_top_soil going to h2osfc, minus evaporation (mm/s) + qflx_rain_plus_snomelt => waterflux_inst%qflx_rain_plus_snomelt_col , & ! Input: [real(r8) (:)] rain plus snow melt falling on the soil (mm/s) + qflx_snow_h2osfc => waterflux_inst%qflx_snow_h2osfc_col , & ! Input: [real(r8) (:)] snow falling on surface water (mm/s) + qflx_floodc => waterflux_inst%qflx_floodc_col , & ! Input: [real(r8) (:)] column flux of flood water from RTM + qflx_ev_soil => waterflux_inst%qflx_ev_soil_col , & ! Input: [real(r8) (:) ] evaporation flux from soil (W/m**2) [+ to atm] + qflx_evap_grnd => waterflux_inst%qflx_evap_grnd_col , & ! Input: [real(r8) (:) ] ground surface evaporation rate (mm H2O/s) [+] + qflx_ev_h2osfc => waterflux_inst%qflx_ev_h2osfc_col , & ! Input: [real(r8) (:) ] evaporation flux from h2osfc (W/m**2) [+ to atm] + + qflx_sat_surf => surf_runoff_sat_inst%qflx_sat_surf_col , & ! Input: [real(r8) (:) ] surface runoff due to saturated surface (mm H2O /s) + + frac_sno => waterstate_inst%frac_sno_eff_col , & ! Input: [real(r8) (:) ] fraction of ground covered by snow (0 to 1) + frac_h2osfc => waterstate_inst%frac_h2osfc_col & ! Input: [real(r8) (:) ] fraction of ground covered by surface water (0 to 1) + ) do fc = 1, num_hydrologyc c = filter_hydrologyc(fc) qflx_top_soil(c) = qflx_rain_plus_snomelt(c) + qflx_snow_h2osfc(c) + qflx_floodc(c) + + ! Partition surface inputs between soil and h2osfc + ! + ! This is only done for soil & crop landunits, because other + ! hydrologically-active landunits (in particular, urban pervious road) do not use + ! h2osfc. + if (lun%itype(col%landunit(c)) == istsoil .or. lun%itype(col%landunit(c))==istcrop) then + + ! explicitly use frac_sno=0 if snl=0 + if (snl(c) >= 0) then + fsno=0._r8 + ! if no snow layers, sublimation is removed from h2osoi_ice in drainage + qflx_evap(c)=qflx_evap_grnd(c) + else + fsno=frac_sno(c) + qflx_evap(c)=qflx_ev_soil(c) + endif + + qflx_in_soil(c) = (1._r8 - frac_h2osfc(c)) * (qflx_top_soil(c) - qflx_sat_surf(c)) + qflx_top_soil_to_h2osfc(c) = frac_h2osfc(c) * (qflx_top_soil(c) - qflx_sat_surf(c)) + + ! remove evaporation (snow treated in SnowHydrology) + qflx_in_soil(c) = qflx_in_soil(c) - (1.0_r8 - fsno - frac_h2osfc(c))*qflx_evap(c) + qflx_top_soil_to_h2osfc(c) = qflx_top_soil_to_h2osfc(c) - frac_h2osfc(c) * qflx_ev_h2osfc(c) + + end if end do end associate - end subroutine SetQflxTopSoil + end subroutine SetQflxInputs !----------------------------------------------------------------------- @@ -220,7 +263,6 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f use shr_const_mod , only : shr_const_pi use clm_varcon , only : denh2o, denice, roverg, wimp, pc, mu, tfrz use column_varcon , only : icol_roof, icol_road_imperv, icol_sunwall, icol_shadewall, icol_road_perv - use landunit_varcon , only : istsoil, istcrop ! ! !ARGUMENTS: type(bounds_type) , intent(in) :: bounds @@ -242,13 +284,10 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f real(r8) :: s1,su,v ! variable to calculate qinmax real(r8) :: qinmax ! maximum infiltration capacity (mm/s) real(r8) :: alpha_evap(bounds%begc:bounds%endc) ! fraction of total evap from h2osfc - real(r8) :: qflx_evap(bounds%begc:bounds%endc) ! local evaporation array real(r8) :: qflx_h2osfc_drain(bounds%begc:bounds%endc) ! bottom drainage from h2osfc - real(r8) :: qflx_in_h2osfc(bounds%begc:bounds%endc) ! surface input to h2osfc - real(r8) :: qflx_in_soil(bounds%begc:bounds%endc) ! surface input to soil + real(r8) :: qflx_in_h2osfc(bounds%begc:bounds%endc) ! net surface input to h2osfc real(r8) :: qflx_infl_excess(bounds%begc:bounds%endc) ! infiltration excess runoff -> h2osfc real(r8) :: frac_infclust ! fraction of submerged area that is connected - real(r8) :: fsno ! copy of frac_sno real(r8) :: k_wet ! linear reservoir coefficient for h2osfc real(r8) :: fac ! soil wetness of surface layer real(r8) :: psit ! negative potential of soil @@ -281,11 +320,11 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f h2osoi_ice => waterstate_inst%h2osoi_ice_col , & ! Input: [real(r8) (:,:) ] ice lens (kg/m2) h2osfc => waterstate_inst%h2osfc_col , & ! Output: [real(r8) (:) ] surface water (mm) - qflx_ev_soil => waterflux_inst%qflx_ev_soil_col , & ! Input: [real(r8) (:) ] evaporation flux from soil (W/m**2) [+ to atm] - qflx_evap_soi => waterflux_inst%qflx_evap_soi_col , & ! Input: [real(r8) (:) ] ground surface evaporation rate (mm H2O/s) [+] + qflx_in_soil => waterflux_inst%qflx_in_soil_col , & ! Input: [real(r8) (:) ] surface input to soil (mm/s) + qflx_top_soil_to_h2osfc => waterflux_inst%qflx_top_soil_to_h2osfc_col, & ! Input: [real(r8) (:)] portion of qflx_top_soil going to h2osfc, minus evaporation (mm/s) + qflx_ev_soil => waterflux_inst%qflx_ev_soil_col , & ! Input: [real(r8) (:) ] evaporation flux from soil (W/m**2) [+ to atm] qflx_evap_grnd => waterflux_inst%qflx_evap_grnd_col , & ! Input: [real(r8) (:) ] ground surface evaporation rate (mm H2O/s) [+] qflx_top_soil => waterflux_inst%qflx_top_soil_col , & ! Input: [real(r8) (:) ] net water input into soil from top (mm/s) - qflx_ev_h2osfc => waterflux_inst%qflx_ev_h2osfc_col , & ! Input: [real(r8) (:) ] evaporation flux from h2osfc (W/m**2) [+ to atm] qflx_infl_excess_surf => waterflux_inst%qflx_infl_excess_surf_col, & ! Output: [real(r8) (:) ] surface runoff due to infiltration excess (mm H2O /s) qflx_h2osfc_surf => waterflux_inst%qflx_h2osfc_surf_col , & ! Output: [real(r8) (:) ] surface water runoff (mm H2O /s) qflx_infl => waterflux_inst%qflx_infl_col , & ! Output: [real(r8) (:) ] infiltration (mm H2O /s) @@ -317,27 +356,8 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f do fc = 1, num_hydrologyc c = filter_hydrologyc(fc) - ! partition moisture fluxes between soil and h2osfc if (lun%itype(col%landunit(c)) == istsoil .or. lun%itype(col%landunit(c))==istcrop) then - ! explicitly use frac_sno=0 if snl=0 - if (snl(c) >= 0) then - fsno=0._r8 - ! if no snow layers, sublimation is removed from h2osoi_ice in drainage - qflx_evap(c)=qflx_evap_grnd(c) - else - fsno=frac_sno(c) - qflx_evap(c)=qflx_ev_soil(c) - endif - - !1. partition surface inputs between soil and h2osfc - qflx_in_soil(c) = (1._r8 - frac_h2osfc(c)) * (qflx_top_soil(c) - qflx_sat_surf(c)) - qflx_in_h2osfc(c) = frac_h2osfc(c) * (qflx_top_soil(c) - qflx_sat_surf(c)) - - !2. remove evaporation (snow treated in SnowHydrology) - qflx_in_soil(c) = qflx_in_soil(c) - (1.0_r8 - fsno - frac_h2osfc(c))*qflx_evap(c) - qflx_in_h2osfc(c) = qflx_in_h2osfc(c) - frac_h2osfc(c) * qflx_ev_h2osfc(c) - !3. determine maximum infiltration rate if (use_vichydro) then top_icefrac = min(1._r8,top_ice(c)/top_max_moist(c)) @@ -364,10 +384,11 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f !4. soil infiltration and h2osfc "run-on" qflx_infl(c) = qflx_in_soil(c) - qflx_infl_excess(c) if (h2osfcflag /= 0) then - qflx_in_h2osfc(c) = qflx_in_h2osfc(c) + qflx_infl_excess(c) + qflx_in_h2osfc(c) = qflx_top_soil_to_h2osfc(c) + qflx_infl_excess(c) qflx_infl_excess_surf(c) = 0._r8 else ! No h2osfc pool, so qflx_infl_excess goes directly to surface runoff + qflx_in_h2osfc(c) = qflx_top_soil_to_h2osfc(c) qflx_infl_excess_surf(c) = qflx_infl_excess(c) end if diff --git a/src/biogeophys/WaterfluxType.F90 b/src/biogeophys/WaterfluxType.F90 index e18065ec31..33c84bf908 100644 --- a/src/biogeophys/WaterfluxType.F90 +++ b/src/biogeophys/WaterfluxType.F90 @@ -80,6 +80,8 @@ module WaterfluxType real(r8), pointer :: qflx_drain_col (:) ! col sub-surface runoff (mm H2O /s) real(r8), pointer :: qflx_rain_plus_snomelt_col(:) ! col rain plus snow melt falling on the soil (mm/s) real(r8), pointer :: qflx_top_soil_col (:) ! col net water input into soil from top (mm/s) + real(r8), pointer :: qflx_in_soil_col (:) ! col surface input to soil (mm/s) + real(r8), pointer :: qflx_top_soil_to_h2osfc_col(:) ! col portion of qflx_top_soil going to h2osfc, minus evaporation (mm/s) real(r8), pointer :: qflx_h2osfc_to_ice_col (:) ! col conversion of h2osfc to ice real(r8), pointer :: qflx_snow_h2osfc_col (:) ! col snow falling on surface water real(r8), pointer :: qflx_drain_perched_col (:) ! col sub-surface runoff from perched wt (mm H2O /s) @@ -222,6 +224,8 @@ subroutine InitAllocate(this, bounds) allocate(this%qflx_drain_col (begc:endc)) ; this%qflx_drain_col (:) = nan allocate(this%qflx_rain_plus_snomelt_col(begc:endc)) ; this%qflx_rain_plus_snomelt_col(:) = nan allocate(this%qflx_top_soil_col (begc:endc)) ; this%qflx_top_soil_col (:) = nan + allocate(this%qflx_in_soil_col (begc:endc)) ; this%qflx_in_soil_col (:) = nan + allocate(this%qflx_top_soil_to_h2osfc_col(begc:endc)) ; this%qflx_top_soil_to_h2osfc_col(:) = nan allocate(this%qflx_h2osfc_to_ice_col (begc:endc)) ; this%qflx_h2osfc_to_ice_col (:) = nan allocate(this%qflx_infl_excess_surf_col(begc:endc)) ; this%qflx_infl_excess_surf_col(:) = nan allocate(this%qflx_h2osfc_surf_col (begc:endc)) ; this%qflx_h2osfc_surf_col (:) = nan From c0240c9ccf442c24abd81704cd101f638eac6cad Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Tue, 15 Aug 2017 08:30:17 -0600 Subject: [PATCH 5/8] Separate calculation of infiltraation excess runoff into its own module --- src/biogeophys/HydrologyNoDrainageMod.F90 | 16 +- .../InfiltrationExcessRunoffMod.F90 | 342 ++++++++++++++++++ src/biogeophys/SoilHydrologyMod.F90 | 81 +---- src/main/clm_driver.F90 | 1 + src/main/clm_instMod.F90 | 3 + 5 files changed, 369 insertions(+), 74 deletions(-) create mode 100644 src/biogeophys/InfiltrationExcessRunoffMod.F90 diff --git a/src/biogeophys/HydrologyNoDrainageMod.F90 b/src/biogeophys/HydrologyNoDrainageMod.F90 index 7df30532fe..ef84b76e98 100644 --- a/src/biogeophys/HydrologyNoDrainageMod.F90 +++ b/src/biogeophys/HydrologyNoDrainageMod.F90 @@ -17,6 +17,7 @@ Module HydrologyNoDrainageMod use SoilHydrologyType , only : soilhydrology_type use SoilStateType , only : soilstate_type use SurfRunoffSatMod , only : surf_runoff_sat_type + use InfiltrationExcessRunoffMod, only : infiltration_excess_runoff_type use WaterfluxType , only : waterflux_type use WaterstateType , only : waterstate_type use CanopyStateType , only : canopystate_type @@ -43,8 +44,8 @@ subroutine HydrologyNoDrainage(bounds, & clm_fates, & atm2lnd_inst, soilstate_inst, energyflux_inst, temperature_inst, & waterflux_inst, waterstate_inst, & - soilhydrology_inst, surf_runoff_sat_inst, aerosol_inst, & - canopystate_inst, soil_water_retention_curve) + soilhydrology_inst, surf_runoff_sat_inst, infiltration_excess_runoff_inst, & + aerosol_inst, canopystate_inst, soil_water_retention_curve) ! ! !DESCRIPTION: ! This is the main subroutine to execute the calculation of soil/snow @@ -93,6 +94,7 @@ subroutine HydrologyNoDrainage(bounds, & type(aerosol_type) , intent(inout) :: aerosol_inst type(soilhydrology_type) , intent(inout) :: soilhydrology_inst type(surf_runoff_sat_type), intent(inout) :: surf_runoff_sat_inst + type(infiltration_excess_runoff_type), intent(inout) :: infiltration_excess_runoff_inst type(canopystate_type) , intent(inout) :: canopystate_inst class(soil_water_retention_curve_type), intent(in) :: soil_water_retention_curve ! @@ -187,9 +189,15 @@ subroutine HydrologyNoDrainage(bounds, & call SetQflxInputs(bounds, num_hydrologyc, filter_hydrologyc, & waterflux_inst, surf_runoff_sat_inst, waterstate_inst) + call infiltration_excess_runoff_inst%InfiltrationExcessRunoff( & + bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, soilstate_inst, surf_runoff_sat_inst, waterflux_inst, & + waterstate_inst) + call Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, filter_urbanc,& - energyflux_inst, soilhydrology_inst, soilstate_inst, surf_runoff_sat_inst, & - temperature_inst, waterflux_inst, waterstate_inst) + infiltration_excess_runoff_inst, & + energyflux_inst, soilhydrology_inst, surf_runoff_sat_inst, & + waterflux_inst, waterstate_inst) call TotalSurfaceRunoff(bounds, num_hydrologyc, filter_hydrologyc, & num_urbanc, filter_urbanc, & diff --git a/src/biogeophys/InfiltrationExcessRunoffMod.F90 b/src/biogeophys/InfiltrationExcessRunoffMod.F90 new file mode 100644 index 0000000000..4d969d8abe --- /dev/null +++ b/src/biogeophys/InfiltrationExcessRunoffMod.F90 @@ -0,0 +1,342 @@ +module InfiltrationExcessRunoffMod + + !----------------------------------------------------------------------- + ! !DESCRIPTION: + ! Type and associated routines for computing infiltration excess runoff and related + ! variables + ! + ! !USES: +#include "shr_assert.h" + use shr_kind_mod , only : r8 => shr_kind_r8 + use shr_log_mod , only : errMsg => shr_log_errMsg + use decompMod , only : bounds_type + use abortutils , only : endrun + use clm_varctl , only : iulog, use_vichydro + use clm_varcon , only : spval, e_ice + use clm_time_manager , only : get_step_size + use SoilHydrologyType, only : soilhydrology_type + use SoilStateType , only : soilstate_type + use SurfRunoffSatMod , only : surf_runoff_sat_type + use WaterfluxType , only : waterflux_type + use WaterstateType , only : waterstate_type + + implicit none + save + private + + ! !PUBLIC TYPES: + + type, public :: infiltration_excess_runoff_type + private + ! Public data members + ! Note: these should be treated as read-only by other modules + + ! These are valid within the hydrology filter. + ! + ! Both of these give averages over the entire column. However, qinmax is implicitly + ! 0 over the fraction of the column given by fsat, and qflx_infl_excess is + ! implicitly 0 over both fsat and frac_h2osfc. + real(r8), pointer, public :: qinmax_col(:) ! maximum infiltration capacity (mm H2O /s) + real(r8), pointer, public :: qflx_infl_excess_col(:) ! infiltration excess runoff (mm H2O /s) + + ! Private data members + integer :: qinmax_method + contains + ! Public routines + procedure, public :: Init + + procedure, public :: InfiltrationExcessRunoff ! Calculate surface runoff due to infiltration excess + + ! Private routines + procedure, private :: InitAllocate + procedure, private :: InitHistory + procedure, private :: InitCold + + procedure, private, nopass :: ComputeQinmaxHksat + procedure, private, nopass :: ComputeQinmaxVic + end type infiltration_excess_runoff_type + + ! !PRIVATE DATA MEMBERS: + + integer, parameter :: QINMAX_METHOD_HKSAT = 1 + integer, parameter :: QINMAX_METHOD_VIC = 2 + + character(len=*), parameter, private :: sourcefile = & + __FILE__ + +contains + + ! ======================================================================== + ! Infrastructure routines + ! ======================================================================== + + !----------------------------------------------------------------------- + subroutine Init(this, bounds) + ! + ! !DESCRIPTION: + ! Initialize this infiltration_excess_runoff_type object + ! + ! !ARGUMENTS: + class(infiltration_excess_runoff_type), intent(inout) :: this + type(bounds_type), intent(in) :: bounds + ! + ! !LOCAL VARIABLES: + + character(len=*), parameter :: subname = 'Init' + !----------------------------------------------------------------------- + + call this%InitAllocate(bounds) + call this%InitHistory(bounds) + call this%InitCold(bounds) + + end subroutine Init + + !----------------------------------------------------------------------- + subroutine InitAllocate(this, bounds) + ! + ! !DESCRIPTION: + ! Allocate memory for this infiltration_excess_runoff_type object + ! + ! !USES: + use shr_infnan_mod , only : nan => shr_infnan_nan, assignment(=) + ! + ! !ARGUMENTS: + class(infiltration_excess_runoff_type), intent(inout) :: this + type(bounds_type), intent(in) :: bounds + ! + ! !LOCAL VARIABLES: + integer :: begc, endc + + character(len=*), parameter :: subname = 'InitAllocate' + !----------------------------------------------------------------------- + + begc = bounds%begc; endc= bounds%endc + allocate(this%qinmax_col (begc:endc)); this%qinmax_col (:) = nan + allocate(this%qflx_infl_excess_col(begc:endc)); this%qflx_infl_excess_col(:) = nan + + end subroutine InitAllocate + + !----------------------------------------------------------------------- + subroutine InitHistory(this, bounds) + ! + ! !DESCRIPTION: + ! Initialize infiltration_excess_runoff_type history variables + ! + ! !USES: + use histFileMod , only : hist_addfld1d + ! + ! !ARGUMENTS: + class(infiltration_excess_runoff_type), intent(inout) :: this + type(bounds_type), intent(in) :: bounds + ! + ! !LOCAL VARIABLES: + + character(len=*), parameter :: subname = 'InitHistory' + !----------------------------------------------------------------------- + + ! Nothing to do for now + + end subroutine InitHistory + + !----------------------------------------------------------------------- + subroutine InitCold(this, bounds) + ! + ! !DESCRIPTION: + ! Perform cold-start initialization for infiltration_excess_runoff_type + ! + ! !ARGUMENTS: + class(infiltration_excess_runoff_type), intent(inout) :: this + type(bounds_type), intent(in) :: bounds + ! + ! !LOCAL VARIABLES: + + character(len=*), parameter :: subname = 'InitCold' + !----------------------------------------------------------------------- + + ! TODO(wjs, 2017-08-14) We'll read qinmax_method from namelist. + if (use_vichydro) then + this%qinmax_method = QINMAX_METHOD_VIC + else + this%qinmax_method = QINMAX_METHOD_HKSAT + end if + + end subroutine InitCold + + ! ======================================================================== + ! Science routines + ! ======================================================================== + + !----------------------------------------------------------------------- + subroutine InfiltrationExcessRunoff(this, bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, soilstate_inst, surf_runoff_sat_inst, waterflux_inst, waterstate_inst) + ! + ! !DESCRIPTION: + ! Calculate surface runoff due to infiltration excess + ! + ! !ARGUMENTS: + class(infiltration_excess_runoff_type) , intent(inout) :: this + type(bounds_type) , intent(in) :: bounds + integer , intent(in) :: num_hydrologyc ! number of column soil points in column filter + integer , intent(in) :: filter_hydrologyc(:) ! column filter for soil points + type(soilhydrology_type) , intent(in) :: soilhydrology_inst + type(soilstate_type) , intent(in) :: soilstate_inst + type(surf_runoff_sat_type) , intent(in) :: surf_runoff_sat_inst + type(waterflux_type) , intent(in) :: waterflux_inst + type(waterstate_type) , intent(in) :: waterstate_inst + ! + ! !LOCAL VARIABLES: + integer :: fc, c + real(r8) :: qinmax_on_unsaturated_area(bounds%begc:bounds%endc) ! maximum infiltration capacity on the unsaturated fraction of the column (mm H2O /s) + + character(len=*), parameter :: subname = 'InfiltrationExcessRunoff' + !----------------------------------------------------------------------- + + associate( & + qinmax => this%qinmax_col , & ! Output: [real(r8) (:) ] maximum infiltration capacity (mm H2O /s) + qflx_infl_excess => this%qflx_infl_excess_col , & ! Output: [real(r8) (:) ] infiltration excess runoff (mm H2O /s) + + fsat => surf_runoff_sat_inst%fsat_col , & ! Input: [real(r8) (:) ] fractional area with water table at surface + + qflx_in_soil => waterflux_inst%qflx_in_soil_col , & ! Input: [real(r8) (:) ] surface input to soil (mm/s) + + frac_h2osfc => waterstate_inst%frac_h2osfc_col & ! Input: [real(r8) (:) ] fraction of ground covered by surface water (0 to 1) + ) + + select case (this%qinmax_method) + case (QINMAX_METHOD_HKSAT) + call this%ComputeQinmaxHksat(bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, soilstate_inst, & + qinmax_on_unsaturated_area = qinmax_on_unsaturated_area(bounds%begc:bounds%endc)) + case (QINMAX_METHOD_VIC) + call this%ComputeQinmaxVic(bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, & + fsat = fsat(bounds%begc:bounds%endc), & + qflx_in_soil = qflx_in_soil(bounds%begc:bounds%endc), & + qinmax_on_unsaturated_area = qinmax_on_unsaturated_area(bounds%begc:bounds%endc)) + case default + write(iulog,*) subname//' ERROR: Unrecognized qinmax_method: ', this%qinmax_method + call endrun(subname//' ERROR: Unrecognized qinmax_method') + end select + + do fc = 1, num_hydrologyc + c = filter_hydrologyc(fc) + qinmax(c) = (1._r8 - fsat(c)) * qinmax_on_unsaturated_area(c) + qflx_infl_excess(c) = max(0._r8, & + (qflx_in_soil(c) - (1.0_r8 - frac_h2osfc(c))*qinmax(c))) + end do + + end associate + + end subroutine InfiltrationExcessRunoff + + !----------------------------------------------------------------------- + subroutine ComputeQinmaxHksat(bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, soilstate_inst, & + qinmax_on_unsaturated_area) + ! + ! !DESCRIPTION: + ! Compute qinmax using a parameterization based on hksat + ! + ! This is the CLM default parameterization + ! + ! !ARGUMENTS: + type(bounds_type), intent(in) :: bounds + integer, intent(in) :: num_hydrologyc ! number of column soil points in column filter + integer, intent(in) :: filter_hydrologyc(:) ! column filter for soil points + type(soilhydrology_type) , intent(in) :: soilhydrology_inst + type(soilstate_type), intent(in) :: soilstate_inst + real(r8), intent(inout) :: qinmax_on_unsaturated_area( bounds%begc: ) ! maximum infiltration capacity on the unsaturated fraction of the column (mm H2O /s) + ! + ! !LOCAL VARIABLES: + integer :: fc, c + + character(len=*), parameter :: subname = 'ComputeQinmaxHksat' + !----------------------------------------------------------------------- + + SHR_ASSERT_ALL((ubound(qinmax_on_unsaturated_area) == (/bounds%endc/)), errMsg(sourcefile, __LINE__)) + + associate( & + icefrac => soilhydrology_inst%icefrac_col , & ! Input: [real(r8) (:,:) ] fraction of ice + + hksat => soilstate_inst%hksat_col & ! Input: [real(r8) (:,:) ] hydraulic conductivity at saturation (mm H2O /s) + ) + + do fc = 1, num_hydrologyc + c = filter_hydrologyc(fc) + qinmax_on_unsaturated_area(c) = minval(10._r8**(-e_ice*(icefrac(c,1:3)))*hksat(c,1:3)) + end do + + end associate + + end subroutine ComputeQinmaxHksat + + !----------------------------------------------------------------------- + subroutine ComputeQinmaxVic(bounds, num_hydrologyc, filter_hydrologyc, & + soilhydrology_inst, & + fsat, qflx_in_soil, qinmax_on_unsaturated_area) + ! + ! !DESCRIPTION: + ! Compute qinmax using the VIC parameterization + ! + ! Citation: Wood et al. 1992, "A land-surface hydrology parameterization with subgrid + ! variability for general circulation models", JGR 97(D3), 2717-2728. + ! + ! !ARGUMENTS: + type(bounds_type), intent(in) :: bounds + integer, intent(in) :: num_hydrologyc ! number of column soil points in column filter + integer, intent(in) :: filter_hydrologyc(:) ! column filter for soil points + type(soilhydrology_type) , intent(in) :: soilhydrology_inst + real(r8) , intent(in) :: fsat( bounds%begc: ) ! fractional area with water table at surface + real(r8) , intent(in) :: qflx_in_soil( bounds%begc: ) ! surface input to soil (mm/s) + real(r8) , intent(inout) :: qinmax_on_unsaturated_area( bounds%begc: ) ! maximum infiltration capacity on the unsaturated fraction of the column (mm H2O /s) + ! + ! !LOCAL VARIABLES: + integer :: fc, c + real(r8) :: dtime ! land model time step (sec) + real(r8) :: top_icefrac ! ice fraction in top VIC layers + real(r8) :: max_infil ! max infiltration capacity in VIC (mm) + real(r8) :: i_0 ! average soil moisture in top VIC layers (mm) + real(r8) :: rsurf_vic ! VIC surface runoff + real(r8) :: basis ! variable soil moisture holding capacity in top VIC layers for runoff calculation + + character(len=*), parameter :: subname = 'ComputeQinmaxVic' + !----------------------------------------------------------------------- + + SHR_ASSERT_ALL((ubound(fsat) == (/bounds%endc/)), errMsg(sourcefile, __LINE__)) + SHR_ASSERT_ALL((ubound(qflx_in_soil) == (/bounds%endc/)), errMsg(sourcefile, __LINE__)) + SHR_ASSERT_ALL((ubound(qinmax_on_unsaturated_area) == (/bounds%endc/)), errMsg(sourcefile, __LINE__)) + + associate( & + top_max_moist => soilhydrology_inst%top_max_moist_col, & ! Input: [real(r8) (:) ] maximum soil moisture in top VIC layers + top_moist => soilhydrology_inst%top_moist_col , & ! Input: [real(r8) (:) ] soil moisture in top VIC layers + top_ice => soilhydrology_inst%top_ice_col , & ! Input: [real(r8) (:) ] ice len in top VIC layers + b_infil => soilhydrology_inst%b_infil_col & ! Input: [real(r8) (:) ] VIC b infiltration parameter + ) + + dtime = get_step_size() + + do fc = 1, num_hydrologyc + c = filter_hydrologyc(fc) + top_icefrac = min(1._r8,top_ice(c)/top_max_moist(c)) + max_infil = (1._r8+b_infil(c)) * top_max_moist(c) + i_0 = max_infil * (1._r8 - (1._r8 - fsat(c))**(1._r8/b_infil(c))) + if(qflx_in_soil(c) <= 0._r8) then + rsurf_vic = 0._r8 + else if(max_infil <= 0._r8) then + rsurf_vic = qflx_in_soil(c) + else if((i_0 + qflx_in_soil(c)*dtime) > max_infil) then !(Eq.(3a) Wood et al. 1992) + rsurf_vic = (qflx_in_soil(c)*dtime - top_max_moist(c) + top_moist(c))/dtime + else !(Eq.(3b) Wood et al. 1992) + basis = 1._r8 - (i_0 + qflx_in_soil(c)*dtime)/max_infil + rsurf_vic = (qflx_in_soil(c)*dtime - top_max_moist(c) + top_moist(c) & + + top_max_moist(c) * basis**(1._r8 + b_infil(c)))/dtime + end if + rsurf_vic = min(qflx_in_soil(c), rsurf_vic) + qinmax_on_unsaturated_area(c) = (1._r8 - fsat(c)) * 10._r8**(-e_ice*top_icefrac)*(qflx_in_soil(c) - rsurf_vic) + end do + + end associate + + end subroutine ComputeQinmaxVic + +end module InfiltrationExcessRunoffMod diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index a2381a78b0..96a729e7b4 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -17,6 +17,7 @@ module SoilHydrologyMod use landunit_varcon , only : istsoil, istcrop use clm_time_manager , only : get_step_size use EnergyFluxType , only : energyflux_type + use InfiltrationExcessRunoffMod, only : infiltration_excess_runoff_type use SoilHydrologyType , only : soilhydrology_type use SoilStateType , only : soilstate_type use SurfRunoffSatMod , only : surf_runoff_sat_type @@ -253,8 +254,9 @@ end subroutine SetQflxInputs !----------------------------------------------------------------------- subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, filter_urbanc, & - energyflux_inst, soilhydrology_inst, soilstate_inst, surf_runoff_sat_inst, & - temperature_inst, waterflux_inst, waterstate_inst) + infiltration_excess_runoff_inst, & + energyflux_inst, soilhydrology_inst, surf_runoff_sat_inst, & + waterflux_inst, waterstate_inst) ! ! !DESCRIPTION: ! Calculate infiltration into surface soil layer (minus the evaporation) @@ -270,54 +272,31 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f integer , intent(in) :: filter_hydrologyc(:) ! column filter for soil points integer , intent(in) :: num_urbanc ! number of column urban points in column filter integer , intent(in) :: filter_urbanc(:) ! column filter for urban points - type(energyflux_type) , intent(in) :: energyflux_inst + type(infiltration_excess_runoff_type), intent(in) :: infiltration_excess_runoff_inst + type(energyflux_type) , intent(in) :: energyflux_inst type(soilhydrology_type) , intent(in) :: soilhydrology_inst - type(soilstate_type) , intent(in) :: soilstate_inst type(surf_runoff_sat_type), intent(in) :: surf_runoff_sat_inst - type(temperature_type) , intent(in) :: temperature_inst type(waterstate_type) , intent(inout) :: waterstate_inst type(waterflux_type) , intent(inout) :: waterflux_inst ! ! !LOCAL VARIABLES: integer :: c,l,fc ! indices real(r8) :: dtime ! land model time step (sec) - real(r8) :: s1,su,v ! variable to calculate qinmax - real(r8) :: qinmax ! maximum infiltration capacity (mm/s) - real(r8) :: alpha_evap(bounds%begc:bounds%endc) ! fraction of total evap from h2osfc real(r8) :: qflx_h2osfc_drain(bounds%begc:bounds%endc) ! bottom drainage from h2osfc real(r8) :: qflx_in_h2osfc(bounds%begc:bounds%endc) ! net surface input to h2osfc - real(r8) :: qflx_infl_excess(bounds%begc:bounds%endc) ! infiltration excess runoff -> h2osfc real(r8) :: frac_infclust ! fraction of submerged area that is connected real(r8) :: k_wet ! linear reservoir coefficient for h2osfc - real(r8) :: fac ! soil wetness of surface layer - real(r8) :: psit ! negative potential of soil - real(r8) :: hr ! relative humidity - real(r8) :: wx ! partial volume of ice and water of surface layer - real(r8) :: z_avg - real(r8) :: rho_avg - real(r8) :: fmelt - real(r8) :: f_sno - real(r8) :: imped - real(r8) :: d - real(r8) :: h2osoi_vol - real(r8) :: basis ! temporary, variable soil moisture holding capacity - ! in top VIC layers for runoff calculation - real(r8) :: rsurf_vic ! temp VIC surface runoff - real(r8) :: i_0(bounds%begc:bounds%endc) ! average soil moisture in top VIC layers (mm) - real(r8) :: max_infil(bounds%begc:bounds%endc) ! max infiltration capacity in VIC (mm) - real(r8) :: top_icefrac ! temporary, ice fraction in top VIC layers !----------------------------------------------------------------------- associate( & snl => col%snl , & ! Input: [integer (:) ] minus number of snow layers - dz => col%dz , & ! Input: [real(r8) (:,:) ] layer depth (m) - t_soisno => temperature_inst%t_soisno_col , & ! Input: [real(r8) (:,:) ] soil temperature (Kelvin) + qflx_infl_excess => infiltration_excess_runoff_inst%qflx_infl_excess_col , & ! Input: [real(r8) (:)] infiltration excess runoff (mm H2O /s) + qinmax => infiltration_excess_runoff_inst%qinmax_col , & ! Input: [real(r8) (:)] maximum infiltration capacity (mm H2O /s) frac_h2osfc => waterstate_inst%frac_h2osfc_col , & ! Input: [real(r8) (:) ] fraction of ground covered by surface water (0 to 1) - frac_h2osfc_nosnow => waterstate_inst%frac_h2osfc_nosnow_col, & ! Output: [real(r8) (:) ] col fractional area with surface water greater than zero (if no snow present) + frac_h2osfc_nosnow => waterstate_inst%frac_h2osfc_nosnow_col, & ! Input: [real(r8) (:) ] col fractional area with surface water greater than zero (if no snow present) frac_sno => waterstate_inst%frac_sno_eff_col , & ! Input: [real(r8) (:) ] fraction of ground covered by snow (0 to 1) - h2osoi_ice => waterstate_inst%h2osoi_ice_col , & ! Input: [real(r8) (:,:) ] ice lens (kg/m2) h2osfc => waterstate_inst%h2osfc_col , & ! Output: [real(r8) (:) ] surface water (mm) qflx_in_soil => waterflux_inst%qflx_in_soil_col , & ! Input: [real(r8) (:) ] surface input to soil (mm/s) @@ -329,25 +308,10 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f qflx_h2osfc_surf => waterflux_inst%qflx_h2osfc_surf_col , & ! Output: [real(r8) (:) ] surface water runoff (mm H2O /s) qflx_infl => waterflux_inst%qflx_infl_col , & ! Output: [real(r8) (:) ] infiltration (mm H2O /s) - smpmin => soilstate_inst%smpmin_col , & ! Input: [real(r8) (:) ] restriction for min of soil potential (mm) - sucsat => soilstate_inst%sucsat_col , & ! Input: [real(r8) (:,:) ] minimum soil suction (mm) - watsat => soilstate_inst%watsat_col , & ! Input: [real(r8) (:,:) ] volumetric soil water at saturation (porosity) - bsw => soilstate_inst%bsw_col , & ! Input: [real(r8) (:,:) ] Clapp and Hornberger "b" - hksat => soilstate_inst%hksat_col , & ! Input: [real(r8) (:,:) ] hydraulic conductivity at saturation (mm H2O /s) - qflx_sat_surf => surf_runoff_sat_inst%qflx_sat_surf_col, & ! Input: [real(r8) (:) ] surface runoff due to saturated surface (mm H2O /s) - fsat => surf_runoff_sat_inst%fsat_col , & ! Input: [real(r8) (:) ] fractional area with water table at surface h2osfc_thresh => soilhydrology_inst%h2osfc_thresh_col, & ! Input: [real(r8) (:) ] level at which h2osfc "percolates" - zwt => soilhydrology_inst%zwt_col , & ! Input: [real(r8) (:) ] water table depth (m) - zwt_perched => soilhydrology_inst%zwt_perched_col , & ! Input: [real(r8) (:) ] perched water table depth (m) - b_infil => soilhydrology_inst%b_infil_col , & ! Input: [real(r8) (:) ] VIC b infiltration parameter - frost_table => soilhydrology_inst%frost_table_col , & ! Input: [real(r8) (:) ] frost table depth (m) - top_moist => soilhydrology_inst%top_moist_col , & ! Input: [real(r8) (:) ] soil moisture in top VIC layers - top_max_moist => soilhydrology_inst%top_max_moist_col, & ! Input: [real(r8) (:) ] maximum soil moisture in top VIC layers - top_ice => soilhydrology_inst%top_ice_col , & ! Input: [real(r8) (:) ] ice len in top VIC layers - h2osfcflag => soilhydrology_inst%h2osfcflag , & ! Input: logical - icefrac => soilhydrology_inst%icefrac_col & ! Input: [real(r8) (:,:) ] fraction of ice + h2osfcflag => soilhydrology_inst%h2osfcflag & ! Input: logical ) dtime = get_step_size() @@ -358,29 +322,6 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f c = filter_hydrologyc(fc) if (lun%itype(col%landunit(c)) == istsoil .or. lun%itype(col%landunit(c))==istcrop) then - !3. determine maximum infiltration rate - if (use_vichydro) then - top_icefrac = min(1._r8,top_ice(c)/top_max_moist(c)) - max_infil(c) = (1._r8+b_infil(c)) * top_max_moist(c) - i_0(c) = max_infil(c) * (1._r8 - (1._r8 - fsat(c))**(1._r8/b_infil(c))) - if(qflx_in_soil(c) <= 0._r8) then - rsurf_vic = 0._r8 - else if(max_infil(c) <= 0._r8) then - rsurf_vic = qflx_in_soil(c) - else if((i_0(c) + qflx_in_soil(c)*dtime) > max_infil(c)) then !(Eq.(3a) Wood et al. 1992) - rsurf_vic = (qflx_in_soil(c)*dtime - top_max_moist(c) + top_moist(c))/dtime - else !(Eq.(3b) Wood et al. 1992) - basis = 1._r8 - (i_0(c) + qflx_in_soil(c)*dtime)/max_infil(c) - rsurf_vic = (qflx_in_soil(c)*dtime - top_max_moist(c) + top_moist(c) & - + top_max_moist(c) * basis**(1._r8 + b_infil(c)))/dtime - end if - rsurf_vic = min(qflx_in_soil(c), rsurf_vic) - qinmax = (1._r8 - fsat(c)) * 10._r8**(-e_ice*top_icefrac)*(qflx_in_soil(c) - rsurf_vic) - else - qinmax=(1._r8 - fsat(c)) * minval(10._r8**(-e_ice*(icefrac(c,1:3)))*hksat(c,1:3)) - end if - qflx_infl_excess(c) = max(0._r8,qflx_in_soil(c) - (1.0_r8 - frac_h2osfc(c))*qinmax) - !4. soil infiltration and h2osfc "run-on" qflx_infl(c) = qflx_in_soil(c) - qflx_infl_excess(c) if (h2osfcflag /= 0) then @@ -427,7 +368,7 @@ subroutine Infiltration(bounds, num_hydrologyc, filter_hydrologyc, num_urbanc, f h2osfc(c) = 0.0 qflx_h2osfc_drain(c)= 0._r8 else - qflx_h2osfc_drain(c)=min(frac_h2osfc(c)*qinmax,h2osfc(c)/dtime) + qflx_h2osfc_drain(c)=min(frac_h2osfc(c)*qinmax(c),h2osfc(c)/dtime) endif if(h2osfcflag==0) then diff --git a/src/main/clm_driver.F90 b/src/main/clm_driver.F90 index 2e4599aa86..0b23e5d572 100644 --- a/src/main/clm_driver.F90 +++ b/src/main/clm_driver.F90 @@ -697,6 +697,7 @@ subroutine clm_drv(doalb, nextsw_cday, declinp1, declin, rstwr, nlend, rdate, ro clm_fates, & atm2lnd_inst, soilstate_inst, energyflux_inst, temperature_inst, & waterflux_inst, waterstate_inst, soilhydrology_inst, surf_runoff_sat_inst, & + infiltration_excess_runoff_inst, & aerosol_inst, canopystate_inst, soil_water_retention_curve) ! The following needs to be done after HydrologyNoDrainage (because it needs diff --git a/src/main/clm_instMod.F90 b/src/main/clm_instMod.F90 index 66c71daa5b..a683f02e8a 100644 --- a/src/main/clm_instMod.F90 +++ b/src/main/clm_instMod.F90 @@ -44,6 +44,7 @@ module clm_instMod use EnergyFluxType , only : energyflux_type use FrictionVelocityMod , only : frictionvel_type use GlacierSurfaceMassBalanceMod , only : glacier_smb_type + use InfiltrationExcessRunoffMod , only : infiltration_excess_runoff_type use IrrigationMod , only : irrigation_type use LakeStateType , only : lakestate_type use OzoneBaseMod , only : ozone_base_type @@ -95,6 +96,7 @@ module clm_instMod type(energyflux_type) :: energyflux_inst type(frictionvel_type) :: frictionvel_inst type(glacier_smb_type) :: glacier_smb_inst + type(infiltration_excess_runoff_type) :: infiltration_excess_runoff_inst type(irrigation_type) :: irrigation_inst type(lakestate_type) :: lakestate_inst class(ozone_base_type), allocatable :: ozone_inst @@ -311,6 +313,7 @@ subroutine clm_instInit(bounds) call SoilHydrologyInitTimeConst(bounds, soilhydrology_inst) ! sets time constant properties call surf_runoff_sat_inst%Init(bounds) + call infiltration_excess_runoff_inst%Init(bounds) call solarabs_inst%Init(bounds) From f2612f1cc429b5b885be14dbb261da45433ba110 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Tue, 15 Aug 2017 12:55:57 -0600 Subject: [PATCH 6/8] Temporary kludgey fix to fix runtime error This is needed to avoid a FPE (or something like that) in this line of code: qflx_infl_excess(c) = max(0._r8, & (qflx_in_soil(c) - (1.0_r8 - frac_h2osfc(c))*qinmax(c))) A more robust fix would be to set qflx_in_soil appropriately for all hydrologically-active columns (including urban hydrology columns). --- src/biogeophys/SoilHydrologyMod.F90 | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index 96a729e7b4..cb3513d7c3 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -244,6 +244,10 @@ subroutine SetQflxInputs(bounds, num_hydrologyc, filter_hydrologyc, & qflx_in_soil(c) = qflx_in_soil(c) - (1.0_r8 - fsno - frac_h2osfc(c))*qflx_evap(c) qflx_top_soil_to_h2osfc(c) = qflx_top_soil_to_h2osfc(c) - frac_h2osfc(c) * qflx_ev_h2osfc(c) + else + ! FIXME(wjs, 2017-08-15) Rather than this kludge, instead set qflx_in_soil and + ! qflx_top_soil_to_h2osfc to reasonable values for other landunits + qflx_in_soil(c) = 0._r8 end if end do From c1245110add833f92f022ca4d3b04a89474b6df2 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Tue, 15 Aug 2017 13:19:50 -0600 Subject: [PATCH 7/8] Fix accidental double-counting of (1-fsat) for vic --- src/biogeophys/InfiltrationExcessRunoffMod.F90 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/biogeophys/InfiltrationExcessRunoffMod.F90 b/src/biogeophys/InfiltrationExcessRunoffMod.F90 index 4d969d8abe..e4392f2b91 100644 --- a/src/biogeophys/InfiltrationExcessRunoffMod.F90 +++ b/src/biogeophys/InfiltrationExcessRunoffMod.F90 @@ -332,7 +332,7 @@ subroutine ComputeQinmaxVic(bounds, num_hydrologyc, filter_hydrologyc, & + top_max_moist(c) * basis**(1._r8 + b_infil(c)))/dtime end if rsurf_vic = min(qflx_in_soil(c), rsurf_vic) - qinmax_on_unsaturated_area(c) = (1._r8 - fsat(c)) * 10._r8**(-e_ice*top_icefrac)*(qflx_in_soil(c) - rsurf_vic) + qinmax_on_unsaturated_area(c) = 10._r8**(-e_ice*top_icefrac)*(qflx_in_soil(c) - rsurf_vic) end do end associate From e44881c84cb94e8f727a47598906fe378ee9e797 Mon Sep 17 00:00:00 2001 From: Bill Sacks Date: Mon, 21 Aug 2017 12:33:04 -0600 Subject: [PATCH 8/8] Avoid divide by zero This should only affect tests with the oldhyd test mod. However, it is needed in order for these tests to pass: ERP_D_Ld5.f10_f10_musgs.I2000Clm50BgcCrop.yellowstone_intel.clm-allActive ERP_D_P15x2_Ld3.f10_f10_musgs.I2000Clm50BgcCrop.yellowstone_intel.clm-cropColdStart ERP_P180x2_D_Ld5.f19_g17.I2000Clm50BgcDvCrop.yellowstone_pgi.clm-crop ERP_D_Ld5.f10_f10_musgs.I2000Clm50BgcCrop.cheyenne_intel.clm-allActive ERP_D_P15x2_Ld3.f10_f10_musgs.I2000Clm50BgcCrop.cheyenne_intel.clm-cropColdStart --- src/biogeophys/SoilHydrologyMod.F90 | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/biogeophys/SoilHydrologyMod.F90 b/src/biogeophys/SoilHydrologyMod.F90 index cb3513d7c3..c14e3e7233 100644 --- a/src/biogeophys/SoilHydrologyMod.F90 +++ b/src/biogeophys/SoilHydrologyMod.F90 @@ -165,7 +165,12 @@ subroutine SetSoilWaterFractions(bounds, num_hydrologyc, filter_hydrologyc, & ! fracice is only used in code with origflag == 1. For this calculation, we use ! the version of icefrac that was used in this original hydrology code. - icefrac_orig = min(1._r8,h2osoi_ice(c,j)/(h2osoi_ice(c,j)+h2osoi_liq(c,j))) + if (h2osoi_ice(c,j) == 0._r8) then + ! Avoid possible divide by zero (in case h2osoi_liq(c,j) is also 0) + icefrac_orig = 0._r8 + else + icefrac_orig = min(1._r8,h2osoi_ice(c,j)/(h2osoi_ice(c,j)+h2osoi_liq(c,j))) + end if fracice(c,j) = max(0._r8,exp(-3._r8*(1._r8-icefrac_orig))- exp(-3._r8))/(1.0_r8-exp(-3._r8)) end do end do