From 8104707966b33187b6b2b6021b5ec9b74e439e37 Mon Sep 17 00:00:00 2001 From: Charlie Koven Date: Tue, 7 May 2024 14:44:35 -0700 Subject: [PATCH 01/11] first set of changes to move growth respiration calculation from half-hourly to daily --- biogeochem/EDCohortDynamicsMod.F90 | 15 +++-- biogeochem/FatesCohortMod.F90 | 47 ++++++++------- biogeochem/FatesSoilBGCFluxMod.F90 | 2 +- biogeophys/EDAccumulateFluxesMod.F90 | 8 +-- biogeophys/FatesPlantRespPhotosynthMod.F90 | 10 +--- main/EDMainMod.F90 | 36 +++++++++--- main/FatesHistoryInterfaceMod.F90 | 4 +- main/FatesRestartInterfaceMod.F90 | 68 ++++++++++++++-------- 8 files changed, 115 insertions(+), 75 deletions(-) diff --git a/biogeochem/EDCohortDynamicsMod.F90 b/biogeochem/EDCohortDynamicsMod.F90 index ef87851ec1..e362cfb96a 100644 --- a/biogeochem/EDCohortDynamicsMod.F90 +++ b/biogeochem/EDCohortDynamicsMod.F90 @@ -1066,11 +1066,16 @@ subroutine fuse_cohorts(currentSite, currentPatch, bc_in) nextc%n*nextc%gpp_acc)/newn currentCohort%npp_acc = (currentCohort%n*currentCohort%npp_acc + & nextc%n*nextc%npp_acc)/newn - currentCohort%resp_acc = (currentCohort%n*currentCohort%resp_acc + & - nextc%n*nextc%resp_acc)/newn - currentCohort%resp_acc_hold = & - (currentCohort%n*currentCohort%resp_acc_hold + & - nextc%n*nextc%resp_acc_hold)/newn + currentCohort%resp_m_acc = (currentCohort%n*currentCohort%resp_m_acc + & + nextc%n*nextc%resp_m_acc)/newn + currentCohort%resp_m_acc_hold = & + (currentCohort%n*currentCohort%resp_m_acc_hold + & + nextc%n*nextc%resp_m_acc_hold)/newn + currentCohort%resp_g_acc = (currentCohort%n*currentCohort%resp_g_acc + & + nextc%n*nextc%resp_g_acc)/newn + currentCohort%resp_g_acc_hold = & + (currentCohort%n*currentCohort%resp_g_acc_hold + & + nextc%n*nextc%resp_g_acc_hold)/newn currentCohort%npp_acc_hold = & (currentCohort%n*currentCohort%npp_acc_hold + & nextc%n*nextc%npp_acc_hold)/newn diff --git a/biogeochem/FatesCohortMod.F90 b/biogeochem/FatesCohortMod.F90 index e98c4a345a..64a4d929af 100644 --- a/biogeochem/FatesCohortMod.F90 +++ b/biogeochem/FatesCohortMod.F90 @@ -142,13 +142,15 @@ module FatesCohortMod real(r8) :: gpp_acc real(r8) :: gpp_acc_hold - real(r8) :: npp_tstep ! Net Primary Production (see above *) real(r8) :: npp_acc real(r8) :: npp_acc_hold - real(r8) :: resp_tstep ! Autotrophic respiration (see above *) - real(r8) :: resp_acc - real(r8) :: resp_acc_hold + real(r8) :: resp_m_tstep ! Maintenance respiration (see above *) + real(r8) :: resp_m_acc + real(r8) :: resp_m_acc_hold + + real(r8) :: resp_g_acc ! Growth respication can only be calculated at the daily timestep + real(r8) :: resp_g_acc_hold real(r8) :: c13disc_clm ! carbon 13 discrimination in new synthesized carbon at each indiv/timestep [ppm] real(r8) :: c13disc_acc ! carbon 13 discrimination in new synthesized carbon at each indiv/day @@ -376,12 +378,13 @@ subroutine NanValues(this) this%gpp_tstep = nan this%gpp_acc = nan this%gpp_acc_hold = nan - this%npp_tstep = nan this%npp_acc = nan this%npp_acc_hold = nan - this%resp_tstep = nan - this%resp_acc = nan - this%resp_acc_hold = nan + this%resp_m_tstep = nan + this%resp_m_acc = nan + this%resp_m_acc_hold = nan + this%resp_g_acc = nan + this%resp_g_acc_hold = nan this%c13disc_clm = nan this%c13disc_acc = nan this%vcmax25top = nan @@ -478,17 +481,18 @@ subroutine ZeroValues(this) this%size_class_lasttimestep = 0 this%gpp_tstep = 0._r8 this%gpp_acc = 0._r8 - this%npp_tstep = 0._r8 this%npp_acc = 0._r8 - this%resp_tstep = 0._r8 - this%resp_acc = 0._r8 + this%resp_m_tstep = 0._r8 + this%resp_m_acc = 0._r8 + this%resp_g_acc = 0._r8 ! do not zero these, they are not built ! so more appropriate to leave unzerod ! to prevent uninitialized use ! this%gpp_acc_hold = nan ! this%npp_acc_hold = nan - ! this%resp_acc_hold = nan + ! this%resp_m_acc_hold = nan + ! this%resp_g_acc_hold = nan this%c13disc_clm = 0._r8 this%c13disc_acc = 0._r8 @@ -707,12 +711,13 @@ subroutine Copy(this, copyCohort) copyCohort%gpp_tstep = this%gpp_tstep copyCohort%gpp_acc = this%gpp_acc copyCohort%gpp_acc_hold = this%gpp_acc_hold - copyCohort%npp_tstep = this%npp_tstep copyCohort%npp_acc = this%npp_acc copyCohort%npp_acc_hold = this%npp_acc_hold - copyCohort%resp_tstep = this%resp_tstep - copyCohort%resp_acc = this%resp_acc - copyCohort%resp_acc_hold = this%resp_acc_hold + copyCohort%resp_m_tstep = this%resp_m_tstep + copyCohort%resp_m_acc = this%resp_m_acc + copyCohort%resp_m_acc_hold = this%resp_m_acc_hold + copyCohort%resp_g_acc = this%resp_g_acc + copyCohort%resp_g_acc_hold = this%resp_g_acc_hold copyCohort%c13disc_clm = this%c13disc_clm copyCohort%c13disc_acc = this%c13disc_acc copyCohort%vcmax25top = this%vcmax25top @@ -1048,14 +1053,14 @@ subroutine Dump(this) write(fates_log(),*) 'cohort%gpp_acc = ', this%gpp_acc write(fates_log(),*) 'cohort%gpp_tstep = ', this%gpp_tstep write(fates_log(),*) 'cohort%npp_acc_hold = ', this%npp_acc_hold - write(fates_log(),*) 'cohort%npp_tstep = ', this%npp_tstep write(fates_log(),*) 'cohort%npp_acc = ', this%npp_acc - write(fates_log(),*) 'cohort%resp_tstep = ', this%resp_tstep - write(fates_log(),*) 'cohort%resp_acc = ', this%resp_acc - write(fates_log(),*) 'cohort%resp_acc_hold = ', this%resp_acc_hold + write(fates_log(),*) 'cohort%resp_m_tstep = ', this%resp_m_tstep + write(fates_log(),*) 'cohort%resp_m_acc = ', this%resp_m_acc + write(fates_log(),*) 'cohort%resp_m_acc_hold = ', this%resp_m_acc_hold + write(fates_log(),*) 'cohort%resp_g_acc = ', this%resp_g_acc + write(fates_log(),*) 'cohort%resp_g_acc_hold = ', this%resp_g_acc_hold write(fates_log(),*) 'cohort%rdark = ', this%rdark write(fates_log(),*) 'cohort%resp_m = ', this%resp_m - write(fates_log(),*) 'cohort%resp_g_tstep = ', this%resp_g_tstep write(fates_log(),*) 'cohort%livestem_mr = ', this%livestem_mr write(fates_log(),*) 'cohort%livecroot_mr = ', this%livecroot_mr write(fates_log(),*) 'cohort%froot_mr = ', this%froot_mr diff --git a/biogeochem/FatesSoilBGCFluxMod.F90 b/biogeochem/FatesSoilBGCFluxMod.F90 index 9d813c32b3..ae102eaf15 100644 --- a/biogeochem/FatesSoilBGCFluxMod.F90 +++ b/biogeochem/FatesSoilBGCFluxMod.F90 @@ -348,7 +348,7 @@ subroutine PrepCH4BCs(csite,bc_in,bc_out) ! this is a best (bad) guess at fine root MR + total root GR ! (kgC/indiv/yr) -> gC/m2/s bc_out%root_resp(1:bc_in%nlevsoil) = bc_out%root_resp(1:bc_in%nlevsoil) + & - ccohort%resp_acc_hold*years_per_day*g_per_kg*days_per_sec* & + (ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold)*years_per_day*g_per_kg*days_per_sec* & ccohort%n*area_inv*(1._r8-prt_params%allom_agb_frac(pft)) * csite%rootfrac_scr(1:bc_in%nlevsoil) end if diff --git a/biogeophys/EDAccumulateFluxesMod.F90 b/biogeophys/EDAccumulateFluxesMod.F90 index 6a1bb001c0..d74c965e31 100644 --- a/biogeophys/EDAccumulateFluxesMod.F90 +++ b/biogeophys/EDAccumulateFluxesMod.F90 @@ -3,7 +3,7 @@ module EDAccumulateFluxesMod !------------------------------------------------------------------------------ ! !DESCRIPTION: ! This routine accumulates NPP, GPP and respiration of each cohort over the course of each 24 hour period. - ! The fluxes are stored per cohort, and the npp_tstep (etc) fluxes are calcualted in EDPhotosynthesis + ! The fluxes are stored per cohort, and the gpp_tstep (etc) fluxes are calculated in EDPhotosynthesis ! This routine cannot be in EDPhotosynthesis because EDPhotosynthesis is a loop and therefore would ! erroneously add these things up multiple times. ! Rosie Fisher. March 2014. @@ -82,15 +82,13 @@ subroutine AccumulateFluxes_ED(nsites, sites, bc_in, bc_out, dt_time) if ( debug ) then - write(fates_log(),*) 'EDAccumFlux 64 ',ccohort%npp_tstep write(fates_log(),*) 'EDAccumFlux 66 ',ccohort%gpp_tstep - write(fates_log(),*) 'EDAccumFlux 67 ',ccohort%resp_tstep + write(fates_log(),*) 'EDAccumFlux 67 ',ccohort%resp_m_tstep endif - ccohort%npp_acc = ccohort%npp_acc + ccohort%npp_tstep ccohort%gpp_acc = ccohort%gpp_acc + ccohort%gpp_tstep - ccohort%resp_acc = ccohort%resp_acc + ccohort%resp_tstep + ccohort%resp_m_acc = ccohort%resp_m_acc + ccohort%resp_m_tstep ccohort%sym_nfix_daily = ccohort%sym_nfix_daily + ccohort%sym_nfix_tstep diff --git a/biogeophys/FatesPlantRespPhotosynthMod.F90 b/biogeophys/FatesPlantRespPhotosynthMod.F90 index 0aad6eb977..430ca555bf 100644 --- a/biogeophys/FatesPlantRespPhotosynthMod.F90 +++ b/biogeophys/FatesPlantRespPhotosynthMod.F90 @@ -987,6 +987,7 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) ! convert from kgC/indiv/s to kgC/indiv/timestep currentCohort%resp_m = currentCohort%resp_m * dtime + currentCohort%resp_m_tstep = currentCohort%resp_m ! these two things are the same. we should get rid of one of them. which? currentCohort%gpp_tstep = currentCohort%gpp_tstep * dtime currentCohort%ts_net_uptake = currentCohort%ts_net_uptake * dtime @@ -995,15 +996,6 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) if ( debug ) write(fates_log(),*) 'EDPhoto 913 ', currentCohort%resp_m - currentCohort%resp_g_tstep = prt_params%grperc(ft) * & - (max(0._r8,currentCohort%gpp_tstep - currentCohort%resp_m)) - - - currentCohort%resp_tstep = currentCohort%resp_m + & - currentCohort%resp_g_tstep ! kgC/indiv/ts - currentCohort%npp_tstep = currentCohort%gpp_tstep - & - currentCohort%resp_tstep ! kgC/indiv/ts - ! Accumulate the combined conductance (stomatal+leaf boundary layer) ! Note that currentCohort%g_sb_laweight is weighted by the leaf area ! of each cohort and has units of [m/s] * [m2 leaf] diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 159e942c6a..22cb94d992 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -335,7 +335,8 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) use FatesConstantsMod, only : itrue use FatesConstantsMod , only : nearzero use EDCanopyStructureMod , only : canopy_structure - + use PRTParametersMod , only : prt_params + ! !ARGUMENTS: type(ed_site_type) , intent(inout) :: currentSite @@ -498,7 +499,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! We don't explicitly define a respiration rate for prescribe phys ! but we do need to pass mass balance. So we say it is zero respiration currentCohort%gpp_acc = currentCohort%npp_acc - currentCohort%resp_acc = 0._r8 + currentCohort%resp_m_acc = 0._r8 end if @@ -514,14 +515,30 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! photosynthesis step ! ----------------------------------------------------------------------------- - currentCohort%npp_acc_hold = currentCohort%npp_acc * real(hlm_days_per_year,r8) currentCohort%gpp_acc_hold = currentCohort%gpp_acc * real(hlm_days_per_year,r8) - currentCohort%resp_acc_hold = currentCohort%resp_acc * real(hlm_days_per_year,r8) + currentCohort%resp_m_acc_hold = currentCohort%resp_m_acc * real(hlm_days_per_year,r8) + + ! at this point we have the info we need to calculate growth respiration + ! as a "tax" on the difference between daily GPP and daily maintenance respiration + if (hlm_use_ed_prescribed_phys .eq. itrue) then + currentCohort%resp_g_acc = prt_params%grperc(ft) * & + max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) + currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) + else + ! set growth respiration to zero in prescribed physiology mode, + ! that way the npp_acc vars will be set to the nominal gpp values set above. + currentCohort%resp_g_acc = 0._r8 + currentCohort%resp_g_acc_hold = 0._r8 + endif + + ! calculate the npp as the difference between gpp and autotrophic respiration + currentCohort%npp_acc = currentCohort%gpp_acc - (currentCohort%resp_m_acc + currentCohort%resp_g_acc) + currentCohort%npp_acc_hold = currentCohort%gpp_acc_hold - (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) ! Passing gpp_acc_hold to HLM bc_out%gpp_site = bc_out%gpp_site + currentCohort%gpp_acc_hold * & AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day - bc_out%ar_site = bc_out%ar_site + currentCohort%resp_acc_hold * & + bc_out%ar_site = bc_out%ar_site + (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) * & AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day ! Conduct Maintenance Turnover (parteh) @@ -630,7 +647,8 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentCohort%gpp_acc * currentCohort%n site_cmass%aresp_acc = site_cmass%aresp_acc + & - (currentCohort%resp_acc+currentCohort%resp_excess) * currentCohort%n + (currentCohort%resp_m_acc+currentCohort%resp_g_acc+currentCohort%resp_excess) * & + currentCohort%n call currentCohort%prt%CheckMassConservation(ft,5) @@ -1050,11 +1068,13 @@ subroutine bypass_dynamics(currentSite) currentCohort%npp_acc_hold = currentCohort%npp_acc * real(hlm_days_per_year,r8) currentCohort%gpp_acc_hold = currentCohort%gpp_acc * real(hlm_days_per_year,r8) - currentCohort%resp_acc_hold = currentCohort%resp_acc * real(hlm_days_per_year,r8) + currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) + currentCohort%resp_m_acc_hold = currentCohort%resp_m_acc * real(hlm_days_per_year,r8) currentCohort%npp_acc = 0.0_r8 currentCohort%gpp_acc = 0.0_r8 - currentCohort%resp_acc = 0.0_r8 + currentCohort%resp_g_acc = 0.0_r8 + currentCohort%resp_m_acc = 0.0_r8 ! No need to set the "net_art" terms to zero ! they are zeroed at the beginning of the daily step diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index c80186a58b..8b8615a75c 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -3918,7 +3918,7 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_gpp_canopy_si_scpf(io_si,scpf) = hio_gpp_canopy_si_scpf(io_si,scpf) + & n_perm2*ccohort%gpp_acc_hold / days_per_year / sec_per_day hio_ar_canopy_si_scpf(io_si,scpf) = hio_ar_canopy_si_scpf(io_si,scpf) + & - n_perm2*ccohort%resp_acc_hold / days_per_year / sec_per_day + n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold) / days_per_year / sec_per_day ! growth increment hio_ddbh_canopy_si_scpf(io_si,scpf) = hio_ddbh_canopy_si_scpf(io_si,scpf) + & ccohort%ddbhdt*ccohort%n * m_per_cm / m2_per_ha @@ -4053,7 +4053,7 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_gpp_understory_si_scpf(io_si,scpf) = hio_gpp_understory_si_scpf(io_si,scpf) + & n_perm2*ccohort%gpp_acc_hold / days_per_year / sec_per_day hio_ar_understory_si_scpf(io_si,scpf) = hio_ar_understory_si_scpf(io_si,scpf) + & - n_perm2*ccohort%resp_acc_hold / days_per_year / sec_per_day + n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold) / days_per_year / sec_per_day ! growth increment hio_ddbh_understory_si_scpf(io_si,scpf) = hio_ddbh_understory_si_scpf(io_si,scpf) + & diff --git a/main/FatesRestartInterfaceMod.F90 b/main/FatesRestartInterfaceMod.F90 index 90e282253b..3300d2297a 100644 --- a/main/FatesRestartInterfaceMod.F90 +++ b/main/FatesRestartInterfaceMod.F90 @@ -132,10 +132,12 @@ module FatesRestartInterfaceMod integer :: ir_nplant_co integer :: ir_gpp_acc_co integer :: ir_npp_acc_co - integer :: ir_resp_acc_co + integer :: ir_resp_m_acc_co + integer :: ir_resp_g_acc_co integer :: ir_gpp_acc_hold_co integer :: ir_npp_acc_hold_co - integer :: ir_resp_acc_hold_co + integer :: ir_resp_m_acc_hold_co + integer :: ir_resp_g_acc_hold_co integer :: ir_resp_excess_co integer :: ir_bmort_co integer :: ir_hmort_co @@ -180,7 +182,7 @@ module FatesRestartInterfaceMod integer :: ir_ddbhdt_co - integer :: ir_resp_tstep_co + integer :: ir_resp_m_tstep_co integer :: ir_pft_co integer :: ir_status_co integer :: ir_efleaf_co @@ -853,10 +855,15 @@ subroutine define_restart_vars(this, initialize_variables) units='kgC/indiv', flushval = flushzero, & hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_npp_acc_co ) - call this%set_restart_var(vname='fates_resp_acc', vtype=cohort_r8, & - long_name='ed cohort - accumulated respiration over dynamics step', & + call this%set_restart_var(vname='fates_resp_m_acc', vtype=cohort_r8, & + long_name='ed cohort - accumulated maintenance respiration over dynamics step', & units='kgC/indiv', flushval = flushzero, & - hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_acc_co ) + hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_m_acc_co ) + + call this%set_restart_var(vname='fates_resp_g_acc', vtype=cohort_r8, & + long_name='ed cohort - accumulated growth respiration over dynamics step', & + units='kgC/indiv', flushval = flushzero, & + hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_g_acc_co ) call this%set_restart_var(vname='fates_gpp_acc_hold', vtype=cohort_r8, & long_name='ed cohort - current step gpp', & @@ -868,10 +875,15 @@ subroutine define_restart_vars(this, initialize_variables) units='kgC/indiv/year', flushval = flushzero, & hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_npp_acc_hold_co ) - call this%set_restart_var(vname='fates_resp_acc_hold', vtype=cohort_r8, & - long_name='ed cohort - current step resp', & + call this%set_restart_var(vname='fates_resp_m_acc_hold', vtype=cohort_r8, & + long_name='ed cohort - current step maint resp', & + units='kgC/indiv/year', flushval = flushzero, & + hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_m_acc_hold_co ) + + call this%set_restart_var(vname='fates_resp_g_acc_hold', vtype=cohort_r8, & + long_name='ed cohort - current step growth resp', & units='kgC/indiv/year', flushval = flushzero, & - hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_acc_hold_co ) + hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_g_acc_hold_co ) call this%set_restart_var(vname='fates_resp_excess', vtype=cohort_r8, & long_name='ed cohort - maintenance respiration deficit', & @@ -935,10 +947,10 @@ subroutine define_restart_vars(this, initialize_variables) units='cm/year', flushval = flushzero, & hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_ddbhdt_co ) - call this%set_restart_var(vname='fates_resp_tstep', vtype=cohort_r8, & - long_name='ed cohort - autotrophic respiration over timestep', & + call this%set_restart_var(vname='fates_resp_m_tstep', vtype=cohort_r8, & + long_name='ed cohort - maintenance respiration over timestep', & units='kgC/indiv/timestep', flushval = flushzero, & - hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_tstep_co ) + hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_m_tstep_co ) call this%set_restart_var(vname='fates_pft', vtype=cohort_int, & long_name='ed cohort - plant functional type', units='index', flushval = flushzero, & @@ -2056,9 +2068,11 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_nplant_co => this%rvars(ir_nplant_co)%r81d, & rio_gpp_acc_co => this%rvars(ir_gpp_acc_co)%r81d, & rio_npp_acc_co => this%rvars(ir_npp_acc_co)%r81d, & - rio_resp_acc_co => this%rvars(ir_resp_acc_co)%r81d, & + rio_resp_m_acc_co => this%rvars(ir_resp_m_acc_co)%r81d, & + rio_resp_g_acc_co => this%rvars(ir_resp_g_acc_co)%r81d, & rio_gpp_acc_hold_co => this%rvars(ir_gpp_acc_hold_co)%r81d, & - rio_resp_acc_hold_co => this%rvars(ir_resp_acc_hold_co)%r81d, & + rio_resp_m_acc_hold_co => this%rvars(ir_resp_m_acc_hold_co)%r81d, & + rio_resp_g_acc_hold_co => this%rvars(ir_resp_g_acc_hold_co)%r81d, & rio_npp_acc_hold_co => this%rvars(ir_npp_acc_hold_co)%r81d, & rio_resp_excess_co => this%rvars(ir_resp_excess_co)%r81d, & rio_bmort_co => this%rvars(ir_bmort_co)%r81d, & @@ -2072,7 +2086,7 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_lmort_collateral_co => this%rvars(ir_lmort_collateral_co)%r81d, & rio_lmort_infra_co => this%rvars(ir_lmort_infra_co)%r81d, & rio_ddbhdt_co => this%rvars(ir_ddbhdt_co)%r81d, & - rio_resp_tstep_co => this%rvars(ir_resp_tstep_co)%r81d, & + rio_resp_m_tstep_co => this%rvars(ir_resp_m_tstep_co)%r81d, & rio_pft_co => this%rvars(ir_pft_co)%int1d, & rio_status_co => this%rvars(ir_status_co)%int1d, & rio_efleaf_co => this%rvars(ir_efleaf_co)%r81d, & @@ -2378,9 +2392,11 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_nplant_co(io_idx_co) = ccohort%n rio_gpp_acc_co(io_idx_co) = ccohort%gpp_acc rio_npp_acc_co(io_idx_co) = ccohort%npp_acc - rio_resp_acc_co(io_idx_co) = ccohort%resp_acc + rio_resp_m_acc_co(io_idx_co) = ccohort%resp_m_acc + rio_resp_g_acc_co(io_idx_co) = ccohort%resp_g_acc rio_gpp_acc_hold_co(io_idx_co) = ccohort%gpp_acc_hold - rio_resp_acc_hold_co(io_idx_co) = ccohort%resp_acc_hold + rio_resp_m_acc_hold_co(io_idx_co) = ccohort%resp_m_acc_hold + rio_resp_g_acc_hold_co(io_idx_co) = ccohort%resp_g_acc_hold rio_npp_acc_hold_co(io_idx_co) = ccohort%npp_acc_hold rio_resp_excess_co(io_idx_co) = ccohort%resp_excess @@ -2400,7 +2416,7 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_lmort_infra_co(io_idx_co) = ccohort%lmort_infra rio_ddbhdt_co(io_idx_co) = ccohort%ddbhdt - rio_resp_tstep_co(io_idx_co) = ccohort%resp_tstep + rio_resp_m_tstep_co(io_idx_co) = ccohort%resp_m_tstep rio_pft_co(io_idx_co) = ccohort%pft rio_status_co(io_idx_co) = ccohort%status_coh rio_efleaf_co(io_idx_co) = ccohort%efleaf_coh @@ -3023,9 +3039,11 @@ subroutine get_restart_vectors(this, nc, nsites, sites) rio_nplant_co => this%rvars(ir_nplant_co)%r81d, & rio_gpp_acc_co => this%rvars(ir_gpp_acc_co)%r81d, & rio_npp_acc_co => this%rvars(ir_npp_acc_co)%r81d, & - rio_resp_acc_co => this%rvars(ir_resp_acc_co)%r81d, & + rio_resp_m_acc_co => this%rvars(ir_resp_m_acc_co)%r81d, & + rio_resp_g_acc_co => this%rvars(ir_resp_g_acc_co)%r81d, & rio_gpp_acc_hold_co => this%rvars(ir_gpp_acc_hold_co)%r81d, & - rio_resp_acc_hold_co => this%rvars(ir_resp_acc_hold_co)%r81d, & + rio_resp_m_acc_hold_co => this%rvars(ir_resp_m_acc_hold_co)%r81d, & + rio_resp_g_acc_hold_co => this%rvars(ir_resp_g_acc_hold_co)%r81d, & rio_npp_acc_hold_co => this%rvars(ir_npp_acc_hold_co)%r81d, & rio_resp_excess_co => this%rvars(ir_resp_excess_co)%r81d, & rio_bmort_co => this%rvars(ir_bmort_co)%r81d, & @@ -3039,7 +3057,7 @@ subroutine get_restart_vectors(this, nc, nsites, sites) rio_lmort_collateral_co => this%rvars(ir_lmort_collateral_co)%r81d, & rio_lmort_infra_co => this%rvars(ir_lmort_infra_co)%r81d, & rio_ddbhdt_co => this%rvars(ir_ddbhdt_co)%r81d, & - rio_resp_tstep_co => this%rvars(ir_resp_tstep_co)%r81d, & + rio_resp_m_tstep_co => this%rvars(ir_resp_m_tstep_co)%r81d, & rio_pft_co => this%rvars(ir_pft_co)%int1d, & rio_status_co => this%rvars(ir_status_co)%int1d, & rio_efleaf_co => this%rvars(ir_efleaf_co)%r81d, & @@ -3319,9 +3337,11 @@ subroutine get_restart_vectors(this, nc, nsites, sites) ccohort%n = rio_nplant_co(io_idx_co) ccohort%gpp_acc = rio_gpp_acc_co(io_idx_co) ccohort%npp_acc = rio_npp_acc_co(io_idx_co) - ccohort%resp_acc = rio_resp_acc_co(io_idx_co) + ccohort%resp_m_acc = rio_resp_m_acc_co(io_idx_co) + ccohort%resp_g_acc = rio_resp_g_acc_co(io_idx_co) ccohort%gpp_acc_hold = rio_gpp_acc_hold_co(io_idx_co) - ccohort%resp_acc_hold = rio_resp_acc_hold_co(io_idx_co) + ccohort%resp_m_acc_hold = rio_resp_m_acc_hold_co(io_idx_co) + ccohort%resp_g_acc_hold = rio_resp_g_acc_hold_co(io_idx_co) ccohort%npp_acc_hold = rio_npp_acc_hold_co(io_idx_co) ccohort%resp_excess = rio_resp_excess_co(io_idx_co) @@ -3341,7 +3361,7 @@ subroutine get_restart_vectors(this, nc, nsites, sites) ccohort%lmort_infra = rio_lmort_infra_co(io_idx_co) ccohort%ddbhdt = rio_ddbhdt_co(io_idx_co) - ccohort%resp_tstep = rio_resp_tstep_co(io_idx_co) + ccohort%resp_m_tstep = rio_resp_m_tstep_co(io_idx_co) ccohort%pft = rio_pft_co(io_idx_co) ccohort%status_coh = rio_status_co(io_idx_co) ccohort%efleaf_coh = rio_efleaf_co(io_idx_co) From d03b44f463ecee3b724de11e8f505c5abfeaca7d Mon Sep 17 00:00:00 2001 From: Charlie Koven Date: Wed, 8 May 2024 16:51:09 -0700 Subject: [PATCH 02/11] starting to modify history vars --- main/FatesHistoryInterfaceMod.F90 | 83 +++++++++++++------------------ 1 file changed, 34 insertions(+), 49 deletions(-) diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index 8b8615a75c..a2faecb9c0 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -2377,6 +2377,8 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) hio_sum_fuel_si => this%hvars(ih_sum_fuel_si)%r81d, & hio_litter_in_si => this%hvars(ih_litter_in_si)%r81d, & hio_litter_out_si => this%hvars(ih_litter_out_si)%r81d, & + hio_npp_si => this%hvars(ih_npp_si)%r82d, & + hio_growth_resp_si => this%hvars(ih_growth_resp_si)%r81d, & hio_seed_bank_si => this%hvars(ih_seed_bank_si)%r81d, & hio_ungerm_seed_bank_si => this%hvars(ih_ungerm_seed_bank_si)%r81d, & hio_seedling_pool_si => this%hvars(ih_seedling_pool_si)%r81d, & @@ -2814,6 +2816,12 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) ! have any meaning, otherwise they are just inialization values notnew: if( .not.(ccohort%isnew) ) then + hio_npp_si(io_si) = hio_npp_si(io_si) + & + ccohort%npp_acc_hold * n_perm2 / days_per_year / sec_per_day + + hio_growth_resp_si(io_si) = hio_growth_resp_si(io_si) + & + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day + ! Turnover pools [kgC/day] * [day/yr] = [kgC/yr] sapw_m_turnover = ccohort%prt%GetTurnover(sapw_organ, carbon12_element) * days_per_year store_m_turnover = ccohort%prt%GetTurnover(store_organ, carbon12_element) * days_per_year @@ -3069,7 +3077,7 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_canopycrownarea_si_pft => this%hvars(ih_canopycrownarea_si_pft)%r82d, & hio_gpp_si_pft => this%hvars(ih_gpp_si_pft)%r82d, & hio_gpp_sec_si_pft => this%hvars(ih_gpp_sec_si_pft)%r82d, & - hio_npp_si_pft => this%hvars(ih_npp_si_pft)%r82d, & + hio_npp_si_pft => this%hvars(ih_npp_si_pft)%r82d, & hio_npp_sec_si_pft => this%hvars(ih_npp_sec_si_pft)%r82d, & hio_fragmentation_scaler_sl => this%hvars(ih_fragmentation_scaler_sl)%r82d, & hio_litter_in_elem => this%hvars(ih_litter_in_elem)%r82d, & @@ -3221,6 +3229,7 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_npatches_si_age => this%hvars(ih_npatches_si_age)%r82d, & hio_zstar_si_age => this%hvars(ih_zstar_si_age)%r82d, & hio_biomass_si_age => this%hvars(ih_biomass_si_age)%r82d, & + hio_npp_si_age => this%hvars(ih_npp_si_age)%r82d, & hio_agesince_anthrodist_si_age => this%hvars(ih_agesince_anthrodist_si_age)%r82d, & hio_secondarylands_area_si_age => this%hvars(ih_secondarylands_area_si_age)%r82d, & hio_area_si_landuse => this%hvars(ih_area_si_landuse)%r82d, & @@ -3852,6 +3861,11 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_biomass_si_scls(io_si,scls) = hio_biomass_si_scls(io_si,scls) + & total_m * ccohort%n * AREA_INV + ! age-resolved cohort-based areas + + hio_npp_si_age(io_si,cpatch%age_class) = hio_npp_si_age(io_si,cpatch%age_class) + & + ccohort%n * ccohort%npp_acc_hold * AREA_INV / days_per_year / sec_per_day + ! update size-class x patch-age related quantities iscag = get_sizeage_class_index(ccohort%dbh,cpatch%age) @@ -4873,13 +4887,11 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) associate( hio_gpp_si => this%hvars(ih_gpp_si)%r81d, & hio_gpp_secondary_si => this%hvars(ih_gpp_secondary_si)%r81d, & - hio_npp_si => this%hvars(ih_npp_si)%r81d, & hio_npp_secondary_si => this%hvars(ih_npp_secondary_si)%r81d, & hio_aresp_si => this%hvars(ih_aresp_si)%r81d, & hio_aresp_secondary_si => this%hvars(ih_aresp_secondary_si)%r81d, & hio_maint_resp_si => this%hvars(ih_maint_resp_si)%r81d, & hio_maint_resp_secondary_si => this%hvars(ih_maint_resp_secondary_si)%r81d, & - hio_growth_resp_si => this%hvars(ih_growth_resp_si)%r81d, & hio_growth_resp_secondary_si => this%hvars(ih_growth_resp_secondary_si)%r81d, & hio_c_stomata_si => this%hvars(ih_c_stomata_si)%r81d, & hio_c_lblayer_si => this%hvars(ih_c_lblayer_si)%r81d, & @@ -5012,10 +5024,7 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! scale up cohort fluxes to the site level ! these fluxes have conversions of [kg/plant/timestep] -> [kg/m2/s] - hio_npp_si(io_si) = hio_npp_si(io_si) + & - ccohort%npp_tstep * n_perm2 * dt_tstep_inv - - ! Net Ecosystem Production [kgC/m2/s] + ! Net Ecosystem Production [kgC/m2/s] !cdk: use yesterday's growth respiration hio_nep_si(io_si) = hio_nep_si(io_si) + & ccohort%npp_tstep * n_perm2 * dt_tstep_inv @@ -5025,9 +5034,6 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_aresp_si(io_si) = hio_aresp_si(io_si) + & ccohort%resp_tstep * n_perm2 * dt_tstep_inv - hio_growth_resp_si(io_si) = hio_growth_resp_si(io_si) + & - ccohort%resp_g_tstep * n_perm2 * dt_tstep_inv - hio_maint_resp_si(io_si) = hio_maint_resp_si(io_si) + & ccohort%resp_m * n_perm2 * dt_tstep_inv @@ -5036,18 +5042,9 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! Secondary forest only if(cpatch%land_use_label .eq. secondaryland) then - hio_npp_secondary_si(io_si) = hio_npp_secondary_si(io_si) + & - ccohort%npp_tstep * n_perm2 * dt_tstep_inv - hio_gpp_secondary_si(io_si) = hio_gpp_secondary_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv - hio_aresp_secondary_si(io_si) = hio_aresp_secondary_si(io_si) + & - ccohort%resp_tstep * n_perm2 * dt_tstep_inv - - hio_growth_resp_secondary_si(io_si) = hio_growth_resp_secondary_si(io_si) + & - ccohort%resp_g_tstep * n_perm2 * dt_tstep_inv - hio_maint_resp_secondary_si(io_si) = hio_maint_resp_secondary_si(io_si) + & ccohort%resp_m * n_perm2 * dt_tstep_inv end if @@ -5069,17 +5066,11 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_gpp_canopy_si(io_si) = hio_gpp_canopy_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv - hio_ar_canopy_si(io_si) = hio_ar_canopy_si(io_si) + & - ccohort%resp_tstep * n_perm2 * dt_tstep_inv - else hio_gpp_understory_si(io_si) = hio_gpp_understory_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv - hio_ar_understory_si(io_si) = hio_ar_understory_si(io_si) + & - ccohort%resp_tstep * n_perm2 * dt_tstep_inv - end if end if if_notnew @@ -5165,7 +5156,6 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_resp_g_understory_si_scls => this%hvars(ih_resp_g_understory_si_scls)%r82d, & hio_resp_m_understory_si_scls => this%hvars(ih_resp_m_understory_si_scls)%r82d, & hio_gpp_si_age => this%hvars(ih_gpp_si_age)%r82d, & - hio_npp_si_age => this%hvars(ih_npp_si_age)%r82d, & hio_c_stomata_si_age => this%hvars(ih_c_stomata_si_age)%r82d, & hio_c_lblayer_si_age => this%hvars(ih_c_lblayer_si_age)%r82d, & hio_parsun_z_si_cnlf => this%hvars(ih_parsun_z_si_cnlf)%r82d, & @@ -5285,9 +5275,6 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_gpp_si_age(io_si,cpatch%age_class) = hio_gpp_si_age(io_si,cpatch%age_class) & + ccohort%gpp_tstep * ccohort%n * dt_tstep_inv - hio_npp_si_age(io_si,cpatch%age_class) = hio_npp_si_age(io_si,cpatch%age_class) & - + npp * ccohort%n * dt_tstep_inv - ! accumulate fluxes on canopy- and understory- separated fluxes if (ccohort%canopy_layer .eq. 1) then @@ -5522,11 +5509,8 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) if (patch_area_by_age(ipa2) .gt. nearzero) then hio_gpp_si_age(io_si, ipa2) = & hio_gpp_si_age(io_si, ipa2) / (patch_area_by_age(ipa2)) - hio_npp_si_age(io_si, ipa2) = & - hio_npp_si_age(io_si, ipa2) / (patch_area_by_age(ipa2)) else hio_gpp_si_age(io_si, ipa2) = 0._r8 - hio_npp_si_age(io_si, ipa2) = 0._r8 endif ! Normalize resistance diagnostics @@ -6367,6 +6351,18 @@ subroutine define_history_vars(this, initialize_variables) upfreq=group_dyna_simple, ivar=ivar, initialize=initialize_variables, & index = ih_reproc_si) + call this%set_history_var(vname='FATES_NPP', units='kg m-2 s-1', & + long='net primary production in kg carbon per m2 per second', & + use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & + upfreq=group_dyna_simple, ivar=ivar, initialize=initialize_variables, index = ih_npp_si) + + call this%set_history_var(vname='FATES_GROWTH_RESP', units='kg m-2 s-1', & + long='growth respiration in kg carbon per m2 per second', & + use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & + upfreq=group_dyna_simple, ivar=ivar, initialize=initialize_variables, & + index = ih_growth_resp_si) + + ! Output specific to the chemical species dynamics used (parteh) call this%set_history_var(vname='FATES_L2FR', units='kg kg-1', & long='The leaf to fineroot biomass multiplier for target allometry', & @@ -7383,6 +7379,12 @@ subroutine define_history_vars(this, initialize_variables) hlms='CLM:ALM', upfreq=group_dyna_complx, ivar=ivar, & initialize=initialize_variables, index = ih_npp_si_agepft) + call this%set_history_var(vname='FATES_NPP_AP', units='kg m-2 s-1', & + long='net primary productivity by age bin in kg carbon per m2 per second', & + use_default='inactive', avgflag='A', vtype=site_age_r8, & + hlms='CLM:ALM', upfreq=group_dyna_complx, ivar=ivar, initialize=initialize_variables, & + index = ih_npp_si_age) + call this%set_history_var(vname='FATES_VEGC_APPF',units = 'kg m-2', & long='biomass per PFT in each age bin in kg carbon per m2', & use_default='inactive', avgflag='A', vtype=site_agepft_r8, & @@ -8487,11 +8489,6 @@ subroutine define_history_vars(this, initialize_variables) ! Ecosystem Carbon Fluxes (updated rapidly, upfreq=group_hifr_simple) - call this%set_history_var(vname='FATES_NPP', units='kg m-2 s-1', & - long='net primary production in kg carbon per m2 per second', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_npp_si) - call this%set_history_var(vname='FATES_NPP_SECONDARY', units='kg m-2 s-1', & long='net primary production in kg carbon per m2 per second, secondary patches', & use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & @@ -8517,12 +8514,6 @@ subroutine define_history_vars(this, initialize_variables) use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_aresp_secondary_si) - call this%set_history_var(vname='FATES_GROWTH_RESP', units='kg m-2 s-1', & - long='growth respiration in kg carbon per m2 per second', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, & - index = ih_growth_resp_si) - call this%set_history_var(vname='FATES_GROWTH_RESP_SECONDARY', units='kg m-2 s-1', & long='growth respiration in kg carbon per m2 per second, secondary patches', & use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & @@ -8669,12 +8660,6 @@ subroutine define_history_vars(this, initialize_variables) ! over the short timestep. We turn off these variables when we want ! to save time (and some space) - call this%set_history_var(vname='FATES_NPP_AP', units='kg m-2 s-1', & - long='net primary productivity by age bin in kg carbon per m2 per second', & - use_default='inactive', avgflag='A', vtype=site_age_r8, & - hlms='CLM:ALM', upfreq=group_hifr_complx, ivar=ivar, initialize=initialize_variables, & - index = ih_npp_si_age) - call this%set_history_var(vname='FATES_GPP_AP', units='kg m-2 s-1', & long='gross primary productivity by age bin in kg carbon per m2 per second', & use_default='inactive', avgflag='A', vtype=site_age_r8, & From 2df2832f64e035d46ae5cd8052b2d1bc84b09d17 Mon Sep 17 00:00:00 2001 From: Charlie Koven Date: Thu, 9 May 2024 14:27:44 -0700 Subject: [PATCH 03/11] more edits to history fields --- biogeophys/FatesPlantRespPhotosynthMod.F90 | 6 +-- main/FatesHistoryInterfaceMod.F90 | 59 +++++++--------------- 2 files changed, 20 insertions(+), 45 deletions(-) diff --git a/biogeophys/FatesPlantRespPhotosynthMod.F90 b/biogeophys/FatesPlantRespPhotosynthMod.F90 index 430ca555bf..e66b9138f9 100644 --- a/biogeophys/FatesPlantRespPhotosynthMod.F90 +++ b/biogeophys/FatesPlantRespPhotosynthMod.F90 @@ -735,7 +735,7 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) ! Zero cohort flux accumulators. currentCohort%npp_tstep = 0.0_r8 - currentCohort%resp_tstep = 0.0_r8 + currentCohort%resp_m_tstep = 0.0_r8 currentCohort%gpp_tstep = 0.0_r8 currentCohort%rdark = 0.0_r8 currentCohort%resp_m = 0.0_r8 @@ -987,12 +987,12 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) ! convert from kgC/indiv/s to kgC/indiv/timestep currentCohort%resp_m = currentCohort%resp_m * dtime - currentCohort%resp_m_tstep = currentCohort%resp_m ! these two things are the same. we should get rid of one of them. which? + currentCohort%resp_m_tstep = currentCohort%resp_m ! these two things are the same. we should get rid of one them currentCohort%gpp_tstep = currentCohort%gpp_tstep * dtime currentCohort%ts_net_uptake = currentCohort%ts_net_uptake * dtime if ( debug ) write(fates_log(),*) 'EDPhoto 911 ', currentCohort%gpp_tstep - if ( debug ) write(fates_log(),*) 'EDPhoto 912 ', currentCohort%resp_tstep + if ( debug ) write(fates_log(),*) 'EDPhoto 912 ', currentCohort%resp_m_tstep if ( debug ) write(fates_log(),*) 'EDPhoto 913 ', currentCohort%resp_m diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index a2faecb9c0..a6934c26ea 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -349,11 +349,9 @@ module FatesHistoryInterfaceMod integer :: ih_understory_biomass_si integer :: ih_maint_resp_unreduced_si - integer :: ih_npp_secondary_si + ! these "secondary" flux diagnostics should be reorganized to be site x land-use type rather than just secondary, and also moved to daily timestep. integer :: ih_gpp_secondary_si - integer :: ih_aresp_secondary_si integer :: ih_maint_resp_secondary_si - integer :: ih_growth_resp_secondary_si integer :: ih_primaryland_fusion_error_si @@ -2378,6 +2376,7 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) hio_litter_in_si => this%hvars(ih_litter_in_si)%r81d, & hio_litter_out_si => this%hvars(ih_litter_out_si)%r81d, & hio_npp_si => this%hvars(ih_npp_si)%r82d, & + hio_aresp_si => this%hvars(ih_aresp_si)%r81d, & hio_growth_resp_si => this%hvars(ih_growth_resp_si)%r81d, & hio_seed_bank_si => this%hvars(ih_seed_bank_si)%r81d, & hio_ungerm_seed_bank_si => this%hvars(ih_ungerm_seed_bank_si)%r81d, & @@ -2822,6 +2821,9 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) hio_growth_resp_si(io_si) = hio_growth_resp_si(io_si) + & ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day + hio_aresp_si(io_si) = hio_aresp_si(io_si) + & + (ccohort%resp_g_acc_hold + ccohort%resp_m_acc_hold) * n_perm2 / days_per_year / sec_per_day + ! Turnover pools [kgC/day] * [day/yr] = [kgC/yr] sapw_m_turnover = ccohort%prt%GetTurnover(sapw_organ, carbon12_element) * days_per_year store_m_turnover = ccohort%prt%GetTurnover(store_organ, carbon12_element) * days_per_year @@ -4887,12 +4889,8 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) associate( hio_gpp_si => this%hvars(ih_gpp_si)%r81d, & hio_gpp_secondary_si => this%hvars(ih_gpp_secondary_si)%r81d, & - hio_npp_secondary_si => this%hvars(ih_npp_secondary_si)%r81d, & - hio_aresp_si => this%hvars(ih_aresp_si)%r81d, & - hio_aresp_secondary_si => this%hvars(ih_aresp_secondary_si)%r81d, & hio_maint_resp_si => this%hvars(ih_maint_resp_si)%r81d, & hio_maint_resp_secondary_si => this%hvars(ih_maint_resp_secondary_si)%r81d, & - hio_growth_resp_secondary_si => this%hvars(ih_growth_resp_secondary_si)%r81d, & hio_c_stomata_si => this%hvars(ih_c_stomata_si)%r81d, & hio_c_lblayer_si => this%hvars(ih_c_lblayer_si)%r81d, & hio_vis_rad_err_si => this%hvars(ih_vis_rad_err_si)%r81d, & @@ -5024,16 +5022,14 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! scale up cohort fluxes to the site level ! these fluxes have conversions of [kg/plant/timestep] -> [kg/m2/s] - ! Net Ecosystem Production [kgC/m2/s] !cdk: use yesterday's growth respiration + ! Net Ecosystem Production [kgC/m2/s]. Use yesterday's growth respiration hio_nep_si(io_si) = hio_nep_si(io_si) + & - ccohort%npp_tstep * n_perm2 * dt_tstep_inv + (ccohort%gpp_tstep-ccohort%resp_m) * n_perm2 * dt_tstep_inv - & + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day hio_gpp_si(io_si) = hio_gpp_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv - hio_aresp_si(io_si) = hio_aresp_si(io_si) + & - ccohort%resp_tstep * n_perm2 * dt_tstep_inv - hio_maint_resp_si(io_si) = hio_maint_resp_si(io_si) + & ccohort%resp_m * n_perm2 * dt_tstep_inv @@ -5118,7 +5114,6 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) real(r8) :: n_density ! individual of cohort per m2. real(r8) :: resp_g ! growth respiration per timestep [kgC/indiv/step] real(r8) :: npp ! npp for this time-step (adjusted for g resp) [kgC/indiv/step] - real(r8) :: aresp ! autotrophic respiration (adjusted for g resp) [kgC/indiv/step] real(r8) :: n_perm2 ! individuals per m2 for the whole column real(r8) :: patch_area_by_age(nlevage) ! patch area in each bin for normalizing purposes real(r8) :: canopy_area_by_age(nlevage) ! canopy area in each bin for normalizing purposes @@ -5234,21 +5229,17 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) if ( .not. ccohort%isnew ) then - npp = ccohort%npp_tstep - resp_g = ccohort%resp_g_tstep - aresp = ccohort%resp_tstep - ! Calculate index for the scpf class associate( scpf => ccohort%size_by_pft_class, & scls => ccohort%size_class ) - ! Total AR (kgC/m2/s) = (kgC/plant/step) / (s/step) * (plant/m2) + ! Total AR (kgC/m2/s) = (kgC/plant/step) / (s/step) * (plant/m2) ! CDK: this should be daily hio_ar_si_scpf(io_si,scpf) = hio_ar_si_scpf(io_si,scpf) + & - (ccohort%resp_tstep*dt_tstep_inv) * n_perm2 + (ccohort%resp_m_tstep*dt_tstep_inv) * n_perm2 - ! Growth AR (kgC/m2/s) + ! Growth AR (kgC/m2/s) ! CDK: this should be daily hio_ar_grow_si_scpf(io_si,scpf) = hio_ar_grow_si_scpf(io_si,scpf) + & - (resp_g*dt_tstep_inv) * n_perm2 + (ccohort%resp_g_tstep*dt_tstep_inv) * n_perm2 ! Maint AR (kgC/m2/s) hio_ar_maint_si_scpf(io_si,scpf) = hio_ar_maint_si_scpf(io_si,scpf) + & @@ -6356,6 +6347,11 @@ subroutine define_history_vars(this, initialize_variables) use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & upfreq=group_dyna_simple, ivar=ivar, initialize=initialize_variables, index = ih_npp_si) + call this%set_history_var(vname='FATES_AUTORESP', units='kg m-2 s-1', & + long='autotrophic respiration in kg carbon per m2 per second', & + use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & + upfreq=group_dyna_simple, ivar=ivar, initialize=initialize_variables, index = ih_aresp_si) + call this%set_history_var(vname='FATES_GROWTH_RESP', units='kg m-2 s-1', & long='growth respiration in kg carbon per m2 per second', & use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & @@ -8489,11 +8485,6 @@ subroutine define_history_vars(this, initialize_variables) ! Ecosystem Carbon Fluxes (updated rapidly, upfreq=group_hifr_simple) - call this%set_history_var(vname='FATES_NPP_SECONDARY', units='kg m-2 s-1', & - long='net primary production in kg carbon per m2 per second, secondary patches', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_npp_secondary_si) - call this%set_history_var(vname='FATES_GPP', units='kg m-2 s-1', & long='gross primary production in kg carbon per m2 per second', & use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & @@ -8504,22 +8495,6 @@ subroutine define_history_vars(this, initialize_variables) use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_gpp_secondary_si) - call this%set_history_var(vname='FATES_AUTORESP', units='kg m-2 s-1', & - long='autotrophic respiration in kg carbon per m2 per second', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_aresp_si) - - call this%set_history_var(vname='FATES_AUTORESP_SECONDARY', units='kg m-2 s-1', & - long='autotrophic respiration in kg carbon per m2 per second, secondary patches', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, index = ih_aresp_secondary_si) - - call this%set_history_var(vname='FATES_GROWTH_RESP_SECONDARY', units='kg m-2 s-1', & - long='growth respiration in kg carbon per m2 per second, secondary patches', & - use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & - upfreq=group_hifr_simple, ivar=ivar, initialize=initialize_variables, & - index = ih_growth_resp_secondary_si) - call this%set_history_var(vname='FATES_MAINT_RESP', units='kg m-2 s-1', & long='maintenance respiration in kg carbon per m2 land area per second', & use_default='active', avgflag='A', vtype=site_r8, hlms='CLM:ALM', & From b6cad147f50f9d4fc586eea77037b0f9cbae5861 Mon Sep 17 00:00:00 2001 From: Jessica Needham <10586303+JessicaNeedham@users.noreply.github.com> Date: Mon, 1 Jul 2024 13:46:59 -0700 Subject: [PATCH 04/11] Update main/FatesHistoryInterfaceMod.F90 --- main/FatesHistoryInterfaceMod.F90 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index a6934c26ea..a1b1010445 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -2375,7 +2375,7 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) hio_sum_fuel_si => this%hvars(ih_sum_fuel_si)%r81d, & hio_litter_in_si => this%hvars(ih_litter_in_si)%r81d, & hio_litter_out_si => this%hvars(ih_litter_out_si)%r81d, & - hio_npp_si => this%hvars(ih_npp_si)%r82d, & + hio_npp_si => this%hvars(ih_npp_si)%r81d, & hio_aresp_si => this%hvars(ih_aresp_si)%r81d, & hio_growth_resp_si => this%hvars(ih_growth_resp_si)%r81d, & hio_seed_bank_si => this%hvars(ih_seed_bank_si)%r81d, & From 9080ba37506445d2bdaa11d68e0b43316cfbc75b Mon Sep 17 00:00:00 2001 From: Charlie Koven Date: Tue, 29 Oct 2024 11:34:17 -0700 Subject: [PATCH 05/11] Apply suggestions from code review Co-authored-by: Jessica Needham <10586303+JessicaNeedham@users.noreply.github.com> --- biogeophys/FatesPlantRespPhotosynthMod.F90 | 2 +- main/EDMainMod.F90 | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/biogeophys/FatesPlantRespPhotosynthMod.F90 b/biogeophys/FatesPlantRespPhotosynthMod.F90 index e66b9138f9..eed9458993 100644 --- a/biogeophys/FatesPlantRespPhotosynthMod.F90 +++ b/biogeophys/FatesPlantRespPhotosynthMod.F90 @@ -734,7 +734,7 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) end do leaf_layer_loop ! Zero cohort flux accumulators. - currentCohort%npp_tstep = 0.0_r8 + currentCohort%resp_m_tstep = 0.0_r8 currentCohort%gpp_tstep = 0.0_r8 currentCohort%rdark = 0.0_r8 diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 22cb94d992..807deb0496 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -520,7 +520,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! at this point we have the info we need to calculate growth respiration ! as a "tax" on the difference between daily GPP and daily maintenance respiration - if (hlm_use_ed_prescribed_phys .eq. itrue) then + if (hlm_use_ed_prescribed_phys .eq. ifalse) then currentCohort%resp_g_acc = prt_params%grperc(ft) * & max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) From a0dea3b51a2370ba13ce811d6c90b662f3c06833 Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 13:27:02 -0400 Subject: [PATCH 06/11] Added respiration excess to sub-daily npp tracking by holding it like gpp does. Fixed some output units. Fixed carbon flux tracking when we bypass (ie st3 and satphen). --- biogeochem/EDCohortDynamicsMod.F90 | 5 ++- biogeochem/FatesCohortMod.F90 | 21 +++++----- biogeophys/EDAccumulateFluxesMod.F90 | 7 ---- biogeophys/FatesPlantRespPhotosynthMod.F90 | 36 +++++----------- main/EDMainMod.F90 | 49 +++++++++++++++++----- main/FatesHistoryInterfaceMod.F90 | 12 +++--- 6 files changed, 69 insertions(+), 61 deletions(-) diff --git a/biogeochem/EDCohortDynamicsMod.F90 b/biogeochem/EDCohortDynamicsMod.F90 index e362cfb96a..ce9691c2ac 100644 --- a/biogeochem/EDCohortDynamicsMod.F90 +++ b/biogeochem/EDCohortDynamicsMod.F90 @@ -1083,8 +1083,9 @@ subroutine fuse_cohorts(currentSite, currentPatch, bc_in) (currentCohort%n*currentCohort%gpp_acc_hold + & nextc%n*nextc%gpp_acc_hold)/newn - currentCohort%resp_excess = (currentCohort%n*currentCohort%resp_excess + & - nextc%n*nextc%resp_excess)/newn + currentCohort%resp_excess_hold = & + (currentCohort%n*currentCohort%resp_excess_hold + & + nextc%n*nextc%resp_excess_hold)/newn currentCohort%dmort = (currentCohort%n*currentCohort%dmort + & nextc%n*nextc%dmort)/newn diff --git a/biogeochem/FatesCohortMod.F90 b/biogeochem/FatesCohortMod.F90 index 64a4d929af..9d3b5522d8 100644 --- a/biogeochem/FatesCohortMod.F90 +++ b/biogeochem/FatesCohortMod.F90 @@ -205,11 +205,16 @@ module FatesCohortMod integer :: twostr_col ! The column index in the two-stream solution that this cohort is part of ! RESPIRATION COMPONENTS + real(r8) :: resp_excess_hold ! respiration of excess carbon [kgC/indiv/day] + ! note: this is flagged "hold" because it is calculated + ! at the end of the day (dynamics) but is used + ! on the following day (like growth respiration) + ! to aid in reporting a more accurate sub-daily + ! NEP + real(r8) :: rdark ! dark respiration [kgC/indiv/s] real(r8) :: resp_g_tstep ! growth respiration [kgC/indiv/timestep] - real(r8) :: resp_m ! maintenance respiration [kgC/indiv/timestep] real(r8) :: resp_m_unreduced ! diagnostic-only unreduced maintenance respiration [kgC/indiv/timestep] - real(r8) :: resp_excess ! respiration of excess carbon [kgC/indiv/day] real(r8) :: livestem_mr ! aboveground live stem maintenance respiration [kgC/indiv/s] real(r8) :: livecroot_mr ! belowground live stem maintenance respiration [kgC/indiv/s] real(r8) :: froot_mr ! live fine root maintenance respiration [kgC/indiv/s] @@ -415,9 +420,8 @@ subroutine NanValues(this) ! RESPIRATION COMPONENTS this%rdark = nan this%resp_g_tstep = nan - this%resp_m = nan this%resp_m_unreduced = nan - this%resp_excess = nan + this%resp_excess_hold = nan this%livestem_mr = nan this%livecroot_mr = nan this%froot_mr = nan @@ -493,6 +497,7 @@ subroutine ZeroValues(this) ! this%npp_acc_hold = nan ! this%resp_m_acc_hold = nan ! this%resp_g_acc_hold = nan + ! this%resp_excess_hold = nan this%c13disc_clm = 0._r8 this%c13disc_acc = 0._r8 @@ -522,9 +527,7 @@ subroutine ZeroValues(this) this%seed_prod = 0._r8 this%rdark = 0._r8 this%resp_g_tstep = 0._r8 - this%resp_m = 0._r8 this%resp_m_unreduced = 0._r8 - this%resp_excess = 0._r8 this%livestem_mr = 0._r8 this%livecroot_mr = 0._r8 this%froot_mr = 0._r8 @@ -751,9 +754,8 @@ subroutine Copy(this, copyCohort) ! RESPIRATION COMPONENTS copyCohort%rdark = this%rdark copyCohort%resp_g_tstep = this%resp_g_tstep - copyCohort%resp_m = this%resp_m copyCohort%resp_m_unreduced = this%resp_m_unreduced - copyCohort%resp_excess = this%resp_excess + copyCohort%resp_excess_hold = this%resp_excess_hold copyCohort%livestem_mr = this%livestem_mr copyCohort%livecroot_mr = this%livecroot_mr copyCohort%froot_mr = this%froot_mr @@ -886,7 +888,7 @@ subroutine InitPRTBoundaryConditions(this) call this%prt%RegisterBCIn(acnp_bc_in_id_cdamage, bc_ival=this%crowndamage) call this%prt%RegisterBCInOut(acnp_bc_inout_id_dbh, bc_rval=this%dbh) - call this%prt%RegisterBCInOut(acnp_bc_inout_id_resp_excess, bc_rval=this%resp_excess) + call this%prt%RegisterBCInOut(acnp_bc_inout_id_resp_excess, bc_rval=this%resp_excess_hold) call this%prt%RegisterBCInOut(acnp_bc_inout_id_l2fr, bc_rval=this%l2fr) call this%prt%RegisterBCInOut(acnp_bc_inout_id_cx_int, bc_rval=this%cx_int) call this%prt%RegisterBCInOut(acnp_bc_inout_id_emadcxdt, bc_rval=this%ema_dcxdt) @@ -1060,7 +1062,6 @@ subroutine Dump(this) write(fates_log(),*) 'cohort%resp_g_acc = ', this%resp_g_acc write(fates_log(),*) 'cohort%resp_g_acc_hold = ', this%resp_g_acc_hold write(fates_log(),*) 'cohort%rdark = ', this%rdark - write(fates_log(),*) 'cohort%resp_m = ', this%resp_m write(fates_log(),*) 'cohort%livestem_mr = ', this%livestem_mr write(fates_log(),*) 'cohort%livecroot_mr = ', this%livecroot_mr write(fates_log(),*) 'cohort%froot_mr = ', this%froot_mr diff --git a/biogeophys/EDAccumulateFluxesMod.F90 b/biogeophys/EDAccumulateFluxesMod.F90 index d74c965e31..5bab6b5191 100644 --- a/biogeophys/EDAccumulateFluxesMod.F90 +++ b/biogeophys/EDAccumulateFluxesMod.F90 @@ -80,13 +80,6 @@ subroutine AccumulateFluxes_ED(nsites, sites, bc_in, bc_out, dt_time) ! Accumulate fluxes from hourly to daily values. ! _tstep fluxes are KgC/indiv/timestep _acc are KgC/indiv/day - if ( debug ) then - - write(fates_log(),*) 'EDAccumFlux 66 ',ccohort%gpp_tstep - write(fates_log(),*) 'EDAccumFlux 67 ',ccohort%resp_m_tstep - - endif - ccohort%gpp_acc = ccohort%gpp_acc + ccohort%gpp_tstep ccohort%resp_m_acc = ccohort%resp_m_acc + ccohort%resp_m_tstep diff --git a/biogeophys/FatesPlantRespPhotosynthMod.F90 b/biogeophys/FatesPlantRespPhotosynthMod.F90 index eed9458993..3578a92e3b 100644 --- a/biogeophys/FatesPlantRespPhotosynthMod.F90 +++ b/biogeophys/FatesPlantRespPhotosynthMod.F90 @@ -738,7 +738,6 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) currentCohort%resp_m_tstep = 0.0_r8 currentCohort%gpp_tstep = 0.0_r8 currentCohort%rdark = 0.0_r8 - currentCohort%resp_m = 0.0_r8 currentCohort%ts_net_uptake = 0.0_r8 currentCohort%c13disc_clm = 0.0_r8 @@ -963,39 +962,24 @@ subroutine FatesPlantRespPhotosynthDrive (nsites, sites,bc_in,bc_out,dtime) ! calcualate some fluxes that are sums and nets of the base fluxes ! ------------------------------------------------------------------ - if ( debug ) write(fates_log(),*) 'EDPhoto 904 ', currentCohort%resp_m - if ( debug ) write(fates_log(),*) 'EDPhoto 905 ', currentCohort%rdark - if ( debug ) write(fates_log(),*) 'EDPhoto 906 ', currentCohort%livestem_mr - if ( debug ) write(fates_log(),*) 'EDPhoto 907 ', currentCohort%livecroot_mr - if ( debug ) write(fates_log(),*) 'EDPhoto 908 ', currentCohort%froot_mr - - - ! add on whole plant respiration values in kgC/indiv/s-1 - currentCohort%resp_m = currentCohort%livestem_mr + & + currentCohort%resp_m_tstep = currentCohort%livestem_mr + & currentCohort%livecroot_mr + & - currentCohort%froot_mr - + currentCohort%froot_mr + & + currentCohort%rdark + ! no drought response right now.. something like: - ! resp_m = resp_m * (1.0_r8 - currentPatch%btran_ft(currentCohort%pft) * & + ! resp_m_tstep = resp_m_tstep * (1.0_r8 - currentPatch%btran_ft(currentCohort%pft) * & ! EDPftvarcon_inst%resp_drought_response(ft)) - currentCohort%resp_m = currentCohort%resp_m + currentCohort%rdark - - ! save as a diagnostic the un-throttled maintenance respiration to be able to know how strong this is - currentCohort%resp_m_unreduced = currentCohort%resp_m / maintresp_reduction_factor - ! convert from kgC/indiv/s to kgC/indiv/timestep - currentCohort%resp_m = currentCohort%resp_m * dtime - currentCohort%resp_m_tstep = currentCohort%resp_m ! these two things are the same. we should get rid of one them + currentCohort%resp_m_tstep = currentCohort%resp_m_tstep * dtime currentCohort%gpp_tstep = currentCohort%gpp_tstep * dtime currentCohort%ts_net_uptake = currentCohort%ts_net_uptake * dtime - - if ( debug ) write(fates_log(),*) 'EDPhoto 911 ', currentCohort%gpp_tstep - if ( debug ) write(fates_log(),*) 'EDPhoto 912 ', currentCohort%resp_m_tstep - if ( debug ) write(fates_log(),*) 'EDPhoto 913 ', currentCohort%resp_m - - + + ! save as a diagnostic the un-throttled maintenance respiration to be able to know how strong this is + currentCohort%resp_m_unreduced = currentCohort%resp_m_tstep / maintresp_reduction_factor + ! Accumulate the combined conductance (stomatal+leaf boundary layer) ! Note that currentCohort%g_sb_laweight is weighted by the leaf area ! of each cohort and has units of [m/s] * [m2 leaf] diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 807deb0496..5aef853476 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -475,7 +475,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentPatch%btran_ft, mean_temp, & currentPatch%land_use_label, & currentPatch%age_since_anthro_disturbance, frac_site_primary, & - harvestable_forest_c, harvest_tag) + harvestable_forest_c, harvest_tag) ! ----------------------------------------------------------------------------- ! Apply Plant Allocation and Reactive Transport @@ -532,14 +532,11 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) endif ! calculate the npp as the difference between gpp and autotrophic respiration + ! (NPP is also updated if there is any excess respiration from nutrient limitations) currentCohort%npp_acc = currentCohort%gpp_acc - (currentCohort%resp_m_acc + currentCohort%resp_g_acc) currentCohort%npp_acc_hold = currentCohort%gpp_acc_hold - (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) - ! Passing gpp_acc_hold to HLM - bc_out%gpp_site = bc_out%gpp_site + currentCohort%gpp_acc_hold * & - AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day - bc_out%ar_site = bc_out%ar_site + (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) * & - AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day + ! Conduct Maintenance Turnover (parteh) if(debug) call currentCohort%prt%CheckMassConservation(ft,3) @@ -567,7 +564,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentCohort%daily_n_gain = currentCohort%daily_nh4_uptake + & currentCohort%daily_no3_uptake + currentCohort%sym_nfix_daily - currentCohort%resp_excess = 0._r8 + currentCohort%resp_excess_hold = 0._r8 end if if_not_newlyrecovered @@ -616,11 +613,28 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) end if call currentCohort%prt%DailyPRT(phase=3) + + ! If nutrients are limiting growth, and carbon continues + ! to accumulate beyond the plant's storage capacity, then + ! it will burn carbon as what we call "excess respiration" + ! We must subtract this term from NPP + + currentCohort%npp_acc_hold = currentCohort%npp_acc_hold - currentCohort%resp_excess_hold + + ! Passing gpp_acc_hold to HLM + bc_out%gpp_site = bc_out%gpp_site + currentCohort%gpp_acc_hold * & + AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day + bc_out%ar_site = bc_out%ar_site + (currentCohort%resp_m_acc_hold + & + currentCohort%resp_g_acc_hold + currentCohort%resp_excess_hold) * & + AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day + ! Update the mass balance tracking for the daily nutrient uptake flux ! Then zero out the daily uptakes, they have been used ! ----------------------------------------------------------------------------- + + call EffluxIntoLitterPools(currentSite, currentPatch, currentCohort, bc_in ) @@ -647,7 +661,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentCohort%gpp_acc * currentCohort%n site_cmass%aresp_acc = site_cmass%aresp_acc + & - (currentCohort%resp_m_acc+currentCohort%resp_g_acc+currentCohort%resp_excess) * & + (currentCohort%resp_m_acc+currentCohort%resp_g_acc+currentCohort%resp_excess_hold) * & currentCohort%n call currentCohort%prt%CheckMassConservation(ft,5) @@ -1004,7 +1018,7 @@ subroutine TotalBalanceCheck (currentSite, call_index ) write(fates_log(),*) 'leaf: ',leaf_m,' structure: ',struct_m,' store: ',store_m write(fates_log(),*) 'fineroot: ',fnrt_m,' repro: ',repro_m,' sapwood: ',sapw_m write(fates_log(),*) 'num plant: ',currentCohort%n - write(fates_log(),*) 'resp excess: ',currentCohort%resp_excess*currentCohort%n + write(fates_log(),*) 'resp excess: ',currentCohort%resp_excess_hold*currentCohort%n if(element_list(el).eq.nitrogen_element) then write(fates_log(),*) 'NH4 uptake: ',currentCohort%daily_nh4_uptake*currentCohort%n @@ -1059,6 +1073,9 @@ subroutine bypass_dynamics(currentSite) type(fates_patch_type), pointer :: currentPatch type(fates_cohort_type), pointer :: currentCohort + bc_out%gpp_site = 0._r8 + bc_out%ar_site = 0._r8 + currentPatch => currentSite%youngest_patch do while(associated(currentPatch)) currentCohort => currentPatch%shortest @@ -1066,11 +1083,16 @@ subroutine bypass_dynamics(currentSite) currentCohort%isnew=.false. + + currentCohort%resp_g_acc = prt_params%grperc(ft) * & + max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) + currentCohort%npp_acc_hold = currentCohort%npp_acc * real(hlm_days_per_year,r8) currentCohort%gpp_acc_hold = currentCohort%gpp_acc * real(hlm_days_per_year,r8) currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) currentCohort%resp_m_acc_hold = currentCohort%resp_m_acc * real(hlm_days_per_year,r8) - + + currentCohort%resp_excess_hold = 0._r8 currentCohort%npp_acc = 0.0_r8 currentCohort%gpp_acc = 0.0_r8 currentCohort%resp_g_acc = 0.0_r8 @@ -1097,6 +1119,13 @@ subroutine bypass_dynamics(currentSite) ! as they should just be zero, no uptake ! in ST3 mode. + ! Passing + bc_out%gpp_site = bc_out%gpp_site + currentCohort%gpp_acc_hold * & + AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day + bc_out%ar_site = bc_out%ar_site + (currentCohort%resp_m_acc_hold + & + currentCohort%resp_g_acc_hold + currentCohort%resp_excess_hold) * & + AREA_INV * currentCohort%n / hlm_days_per_year / sec_per_day + currentCohort => currentCohort%taller enddo currentPatch => currentPatch%older diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index a1b1010445..cc6a61c157 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -5024,14 +5024,14 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! Net Ecosystem Production [kgC/m2/s]. Use yesterday's growth respiration hio_nep_si(io_si) = hio_nep_si(io_si) + & - (ccohort%gpp_tstep-ccohort%resp_m) * n_perm2 * dt_tstep_inv - & + (ccohort%gpp_tstep-ccohort%resp_m_tstep) * n_perm2 * dt_tstep_inv - & ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day hio_gpp_si(io_si) = hio_gpp_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv hio_maint_resp_si(io_si) = hio_maint_resp_si(io_si) + & - ccohort%resp_m * n_perm2 * dt_tstep_inv + ccohort%resp_m_tstep * n_perm2 * dt_tstep_inv hio_maint_resp_unreduced_si(io_si) = hio_maint_resp_unreduced_si(io_si) + & ccohort%resp_m_unreduced * n_perm2 * dt_tstep_inv @@ -5042,7 +5042,7 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ccohort%gpp_tstep * n_perm2 * dt_tstep_inv hio_maint_resp_secondary_si(io_si) = hio_maint_resp_secondary_si(io_si) + & - ccohort%resp_m * n_perm2 * dt_tstep_inv + ccohort%resp_m_tstep * n_perm2 * dt_tstep_inv end if ! Maintenance respiration of different organs @@ -5243,7 +5243,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! Maint AR (kgC/m2/s) hio_ar_maint_si_scpf(io_si,scpf) = hio_ar_maint_si_scpf(io_si,scpf) + & - (ccohort%resp_m*dt_tstep_inv) * n_perm2 + (ccohort%resp_m_tstep*dt_tstep_inv) * n_perm2 ! Maintenance AR partition variables are stored as rates (kgC/plant/s) ! (kgC/m2/s) = (kgC/plant/s) * (plant/m2) @@ -5281,7 +5281,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_resp_g_canopy_si_scls(io_si,scls) = hio_resp_g_canopy_si_scls(io_si,scls) + & resp_g * ccohort%n * dt_tstep_inv * ha_per_m2 hio_resp_m_canopy_si_scls(io_si,scls) = hio_resp_m_canopy_si_scls(io_si,scls) + & - ccohort%resp_m * ccohort%n * dt_tstep_inv * ha_per_m2 + ccohort%resp_m_tstep * ccohort%n * dt_tstep_inv * ha_per_m2 else ! size-resolved respiration fluxes are in kg C / m2 / s @@ -5296,7 +5296,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_resp_g_understory_si_scls(io_si,scls) = hio_resp_g_understory_si_scls(io_si,scls) + & resp_g * ccohort%n * dt_tstep_inv * ha_per_m2 hio_resp_m_understory_si_scls(io_si,scls) = hio_resp_m_understory_si_scls(io_si,scls) + & - ccohort%resp_m * ccohort%n * dt_tstep_inv * ha_per_m2 + ccohort%resp_m_tstep * ccohort%n * dt_tstep_inv * ha_per_m2 endif end associate endif From f5a7caad435700def48cf5ff8ac7ba3853e9d8be Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 14:04:24 -0400 Subject: [PATCH 07/11] removing resp_g_acc --- biogeochem/FatesCohortMod.F90 | 4 ---- main/EDMainMod.F90 | 27 +++++++++++++-------------- main/FatesHistoryInterfaceMod.F90 | 28 ++++++++++++++-------------- 3 files changed, 27 insertions(+), 32 deletions(-) diff --git a/biogeochem/FatesCohortMod.F90 b/biogeochem/FatesCohortMod.F90 index 9d3b5522d8..495b1818cf 100644 --- a/biogeochem/FatesCohortMod.F90 +++ b/biogeochem/FatesCohortMod.F90 @@ -213,7 +213,6 @@ module FatesCohortMod ! NEP real(r8) :: rdark ! dark respiration [kgC/indiv/s] - real(r8) :: resp_g_tstep ! growth respiration [kgC/indiv/timestep] real(r8) :: resp_m_unreduced ! diagnostic-only unreduced maintenance respiration [kgC/indiv/timestep] real(r8) :: livestem_mr ! aboveground live stem maintenance respiration [kgC/indiv/s] real(r8) :: livecroot_mr ! belowground live stem maintenance respiration [kgC/indiv/s] @@ -419,7 +418,6 @@ subroutine NanValues(this) ! RESPIRATION COMPONENTS this%rdark = nan - this%resp_g_tstep = nan this%resp_m_unreduced = nan this%resp_excess_hold = nan this%livestem_mr = nan @@ -526,7 +524,6 @@ subroutine ZeroValues(this) this%daily_p_demand = -9._r8 this%seed_prod = 0._r8 this%rdark = 0._r8 - this%resp_g_tstep = 0._r8 this%resp_m_unreduced = 0._r8 this%livestem_mr = 0._r8 this%livecroot_mr = 0._r8 @@ -753,7 +750,6 @@ subroutine Copy(this, copyCohort) ! RESPIRATION COMPONENTS copyCohort%rdark = this%rdark - copyCohort%resp_g_tstep = this%resp_g_tstep copyCohort%resp_m_unreduced = this%resp_m_unreduced copyCohort%resp_excess_hold = this%resp_excess_hold copyCohort%livestem_mr = this%livestem_mr diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 5aef853476..d49d7c40d1 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -521,21 +521,22 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! at this point we have the info we need to calculate growth respiration ! as a "tax" on the difference between daily GPP and daily maintenance respiration if (hlm_use_ed_prescribed_phys .eq. ifalse) then - currentCohort%resp_g_acc = prt_params%grperc(ft) * & - max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) - currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) + + currentCohort%resp_g_acc_hold = prt_params%grperc(ft) * & + max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) * real(hlm_days_per_year,r8) + else ! set growth respiration to zero in prescribed physiology mode, ! that way the npp_acc vars will be set to the nominal gpp values set above. - currentCohort%resp_g_acc = 0._r8 currentCohort%resp_g_acc_hold = 0._r8 endif ! calculate the npp as the difference between gpp and autotrophic respiration ! (NPP is also updated if there is any excess respiration from nutrient limitations) - currentCohort%npp_acc = currentCohort%gpp_acc - (currentCohort%resp_m_acc + currentCohort%resp_g_acc) - currentCohort%npp_acc_hold = currentCohort%gpp_acc_hold - (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) - + currentCohort%npp_acc = currentCohort%gpp_acc - & + (currentCohort%resp_m_acc + currentCohort%resp_g_acc_hold/real(hlm_days_per_year,r8)) + currentCohort%npp_acc_hold = currentCohort%gpp_acc_hold - & + (currentCohort%resp_m_acc_hold + currentCohort%resp_g_acc_hold) ! Conduct Maintenance Turnover (parteh) @@ -661,8 +662,9 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentCohort%gpp_acc * currentCohort%n site_cmass%aresp_acc = site_cmass%aresp_acc + & - (currentCohort%resp_m_acc+currentCohort%resp_g_acc+currentCohort%resp_excess_hold) * & - currentCohort%n + currentCohort%resp_m_acc*currentCohort%n + & + (currentCohort%resp_g_acc_hold+currentCohort%resp_excess_hold) * & + currentCohort%n/real( hlm_days_per_year,r8) call currentCohort%prt%CheckMassConservation(ft,5) @@ -1083,19 +1085,16 @@ subroutine bypass_dynamics(currentSite) currentCohort%isnew=.false. - - currentCohort%resp_g_acc = prt_params%grperc(ft) * & - max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc)) + currentCohort%resp_g_acc_hold = prt_params%grperc(ft) * & + max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc))*real(hlm_days_per_year,r8) currentCohort%npp_acc_hold = currentCohort%npp_acc * real(hlm_days_per_year,r8) currentCohort%gpp_acc_hold = currentCohort%gpp_acc * real(hlm_days_per_year,r8) - currentCohort%resp_g_acc_hold = currentCohort%resp_g_acc * real(hlm_days_per_year,r8) currentCohort%resp_m_acc_hold = currentCohort%resp_m_acc * real(hlm_days_per_year,r8) currentCohort%resp_excess_hold = 0._r8 currentCohort%npp_acc = 0.0_r8 currentCohort%gpp_acc = 0.0_r8 - currentCohort%resp_g_acc = 0.0_r8 currentCohort%resp_m_acc = 0.0_r8 ! No need to set the "net_art" terms to zero diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index cc6a61c157..d8ed8987ba 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -2822,7 +2822,8 @@ subroutine update_history_dyn1(this,nc,nsites,sites,bc_in) ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day hio_aresp_si(io_si) = hio_aresp_si(io_si) + & - (ccohort%resp_g_acc_hold + ccohort%resp_m_acc_hold) * n_perm2 / days_per_year / sec_per_day + (ccohort%resp_g_acc_hold + ccohort%resp_m_acc_hold + & + ccohort%resp_excess_hold) * n_perm2 / days_per_year / sec_per_day ! Turnover pools [kgC/day] * [day/yr] = [kgC/yr] sapw_m_turnover = ccohort%prt%GetTurnover(sapw_organ, carbon12_element) * days_per_year @@ -3934,7 +3935,8 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_gpp_canopy_si_scpf(io_si,scpf) = hio_gpp_canopy_si_scpf(io_si,scpf) + & n_perm2*ccohort%gpp_acc_hold / days_per_year / sec_per_day hio_ar_canopy_si_scpf(io_si,scpf) = hio_ar_canopy_si_scpf(io_si,scpf) + & - n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold) / days_per_year / sec_per_day + n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold + & + ccohort%resp_excess_hold) / days_per_year / sec_per_day ! growth increment hio_ddbh_canopy_si_scpf(io_si,scpf) = hio_ddbh_canopy_si_scpf(io_si,scpf) + & ccohort%ddbhdt*ccohort%n * m_per_cm / m2_per_ha @@ -4069,7 +4071,8 @@ subroutine update_history_dyn2(this,nc,nsites,sites,bc_in) hio_gpp_understory_si_scpf(io_si,scpf) = hio_gpp_understory_si_scpf(io_si,scpf) + & n_perm2*ccohort%gpp_acc_hold / days_per_year / sec_per_day hio_ar_understory_si_scpf(io_si,scpf) = hio_ar_understory_si_scpf(io_si,scpf) + & - n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold) / days_per_year / sec_per_day + n_perm2*(ccohort%resp_m_acc_hold + ccohort%resp_g_acc_hold + & + ccohort%resp_excess_hold) / days_per_year / sec_per_day ! growth increment hio_ddbh_understory_si_scpf(io_si,scpf) = hio_ddbh_understory_si_scpf(io_si,scpf) + & @@ -5025,7 +5028,7 @@ subroutine update_history_hifrq1(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! Net Ecosystem Production [kgC/m2/s]. Use yesterday's growth respiration hio_nep_si(io_si) = hio_nep_si(io_si) + & (ccohort%gpp_tstep-ccohort%resp_m_tstep) * n_perm2 * dt_tstep_inv - & - ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day + (ccohort%resp_g_acc_hold+ccohort%resp_excess_hold) * n_perm2 / days_per_year / sec_per_day hio_gpp_si(io_si) = hio_gpp_si(io_si) + & ccohort%gpp_tstep * n_perm2 * dt_tstep_inv @@ -5112,14 +5115,12 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) integer :: ivar ! index of IO variable object vector integer :: ft ! functional type index real(r8) :: n_density ! individual of cohort per m2. - real(r8) :: resp_g ! growth respiration per timestep [kgC/indiv/step] - real(r8) :: npp ! npp for this time-step (adjusted for g resp) [kgC/indiv/step] real(r8) :: n_perm2 ! individuals per m2 for the whole column real(r8) :: patch_area_by_age(nlevage) ! patch area in each bin for normalizing purposes real(r8) :: canopy_area_by_age(nlevage) ! canopy area in each bin for normalizing purposes real(r8) :: site_area_veg_inv ! 1/area of the site that is not bare-ground integer :: ipa2 ! patch incrementer - integer :: clllpf_indx, cnlf_indx, ipft, ican, ileaf ! more iterators and indices + integer :: clllpf_indx, cnlf_indx, ipft, ican, ileaf ! more iterators and indices real(r8) :: clllpf_area ! area footprint (m2) for the current cl x ll x pft bin real(r8) :: clll_area ! area footprint (m2) for the cl x ll bin (ie adds up pfts in parallel) real(r8) :: cl_area ! total weight of all ll x pft bins in the canopy layer @@ -5211,8 +5212,6 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) canopy_area_by_age(cpatch%age_class) = & canopy_area_by_age(cpatch%age_class) + cpatch%total_canopy_area - - ! Canopy resitance terms hio_c_stomata_si_age(io_si,cpatch%age_class) = & hio_c_stomata_si_age(io_si,cpatch%age_class) + & @@ -5233,13 +5232,14 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) associate( scpf => ccohort%size_by_pft_class, & scls => ccohort%size_class ) - ! Total AR (kgC/m2/s) = (kgC/plant/step) / (s/step) * (plant/m2) ! CDK: this should be daily + ! Total AR (kgC/m2/s) = (kgC/plant/step) / (s/step) * (plant/m2) hio_ar_si_scpf(io_si,scpf) = hio_ar_si_scpf(io_si,scpf) + & - (ccohort%resp_m_tstep*dt_tstep_inv) * n_perm2 + (ccohort%resp_m_tstep*dt_tstep_inv) * n_perm2 + & + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day ! Growth AR (kgC/m2/s) ! CDK: this should be daily hio_ar_grow_si_scpf(io_si,scpf) = hio_ar_grow_si_scpf(io_si,scpf) + & - (ccohort%resp_g_tstep*dt_tstep_inv) * n_perm2 + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day ! Maint AR (kgC/m2/s) hio_ar_maint_si_scpf(io_si,scpf) = hio_ar_maint_si_scpf(io_si,scpf) + & @@ -5279,7 +5279,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_froot_mr_canopy_si_scls(io_si,scls) = hio_froot_mr_canopy_si_scls(io_si,scls) + & ccohort%froot_mr * ccohort%n * ha_per_m2 hio_resp_g_canopy_si_scls(io_si,scls) = hio_resp_g_canopy_si_scls(io_si,scls) + & - resp_g * ccohort%n * dt_tstep_inv * ha_per_m2 + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day hio_resp_m_canopy_si_scls(io_si,scls) = hio_resp_m_canopy_si_scls(io_si,scls) + & ccohort%resp_m_tstep * ccohort%n * dt_tstep_inv * ha_per_m2 else @@ -5294,7 +5294,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) hio_froot_mr_understory_si_scls(io_si,scls) = hio_froot_mr_understory_si_scls(io_si,scls) + & ccohort%froot_mr * ccohort%n * ha_per_m2 hio_resp_g_understory_si_scls(io_si,scls) = hio_resp_g_understory_si_scls(io_si,scls) + & - resp_g * ccohort%n * dt_tstep_inv * ha_per_m2 + ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day hio_resp_m_understory_si_scls(io_si,scls) = hio_resp_m_understory_si_scls(io_si,scls) + & ccohort%resp_m_tstep * ccohort%n * dt_tstep_inv * ha_per_m2 endif From 1d520346dfda16a5a9ef61391829935f6c12cc73 Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 15:02:42 -0400 Subject: [PATCH 08/11] removing resp_g_acc, it is unnecessary --- biogeochem/EDCohortDynamicsMod.F90 | 2 -- biogeochem/FatesCohortMod.F90 | 6 ------ main/FatesRestartInterfaceMod.F90 | 10 ---------- 3 files changed, 18 deletions(-) diff --git a/biogeochem/EDCohortDynamicsMod.F90 b/biogeochem/EDCohortDynamicsMod.F90 index ce9691c2ac..e8597397ea 100644 --- a/biogeochem/EDCohortDynamicsMod.F90 +++ b/biogeochem/EDCohortDynamicsMod.F90 @@ -1071,8 +1071,6 @@ subroutine fuse_cohorts(currentSite, currentPatch, bc_in) currentCohort%resp_m_acc_hold = & (currentCohort%n*currentCohort%resp_m_acc_hold + & nextc%n*nextc%resp_m_acc_hold)/newn - currentCohort%resp_g_acc = (currentCohort%n*currentCohort%resp_g_acc + & - nextc%n*nextc%resp_g_acc)/newn currentCohort%resp_g_acc_hold = & (currentCohort%n*currentCohort%resp_g_acc_hold + & nextc%n*nextc%resp_g_acc_hold)/newn diff --git a/biogeochem/FatesCohortMod.F90 b/biogeochem/FatesCohortMod.F90 index 495b1818cf..0b359fe2c7 100644 --- a/biogeochem/FatesCohortMod.F90 +++ b/biogeochem/FatesCohortMod.F90 @@ -148,8 +148,6 @@ module FatesCohortMod real(r8) :: resp_m_tstep ! Maintenance respiration (see above *) real(r8) :: resp_m_acc real(r8) :: resp_m_acc_hold - - real(r8) :: resp_g_acc ! Growth respication can only be calculated at the daily timestep real(r8) :: resp_g_acc_hold real(r8) :: c13disc_clm ! carbon 13 discrimination in new synthesized carbon at each indiv/timestep [ppm] @@ -387,7 +385,6 @@ subroutine NanValues(this) this%resp_m_tstep = nan this%resp_m_acc = nan this%resp_m_acc_hold = nan - this%resp_g_acc = nan this%resp_g_acc_hold = nan this%c13disc_clm = nan this%c13disc_acc = nan @@ -486,7 +483,6 @@ subroutine ZeroValues(this) this%npp_acc = 0._r8 this%resp_m_tstep = 0._r8 this%resp_m_acc = 0._r8 - this%resp_g_acc = 0._r8 ! do not zero these, they are not built ! so more appropriate to leave unzerod @@ -716,7 +712,6 @@ subroutine Copy(this, copyCohort) copyCohort%resp_m_tstep = this%resp_m_tstep copyCohort%resp_m_acc = this%resp_m_acc copyCohort%resp_m_acc_hold = this%resp_m_acc_hold - copyCohort%resp_g_acc = this%resp_g_acc copyCohort%resp_g_acc_hold = this%resp_g_acc_hold copyCohort%c13disc_clm = this%c13disc_clm copyCohort%c13disc_acc = this%c13disc_acc @@ -1055,7 +1050,6 @@ subroutine Dump(this) write(fates_log(),*) 'cohort%resp_m_tstep = ', this%resp_m_tstep write(fates_log(),*) 'cohort%resp_m_acc = ', this%resp_m_acc write(fates_log(),*) 'cohort%resp_m_acc_hold = ', this%resp_m_acc_hold - write(fates_log(),*) 'cohort%resp_g_acc = ', this%resp_g_acc write(fates_log(),*) 'cohort%resp_g_acc_hold = ', this%resp_g_acc_hold write(fates_log(),*) 'cohort%rdark = ', this%rdark write(fates_log(),*) 'cohort%livestem_mr = ', this%livestem_mr diff --git a/main/FatesRestartInterfaceMod.F90 b/main/FatesRestartInterfaceMod.F90 index 3300d2297a..67f7be996e 100644 --- a/main/FatesRestartInterfaceMod.F90 +++ b/main/FatesRestartInterfaceMod.F90 @@ -133,7 +133,6 @@ module FatesRestartInterfaceMod integer :: ir_gpp_acc_co integer :: ir_npp_acc_co integer :: ir_resp_m_acc_co - integer :: ir_resp_g_acc_co integer :: ir_gpp_acc_hold_co integer :: ir_npp_acc_hold_co integer :: ir_resp_m_acc_hold_co @@ -860,11 +859,6 @@ subroutine define_restart_vars(this, initialize_variables) units='kgC/indiv', flushval = flushzero, & hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_m_acc_co ) - call this%set_restart_var(vname='fates_resp_g_acc', vtype=cohort_r8, & - long_name='ed cohort - accumulated growth respiration over dynamics step', & - units='kgC/indiv', flushval = flushzero, & - hlms='CLM:ALM', initialize=initialize_variables, ivar=ivar, index = ir_resp_g_acc_co ) - call this%set_restart_var(vname='fates_gpp_acc_hold', vtype=cohort_r8, & long_name='ed cohort - current step gpp', & units='kgC/indiv/year', flushval = flushzero, & @@ -2069,7 +2063,6 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_gpp_acc_co => this%rvars(ir_gpp_acc_co)%r81d, & rio_npp_acc_co => this%rvars(ir_npp_acc_co)%r81d, & rio_resp_m_acc_co => this%rvars(ir_resp_m_acc_co)%r81d, & - rio_resp_g_acc_co => this%rvars(ir_resp_g_acc_co)%r81d, & rio_gpp_acc_hold_co => this%rvars(ir_gpp_acc_hold_co)%r81d, & rio_resp_m_acc_hold_co => this%rvars(ir_resp_m_acc_hold_co)%r81d, & rio_resp_g_acc_hold_co => this%rvars(ir_resp_g_acc_hold_co)%r81d, & @@ -2393,7 +2386,6 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_gpp_acc_co(io_idx_co) = ccohort%gpp_acc rio_npp_acc_co(io_idx_co) = ccohort%npp_acc rio_resp_m_acc_co(io_idx_co) = ccohort%resp_m_acc - rio_resp_g_acc_co(io_idx_co) = ccohort%resp_g_acc rio_gpp_acc_hold_co(io_idx_co) = ccohort%gpp_acc_hold rio_resp_m_acc_hold_co(io_idx_co) = ccohort%resp_m_acc_hold rio_resp_g_acc_hold_co(io_idx_co) = ccohort%resp_g_acc_hold @@ -3040,7 +3032,6 @@ subroutine get_restart_vectors(this, nc, nsites, sites) rio_gpp_acc_co => this%rvars(ir_gpp_acc_co)%r81d, & rio_npp_acc_co => this%rvars(ir_npp_acc_co)%r81d, & rio_resp_m_acc_co => this%rvars(ir_resp_m_acc_co)%r81d, & - rio_resp_g_acc_co => this%rvars(ir_resp_g_acc_co)%r81d, & rio_gpp_acc_hold_co => this%rvars(ir_gpp_acc_hold_co)%r81d, & rio_resp_m_acc_hold_co => this%rvars(ir_resp_m_acc_hold_co)%r81d, & rio_resp_g_acc_hold_co => this%rvars(ir_resp_g_acc_hold_co)%r81d, & @@ -3338,7 +3329,6 @@ subroutine get_restart_vectors(this, nc, nsites, sites) ccohort%gpp_acc = rio_gpp_acc_co(io_idx_co) ccohort%npp_acc = rio_npp_acc_co(io_idx_co) ccohort%resp_m_acc = rio_resp_m_acc_co(io_idx_co) - ccohort%resp_g_acc = rio_resp_g_acc_co(io_idx_co) ccohort%gpp_acc_hold = rio_gpp_acc_hold_co(io_idx_co) ccohort%resp_m_acc_hold = rio_resp_m_acc_hold_co(io_idx_co) ccohort%resp_g_acc_hold = rio_resp_g_acc_hold_co(io_idx_co) From 3f688d70d1563e3688434a3c923393c088f408e3 Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 22:32:43 -0400 Subject: [PATCH 09/11] various bug fixes for growth respiration update --- biogeochem/FatesSoilBGCFluxMod.F90 | 3 --- main/EDMainMod.F90 | 18 ++++++++++-------- main/FatesHistoryInterfaceMod.F90 | 2 +- main/FatesRestartInterfaceMod.F90 | 4 ++-- 4 files changed, 13 insertions(+), 14 deletions(-) diff --git a/biogeochem/FatesSoilBGCFluxMod.F90 b/biogeochem/FatesSoilBGCFluxMod.F90 index efff982d8c..2dd3f816a4 100644 --- a/biogeochem/FatesSoilBGCFluxMod.F90 +++ b/biogeochem/FatesSoilBGCFluxMod.F90 @@ -123,9 +123,7 @@ subroutine UnPackNutrientAquisitionBCs(sites, bc_in) ! Locals integer :: nsites ! number of sites integer :: s ! site loop index - integer :: j ! soil layer integer :: icomp ! competitor index - integer :: id ! decomp layer index integer :: pft ! pft index type(fates_patch_type), pointer :: cpatch ! current patch pointer type(fates_cohort_type), pointer :: ccohort ! current cohort pointer @@ -647,7 +645,6 @@ subroutine FluxIntoLitterPools(csite, bc_in, bc_out) integer :: nlev_eff_decomp ! number of effective decomp layers real(r8) :: area_frac ! fraction of site's area of current patch real(r8) :: z_decomp ! Used for calculating depth midpoints of decomp layers - integer :: s ! Site index integer :: el ! Element index (C,N,P,etc) integer :: j ! Soil layer index integer :: id ! Decomposition layer index diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 983d830356..19569735e1 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -102,6 +102,7 @@ module EDMainMod use PRTGenericMod, only : repro_organ use PRTGenericMod, only : struct_organ use PRTLossFluxesMod, only : PRTMaintTurnover + use PRTParametersMod , only : prt_params use EDPftvarcon, only : EDPftvarcon_inst use FatesHistoryInterfaceMod, only : fates_hist @@ -232,7 +233,7 @@ subroutine ed_ecosystem_dynamics(currentSite, bc_in, bc_out) ! values. If we aren't entering that sequence, we need to set the flag ! Make sure cohorts are marked as non-recruits - call bypass_dynamics(currentSite) + call bypass_dynamics(currentSite,bc_out) end if @@ -340,7 +341,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) use FatesConstantsMod, only : itrue use FatesConstantsMod , only : nearzero use EDCanopyStructureMod , only : canopy_structure - use PRTParametersMod , only : prt_params + ! !ARGUMENTS: @@ -666,7 +667,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! Save NPP diagnostic for flux accounting [kg/m2/day] currentSite%flux_diags%npp = currentSite%flux_diags%npp + & - (currentCohort%npp_acc_hold/hlm_days_per_year - currentCohort%resp_excess) * currentCohort%n * area_inv + currentCohort%npp_acc_hold/hlm_days_per_year * currentCohort%n * area_inv ! And simultaneously add the input fluxes to mass balance accounting site_cmass%gpp_acc = site_cmass%gpp_acc + & @@ -699,7 +700,7 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) currentCohort%npp_acc = 0.0_r8 currentCohort%gpp_acc = 0.0_r8 - currentCohort%resp_acc = 0.0_r8 + currentCohort%resp_m_acc = 0.0_r8 ! BOC...update tree 'hydraulic geometry' ! (size --> heights of elements --> hydraulic path lengths --> @@ -1075,7 +1076,7 @@ end subroutine TotalBalanceCheck ! ===================================================================================== - subroutine bypass_dynamics(currentSite) + subroutine bypass_dynamics(currentSite, bc_out) ! ---------------------------------------------------------------------------------- ! If dynamics are bypassed, various fluxes, rates and flags need to be set @@ -1085,8 +1086,9 @@ subroutine bypass_dynamics(currentSite) ! ---------------------------------------------------------------------------------- ! Arguments - type(ed_site_type) , intent(inout), target :: currentSite - + type(ed_site_type) , intent(inout) :: currentSite + type(bc_out_type) , intent(inout) :: bc_out + ! Locals type(fates_patch_type), pointer :: currentPatch type(fates_cohort_type), pointer :: currentCohort @@ -1101,7 +1103,7 @@ subroutine bypass_dynamics(currentSite) currentCohort%isnew=.false. - currentCohort%resp_g_acc_hold = prt_params%grperc(ft) * & + currentCohort%resp_g_acc_hold = prt_params%grperc(currentCohort%pft) * & max(0._r8,(currentCohort%gpp_acc - currentCohort%resp_m_acc))*real(hlm_days_per_year,r8) currentCohort%npp_acc_hold = currentCohort%npp_acc * real(hlm_days_per_year,r8) diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index 9a1a85dbdf..b460b18cfc 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -2157,7 +2157,7 @@ subroutine update_history_nutrflux(this,csite) ! Excess carbon respired this%hvars(ih_excess_resp_si)%r81d(io_si) = & this%hvars(ih_excess_resp_si)%r81d(io_si) + & - ccohort%resp_excess*uconv + ccohort%resp_excess_hold*uconv/days_per_year case (nitrogen_element) diff --git a/main/FatesRestartInterfaceMod.F90 b/main/FatesRestartInterfaceMod.F90 index 6e53316f41..c4117acae4 100644 --- a/main/FatesRestartInterfaceMod.F90 +++ b/main/FatesRestartInterfaceMod.F90 @@ -2476,7 +2476,7 @@ subroutine set_restart_vectors(this,nc,nsites,sites) rio_resp_g_acc_hold_co(io_idx_co) = ccohort%resp_g_acc_hold rio_npp_acc_hold_co(io_idx_co) = ccohort%npp_acc_hold - rio_resp_excess_co(io_idx_co) = ccohort%resp_excess + rio_resp_excess_co(io_idx_co) = ccohort%resp_excess_hold rio_bmort_co(io_idx_co) = ccohort%bmort rio_hmort_co(io_idx_co) = ccohort%hmort @@ -3442,7 +3442,7 @@ subroutine get_restart_vectors(this, nc, nsites, sites) ccohort%resp_m_acc_hold = rio_resp_m_acc_hold_co(io_idx_co) ccohort%resp_g_acc_hold = rio_resp_g_acc_hold_co(io_idx_co) ccohort%npp_acc_hold = rio_npp_acc_hold_co(io_idx_co) - ccohort%resp_excess = rio_resp_excess_co(io_idx_co) + ccohort%resp_excess_hold = rio_resp_excess_co(io_idx_co) ccohort%bmort = rio_bmort_co(io_idx_co) ccohort%hmort = rio_hmort_co(io_idx_co) From a9758b3f4aa92d76e0ed983ea3d923b83dc344a3 Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 22:33:43 -0400 Subject: [PATCH 10/11] Added hard-coded override to revert to original format for inventory files --- main/FatesInventoryInitMod.F90 | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/main/FatesInventoryInitMod.F90 b/main/FatesInventoryInitMod.F90 index 22a48537b5..5e90d8254a 100644 --- a/main/FatesInventoryInitMod.F90 +++ b/main/FatesInventoryInitMod.F90 @@ -798,6 +798,7 @@ subroutine set_inventory_cohort_type1(csite,bc_in,css_file_unit,npatches, & class(prt_vartypes), pointer :: prt_obj real(r8) :: c_time ! Time patch was recorded character(len=patchname_strlen) :: p_name ! The patch associated with this cohort + character(len=patchname_strlen) :: c_name ! Cohort name real(r8) :: c_dbh ! diameter at breast height (cm) real(r8) :: c_height ! tree height (m) integer :: c_pft ! plant functional type index @@ -840,11 +841,17 @@ subroutine set_inventory_cohort_type1(csite,bc_in,css_file_unit,npatches, & real(r8), parameter :: abnormal_large_dbh = 500.0_r8 ! I've never heard of a tree > 3m real(r8), parameter :: abnormal_large_height = 500.0_r8 ! I've never heard of a tree > 500m tall integer, parameter :: recruitstatus = 0 + logical, parameter :: old_type1_override = .false. - - read(css_file_unit,fmt=*,iostat=ios) c_time, p_name, c_dbh, & - c_height, c_pft, c_nplant - + if(old_type1_override) then + ! time patch cohort dbh hite pft nplant bdead alive Avgrg + read(css_file_unit,fmt=*,iostat=ios) c_time, p_name, c_name, c_dbh, & + c_height, c_pft, c_nplant + else + read(css_file_unit,fmt=*,iostat=ios) c_time, p_name, c_dbh, & + c_height, c_pft, c_nplant + end if + if( debug_inv) then write(*,fmt=wr_fmt) & c_time, p_name, c_dbh, c_height, c_pft, c_nplant From 50bf6b93050fe3675f253418afdd6d275d5457b3 Mon Sep 17 00:00:00 2001 From: Ryan Knox Date: Wed, 30 Oct 2024 23:15:04 -0400 Subject: [PATCH 11/11] syntax updates and minor bug fix for respiration update --- biogeochem/FatesCohortMod.F90 | 9 ++++++--- main/EDMainMod.F90 | 4 +++- main/FatesHistoryInterfaceMod.F90 | 2 +- 3 files changed, 10 insertions(+), 5 deletions(-) diff --git a/biogeochem/FatesCohortMod.F90 b/biogeochem/FatesCohortMod.F90 index 28d95650c3..96669a1112 100644 --- a/biogeochem/FatesCohortMod.F90 +++ b/biogeochem/FatesCohortMod.F90 @@ -133,8 +133,11 @@ module FatesCohortMod ! after the dynamics call-sequence is completed. [kgC/indiv/day] ! _acc_hold: While _acc is zero'd after the dynamics call sequence and then integrated, ! _acc_hold "holds" the integrated value until the next time dynamics is - ! called. This is necessary for restarts. This variable also has units - ! converted to a useful rate [kgC/indiv/yr] + ! called. This is useful because growth and excess respiration + ! are calculated once daily, but we want to remove the average + ! flux from the daily NEP signal, so we remove it from the next day. + ! The hold variables are also useful for rebuilding history on restart. + ! Units converted to a useful rate [kgC/indiv/yr] ! -------------------------------------------------------------------------- real(r8) :: gpp_tstep ! Gross Primary Production (see above *) @@ -202,7 +205,7 @@ module FatesCohortMod integer :: twostr_col ! The column index in the two-stream solution that this cohort is part of ! RESPIRATION COMPONENTS - real(r8) :: resp_excess_hold ! respiration of excess carbon [kgC/indiv/day] + real(r8) :: resp_excess_hold ! respiration of excess carbon [kgC/indiv/yr] ! note: this is flagged "hold" because it is calculated ! at the end of the day (dynamics) but is used ! on the following day (like growth respiration) diff --git a/main/EDMainMod.F90 b/main/EDMainMod.F90 index 19569735e1..de1aa810a9 100644 --- a/main/EDMainMod.F90 +++ b/main/EDMainMod.F90 @@ -625,7 +625,9 @@ subroutine ed_integrate_state_variables(currentSite, bc_in, bc_out ) ! If nutrients are limiting growth, and carbon continues ! to accumulate beyond the plant's storage capacity, then ! it will burn carbon as what we call "excess respiration" - ! We must subtract this term from NPP + ! We must subtract this term from NPP. We do not need to subtract it from + ! currentCohort%npp_acc, it has already been removed from this in + ! the daily growth code (PARTEH). currentCohort%npp_acc_hold = currentCohort%npp_acc_hold - currentCohort%resp_excess_hold diff --git a/main/FatesHistoryInterfaceMod.F90 b/main/FatesHistoryInterfaceMod.F90 index b460b18cfc..87137b92a7 100644 --- a/main/FatesHistoryInterfaceMod.F90 +++ b/main/FatesHistoryInterfaceMod.F90 @@ -5306,7 +5306,7 @@ subroutine update_history_hifrq2(this,nc,nsites,sites,bc_in,bc_out,dt_tstep) ! Total AR (kgC/m2/s) = (kgC/plant/step) / (s/step) * (plant/m2) hio_ar_si_scpf(io_si,scpf) = hio_ar_si_scpf(io_si,scpf) + & (ccohort%resp_m_tstep*dt_tstep_inv) * n_perm2 + & - ccohort%resp_g_acc_hold * n_perm2 / days_per_year / sec_per_day + (ccohort%resp_g_acc_hold + ccohort%resp_excess_hold)* n_perm2 / days_per_year / sec_per_day ! Growth AR (kgC/m2/s) ! CDK: this should be daily hio_ar_grow_si_scpf(io_si,scpf) = hio_ar_grow_si_scpf(io_si,scpf) + &