!======================================================================= !BOP ! ! !MODULE: ice_fileunits ! ! !DESCRIPTION: ! This module contains an I/O unit manager for tracking, assigning ! and reserving I/O unit numbers. ! ! There are three reserved I/O units set as parameters in this ! module. The default units for standard input (stdin), standard ! output (stdout) and standard error (stderr). These are currently ! set as units 5,6,6, respectively as that is the most commonly ! used among vendors. However, the user may change these if those ! default units are conflicting with other models or if the ! vendor is using different values. ! ! The maximum number of I/O units per node is currently set by ! the parameter ice\_IOMaxUnits. ! ! !REVISION HISTORY: ! SVN:$Id: ice_fileunits.F90 58 2007-03-29 15:56:53Z eclare $ ! ! author: Elizabeth C. Hunke, LANL ! 2006: ECH converted to free source form (F90) ! 2007: ECH added dynamic file units, modified from POP_IOUnitsMod.F90 ! ! !INTERFACE: ! module ice_fileunits 37,2 ! ! !USES: use ice_kinds_mod #ifdef CCSMCOUPLED use shr_file_mod #endif ! !EOP !======================================================================= implicit none save character (len=char_len) :: & diag_type ! 'stdout' or 'file' integer (kind=int_kind) :: & nu_grid , & ! grid file nu_kmt , & ! land mask file nu_nml , & ! namelist input file nu_forcing , & ! forcing data file nu_dump , & ! dump file for restarting nu_restart , & ! restart input file nu_dump_aero , & ! dump file for restarting ice aerosol tracer MH nu_restart_aero,& ! restart input file for ice aerosol tracer MH nu_dump_age , & ! dump file for restarting ice age tracer nu_restart_age, & ! restart input file for ice age tracer nu_dump_FY , & ! dump file for restarting FY ice tracer nu_restart_FY , & ! restart input file for FY ice tracer nu_dump_lvl , & ! dump file for restarting level ice tracer nu_restart_lvl, & ! restart input file for level ice tracer nu_dump_pond , & ! dump file for restarting melt pond tracer nu_restart_pond,& ! restart input file for melt pond tracer nu_rst_pointer, & ! pointer to latest restart file nu_history , & ! binary history output file nu_hdr , & ! header file for binary history output nu_diag , & ! diagnostics output file nu_timing ! timing output file integer (kind=int_kind) :: & diag_level ! per-processor diagnostics level character (6), parameter :: & nml_filename = 'ice_in' ! namelist input file name integer (kind=int_kind), parameter :: & ice_stdin = 5, & ! reserved unit for standard input ice_stdout = 6, & ! reserved unit for standard output ice_stderr = 6 ! reserved unit for standard error !EOP !BOC integer (kind=int_kind), parameter :: & ice_IOUnitsMinUnits = 11, & ! do not use unit numbers below this ice_IOUnitsMaxUnits = 99 ! maximum number of open units logical (kind=log_kind), dimension(ice_IOUnitsMaxUnits) :: & ice_IOUnitsInUse ! flag=.true. if unit currently open !EOC !======================================================================= contains !======================================================================= !BOP ! !IROUTINE: init_fileunits ! !INTERFACE: subroutine init_fileunits 1,38 ! !DESCRIPTION: ! This routine grabs needed unit numbers. ! nu_diag is set to 6 (stdout) but may be reset later by the namelist. ! nu_nml is obtained separately. nu_diag = ice_stdout ! default #ifdef CCSMCOUPLED ice_IOUnitsInUse = .false. ice_IOUnitsInUse(ice_stdin) = .true. ! reserve unit 5 ice_IOUnitsInUse(ice_stdout) = .true. ! reserve unit 6 ice_IOUnitsInUse(ice_stderr) = .true. nu_grid = shr_file_getUnit() nu_kmt = shr_file_getUnit() nu_forcing = shr_file_getUnit() nu_dump = shr_file_getUnit() nu_restart = shr_file_getUnit() nu_dump_aero = shr_file_getUnit() nu_restart_aero= shr_file_getUnit() nu_dump_age = shr_file_getUnit() nu_restart_age = shr_file_getUnit() nu_dump_FY = shr_file_getUnit() nu_restart_FY = shr_file_getUnit() nu_dump_lvl = shr_file_getUnit() nu_restart_lvl = shr_file_getUnit() nu_dump_pond = shr_file_getUnit() nu_restart_pond = shr_file_getUnit() nu_rst_pointer = shr_file_getUnit() nu_history = shr_file_getUnit() nu_hdr = shr_file_getUnit() nu_timing = shr_file_getUnit() #else call get_fileunit(nu_grid) call get_fileunit(nu_kmt) call get_fileunit(nu_forcing) call get_fileunit(nu_dump) call get_fileunit(nu_restart) call get_fileunit(nu_dump_aero) call get_fileunit(nu_restart_aero) call get_fileunit(nu_dump_age) call get_fileunit(nu_restart_age) call get_fileunit(nu_dump_FY) call get_fileunit(nu_restart_FY) call get_fileunit(nu_dump_lvl) call get_fileunit(nu_restart_lvl) call get_fileunit(nu_dump_pond) call get_fileunit(nu_restart_pond) call get_fileunit(nu_rst_pointer) call get_fileunit(nu_history) call get_fileunit(nu_hdr) call get_fileunit(nu_timing) #endif end subroutine init_fileunits !======================================================================= !BOP ! !IROUTINE: get_fileunit ! !INTERFACE: subroutine get_fileunit(iunit) 25,1 ! !DESCRIPTION: ! This routine returns the next available I/O unit and marks it as ! in use to prevent any later use. ! Note that {\em all} processors must call this routine even if only ! the master task is doing the I/O. This is necessary insure that ! the units remain synchronized for other parallel I/O functions. ! ! !REVISION HISTORY: ! same as module ! !OUTPUT PARAMETERS: integer (kind=int_kind), intent(out) :: & iunit ! next free I/O unit !EOP !BOC integer (kind=int_kind) :: n ! dummy loop index logical (kind=log_kind) :: alreadyInUse #ifdef CCSMCOUPLED iunit = shr_file_getUnit() #else srch_units: do n=ice_IOUnitsMinUnits, ice_IOUnitsMaxUnits if (.not. ice_IOUnitsInUse(n)) then ! I found one, I found one !*** make sure not in use by library or calling routines INQUIRE (unit=n,OPENED=alreadyInUse) if (.not. alreadyInUse) then iunit = n ! return the free unit number ice_IOUnitsInUse(n) = .true. ! mark iunit as being in use exit srch_units else !*** if inquire shows this unit in use, mark it as !*** in use to prevent further queries ice_IOUnitsInUse(iunit) = .true. endif endif end do srch_units if (iunit > ice_IOUnitsMaxUnits) stop 'ice_IOUnitsGet: No free units' #endif !EOC end subroutine get_fileunit !======================================================================= !BOP ! !IROUTINE: release_all_fileunits ! !INTERFACE: subroutine release_all_fileunits 1,20 ! !DESCRIPTION: ! This routine releases unit numbers at the end of a run. call release_fileunit(nu_grid) call release_fileunit(nu_kmt) call release_fileunit(nu_forcing) call release_fileunit(nu_dump) call release_fileunit(nu_restart) call release_fileunit(nu_dump_aero) call release_fileunit(nu_restart_aero) call release_fileunit(nu_dump_age) call release_fileunit(nu_restart_age) call release_fileunit(nu_dump_FY) call release_fileunit(nu_restart_FY) call release_fileunit(nu_dump_lvl) call release_fileunit(nu_restart_lvl) call release_fileunit(nu_dump_pond) call release_fileunit(nu_restart_pond) call release_fileunit(nu_rst_pointer) call release_fileunit(nu_history) call release_fileunit(nu_hdr) call release_fileunit(nu_timing) if (nu_diag /= ice_stdout) call release_fileunit(nu_diag) end subroutine release_all_fileunits !======================================================================= !BOP ! !IROUTINE: release_fileunit ! !INTERFACE: subroutine release_fileunit(iunit) 25,1 ! !DESCRIPTION: ! This routine releases an I/O unit (marks it as available). ! Note that {\em all} processors must call this routine even if only ! the master task is doing the I/O. This is necessary insure that ! the units remain synchronized for other parallel I/O functions. ! ! !REVISION HISTORY: ! same as module ! !INPUT PARAMETER: integer (kind=int_kind), intent(in) :: & iunit ! I/O unit to be released !EOP !BOC #ifdef CCSMCOUPLED call shr_file_freeUnit(iunit) #else ! check for proper unit number if (iunit < 1 .or. iunit > ice_IOUnitsMaxUnits) then stop 'release_fileunit: bad unit' endif ! mark the unit as not in use ice_IOUnitsInUse(iunit) = .false. ! that was easy... #endif !EOC end subroutine release_fileunit !======================================================================= !BOP ! !IROUTINE: flush_fileunit ! !INTERFACE: subroutine flush_fileunit(iunit) 3,2 ! !DESCRIPTION: ! This routine enables a user to flush the output from an IO unit ! (typically stdout) to force output when the system is buffering ! such output. Because this system function is system dependent, ! we only support this wrapper and users are welcome to insert the ! code relevant to their local machine. In the case where the CCSM ! libraries are available, the shared routine for sys flush can be ! used (and is provided here under a preprocessor option). ! ! !REVISION HISTORY: ! same as module ! ! !USES: #ifdef CCSMCOUPLED use shr_sys_mod, only : shr_sys_flush #endif ! !INPUT PARAMETER: integer (kind=int_kind), intent(in) :: & iunit ! I/O unit to be flushed !EOP !BOC !----------------------------------------------------------------------- ! ! insert your system code here ! !----------------------------------------------------------------------- #if (defined IRIX64 || defined CRAY || defined OSF1 || defined SUNOS || defined LINUX || defined NEC_SX | defined UNICOSMP) call flush(iunit) #endif #if (defined AIX) call flush_(iunit) #endif #ifdef CCSMCOUPLED call shr_sys_flush(iunit) #endif !EOC end subroutine flush_fileunit !======================================================================= end module ice_fileunits !=======================================================================