!=============================================================================== ! SVN $Id: shr_date_mod.F90 11998 2008-09-30 05:47:54Z erik $ ! SVN $URL: https://svn-ccsm-models.cgd.ucar.edu/csm_share/branch_tags/cesm1_0_rel_tags/cesm1_0_rel01_share3_100616/shr/shr_date_mod.F90 $ !=============================================================================== !BOP =========================================================================== ! ! !MODULE: shr_date_mod -- date/time module, built upon a calendar module ! ! !DESCRIPTION: ! Keeps track of model date, including elapsed seconds in current date. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - created initial version ! ! !REMARKS: ! This module is independant of a particular calendar, e.g. is ignorant of ! whether the underlying calendar does or doesn't implement leap years. ! ! !INTERFACE: ------------------------------------------------------------------ module shr_date_mod 2,9 ! !USES: use shr_cal_mod ! underlying calendar use shr_sys_mod ! system call wrappers use shr_kind_mod ! kinds use shr_log_mod, only: s_loglev => shr_log_Level use shr_log_mod, only: s_logunit => shr_log_Unit implicit none private ! except ! !PUBLIC TYPES: public :: shr_date type shr_date sequence ! place in contiguous memory private ! no public access to internal components integer(SHR_KIND_IN) :: y ! calendar year integer(SHR_KIND_IN) :: m ! calendar month integer(SHR_KIND_IN) :: d ! calendar day integer(SHR_KIND_IN) :: s ! elapsed seconds in current day integer(SHR_KIND_IN) :: cDate ! coded calendar date (yymmdd) integer(SHR_KIND_IN) :: eDay ! elsapsed days relative to calendar's reference date integer(SHR_KIND_IN) :: stepInDay ! current time-step in current day integer(SHR_KIND_IN) :: stepsPerDay ! number of time-steps per day end type shr_date ! !PUBLIC MEMBER FUNCTIONS: public :: shr_date_adv1step ! advance the date one time step public :: shr_date_advNextDay ! advance date to next day, 0 seconds public :: shr_date_initYMD ! init date given YMD public :: shr_date_initEDay ! init date given elapsed days public :: shr_date_initCDate ! init date given coded date (yymmdd) public :: shr_date_getYMD ! returns integers yy,mm,dd,sssss public :: shr_date_getEDay ! returns elased day, sssss public :: shr_date_getCDate ! returns coded date, ssssS public :: shr_date_getStepsPerDay ! returns steps per day public :: shr_date_getStepInDay ! returns step in the day public :: shr_date_getJulian ! returns julian day number public :: assignment(=) ! sets (date a) equal to (date b) public :: operator(==) ! true iff (date a) == (date b) public :: operator(<) ! true iff (date a) < (date b) public :: operator(>) ! true iff (date a) > (date b) ! !PUBLIC DATA MEMBERS: real(SHR_KIND_R8), parameter,public :: shr_date_secsPerDay = 86400.0_SHR_KIND_R8 ! seconds per day integer(SHR_KIND_IN), parameter :: shr_date_nsecsPerDay = shr_date_secsPerDay !EOP interface assignment(=) module procedure shr_date_assign end interface interface operator(==) module procedure shr_date_equals end interface interface operator(>) module procedure shr_date_greater end interface interface operator(<) module procedure shr_date_less end interface !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ contains !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ !=============================================================================== !=============================================================================== subroutine shr_date_assign(a,b) 1 type(shr_date),intent(out) :: a type(shr_date),intent(in ) :: b !------------------------------------------------------------------------------- ! Make date a equal to date b !------------------------------------------------------------------------------- a%y = b%y a%m = b%m a%d = b%d a%s = b%s a%eday = b%eday a%cDate = b%cDate a%stepInDay = b%stepInDay a%stepsPerDay = b%stepsPerDay end subroutine shr_date_assign !=============================================================================== !=============================================================================== function shr_date_equals(a,b) 1 type(shr_date),intent(in) :: a,b logical :: shr_date_equals !------------------------------------------------------------------------------- ! Is date a equal to date b ?? !------------------------------------------------------------------------------- if (a%eday == b%eday .and. a%s == b%s) then shr_date_equals=.true. else shr_date_equals =.false. end if end function shr_date_equals !=============================================================================== !=============================================================================== function shr_date_greater(a,b) 1 type(shr_date),intent(in) :: a,b logical :: shr_date_greater !------------------------------------------------------------------------------- ! Is date a greater than date b ?? !------------------------------------------------------------------------------- if (a%eday < b%eday) then shr_date_greater = .false. else if (a%eday > b%eday) then shr_date_greater = .true. else if (a%s > b%s ) then shr_date_greater = .true. else shr_date_greater = .false. end if end function shr_date_greater !=============================================================================== !=============================================================================== function shr_date_less(a,b) 1 type(shr_date),intent(in) :: a,b logical :: shr_date_less !------------------------------------------------------------------------------- ! Is date a less than date b ?? !------------------------------------------------------------------------------- if (a%eday < b%eday) then shr_date_less = .true. else if (a%eday > b%eday) then shr_date_less = .false. else if (a%s < b%s ) then shr_date_less = .true. else shr_date_less = .false. end if end function shr_date_less !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_initYMD - Initialize date given y,m,d, steps per day. ! ! !DESCRIPTION: ! Initialize date given y,m,d, and steps per day. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- type(shr_date) function shr_date_initYMD(y,m,d,ns,sec) 12,4 implicit none ! !INPUT/OUTPUT PARAMETERS: integer(SHR_KIND_IN),intent(in) :: y ! year integer(SHR_KIND_IN),intent(in) :: m ! month integer(SHR_KIND_IN),intent(in) :: d ! day integer(SHR_KIND_IN),intent(in) :: ns ! number of steps per day. integer(SHR_KIND_IN),intent(in), optional :: sec ! seconds into day !EOP !----- local ----- type(shr_date) :: date !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- if (shr_cal_validYMD(y,m,d) ) then date%y = y date%m = m date%d = d date%s = 0 date%stepInDay = 0 date%stepsPerDay = ns if (present(sec)) then date%s = sec date%stepInDay = nint(date%s*ns/shr_date_secsPerDay) end if call shr_cal_ymd2date(y,m,d,date%cDate) call shr_cal_ymd2eDay(y,m,d,date%eDay ) else write(s_logunit,*) 'ERROR: invalid y,m,d = ', y,m,d call shr_sys_abort('(shr_date_initYMD) invalid date') end if shr_date_initYMD = date end function shr_date_initYMD !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_initEDay - Initialize date given an elapsed day ! ! !DESCRIPTION: ! Initialize date given elapsed days and the number of steps per day. ! ! !REVISION HISTORY: ! 2003-May-27 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- type(shr_date) function shr_date_initEDay(eDay,ns),4 implicit none ! !INPUT/OUTPUT PARAMETERS: integer(SHR_KIND_IN),intent(in) :: eDay ! elpased days integer(SHR_KIND_IN),intent(in) :: ns ! number of steps per day. !EOP !----- local ----- type(shr_date) :: date ! date to return integer(SHR_KIND_IN) :: y,m,d ! year, month, day !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- call shr_cal_eDay2ymd(eDay,y,m,d) ! convert eDay to y,m,d if ( shr_cal_validYMD(y,m,d) ) then date = shr_date_initYMD(y,m,d,ns) else write(s_logunit,*) 'ERROR: invalid eDay = ', eDay write(s_logunit,*) 'ERROR: invalid y,m,d = ', y,m,d call shr_sys_abort('(shr_date_initEDay) invalid eDay') end if shr_date_initEDay = date end function shr_date_initEDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_initCDate - Initialize date given coded date (yymmdd). ! ! !DESCRIPTION: ! Initialize date given coded date (yymmdd) and the number of steps per day. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- type(shr_date) function shr_date_initCDate(cDate,ns,sec),5 implicit none ! !INPUT/OUTPUT PARAMETERS: integer(SHR_KIND_IN),intent(in) :: cDate ! coded date (yymmdd) integer(SHR_KIND_IN),intent(in) :: ns ! number of steps per day. integer(SHR_KIND_IN),intent(in), optional :: sec ! seconds into day !EOP !----- local ----- type(shr_date) :: date ! date to return integer(SHR_KIND_IN) :: y,m,d ! year, month, day !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- if (shr_cal_validDate(cDate) ) then call shr_cal_date2ymd(cDate,y,m,d) date = shr_date_initYMD(y,m,d,ns) if (present(sec)) then date = shr_date_initYMD(y,m,d,ns,sec) end if else write(s_logunit,*) 'ERROR: invalid cDate = ', cDate call shr_sys_abort('(shr_date_initCDate) invalid date') end if shr_date_initCDate = date end function shr_date_initCDate !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_adv1step - advance the date by one time step ! ! !DESCRIPTION: ! Advance the date by one time step. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_adv1step(date) 1,1 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(inout) :: date ! number of elapsed days !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- date%stepInDay = date%stepInDay + 1 if (date%stepInDay < date%stepsPerDay) then date%s = nint((shr_date_secsPerDay*date%stepInDay)/date%stepsPerDay) else call shr_date_advNextDay(date) end if end subroutine shr_date_adv1step !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_rev1step - reverse the date by one time step ! ! !DESCRIPTION: ! Reverse the date by one time step. ! ! !REVISION HISTORY: ! 2008-Jul-01 - E. Kluzek - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_rev1step(date) 1,1 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(inout) :: date ! number of elapsed days !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- date%stepInDay = date%stepInDay - 1 if (date%stepInDay >= 0) then date%s = nint((shr_date_secsPerDay*date%stepInDay)/date%stepsPerDay) else call shr_date_revPrevDay(date) date%stepInDay = date%stepsPerDay - 1 date%s = nint((shr_date_secsPerDay*date%stepInDay)/date%stepsPerDay) end if end subroutine shr_date_rev1step !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_advNextDay - advance the date to start of next day. ! ! !DESCRIPTION: ! Advance the date to the start of next day. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_advNextDay(date) 1,2 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(inout) :: date ! number of elapsed days !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- date%eDay = date%eDay + 1 date%stepInDay = 0 date%s = 0 call shr_cal_eDay2ymd (date%eDay,date%y,date%m,date%d) call shr_cal_eDay2date(date%eDay,date%cDate) end subroutine shr_date_advNextDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_revPrevDay - reverse the date to start of previous day. ! ! !DESCRIPTION: ! Reverse the date to the start of previous day. ! ! !REVISION HISTORY: ! 2008-Jul-01 - E. Kluzek - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_revPrevDay(date) 1,2 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(inout) :: date ! number of elapsed days !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- date%eDay = date%eDay - 1 date%stepInDay = 0 date%s = 0 call shr_cal_eDay2ymd (date%eDay,date%y,date%m,date%d) call shr_cal_eDay2date(date%eDay,date%cDate) end subroutine shr_date_revPrevDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getYMD - return yy,mm,dd,ss of date ! ! !DESCRIPTION: ! return yy,mm,dd,ss of date. ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_getYMD(date,y,m,d,s) 8 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! input date integer(SHR_KIND_IN),intent(out) :: y ! year integer(SHR_KIND_IN),intent(out) :: m ! month integer(SHR_KIND_IN),intent(out) :: d ! day integer(SHR_KIND_IN),intent(out) :: s ! elapsed seconds on date !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- y = date%y m = date%m d = date%d s = date%s end subroutine shr_date_getYMD !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getCDate - return coded date of a date ! ! !DESCRIPTION: ! return coded date of a date ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_getCDATE(date,cDate,s,previous) 1,1 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! number of elapsed days integer(SHR_KIND_IN),intent(out) :: cDate ! coded date integer(SHR_KIND_IN),intent(out) :: s ! elapsed seconds on date logical, optional ,intent(in) :: previous ! flag if true return values for previous time step !EOP type(shr_date) :: ldate ! local date !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- ldate = date if ( present(previous) )then if ( previous )then call shr_date_rev1step(ldate) end if end if cDate = ldate%cDate s = ldate%s end subroutine shr_date_getCDate !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getEDay - return elapsed days of a date ! ! !DESCRIPTION: ! return elapsed days of a date ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- subroutine shr_date_getEDay(date,eDay,s) 3 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! input date integer(SHR_KIND_IN),intent(out) :: eDay ! elapsed days of date integer(SHR_KIND_IN),intent(out) :: s ! elapsed seconds on date !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- eDay = date%eDay s = date%s end subroutine shr_date_getEDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getStepsPerDay - return elapsed days of a date ! ! !DESCRIPTION: ! return elapsed days of a date ! ! !REVISION HISTORY: ! 2001-Sep-13 - B. Kauffman - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- integer function shr_date_getStepsPerDay(date) 1 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! input date !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- shr_date_getStepsPerDay = date%stepsPerDay end function shr_date_getStepsPerDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getStepInDay - return elapsed days of a date ! ! !DESCRIPTION: ! return timestep in this day ! ! !REVISION HISTORY: ! 2001-Nov-13 - T. Craig - initial version. ! ! !INTERFACE: ----------------------------------------------------------------- integer function shr_date_getStepInDay(date) 1 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! input date !EOP !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- shr_date_getStepInDay = date%StepInDay end function shr_date_getStepInDay !=============================================================================== !BOP =========================================================================== ! ! !IROUTINE: shr_date_getJulian - return julian day ! ! !DESCRIPTION: ! return julian day ! ! !REVISION HISTORY: ! 2002-Oct-28 - R. Jacob -initial version ! ! !INTERFACE: ----------------------------------------------------------------- real(SHR_KIND_R8) function shr_date_getJulian(date,shift),3 implicit none ! !INPUT/OUTPUT PARAMETERS: type(shr_date),intent(in) :: date ! input date integer(SHR_KIND_IN),intent(in),optional :: shift ! seconds to shift calculation !EOP ! local type(shr_date) :: ldate ! local date integer(SHR_KIND_IN) :: nsteps ! number of steps to advance integer(SHR_KIND_IN) :: n ! step index !------------------------------------------------------------------------------- ! !------------------------------------------------------------------------------- ldate = date if( present(shift) )then if ( mod(shift,int(shr_date_nsecsPerDay/date%stepsPerDay) ) /= 0 )then write(s_logunit,*) 'ERROR: invalid shift = ', shift, ' needs to be divisible by: ', & shr_date_nsecsPerDay/date%stepsPerDay call shr_sys_abort('(shr_date_getJulian) invalid amount to shift') end if nsteps = shift * date%stepsPerDay / shr_date_nsecsPerDay do n = 1, nsteps call shr_date_adv1step( ldate ) end do end if shr_date_getJulian = shr_cal_elapsDaysStrtMonth(ldate%y,ldate%m) & + ldate%d + ldate%s/shr_date_secsPerDay end function shr_date_getJulian !=============================================================================== !=============================================================================== end module shr_date_mod