source: trunk/LMDZ.COMMON/libf/evolution/ice_table_mod.F90 @ 3599

Last change on this file since 3599 was 3571, checked in by jbclement, 2 weeks ago

PEM:

  • New way to manage the pressure: now the PEM manages only the average pressure and keeps the pressure deviation with the instantaneous pressure from the start to reconstruct the pressure at the end ('ps_avg = ps_start + ps_dev'). As a consequence, everything related to pressure in the PEM is modified accordingly.
  • Surface temperatures management is now simpler. It follows the strategy for the pressure (and soil temperature) described above.
  • Soil temperatures are now adapted to match the surface temperature changes occured during the PEM by modifying the soil temperature deviation at the end.
  • Few simplifications/optimizations: notably, the two PCM years are now read in one go in 'read_data_PCM_mod.F90' and only the needed variables are extracted.
  • Deletion of unused variables and unnecessary intermediate variables (memory saving and loop deletion in some cases).
  • Renaming of variables and subroutines to make everything clearer. In particular, the suffixes: '_avg' = average, '_start' = PCM start file, '_dev' = deviation, '_ini' or '0' = initial, '_dyn' = dynamical grid, '_timeseries' = daily average of last PCM year.
  • Cosmetic cleanings for readability.

JBC

File size: 20.8 KB
RevLine 
[3320]1MODULE ice_table_mod
[2888]2
[3320]3implicit none
[2888]4
5!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
6!!!
[2961]7!!! Purpose: Ice table (pore-filling) variables and methods to compute it (dynamic and static)
[2888]8!!! Author:  LL, 02/2023
9!!!
10!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
11
[3498]12logical                             :: icetable_equilibrium ! Boolean to say if the PEM needs to recompute the icetable depth when at equilibrium
13logical                             :: icetable_dynamic     ! Boolean to say if the PEM needs to recompute the icetable depth with the dynamic method
14real, allocatable, dimension(:,:)   :: icetable_depth       ! ngrid x nslope: Depth of the ice table [m]
15real, allocatable, dimension(:,:)   :: icetable_thickness   ! ngrid x nslope: Thickness of the ice table [m]
16real, allocatable, dimension(:,:,:) :: ice_porefilling      ! the amout of porefilling in each layer in each grid [m^3/m^3]
[2888]17
[3321]18!-----------------------------------------------------------------------
[3327]19contains
[3525]20
[3321]21!-----------------------------------------------------------------------
[3493]22SUBROUTINE ini_ice_table(ngrid,nslope,nsoil)
[2888]23
[3320]24implicit none
[2961]25
[3320]26integer, intent(in) :: ngrid  ! number of atmospheric columns
27integer, intent(in) :: nslope ! number of slope within a mesh
[3493]28integer, intent(in) :: nsoil  ! number of soil layers
[2961]29
[3493]30allocate(icetable_depth(ngrid,nslope))
[3512]31allocate(icetable_thickness(ngrid,nslope))
32allocate(ice_porefilling(ngrid,nsoil,nslope))
[2961]33
[3493]34END SUBROUTINE ini_ice_table
[2961]35
[3321]36!-----------------------------------------------------------------------
[3493]37SUBROUTINE end_ice_table
[2961]38
[3320]39implicit none
[2961]40
[3493]41if (allocated(icetable_depth)) deallocate(icetable_depth)
42if (allocated(icetable_thickness)) deallocate(icetable_thickness)
43if (allocated(ice_porefilling)) deallocate(ice_porefilling)
[2961]44
[3493]45END SUBROUTINE end_ice_table
[2961]46
[3321]47!------------------------------------------------------------------------------------------------------
[3525]48SUBROUTINE computeice_table_equilibrium(ngrid,nslope,nsoil,watercaptag,rhowatersurf_ave,rhowatersoil_ave,regolith_inertia,ice_table_beg,ice_table_thickness)
[2888]49!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
50!!!
[2961]51!!! Purpose: Compute the ice table depth (pore-filling) knowing the yearly average water
[2888]52!!! density at the surface and at depth.
53!!! Computations are made following the methods in Schorgofer et al., 2005
[3525]54!!! This subroutine only gives the ice table at equilibrium and does not consider exchange with the atmosphere
[3320]55!!!
[2888]56!!! Author: LL
57!!!
58!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
[3320]59use math_mod,                   only: findroot
60use comsoil_h_PEM,              only: mlayer_PEM, layer_PEM ! Depth of the vertical grid
61use soil_thermalproperties_mod, only: ice_thermal_properties
62
63implicit none
64
[3327]65! Inputs
66! ------
[3525]67integer,                             intent(in) :: ngrid, nslope, nsoil ! Size of the physical grid, number of subslope, number of soil layer in the PEM
68logical, dimension(ngrid),           intent(in) :: watercaptag          ! Boolean to check the presence of a perennial glacier
69real, dimension(ngrid,nslope),       intent(in) :: rhowatersurf_ave     ! Water density at the surface, yearly averaged [kg/m^3]
70real, dimension(ngrid,nsoil,nslope), intent(in) :: rhowatersoil_ave     ! Water density at depth, computed from clapeyron law's (Murchy and Koop 2005), yearly averaged [kg/m^3]
71real, dimension(ngrid,nslope),       intent(in) :: regolith_inertia     ! Thermal inertia of the regolith layer [SI]
[3327]72! Ouputs
73! ------
[3320]74real, dimension(ngrid,nslope), intent(out) :: ice_table_beg       ! ice table depth [m]
75real, dimension(ngrid,nslope), intent(out) :: ice_table_thickness ! ice table thickness [m]
[3327]76! Locals
77! ------
[3321]78integer                       :: ig, islope, isoil, isoilend ! loop variables
[3525]79real, dimension(nsoil)        :: diff_rho                    ! difference of water vapor density between the surface and at depth [kg/m^3]
[3321]80real                          :: ice_table_end               ! depth of the end of the ice table  [m]
81real, dimension(ngrid,nslope) :: previous_icetable_depth     ! Ice table computed at previous ice depth [m]
82real                          :: stretch                     ! stretch factor to improve the convergence of the ice table
83real                          :: wice_inertia                ! Water Ice thermal Inertia [USI]
[3327]84! Code
85! ----
[3321]86previous_icetable_depth = ice_table_beg
[3320]87do ig = 1,ngrid
88    if (watercaptag(ig)) then
89        ice_table_beg(ig,:) = 0.
[3525]90        ice_table_thickness(ig,:) = layer_PEM(nsoil) ! Let's assume an infinite ice table (true when geothermal flux is set to 0.)
[3320]91    else
[2888]92        do islope = 1,nslope
[3320]93            ice_table_beg(ig,islope) = -1.
94            ice_table_thickness(ig,islope) = 0.
[3525]95            do isoil = 1,nsoil
[3320]96                diff_rho(isoil) = rhowatersurf_ave(ig,islope) - rhowatersoil_ave(ig,isoil,islope)
97            enddo
98            if (diff_rho(1) > 0) then ! ice is at the surface
99                ice_table_beg(ig,islope) = 0.
[3525]100                do isoilend = 2,nsoil - 1
[3321]101                    if (diff_rho(isoilend) > 0 .and. diff_rho(isoilend + 1) < 0.) then
[3320]102                        call findroot(diff_rho(isoilend),diff_rho(isoilend + 1),mlayer_PEM(isoilend),mlayer_PEM(isoilend + 1),ice_table_end)
103                        ice_table_thickness(ig,islope) = ice_table_end - ice_table_beg(ig,islope)
104                        exit
105                    endif
106                enddo
107            else
[3525]108                do isoil = 1,nsoil - 1 ! general case, we find the ice table depth by doing a linear approximation between the two depth, and then solve the first degree equation to find the root
[3320]109                    if (diff_rho(isoil) < 0 .and. diff_rho(isoil + 1) > 0.) then
110                        call findroot(diff_rho(isoil),diff_rho(isoil + 1),mlayer_PEM(isoil),mlayer_PEM(isoil + 1),ice_table_beg(ig,islope))
111                        ! Now let's find the end of the ice table
[3525]112                        ice_table_thickness(ig,islope) = layer_PEM(nsoil) ! Let's assume an infinite ice table (true when geothermal flux is set to 0.)
113                        do isoilend = isoil + 1,nsoil - 1
[3320]114                            if (diff_rho(isoilend) > 0 .and. diff_rho(isoilend + 1) < 0.) then
115                                call findroot(diff_rho(isoilend),diff_rho(isoilend + 1),mlayer_PEM(isoilend),mlayer_PEM(isoilend + 1),ice_table_end)
116                                ice_table_thickness(ig,islope) = ice_table_end - ice_table_beg(ig,islope)
117                                exit
118                            endif
119                        enddo
[3321]120                    endif ! diff_rho(z) <0 & diff_rho(z+1) > 0
[3320]121                enddo
122            endif ! diff_rho(1) > 0
[2888]123        enddo
[3320]124    endif ! watercaptag
125enddo
[2961]126
127! Small trick to speed up the convergence, Oded's idea.
[3320]128do islope = 1,nslope
129    do ig = 1,ngrid
130        if (ice_table_beg(ig,islope) > previous_icetable_depth(ig,islope) .and. previous_icetable_depth(ig,islope) >= 0) then
[2961]131            call ice_thermal_properties(.false.,1.,regolith_inertia(ig,islope),wice_inertia)
132            stretch = (regolith_inertia(ig,islope)/wice_inertia)**2
133            ice_table_thickness(ig,islope) = ice_table_thickness(ig,islope) + (ice_table_beg(ig,islope) - &
[3320]134                                             previous_icetable_depth(ig,islope) + (ice_table_beg(ig,islope) - previous_icetable_depth(ig,islope))/stretch)
135            ice_table_beg(ig,islope) = previous_icetable_depth(ig,islope) + (ice_table_beg(ig,islope) - previous_icetable_depth(ig,islope))/stretch
136        endif
137    enddo
138enddo
[2961]139
[3321]140END SUBROUTINE computeice_table_equilibrium
[2888]141
[3321]142!-----------------------------------------------------------------------
[3525]143SUBROUTINE compute_massh2o_exchange_ssi(ngrid,nslope,nsoil,icetable_thickness_old,ice_porefilling_old,tsurf,tsoil,delta_m_h2o)
[3031]144!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
145!!!
[3320]146!!! Purpose: Compute the mass of H2O that has sublimated from the ice table / condensed
147!!!
[3031]148!!! Author: LL
149!!!
150!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
[3320]151use comsoil_h_PEM,         only: mlayer_PEM
152use comslope_mod,          only: subslope_dist, def_slope_mean
153use constants_marspem_mod, only: porosity
[3082]154#ifndef CPP_STD
155    use comcstfi_h,   only: pi
156#else
157    use comcstfi_mod, only: pi
158#endif
159
[3320]160implicit none
[3031]161
[3327]162! Inputs
163! ------
[3525]164integer,                             intent(in) :: ngrid, nslope, nsoil   ! Size of the physical grid, number of subslope, number of soil layer in the PEM
165real, dimension(ngrid,nslope),       intent(in) :: icetable_thickness_old ! Ice table thickness at the previous iteration [m]
166real, dimension(ngrid,nsoil,nslope), intent(in) :: ice_porefilling_old    ! Ice pore filling at the previous iteration [m]
167real, dimension(ngrid,nslope),       intent(in) :: tsurf                  ! Surface temperature [K]
168real, dimension(ngrid,nsoil,nslope), intent(in) :: tsoil                  ! Soil temperature [K]
[3327]169! Outputs
170! -------
[3320]171real, dimension(ngrid), intent(out) :: delta_m_h2o ! Mass of H2O ice that has been condensed on the ice table / sublimates from the ice table [kg/m^2]
[3327]172! Locals
173!-------
[3525]174integer :: ig, islope, isoil ! loop index
175real    :: rho               ! density of water ice [kg/m^3]
176real    :: Tice              ! ice temperature [k]
[3327]177! Code
178! ----
[3031]179
[3525]180! Let's compute the amount of ice that has sublimated in each subslope
181if (icetable_equilibrium) then
182    delta_m_h2o = 0.
183    do ig = 1,ngrid
184        do islope = 1,nslope
185            call compute_Tice_pem(nsoil,tsoil(ig,:,islope),tsurf(ig,islope),icetable_depth(ig,islope),Tice)
[3527]186            delta_m_h2o(ig) = delta_m_h2o(ig) + porosity*rho_ice(Tice,'h2o')*(icetable_thickness(ig,islope) - icetable_thickness_old(ig,islope)) & ! convention > 0. <=> it condenses
[3525]187                              *subslope_dist(ig,islope)/cos(def_slope_mean(islope)*pi/180.)
188        enddo
[3320]189    enddo
[3525]190else if (icetable_dynamic) then
191    delta_m_h2o = 0.
192    do ig = 1,ngrid
193        do islope = 1,nslope
194            do isoil = 1,nsoil
195                call compute_Tice_pem(nsoil,tsoil(ig,:,islope),tsurf(ig,islope),mlayer_PEM(isoil - 1),Tice)
[3527]196                delta_m_h2o(ig) = delta_m_h2o(ig) + rho_ice(Tice,'h2o')*(ice_porefilling(ig,isoil,islope) - ice_porefilling_old(ig,isoil,islope)) & ! convention > 0. <=> it condenses
[3525]197                                  *subslope_dist(ig,islope)/cos(def_slope_mean(islope)*pi/180.)
198            enddo
199        enddo
200    enddo
201endif
[3031]202
[3321]203END SUBROUTINE compute_massh2o_exchange_ssi
[3031]204
[3321]205!-----------------------------------------------------------------------
[3320]206SUBROUTINE find_layering_icetable(porefill,psat_soil,psat_surf,pwat_surf,psat_bottom,B,index_IS,depth_filling,index_filling,index_geothermal,depth_geothermal,dz_etadz_rho)
[2902]207!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
208!!!
[3320]209!!! Purpose: Compute layering between dry soil, pore filling ice, and ice sheet based on Schorgofer, Icarus (2010). Adapted from NS MSIM
210!!!
[2902]211!!! Author: LL
212!!!
213!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
[3320]214use comsoil_h_PEM, only: nsoilmx_PEM, mlayer_PEM
215use math_mod,      only: deriv1, deriv1_onesided, colint, findroot, deriv2_simple
216
[2902]217implicit none
[3320]218
219! Inputs
[2902]220! ------
[3320]221real, dimension(nsoilmx_PEM), intent(in) :: porefill    ! Fraction of pore space filled with ice [Unitless] 0 <= f <= 1 for pore ice
222real, dimension(nsoilmx_PEM), intent(in) :: psat_soil   ! Soil water pressure at saturation, yearly averaged [Pa]
223real,                         intent(in) :: psat_surf   ! surface water pressure at saturation, yearly averaged [Pa]
224real,                         intent(in) :: pwat_surf   ! Water vapor pressure  at the surface, not necesseraly at saturation, yearly averaged [Pa]
225real,                         intent(in) :: psat_bottom ! Boundary conditions for soil vapor pressure [Pa]
226real,                         intent(in) :: B           ! constant (Eq. 8 from  Schorgofer, Icarus (2010).)
227integer,                      intent(in) :: index_IS    ! index of the soil layer where the ice sheet begins [1]
[2902]228real, intent(inout) :: depth_filling ! depth where pore filling begins [m]
[3320]229! Outputs
[2902]230! -------
[3320]231integer,                      intent(out) :: index_filling    ! index where the pore filling begins [1]
232integer,                      intent(out) :: index_geothermal ! index where the ice table stops [1]
233real,                         intent(out) :: depth_geothermal ! depth where the ice table stops [m]
234real, dimension(nsoilmx_PEM), intent(out) :: dz_etadz_rho     ! \partial z(eta \partial z rho), eta is the constriction, used later for pore filling increase
[3327]235! Locals
236!-------
[3320]237real, dimension(nsoilmx_PEM) :: eta                          ! constriction
238real, dimension(nsoilmx_PEM) :: dz_psat                      ! first derivative of the vapor pressure at saturation
239real, dimension(nsoilmx_PEM) :: dz_eta                       ! \partial z \eta
240real, dimension(nsoilmx_PEM) :: dzz_psat                     ! \partial \partial psat
241integer                      :: ilay, index_tmp              ! index for loop
242real                         :: old_depth_filling            ! depth_filling saved
243real                         :: Jdry                         ! flux trought the dry layer
244real                         :: Jsat                         ! flux trought the ice layer
245real                         :: Jdry_prevlay, Jsat_prevlay   ! same but for the previous ice layer
246integer                      :: index_firstice               ! first index where ice appears (i.e., f > 0)
247real                         :: massfillabove, massfillafter ! h2O mass above and after index_geothermal
[3327]248! Constant
249!---------
[3320]250real, parameter :: pvap2rho = 18.e-3/8.314
[3327]251! Code
252!-----
[2902]253! 0. Compute constriction over the layer
254! Within the ice sheet, constriction is set to 0. Elsewhere, constriction =  (1-porefilling)**2
[3321]255if (index_IS < 0) then
[3320]256    index_tmp = nsoilmx_PEM
257    do ilay = 1,nsoilmx_PEM
[3527]258        eta(ilay) = constriction(porefill(ilay))
[3320]259    enddo
[2902]260else
[3320]261    index_tmp = index_IS
[3321]262    do ilay = 1,index_IS - 1
[3527]263        eta(ilay) = constriction(porefill(ilay))
[3320]264    enddo
265    do ilay = index_IS,nsoilmx_PEM
266        eta(ilay) = 0.
267    enddo
[2902]268endif
269
[3320]270! 1. Depth at which pore filling occurs. We solve Eq. 9 from  Schorgofer, Icarus  (2010)
[2902]271old_depth_filling = depth_filling
272
[3320]273call deriv1(mlayer_PEM,nsoilmx_PEM,psat_soil,psat_surf,psat_bottom,dz_psat)
[2902]274
275do ilay = 1,index_tmp
[3320]276    Jdry = (psat_soil(ilay) - pwat_surf)/mlayer_PEM(ilay) ! left member of Eq. 9 from Schorgofer, Icarus  (2010)
277    Jsat =  eta(ilay)*dz_psat(ilay) !right member of Eq. 9 from Schorgofer, Icarus (2010)
278    if (Jdry - Jsat <= 0) then
279        index_filling = ilay
280        exit
281    endif
[2902]282enddo
283
[3320]284if (index_filling == 1) then
285    depth_filling = mlayer_PEM(1)
286else if (index_filling > 1) then
287    Jdry_prevlay = (psat_soil(index_filling - 1) - pwat_surf)/mlayer_PEM(index_filling - 1)
288    Jsat_prevlay = eta(index_filling - 1)*dz_psat(index_filling - 1)
289    call findroot(Jdry - Jsat,Jdry_prevlay - Jsat_prevlay,mlayer_PEM(index_filling),mlayer_PEM(index_filling - 1),depth_filling)
[2902]290endif
291
292! 2. Compute d_z (eta* d_z(rho)) (last term in Eq. 13 of Schorgofer, Icarus (2010))
[3525]293! 2.0 preliminary: depth to shallowest ice (discontinuity at interface)
[2902]294index_firstice = -1
295do ilay = 1,nsoilmx_PEM
[3320]296    if (porefill(ilay) <= 0.) then
[2902]297        index_firstice = ilay  ! first point with ice
298        exit
299    endif
300enddo
301
[3525]302! 2.1: now we can compute
[3320]303call deriv1(mlayer_PEM,nsoilmx_PEM,eta,1.,eta(nsoilmx_PEM - 1),dz_eta)
[2902]304
[3320]305if (index_firstice > 0 .and. index_firstice < nsoilmx_PEM - 2) call deriv1_onesided(index_firstice,mlayer_PEM,nsoilmx_PEM,eta,dz_eta(index_firstice))
[2902]306
307call deriv2_simple(mlayer_PEM,nsoilmx_PEM,psat_soil,psat_surf,psat_bottom,dzz_psat)
[3320]308dz_etadz_rho = pvap2rho*(dz_eta*dz_psat + eta*dzz_psat)
[2902]309
310! 3. Ice table boundary due to geothermal heating
[3320]311if (index_IS > 0) index_geothermal = -1
312if (index_geothermal < 0) depth_geothermal = -1.
313if ((index_geothermal > 0).and.(index_IS < 0)) then ! Eq. 21 from Schorfoger, Icarus (2010)
314    index_geothermal = -1
315    do ilay = 2,nsoilmx_PEM
316        if (dz_psat(ilay) > 0.) then  ! first point with reversed flux
317            index_geothermal = ilay
318            call findroot(dz_psat(ilay - 1),dz_psat(ilay),mlayer_PEM(ilay - 1),mlayer_PEM(ilay),depth_geothermal)
319            exit
[2902]320        endif
[3320]321    enddo
322else
323    index_geothermal = -1
324endif
325if (index_geothermal > 0 .and. index_IS < 0) then ! Eq. 24 from Schorgofer, Icarus (2010)
[3321]326    call  colint(porefill/eta,mlayer_PEM,nsoilmx_PEM,index_geothermal - 1,nsoilmx_PEM,massfillabove)
[3320]327    index_tmp = -1
328    do ilay = index_geothermal,nsoilmx_PEM
329        if (minval(eta(ilay:nsoilmx_PEM)).le.0.) cycle ! eta=0 means completely full
[3321]330        call colint(porefill/eta,mlayer_PEM,nsoilmx_PEM,ilay,nsoilmx_PEM,massfillafter)
[3320]331        if (massfillafter < dz_psat(ilay)*pvap2rho*B) then ! usually executes on i=typeG
332            if (ilay > index_geothermal) then
333!                write(34,*) '# adjustment to geotherm depth by',ilay-index_geothermal
334                call findroot(dz_psat(ilay - 1)*pvap2rho*B - massfillabove, &
335                              dz_psat(ilay)*pvap2rho*B - massfillafter,mlayer_PEM(ilay - 1),mlayer_PEM(ilay),depth_geothermal)
336!                if (depth_geothermal > mlayer_PEM(ilay) .or. depth_geothermal < mlayer_PEM(ilay - 1)) write(*,*) '# WARNING: zdepthG interpolation failed',ilay,mlayer_PEM(ilay - 1),depth_geothermal,mlayer_PEM(ilay)
337              index_tmp = ilay
[2902]338           endif
339           ! otherwise leave depth_geothermal unchanged
340           exit
341        endif
342        massfillabove = massfillafter
[3320]343    enddo
344    if (index_tmp > 0) index_geothermal = index_tmp
345end if
[2902]346
[3321]347END SUBROUTINE find_layering_icetable
[2902]348
[3321]349!-----------------------------------------------------------------------
[3527]350FUNCTION constriction(porefill) RESULT(eta)
[2902]351
[3320]352!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
353!!!
354!!! Purpose: Compute the constriction of vapor flux by pore ice
355!!!
356!!! Author: LL
357!!!
358!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
[2902]359
[3320]360implicit none
[2902]361
[3320]362real, intent(in) :: porefill ! pore filling fraction
[3527]363real :: eta ! constriction
[2902]364
[3320]365if (porefill <= 0.) then
366    eta = 1.
367else if (0 < porefill .and. porefill < 1.) then
368    eta = (1-porefill)**2 ! Hudson et al., JGR, 2009
369else
370    eta = 0.
371endif
372
[3527]373END FUNCTION constriction
[3320]374
[3327]375!-----------------------------------------------------------------------
[3525]376SUBROUTINE compute_Tice_pem(nsoil,ptsoil,ptsurf,ice_depth,Tice)
[3327]377!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
378!!!
379!!! Purpose: Compute subsurface ice temperature by interpolating the temperatures between the two adjacent cells.
380!!!
381!!! Author: LL
382!!!
383!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
384
[3571]385use comsoil_h_PEM, only: mlayer_PEM
[3527]386use abort_pem_mod, only: abort_pem
[3327]387
388implicit none
389
390! Inputs
391! ------
[3525]392integer,                intent(in) :: nsoil     ! Number of soil layers
393real, dimension(nsoil), intent(in) :: ptsoil    ! Soil temperature (K)
394real,                   intent(in) :: ptsurf    ! Surface temperature (K)
395real,                   intent(in) :: ice_depth ! Ice depth (m)
[3327]396
397! Outputs
[3525]398! -------
[3327]399real, intent(out) :: Tice ! Ice temperatures (K)
400
401! Locals
402! ------
[3532]403integer :: ik       ! Loop variables
[3327]404integer :: indexice ! Index of the ice
405
406! Code
407!-----
408indexice = -1
409if (ice_depth >= mlayer_PEM(nsoil - 1)) then
410    Tice = ptsoil(nsoil)
411else
412    if(ice_depth < mlayer_PEM(0)) then
413        indexice = 0.
[3532]414    else
[3327]415        do ik = 0,nsoil - 2 ! go through all the layers to find the ice locations
416            if (mlayer_PEM(ik) <= ice_depth .and. mlayer_PEM(ik + 1) > ice_depth) then
417                indexice = ik + 1
418                exit
419            endif
420        enddo
421    endif
422    if (indexice < 0) then
[3527]423        call abort_pem("compute_Tice_pem","Subsurface ice is below the last soil layer!",1)
[3327]424    else
425        if(indexice >= 1) then ! Linear inteprolation between soil temperature
426            Tice = (ptsoil(indexice) - ptsoil(indexice + 1))/(mlayer_PEM(indexice - 1) - mlayer_PEM(indexice))*(ice_depth - mlayer_PEM(indexice)) + ptsoil(indexice + 1)
427        else ! Linear inteprolation between the 1st soil temperature and the surface temperature
428            Tice = (ptsoil(1) - ptsurf)/mlayer_PEM(0)*(ice_depth - mlayer_PEM(0)) + ptsoil(1)
429        endif ! index ice >= 1
430    endif !indexice < 0
431endif ! icedepth > mlayer_PEM(nsoil - 1)
432
433END SUBROUTINE compute_Tice_pem
434
[3527]435!-----------------------------------------------------------------------
436FUNCTION rho_ice(T,ice) RESULT(rho)
437!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
438!!!
439!!! Purpose: Compute subsurface ice density
440!!!
441!!! Author: JBC
442!!!
443!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
444
445use abort_pem_mod, only: abort_pem
446
447implicit none
448
449real,         intent(in) :: T
450character(3), intent(in) :: ice
451real :: rho
452
453select case (trim(adjustl(ice)))
454    case('h2o')
455        rho = -3.5353e-4*T**2 + 0.0351*T + 933.5030 ! Rottgers 2012
456    case('co2')
457        rho = (1.72391 - 2.53e-4*T-2.87*1e-7*T**2)*1.e3 ! Mangan et al. 2017
458    case default
459        call abort_pem("rho_ice","Type of ice not known!",1)
460end select
461
462END FUNCTION rho_ice
463
[3321]464END MODULE ice_table_mod
Note: See TracBrowser for help on using the repository browser.