CNRootDynMod.F90 Source File


Source Code

module CNRootDynMod

!-----------------------------------------------------------------------
! !DESCRIPTION:
! Module holding routines used for determining fine root distribution for all pfts.
! Includes dynamic root depth for crops
!
! !USES:
   use shr_kind_mod                    , only : r8 => shr_kind_r8
   use clm_time_manager                , only : get_step_size
   use clm_varpar                      , only : nlevsoi, nlevgrnd
   use clm_varctl                      , only : use_vertsoilc, use_bedrock
   use decompMod                       , only : bounds_type
   use pftconMod                       , only : noveg, npcropmin, pftcon 
   use ColumnType                      , only : col
   use PatchType                       , only : patch
   use CNVegStateType                  , only : cnveg_state_type
   use CNVegCarbonStateType            , only : cnveg_carbonstate_type
   use CNVegCarbonFluxType             , only : cnveg_carbonflux_type
   use CNVegnitrogenstateType          , only : cnveg_nitrogenstate_type
   use SoilStateType                   , only : soilstate_type
   use SoilBiogeochemNitrogenStateType , only : soilbiogeochem_nitrogenstate_type
   use CropType                        , only : crop_type

! !PUBLIC TYPES:
  implicit none
  save
  private
  public :: CNRootDyn
!-----------------------------------------------------------------------

contains

!-----------------------------------------------------------------------
!
subroutine CNRootDyn(bounds, num_soilc, filter_soilc, num_soilp, filter_soilp, &
           cnveg_carbonstate_inst, cnveg_nitrogenstate_inst, cnveg_carbonflux_inst,   &
           cnveg_state_inst, crop_inst,  soilstate_inst, soilbiogeochem_nitrogenstate_inst)
!
! !DESCRIPTION:
! This routine determine the fine root distribution
! Needs to be called after the photosynthesis calculation
! May need to update other subroutines that use the fixed root profile for calculations
! i.e. CNVerticalProfileMod
!
! !USES:


! !ARGUMENTS:
   type(bounds_type), intent(in) :: bounds  ! bounds
   integer, intent(in) :: num_soilc
   integer, intent(in) :: filter_soilc(:)
   integer, intent(in) :: num_soilp              ! number of soil pfts in filter
   integer, intent(in) :: filter_soilp(:)        ! filter for soil pfts
   type(cnveg_state_type)                  , intent(in)    :: cnveg_state_inst
   type(cnveg_carbonstate_type)            , intent(in)    :: cnveg_carbonstate_inst
   type(cnveg_carbonflux_type)             , intent(in)    :: cnveg_carbonflux_inst
   type(cnveg_nitrogenstate_type)          , intent(in)    :: cnveg_nitrogenstate_inst
   type(crop_type)                         , intent(in)    :: crop_inst
   type(soilbiogeochem_nitrogenstate_type) , intent(in)    :: soilbiogeochem_nitrogenstate_inst
   type(soilstate_type)                    , intent(inout) :: soilstate_inst

!
! !LOCAL VARIABLES:
       
   integer :: f,c,p,lev,j                        ! indices
   real(r8):: dt                                 ! radiation time step delta t (seconds)
   real(r8), allocatable :: w_limit(:)
   real(r8), allocatable :: rswa(:,:)            ! soil water availability in each soil layer 
   real(r8), allocatable :: rsmn(:,:)            ! soil nitrogen availability in each soil layer 
   real(r8), allocatable :: sumrswa(:)           ! scaling soil water availability in each soil layer 
   real(r8), allocatable :: sumrsmn(:)           ! scaling  soil mineral N availability in each soil layer
   real(r8) :: frootc_dz(bounds%begp:bounds%endp, 1:nlevgrnd)
   real(r8), allocatable :: sumfrootc(:)         ! fine root carbon total before turnover in each step
   real(r8):: minpsi                             ! minimum soil moisture potential
   real(r8):: psi
   real(r8):: maxpsi
   real(r8):: new_growth

!-----------------------------------------------------------------------
   ! Assign local pointers to derived type arrays (in)
    associate(&
    ivt                    => patch%itype                                       , & ! Input:  [integer (:)]  pft vegetation type
    pcolumn                => patch%column                                      , & ! Input:  [integer (:)]  pft's column index
    roota_par              => pftcon%roota_par                                  , & ! Input:  [real(r8) (:)] pft's roota index
    rootb_par              => pftcon%rootb_par                                  , & ! Input:  [real(r8) (:)] pft's rootb index
    root_dmx               => pftcon%root_dmx                                   , & ! Input:  [real(r8) (:)] crop maximum root depth
    cpool_to_frootc        => cnveg_carbonflux_inst%cpool_to_frootc_patch       , & ! Input:  [real(r8) (:)] allocation to fine root C (gC/m2/s)
    frootc_xfer_to_frootc  => cnveg_carbonflux_inst%frootc_xfer_to_frootc_patch , & ! Input:  [real(r8) (:)] fine root C growth from storage (gC/m2/s)
    dormant_flag           => cnveg_state_inst%dormant_flag_patch               , & ! Input:  [real(r8) (:)]  dormancy flag
    root_depth             => soilstate_inst%root_depth_patch                   , & ! InOut:  [real(r8) (:)] current root depth
    dz                     => col%dz                                            , & ! Input:  layer thickness (m)  (-nlevsno+1:nlevgrnd)
    zi                     => col%zi                                            , & ! Input:  interface level below a "z" level (m) (-nlevsno+0:nlevgrnd)
    rootfr                 => soilstate_inst%rootfr_patch                       , & ! Output: [real(r8) (:,:)]  fraction of roots in each soil layer
    sucsat                 => soilstate_inst%sucsat_col                         , & ! Input:  minimum soil suction (mm)
    soilpsi                => soilstate_inst%soilpsi_col                        , & ! Input:  soil water potential in each soil layer (MPa)
    sminn_vr               => soilbiogeochem_nitrogenstate_inst%sminn_vr_col    , & ! Iniput:  [real(r8) (:,:)]  (gN/m3) soil mineral N
    frootc                 => cnveg_carbonstate_inst%frootc_patch               , & ! Input:  [real(r8) (:)]  (gC/m2) fine root C
    hui                    => crop_inst%gddplant_patch                          , & ! Input:  [real(r8) (:)]  =gdd since planting (gddplant)
    croplive               => crop_inst%croplive_patch                          , & ! Input:  [logical (:)]  flag, true if planted, not harvested
    huigrain               => cnveg_state_inst%huigrain_patch                     & ! Input: [real(r8) (:)]  same to reach vegetative maturity
    )
   
! set time steps
   dt = get_step_size()

! set minpsi to permanent wilting point
   minpsi = -1.5_r8
 
   allocate(sumrswa(bounds%begp:bounds%endp)) 
   allocate(sumrsmn(bounds%begp:bounds%endp))
   allocate(sumfrootc(bounds%begp:bounds%endp))
   allocate(rswa(bounds%begp:bounds%endp,nlevgrnd))
   allocate(rsmn(bounds%begp:bounds%endp,nlevgrnd))
   allocate(w_limit(bounds%begp:bounds%endp))

!initialize to 0
   w_limit(bounds%begp:bounds%endp)   = 0._r8
   sumrswa(bounds%begp:bounds%endp)   = 0._r8 
   sumrsmn(bounds%begp:bounds%endp)   = 0._r8 
   sumfrootc(bounds%begp:bounds%endp) = 0._r8
   rswa(bounds%begp:bounds%endp,:) = 0._r8
   rsmn(bounds%begp:bounds%endp,:) = 0._r8
    
   frootc_dz(bounds%begp:bounds%endp,1:nlevgrnd) = 0._r8


!---------------------------------------------------------------
! Set root depth, dynamic for crops, fixed for other vegetation
!---------------------------------------------------------------

         do f = 1, num_soilp
            p = filter_soilp(f)
            c = pcolumn(p)
            if (ivt(p) /= noveg) then
                if((ivt(p)) >= npcropmin)then !skip generic crop types
                    if(huigrain(p) > 0._r8)then
                        root_depth(p) = max(zi(c,2), min(hui(p)/huigrain(p)* root_dmx(ivt(p)), root_dmx(ivt(p))))
                    end if
                else
                ! this can be changed to any depth (i.e. the maximum soil depth)
                    root_depth(p) = zi(c,nlevsoi) 
                end if
                if (use_bedrock) then
                   root_depth(p) = min(root_depth(p),zi(c,col%nbedrock(c)))
                end if
            else
                root_depth(p) = 0._r8
            end if
         end do

!----------------------------------------------------------------
!  ! calculate a weighting function by soil depth that depends on the
   ! fine root distribution per pft and depth and the pft weight on the column.
   ! This will be used to weight the temperature and water potential scalars
   ! for decomposition control.  

   ! calculate the rate constant scalar for soil water content.
   ! Uses the log relationship with water potential given in
   ! Andren, O., and K. Paustian, 1987. Barley straw decomposition in the field:
   ! a comparison of models. Ecology, 68(5):1190-1200.
   ! and supported by data in
   ! Orchard, V.A., and F.J. Cook, 1983. Relationship between soil respiration
   ! and soil moisture. Soil Biol. Biochem., 15(4):447-453.

   do j = 1,nlevsoi
      do f = 1,num_soilp
         p = filter_soilp(f)
         c = pcolumn(p)
         maxpsi = sucsat(c,j) * (-9.8e-6_r8)
         psi = min(soilpsi(c,j),maxpsi)
         if (psi > minpsi) then
! First calculate water in the root zone
            if(root_depth(p) >  0.15_r8 .and. (zi(c,j) <= root_depth(p) .or. &
              (zi(c,j-1) < root_depth(p) .and. zi(c,j) > root_depth(p)))) then
                 w_limit(p) = w_limit(p) + max(0._r8,log(minpsi/psi)/log(minpsi/maxpsi))*rootfr(p,j)
             end if
! Calculate the water in each soil layer
             if (root_depth(p) >= zi(c,j) .or. &
               (zi(c,j-1) < root_depth(p) .and. zi(c,j) > root_depth(p))) then
                 rswa(p,j) = max(0._r8, (log(minpsi/psi)/log(minpsi/maxpsi)))
             end if
         end if
         sumrswa(p) = sumrswa(p) + rswa(p,j)

! Calculate the nitrogen profile in each layer
! For now, the profile for each PFT is equivilent to the
! column profile, in the future, this could be changed to a weighted profile
         if(use_vertsoilc) then !for vertical soil profile
            rsmn(p,j) = sminn_vr(c,j) 
         else ! need to calculate a profile, top 0.2m are constant, and decrease linearly
            if(zi(c,j) <= 0.2_r8)then
                rsmn(p,j) = dz(c,j)
            end if
            if(zi(c,j) > 0.2_r8)then
                rsmn(p,j) = dz(c,j) * (zi(c,nlevsoi) - zi(c,j)) / (zi(c,nlevsoi) - 0.2_r8)
            end if 
         end if
         if (root_depth(p) >= zi(c,j).or. &
               (zi(c,j-1) < root_depth(p) .and. zi(c,j) > root_depth(p))) then
            sumrsmn(p) = sumrsmn(p) + rsmn(p,j)      
         end if
      end do
   end do


!--------------------------------------------------------------------
! Now calculate the density of roots in each soil layer for each pft
! based on this timesteps growth
!--------------------------------------------------------------------
    do lev = 1, nlevgrnd
    
         do f = 1, num_soilp
            p = filter_soilp(f) 
            c = pcolumn(p)

            new_growth = (cpool_to_frootc(p) + frootc_xfer_to_frootc(p))*dt
            if(zi(c,lev) <= root_depth(p) .or. &
               (zi(c,lev-1) < root_depth(p) .and. zi(c,lev) > root_depth(p))) then 
               if(sumrswa(p) <= 0._r8 .or. sumrsmn(p) <= 0._r8) then
!                   when sumrswa or sumrsmn are less than or equal to 0 rootfr will not be updated
               else
                   frootc_dz(p,lev) = (frootc(p))*rootfr(p,lev) & 
                                 + new_growth * ((1._r8 - w_limit(p)) * rswa(p,lev) / sumrswa(p) &
                                 + w_limit(p) * rsmn(p,lev) / sumrsmn(p))
               end if
            else
               frootc_dz(p,lev) = 0._r8
            end if
     
            sumfrootc(p) = sumfrootc(p) + frootc_dz(p,lev)
       
         end do
    end do
!----------------------------------
!Calculate root fraction
!----------------------------------

     do lev = 1, nlevgrnd
        do f = 1, num_soilp
           p = filter_soilp(f)
           c = pcolumn(p)
             if(sumfrootc(p) > 0._r8)then 
                rootfr(p,lev) = frootc_dz(p,lev)/sumfrootc(p)
             end if
             if(ivt(p) >= npcropmin .and. .not. croplive(p))then
! CROPS are dormant, there are no roots! 
! but, need an initial frootr so crops can start root production
                 if (lev <  2)then
                    rootfr(p,lev) = .5_r8*( exp(-roota_par(patch%itype(p)) * zi(c,lev-1))  &
                                   + exp(-rootb_par(patch%itype(p)) * zi(c,lev-1))  &
                                   - exp(-roota_par(patch%itype(p)) * zi(c,lev  ))  &
                                   - exp(-rootb_par(patch%itype(p)) * zi(c,lev  )) )
                 elseif (lev == 2) then
                     rootfr(p,lev) = .5_r8*( exp(-roota_par(patch%itype(p)) * zi(c,lev-1))  &
                                    + exp(-rootb_par(patch%itype(p)) * zi(c,lev-1)) )
                 else
                     rootfr(p,lev) =  0.0_r8
                 end if
      
             end if
          end do
     end do

!**********************
     deallocate(sumrswa)
     deallocate(sumrsmn)
     deallocate(sumfrootc)
     deallocate(rsmn)
     deallocate(rswa)
     deallocate(w_limit)

     end associate

 end subroutine CNRootDyn

end module CNRootDynMod