10#include "scaleFElib.h"
16 use scale_const,
only: &
17 undef8 => const_undef8
22 use scale_time,
only: &
33 time_dtsec_wallclock_check, &
34 time_dstep_wallclock_check
37 use scale_calendar,
only: &
38 calendar_date2daysec, &
39 calendar_daysec2date, &
40 calendar_combine_daysec, &
41 calendar_adjust_daysec, &
43 calendar_cfunits2sec, &
62 type :: time_manager_process
67 integer :: inner_itr_num
68 character(len=H_SHORT) :: process_name
70 procedure,
public :: Init => time_manager_process_init
71 procedure,
public :: Check_state => time_manager_process_checkstate
72 procedure,
public :: Final => time_manager_process_final
73 end type time_manager_process
75 integer,
private,
parameter :: TIME_MANAGER_PROCESS_MAX_NUM = 10
82 real(dp) :: dtsec_restart
83 integer :: dstep_restart
84 integer :: res_step_restart
86 character(len=H_SHORT) :: comp_name
88 type(time_manager_process) :: process_list(time_manager_process_max_num)
89 integer :: process_num
91 procedure,
public :: init => time_manager_component_init
92 procedure,
public :: regist_process => time_manager_component_regist_process
93 procedure,
public :: check_state => time_manager_component_checkstate
94 procedure,
public :: do_process => time_manager_component_do_process
95 procedure,
public :: get_process_inner_itr_num => time_manager_component_get_process_inner_itr_num
96 procedure,
public :: final => time_manager_component_final
122 public :: time_startdaysec
123 public :: time_nowdate, time_nowsubsec, time_nowday, time_nowdaysec, time_nowsec
124 public :: time_nowstep, time_nstep
126 public :: time_offset_year
141 integer,
private :: time_res_resume
143 real(dp),
private :: time_wallclock_start
144 real(dp),
private :: time_wallclock_limit = -1.0_dp
145 real(dp),
private :: time_wallclock_safe = 0.9_dp
146 real(dp),
private :: time_wallclock_safelim
148 real(dp),
private,
parameter :: eps = 1.e-6_dp
151 integer,
private,
parameter :: time_manager_component_max_num = 20
152 type time_manager_component_ptr
154 end type time_manager_component_ptr
155 type(time_manager_component_ptr),
private :: time_manager_comp_ptr_list(time_manager_component_max_num)
156 integer,
private :: time_manager_component_num
158 logical :: setup_tinteg_flag
163 setup_TimeIntegration, &
164 restart_in_basename )
166 use scale_file,
only: &
171 logical,
intent(in),
optional :: setup_timeintegration
172 character(len=*),
intent(in),
optional :: restart_in_basename
174 real(dp) :: time_duration = undef8
175 character(len=H_SHORT) :: time_duration_unit =
"SEC"
176 real(dp) :: time_dt = undef8
177 character(len=H_SHORT) :: time_dt_unit =
"SEC"
179 real(dp) :: time_dt_resume = undef8
180 character(len=H_SHORT) :: time_dt_resume_unit =
""
182 real(dp) :: time_dt_wallclock_check = undef8
183 character(len=H_SHORT) :: time_dt_wallclock_check_unit =
""
185 namelist /param_time/ &
189 time_duration_unit, &
193 time_dt_resume_unit, &
194 time_dt_wallclock_check, &
195 time_dt_wallclock_check_unit, &
196 time_wallclock_limit, &
201 real(dp) :: cftime(1)
202 character(len=H_MID) :: cfunits
204 character(len=27) :: startchardate
205 character(len=27) :: endchardate
212 log_info(
"TIME_manager_setup",*)
'Setup'
216 read(io_fid_conf,nml=param_time,iostat=ierr)
218 log_info(
"TIME_manager_setup",*)
'Not found namelist. Default used.'
219 elseif( ierr > 0 )
then
220 log_error(
"TIME_manager_setup",*)
'Not appropriate names in namelist PARAM_TIME. Check!'
225 if (
present(setup_timeintegration) )
then
226 setup_tinteg_flag = setup_timeintegration
228 setup_tinteg_flag = .true.
231 if ( setup_tinteg_flag )
then
232 if ( time_dt == undef8 )
then
233 log_error(
"TIME_manager_setup",*)
'Not found TIME_DT. STOP.'
236 if ( time_duration == undef8 )
then
237 log_error(
"TIME_manager_setup",*)
'Not found TIME_DURATION. STOP.'
241 if ( time_dt_resume == undef8 )
then
242 time_dt_resume = time_duration
244 if ( time_dt_resume_unit ==
'' )
then
245 log_info_cont(*)
'Not found TIME_DT_RESUME_UNIT. TIME_DURATION_UNIT is used.'
246 time_dt_resume_unit = time_duration_unit
252 if (
present(restart_in_basename) )
then
253 if ( restart_in_basename /=
'' )
then
254 call file_get_attribute( restart_in_basename, &
258 rankid = prc_myrank, &
261 call file_get_attribute( restart_in_basename, &
265 rankid = prc_myrank, &
269 datesec = calendar_cfunits2sec( cftime(1), cfunits, 0 )
271 call calendar_adjust_daysec( dateday, datesec )
292 call calendar_date2char( startchardate, &
301 time_nowdaysec = calendar_combine_daysec( time_nowday, time_nowsec )
304 if (setup_tinteg_flag)
then
305 call calendar_unit2sec(
time_durationsec, time_duration, time_duration_unit )
319 call calendar_date2char( endchardate, &
324 log_info(
"TIME_manager_setup",*)
'Global date / time setting '
325 log_info_cont(
'(1x,A,A)')
'START Date : ', startchardate
326 log_info_cont(
'(1x,A,A)')
'END Date : ', endchardate
328 if (setup_tinteg_flag)
then
329 call calendar_unit2sec( time_dtsec, time_dt, time_dt_unit )
341 time_manager_component_num = 0
342 do n=1, time_manager_component_max_num
343 nullify( time_manager_comp_ptr_list(n)%ptr )
347 time_wallclock_start = prc_mpitime()
349 if ( time_wallclock_limit > 0.0_dp )
then
351 log_info(
"TIME_manager_setup",*)
'Wall clock time limit of execution is specified.'
353 if ( time_dt_wallclock_check == undef8 )
then
354 log_info_cont(*)
'Not found TIME_DT_WALLCLOCK_CHECK. TIME_DT is used.'
355 time_dtsec_wallclock_check = time_dtsec
357 if ( time_dt_wallclock_check_unit ==
'' )
then
358 log_info_cont(*)
'Not found TIME_DT_WALLCLOCK_CHECK_UNIT. TIME_DURATION_UNIT is used.'
359 time_dt_wallclock_check_unit = time_duration_unit
361 call calendar_unit2sec( time_dtsec_wallclock_check, time_dt_wallclock_check, time_dt_wallclock_check_unit )
362 time_dtsec_wallclock_check = max( time_dtsec_wallclock_check, time_dtsec )
365 time_dstep_wallclock_check = int( time_dtsec_wallclock_check / time_dtsec )
367 time_wallclock_safe = max( min( time_wallclock_safe, 1.0_dp ), 0.0_dp )
368 time_wallclock_safelim = time_wallclock_limit * time_wallclock_safe
370 log_info_cont(
'(1x,A,F10.1,A)')
'This job stops after ', time_wallclock_safelim,
' seconds.'
371 log_info_cont(
'(1x,A,F10.3,A,I8,A)')
'Time interval for check : ', time_dtsec_wallclock_check, &
372 ' (step interval=', time_dstep_wallclock_check,
')'
384 do n=1, time_manager_component_num
385 nullify( time_manager_comp_ptr_list(n)%ptr )
387 time_manager_component_num = 0
396 time_nowstep = time_nowstep + 1
398 time_nowsec =
time_startsec + real(time_nowstep-1,kind=dp) * time_dtsec
401 call calendar_adjust_daysec( time_nowday, time_nowsec )
403 call calendar_daysec2date( time_nowdate(:), time_nowsubsec, &
404 time_nowday, time_nowsec, time_offset_year )
406 time_nowdaysec = calendar_combine_daysec( time_nowday, time_nowsec )
408 if (time_nowstep > time_nstep)
then
416 use scale_calendar,
only: calendar_date2char
422 real(dp) :: wallclock_elapse
423 character(len=27) :: nowchardate
427 do n=1, time_manager_component_num
428 tm_comp => time_manager_comp_ptr_list(n)%ptr
429 call tm_comp%Check_state()
433 time_res_resume = time_res_resume + 1
443 if ( io_step_to_stdout > 0 )
then
444 if( mod(time_nowstep-1,io_step_to_stdout) == 0 ) to_stdout = .true.
447 call calendar_date2char( nowchardate, &
451 wallclock_elapse = prc_mpitime() - time_wallclock_start
454 if ( time_wallclock_limit > 0.0_dp )
then
455 log_progress(
'(1x,2A,2(A,I7),2(A,F10.1))')
'TIME: ', nowchardate,
' STEP:',time_nowstep,
'/', time_nstep, &
456 ' WCLOCK:', wallclock_elapse,
'/', time_wallclock_safelim
457 if ( prc_universal_ismaster .AND. to_stdout )
then
458 write(*,
'(1x,2A,2(A,I7),2(A,F10.1))')
'TIME: ', nowchardate,
' STEP:',time_nowstep,
'/', time_nstep, &
459 ' WCLOCK:', wallclock_elapse,
'/', time_wallclock_safelim
462 log_progress(
'(1x,2A,2(A,I7),A,F10.1)')
'TIME: ', nowchardate,
' STEP:',time_nowstep,
'/', time_nstep, &
463 ' WCLOCK:', wallclock_elapse
464 if ( prc_universal_ismaster .AND. to_stdout )
then
465 write(*,
'(1x,2A,2(A,I7),A,F10.1)')
'TIME: ', nowchardate,
' STEP:',time_nowstep,
'/', time_nstep, &
466 ' WCLOCK:', wallclock_elapse
478 type(time_manager_process),
pointer :: tm_process
482 log_info(
"TIME_manager_report_timeintervals",*)
'Time interval for each component (sec.)'
484 do n=1, time_manager_component_num
485 tm_comp => time_manager_comp_ptr_list(n)%ptr
486 log_info_cont(*) trim(tm_comp%comp_name)
487 do p=1, tm_comp%process_num
488 tm_process => tm_comp%process_list(p)
489 log_info_cont(
'(1x,A,F10.3)') trim(tm_process%process_name)//
' (time) : ', tm_process%dtsec
490 if (tm_process%inner_itr_num > 1)
then
491 log_info_cont(
'(1x,A,I10)')
' (step) : ', tm_process%inner_itr_num
493 log_info_cont(
'(1x,A,I8,A)')
' (step interval=', tm_process%dstep,
')'
498 log_info_cont(*)
'Time interval for restart (sec.)'
500 do n=1, time_manager_component_num
501 tm_comp => time_manager_comp_ptr_list(n)%ptr
502 log_info_cont(
'(1x,A,F10.3,A,I8,A)') trim(tm_comp%comp_name)//
' variables : ', tm_comp%dtsec_restart, &
503 ' (step interval=', tm_comp%dstep_restart,
')'
517 time_manager_component_num = time_manager_component_num + 1
518 if (time_manager_component_num > time_manager_component_max_num)
then
519 log_error(
"TIME_manager_regist_component",*)
'The number of TIME_manager_component registered exceeds ', &
520 time_manager_component_num, time_manager_component_max_num
524 time_manager_comp_ptr_list(time_manager_component_num)%ptr => tmanager_comp
529 subroutine time_manager_component_init( this, comp_name, &
530 dt, dt_unit, dt_restart, dt_restart_unit )
534 character(*),
intent(in) :: comp_name
535 real(dp),
intent(in) :: dt
536 character(*),
intent(in) :: dt_unit
537 real(dp),
intent(in) :: dt_restart
538 character(*),
intent(in) :: dt_restart_unit
540 real(rp) :: start_sec, end_sec
541 real(rp) :: absday, absdaysec, abssec
546 this%res_step_restart = 0
547 this%comp_name = comp_name
550 if (.not. setup_tinteg_flag)
return
552 if (dt == undef8)
then
553 log_info_cont(*)
'Not found TIME_DT_'//trim(comp_name)//
'. TIME_DTSEC is used.'
554 this%dtsec = time_dtsec
556 call calendar_unit2sec( this%dtsec, dt, dt_unit )
559 this%dstep = nint( this%dtsec / time_dtsec )
561 if ( abs(this%dtsec - real(this%dstep,kind=dp)*time_dtsec) > eps )
then
562 log_error(
"TIME_manager_component_Init",*)
'delta t('//trim(comp_name)//
') must be a multiple of delta t ', &
563 this%dtsec, real(this%dstep,kind=dp)*time_dtsec
569 if (dt_restart == undef8)
then
570 log_info_cont(*)
'Not found TIME_DT_'//trim(comp_name)//
'_RESTART. TIME_DURATION is used.'
573 call calendar_unit2sec( this%dtsec_restart, dt_restart, dt_restart_unit )
576 this%dstep_restart = nint( this%dtsec_restart / time_dtsec )
578 if ( abs(this%dtsec_restart - real(this%dstep_restart,kind=dp)*time_dtsec) > eps )
then
579 log_error(
"TIME_manager_component_Init",*)
'delta t('//trim(comp_name)//
'_RESTART) must be a multiple of delta t ', &
580 this%dtsec_restart, real(this%dstep_restart,kind=dp)*time_dtsec
587 end subroutine time_manager_component_init
589 subroutine time_manager_component_checkstate( this )
594 type(time_manager_process),
pointer :: tm_process
597 this%do_step = .false.
599 if (this%process_num > 0)
then
600 do n=1, this%process_num
601 tm_process => this%process_list(n)
602 call tm_process%Check_state()
603 if (tm_process%do_step) this%do_step = .true.
606 this%res_step = this%res_step + 1
607 if ( this%res_step == this%dstep )
then
608 this%do_step = .true.
614 this%do_restart = .false.
616 this%res_step_restart = this%res_step_restart + 1
617 if ( this%res_step_restart == this%dstep_restart )
then
618 this%do_restart = .true.
619 this%res_step_restart = 0
623 end subroutine time_manager_component_checkstate
625 subroutine time_manager_component_regist_process( this, &
626 process_name, dt, dt_unit, &
631 character(*),
intent(in) :: process_name
632 real(dp),
intent(in) :: dt
633 character(*),
intent(in) :: dt_unit
634 integer,
intent(out) :: tm_process_id
636 type(time_manager_process),
pointer :: tm_process
639 this%process_num = this%process_num + 1
640 tm_process_id = this%process_num
642 if( this%process_num > time_manager_process_max_num)
then
643 log_error(
"TIME_manager_component_Regist_process",*)
'The number of TIME_manager_process registered exceeds ', &
644 this%process_num, time_manager_process_max_num
648 tm_process => this%process_list(this%process_num)
649 call tm_process%Init( process_name, dt, dt_unit )
652 end subroutine time_manager_component_regist_process
654 subroutine time_manager_component_final( this )
661 do n=1, this%process_num
662 call this%process_list(n)%Final()
667 end subroutine time_manager_component_final
669 function time_manager_component_do_process( this, tm_process_id )
result(do_step)
672 integer,
intent(in) :: tm_process_id
675 do_step = this%process_list(tm_process_id)%do_step
677 end function time_manager_component_do_process
679 function time_manager_component_get_process_inner_itr_num( this, tm_process_id )
result(itr_num)
682 integer,
intent(in) :: tm_process_id
685 itr_num = this%process_list(tm_process_id)%inner_itr_num
687 end function time_manager_component_get_process_inner_itr_num
691 subroutine time_manager_process_init( this, process_name, &
695 class(time_manager_process),
intent(inout) :: this
696 character(*),
intent(in) :: process_name
697 real(dp),
intent(in) :: dt
698 character(*),
intent(in) :: dt_unit
702 this%process_name = process_name
704 if (.not. setup_tinteg_flag)
return
706 if (dt == undef8)
then
707 log_info_cont(*)
'Not found TIME_DT_'//trim(process_name)//
'. TIME_DTSEC is used.'
708 this%dtsec = time_dtsec
710 call calendar_unit2sec( this%dtsec, dt, dt_unit )
713 this%inner_itr_num = max( nint(time_dtsec/this%dtsec), 1 )
714 this%dstep = nint( this%dtsec / time_dtsec * dble(this%inner_itr_num) )
716 if ( abs( real(this%inner_itr_num, kind=dp)*this%dtsec &
717 - real(this%dstep ,kind=dp)*time_dtsec ) > eps )
then
718 log_error(
"TIME_manager_process_Init",*)
'delta t('//trim(process_name)//
') must be a multiple of delta t ', &
719 this%dtsec, real(this%dstep,kind=dp)*time_dtsec
724 end subroutine time_manager_process_init
727 subroutine time_manager_process_final( this )
729 class(time_manager_process),
intent(inout) :: this
733 end subroutine time_manager_process_final
735 subroutine time_manager_process_checkstate( this )
737 class(time_manager_process),
intent(inout) :: this
740 this%do_step = .false.
741 this%res_step = this%res_step + 1
742 if ( this%res_step == this%dstep )
then
743 this%do_step = .true.
748 end subroutine time_manager_process_checkstate
752end module scale_time_manager
subroutine, public time_manager_advance()
real(dp), public time_endsec
integer, public time_endday
subroutine, public time_manager_report_timeintervals()
real(dp), public time_startsubsec
integer, dimension(6), public time_startdate
real(dp), public time_durationsec
subroutine, public time_manager_checkstate()
real(dp), public time_endsubsec
real(dp), public time_dtsec_resume
subroutine, public time_manager_final()
integer, dimension(6), public time_enddate
logical, public time_doend
real(dp), public time_startsec
integer, public time_startday
logical, public time_doresume
integer, public time_dstep_resume
real(dp), public time_startms
[millisec]
subroutine, public time_manager_regist_component(tmanager_comp)
subroutine, public time_manager_init(setup_timeintegration, restart_in_basename)