module phyetat0_mod implicit none real, save :: tab_cntrl_mod(100) integer,save :: nid_start ! NetCDF file identifier for startfi.nc file !$OMP THREADPRIVATE(tab_cntrl_mod) contains subroutine phyetat0 (startphy_file, & ngrid,nlayer,fichnom,tab0,Lmodif,nsoil,nq, & day_ini,time,tsurf,tsoil,emis,albedo, & q2,qsurf,cloudfrac,totcloudfrac,hice, & rnat,pctsrf_sic,tslab,tsea_ice,tice,sea_ice) ! to use 'getin_p' use ioipsl_getin_p_mod, only: getin_p !! use write_field_phy, only: Writefield_phy !! use tabfi_mod, only: tabfi USE tracer_h, ONLY: noms, igcm_h2o_vap USE radinc_h, ONLY: L_NSPECTV USE surfdat_h, only: phisfi, albedodat, zmea, zstd, zsig, zgam, zthe use iostart, only: nid_start, open_startphy, close_startphy, & get_field, get_var, inquire_field, & inquire_dimension, inquire_dimension_length ! use slab_ice_h, only: noceanmx USE ocean_slab_mod, ONLY: nslay use callkeys_mod, only: CLFvarying,surfalbedo,surfemis, callsoil use nonoro_gwd_ran_mod, only: du_nonoro_gwd, dv_nonoro_gwd, & east_gwstress, west_gwstress implicit none !====================================================================== ! Arguments: ! --------- ! inputs: logical,intent(in) :: startphy_file ! .true. if reading start file integer,intent(in) :: ngrid integer,intent(in) :: nlayer character*(*),intent(in) :: fichnom ! "startfi.nc" file integer,intent(in) :: tab0 integer,intent(in) :: Lmodif integer,intent(in) :: nsoil ! # of soil layers integer,intent(in) :: nq integer,intent(out) :: day_ini real,intent(out) :: time ! outputs: real,intent(out) :: tsurf(ngrid) ! surface temperature real,intent(out) :: tsoil(ngrid,nsoil) ! soil temperature real,intent(out) :: emis(ngrid) ! surface emissivity real,intent(out) :: albedo(ngrid,L_NSPECTV) ! albedo of the surface real,intent(out) :: q2(ngrid,nlayer+1) ! real,intent(out) :: qsurf(ngrid,nq) ! tracers on surface ! real co2ice(ngrid) ! co2 ice cover real,intent(out) :: cloudfrac(ngrid,nlayer) real,intent(out) :: hice(ngrid), totcloudfrac(ngrid) real,intent(out) :: pctsrf_sic(ngrid),tslab(ngrid,nslay) real,intent(out) :: tsea_ice(ngrid),sea_ice(ngrid) real,intent(out) :: rnat(ngrid) real,intent(out) :: tice(ngrid) !====================================================================== ! Local variables: ! INTEGER radpas ! REAL co2_ppm ! REAL solaire real xmin,xmax ! to display min and max of a field ! INTEGER ig,iq,lmax INTEGER nid, nvarid INTEGER ierr, i, nsrf ! integer isoil ! INTEGER length ! PARAMETER (length=100) CHARACTER*7 str7 CHARACTER*2 str2 CHARACTER*1 yes ! REAL p_rad,p_omeg,p_g,p_cpp,p_mugaz,p_daysec INTEGER nqold ! flag which identifies if 'startfi.nc' file is using old names (qsurf01,...) ! logical :: oldtracernames=.false. integer :: count character(len=30) :: txt ! to store some text INTEGER :: indextime=1 ! index of selected time, default value=1 logical :: found character(len=8) :: modname="phyetat0" ! ! ALLOCATE ARRAYS IN surfdat_h ! IF (.not. ALLOCATED(albedodat)) ALLOCATE(albedodat(ngrid)) IF (.not. ALLOCATED(phisfi)) ALLOCATE(phisfi(ngrid)) IF (.not. ALLOCATED(zmea)) ALLOCATE(zmea(ngrid)) IF (.not. ALLOCATED(zstd)) ALLOCATE(zstd(ngrid)) IF (.not. ALLOCATED(zsig)) ALLOCATE(zsig(ngrid)) IF (.not. ALLOCATED(zgam)) ALLOCATE(zgam(ngrid)) IF (.not. ALLOCATED(zthe)) ALLOCATE(zthe(ngrid)) if (startphy_file) then ! open physics initial state file: call open_startphy(fichnom,nid_start) ! possibility to modify tab_cntrl in tabfi write(*,*) write(*,*) 'TABFI in phyeta0: Lmodif=',Lmodif," tab0=",tab0 call tabfi (ngrid,nid_start,Lmodif,tab0,day_ini,lmax,p_rad, & p_omeg,p_g,p_cpp,p_mugaz,p_daysec,time) else ! "academic" initialization of planetary parameters via tabfi call tabfi (ngrid,0,0,0,day_ini,lmax,p_rad, & p_omeg,p_g,p_cpp,p_mugaz,p_daysec,time) endif ! of if (startphy_file) if (startphy_file) then ! Load surface geopotential: call get_field(nid_start,"phisfi",phisfi,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else phisfi(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: surface geopotential range:", & minval(phisfi), maxval(phisfi) if (startphy_file) then ! Load bare ground albedo: (will be stored in surfdat_h) call get_field(nid_start,"albedodat",albedodat,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else ! If no startfi file, use parameter surfalbedo in def file surfalbedo=0.5 call getin_p("surfalbedo",surfalbedo) print*,"surfalbedo",surfalbedo albedodat(:)=surfalbedo endif ! of if (startphy_file) write(*,*) "phyetat0: Bare ground albedo range:", & minval(albedodat), maxval(albedodat) if (startphy_file) then ! Load surface albedo (for now assume it is spectrally homogeneous) call get_field(nid_start,"albedo",albedo(:,1),found) if (.not.found) then write(*,*) modname//": Failed loading " write(*,*) " setting it to bare ground albedo" albedo(1:ngrid,1)=albedodat(1:ngrid) endif ! copy value to all spectral bands do i=2,L_NSPECTV albedo(1:ngrid,i)=albedo(1:ngrid,1) enddo else ! If no startfi file, use bare ground value do i=1,L_NSPECTV albedo(1:ngrid,i)=albedodat(1:ngrid) enddo endif ! of if (startphy_file) write(*,*) "phyetat0: Surface albedo range:", & minval(albedo), maxval(albedo) ! ZMEA if (startphy_file) then call get_field(nid_start,"ZMEA",zmea,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else zmea(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: range:", & minval(zmea), maxval(zmea) ! ZSTD if (startphy_file) then call get_field(nid_start,"ZSTD",zstd,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else zstd(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: range:", & minval(zstd), maxval(zstd) ! ZSIG if (startphy_file) then call get_field(nid_start,"ZSIG",zsig,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else zsig(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: range:", & minval(zsig), maxval(zsig) ! ZGAM if (startphy_file) then call get_field(nid_start,"ZGAM",zgam,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else zgam(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: range:", & minval(zgam), maxval(zgam) ! ZTHE if (startphy_file) then call get_field(nid_start,"ZTHE",zthe,found) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else zthe(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: range:", & minval(zthe), maxval(zthe) ! Surface temperature : if (startphy_file) then call get_field(nid_start,"tsurf",tsurf,found,indextime) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else tsurf(:)=0. ! will be updated afterwards in physiq ! endif ! of if (startphy_file) write(*,*) "phyetat0: Surface temperature range:", & minval(tsurf), maxval(tsurf) ! Surface emissivity if (startphy_file) then call get_field(nid_start,"emis",emis,found,indextime) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else ! If no startfi file, use parameter surfemis in def file surfemis=1.0 call getin_p("surfemis",surfemis) print*,"surfemis",surfemis emis(:)=surfemis endif ! of if (startphy_file) write(*,*) "phyetat0: Surface emissivity range:", & minval(emis), maxval(emis) ! Cloud fraction (added by BC 2010) if (CLFvarying) then if (startphy_file) then call get_field(nid_start,"cloudfrac",cloudfrac,found,indextime) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else cloudfrac(:,:)=0.0 endif ! of if (startphy_file) write(*,*) "phyetat0: Cloud fraction range:", & minval(cloudfrac), maxval(cloudfrac) else cloudfrac(:,:)=0.0 endif ! of if (CLFvarying) ! Total cloud fraction (added by BC 2010) if (CLFvarying) then if (startphy_file) then call get_field(nid_start,"totcloudfrac",totcloudfrac,found,indextime) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else totcloudfrac(:)=0.0 endif ! of if (startphy_file) write(*,*) "phyetat0: Total cloud fraction range:", & minval(totcloudfrac), maxval(totcloudfrac) else totcloudfrac(:)=0.0 endif ! of if (CLFvarying) ! Height of oceanic ice (added by BC 2010) if (startphy_file) then call get_field(nid_start,"hice",hice,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " ! call abort hice(:)=0. endif else hice(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: Height of oceanic ice range:", & minval(hice), maxval(hice) ! SLAB OCEAN (added by BC 2014) if (startphy_file) then ! nature of the surface call get_field(nid_start,"rnat",rnat,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " rnat(1:ngrid)=1. else do ig=1,ngrid if((nint(rnat(ig)).eq.2).or.(nint(rnat(ig)).eq.0))then rnat(ig)=0. else rnat(ig)=1. endif enddo endif ! of if (.not.found) else rnat(:)=1. endif ! of if (startphy_file) write(*,*) "phyetat0: Nature of surface range:", & minval(rnat), maxval(rnat) if (startphy_file) then ! Pourcentage of sea ice cover call get_field(nid_start,"pctsrf_sic",pctsrf_sic,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " pctsrf_sic(1:ngrid)=0. endif else pctsrf_sic(:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: Pourcentage of sea ice cover range:", & minval(pctsrf_sic), maxval(pctsrf_sic) if (startphy_file) then ! Slab ocean temperature (2 layers) call get_field(nid_start,"tslab",tslab,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " do iq=1,nslay tslab(1:ngrid,iq)=tsurf(1:ngrid) enddo endif else do iq=1,nslay tslab(1:ngrid,iq)=tsurf(1:ngrid) enddo endif ! of if (startphy_file) write(*,*) "phyetat0: Slab ocean temperature range:", & minval(tslab), maxval(tslab) if (startphy_file) then ! Oceanic ice temperature call get_field(nid_start,"tsea_ice",tsea_ice,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " tsea_ice(1:ngrid)=273.15-1.8 endif else tsea_ice(1:ngrid)=273.15-1.8 endif ! of if (startphy_file) write(*,*) "phyetat0: Oceanic ice/snow temperature range:", & minval(tsea_ice), maxval(tsea_ice) if (startphy_file) then ! Oceanic ice temperature call get_field(nid_start,"tice",tice,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " tice(1:ngrid)=273.15-1.8 endif else tice(1:ngrid)=273.15-1.8 endif ! of if (startphy_file) write(*,*) "phyetat0: Oceanic ice surface temperature range:", & minval(tice), maxval(tice) if (startphy_file) then ! Oceanic ice quantity (kg/m^2) call get_field(nid_start,"sea_ice",sea_ice,found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading " sea_ice(1:ngrid)=0. endif else sea_ice(1:ngrid)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: Oceanic ice quantity range:", & minval(sea_ice), maxval(sea_ice) ! pbl wind variance if (startphy_file) then call get_field(nid_start,"q2",q2,found,indextime) if (.not.found) then call abort_physic(modname,"Failed loading ",1) endif else q2(:,:)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: PBL wind variance range:", & minval(q2), maxval(q2) ! tracer on surface if (nq.ge.1) then do iq=1,nq txt=noms(iq) if (startphy_file) then call get_field(nid_start,txt,qsurf(:,iq),found,indextime) if (.not.found) then write(*,*) "phyetat0: Failed loading <",trim(txt),">" write(*,*) " ",trim(txt)," is set to zero" qsurf(:,iq) = 0. endif else qsurf(:,iq)=0. endif ! of if (startphy_file) write(*,*) "phyetat0: Surface tracer <",trim(txt),"> range:", & minval(qsurf(:,iq)), maxval(qsurf(:,iq)) enddo! of do iq=1,nq endif ! of if (nq.ge.1) !!call WriteField_phy("in_phyetat0_qsurf",qsurf(1:ngrid,igcm_h2o_vap),1) if (startphy_file) then ! Call to soil_settings, in order to read soil temperatures, ! as well as thermal inertia and volumetric heat capacity if (callsoil) then call soil_settings(nid_start,ngrid,nsoil,tsurf,tsoil,indextime) endif endif ! of if (startphy_file) ! Non-orographic gravity waves if (startphy_file) then call get_field(nid_start,"du_nonoro_gwd",du_nonoro_gwd,found,indextime) if (.not.found) then write(*,*) "phyetat0: not in file" du_nonoro_gwd(:,:)=0. else write(*,*) "phyetat0: Memory of zonal wind tendency due to non-orographic GW" write(*,*) " range:", & minval(du_nonoro_gwd), maxval(du_nonoro_gwd) endif endif ! of if (startphy_file) if (startphy_file) then call get_field(nid_start,"dv_nonoro_gwd",dv_nonoro_gwd,found,indextime) if (.not.found) then write(*,*) "phyetat0: not in file" dv_nonoro_gwd(:,:)=0. else write(*,*) "phyetat0: Memory of meridional wind tendency due to non-orographic GW" write(*,*) " range:", & minval(dv_nonoro_gwd), maxval(dv_nonoro_gwd) endif endif ! of if (startphy_file) if (startphy_file) then call get_field(nid_start,"east_gwstress",east_gwstress,found,indextime) if (.not.found) then write(*,*) "phyetat0: not in file" east_gwstress(:,:)=0. else write(*,*) "phyetat0: Memory of eastward stress profile due to non-orographic GW" write(*,*) " range:", & minval(east_gwstress), maxval(east_gwstress) endif endif ! of if (startphy_file) if (startphy_file) then call get_field(nid_start,"west_gwstress",west_gwstress,found,indextime) if (.not.found) then write(*,*) "phyetat0: not in file" west_gwstress(:,:)=0. else write(*,*) "phyetat0: Memory of westward stress profile due to non-orographic GW" write(*,*) " range:", & minval(west_gwstress), maxval(west_gwstress) endif endif ! of if (startphy_file) ! ! close file: ! if (startphy_file) call close_startphy(nid_start) end subroutine phyetat0 !====================================================================== subroutine ini_tab_controle_dyn_xios(idayref) use comcstfi_mod, only: g, mugaz, omeg, rad, rcp use time_phylmdz_mod, only: daysec, dtphys use mod_grid_phy_lmdz, only: nbp_lon, nbp_lat, nbp_lev implicit none integer*4, intent(in) :: idayref ! date (initial date for this run) integer :: length, l parameter (length = 100) real :: tab_cntrl(length) ! run parameters are stored in this array do l = 1,length tab_cntrl(l) = 0. enddo tab_cntrl(1) = real(nbp_lon) tab_cntrl(2) = real(nbp_lat-1) tab_cntrl(3) = real(nbp_lev) tab_cntrl(4) = real(idayref) tab_cntrl(5) = rad tab_cntrl(6) = omeg tab_cntrl(7) = g tab_cntrl(8) = mugaz tab_cntrl(9) = rcp tab_cntrl(10) = daysec tab_cntrl(11) = dtphys tab_cntrl_mod = tab_cntrl end subroutine ini_tab_controle_dyn_xios end module phyetat0_mod