From b8075ae417417d6c9d126000b40893143c470d82 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:01:51 +0100 Subject: [PATCH 01/12] Add 'otracers' which are necessary for transient tracer output. --- config/namelist.io | 1 + 1 file changed, 1 insertion(+) diff --git a/config/namelist.io b/config/namelist.io index 4a83228e9..abeef5b92 100644 --- a/config/namelist.io +++ b/config/namelist.io @@ -56,4 +56,5 @@ io_list = 'sst ',1, 'm', 4, 'bolus_w ',1, 'y', 4, 'fw ',1, 'm', 4, 'fh ',1, 'm', 4, + 'otracers ',1, 'y', 4, / From a671797482ff68b2da85afada65e3ffa8e6a08fe Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:03:58 +0100 Subject: [PATCH 02/12] Include logical variables to swich on/off specific transient tracers. --- config/namelist.transit | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/config/namelist.transit b/config/namelist.transit index 3ca5c87cf..5510ef1ad 100644 --- a/config/namelist.transit +++ b/config/namelist.transit @@ -1,11 +1,16 @@ ! The namelist file for transient tracers &transit_param +l_r14c = .false. +l_r39ar = .false. +l_f11 = .false. +l_f12 = .false. +l_sf6 = .false. anthro_transit=.false. paleo_transit=.false. length_transit=1 ! 166 for anthro_transit=.true. ti_start_transit=1 ! 1 for D14C, 80 for CFC-12 -ifile_transit='/work/ollie/mbutzin/fesom2/input/trace_gases/Table_CO2_isoC_CFC12_SF6.txt' +ifile_transit='/work/ab0246/a270108/fesom2_recom_config/input-for-awiesm/Table_CO2_isoC_CFCs1112_SF6.txt' r14c_a = 1.0000 ! atm. 14C/C ratio, global mean r39ar_a = 1.0000 ! atm. 39Ar/Ar ratio, global mean xarg_a = 9.34e-3 ! atm. Argon concn. (mole fraction), global mean From 092e333c33a30207eca3cf309ca48f6a622443a7 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:07:32 +0100 Subject: [PATCH 03/12] Bugfixes: include missing transit tracer code and control output. --- src/fesom_module.F90 | 46 +++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 45 insertions(+), 1 deletion(-) diff --git a/src/fesom_module.F90 b/src/fesom_module.F90 index dffef9186..8f0113eb2 100755 --- a/src/fesom_module.F90 +++ b/src/fesom_module.F90 @@ -53,11 +53,14 @@ module fesom_main_storage_module use recom_interface #endif +! Transient tracers +use mod_transit, only: year_ce, r14c_nh, r14c_tz, r14c_sh, r14c_ti, xCO2_ti, xf11_nh, xf11_sh, xf12_nh, xf12_sh, xsf6_nh, xsf6_sh, ti_transit, anthro_transit + implicit none type :: fesom_main_storage_type - integer :: n, from_nstep, offset, row, i, provided + integer :: n, from_nstep, offset, row, i, provided, id integer :: which_readr ! read which restart files (0=netcdf, 1=core dump,2=dtype) integer :: total_nsteps integer, pointer :: mype, npes, MPIerr, MPI_COMM_FESOM, MPI_COMM_WORLD, MPI_COMM_FESOM_IB @@ -186,6 +189,21 @@ subroutine fesom_init(fesom_total_nsteps) if (f%mype==0) write(*,*) 'FESOM mesh_setup... complete' +! Transient tracers: control output of initial input values + if(use_transit .and. anthro_transit .and. f%mype==0) then + write (*,*) + write (*,*) "*** Transient tracers: Initial atmospheric input values >>>" + write (*,*) "Year CE, xCO2, D14C_NH, D14C_TZ, D14C_SH, xCFC-11_NH, xCFC-11_SH, xCFC-12_NH, xCFC-12_SH, xSF6_NH, xSF6_SH" + write (*, fmt="(2x,i4,10(2x,f6.2))"), & + year_ce(ti_transit), xCO2_ti(ti_transit) * 1.e6, & + (r14c_nh(ti_transit) - 1.) * 1000., (r14c_tz(ti_transit) - 1.) * 1000., (r14c_sh(ti_transit) - 1.) * 1000., & + xf11_nh(ti_transit) * 1.e12, xf11_sh(ti_transit) * 1.e12, & + xf12_nh(ti_transit) * 1.e12, xf12_sh(ti_transit) * 1.e12, & + xsf6_nh(ti_transit) * 1.e12, xsf6_sh(ti_transit) * 1.e12 + write (*,*) + end if + + !===================== ! Allocate field variables ! and additional arrays needed for @@ -591,6 +609,14 @@ subroutine fesom_runloop(current_nsteps) !___model ocean step____________________________________________________ if (flag_debug .and. f%mype==0) print *, achar(27)//'[34m'//' --> call oce_timestep_ale'//achar(27)//'[0m' call oce_timestep_ale(n, f%ice, f%dynamics, f%tracers, f%partit, f%mesh) + if (use_transit) then + ! Prevent negative concentrations of SF6, CFC-11 and CFC-12 during the first years (inital values are zero) + do tr_num=1, f%tracers%num_tracers + if ((f%tracers%data(tr_num)%ID==6) .or. (f%tracers%data(tr_num)%ID==11) .or. (f%tracers%data(tr_num)%ID==12)) then + f%tracers%data(tr_num)%values(:,:) = max(f%tracers%data(tr_num)%values(:,:), 0._WP) + end if + end do + end if ! use_transit f%t3 = MPI_Wtime() !___compute energy diagnostics..._______________________________________ @@ -620,6 +646,24 @@ subroutine fesom_runloop(current_nsteps) #if defined (__recom) f%rtime_compute_recom = f%rtime_compute_recom + f%t1_recom - f%t0_recom #endif + +! Transient tracers: update of input values between restarts + if(use_transit .and. anthro_transit .and. (daynew == ndpyr) .and. (timenew==86400.)) then + ti_transit = ti_transit + 1 + if (f%mype==0) then + write (*,*) + write (*,*) "*** Transient tracers: Updated atmospheric input values >>>" + write (*,*) "Year CE, xCO2, D14C_NH, D14C_TZ, D14C_SH, xCFC-11_NH, xCFC-11_SH, xCFC-12_NH, xCFC-12_SH, xSF6_NH, xSF6_SH" + write (*, fmt="(2x,i4,10(2x,f6.2))"), & + year_ce(ti_transit), xCO2_ti(ti_transit) * 1.e6, & + (r14c_nh(ti_transit) - 1.) * 1000., (r14c_tz(ti_transit) - 1.) * 1000., (r14c_sh(ti_transit) - 1.) * 1000., & + xf11_nh(ti_transit) * 1.e12, xf11_sh(ti_transit) * 1.e12, & + xf12_nh(ti_transit) * 1.e12, xf12_sh(ti_transit) * 1.e12, & + xsf6_nh(ti_transit) * 1.e12, xsf6_sh(ti_transit) * 1.e12 + write (*,*) + end if + endif + end do !call cray_acc_set_debug_global_level(3) f%from_nstep = f%from_nstep+current_nsteps From 23819e5d8fa8546129b76cdfaa96585c69d98aad Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:09:21 +0100 Subject: [PATCH 04/12] Bugfix: amend incorrect transit tracer input. --- src/gen_model_setup.F90 | 26 ++++++++++++++++---------- 1 file changed, 16 insertions(+), 10 deletions(-) diff --git a/src/gen_model_setup.F90 b/src/gen_model_setup.F90 index 13a669a3a..4af143fea 100755 --- a/src/gen_model_setup.F90 +++ b/src/gen_model_setup.F90 @@ -125,18 +125,24 @@ subroutine setup_model(partit) #endif if (use_transit) then -! Transient tracer input, input file names have to be specified in -! namelist.config, nml=run_config - if(partit%mype==0) print *, "Transient tracers are ON. Tracer input file: ", ifile_transit - open (20,file=ifile_transit) - if (anthro_transit .or. paleo_transit) then - call read_transit_input + if(partit%mype==0) print *, "Transient tracers are ON." + nmlfile = 'namelist.transit' ! name of transient tracers namelist file + open (newunit=fileunit, file=nmlfile) + read (fileunit, nml=transit_param) + close (fileunit) + if (anthro_transit) then +! transient values of historical CO2, bomb radiocarbon, CFC-12, and SF6 + if(partit%mype==0) print *, "Reading transient input values from file: ", ifile_transit + open (newunit=fileunit, file=ifile_transit) + call read_transit_input(fileunit) + close (fileunit) + elseif (paleo_transit) then +! transient values of atmospheric CO2 and radiocarbon reconstructions +! under construction / not yet realized else -! Spinup / equilibrium runs with constant tracer input, -! read parameter values from namelist.oce - read (20,nml=transit_param) +! Spinup / equilibrium runs with constant tracer input specified in namelist.transit + if(partit%mype==0) print *, "Reading constant input values from file: ", nmlfile end if - close (20) end if if(partit%mype==0) write(*,*) 'Namelist files are read in' From 8884e7936e892f139a5de5efb2c5b1017e666483 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:12:35 +0100 Subject: [PATCH 05/12] This version comes from FESOM in AWIESM-2.6 but does not include any changes wrt transient tracers. --- src/gen_modules_clock.F90 | 41 ++++++++++----------------------------- 1 file changed, 10 insertions(+), 31 deletions(-) diff --git a/src/gen_modules_clock.F90 b/src/gen_modules_clock.F90 index 99f861e0c..0a287ab7e 100755 --- a/src/gen_modules_clock.F90 +++ b/src/gen_modules_clock.F90 @@ -2,8 +2,6 @@ module g_clock !combining RT and Lars version ! use g_config - use iso_fortran_env, only: error_unit - use mpi implicit none save real(kind=WP) :: timeold, timenew !time in a day, unit: sec @@ -71,15 +69,13 @@ end subroutine clock ! subroutine clock_init(partit) USE MOD_PARTIT + USE MOD_PARSUP use g_config use mod_transit, only: ti_transit, ti_start_transit implicit none type(t_partit), intent(in), target :: partit integer :: i, daystart, yearstart real(kind=WP) :: aux1, aux2, timestart - integer :: ierr - integer :: file_unit - character(512) :: errmsg ! the model initialized at timestart=timenew @@ -87,17 +83,10 @@ subroutine clock_init(partit) yearstart=yearnew ! init clock for this run - open(newunit=file_unit, file=trim(ResultPath)//trim(runid)//'.clock', action='read', & - status='old', iostat=ierr, iomsg=errmsg) - if (ierr /= 0) then - write (unit=error_unit, fmt='(3A)') & - '### error: can not open file ', trim(ResultPath)//trim(runid)//'.clock', & - ', error: ' // trim(errmsg) - call MPI_Abort(MPI_COMM_WORLD, 1, ierr) - end if - read(unit=file_unit, fmt=*) timeold, dayold, yearold - read(unit=file_unit, fmt=*) timenew, daynew, yearnew - close(unit=file_unit) + open(99,file=trim(ResultPath)//trim(runid)//'.clock',status='old') + read(99,*) timeold, dayold, yearold + read(99,*) timenew, daynew, yearnew + close(99) if(daynew==0) daynew=1 ! check if this is a restart or not @@ -168,9 +157,6 @@ subroutine clock_finish real(kind=WP) :: dum_timenew !time in a day, unit: sec integer :: dum_daynew !day in a year integer :: dum_yearnew !year before and after time step - integer :: ierr - integer :: file_unit - character(512) :: errmsg dum_timenew = timenew dum_daynew = daynew @@ -181,17 +167,10 @@ subroutine clock_finish dum_yearnew=yearold+1 endif - open(newunit=file_unit, file=trim(ResultPath)//trim(runid)//'.clock', action='write', & - status='unknown', iostat=ierr, iomsg=errmsg) - if (ierr /= 0) then - write (unit=error_unit, fmt='(3A)') & - '### error: can not open file ', trim(ResultPath)//trim(runid)//'.clock', & - ', error: ' // trim(errmsg) - call MPI_Abort(MPI_COMM_WORLD, 1, ierr) - end if - write(unit=file_unit, fmt=*) timeold, dayold, yearold - write(unit=file_unit, fmt=*) dum_timenew, dum_daynew, dum_yearnew - close(unit=file_unit) + open(99,file=trim(ResultPath)//trim(runid)//'.clock',status='unknown') + write(99,*) timeold, dayold, yearold + write(99,*) dum_timenew, dum_daynew, dum_yearnew + close(99) end subroutine clock_finish ! !---------------------------------------------------------------------------- @@ -232,4 +211,4 @@ end subroutine is_fleapyr ! !---------------------------------------------------------------------------- ! -end module g_clock \ No newline at end of file +end module g_clock From 78836ca98a0f9e362f8f0ce6262c0368ebd4c402 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:15:51 +0100 Subject: [PATCH 06/12] Change output file names of transient tracers from 'tra_ ...' to explicit tracer names, e.g., 'cfc11'. --- src/io_meandata.F90 | 53 +++++++++++++++++++++++---------------------- src/io_restart.F90 | 9 ++++++-- 2 files changed, 34 insertions(+), 28 deletions(-) diff --git a/src/io_meandata.F90 b/src/io_meandata.F90 index c5f958320..fe42eb7db 100644 --- a/src/io_meandata.F90 +++ b/src/io_meandata.F90 @@ -152,7 +152,7 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) #endif use g_forcing_param, only: use_virt_salt, use_landice_water, use_age_tracer !---fwf-code, age-code use g_config, only : lwiso !---wiso-code - use mod_transit, only : index_transit + use mod_transit, only : index_transit_r14c, index_transit_r39ar, index_transit_f11, index_transit_f12, index_transit_sf6 implicit none integer :: i, j @@ -241,7 +241,7 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) end if CASE ('m_ice ') if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'm_ice', 'ice height per unit area', 'm', ice%data(2)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'm_ice', 'ice height', 'm', ice%data(2)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) end if CASE ('thdgr ') if (use_ice) then @@ -257,17 +257,9 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) end if CASE ('m_snow ') if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'm_snow', 'snow height per unit area', 'm', ice%data(3)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'm_snow', 'snow height', 'm', ice%data(3)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) end if -CASE ('h_ice ') - if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'h_ice', 'ice thickness over ice-covered fraction', 'm', ice%h_ice(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) - end if -CASE ('h_snow ') - if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'h_snow', 'snow thickness over ice-covered fraction', 'm', ice%h_snow(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) - end if - + ! Debug ice variables CASE ('strength_ice') if (use_ice) then @@ -385,7 +377,7 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) !_______________________________________________________________________________ ! output surface forcing CASE ('fh ') - call def_stream(nod2D, myDim_nod2D, 'fh', 'heat flux', 'W/m2', heat_flux_in(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'fh', 'heat flux', 'W', heat_flux_in(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) CASE ('fw ') call def_stream(nod2D, myDim_nod2D, 'fw', 'fresh water flux', 'm/s', water_flux(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) CASE ('atmice_x ') @@ -873,13 +865,33 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) endif else #endif - call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'tra_'//id_string, 'passive tracer ID='//id_string, 'n/a', tracers%data(j)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + + if (use_transit) then +! Transient tracers + select case (tracers%data(j)%ID) + case(6) + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'sf6', 'sulfur hexafluoride', 'mol / m**3', tracers%data(index_transit_sf6)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + case(11) + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'cfc11', 'chlorofluorocarbon CFC-11', 'mol / m**3', tracers%data(index_transit_f11)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + case(12) + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'cfc12', 'chlorofluorocarbon CFC-12', 'mol / m**3', tracers%data(index_transit_f12)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + case(14) + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'r14c', '14C / C ratio of DIC', 'none', tracers%data(index_transit_r14c)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + case(39) + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'r39ar', '39Ar / Ar ratio', 'none', tracers%data(index_transit_r39ar)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + case default + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'tra_'//id_string, 'passive tracer ID='//id_string, 'n/a', tracers%data(j)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + end select + else +! Other passive but not transient tracers + call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'tra_'//id_string, 'passive tracer ID='//id_string, 'n/a', tracers%data(j)%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + end if #if defined(__recom) end if #endif - end do + end do CASE ('sigma0 ') call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'sigma0', 'potential density', 'kg/m3', density_m_rho0(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) @@ -915,17 +927,6 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) CASE ('slope_z ') call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'slope_z', 'neutral slope Z', 'none', neutral_slope(3,:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) -! Transient tracers -CASE('SF6 ') - if (use_transit) call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'sf6', 'sulfur hexafluoride', 'mol / m**3', tracers%data(index_transit(1))%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) -CASE('CFC-12 ') - if (use_transit) call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'cfc12', 'chlorofluorocarbon CFC-12', 'mol / m**3', tracers%data(index_transit(2))%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) -CASE('R14C ') - if (use_transit) call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'r14c', '14C / C ratio of DIC', 'none', tracers%data(index_transit(3))%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) -CASE('R39Ar ') - if (use_transit) call def_stream((/nl-1, nod2D/), (/nl-1, myDim_nod2D/), 'r39ar', '39Ar / Ar ratio', 'none', tracers%data(index_transit(4))%values(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) -! Transient tracers end - CASE ('N2 ') call def_stream((/nl, nod2D/), (/nl, myDim_nod2D/), 'N2', 'brunt väisälä', '1/s2', bvfreq(:,:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) CASE ('Kv ') diff --git a/src/io_restart.F90 b/src/io_restart.F90 index a51be4bd6..c23b55843 100644 --- a/src/io_restart.F90 +++ b/src/io_restart.F90 @@ -62,7 +62,7 @@ subroutine ini_ocean_io(year, dynamics, tracers, partit, mesh) use nvfortran_subarray_workaround_module #endif integer, intent(in) :: year - integer :: j + integer :: j, id character(500) :: longname character(500) :: trname, units character(4) :: cyear @@ -118,7 +118,8 @@ subroutine ini_ocean_io(year, dynamics, tracers, partit, mesh) endif do j=1,tracers%num_tracers - SELECT CASE (j) + id=tracers%data(j)%ID !MB: Avoid hard-wired tracer assignments like SELECT CASE(j) + SELECT CASE (id) CASE(1) trname='temp' longname='potential temperature' @@ -131,6 +132,10 @@ subroutine ini_ocean_io(year, dynamics, tracers, partit, mesh) trname='sf6' longname='sulfur hexafluoride' units='mol / m**3' + CASE(11) + trname='cfc11' + longname='chlorofluorocarbon CFC-11' + units='mol / m**3' CASE(12) trname='cfc12' longname='chlorofluorocarbon CFC-12' From 76207032d9bff865c10e3fb49b1240fee448124b Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:18:56 +0100 Subject: [PATCH 07/12] Include logical variables to swich on/off specific transient tracers. Include CFC-11. Change from hard-wired I/O unit numbers with variable I/O unit values. --- src/mod_transit.F90 | 43 ++++++++++++++++++++++++++++++++++--------- 1 file changed, 34 insertions(+), 9 deletions(-) diff --git a/src/mod_transit.F90 b/src/mod_transit.F90 index 6713032eb..7fbc8e018 100644 --- a/src/mod_transit.F90 +++ b/src/mod_transit.F90 @@ -16,6 +16,7 @@ MODULE mod_transit r39ar_a = 1.0, & ! 39Ar / 40 Ar (homogeneous) xarg_a = 9.34e-3, & ! Argon (homogeneous) xCO2_a = 284.32e-6, & ! CO2 (CMIP6 & OMIP-BGC: 284.32e-6 for 1700-1850, PMIP4: 190.00e-6 for 21 ka BP) + xf11_a = 0.0, & ! CFC-11 (latitude dependent) xf12_a = 0.0, & ! CFC-12 (latitude dependent) xsf6_a = 0.0 ! SF6 (latitude dependent) @@ -23,20 +24,27 @@ MODULE mod_transit real(kind=8), allocatable, dimension(:) :: r14c_nh, r14c_tz, r14c_sh, & ! 14CO2 / 12CO2, latitude-dependent (e.g., bomb 14C) r14c_ti, & ! 14CO2 / 12CO2, homogenous (e.g., IntCal) xCO2_ti, & ! CO2 + xf11_nh, xf11_sh, & ! CFC-11, latitude-dependent xf12_nh, xf12_sh, & ! CFC-12, latitude-dependent xsf6_nh, xsf6_sh ! SF6, latitude-dependent integer, allocatable, dimension(:) :: year_ce ! current year in anthropenic runs (control output) integer :: length_transit = 1, & ! length (years) of transient tracer input ti_start_transit = 1 ! index of the first tracer input year in ifile_transit - logical :: anthro_transit = .false., & + logical :: l_r14c = .false., & ! switch on / off specific tracerss + l_r39ar = .false. , & + l_f11 = .false., & + l_f12 = .false., & + l_sf6 = .false., & + anthro_transit = .false., & paleo_transit = .false. ! specify tracer input scenario - character(300) :: ifile_transit ='Table_CO2_isoC_CFC12_SF6.txt'! tracer input file; not neccessary for steady state simulations + character(300) :: ifile_transit ='Table_CO2_isoC_CFCs_SF6.txt'! tracer input file; not neccessary for steady state simulations ! Parameters which can be changed via namelist.oce (-> transit_param) ! Global-mean concentrations of DIC and Argon in the mixed layer (mol / m**3) real(kind=8) :: dic_0 = 2.00, & ! GLODAPv2, 0-50 m: TCO2 ~ 2050 umol / kg arg_0 = 0.01 ! Hamme et al. 2019, doi:10.1146/annurev-marine-121916-063604 + ! Radioactive decay constants (1 / s; default values assume that 1 year = 365.00 days) real(kind=8) :: decay14 = 3.8561e-12 , & ! 14C; t1/2 = 5700 a following OMIP-BGC decay39 = 8.1708e-11 ! 39Ar; t1/2 = 269 a @@ -45,16 +53,20 @@ MODULE mod_transit ! Latitude of atmospheric boundary conditions and latitudinal interpolation weight real(kind=8) :: y_abc, yy_nh ! Tracer indices of transient tracers - integer :: id_r14c, id_r39ar, id_f12, id_sf6 - integer, dimension(4) :: index_transit = (/-1, -1, -1, -1/) + integer :: id_r14c, id_r39ar, id_f11, id_f12, id_sf6, index_transit_r14c, index_transit_r39ar, index_transit_f11, index_transit_f12, index_transit_sf6 ! Time index (=year) in transient simulations integer :: ti_transit ! Namelist to modify default parameter settings namelist / transit_param / & - anthro_transit, & ! anthoropogenic transient tracers + l_r14c, & ! switch on R14C + l_r39ar, & ! switch on R39Ar + l_f11, & ! switch on CFC-11 + l_f12, & ! switch on CFC-12 + l_sf6, & ! switch on SF6 + anthro_transit, & ! anthropogenic transient tracers paleo_transit, & ! paleo transient tracers - length_transit, & ! 166 for anthro_transit=.true. + length_transit, & ! 166 if (anthro_transit=.true.) ti_start_transit, & ! 1 for D14C, 80 for CFC-12 ifile_transit, & ! forcing file r14c_a, & ! atm. 14C/C ratio, global mean @@ -135,6 +147,11 @@ function solub(which_gas, temp_c, sal) a1 = -160.7333; a2 = 215.4152; a3 = 89.8920; a4 = -1.47759; pow = 2 b1 = 0.029941; b2 = -0.027455; b3 = 0.0053407; c1 = 0. con2con = 1000. ! convert to mol / (m**3 * atm) + case ("f11") +! CFC-11 in mol / (L * atm) (Warner & Weiss 1985, doi:10.1016/0198-0149(85)90099-8, Table 5) + a1 = -229.9261; a2 = 319.6552; a3 = 119.4471; a4 = -1.39165; pow = 2 + b1 = -0.142382; b2 = 0.091459; b3 = -0.0157274; c1 = 0. + con2con = 1000. ! convert to mol / (m**3 * atm) case ("f12") ! CFC-12 in mol / (L * atm) (Warner & Weiss 1985, doi:10.1016/0198-0149(85)90099-8, Table 5) a1 = -218.0971; a2 = 298.9702; a3 = 113.8049; a4 = -1.39165; pow = 2 @@ -175,6 +192,8 @@ function sc_660(which_gas, temp_c) select case (which_gas) case ("co2") ! CO2 as = 2116.8; bs = -136.25; cs = 4.7353; ds = -0.092307; es = 0.0007555 + case ("f11") ! CFC-11 + as = 3579.2; bs = -222.63; cs = 7.5749; ds = -0.145950; es = 0.0011870 case ("f12") ! CFC-12 as = 3828.1; bs = -249.86; cs = 8.7603; ds = -0.171600; es = 0.0014080 case ("sf6") ! SF6 @@ -246,11 +265,12 @@ function speed_2(windstr_x, windstr_y) end function speed_2 - subroutine read_transit_input + subroutine read_transit_input(ifileunit) ! Read atmospheric input of isoCO2 and / or other tracers implicit none ! Internal variables + integer, intent(in) :: ifileunit integer :: jj real(kind=8), allocatable, dimension(:) :: d14c_nh, d14c_tz, d14c_sh, d14c_ti, d13c_dummy @@ -263,6 +283,8 @@ subroutine read_transit_input allocate(r14c_tz(length_transit)) allocate(r14c_sh(length_transit)) allocate(xCO2_ti(length_transit)) + allocate(xf11_nh(length_transit)) + allocate(xf11_sh(length_transit)) allocate(xf12_nh(length_transit)) allocate(xf12_sh(length_transit)) allocate(xsf6_nh(length_transit)) @@ -272,14 +294,15 @@ subroutine read_transit_input ! Skip header lines do jj = 1,15 - read (20, fmt=*) + read (ifileunit, fmt=*) end do ! Read input values do jj = 1, length_transit - read (20, fmt=*) year_ce(jj), & + read (ifileunit, fmt=*) year_ce(jj), & xCO2_ti(jj), & d14c_nh(jj), d14c_tz(jj), d14c_sh(jj), & d13c_dummy(jj), & + xf11_nh(jj), xf11_sh(jj), & xf12_nh(jj), xf12_sh(jj), & xsf6_nh(jj), xsf6_sh(jj) end do @@ -290,6 +313,8 @@ subroutine read_transit_input r14c_sh = 1. + 0.001 * d14c_sh ! Convert volume mixing ratios xCO2_ti = xCO2_ti * 1.e-6 + xf11_nh = xf11_nh * 1.e-12 + xf11_sh = xf11_sh * 1.e-12 xf12_nh = xf12_nh * 1.e-12 xf12_sh = xf12_sh * 1.e-12 xsf6_nh = xsf6_nh * 1.e-12 From 6db046f287eb83cfe22445dfb7b36bcc8051dfb6 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:23:02 +0100 Subject: [PATCH 08/12] Boundary conditions of transient tracers are now handled by function 'bc_surface', remove the now obsolete function 'transit_bc_surface' and its interface module. --- src/oce_ale_tracer.F90 | 279 +++++++++++++++++------------------------ 1 file changed, 112 insertions(+), 167 deletions(-) diff --git a/src/oce_ale_tracer.F90 b/src/oce_ale_tracer.F90 index bb40ff335..83e28d74a 100644 --- a/src/oce_ale_tracer.F90 +++ b/src/oce_ale_tracer.F90 @@ -81,29 +81,15 @@ subroutine diff_tracers_ale(tr_num, dynamics, tracer, ice, partit, mesh) module bc_surface_interface interface - function bc_surface(n, id, sval, nzmin, partit) - use mod_mesh - USE MOD_PARTIT - USE MOD_PARSUP - integer , intent(in) :: n, id, nzmin - type(t_partit), intent(inout), target :: partit - real(kind=WP) :: bc_surface - real(kind=WP), intent(in) :: sval - end function - end interface -end module - -module transit_bc_surface_interface - interface - function transit_bc_surface(n, id, sst, sss, a_ice, sval, nzmin, partit, mesh) + function bc_surface(n, id, sval, nzmin, partit, mesh, sst, sss, a_ice) use mod_mesh USE MOD_PARTIT USE MOD_PARSUP integer , intent(in) :: n, id, nzmin type(t_partit), intent(inout), target :: partit type(t_mesh), intent(in), target :: mesh - real(kind=WP) :: transit_bc_surface - real(kind=WP), intent(in) :: sst, sss, a_ice, sval + real(kind=WP) :: bc_surface + real(kind=WP), intent(in) :: sval, sst, sss, a_ice end function end interface end module @@ -992,11 +978,10 @@ subroutine diff_ver_part_impl_ale(tr_num, dynamics, tracers, ice, partit, mesh) !_______________________________________________________________________ ! case of activated shortwave penetration into the ocean, ad 3d contribution if (use_icebergs .and. (.not. turn_off_hf) .and. tracers%data(tr_num)%ID==1) then - do nz=nzmin, nzmax - + do nz=nzmin, nzmax-1 zinv=1.0_WP*dt !/(zbar(nz)-zbar(nz+1)) ale! !!PS tr(nz)=tr(nz)+(sw_3d(nz, n)-sw_3d(nz+1, n) * ( area(nz+1,n)/areasvol(nz,n)) ) * zinv - tr(nz)=tr(nz) + ibhf_n(nz, n) * zinv / vcpw + tr(nz)=tr(nz)+(ibhf_n(nz, n)-ibhf_n(nz+1, n) * area(nz+1,n)/areasvol(nz,n)) * zinv / vcpw end do end if @@ -1012,12 +997,8 @@ subroutine diff_ver_part_impl_ale(tr_num, dynamics, tracers, ice, partit, mesh) ! (BUT CHECK!) | | | | ! v (+) v (+) ! - tr(nzmin)= tr(nzmin)+bc_surface(n, tracers%data(tr_num)%ID, trarr(nzmin,n), nzmin, partit) + tr(nzmin)= tr(nzmin)+bc_surface(n, tracers%data(tr_num)%ID, trarr(nzmin,n), nzmin, partit, mesh, sst(nzmin,n), sss(nzmin,n), a_ice(n)) - if ((tracers%data(tr_num)%ID .ge. 6) .and.(tracers%data(tr_num)%ID .le. 40)) then - tr(nzmin)= tr(nzmin)+transit_bc_surface(n, tracers%data(tr_num)%ID, sst(nzmin,n), sss(nzmin,n), a_ice(n), trarr(nzmin,n), nzmin, partit, mesh) - end if - !_______________________________________________________________________ ! The forward sweep algorithm to solve the three-diagonal matrix ! problem @@ -1460,7 +1441,8 @@ end subroutine diff_part_bh !=============================================================================== ! this function returns a boundary conditions for a specified tracer ID and surface node ! ID = 0 and 1 are reserved for temperature and salinity -FUNCTION bc_surface(n, id, sval, nzmin, partit) +! MB: mesh, sst, sss, and aice are only needed for transient tracers +FUNCTION bc_surface(n, id, sval, nzmin, partit, mesh, sst, sss, aice) use MOD_MESH USE MOD_PARTIT USE MOD_PARSUP @@ -1472,14 +1454,32 @@ FUNCTION bc_surface(n, id, sval, nzmin, partit) use recom_glovar #endif use mod_transit + use g_clock implicit none integer, intent(in) :: n, id, nzmin - real(kind=WP), intent(in) :: sval + real(kind=WP), intent(in) :: sval, sst, sss, aice type(t_partit),intent(inout), target :: partit + type(t_mesh), intent(in), target :: mesh + REAL(kind=WP) :: bc_surface character(len=10) :: id_string - real(kind=WP), dimension(:), pointer :: a_ice +! real(kind=WP), dimension(:), pointer :: a_ice !! MB: where is this needed? + + if (use_transit) then +#if defined (__oasis) + ! SLP and wind speed in coupled setups. This is a makeshift solution + ! as long as the true values are not provided by the AGCM / OASIS. + press_a = mean_slp + wind_2 = speed_2(stress_atmoce_x(n), stress_atmoce_y(n)) +#else + press_a = press_air(n) + wind_2 = u_wind(n)**2 + v_wind(n)**2 +#endif + ! Atmospheric input of bomb-14C, CFC-12, and SF6 varies with latitude. To that effect specify + y_abc = mesh%geo_coord_nod2D(2,n) / rad ! latitude of atmospheric tracer input + yy_nh = (10. - y_abc) * 0.05 ! interpolation weight for tropical CFC-12 and SF6 values + end if ! use_transit SELECT CASE (id) CASE (1) @@ -1489,6 +1489,90 @@ FUNCTION bc_surface(n, id, sval, nzmin, partit) ! by forming/melting of sea ice bc_surface= dt*(virtual_salt(n) & !--> is zeros for zlevel/zstar + relax_salt(n) - real_salt_flux(n)*is_nonlinfs) +!---Transient tracers (cases ##6,12,14,39) + CASE (6) ! SF6 + if (anthro_transit) then + ! Select atmospheric input values corresponding to the latitude + ! Annual values are interpolated to monthly values, this is omitted in the last simulation year + if (y_abc > 10.) then ! Northern hemisphere + xsf6_a = xsf6_nh(ti_transit) + if (ti_transit < length_transit) xsf6_a = xsf6_a + month * (xsf6_nh(ti_transit + 1) - xsf6_a) / 12. + else if (y_abc <- 10.) then ! Southern hemisphere + xsf6_a = xsf6_sh(ti_transit) + if (ti_transit < length_transit) xsf6_a = xsf6_a + month * (xsf6_sh(ti_transit + 1) - xsf6_a) / 12. + else ! Tropical zone, interpolate between NH and SH + xsf6_a = (1 - yy_nh) * xsf6_nh(ti_transit) + yy_nh * xsf6_sh(ti_transit) + if (ti_transit < length_transit) & + xsf6_a = xsf6_a + month * ((1 - yy_nh) * xsf6_nh(ti_transit + 1) + yy_nh * xsf6_sh(ti_transit + 1) - xsf6_a) / 12. + end if + else + ! Constant (global-mean) namelist values are taken + end if + ! Local air-sea exchange gas flux of SF6 (m / s): + bc_surface = dt * (gas_flux("sf6", sst, sss, wind_2, aice, press_a, xsf6_a, sval) - sval * water_flux(n) * is_nonlinfs) + CASE (11) ! CFC-11 + if (anthro_transit) then + ! Select atmospheric input values corresponding to the latitude + ! Annual values are interpolated to monthly values, this is omitted in the last simulation year + if (y_abc > 10.) then ! Northern hemisphere + xf11_a = xf11_nh(ti_transit) + if (ti_transit < length_transit) xf11_a = xf11_a + month * (xf11_nh(ti_transit + 1) - xf11_a) / 12. + else if (y_abc <- 10.) then ! Southern hemisphere + xf11_a = xf12_sh(ti_transit) + if (ti_transit < length_transit) xf11_a = xf11_a + month * (xf11_sh(ti_transit + 1) - xf11_a) / 12. + else ! Tropical zone, interpolate between NH and SH + xf11_a = (1 - yy_nh) * xf11_nh(ti_transit) + yy_nh * xf11_sh(ti_transit) + if (ti_transit < length_transit) & + xf11_a = xf11_a + month * ((1 - yy_nh) * xf11_nh(ti_transit + 1) + yy_nh * xf11_sh(ti_transit + 1) - xf11_a) / 12. + end if + else + ! Constant (global-mean) namelist values are taken + end if + ! Local air-sea exchange gas flux of CFC-12 (m / s): + bc_surface = dt * (gas_flux("f11", sst, sss, wind_2, aice, press_a, xf11_a, sval) - sval * water_flux(n) * is_nonlinfs) + CASE (12) ! CFC-12 + if (anthro_transit) then + ! Select atmospheric input values corresponding to the latitude + ! Annual values are interpolated to monthly values, this is omitted in the last simulation year + if (y_abc > 10.) then ! Northern hemisphere + xf12_a = xf12_nh(ti_transit) + if (ti_transit < length_transit) xf12_a = xf12_a + month * (xf12_nh(ti_transit + 1) - xf12_a) / 12. + else if (y_abc <- 10.) then ! Southern hemisphere + xf12_a = xf12_sh(ti_transit) + if (ti_transit < length_transit) xf12_a = xf12_a + month * (xf12_sh(ti_transit + 1) - xf12_a) / 12. + else ! Tropical zone, interpolate between NH and SH + xf12_a = (1 - yy_nh) * xf12_nh(ti_transit) + yy_nh * xf12_sh(ti_transit) + if (ti_transit < length_transit) & + xf12_a = xf12_a + month * ((1 - yy_nh) * xf12_nh(ti_transit + 1) + yy_nh * xf12_sh(ti_transit + 1) - xf12_a) / 12. + end if + else + ! Constant (global-mean) namelist values are taken + end if + ! Local air-sea exchange gas flux of CFC-12 (m / s): + bc_surface = dt * (gas_flux("f12", sst, sss, wind_2, aice, press_a, xf12_a, sval) - sval * water_flux(n) * is_nonlinfs) + CASE (14) ! Radiocarbon (more precisely, fractionation-corrected 14C/C): + if (anthro_transit) then + ! Select atmospheric input values corresponding to the latitude + if (y_abc > 30.) then ! Northern hemisphere + r14c_a = r14c_nh(ti_transit) + else if (y_abc <- 30.) then ! Southern hemisphere + r14c_a = r14c_sh(ti_transit) + else ! Tropical zone (values prescribed in input file) + r14c_a = r14c_tz(ti_transit) + end if + xCO2_a = xCO2_ti(ti_transit) + else if (paleo_transit) then + r14c_a = r14c_ti(ti_transit) + xCO2_a = xCO2_ti(ti_transit) + else + ! Constant (global-mean) namelist values are taken + end if + ! Local isotopic 14CO2/CO2 air-sea exchange flux (m / s) + bc_surface = dt * (iso_flux("co2", sst, sss, wind_2, aice, press_a, xco2_a, r14c_a, sval, dic_0) - sval * water_flux(n) * is_nonlinfs) + CASE (39) ! Argon-39 (fractionationation-corrected 39Ar/Ar) + ! Local isotopic 39Ar/Ar air-sea exchange flux (m / s) + bc_surface = dt * (iso_flux("arg", sst, sss, wind_2, aice, press_a, xarg_a, r39ar_a, sval, arg_0) - sval * water_flux(n) * is_nonlinfs) +!--- Done with boundary conditions for transient tracers. #if defined(__recom) CASE (1001) ! DIN bc_surface= dt*(AtmNInput(n) + RiverDIN2D(n) * is_riverinput & @@ -1523,8 +1607,6 @@ FUNCTION bc_surface(n, id, sval, nzmin, partit) #endif CASE (101) ! apply boundary conditions to tracer ID=101 bc_surface= dt*(prec_rain(n))! - real_salt_flux(n)*is_nonlinfs) -!---Transient tracers (case ##6,12,14,39) need additional input parameters -! and are considered in the separate function transit_bc_surface !---wiso-code CASE (102) ! apply boundary conditions to tracer ID=101 (H218O) bc_surface = dt*wiso_flux_oce(n,1) @@ -1562,140 +1644,3 @@ FUNCTION bc_surface(n, id, sval, nzmin, partit) RETURN END FUNCTION - - -!=============================================================================== -! This function returns a boundary conditions for a specified transient tracer ID and surface node. -! Different to function bc_surface, SST, SSS, and sea ice concentrations are always needed as -! auxiliary variable -FUNCTION transit_bc_surface(n, id, sst, sss, aice, sval, nzmin, partit, mesh) - use MOD_MESH - USE MOD_PARTIT - USE MOD_PARSUP - USE o_ARRAYS - USE g_forcing_arrays - USE g_config - use g_clock - use mod_transit - implicit none - - integer, intent(in) :: n, id, nzmin - real(kind=WP), intent(in) :: sst, sss, aice, sval - type(t_partit),intent(inout), target :: partit - type(t_mesh), intent(in), target :: mesh - REAL(kind=WP) :: transit_bc_surface - character(len=10) :: id_string - - - ! --> is_nonlinfs=1.0 for zelvel,zstar .... - ! --> is_nonlinfs=0.0 for linfs - -#if defined (__oasis) -! SLP and wind speed in coupled setups. This is a makeshift solution -! as long as the true values are not provided by the AGCM / OASIS. - press_a = mean_slp - wind_2 = speed_2(stress_atmoce_x(n), stress_atmoce_y(n)) -#else - press_a = press_air(n) - wind_2 = u_wind(n)**2 + v_wind(n)**2 -#endif - -! The atmospheric input of bomb 14C, CFC-12, and SF6 depends on latitude. To that effect specify - y_abc = mesh%geo_coord_nod2D(2,n) / rad ! latitude of atmospheric tracer input - yy_nh = (10. - y_abc) * 0.05 ! interpolation weight for tropical tracer values - - - SELECT CASE (id) - -! Boundary conditions for additional (transient) tracers (14C, 39Ar, CFC-12, and SF6) - CASE (14) ! Radiocarbon (more precisely, fractionation-corrected 14C/C): - if (anthro_transit) then -! Select atmospheric input values corresponding to the latitude - if (y_abc > 30.) then -! Northern Hemisphere - r14c_a = r14c_nh(ti_transit) - else if (y_abc <- 30.) then -! Southern Hemisphere - r14c_a = r14c_sh(ti_transit) - else -! Tropical zone - r14c_a = r14c_tz(ti_transit) - end if - xCO2_a = xCO2_ti(ti_transit) - else if (paleo_transit) then - r14c_a = r14c_ti(ti_transit) - xCO2_a = xCO2_ti(ti_transit) - else -! Constant (global-mean) namelist values are taken - end if -! Local isotopic 14CO2/CO2 air-sea exchange flux (in m / s), -! since F14C is normalized to atmospheric (water) values the isotopic flux has to be -! corrected for precipitation or evaporation fluxes with different isotopic signatures. - transit_bc_surface = dt * (iso_flux("co2", sst, sss, wind_2, aice, press_a, xco2_a, r14c_a, sval, dic_0) & - - sval * water_flux(n) * is_nonlinfs) - - CASE (39) ! Argon-39 (fractionationation-corrected 39Ar/Ar) -! Local isotopic 39Ar/Ar air-sea exchange flux (in m / s), -! since F39Ar is normalized to atmospheric (water) values the isotopic flux has to be -! corrected for precipitation or evaporation fluxes with different isotopic signatures. - transit_bc_surface = dt * (iso_flux("arg", sst, sss, wind_2, aice, press_a, xarg_a, r39ar_a, sval, arg_0) & - - sval * water_flux(n) * is_nonlinfs) - - CASE (12) ! CFC-12 - if (anthro_transit) then -! Select atmospheric input values corresponding to the latitude -! Annual values are interpolated to monthly values, this is omitted in the last simulation year - if (y_abc > 10.) then ! Northern Hemisphere -! Northern Hemisphere - xf12_a = xf12_nh(ti_transit) - if (ti_transit < length_transit) xf12_a = xf12_a + month * (xf12_nh(ti_transit + 1) - xf12_a) / 12. - else if (y_abc <- 10.) then -! Southern Hemisphere - xf12_a = xf12_sh(ti_transit) - if (ti_transit < length_transit) xf12_a = xf12_a + month * (xf12_sh(ti_transit + 1) - xf12_a) / 12. - else -! Tropical zone, interpolate between NH and SH - xf12_a = (1 - yy_nh) * xf12_nh(ti_transit) + yy_nh * xf12_sh(ti_transit) - if (ti_transit < length_transit) & - xf12_a = xf12_a + month * ((1 - yy_nh) * xf12_nh(ti_transit + 1) + yy_nh * xf12_sh(ti_transit + 1) - xf12_a) / 12. - end if - else -! Constant (global-mean) namelist values are taken - end if - -! Local air-sea exchange gas flux of CFC-12 (in m / s): - transit_bc_surface = dt * (gas_flux("f12", sst, sss, wind_2, aice, press_a, xf12_a, sval) & - - sval * water_flux(n) * is_nonlinfs) - - CASE (6) ! SF6 - if (anthro_transit) then -! Select atmospheric input values corresponding to the latitude -! Annual values are interpolated to monthly values, this is omitted in the last simulation year - if (y_abc > 10.) then ! Northern Hemisphere -! Northern Hemisphere - xsf6_a = xsf6_nh(ti_transit) - if (ti_transit < length_transit) xsf6_a = xsf6_a + month * (xsf6_nh(ti_transit + 1) - xsf6_a) / 12. - else if (y_abc <- 10.) then -! Southern Hemisphere - xsf6_a = xsf6_sh(ti_transit) - if (ti_transit < length_transit) xsf6_a = xsf6_a + month * (xsf6_sh(ti_transit + 1) - xsf6_a) / 12. - else -! Tropical zone, interpolate between NH and SH - xsf6_a = (1 - yy_nh) * xsf6_nh(ti_transit) + yy_nh * xsf6_sh(ti_transit) - if (ti_transit < length_transit) & - xsf6_a = xsf6_a + month * ((1 - yy_nh) * xsf6_nh(ti_transit + 1) + yy_nh * xsf6_sh(ti_transit + 1) - xsf6_a) / 12. - end if - else -! Constant (global-mean) namelist values are taken - end if - -! Local air-sea exchange gas flux of SF6 (in m / s): - transit_bc_surface = dt * (gas_flux("sf6", sst, sss, wind_2, aice, press_a, xsf6_a, sval) & - - sval * water_flux(n) * is_nonlinfs) - -! Done with boundary conditions for (transient) tracers. - END SELECT - RETURN - -END FUNCTION - From ca08d96e14dfe98268c827235a120e58e573b6b2 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 12:25:07 +0100 Subject: [PATCH 09/12] IDs of transient tracers are no longer hard-wired. Include initialization of CFC-11. --- src/oce_setup_step.F90 | 92 +++++++++++++++++++++++++++--------------- 1 file changed, 59 insertions(+), 33 deletions(-) diff --git a/src/oce_setup_step.F90 b/src/oce_setup_step.F90 index 4a1e96d9d..f465155fd 100755 --- a/src/oce_setup_step.F90 +++ b/src/oce_setup_step.F90 @@ -277,7 +277,7 @@ SUBROUTINE tracer_init(tracers, partit, mesh) USE g_ic3d use g_forcing_param, only: use_age_tracer !---age-code use g_config, only : lwiso, use_transit ! add lwiso switch and switch for transient tracers - use mod_transit, only : index_transit + use mod_transit, only : index_transit_r14c, index_transit_r39ar, index_transit_f11, index_transit_f12, index_transit_sf6, l_r14c, l_r39ar, l_f11, l_f12, l_sf6 IMPLICIT NONE type(t_tracer), intent(inout), target :: tracers type(t_partit), intent(inout), target :: partit @@ -375,24 +375,42 @@ SUBROUTINE tracer_init(tracers, partit, mesh) !---age-code-end ! Transient tracers -!! UNDER CONSTRUCTION - Actually we do not want to hardwire the number of transient tracers if (use_transit) then ! add transient tracers to the model - nml_tracer_list(num_tracers+1) = nml_tracer_list(1) - nml_tracer_list(num_tracers+2) = nml_tracer_list(1) - nml_tracer_list(num_tracers+3) = nml_tracer_list(1) - nml_tracer_list(num_tracers+4) = nml_tracer_list(1) - nml_tracer_list(num_tracers+1)%id = 6 - nml_tracer_list(num_tracers+1)%id = 12 - nml_tracer_list(num_tracers+1)%id = 14 - nml_tracer_list(num_tracers+1)%id = 39 - - index_transit(1) = num_tracers+1 - index_transit(2) = num_tracers+2 - index_transit(3) = num_tracers+3 - index_transit(4) = num_tracers+4 - - num_tracers = num_tracers + 4 + if (l_sf6) then + nml_tracer_list(num_tracers+1) = nml_tracer_list(1) + nml_tracer_list(num_tracers+1)%id = 6 + index_transit_sf6 = num_tracers+1 + num_tracers = num_tracers + 1 + endif + + if (l_f11) then + nml_tracer_list(num_tracers+1) = nml_tracer_list(1) + nml_tracer_list(num_tracers+1)%id = 11 + index_transit_f11 = num_tracers+1 + num_tracers = num_tracers + 1 + endif + + if (l_f12) then + nml_tracer_list(num_tracers+1) = nml_tracer_list(1) + nml_tracer_list(num_tracers+1)%id = 12 + index_transit_f12 = num_tracers+1 + num_tracers = num_tracers + 1 + endif + + if (l_r14c) then + nml_tracer_list(num_tracers+1) = nml_tracer_list(1) + nml_tracer_list(num_tracers+1)%id = 14 + index_transit_r14c = num_tracers+1 + num_tracers = num_tracers + 1 + endif + + if (l_r39ar) then + nml_tracer_list(num_tracers+1) = nml_tracer_list(1) + nml_tracer_list(num_tracers+1)%id = 39 + index_transit_r39ar = num_tracers+1 + num_tracers = num_tracers + 1 + endif ! tracers initialised from file idlist((n_ic3d+1):(n_ic3d+1)) = (/14/) @@ -1161,19 +1179,16 @@ SUBROUTINE oce_initial_state(tracers, partit, mesh) !---wiso-code-end ! Transient tracers - CASE (14) ! initialize tracer ID=14, fractionation-corrected 14C/C -! this initialization can be overwritten by calling do_ic3d -!! if (.not. any(idlist == 14)) then ! CHECK IF THIS LINE IS STILL NECESSARY - tracers%data(i)%values(:,:) = 0.85 - if (mype==0) then - write (i_string, "(I3)") i - write (id_string, "(I3)") id - write(*,*) 'initializing '//trim(i_string)//'th tracer with ID='//trim(id_string) - write (*,*) tracers%data(i)%values(1,1) - end if -!! end if - CASE (39) ! initialize tracer ID=39, fractionation-corrected 39Ar/Ar - tracers%data(i)%values(:,:) = 0.85 + CASE (6) ! initialize tracer ID=6, SF6 + tracers%data(i)%values(:,:) = 0.0_WP + if (mype==0) then + write (i_string, "(I3)") i + write (id_string, "(I3)") id + write(*,*) 'initializing '//trim(i_string)//'th tracer with ID='//trim(id_string) + write (*,*) tracers%data(i)%values(1,1) + end if + CASE (11) ! initialize tracer ID=11, CFC-11 + tracers%data(i)%values(:,:) = 0.0_WP if (mype==0) then write (i_string, "(I3)") i write (id_string, "(I3)") id @@ -1181,15 +1196,26 @@ SUBROUTINE oce_initial_state(tracers, partit, mesh) write (*,*) tracers%data(i)%values(1,1) end if CASE (12) ! initialize tracer ID=12, CFC-12 - tracers%data(i)%values(:,:) = 0. + tracers%data(i)%values(:,:) = 0.0_WP if (mype==0) then write (i_string, "(I3)") i write (id_string, "(I3)") id write(*,*) 'initializing '//trim(i_string)//'th tracer with ID='//trim(id_string) write (*,*) tracers%data(i)%values(1,1) end if - CASE (6) ! initialize tracer ID=6, SF6 - tracers%data(i)%values(:,:) = 0. + CASE (14) ! initialize tracer ID=14, fractionation-corrected 14C/C +! this initialization can be overwritten by calling do_ic3d + if (.not. any(idlist == 14)) then ! CHECK IF THIS LINE IS STILL NECESSARY + tracers%data(i)%values(:,:) = 0.85 + if (mype==0) then + write (i_string, "(I3)") i + write (id_string, "(I3)") id + write(*,*) 'initializing '//trim(i_string)//'th tracer with ID='//trim(id_string) + write (*,*) tracers%data(i)%values(1,1) + end if + end if + CASE (39) ! initialize tracer ID=39, fractionation-corrected 39Ar/Ar + tracers%data(i)%values(:,:) = 0.50 if (mype==0) then write (i_string, "(I3)") i write (id_string, "(I3)") id From 3025ad508b1f690315940eb5d06b24494799ca84 Mon Sep 17 00:00:00 2001 From: Martin Butzin Date: Thu, 19 Dec 2024 14:23:24 +0100 Subject: [PATCH 10/12] Remove obsolete reference to module 'transit_bc_surface_interface.mod'.. --- src/oce_ale_tracer.F90 | 1 - 1 file changed, 1 deletion(-) diff --git a/src/oce_ale_tracer.F90 b/src/oce_ale_tracer.F90 index 83e28d74a..a8ad8fc07 100644 --- a/src/oce_ale_tracer.F90 +++ b/src/oce_ale_tracer.F90 @@ -557,7 +557,6 @@ subroutine diff_ver_part_impl_ale(tr_num, dynamics, tracers, ice, partit, mesh) use o_mixing_KPP_mod !for ghats _GO_ use g_cvmix_kpp, only: kpp_nonlcltranspT, kpp_nonlcltranspS, kpp_oblmixc use bc_surface_interface - use transit_bc_surface_interface use mod_ice use iceberg_params implicit none From 58d1a93ddcea7ad869935e55c60d49bd10552854 Mon Sep 17 00:00:00 2001 From: ackerlar Date: Fri, 10 Jan 2025 16:11:14 +0100 Subject: [PATCH 11/12] make changes consistent with main --- src/gen_modules_clock.F90 | 41 +++++++++++++++++++++++++++++---------- src/io_meandata.F90 | 15 ++++++++++---- 2 files changed, 42 insertions(+), 14 deletions(-) diff --git a/src/gen_modules_clock.F90 b/src/gen_modules_clock.F90 index 0a287ab7e..99f861e0c 100755 --- a/src/gen_modules_clock.F90 +++ b/src/gen_modules_clock.F90 @@ -2,6 +2,8 @@ module g_clock !combining RT and Lars version ! use g_config + use iso_fortran_env, only: error_unit + use mpi implicit none save real(kind=WP) :: timeold, timenew !time in a day, unit: sec @@ -69,13 +71,15 @@ end subroutine clock ! subroutine clock_init(partit) USE MOD_PARTIT - USE MOD_PARSUP use g_config use mod_transit, only: ti_transit, ti_start_transit implicit none type(t_partit), intent(in), target :: partit integer :: i, daystart, yearstart real(kind=WP) :: aux1, aux2, timestart + integer :: ierr + integer :: file_unit + character(512) :: errmsg ! the model initialized at timestart=timenew @@ -83,10 +87,17 @@ subroutine clock_init(partit) yearstart=yearnew ! init clock for this run - open(99,file=trim(ResultPath)//trim(runid)//'.clock',status='old') - read(99,*) timeold, dayold, yearold - read(99,*) timenew, daynew, yearnew - close(99) + open(newunit=file_unit, file=trim(ResultPath)//trim(runid)//'.clock', action='read', & + status='old', iostat=ierr, iomsg=errmsg) + if (ierr /= 0) then + write (unit=error_unit, fmt='(3A)') & + '### error: can not open file ', trim(ResultPath)//trim(runid)//'.clock', & + ', error: ' // trim(errmsg) + call MPI_Abort(MPI_COMM_WORLD, 1, ierr) + end if + read(unit=file_unit, fmt=*) timeold, dayold, yearold + read(unit=file_unit, fmt=*) timenew, daynew, yearnew + close(unit=file_unit) if(daynew==0) daynew=1 ! check if this is a restart or not @@ -157,6 +168,9 @@ subroutine clock_finish real(kind=WP) :: dum_timenew !time in a day, unit: sec integer :: dum_daynew !day in a year integer :: dum_yearnew !year before and after time step + integer :: ierr + integer :: file_unit + character(512) :: errmsg dum_timenew = timenew dum_daynew = daynew @@ -167,10 +181,17 @@ subroutine clock_finish dum_yearnew=yearold+1 endif - open(99,file=trim(ResultPath)//trim(runid)//'.clock',status='unknown') - write(99,*) timeold, dayold, yearold - write(99,*) dum_timenew, dum_daynew, dum_yearnew - close(99) + open(newunit=file_unit, file=trim(ResultPath)//trim(runid)//'.clock', action='write', & + status='unknown', iostat=ierr, iomsg=errmsg) + if (ierr /= 0) then + write (unit=error_unit, fmt='(3A)') & + '### error: can not open file ', trim(ResultPath)//trim(runid)//'.clock', & + ', error: ' // trim(errmsg) + call MPI_Abort(MPI_COMM_WORLD, 1, ierr) + end if + write(unit=file_unit, fmt=*) timeold, dayold, yearold + write(unit=file_unit, fmt=*) dum_timenew, dum_daynew, dum_yearnew + close(unit=file_unit) end subroutine clock_finish ! !---------------------------------------------------------------------------- @@ -211,4 +232,4 @@ end subroutine is_fleapyr ! !---------------------------------------------------------------------------- ! -end module g_clock +end module g_clock \ No newline at end of file diff --git a/src/io_meandata.F90 b/src/io_meandata.F90 index fe42eb7db..94d8e3a16 100644 --- a/src/io_meandata.F90 +++ b/src/io_meandata.F90 @@ -241,7 +241,7 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) end if CASE ('m_ice ') if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'm_ice', 'ice height', 'm', ice%data(2)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'm_ice', 'ice height per unit area', 'm', ice%data(2)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) end if CASE ('thdgr ') if (use_ice) then @@ -257,9 +257,16 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) end if CASE ('m_snow ') if (use_ice) then - call def_stream(nod2D, myDim_nod2D, 'm_snow', 'snow height', 'm', ice%data(3)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'm_snow', 'snow height per unit area', 'm', ice%data(3)%values(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) end if - +CASE ('h_ice ') + if (use_ice) then + call def_stream(nod2D, myDim_nod2D, 'h_ice', 'ice thickness over ice-covered fraction', 'm', ice%h_ice(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + end if +CASE ('h_snow ') + if (use_ice) then + call def_stream(nod2D, myDim_nod2D, 'h_snow', 'snow thickness over ice-covered fraction', 'm', ice%h_snow(1:myDim_nod2D), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + end if ! Debug ice variables CASE ('strength_ice') if (use_ice) then @@ -377,7 +384,7 @@ subroutine ini_mean_io(ice, dynamics, tracers, partit, mesh) !_______________________________________________________________________________ ! output surface forcing CASE ('fh ') - call def_stream(nod2D, myDim_nod2D, 'fh', 'heat flux', 'W', heat_flux_in(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) + call def_stream(nod2D, myDim_nod2D, 'fh', 'heat flux', 'W/m2', heat_flux_in(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) CASE ('fw ') call def_stream(nod2D, myDim_nod2D, 'fw', 'fresh water flux', 'm/s', water_flux(:), io_list(i)%freq, io_list(i)%unit, io_list(i)%precision, partit, mesh) CASE ('atmice_x ') From 147b2f8ebf21f3a761ca92191c764047cea17911 Mon Sep 17 00:00:00 2001 From: ackerlar Date: Mon, 13 Jan 2025 17:43:07 +0100 Subject: [PATCH 12/12] uncomment a_ice in bc_surface header --- src/oce_ale_tracer.F90 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/oce_ale_tracer.F90 b/src/oce_ale_tracer.F90 index a8ad8fc07..0a07f26b2 100644 --- a/src/oce_ale_tracer.F90 +++ b/src/oce_ale_tracer.F90 @@ -1463,7 +1463,7 @@ FUNCTION bc_surface(n, id, sval, nzmin, partit, mesh, sst, sss, aice) REAL(kind=WP) :: bc_surface character(len=10) :: id_string -! real(kind=WP), dimension(:), pointer :: a_ice !! MB: where is this needed? + real(kind=WP), dimension(:), pointer :: a_ice !! MB: where is this needed? if (use_transit) then #if defined (__oasis)