diff --git a/.circleci/config.yml b/.circleci/config.yml index e61430e7f74a..c1d9deaf44b9 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -21,7 +21,7 @@ bcs_version: &bcs_version v11.3.0 tag_build_arg_name: &tag_build_arg_name maplversion orbs: - ci: geos-esm/circleci-tools@1 + ci: geos-esm/circleci-tools@2 workflows: build-and-test: @@ -221,7 +221,7 @@ workflows: when: equal: [ "release", << pipeline.parameters.GHA_Event >> ] jobs: - - ci/publish-docker: + - ci/publish_docker: filters: tags: only: /^v.*$/ @@ -238,7 +238,7 @@ workflows: compiler_version: 2022.1.0 image_name: geos-env tag_build_arg_name: *tag_build_arg_name - - ci/publish-docker: + - ci/publish_docker: filters: tags: only: /^v.*$/ diff --git a/.github/actions/deploy-ford-docs/action.yml b/.github/actions/deploy-ford-docs/action.yml index 0fff600e9c9e..26ca971441d3 100644 --- a/.github/actions/deploy-ford-docs/action.yml +++ b/.github/actions/deploy-ford-docs/action.yml @@ -37,6 +37,10 @@ runs: run: ford --version shell: bash + - name: cpp version + run: cpp --version + shell: bash + - name: Checkout gFTL uses: actions/checkout@v3 with: diff --git a/.github/workflows/validate_yaml_files.yml b/.github/workflows/validate_yaml_files.yml index 8dc81d4f2b54..449db6e674da 100644 --- a/.github/workflows/validate_yaml_files.yml +++ b/.github/workflows/validate_yaml_files.yml @@ -24,7 +24,7 @@ jobs: format: colored config_file: .yamllint.yml - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 if: always() with: name: yamllint-logfile diff --git a/CHANGELOG.md b/CHANGELOG.md index a84b590ac6e1..73f4ad93dc70 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,18 +17,48 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Made changes to allocate fields to use farray instead of farrayPtr. This allows explicit specification of indexflag required by the new MAPL field split functionality. This functionality allows a clean way to create a new field from an exiting field where the new field is a 'slice' of the existing field with the slicing index being that of the trailing ungiridded dim of the existing field. - Replaced RC=STATUS plus `_VERIFY(RC)` in `Base_Base_implementation.F90` with just `_RC` in line with our new convention. - Updated CI to use ESMF 8.6.0 (Baselibs 7.16.0) and updated ESMF required version to 8.6.0 +- Swath grid step 1: allow for destroying and regenerating swath grid and regenerating regridder route handle, and creating + allocatable metadata in griddedIO. Modifications are made to GriddedIO.F90, MAPL_AbstractRegridder.F90, and MAPL_EsmfRegridder.F90. +- Swath grid step 2: add control keywords for swath grid. Allow for filename template with `'*'` and DOY. Allow for missing obs files. Specify index_name_lon/lat, var_name_lon/lat/time, tunit, obs_file_begin/end/interval, Epoch and Epoch_init. - Update CI to Baselibs 7.17.0 (for future MAPL3 work) and the BCs v11.3.0 (to fix coupled run) - Update `components.yaml` - - ESMA_env v4.22.0 (Baselibs 7.15.1) + - ESMA_env v4.24.0 (Baselibs 7.17.0) +- Update CI to use circleci-tools v2 +- Made changes to allocate fields to use farray instead of farrayPtr. This allows explicit specification of indexflag required by the new MAPL field split functionality. This functionality allows a clean way to create a new field from an exiting field where the new field is a 'slice' of the existing field with the slicing index being that of the trailing ungiridded dim of the existing field. ### Fixed +- Fixed bug broken multi-step file output in History under certain template conditions - [#2433] Implemented workarounds for gfortran-13 +- Missing TARGET in GriddedIO - exposed runtime error when using NAG + debug. ### Removed ### Deprecated +## [2.42.4] - 2023-12-10 + +### Changed + +- Improved error message for missing labels in GridManager. + +### Fixed + +- Corrected some unit tests (and test utilities) to fix dangling pointers detected by NAG. Most (possibly all) of these changes are already on release/MAPL-v3, but it was getting annoying to have NAG fail unit tests with develop branch. +- Fix for CMake an Apple. Needs to set `__DARWIN` as an fpp flag. (Only used by NAG, but ...) + +## [2.42.3] - 2023-12-06 + +### Fixed + +- `MAPL_Abort()` was passing an uninitialized integer to `MPI_Abort()` resulting in spurious false successes when running ctest. Maybe was happening frequently, but CI would be blind to this. + +## [2.42.2] - 2023-12-05 + +### Fixed + +- Corrected some unit tests (and test utilities) to fix dangling pointers detected by NAG. Most (possibly all) of these changes are already on release/MAPL-v3, but it was getting annoying to have NAG fail unit tests with develop branch. + ## [2.42.1] - 2023-11-20 ### Fixed diff --git a/CMakeLists.txt b/CMakeLists.txt index 58919671b5ae..ec49ecc157da 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -8,7 +8,7 @@ endif () project ( MAPL - VERSION 2.42.1 + VERSION 2.42.4 LANGUAGES Fortran CXX C) # Note - CXX is required for ESMF # Set the possible values of build type for cmake-gui @@ -225,6 +225,9 @@ include(CheckCompilerCapabilities) if(SUPPORT_FOR_MPI_IERROR_KEYWORD) add_compile_definitions(SUPPORT_FOR_MPI_IERROR_KEYWORD) endif() +if (APPLE) + add_compile_definitions("-D__DARWIN") +endif() add_subdirectory (pfio) add_subdirectory (profiler) diff --git a/base/CMakeLists.txt b/base/CMakeLists.txt index a08dacd1250a..268d7291f6f4 100644 --- a/base/CMakeLists.txt +++ b/base/CMakeLists.txt @@ -32,7 +32,7 @@ set (srcs MAPL_IO.F90 MAPL_LatLonGridFactory.F90 MAPL_TransposeRegridder.F90 MAPL_Comms.F90 MAPL_LatLonToLatLonRegridder.F90 MAPL_TripolarGridFactory.F90 - MAPL_LlcGridFactory.F90 + MAPL_LlcGridFactory.F90 MAPL_SwathGridFactory.F90 MAPL_Config.F90 MAPL_LocStreamMod.F90 MAPL_ConservativeRegridder.F90 MAPL_MaxMinMod.F90 MAPL_VerticalInterpMod.F90 MAPL_CubedSphereGridFactory.F90 MAPL_MemUtils.F90 MAPL_VerticalMethods.F90 @@ -55,7 +55,7 @@ set (srcs MAPL_Resource.F90 MAPL_XYGridFactory.F90 MAPL_NetCDF.F90 Plain_netCDF_Time.F90 - MAPL_DateTime_Parsing_ESMF.F90 + MAPL_DateTime_Parsing_ESMF.F90 MAPL_ObsUtil.F90 # Orphaned program: should not be in this library. # tstqsat.F90 ) diff --git a/base/MAPL_AbstractGridFactory.F90 b/base/MAPL_AbstractGridFactory.F90 index 2a422d617991..fabe78cfa803 100644 --- a/base/MAPL_AbstractGridFactory.F90 +++ b/base/MAPL_AbstractGridFactory.F90 @@ -82,6 +82,11 @@ module MAPL_AbstractGridFactoryMod procedure(get_file_format_vars), deferred :: get_file_format_vars procedure(decomps_are_equal), deferred :: decomps_are_equal procedure(physical_params_are_equal), deferred :: physical_params_are_equal + + procedure :: get_xy_subset + procedure :: get_xy_mask + procedure :: destroy + procedure :: get_obs_time end type AbstractGridFactory abstract interface @@ -238,6 +243,7 @@ function generate_file_reference3D(this,fpointer,metadata) result(ref) type(FileMetadata), intent(in), optional :: metaData end function generate_file_reference3D + end interface character(len=*), parameter :: MOD_NAME = 'MAPL_AbstractGridFactory::' @@ -1030,5 +1036,49 @@ function get_grid(this, unusable, rc) result(grid) end if end function get_grid - + + + ! This procedure should only be called for time dependent grids. + ! A default implementation is to fail for other grid types, so we do not + ! have to explicitly add methods to all of the existing subclasses. + subroutine get_xy_subset(this, interval, xy_subset, rc) + class(AbstractGridFactory), intent(in) :: this + type(ESMF_Time), intent(in) :: interval(2) + integer, intent(out) :: xy_subset(2,2) + integer, optional, intent(out) :: rc + integer :: status + + _RETURN(_FAILURE) + end subroutine get_xy_subset + + subroutine get_xy_mask(this, interval, xy_mask, rc) + class(AbstractGridFactory), intent(inout) :: this + type(ESMF_Time), intent(in) :: interval(2) + integer, allocatable, intent(out) :: xy_mask(:,:) + integer, optional, intent(out) :: rc + integer :: status + + _RETURN(_FAILURE) + end subroutine get_xy_mask + + ! Probably don't need to do anything more for subclasses unless they have + ! other objects that don't finalize well. (NetCDF, ESMF, MPI, ...) + subroutine destroy(this, rc) + class(AbstractGridFactory), intent(inout) :: this + integer, optional, intent(out) :: rc + integer :: status + + call ESMF_GridDestroy(this%grid, noGarbage=.true., _RC) + _RETURN(_SUCCESS) + end subroutine destroy + + subroutine get_obs_time(this, grid, obs_time, rc) + class(AbstractGridFactory), intent(inout) :: this + type (ESMF_Grid), intent(in) :: grid + real(ESMF_KIND_R4), intent(out) :: obs_time(:,:) + integer, optional, intent(out) :: rc + + _RETURN(_SUCCESS) + end subroutine get_obs_time + end module MAPL_AbstractGridFactoryMod diff --git a/base/MAPL_AbstractRegridder.F90 b/base/MAPL_AbstractRegridder.F90 index 52aa6364a388..86086af152d3 100644 --- a/base/MAPL_AbstractRegridder.F90 +++ b/base/MAPL_AbstractRegridder.F90 @@ -8,6 +8,7 @@ module MAPL_AbstractRegridderMod use ESMF use MAPL_MemUtilsMod use MAPL_ExceptionHandling + use MAPL_RegridderSpecRouteHandleMap use, intrinsic :: iso_fortran_env, only: REAL32, REAL64 implicit none private @@ -92,6 +93,9 @@ module MAPL_AbstractRegridderMod procedure :: has_undef_value procedure :: get_regrid_method + procedure :: destroy + procedure :: destroy_route_handle + end type AbstractRegridder @@ -1006,4 +1010,21 @@ integer function get_regrid_method(this) result(method) method = this%spec%regrid_method end function get_regrid_method + + subroutine destroy(this, rc) + class(AbstractRegridder), intent(inout) :: this + integer, optional, intent(out) :: rc + integer :: status + + _RETURN(_SUCCESS) + end subroutine destroy + + subroutine destroy_route_handle(this, kind, rc) + class(AbstractRegridder), intent(inout) :: this + type(ESMF_TypeKind_Flag), intent(in) :: kind + integer, optional, intent(out) :: rc + + _RETURN(_SUCCESS) + end subroutine destroy_route_handle + end module MAPL_AbstractRegridderMod diff --git a/base/MAPL_Config.F90 b/base/MAPL_Config.F90 index 9cae1e8cdb2c..f31daf1cc93d 100644 --- a/base/MAPL_Config.F90 +++ b/base/MAPL_Config.F90 @@ -72,7 +72,6 @@ function MAPL_ConfigCreate(unusable, rc) result(config) #endif character, parameter :: NUL = achar(00) !! what it says - _UNUSED_DUMMY(unusable) config = ESMF_ConfigCreate(rc=rc) config%cptr%buffer(1:1) = EOL config%cptr%buffer(2:2) = EOB @@ -80,6 +79,8 @@ function MAPL_ConfigCreate(unusable, rc) result(config) config%cptr%next_line = 1 config%cptr%value_begin = 1 + _RETURN(_SUCCESS) + _UNUSED_DUMMY(unusable) end function MAPL_ConfigCreate !------------------------------------------------------------------------------ diff --git a/base/MAPL_EsmfRegridder.F90 b/base/MAPL_EsmfRegridder.F90 index 382ec9cc2c4f..581545b41c57 100644 --- a/base/MAPL_EsmfRegridder.F90 +++ b/base/MAPL_EsmfRegridder.F90 @@ -53,6 +53,8 @@ module MAPL_EsmfRegridderMod procedure :: do_regrid procedure :: create_route_handle procedure :: select_route_handle + procedure :: destroy + procedure :: destroy_route_handle end type EsmfRegridder @@ -1600,4 +1602,54 @@ function select_route_handle(this, kind, do_transpose, rc) result(route_handle) end function select_route_handle + subroutine destroy(this, rc) + class(EsmfRegridder), intent(inout) :: this + integer, optional, intent(out) :: rc + integer :: status + + call this%destroy_route_handle(ESMF_TYPEKIND_R4, _RC) + + _RETURN(_SUCCESS) + end subroutine destroy + + + subroutine destroy_route_handle(this, kind, rc) + class(EsmfRegridder), intent(inout) :: this + type(ESMF_TypeKind_Flag), intent(in) :: kind + integer, optional, intent(out) :: rc + + type (RegridderSpec) :: spec + type(ESMF_RouteHandle) :: dummy_rh + type(RegridderSpecRouteHandleMap), pointer :: route_handles, transpose_route_handles + type(ESMF_RouteHandle) :: route_handle + type(RegridderSpecRouteHandleMapIterator) :: iter + integer :: status + + if (kind == ESMF_TYPEKIND_R4) then + route_handles => route_handles_r4 + transpose_route_handles => transpose_route_handles_r4 + else if(kind == ESMF_TYPEKIND_R8) then + route_handles => route_handles_r8 + transpose_route_handles => transpose_route_handles_r8 + else + _FAIL('unsupported type kind (must be R4 or R8)') + end if + + spec = this%get_spec() + + _ASSERT(route_handles%count(spec) == 1, 'Did not find this spec in route handle table.') + route_handle = route_handles%at(spec) + call ESMF_RouteHandleDestroy(route_handle, noGarbage=.true.,_RC) + iter = route_handles%find(spec) + call route_handles%erase(iter) + + _ASSERT(transpose_route_handles%count(spec) == 1, 'Did not find this spec in route handle table.') + route_handle = transpose_route_handles%at(spec) + call ESMF_RouteHandleDestroy(route_handle, noGarbage=.true., _RC) + iter = transpose_route_handles%find(spec) + call transpose_route_handles%erase(iter) + + _RETURN(_SUCCESS) + end subroutine destroy_route_handle + end module MAPL_EsmfRegridderMod diff --git a/base/MAPL_GridManager.F90 b/base/MAPL_GridManager.F90 index eb2bd07b782b..c26cf5db567e 100644 --- a/base/MAPL_GridManager.F90 +++ b/base/MAPL_GridManager.F90 @@ -32,6 +32,9 @@ module MAPL_GridManager_private type (Integer64GridFactoryMap) :: factories contains procedure :: add_prototype + procedure :: destroy_grid + generic :: destroy => destroy_grid + procedure :: delete !!$ procedure :: make_field !!$ procedure :: delete_field @@ -120,6 +123,7 @@ subroutine initialize_prototypes(this, unusable, rc) use MAPL_LlcGridFactoryMod, only: LlcGridFactory use MAPL_ExternalGridFactoryMod, only: ExternalGridFactory use MAPL_XYGridFactoryMod, only: XYGridFactory + use MAPL_SwathGridFactoryMod, only : SwathGridFactory class (GridManager), intent(inout) :: this class (KeywordEnforcer), optional, intent(in) :: unusable @@ -131,7 +135,8 @@ subroutine initialize_prototypes(this, unusable, rc) type (LlcGridFactory) :: llc_factory type (ExternalGridFactory) :: external_factory type (XYGridFactory) :: xy_factory - + type (SwathGridFactory) :: swath_factory + ! This is a local variable to prevent the subroutine from running ! initialiazation twice. Calling functions have their own local variables ! to prevent calling this subroutine twice, but the initialization status @@ -147,6 +152,7 @@ subroutine initialize_prototypes(this, unusable, rc) call this%prototypes%insert('llc', llc_factory) call this%prototypes%insert('External', external_factory) call this%prototypes%insert('XY', xy_factory) + call this%prototypes%insert('Swath', swath_factory) initialized = .true. end if @@ -291,7 +297,7 @@ function make_grid_from_config(this, config, unusable, prefix, rc) result(grid) end if call ESMF_ConfigGetAttribute(config, label=label, value=grid_type, rc=status) - _ASSERT(status==0,'label not found') + _ASSERT(status==0,'label ['//label//'] not found') allocate(factory, source=this%make_factory(trim(grid_type), config, prefix=prefix, rc=status)) _VERIFY(status) @@ -397,6 +403,27 @@ function make_factory_from_distGrid(this, grid_type, dist_grid, lon_array, lat_a end function make_factory_from_distGrid + subroutine destroy_grid(this, grid, unusable, rc) + use ESMF + class (GridManager), target, intent(inout) :: this + type (ESMF_Grid), intent(inout) :: grid + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + + integer :: status + integer (kind=ESMF_KIND_I8) :: id + class(AbstractGridFactory), pointer :: factory + type(Integer64GridFactoryMapIterator) :: iter + + call ESMF_AttributeGet(grid, factory_id_attribute, id, _RC) + factory => this%factories%at(id) + call factory%destroy(_RC) + iter = this%factories%find(id) + call this%factories%erase(iter) + + _RETURN(_SUCCESS) + _UNUSED_DUMMY(unusable) + end subroutine destroy_grid ! Clients should use this procedure to release ESMF resources when a grid ! is no longer being used. @@ -413,15 +440,13 @@ subroutine delete(this, grid, unusable, rc) integer :: status character(len=*), parameter :: Iam= MOD_NAME // 'destroy_grid' - _UNUSED_DUMMY(unusable) - if (.not. this%keep_grids) then - call ESMF_GridDestroy(grid, rc=status) + call ESMF_GridDestroy(grid, noGarbage=.true., rc=status) _ASSERT(status==0,'failed to destroy grid') end if _RETURN(_SUCCESS) - + _UNUSED_DUMMY(unusable) end subroutine delete diff --git a/base/MAPL_ObsUtil.F90 b/base/MAPL_ObsUtil.F90 new file mode 100644 index 000000000000..8a797c94577e --- /dev/null +++ b/base/MAPL_ObsUtil.F90 @@ -0,0 +1,544 @@ +#include "MAPL_ErrLog.h" +#include "unused_dummy.H" + +module MAPL_ObsUtilMod + use ESMF + use Plain_netCDF_Time + use netCDF + use MAPL_CommsMod, only : MAPL_AM_I_ROOT + use, intrinsic :: iso_fortran_env, only: REAL32, REAL64 + implicit none + + interface sort_multi_arrays_by_time + module procedure sort_three_arrays_by_time + module procedure sort_four_arrays_by_time + end interface sort_multi_arrays_by_time + +contains + + subroutine get_obsfile_Tbracket_from_epoch(currTime, & + obsfile_start_time, obsfile_end_time, obsfile_interval, & + epoch_frequency, obsfile_Ts_index, obsfile_Te_index, rc) + implicit none + type(ESMF_Time), intent(in) :: currTime + type(ESMF_Time), intent(in) :: obsfile_start_time, obsfile_end_time + type(ESMF_TimeInterval), intent(in) :: obsfile_interval, epoch_frequency + integer, intent(out) :: obsfile_Ts_index + integer, intent(out) :: obsfile_Te_index + integer, optional, intent(out) :: rc + + type(ESMF_Time) :: T1, Tn + type(ESMF_Time) :: cT1 + type(ESMF_Time) :: Ts, Te + type(ESMF_TimeInterval) :: dT1, dT2, dTs, dTe + real(ESMF_KIND_R8) :: dT0_s, dT1_s, dT2_s + real(ESMF_KIND_R8) :: s1, s2 + integer :: n1, n2 + integer :: status + + T1 = obsfile_start_time + Tn = obsfile_end_time + + cT1 = currTime + dT1 = currTime - T1 + dT2 = currTime + epoch_frequency - T1 + + call ESMF_TimeIntervalGet(obsfile_interval, s_r8=dT0_s, rc=status) + call ESMF_TimeIntervalGet(dT1, s_r8=dT1_s, rc=status) + call ESMF_TimeIntervalGet(dT2, s_r8=dT2_s, rc=status) + n1 = floor (dT1_s / dT0_s) + n2 = floor (dT2_s / dT0_s) + s1 = n1 * dT0_s + s2 = n2 * dT0_s + call ESMF_TimeIntervalSet(dTs, s_r8=s1, rc=status) + call ESMF_TimeIntervalSet(dTe, s_r8=s2, rc=status) + Ts = T1 + dTs + Te = T1 + dTe + + obsfile_Ts_index = n1 + if ( dT2_s - n2*dT0_s < 1 ) then + obsfile_Te_index = n2 - 1 + else + obsfile_Te_index = n2 + end if + + _RETURN(ESMF_SUCCESS) + + end subroutine get_obsfile_Tbracket_from_epoch + + + function get_filename_from_template (time, file_template, rc) result(filename) + use Plain_netCDF_Time, only : ESMF_time_to_two_integer + use MAPL_StringTemplate, only : fill_grads_template + type(ESMF_Time), intent(in) :: time + character(len=*), intent(in) :: file_template + character(len=ESMF_MAXSTR) :: filename + integer, optional, intent(out) :: rc + + integer :: itime(2) + integer :: nymd, nhms + integer :: status + + _FAIL ('DO not use get_filename_from_template') + call ESMF_time_to_two_integer(time, itime, _RC) + print*, 'two integer time, itime(:)', itime(1:2) + nymd = itime(1) + nhms = itime(2) + call fill_grads_template ( filename, file_template, & + experiment_id='', nymd=nymd, nhms=nhms, _RC ) + print*, 'ck: obsFile_T=', trim(filename) + _RETURN(ESMF_SUCCESS) + + end function get_filename_from_template + + + + subroutine time_real_to_ESMF (times_R8_1d, times_esmf_1d, datetime_units, rc) + use MAPL_NetCDF, only : convert_NetCDF_DateTime_to_ESMF + + real(kind=REAL64), intent(in) :: times_R8_1d(:) + type(ESMF_Time), intent(inout) :: times_esmf_1d(:) + character(len=*), intent(in) :: datetime_units + integer, optional, intent(out) :: rc + + type(ESMF_TimeInterval) :: interval + type(ESMF_Time) :: time0 + type(ESMF_Time) :: time1 + character(len=:), allocatable :: tunit + + integer :: i, len + integer :: int_time + integer :: status + + len = size (times_R8_1d) + do i=1, len + int_time = times_R8_1d(i) + call convert_NetCDF_DateTime_to_ESMF(int_time, datetime_units, interval, & + time0, time=time1, time_unit=tunit, _RC) + times_esmf_1d(i) = time1 + enddo + + _RETURN(_SUCCESS) + end subroutine time_real_to_ESMF + + + subroutine reset_times_to_current_day(current_time, times_1d, rc) + type(ESMF_Time), intent(in) :: current_time + type(ESMF_Time), intent(inout) :: times_1d(:) + integer, optional, intent(out) :: rc + integer :: i,status,h,m,yp,mp,dp,s,ms,us,ns + integer :: year,month,day + + call ESMF_TimeGet(current_time,yy=year,mm=month,dd=day,_RC) + do i=1,size(times_1d) + call ESMF_TimeGet(times_1d(i),yy=yp,mm=mp,dd=dp,h=h,m=m,s=s,ms=ms,us=us,ns=ns,_RC) + call ESMF_TimeSet(times_1d(i),yy=year,mm=month,dd=day,h=h,m=m,s=s,ms=ms,us=us,ns=ns,_RC) + enddo + _RETURN(_SUCCESS) + end subroutine reset_times_to_current_day + + + + + ! --//-------------------------------------//-> + ! files + ! o o o o o o o o o o T: filename + ! <--- off set + ! o o o o o o o o o o T: file content start + ! | | + ! curr curr+Epoch + ! + + subroutine Find_M_files_for_currTime (currTime, & + obsfile_start_time, obsfile_end_time, obsfile_interval, & + epoch_frequency, file_template, M, filenames, & + T_offset_in_file_content, rc) + implicit none + type(ESMF_Time), intent(in) :: currTime + type(ESMF_Time), intent(in) :: obsfile_start_time, obsfile_end_time + type(ESMF_TimeInterval), intent(in) :: obsfile_interval, epoch_frequency + character(len=*), intent(in) :: file_template + integer, intent(out) :: M + character(len=ESMF_MAXSTR), intent(inout) :: filenames(:) + type(ESMF_TimeInterval), intent(in), optional :: T_offset_in_file_content + integer, optional, intent(out) :: rc + + type(ESMF_Time) :: T1, Tn + type(ESMF_Time) :: cT1 + type(ESMF_Time) :: Ts, Te + type(ESMF_TimeInterval) :: dT1, dT2, dTs, dTe + type(ESMF_TimeInterval) :: Toff + real(ESMF_KIND_R8) :: dT0_s, dT1_s, dT2_s + real(ESMF_KIND_R8) :: s1, s2 + character(len=ESMF_MAXSTR) :: test_file + + integer :: obsfile_Ts_index, obsfile_Te_index + integer :: n1, n2 + integer :: i, j + integer :: status + + !__ s1. Arithmetic index list based on s,e,interval + ! + if (present(T_offset_in_file_content)) then + Toff = T_offset_in_file_content + else + call ESMF_TimeIntervalSet(Toff, h=0, m=0, s=60, rc=status) + endif + + ! T1 = obsfile_start_time + Toff + ! Tn = obsfile_end_time + Toff + + T1 = obsfile_start_time + Tn = obsfile_end_time + + cT1 = currTime + dT1 = currTime - T1 + dT2 = currTime + epoch_frequency - T1 + + call ESMF_TimeIntervalGet(obsfile_interval, s_r8=dT0_s, rc=status) + call ESMF_TimeIntervalGet(dT1, s_r8=dT1_s, rc=status) + call ESMF_TimeIntervalGet(dT2, s_r8=dT2_s, rc=status) + + n1 = floor (dT1_s / dT0_s) + n2 = floor (dT2_s / dT0_s) + +! print*, 'ck dT0_s, dT1_s, dT2_s', dT0_s, dT1_s, dT2_s +! print*, '1st n1, n2', n1, n2 + + obsfile_Ts_index = n1 + if ( dT2_s - n2*dT0_s < 1 ) then + obsfile_Te_index = n2 - 1 + else + obsfile_Te_index = n2 + end if + + ! put back + n1 = obsfile_Ts_index + n2 = obsfile_Te_index + +! print*, __LINE__, __FILE__ +! print*, '2nd n1, n2', n1, n2 + + !__ s2. further test file existence + ! + j=0 + do i= n1, n2 + test_file = get_filename_from_template_use_index & + (obsfile_start_time, obsfile_interval, & + i, file_template, rc=rc) + if (test_file /= '') then + j=j+1 + filenames(j) = test_file + end if + end do + M=j + + _ASSERT ( M < size(filenames) , 'code crash, number of files exceeds upper bound') + _ASSERT (M/=0, 'M is zero, no files found for currTime') + + + _RETURN(_SUCCESS) + + end subroutine Find_M_files_for_currTime + + + subroutine read_M_files_4_swath ( filenames, Xdim, Ydim, & + index_name_lon, index_name_lat,& + var_name_lon, var_name_lat, var_name_time, & + lon, lat, time, rc ) + use pFlogger, only: logging, Logger + character(len=ESMF_MAXSTR), intent(in) :: filenames(:) + integer, intent(out) :: Xdim + integer, intent(out) :: Ydim + character(len=ESMF_MAXSTR), intent(in) :: index_name_lon + character(len=ESMF_MAXSTR), intent(in) :: index_name_lat + character(len=ESMF_MAXSTR), optional, intent(in) :: var_name_lon + character(len=ESMF_MAXSTR), optional, intent(in) :: var_name_lat + character(len=ESMF_MAXSTR), optional, intent(in) :: var_name_time + + real, optional, intent(inout) :: lon(:,:) + real, optional, intent(inout) :: lat(:,:) + !! real(ESMF_KIND_R8), optional, intent(inout) :: time_R8(:,:) + real, optional, intent(inout) :: time(:,:) + + integer, optional, intent(out) :: rc + + integer :: M + integer :: i, j, jx, status + integer :: nlon, nlat + integer :: ncid, ncid2 + character(len=ESMF_MAXSTR) :: grp1, grp2 + integer :: varid + logical :: found_group + + character(len=ESMF_MAXSTR) :: filename + integer, allocatable :: nlons(:), nlats(:) + real(ESMF_KIND_R8), allocatable :: time_loc_R8(:,:) + real(ESMF_KIND_R8), allocatable :: lon_loc(:,:) + real(ESMF_KIND_R8), allocatable :: lat_loc(:,:) + class(Logger), pointer :: lgr + + !__ s1. get Xdim Ydim + M = size(filenames) + _ASSERT(M/=0, 'M is zero, no files found') + lgr => logging%get_logger('MAPL.Sampler') + + allocate(nlons(M), nlats(M)) + jx=0 + do i = 1, M + filename = filenames(i) + CALL get_ncfile_dimension(filename, nlon=nlon, nlat=nlat, & + key_lon=index_name_lon, key_lat=index_name_lat, _RC) + nlons(i)=nlon + nlats(i)=nlat + jx=jx+nlat + + call lgr%debug('Input filename: %a', trim(filename)) + call lgr%debug('Input file : nlon, nlat= %i6 %i6', nlon, nlat) + end do + Xdim=nlon + Ydim=jx + + + !__ s2. get fields + jx=0 + do i = 1, M + filename = filenames(i) + nlon = nlons(i) + nlat = nlats(i) + + if (present(var_name_time).AND.present(time)) then + allocate (time_loc_R8(nlon, nlat)) + call get_var_from_name_w_group (var_name_time, time_loc_R8, filename, _RC) + time(1:nlon,jx+1:jx+nlat) = time_loc_R8(1:nlon,1:nlat) + deallocate(time_loc_R8) + end if + + if (present(var_name_lon).AND.present(lon)) then + allocate (lon_loc(nlon, nlat)) + call get_var_from_name_w_group (var_name_lon, lon_loc, filename, _RC) + lon(1:nlon,jx+1:jx+nlat) = lon_loc(1:nlon,1:nlat) + deallocate(lon_loc) + end if + + if (present(var_name_lat).AND.present(lat)) then + allocate (lat_loc(nlon, nlat)) + call get_var_from_name_w_group (var_name_lat, lat_loc, filename, _RC) + lat(1:nlon,jx+1:jx+nlat) = lat_loc(1:nlon,1:nlat) + deallocate(lat_loc) + end if + + jx = jx + nlat + + end do + + _RETURN(_SUCCESS) + end subroutine read_M_files_4_swath + + + ! + !-- caveat: note call this subr. on head node + ! because of (bash ls) command therein + ! + function get_filename_from_template_use_index (obsfile_start_time, obsfile_interval, & + f_index, file_template, rc) result(filename) + use Plain_netCDF_Time, only : ESMF_time_to_two_integer + use MAPL_StringTemplate, only : fill_grads_template + character(len=ESMF_MAXSTR) :: filename + type(ESMF_Time), intent(in) :: obsfile_start_time + type(ESMF_TimeInterval), intent(in) :: obsfile_interval + character(len=*), intent(in) :: file_template + integer, intent(in) :: f_index + integer, optional, intent(out) :: rc + + integer :: itime(2) + integer :: nymd, nhms + integer :: status + real(ESMF_KIND_R8) :: dT0_s + real(ESMF_KIND_R8) :: s + type(ESMF_TimeInterval) :: dT + type(ESMF_Time) :: time + integer :: i, j, u + + character(len=ESMF_MAXSTR) :: file_template_left + character(len=ESMF_MAXSTR) :: file_template_right + character(len=ESMF_MAXSTR) :: filename_left + character(len=ESMF_MAXSTR) :: filename_full + character(len=ESMF_MAXSTR) :: filename2 + character(len=ESMF_MAXSTR) :: cmd + + call ESMF_TimeIntervalGet(obsfile_interval, s_r8=dT0_s, rc=status) + s = dT0_s * f_index + call ESMF_TimeIntervalSet(dT, s_r8=s, rc=status) + time = obsfile_start_time + dT + + call ESMF_time_to_two_integer(time, itime, _RC) + nymd = itime(1) + nhms = itime(2) + + + j= index(file_template, '*') + if (j>0) then + ! wild char exist + !!print*, 'pos of * in template =', j + file_template_left = file_template(1:j-1) + call fill_grads_template ( filename_left, file_template_left, & + experiment_id='', nymd=nymd, nhms=nhms, _RC ) + filename= trim(filename_left)//trim(file_template(j:)) + cmd="bash -c 'ls "//trim(filename)//"' &> zzz_MAPL" + CALL execute_command_line(trim(cmd)) + open(newunit=u, file='zzz_MAPL', status='unknown') + read(u, '(a)') filename + i=index(trim(filename), 'ls') + if (i==1) then + filename='' + end if + ! cmd="rm -f ./zzz_MAPL" + ! CALL execute_command_line(trim(cmd)) + close(u) + else + ! exact file name + call fill_grads_template ( filename, file_template, & + experiment_id='', nymd=nymd, nhms=nhms, _RC ) + end if + + + _RETURN(_SUCCESS) + + end function get_filename_from_template_use_index + + + + subroutine get_var_from_name_w_group (var_name, var2d, filename, rc) + character(len=ESMF_MAXSTR), intent(in) :: var_name, filename + real(ESMF_KIND_R8), intent(inout) :: var2d(:,:) + integer, optional, intent(out) :: rc + + integer :: i, j + character(len=ESMF_MAXSTR) :: grp1, grp2 + character(len=ESMF_MAXSTR) :: short_name + integer :: ncid, ncid2, varid + logical :: found_group + integer :: status + + + i=index(var_name, '/') + if (i>0) then + found_group = .true. + grp1 = var_name(1:i-1) + j=index(var_name(i+1:), '/') + if (j>0) then + grp2=var_name(i+1:i+j-1) + short_name=var_name(i+j+1:) + else + grp2='' + short_name=var_name(i+1:) + endif + i=i+j + else + found_group = .false. + grp1 = '' + grp2='' + short_name=var_name + endif + + call check_nc_status(nf90_open(filename, NF90_NOWRITE, ncid2), _RC) + if ( found_group ) then + call check_nc_status(nf90_inq_ncid(ncid2, grp1, ncid), _RC) + if (j>0) then + call check_nc_status(nf90_inq_ncid(ncid, grp2, ncid2), _RC) + ncid=ncid2 + endif + else + print*, 'no grp name' + ncid=ncid2 + endif + call check_nc_status(nf90_inq_varid(ncid, short_name, varid), _RC) + call check_nc_status(nf90_get_var(ncid, varid, var2d), _RC) +!! call check_nc_status(nf90_close(ncid), _RC) + + _RETURN(_SUCCESS) + + end subroutine get_var_from_name_w_group + + + subroutine sort_three_arrays_by_time(U,V,T,rc) + use MAPL_SortMod + real(ESMF_KIND_R8), intent(inout) :: U(:), V(:), T(:) + integer, optional, intent(out) :: rc + + integer :: i, len + integer, allocatable :: IA(:) + integer(ESMF_KIND_I8), allocatable :: IX(:) + real(ESMF_KIND_R8), allocatable :: X(:) + + _ASSERT (size(U)==size(V), 'U,V different dimension') + _ASSERT (size(U)==size(T), 'U,T different dimension') + len = size (T) + + allocate (IA(len), IX(len), X(len)) + do i=1, len + IX(i)=T(i) + IA(i)=i + enddo + call MAPL_Sort(IX,IA) + + X = U + do i=1, len + U(i) = X(IA(i)) + enddo + X = V + do i=1, len + V(i) = X(IA(i)) + enddo + X = T + do i=1, len + T(i) = X(IA(i)) + enddo + _RETURN(_SUCCESS) + end subroutine sort_three_arrays_by_time + + + subroutine sort_four_arrays_by_time(U,V,T,ID,rc) + use MAPL_SortMod + real(ESMF_KIND_R8) :: U(:), V(:), T(:) + integer :: ID(:) + integer, optional, intent(out) :: rc + + integer :: i, len + integer, allocatable :: IA(:) + integer(ESMF_KIND_I8), allocatable :: IX(:) + real(ESMF_KIND_R8), allocatable :: X(:) + integer, allocatable :: NX(:) + + _ASSERT(size(U)==size(V), 'U,V different dimension') + _ASSERT(size(U)==size(T), 'U,T different dimension') + len = size (T) + + allocate (IA(len), IX(len), X(len), NX(len)) + do i=1, len + IX(i)=T(i) + IA(i)=i + enddo + call MAPL_Sort(IX,IA) + + X = U + do i=1, len + U(i) = X(IA(i)) + enddo + X = V + do i=1, len + V(i) = X(IA(i)) + enddo + X = T + do i=1, len + T(i) = X(IA(i)) + enddo + NX = ID + do i=1, len + ID(i) = NX(IA(i)) + enddo + _RETURN(_SUCCESS) + end subroutine sort_four_arrays_by_time + +end module MAPL_ObsUtilMod diff --git a/base/MAPL_SwathGridFactory.F90 b/base/MAPL_SwathGridFactory.F90 new file mode 100644 index 000000000000..591c9eb562cc --- /dev/null +++ b/base/MAPL_SwathGridFactory.F90 @@ -0,0 +1,1445 @@ +#include "MAPL_Exceptions.h" +#include "MAPL_ErrLog.h" +#include "unused_dummy.H" + +module MAPL_SwathGridFactoryMod + use MAPL_AbstractGridFactoryMod + use MAPL_MinMaxMod + use MAPL_KeywordEnforcerMod + use MAPL_ExceptionHandling + use MAPL_ShmemMod + use mapl_ErrorHandlingMod + use MAPL_Constants + use MAPL_Base, only : MAPL_GridGetInterior + use ESMF + use pFIO + use MAPL_CommsMod + !!use netcdf + !!use Plain_netCDF_Time + use MAPL_ObsUtilMod + use pflogger, only : Logger, logging + use, intrinsic :: iso_fortran_env, only: REAL32 + use, intrinsic :: iso_fortran_env, only: REAL64 + implicit none + integer, parameter :: gridLabel_max = 20 + integer, parameter :: mx_file = 300 + private + + public :: SwathGridFactory + + type, extends(AbstractGridFactory) :: SwathGridFactory + private + character(len=:), allocatable :: grid_name + character(len=:), allocatable :: grid_file_name + character(len=ESMF_MAXSTR) :: filenames(mx_file) + integer :: M_file + + integer :: cell_across_swath + integer :: cell_along_swath + integer :: im_world = MAPL_UNDEFINED_INTEGER + integer :: jm_world = MAPL_UNDEFINED_INTEGER + integer :: lm = MAPL_UNDEFINED_INTEGER + logical :: force_decomposition = .false. + + integer :: epoch ! unit: second + integer(ESMF_KIND_I8) :: epoch_index(4) ! is,ie,js,je + real(ESMF_KIND_R8), allocatable:: t_alongtrack(:) + ! note: this var is not deallocated in swathfactory, use caution + character(len=ESMF_MAXSTR) :: tunit + character(len=ESMF_MAXSTR) :: index_name_lon + character(len=ESMF_MAXSTR) :: index_name_lat + character(len=ESMF_MAXSTR) :: var_name_lon + character(len=ESMF_MAXSTR) :: var_name_lat + character(len=ESMF_MAXSTR) :: var_name_time + character(len=ESMF_MAXSTR) :: input_template + logical :: found_group + + type(ESMF_Time) :: obsfile_start_time ! user specify + type(ESMF_Time) :: obsfile_end_time + type(ESMF_TimeInterval) :: obsfile_interval + type(ESMF_TimeInterval) :: EPOCH_FREQUENCY + integer :: obsfile_Ts_index ! for epoch + integer :: obsfile_Te_index + logical :: is_valid + + ! Domain decomposition: + integer :: nx = MAPL_UNDEFINED_INTEGER + integer :: ny = MAPL_UNDEFINED_INTEGER + integer, allocatable :: ims(:) + integer, allocatable :: jms(:) + ! Used for halo + type (ESMF_DELayout) :: layout + logical :: initialized_from_metadata = .false. + contains + procedure :: make_new_grid + procedure :: create_basic_grid + procedure :: add_horz_coordinates_from_file + procedure :: init_halo + procedure :: halo + + procedure :: initialize_from_file_metadata + procedure :: initialize_from_config_with_prefix + procedure :: initialize_from_esmf_distGrid + + procedure :: equals + procedure :: check_and_fill_consistency + procedure :: generate_grid_name + procedure :: to_string + + procedure :: append_metadata + procedure :: get_grid_vars + procedure :: get_file_format_vars + procedure :: append_variable_metadata + procedure :: check_decomposition + procedure :: generate_newnxy + procedure :: generate_file_bounds + procedure :: generate_file_corner_bounds + procedure :: generate_file_reference2D + procedure :: generate_file_reference3D + procedure :: decomps_are_equal + procedure :: physical_params_are_equal + + procedure :: get_xy_subset + procedure :: destroy + procedure :: get_obs_time + end type SwathGridFactory + + character(len=*), parameter :: MOD_NAME = 'MAPL_SwathGridFactory::' + + interface SwathGridFactory + module procedure SwathGridFactory_from_parameters + end interface SwathGridFactory + + interface set_with_default + module procedure set_with_default_integer + module procedure set_with_default_real + module procedure set_with_default_real64 + module procedure set_with_default_character + module procedure set_with_default_bounds + end interface set_with_default + +contains + + function SwathGridFactory_from_parameters(unusable, grid_name, & + & im_world, jm_world, lm, nx, ny, ims, jms, rc) result(factory) + type (SwathGridFactory) :: factory + class (KeywordEnforcer), optional, intent(in) :: unusable + character(len=*), optional, intent(in) :: grid_name + + ! grid details: + integer, optional, intent(in) :: im_world + integer, optional, intent(in) :: jm_world + integer, optional, intent(in) :: lm + + ! decomposition: + integer, optional, intent(in) :: nx + integer, optional, intent(in) :: ny + integer, optional, intent(in) :: ims(:) + integer, optional, intent(in) :: jms(:) + + integer, optional, intent(out) :: rc + + integer :: status + + _UNUSED_DUMMY(unusable) + + call set_with_default(factory%grid_name, grid_name, MAPL_GRID_NAME_DEFAULT) + call set_with_default(factory%nx, nx, MAPL_UNDEFINED_INTEGER) + call set_with_default(factory%ny, ny, MAPL_UNDEFINED_INTEGER) + call set_with_default(factory%im_world, im_world, MAPL_UNDEFINED_INTEGER) + call set_with_default(factory%jm_world, jm_world, MAPL_UNDEFINED_INTEGER) + call set_with_default(factory%lm, lm, MAPL_UNDEFINED_INTEGER) + + ! default is unallocated + if (present(ims)) factory%ims = ims + if (present(jms)) factory%jms = jms + + call factory%check_and_fill_consistency(_RC) + + _RETURN(_SUCCESS) + end function SwathGridFactory_from_parameters + + + function make_new_grid(this, unusable, rc) result(grid) + type (ESMF_Grid) :: grid + class (SwathGridFactory), intent(in) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + integer :: status + + _UNUSED_DUMMY(unusable) + grid = this%create_basic_grid(_RC) + call this%add_horz_coordinates_from_file(grid,_RC) + _RETURN(_SUCCESS) + end function make_new_grid + + + function create_basic_grid(this, unusable, rc) result(grid) + type (ESMF_Grid) :: grid + class (SwathGridFactory), intent(in) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + integer :: status + + _UNUSED_DUMMY(unusable) + + grid = ESMF_GridCreateNoPeriDim( & + & name = this%grid_name, & + & countsPerDEDim1=this%ims, & + & countsPerDEDim2=this%jms, & + & indexFlag=ESMF_INDEX_DELOCAL, & + & coordDep1=[1,2], & + & coordDep2=[1,2], & + & coordSys=ESMF_COORDSYS_SPH_RAD, & + & _RC) + + ! Allocate coords at default stagger location + call ESMF_GridAddCoord(grid, _RC) + + if (this%lm /= MAPL_UNDEFINED_INTEGER) then + call ESMF_AttributeSet(grid, name='GRID_LM', value=this%lm, _RC) + end if + call ESMF_AttributeSet(grid, 'GridType', 'LatLon', _RC) + call ESMF_AttributeSet(grid, 'Global', .false., _RC) + + _RETURN(_SUCCESS) + end function create_basic_grid + + + subroutine add_horz_coordinates_from_file(this, grid, unusable, rc) + use MAPL_BaseMod, only: MAPL_grid_interior + implicit none + class (SwathGridFactory), intent(in) :: this + type (ESMF_Grid), intent(inout) :: grid + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + integer :: status + + real(kind=ESMF_KIND_R8), pointer :: fptr(:,:) + real, pointer :: centers(:,:) + real, allocatable :: centers_full(:,:) + + integer :: i, j, k + integer :: Xdim, Ydim + integer :: Xdim_full, Ydim_full + integer :: nx, ny + + integer :: IM, JM + integer :: IM_WORLD, JM_WORLD + integer :: COUNTS(3), DIMS(3) + integer :: i_1, i_n, j_1, j_n ! regional array bounds + type(Logger), pointer :: lgr + + _UNUSED_DUMMY(unusable) + + + Xdim=this%im_world + Ydim=this%jm_world + Xdim_full=this%cell_across_swath + Ydim_full=this%cell_along_swath + + call MAPL_grid_interior(grid, i_1, i_n, j_1, j_n) + call MAPL_AllocateShared(centers,[Xdim,Ydim],transroot=.true.,_RC) + call MAPL_SyncSharedMemory(_RC) + + +! if (mapl_am_I_root()) then +! write(6,'(2x,a,10i8)') & +! 'ck: Xdim, Ydim, Xdim_full, Ydim_full', Xdim, Ydim, Xdim_full, Ydim_full +! write(6,'(2x,a,10i8)') & +! 'ck: i_1, i_n, j_1, j_n', i_1, i_n, j_1, j_n +! end if + + + ! read longitudes + if (MAPL_AmNodeRoot .or. (.not. MAPL_ShmInitialized)) then + allocate( centers_full(Xdim_full, Ydim_full)) + call read_M_files_4_swath (this%filenames(1:this%M_file), nx, ny, & + this%index_name_lon, this%index_name_lat, & + var_name_lon=this%var_name_lon, lon=centers_full, _RC) + k=0 + do j=this%epoch_index(3), this%epoch_index(4) + k=k+1 + centers(1:Xdim, k) = centers_full(1:Xdim, j) + enddo + centers=centers*MAPL_DEGREES_TO_RADIANS_R8 + deallocate (centers_full) + end if + call MAPL_SyncSharedMemory(_RC) + call ESMF_GridGetCoord(grid, coordDim=1, localDE=0, & + staggerloc=ESMF_STAGGERLOC_CENTER, farrayPtr=fptr, _RC) + fptr=real(centers(i_1:i_n,j_1:j_n), kind=ESMF_KIND_R8) + + + ! read latitudes + if (MAPL_AmNodeRoot .or. (.not. MAPL_ShmInitialized)) then + allocate( centers_full(Xdim_full, Ydim_full)) + call read_M_files_4_swath (this%filenames(1:this%M_file), nx, ny, & + this%index_name_lon, this%index_name_lat, & + var_name_lat=this%var_name_lat, lat=centers_full, _RC) + k=0 + do j=this%epoch_index(3), this%epoch_index(4) + k=k+1 + centers(1:Xdim, k) = centers_full(1:Xdim, j) + enddo + centers=centers*MAPL_DEGREES_TO_RADIANS_R8 + deallocate (centers_full) + end if + call MAPL_SyncSharedMemory(_RC) + call ESMF_GridGetCoord(grid, coordDim=2, localDE=0, & + staggerloc=ESMF_STAGGERLOC_CENTER, & + farrayPtr=fptr, rc=status) + fptr=real(centers(i_1:i_n,j_1:j_n), kind=ESMF_KIND_R8) + + if(MAPL_ShmInitialized) then + call MAPL_DeAllocNodeArray(centers,_RC) + else + deallocate(centers) + end if + + _RETURN(_SUCCESS) + end subroutine add_horz_coordinates_from_file + + + subroutine initialize_from_file_metadata(this, file_metadata, unusable, force_file_coordinates, rc) + use MAPL_KeywordEnforcerMod + use MAPL_BaseMod, only: MAPL_DecomposeDim + + class (SwathGridFactory), intent(inout) :: this + type (FileMetadata), target, intent(in) :: file_metadata + class (KeywordEnforcer), optional, intent(in) :: unusable + logical, optional, intent(in) :: force_file_coordinates + integer, optional, intent(out) :: rc + + integer :: status + + class (CoordinateVariable), pointer :: v + class (*), pointer :: ptr(:) + + character(:), allocatable :: lon_name + character(:), allocatable :: lat_name + character(:), allocatable :: lev_name + integer :: i + logical :: hasLon, hasLat, hasLongitude, hasLatitude, hasLev,hasLevel,regLat,regLon + real(kind=REAL64) :: del12,delij + + integer :: i_min, i_max + real(kind=REAL64) :: d_lat, d_lat_temp, extrap_lat + logical :: is_valid, use_file_coords, compute_lons, compute_lats + + _UNUSED_DUMMY(unusable) + + if (present(force_file_coordinates)) then + use_file_coords = force_file_Coordinates + else + use_file_coords = .false. + end if + + ! Cannot assume that lats and lons are evenly spaced + + associate (im => this%im_world, jm => this%jm_world, lm => this%lm) + lon_name = 'lon' + hasLon = file_metadata%has_dimension(lon_name) + if (hasLon) then + im = file_metadata%get_dimension(lon_name, _RC) + else + lon_name = 'longitude' + hasLongitude = file_metadata%has_dimension(lon_name) + if (hasLongitude) then + im = file_metadata%get_dimension(lon_name, _RC) + else + _FAIL('no longitude coordinate') + end if + end if + lat_name = 'lat' + hasLat = file_metadata%has_dimension(lat_name) + if (hasLat) then + jm = file_metadata%get_dimension(lat_name, _RC) + else + lat_name = 'latitude' + hasLatitude = file_metadata%has_dimension(lat_name) + if (hasLatitude) then + jm = file_metadata%get_dimension(lat_name, _RC) + else + _FAIL('no latitude coordinate') + end if + end if + hasLev=.false. + hasLevel=.false. + lev_name = 'lev' + hasLev = file_metadata%has_dimension(lev_name) + if (hasLev) then + lm = file_metadata%get_dimension(lev_name,_RC) + else + lev_name = 'levels' + hasLevel = file_metadata%has_dimension(lev_name) + if (hasLevel) then + lm = file_metadata%get_dimension(lev_name,_RC) + end if + end if + end associate + + call this%make_arbitrary_decomposition(this%nx, this%ny, _RC) + + ! Determine IMS and JMS with constraint for ESMF that each DE has at least an extent + ! of 2. Required for ESMF_FieldRegrid(). + allocate(this%ims(0:this%nx-1)) + allocate(this%jms(0:this%ny-1)) + call MAPL_DecomposeDim(this%im_world, this%ims, this%nx, min_DE_extent=2) + call MAPL_DecomposeDim(this%jm_world, this%jms, this%ny, min_DE_extent=2) + + call this%check_and_fill_consistency(_RC) + + _RETURN(_SUCCESS) + + end subroutine initialize_from_file_metadata + + + subroutine initialize_from_config_with_prefix(this, config, prefix, unusable, rc) + use MPI + implicit none + class (SwathGridFactory), intent(inout) :: this + type (ESMF_Config), intent(inout) :: config + character(len=*), intent(in) :: prefix + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + integer :: status + + type(ESMF_VM) :: VM + integer:: mpic + integer:: irank, ierror + integer :: nlon, nlat, tdim + integer :: Xdim, Ydim, ntime + integer :: nx, ny + character(len=ESMF_MAXSTR) :: key_lon, key_lat, key_time + character(len=ESMF_MAXSTR) :: tunit, grp1, grp2 + character(len=ESMF_MAXSTR) :: filename, STR1, tmp + character(len=ESMF_MAXSTR) :: symd, shms + + + ! real(ESMF_KIND_R8), allocatable :: scanTime(:,:) + real, allocatable :: scanTime(:,:) + integer :: yy, mm, dd, h, m, s, sec, second + integer :: i, j, L + integer :: ncid, ncid2, varid + integer :: fid_s, fid_e + integer :: M_file + + type(ESMF_Time) :: currTime + integer (ESMF_KIND_I8) :: j0, j1, jt, jt1, jt2 + real(ESMF_KIND_R8) :: jx0, jx1 + real(ESMF_KIND_R8) :: x0, x1 + integer :: khi, klo, k, nstart, max_iter + type(Logger), pointer :: lgr + logical :: ispresent + + type(ESMF_TimeInterval) :: Toff + + _UNUSED_DUMMY(unusable) + lgr => logging%get_logger('HISTORY.sampler') + + call ESMF_VmGetCurrent(VM, _RC) + + ! input : config + ! output: this%epoch_index, nx, ny + ! + ! Read in specs, crop epoch_index based on scanTime + ! + + + !__ s1. read in file spec. + ! + call ESMF_ConfigGetAttribute(config, tmp, label=prefix//'GRIDNAME:', default=MAPL_GRID_NAME_DEFAULT) + this%grid_name = trim(tmp) + call ESMF_ConfigGetAttribute(config, this%nx, label=prefix//'NX:', default=MAPL_UNDEFINED_INTEGER) + call ESMF_ConfigGetAttribute(config, this%ny, label=prefix//'NY:', default=MAPL_UNDEFINED_INTEGER) + call ESMF_ConfigGetAttribute(config, this%lm, label=prefix//'LM:', default=MAPL_UNDEFINED_INTEGER) + call ESMF_ConfigGetAttribute(config, this%input_template, label=prefix//'GRID_FILE:', default='unknown.txt', _RC) + call ESMF_ConfigGetAttribute(config, this%epoch, label=prefix//'Epoch:', default=300, _RC) + call ESMF_ConfigGetAttribute(config, tmp, label=prefix//'Epoch_init:', default='2006', _RC) + + + call ESMF_ConfigGetAttribute(config, value=STR1, default="", & + label= prefix// 'obs_file_begin:', _RC) + + if (trim(STR1)=='') then + _FAIL('obs_file_begin missing, code crash') + else + call ESMF_TimeSet(this%obsfile_start_time, timestring=STR1, _RC) + end if + + call ESMF_ConfigGetAttribute(config, value=STR1, default="", & + label=prefix // 'obs_file_end:', _RC) + + if (trim(STR1)=='') then + _FAIL('obs_file_end missing, code crash') + else + call ESMF_TimeSet(this%obsfile_end_time, timestring=STR1, _RC) + end if + + call ESMF_ConfigGetAttribute(config, value=STR1, default="", & + label= prefix// 'obs_file_interval:', _RC) + _ASSERT(STR1/='', 'fatal error: obs_file_interval not provided in RC file') + + +! if (mapl_am_I_root()) then +! write(6,'(//2x, a)') 'SWATH initialize_from_config_with_prefix' +! print*, 'obs_file_begin: str1=', trim(STR1) +! write(6,105) 'obs_file_begin provided: ', trim(STR1) +! print*, 'obs_file_end: str1=', trim(STR1) +! write(6,105) 'obs_file_end provided:', trim(STR1) +! write(6,105) 'obs_file_interval:', trim(STR1) +! write(6,106) 'Epoch (hhmmss) :', this%epoch +! end if + + + i= index( trim(STR1), ' ' ) + if (i>0) then + symd=STR1(1:i-1) + shms=STR1(i+1:) + else + symd='' + shms=trim(STR1) + endif + call convert_twostring_2_esmfinterval (symd, shms, this%obsfile_interval, _RC) + + second = hms_2_s(this%Epoch) + call ESMF_TimeIntervalSet(this%epoch_frequency, s=second, _RC) + + if ( index(tmp, 'T') /= 0 .OR. index(tmp, '-') /= 0 ) then + call ESMF_TimeSet(currTime, timeString=tmp, _RC) + else + read(tmp,'(i4,5i2)') yy,mm,dd,h,m,s + call ESMF_Timeset(currTime, yy=yy, mm=mm, dd=dd, h=h, m=m, s=s, _RC) + endif + + call lgr%debug(' %a %a', 'input_template =', trim(this%input_template)) + !!write(6,'(2x,a,/,4i8,/,5(2x,a))') 'nx,ny,lm,epoch -- filename,tmp', & + !! this%nx,this%ny,this%lm,this%epoch,& + !! trim(filename),trim(tmp) + !!print*, 'ck: Epoch_init:', trim(tmp) + + + call ESMF_ConfigGetAttribute(config, value=this%index_name_lon, default="", & + label=prefix // 'index_name_lon:', _RC) + call ESMF_ConfigGetAttribute(config, value=this%index_name_lat, default="", & + label=prefix // 'index_name_lat:', _RC) + call ESMF_ConfigGetAttribute(config, this%var_name_lon, & + label=prefix // 'var_name_lon:', default="", _RC) + call ESMF_ConfigGetAttribute(config, this%var_name_lat, & + label=prefix // 'var_name_lat:', default="", _RC) + call ESMF_ConfigGetAttribute(config, this%var_name_time, default="", & + label=prefix//'var_name_time:', _RC) + call ESMF_ConfigGetAttribute(config, this%tunit, default="", & + label=prefix//'tunit:', _RC) + + + + !__ s2. find obsFile even if missing on disk and get array: this%t_alongtrack(:) + ! + call ESMF_VMGet(vm, mpiCommunicator=mpic, _RC) + call MPI_COMM_RANK(mpic, irank, ierror) + + if (irank==0) & + write(6,'(10(2x,a20,2x,a40,/))') & + 'index_name_lon:', trim(this%index_name_lon), & + 'index_name_lat:', trim(this%index_name_lat), & + 'var_name_lon:', trim(this%var_name_lon), & + 'var_name_lat:', trim(this%var_name_lat), & + 'var_name_time:', trim(this%var_name_time), & + 'tunit:', trim(this%tunit) + + if (irank==0) then + call ESMF_TimeIntervalSet(Toff, h=0, m=0, s=0, _RC) + call Find_M_files_for_currTime (currTime, & + this%obsfile_start_time, this%obsfile_end_time, this%obsfile_interval, & + this%epoch_frequency, this%input_template, M_file, this%filenames, & + T_offset_in_file_content = Toff, _RC) + this%M_file = M_file + write(6,'(10(2x,a20,2x,i40))') & + 'M_file:', M_file + do i=1, M_file + write(6,'(10(2x,a20,2x,a))') & + 'filenames(i):', trim(this%filenames(i)) + end do + + call read_M_files_4_swath (this%filenames(1:M_file), nx, ny, & + this%index_name_lon, this%index_name_lat, _RC) + nlon=nx + nlat=ny + allocate(scanTime(nlon, nlat)) + allocate(this%t_alongtrack(nlat)) + + call read_M_files_4_swath (this%filenames(1:M_file), nx, ny, & + this%index_name_lon, this%index_name_lat, & + var_name_time=this%var_name_time, time=scanTime, _RC) + + + do j=1, nlat + this%t_alongtrack(j)= scanTime(1,j) + enddo + nstart = 1 + ! + ! redefine nstart to skip un-defined time value + ! If the t_alongtrack contains undefined values, use this code + ! + x0 = this%t_alongtrack(1) + x1 = 1.d16 + if (x0 > x1) then + ! + ! bisect backward finding the first index arr[n] < x1 + klo=1 + khi=nlat + max_iter = int( log( real(nlat) ) / log(2.d0) ) + 2 + do i=1, max_iter + k = (klo+khi)/2 + if ( this%t_alongtrack(k) < x1 ) then + khi=k + else + nstart = khi + exit + endif + enddo + call lgr%debug('%a %i4', 'nstart', nstart) + call lgr%debug('%a %i4', 'this%t_alongtrack(nstart)', this%t_alongtrack(nstart)) + endif + + this%cell_across_swath = nlon + this%cell_along_swath = nlat + deallocate(scanTime) +!! write(6,*) 'this%t_alongtrack(j)=', this%t_alongtrack(::100) + + + ! P2. + ! determine im_world from Epoch + ! ----------------------------- + ! t_axis = t_alongtrack = t_a + ! convert currTime to j0 + ! use Epoch to find j1 + ! search j0, j1 in t_a + + call time_esmf_2_nc_int (currTime, this%tunit, j0, _RC) + sec = hms_2_s (this%Epoch) + j1= j0 + sec + jx0= j0 + jx1= j1 + call lgr%debug ('%a %i16 %i16', 'j0, j1 ', j0, j1) + + + this%epoch_index(1)= 1 + this%epoch_index(2)= this%cell_across_swath + call bisect( this%t_alongtrack, jx0, jt1, n_LB=int(nstart, ESMF_KIND_I8), n_UB=int(this%cell_along_swath, ESMF_KIND_I8), rc=rc) + call bisect( this%t_alongtrack, jx1, jt2, n_LB=int(nstart, ESMF_KIND_I8), n_UB=int(this%cell_along_swath, ESMF_KIND_I8), rc=rc) + + + if (jt1==jt2) then + _FAIL('Epoch Time is too small, empty swath grid is generated, increase Epoch') + endif + jt1 = jt1 + 1 ! (x1,x2] design + this%epoch_index(3)= jt1 + this%epoch_index(4)= jt2 + Xdim = this%cell_across_swath + Ydim = this%epoch_index(4) - this%epoch_index(3) + 1 + + call lgr%debug ('%a %i4 %i4', 'bisect for j0: rc, jt', rc, jt1) + call lgr%debug ('%a %i4 %i4', 'bisect for j1: rc, jt', rc, jt2) + call lgr%debug ('%a %i4 %i4', 'Xdim, Ydim', Xdim, Ydim) + call lgr%debug ('%a %i4 %i4 %i4 %i4', 'this%epoch_index(4)', & + this%epoch_index(1), this%epoch_index(2), & + this%epoch_index(3), this%epoch_index(4)) + + this%im_world = Xdim + this%jm_world = Ydim + end if + + call MPI_bcast(this%M_file, 1, MPI_INTEGER, 0, mpic, ierror) + do i=1, this%M_file + call MPI_bcast(this%filenames(i), ESMF_MAXSTR, MPI_CHARACTER, 0, mpic, ierror) + end do + call MPI_bcast(this%epoch_index, 4, MPI_INTEGER8, 0, mpic, ierror) + call MPI_bcast(this%im_world, 1, MPI_INTEGER, 0, mpic, ierror) + call MPI_bcast(this%jm_world, 1, MPI_INTEGER, 0, mpic, ierror) + call MPI_bcast(this%cell_across_swath, 1, MPI_INTEGER, 0, mpic, ierror) + call MPI_bcast(this%cell_along_swath, 1, MPI_INTEGER, 0, mpic, ierror) + ! donot need to bcast this%along_track (root only) + + call ESMF_ConfigGetAttribute(config, tmp, label=prefix//'IMS_FILE:', rc=status) + if ( status == _SUCCESS ) then + call get_ims_from_file(this%ims, trim(tmp),this%nx, _RC) + else + call get_multi_integer(this%ims, 'IMS:', _RC) + endif + call ESMF_ConfigGetAttribute(config, tmp, label=prefix//'JMS_FILE:', rc=status) + if ( status == _SUCCESS ) then + call get_ims_from_file(this%jms, trim(tmp),this%ny, _RC) + else + call get_multi_integer(this%jms, 'JMS:', _RC) + endif + ! ims is set at here + call this%check_and_fill_consistency(_RC) + + + _RETURN(_SUCCESS) + +105 format (1x,a,2x,a) +106 format (1x,a,2x,10i8) + + contains + + subroutine get_multi_integer(values, label, rc) + integer, allocatable, intent(out) :: values(:) + character(len=*) :: label + integer, optional, intent(out) :: rc + + integer :: i + integer :: n + integer :: tmp + integer :: status + logical :: isPresent + + call ESMF_ConfigFindLabel(config, label=prefix//label, isPresent=isPresent, _RC) + + if (.not. isPresent) then + _RETURN(_SUCCESS) + end if + + ! First pass: count values + n = 0 + do + call ESMF_ConfigGetAttribute(config, tmp, rc=status) + if (status /= _SUCCESS) then + exit + else + n = n + 1 + end if + end do + + + ! Second pass: allocate and fill + allocate(values(n), stat=status) ! no point in checking status + _VERIFY(status) + call ESMF_ConfigFindLabel(config, label=prefix//label,_RC) + do i = 1, n + call ESMF_ConfigGetAttribute(config, values(i), _RC) + write(6,*) 'values(i)=', values(i) + end do + + _RETURN(_SUCCESS) + + end subroutine get_multi_integer + + subroutine get_ims_from_file(values, file_name, n, rc) + integer, allocatable, intent(out) :: values(:) + character(len=*), intent(in) :: file_name + integer, intent(in) :: n + integer, optional, intent(out) :: rc + + logical :: FileExists + integer :: i, total, unit + integer :: status + + inquire(FILE = trim(file_name), EXIST=FileExists) + allocate(values(n), stat=status) ! no point in checking status + _VERIFY(status) + + _ASSERT(FileExists, "File <"//trim(file_name)//"> not found") + if (MAPL_AM_I_Root(VM)) then + open(newunit=UNIT, file=trim(file_name), form="formatted", iostat=status ) + _VERIFY(STATUS) + read(UNIT,*) total + _ASSERT(total == n, trim(file_name) // " n is different from total") + do i = 1,total + read(UNIT,*) values(i) + enddo + close(UNIT) + endif + + call MAPL_CommsBcast(VM, values, n=N, ROOT=MAPL_Root, _RC) + _RETURN(_SUCCESS) + + end subroutine get_ims_from_file + + subroutine get_range(range, label, rc) + type(RealMinMax), intent(out) :: range + character(len=*) :: label + integer, optional, intent(out) :: rc + + integer :: i + integer :: n + integer :: status + logical :: isPresent + + call ESMF_ConfigFindLabel(config, label=prefix//label,isPresent=isPresent,_RC) + if (.not. isPresent) then + _RETURN(_SUCCESS) + end if + + ! Must be 2 values: min and max + call ESMF_ConfigGetAttribute(config, range%min, _RC) + call ESMF_ConfigGetAttribute(config, range%max, _RC) + + _RETURN(_SUCCESS) + + end subroutine get_range + + + end subroutine initialize_from_config_with_prefix + + + + function to_string(this) result(string) + character(len=:), allocatable :: string + class (SwathGridFactory), intent(in) :: this + + _UNUSED_DUMMY(this) + string = 'SwathGridFactory' + + end function to_string + + + subroutine check_and_fill_consistency(this, unusable, rc) + use MAPL_BaseMod, only: MAPL_DecomposeDim + class (SwathGridFactory), intent(inout) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + + integer :: status + logical :: verify_decomp + + _UNUSED_DUMMY(unusable) + + if (.not. allocated(this%grid_name)) then + this%grid_name = MAPL_GRID_NAME_DEFAULT + end if + + ! Check decomposition/bounds + ! WY notes: should not have this assert + !_ASSERT(allocated(this%ims) .eqv. allocated(this%jms), 'inconsistent options') + call verify(this%nx, this%im_world, this%ims, rc=status) + call verify(this%ny, this%jm_world, this%jms, rc=status) + + if (.not.this%force_decomposition) then + verify_decomp = this%check_decomposition(_RC) + if ( (.not.verify_decomp) ) then + call this%generate_newnxy(_RC) + end if + end if + + _RETURN(_SUCCESS) + + contains + + subroutine verify(n, m_world, ms, rc) + integer, intent(inout) :: n + integer, intent(inout) :: m_world + integer, allocatable, intent(inout) :: ms(:) + integer, optional, intent(out) :: rc + + integer :: status + + if (allocated(ms)) then + _ASSERT(size(ms) > 0, 'degenerate topology') + + if (n == MAPL_UNDEFINED_INTEGER) then + n = size(ms) + else + _ASSERT(n == size(ms), 'inconsistent topology') + end if + + if (m_world == MAPL_UNDEFINED_INTEGER) then + m_world = sum(ms) + else + _ASSERT(m_world == sum(ms), 'inconsistent decomponsition') + end if + + else + + _ASSERT(n /= MAPL_UNDEFINED_INTEGER, 'uninitialized topology') + _ASSERT(m_world /= MAPL_UNDEFINED_INTEGER,'uninitialized dimension') + allocate(ms(n), stat=status) + _VERIFY(status) + !call MAPL_DecomposeDim(m_world, ms, n, min_DE_extent=2) + call MAPL_DecomposeDim(m_world, ms, n) + + end if + + _RETURN(_SUCCESS) + + end subroutine verify + + end subroutine check_and_fill_consistency + + + elemental subroutine set_with_default_integer(to, from, default) + integer, intent(out) :: to + integer, optional, intent(in) :: from + integer, intent(in) :: default + + if (present(from)) then + to = from + else + to = default + end if + + end subroutine set_with_default_integer + + elemental subroutine set_with_default_real64(to, from, default) + real(REAL64), intent(out) :: to + real(REAL64), optional, intent(in) :: from + real(REAL64), intent(in) :: default + + if (present(from)) then + to = from + else + to = default + end if + + end subroutine set_with_default_real64 + + elemental subroutine set_with_default_real(to, from, default) + real, intent(out) :: to + real, optional, intent(in) :: from + real, intent(in) :: default + + if (present(from)) then + to = from + else + to = default + end if + + end subroutine set_with_default_real + + subroutine set_with_default_character(to, from, default) + character(len=:), allocatable, intent(out) :: to + character(len=*), optional, intent(in) :: from + character(len=*), intent(in) :: default + + if (present(from)) then + to = from + else + to = default + end if + + end subroutine set_with_default_character + + + elemental subroutine set_with_default_bounds(to, from, default) + type (RealMinMax), intent(out) :: to + type (RealMinMax), optional, intent(in) :: from + type (RealMinMax), intent(in) :: default + + if (present(from)) then + to = from + else + to = default + end if + + end subroutine set_with_default_bounds + + + ! MAPL uses values in lon_array and lat_array only to determine the + ! general positioning. Actual coordinates are then recomputed. + ! This helps to avoid roundoff differences from slightly different + ! input files. + subroutine initialize_from_esmf_distGrid(this, dist_grid, lon_array, lat_array, unusable, rc) + use MAPL_ConfigMod + use MAPL_Constants, only: PI => MAPL_PI_R8 + class (SwathGridFactory), intent(inout) :: this + type (ESMF_DistGrid), intent(in) :: dist_grid + type (ESMF_LocalArray), intent(in) :: lon_array + type (ESMF_LocalArray), intent(in) :: lat_array + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + + integer :: dim_count, tile_count + integer, allocatable :: max_index(:,:) + integer :: status + character(len=2) :: pole ,dateline + + type (ESMF_Config) :: config + type (ESMF_VM) :: vm + integer :: nPet + real(kind=REAL32), pointer :: lon(:) + real(kind=REAL32), pointer :: lat(:) + integer :: nx_guess,nx,ny + integer :: i + + real, parameter :: tiny = 1.e-4 + + _FAIL ('stop: not implemented: subroutine initialize_from_esmf_distGrid') + + _UNUSED_DUMMY(unusable) + + call ESMF_DistGridGet(dist_grid, dimCount=dim_count, tileCount=tile_count) + allocate(max_index(dim_count, tile_count)) + call ESMF_DistGridGet(dist_grid, maxindexPTile=max_index) + + config = MAPL_ConfigCreate(_RC) + call MAPL_ConfigSetAttribute(config, max_index(1,1), 'IM_WORLD:', _RC) + call MAPL_ConfigSetAttribute(config, max_index(2,1), 'JM_WORLD:', _RC) + call MAPL_ConfigSetAttribute(config, max_index(3,1), 'LM:', _RC) + + lon => null() + lat => null() + call ESMF_LocalArrayGet(lon_array, farrayPtr=lon, _RC) + call ESMF_LocalArrayGet(lat_array, farrayPtr=lat, _RC) + + call ESMF_VMGetCurrent(vm, _RC) + call ESMF_VMGet(vm, PETcount=nPet, _RC) + + nx_guess = nint(sqrt(real(nPet))) + do nx = nx_guess,1,-1 + ny=nPet/nx + if (nx*ny==nPet) then + call MAPL_ConfigSetAttribute(config, nx, 'NX:') + call MAPL_ConfigSetAttribute(config, ny, 'NY:') + exit + end if + enddo + + call this%initialize(config, _RC) + + end subroutine initialize_from_esmf_distGrid + + + function decomps_are_equal(this,a) result(equal) + class (SwathGridFactory), intent(in) :: this + class (AbstractGridFactory), intent(in) :: a + logical :: equal + + select type (a) + class default + equal = .false. + return + class is (SwathGridFactory) + equal = .true. + + + equal = size(a%ims)==size(this%ims) .and. size(a%jms)==size(this%jms) + if (.not. equal) return + + ! same decomposition + equal = all(a%ims == this%ims) .and. all(a%jms == this%jms) + if (.not. equal) return + + end select + + end function decomps_are_equal + + + function physical_params_are_equal(this, a) result(equal) + class (SwathGridFactory), intent(in) :: this + class (AbstractGridFactory), intent(in) :: a + logical :: equal + + select type (a) + class default + equal = .false. + return + class is (SwathGridFactory) + equal = .true. + + equal = (a%im_world == this%im_world) .and. (a%jm_world == this%jm_world) + if (.not. equal) return + end select + + end function physical_params_are_equal + + logical function equals(a, b) + class (SwathGridFactory), intent(in) :: a + class (AbstractGridFactory), intent(in) :: b + + select type (b) + class default + equals = .false. + return + class is (SwathGridFactory) + equals = .true. + + equals = (a%lm == b%lm) + if (.not. equals) return + + equals = a%decomps_are_equal(b) + if (.not. equals) return + + equals = a%physical_params_are_equal(b) + if (.not. equals) return + + end select + + end function equals + + + function generate_grid_name(this) result(name) + character(len=:), allocatable :: name + class (SwathGridFactory), intent(in) :: this +! from tclune: This needs thought. I suspect we want something that indicates this is a swath grid. + character(len=4) :: im_string, jm_string + name = im_string // 'x' // jm_string + end function generate_grid_name + + + function check_decomposition(this,unusable,rc) result(can_decomp) + class (SwathGridFactory), target, intent(inout) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + logical :: can_decomp + integer :: n + _UNUSED_DUMMY(unusable) + + can_decomp = .true. + if (this%im_world==1 .and. this%jm_world==1) then + _RETURN(_SUCCESS) + end if + n = this%im_world/this%nx + if (n < 2) can_decomp = .false. + n = this%jm_world/this%ny + if (n < 2) can_decomp = .false. + _RETURN(_SUCCESS) + end function check_decomposition + + + subroutine generate_newnxy(this,unusable,rc) + use MAPL_BaseMod, only: MAPL_DecomposeDim + class (SwathGridFactory), target, intent(inout) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + integer :: n + + _UNUSED_DUMMY(unusable) + + n = this%im_world/this%nx + if (n < 2) then + this%nx = generate_new_decomp(this%im_world,this%nx) + deallocate(this%ims) + allocate(this%ims(0:this%nx-1)) + call MAPL_DecomposeDim(this%im_world, this%ims, this%nx) + end if + n = this%jm_world/this%ny + if (n < 2) then + this%ny = generate_new_decomp(this%jm_world,this%ny) + deallocate(this%jms) + allocate(this%jms(0:this%ny-1)) + call MAPL_DecomposeDim(this%jm_world, this%jms, this%ny) + end if + + _RETURN(_SUCCESS) + + end subroutine generate_newnxy + + function generate_new_decomp(im,nd) result(n) + integer, intent(in) :: im, nd + integer :: n + logical :: canNotDecomp + + canNotDecomp = .true. + n = nd + do while(canNotDecomp) + if ( (im/n) < 2) then + n = n/2 + else + canNotDecomp = .false. + end if + enddo + end function generate_new_decomp + + subroutine init_halo(this, unusable, rc) + class (SwathGridFactory), target, intent(inout) :: this + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc + _FAIL('Stop: subroutine init_halo is not needed for SwathGridFactory') + end subroutine init_halo + + subroutine halo(this, array, unusable, halo_width, rc) + use MAPL_CommsMod + class (SwathGridFactory), intent(inout) :: this + real(kind=REAL32), intent(inout) :: array(:,:) + class (KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(in) :: halo_width + integer, optional, intent(out) :: rc + _FAIL( 'Stop: subroutine halo is not needed for SwathGridFactory') + end subroutine halo + + + subroutine append_metadata(this, metadata) + use MAPL_Constants + class (SwathGridFactory), intent(inout) :: this + type (FileMetadata), intent(inout) :: metadata + + type (Variable) :: v + real(kind=REAL64), allocatable :: temp_coords(:) + + character(len=ESMF_MAXSTR) :: key_lon + character(len=ESMF_MAXSTR) :: key_lat + + ! Horizontal grid dimensions + call metadata%add_dimension('lon', this%im_world) + call metadata%add_dimension('lat', this%jm_world) + + ! Coordinate variables + v = Variable(type=PFIO_REAL64, dimensions='lon,lat') + call v%add_attribute('long_name', 'longitude') + call v%add_attribute('units', 'degrees_east') + call metadata%add_variable('lons', v) + + v = Variable(type=PFIO_REAL64, dimensions='lon,lat') + call v%add_attribute('long_name', 'latitude') + call v%add_attribute('units', 'degrees_north') + call metadata%add_variable('lats', v) + + end subroutine append_metadata + + + function get_grid_vars(this) result(vars) + class (SwathGridFactory), intent(inout) :: this + + character(len=:), allocatable :: vars + character(len=ESMF_MAXSTR) :: key_lon + character(len=ESMF_MAXSTR) :: key_lat + _UNUSED_DUMMY(this) + + !!key_lon=trim(this%var_name_lon) + !!key_lat=trim(this%var_name_lat) + vars = 'lon,lat' + + end function get_grid_vars + + + function get_file_format_vars(this) result(vars) + class (SwathGridFactory), intent(inout) :: this + + character(len=:), allocatable :: vars + _UNUSED_DUMMY(this) + + vars = 'lon,lat' + end function get_file_format_vars + + + subroutine append_variable_metadata(this,var) + class (SwathGridFactory), intent(inout) :: this + type(Variable), intent(inout) :: var + _UNUSED_DUMMY(this) + _UNUSED_DUMMY(var) + end subroutine append_variable_metadata + + + subroutine generate_file_bounds(this,grid,local_start,global_start,global_count,metadata,rc) + use MAPL_BaseMod + class(SwathGridFactory), intent(inout) :: this + type(ESMF_Grid), intent(inout) :: grid + integer, allocatable, intent(out) :: local_start(:) + integer, allocatable, intent(out) :: global_start(:) + integer, allocatable, intent(out) :: global_count(:) + type(FileMetaData), intent(in), optional :: metaData + integer, optional, intent(out) :: rc + + integer :: status + integer :: global_dim(3), i1,j1,in,jn + + _UNUSED_DUMMY(this) + + call MAPL_GridGet(grid,globalCellCountPerDim=global_dim,_RC) + call MAPL_GridGetInterior(grid,i1,in,j1,jn) + allocate(local_start,source=[i1,j1]) + allocate(global_start,source=[1,1]) + allocate(global_count,source=[global_dim(1),global_dim(2)]) + + _RETURN(_SUCCESS) + + end subroutine generate_file_bounds + + + subroutine generate_file_corner_bounds(this,grid,local_start,global_start,global_count,rc) + use esmf + class (SwathGridFactory), intent(inout) :: this + type(ESMF_Grid), intent(inout) :: grid + integer, allocatable, intent(out) :: local_start(:) + integer, allocatable, intent(out) :: global_start(:) + integer, allocatable, intent(out) :: global_count(:) + integer, optional, intent(out) :: rc + + _UNUSED_DUMMY(this) + _UNUSED_DUMMY(grid) + _UNUSED_DUMMY(local_start) + _UNUSED_DUMMY(global_start) + _UNUSED_DUMMY(global_count) + + _FAIL('unimplemented') + _RETURN(_SUCCESS) + end subroutine generate_file_corner_bounds + + function generate_file_reference2D(this,fpointer) result(ref) + use pFIO + type(ArrayReference) :: ref + class(SwathGridFactory), intent(inout) :: this + real, pointer, intent(in) :: fpointer(:,:) + _UNUSED_DUMMY(this) + ref = ArrayReference(fpointer) + end function generate_file_reference2D + + function generate_file_reference3D(this,fpointer,metaData) result(ref) + use pFIO + type(ArrayReference) :: ref + class(SwathGridFactory), intent(inout) :: this + real, pointer, intent(in) :: fpointer(:,:,:) + type(FileMetaData), intent(in), optional :: metaData + _UNUSED_DUMMY(this) + ref = ArrayReference(fpointer) + end function generate_file_reference3D + + + subroutine get_xy_subset(this, interval, xy_subset, rc) + use MPI + class(SwathGridFactory), intent(in) :: this + type(ESMF_Time), intent(in) :: interval(2) + integer, intent(out) :: xy_subset(2,2) + integer, optional, intent(out) :: rc + + type(ESMF_VM) :: VM + integer:: mpic + integer:: irank, ierror + + integer :: status + type(ESMF_Time) :: T1, T2 + integer(ESMF_KIND_I8) :: i1, i2 + real(ESMF_KIND_R8) :: iT1, iT2 + integer(ESMF_KIND_I8) :: index1, index2 + integer :: jlo, jhi, je + + + call ESMF_VmGetCurrent(VM, _RC) + call ESMF_VMGet(vm, mpiCommunicator=mpic, _RC) + call MPI_COMM_RANK(mpic, irank, ierror) + + if (irank==0) then + ! xtrack + xy_subset(1:2,1)=this%epoch_index(1:2) + + ! atrack + T1= interval(1) + T2= interval(2) + + ! this%t_alongtrack + ! + call time_esmf_2_nc_int (T1, this%tunit, i1, _RC) + call time_esmf_2_nc_int (T2, this%tunit, i2, _RC) + iT1 = i1 ! int to real*8 + iT2 = i2 + if (this%epoch_index(3) > 2) then + jlo = this%epoch_index(3) - 2 + else + jlo = this%epoch_index(3) + end if + jhi = this%epoch_index(4) + 1 + ! + ! -- it is possible some obs files are missing + ! + call bisect( this%t_alongtrack, iT1, index1, n_LB=int(jlo, ESMF_KIND_I8), n_UB=int(jhi, ESMF_KIND_I8), rc=rc) + call bisect( this%t_alongtrack, iT2, index2, n_LB=int(jlo, ESMF_KIND_I8), n_UB=int(jhi, ESMF_KIND_I8), rc=rc) + + !! complex version + !! ! (x1, x2] design in bisect + !! if (index1==jlo-1) then + !! je = index1 + 1 + !! else + !! je = index1 + !! end if + !! xy_subset(1, 2) = je + !! if (index2==jlo-1) then + !! je = index2 + 1 + !! else + !! je = index2 + !! end if + !! xy_subset(2, 2) = je + + ! simple version + xy_subset(1, 2)=index1+1 ! atrack + xy_subset(2, 2)=index2 + + ! + !- relative + ! + xy_subset(1,2)= xy_subset(1,2) - this%epoch_index(3) + 1 + xy_subset(2,2)= xy_subset(2,2) - this%epoch_index(3) + 1 + end if + + call MPI_bcast(xy_subset, 4, MPI_INTEGER, 0, mpic, ierror) + + _RETURN(_SUCCESS) + end subroutine get_xy_subset + + + subroutine destroy(this, rc) + class(SwathGridFactory), intent(inout) :: this + integer, optional, intent(out) :: rc + integer :: i + return + end subroutine destroy + + + ! here grid == external_grid + ! because this%grid is protected in AbstractGridFactory + subroutine get_obs_time(this, grid, obs_time, rc) + use MAPL_BaseMod, only: MAPL_grid_interior + class(SwathGridFactory), intent(inout) :: this + type (ESMF_Grid), intent(in) :: grid + real(ESMF_KIND_R4), intent(out) :: obs_time(:,:) + integer, optional, intent(out) :: rc + integer :: status + + integer :: i_1, i_n, j_1, j_n ! regional array bounds + + !! shared mem + real(kind=ESMF_KIND_R8), pointer :: fptr(:,:) + real, pointer :: centers(:,:) + real, allocatable :: centers_full(:,:) + + integer :: i, j, k + integer :: Xdim, Ydim + integer :: Xdim_full, Ydim_full + integer :: nx, ny + integer :: IM_WORLD, JM_WORLD + + + !- shared mem case in MPI + ! + Xdim=this%im_world + Ydim=this%jm_world + Xdim_full=this%cell_across_swath + Ydim_full=this%cell_along_swath + + call MAPL_grid_interior(grid, i_1, i_n, j_1, j_n) + call MAPL_AllocateShared(centers,[Xdim,Ydim],transroot=.true.,_RC) + call MAPL_SyncSharedMemory(_RC) + + + ! read Time and set + if (MAPL_AmNodeRoot .or. (.not. MAPL_ShmInitialized)) then + allocate( centers_full(Xdim_full, Ydim_full)) + call read_M_files_4_swath (this%filenames(1:this%M_file), nx, ny, & + this%index_name_lon, this%index_name_lat, & + var_name_time=this%var_name_time, time=centers_full, _RC) + !!call get_v2d_netcdf(this%grid_file_name, time_name, centers_full, Xdim_full, Ydim_full) + k=0 + do j=this%epoch_index(3), this%epoch_index(4) + k=k+1 + centers(1:Xdim, k) = centers_full(1:Xdim, j) + enddo + deallocate (centers_full) + end if + call MAPL_SyncSharedMemory(_RC) + + !(Xdim, Ydim) + obs_time = centers(i_1:i_n,j_1:j_n) + + if(MAPL_ShmInitialized) then + call MAPL_DeAllocNodeArray(centers,_RC) + else + deallocate(centers) + end if + + _RETURN(_SUCCESS) + end subroutine get_obs_time + + +end module MAPL_SwathGridFactoryMod diff --git a/base/Plain_netCDF_Time.F90 b/base/Plain_netCDF_Time.F90 index 85ff1507b407..215cfdeeb31e 100644 --- a/base/Plain_netCDF_Time.F90 +++ b/base/Plain_netCDF_Time.F90 @@ -87,7 +87,6 @@ subroutine get_ncfile_dimension(filename, nlon, nlat, tdim, key_lon, key_lat, ke lat_name=trim(key_lat) call check_nc_status(nf90_inq_dimid(ncid, trim(lat_name), dimid), _RC) call check_nc_status(nf90_inquire_dimension(ncid, dimid, len=nlat), _RC) - call check_nc_status(nf90_close(ncid), _RC) endif if(present(key_time)) then @@ -446,10 +445,10 @@ subroutine bisect_find_LB_R8_I8(xa, x, n, n_LB, n_UB, rc) integer :: i, nmax LB=1; UB=size(xa,1) - if(present(n_LB)) LB=n_LB - if(present(n_UB)) UB=n_UB + if(present(n_LB)) LB=max(LB, n_LB) + if(present(n_UB)) UB=min(UB, n_UB) klo=LB; khi=UB; dk=1 - + if ( xa(LB ) > xa(UB) ) then klo= UB khi= LB diff --git a/base/StringTemplate.F90 b/base/StringTemplate.F90 index 1b13af15edfa..c3efbdeecece 100644 --- a/base/StringTemplate.F90 +++ b/base/StringTemplate.F90 @@ -5,13 +5,14 @@ module MAPL_StringTemplate use MAPL_ExceptionHandling use MAPL_KeywordEnforcerMod + implicit none private public fill_grads_template public StrTemplate -character(len=2), parameter :: valid_tokens(14) = ["y4","y2","m1","m2","mc","Mc","MC","d1","d2","h1","h2","h3","n2","S2"] +character(len=2), parameter :: valid_tokens(15) = ["y4","y2","m1","m2","mc","Mc","MC","d1","d2","h1","h2","h3","n2","S2","D3"] character(len=3),parameter :: mon_lc(12) = [& 'jan','feb','mar','apr','may','jun', & 'jul','aug','sep','oct','nov','dec'] @@ -165,6 +166,10 @@ function evaluate_token(token,year,month,day,hour,minute,second,preserve) result logical, intent(in) :: preserve character(len=4) :: buffer character(len=1) :: c1,c2 + type(ESMF_Time) :: time + integer(ESMF_KIND_I4) :: doy + integer :: status, rc + type(ESMF_Calendar) :: gregorianCalendar c1=token(1:1) c2=token(2:2) select case(c1) @@ -208,6 +213,22 @@ function evaluate_token(token,year,month,day,hour,minute,second,preserve) result else buffer="%"//token end if + case("D") ! dayOfYear + if (.not.skip_token(day,preserve)) then + if (c2 == "3") then + gregorianCalendar = ESMF_CalendarCreate(ESMF_CALKIND_GREGORIAN, & + name='Gregorian_obs' , rc=rc) + call ESMF_TimeSet(time, yy=year, mm=month, dd=day, & + calendar=gregorianCalendar, _RC) + call ESMF_TimeGet(time, dayOfYear=doy, _RC) + call ESMF_CalendarDestroy(gregorianCalendar) + write(buffer,'(i3.3)')doy + else + _FAIL('Day of Year must be %D3') + end if + else + buffer="%"//token + end if case("h") if (.not.skip_token(hour,preserve)) then if (c2 == "3") then diff --git a/base/tests/Test_GridManager.pf b/base/tests/Test_GridManager.pf index 9e7e6f17fcb9..540437006763 100644 --- a/base/tests/Test_GridManager.pf +++ b/base/tests/Test_GridManager.pf @@ -52,7 +52,7 @@ contains if (status == 0) then call grid_manager%delete(grid) end if - @assertExceptionRaised('label not found') + @assertExceptionRaised('label [GRID_TYPE:] not found') ! Check that it actually failed @assertFalse(0 == status, 'made a grid even though there is no prototype') diff --git a/base/tests/Test_SimpleMAPLcomp.pf b/base/tests/Test_SimpleMAPLcomp.pf index f82c70158b71..6e0081f65644 100644 --- a/base/tests/Test_SimpleMAPLcomp.pf +++ b/base/tests/Test_SimpleMAPLcomp.pf @@ -7,7 +7,7 @@ module Test_SimpleMAPLcomp contains - @test(npes=[1,2,0],type=newESMF_TestMethod) + @test(npes=[1,2,0],type=ESMF_TestMethod) subroutine test_one(this) class (ESMF_TestMethod), intent(inout) :: this diff --git a/base/tests/Test_SphericalToCartesian.pf b/base/tests/Test_SphericalToCartesian.pf index d0e5bc11af5d..077577fb92b6 100644 --- a/base/tests/Test_SphericalToCartesian.pf +++ b/base/tests/Test_SphericalToCartesian.pf @@ -15,7 +15,7 @@ module Test_SphericalToCartesian contains - @test(npes=[1],type=newESMF_TestMethod) + @test(npes=[1],type=ESMF_TestMethod) subroutine test_spherical_to_cartesian_east_wind(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory @@ -55,7 +55,7 @@ contains end subroutine test_spherical_to_cartesian_east_wind - @test(npes=[1],type=newESMF_TestMethod) + @test(npes=[1],type=ESMF_TestMethod) subroutine test_spherical_to_cartesian_north_wind(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory @@ -93,7 +93,7 @@ contains end subroutine test_spherical_to_cartesian_north_wind - @test(npes=[1],type=newESMF_TestMethod) + @test(npes=[1],type=ESMF_TestMethod) subroutine test_cartesian_to_spherical_X(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory @@ -132,7 +132,7 @@ contains end subroutine test_cartesian_to_spherical_X - @test(npes=[1],type=newESMF_TestMethod) + @test(npes=[1],type=ESMF_TestMethod) subroutine test_cartesian_to_spherical_Y(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory @@ -172,7 +172,7 @@ contains end subroutine test_cartesian_to_spherical_Y - @test(npes=[1],type=newESMF_TestMethod) + @test(npes=[1],type=ESMF_TestMethod) subroutine test_cartesian_to_spherical_Z(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory @@ -215,7 +215,7 @@ contains ! No good place to put this test, so putting it here for now. ! Testing a static method on abstract class (AbstractGridFactory) - @test(npes=[1,2,3,4,6],type=newESMF_TestMethod) + @test(npes=[1,2,3,4,6],type=ESMF_TestMethod) subroutine test_make_arbitrary_decomposition(this) class (ESMF_TestMethod), intent(inout) :: this type (LatLonGridFactory) :: factory diff --git a/base/tests/mapl_bundleio_test.F90 b/base/tests/mapl_bundleio_test.F90 index be606ce717a5..ad9981c858c6 100644 --- a/base/tests/mapl_bundleio_test.F90 +++ b/base/tests/mapl_bundleio_test.F90 @@ -150,7 +150,7 @@ function create_cf(grid_name,im_world,jm_world,nx,ny,lm,cs_stretch_param,rc) res call MAPL_ConfigSetAttribute(cf,value=dateline, label=trim(grid_name)//".DATELINE:",_RC) end if - + _RETURN(_SUCCESS) end function create_cf function create_gridname(im,jm,date,pole) result(gridname) @@ -260,7 +260,7 @@ program ut_ReGridding call UnpackGridName(Gridname,im_world_new,jm_world_new,dateline_new,pole_new) lm_world=3 - cfoutput = create_cf(gridname,im_world_new,jm_world_new,nx,ny,lm_world,cs_stretch_param,_RC) + cfoutput = create_cf(trim(gridname),im_world_new,jm_world_new,nx,ny,lm_world,cs_stretch_param,_RC) grid_new=grid_manager%make_grid(cfoutput,prefix=trim(gridname)//".",_RC) bundle=ESMF_FieldBundleCreate(name="cfio_bundle",_RC) call ESMF_FieldBundleSet(bundle,grid=grid_new,_RC) diff --git a/components.yaml b/components.yaml index e1bf5d321ef8..e0bbd84af99c 100644 --- a/components.yaml +++ b/components.yaml @@ -5,7 +5,7 @@ MAPL: ESMA_env: local: ./ESMA_env remote: ../ESMA_env.git - tag: v4.22.0 + tag: v4.24.0 develop: main ESMA_cmake: diff --git a/docs/Ford/README.md b/docs/Ford/README.md new file mode 100644 index 000000000000..d32e83e13f3a --- /dev/null +++ b/docs/Ford/README.md @@ -0,0 +1,33 @@ +# Ford Documentation + +These are control files for the [Ford documentation generator](https://github.com/Fortran-FOSS-Programmers/ford). MAPL currently has +two: + +1. `docs-with-remote-esmf.md` - This is the main documentation file. It + includes the documentation for MAPL and ESMF. It is used to generate the + [MAPL documentation](https://geos-esm.github.io/MAPL/). +2. `docs-with-remote-esmf.public_private_protected.md` - This generates the "developer" version of the MAPL documentation. It includes + the documentation for MAPL and ESMF, but also includes the private and + protected members of the MAPL classes. It is used to generate the + [developer version of the MAPL documentation](https://geos-esm.github.io/MAPL/dev-doc/). + +## Issue with `pcpp` + +Note that currently MAPL does not work with `pcpp` which is the default +preprocessor for Ford 7. Instead, we must use `cpp` with the `-traditional-cpp` +flag. This is done by setting the `preprocessor:` in the Ford markdown files. + +```markdown +preprocessor: cpp -traditional-cpp -E +``` + +### cpp on macOS + +Note that on macOS, `cpp` is by default clang's preprocessor. To use the GNU +preprocessor, you must install it with Homebrew and then use the full path to +the executable, e.g., + +```markdown +preprocessor: /opt/homebrew/bin/cpp-13 -traditional-cpp -E +``` + diff --git a/docs/Ford/docs-with-remote-esmf.md b/docs/Ford/docs-with-remote-esmf.md index 2bbba56ba936..641f0c6ed2be 100644 --- a/docs/Ford/docs-with-remote-esmf.md +++ b/docs/Ford/docs-with-remote-esmf.md @@ -1,4 +1,5 @@ --- +preprocessor: cpp -traditional-cpp -E src_dir: ../../ search: true graph: true @@ -6,14 +7,14 @@ coloured_edges: true graph_maxdepth: 4 graph_maxnodes: 32 include: ../../include/ - ../../gFTL/install/GFTL-1.10/include/v1 - ../../gFTL/install/GFTL-1.10/include/v2 -exclude: EsmfRegridder.F90 - FieldBLAS_IntrinsicFunctions.F90 - GeomManager.F90 - MaplGeom.F90 - Regridder.F90 - StateSupplement.F90 + ../../gFTL/install/GFTL-1.11/include/v1 + ../../gFTL/install/GFTL-1.11/include/v2 +exclude: **/EsmfRegridder.F90 + **/FieldBLAS_IntrinsicFunctions.F90 + **/GeomManager.F90 + **/MaplGeom.F90 + **/Regridder.F90 + **/StateSupplement.F90 exclude_dir: ../../docs ../../Doxygen ../../ESMA_cmake diff --git a/docs/Ford/docs-with-remote-esmf.public_private_protected.md b/docs/Ford/docs-with-remote-esmf.public_private_protected.md index fa6a15138b16..24eb948ee0a1 100644 --- a/docs/Ford/docs-with-remote-esmf.public_private_protected.md +++ b/docs/Ford/docs-with-remote-esmf.public_private_protected.md @@ -1,4 +1,5 @@ --- +preprocessor: cpp -traditional-cpp -E src_dir: ../../ output_dir: dev-doc search: true @@ -7,14 +8,14 @@ coloured_edges: true graph_maxdepth: 4 graph_maxnodes: 32 include: ../../include/ - ../../gFTL/install/GFTL-1.10/include/v1 - ../../gFTL/install/GFTL-1.10/include/v2 -exclude: EsmfRegridder.F90 - FieldBLAS_IntrinsicFunctions.F90 - GeomManager.F90 - MaplGeom.F90 - Regridder.F90 - StateSupplement.F90 + ../../gFTL/install/GFTL-1.11/include/v1 + ../../gFTL/install/GFTL-1.11/include/v2 +exclude: **/EsmfRegridder.F90 + **/FieldBLAS_IntrinsicFunctions.F90 + **/GeomManager.F90 + **/MaplGeom.F90 + **/Regridder.F90 + **/StateSupplement.F90 exclude_dir: ../../docs ../../Doxygen ../../ESMA_cmake @@ -64,4 +65,4 @@ fpp_extensions: F90 externalize: true --- -{!../README.md!} +{!../../README.md!} diff --git a/gridcomps/ExtData2G/ExtDataConfig.F90 b/gridcomps/ExtData2G/ExtDataConfig.F90 index 086476a761fe..6ee6f96af98f 100644 --- a/gridcomps/ExtData2G/ExtDataConfig.F90 +++ b/gridcomps/ExtData2G/ExtDataConfig.F90 @@ -87,8 +87,8 @@ recursive subroutine new_ExtDataConfig_from_yaml(ext_config,config_file,current_ if (ESMF_HConfigIsDefined(input_config,keyString="Samplings")) then temp_configs = ESMF_HConfigCreateAt(input_config,keyString="Samplings",_RC) - hconfigIter = ESMF_HConfigIterBegin(temp_configs) hconfigIterBegin = ESMF_HConfigIterBegin(temp_configs) + hconfigIter = hconfigIterBegin hconfigIterEnd = ESMF_HConfigIterEnd(temp_configs) do while (ESMF_HConfigIterLoop(hconfigIter,hconfigIterBegin,hconfigIterEnd)) hconfig_key = ESMF_HConfigAsStringMapKey(hconfigIter,_RC) @@ -96,12 +96,13 @@ recursive subroutine new_ExtDataConfig_from_yaml(ext_config,config_file,current_ ts = ExtDataTimeSample(single_sample,_RC) call ext_config%sample_map%insert(trim(hconfig_key),ts) enddo + call ESMF_HConfigDestroy(temp_configs) end if if (ESMF_HConfigIsDefined(input_config,keyString="Collections")) then temp_configs = ESMF_HConfigCreateAt(input_config,keyString="Collections",_RC) - hconfigIter = ESMF_HConfigIterBegin(temp_configs) hconfigIterBegin = ESMF_HConfigIterBegin(temp_configs) + hconfigIter = hconfigIterBegin hconfigIterEnd = ESMF_HConfigIterEnd(temp_configs) do while (ESMF_HConfigIterLoop(hconfigIter,hconfigIterBegin,hconfigIterEnd)) hconfig_key = ESMF_HConfigAsStringMapKey(hconfigIter,_RC) @@ -111,12 +112,13 @@ recursive subroutine new_ExtDataConfig_from_yaml(ext_config,config_file,current_ ds = ExtDataFileStream(single_collection,current_time,_RC) call ext_config%file_stream_map%insert(trim(hconfig_key),ds) enddo + call ESMF_HConfigDestroy(temp_configs) end if if (ESMF_HConfigIsDefined(input_config,keyString="Exports")) then temp_configs = ESMF_HConfigCreateAt(input_config,keyString="Exports",_RC) - hconfigIter = ESMF_HConfigIterBegin(temp_configs) hconfigIterBegin = ESMF_HConfigIterBegin(temp_configs) + hconfigIter = hconfigIterBegin hconfigIterEnd = ESMF_HConfigIterEnd(temp_configs) do while (ESMF_HConfigIterLoop(hconfigIter,hconfigIterBegin,hconfigIterEnd)) hconfig_key = ESMF_HConfigAsStringMapKey(hconfigIter,_RC) @@ -140,8 +142,8 @@ recursive subroutine new_ExtDataConfig_from_yaml(ext_config,config_file,current_ if (ESMF_HConfigIsDefined(input_config,keyString="Derived")) then temp_configs = ESMF_HConfigCreateAt(input_config,keyString="Derived",_RC) - hconfigIter = ESMF_HConfigIterBegin(temp_configs) hconfigIterBegin = ESMF_HConfigIterBegin(temp_configs) + hconfigIter = hconfigIterBegin hconfigIterEnd = ESMF_HConfigIterEnd(temp_configs) do while (ESMF_HConfigIterLoop(hconfigIter,hconfigIterBegin,hconfigIterEnd)) hconfig_key = ESMF_HConfigAsStringMapKey(hconfigIter,_RC) @@ -162,7 +164,7 @@ end subroutine new_ExtDataConfig_from_yaml function count_rules_for_item(this,item_name,rc) result(number_of_rules) integer :: number_of_rules - class(ExtDataConfig), intent(in) :: this + class(ExtDataConfig), target, intent(in) :: this character(len=*), intent(in) :: item_name integer, optional, intent(out) :: rc @@ -187,7 +189,7 @@ end function count_rules_for_item function get_time_range(this,item_name,rc) result(time_range) type(ESMF_Time), allocatable :: time_range(:) - class(ExtDataConfig), intent(in) :: this + class(ExtDataConfig), target, intent(in) :: this character(len=*), intent(in) :: item_name integer, optional, intent(out) :: rc @@ -265,7 +267,7 @@ function sort_rules_by_start(hconfig_sequence,rc) result(sorted_index) end function sort_rules_by_start function get_item_type(this,item_name,unusable,rc) result(item_type) - class(ExtDataConfig), intent(inout) :: this + class(ExtDataConfig), target, intent(inout) :: this character(len=*), intent(in) :: item_name class(KeywordEnforcer), optional, intent(in) :: unusable integer, optional, intent(out) :: rc diff --git a/gridcomps/ExtData2G/ExtDataGridCompNG.F90 b/gridcomps/ExtData2G/ExtDataGridCompNG.F90 index ce335399f252..0da681786a62 100644 --- a/gridcomps/ExtData2G/ExtDataGridCompNG.F90 +++ b/gridcomps/ExtData2G/ExtDataGridCompNG.F90 @@ -256,7 +256,7 @@ SUBROUTINE Initialize_ ( GC, IMPORT, EXPORT, CLOCK, rc ) integer :: idx type(MAPL_MetaComp),pointer :: MAPLSTATE - type(ExtDataOldTypesCreator),target :: config_yaml + type(ExtDataOldTypesCreator), target :: config_yaml character(len=ESMF_MAXSTR) :: new_rc_file logical :: found_in_config integer :: num_primary,num_derived,num_rules @@ -308,7 +308,7 @@ SUBROUTINE Initialize_ ( GC, IMPORT, EXPORT, CLOCK, rc ) _RETURN(ESMF_SUCCESS) end if - config_yaml = ExtDataOldTypesCreator(new_rc_file,time,_RC) + call new_ExtDataOldTypesCreator(config_yaml, new_rc_file, time, _RC) allocate(ITEMNAMES(ITEMCOUNT), STAT=STATUS) _VERIFY(STATUS) @@ -1449,7 +1449,7 @@ end subroutine createFileLevBracket subroutine IOBundle_Add_Entry(IOBundles,item,entry_num,rc) type(IOBundleNGVector), intent(inout) :: IOBundles - type(primaryExport), intent(inout) :: item + type(primaryExport), target, intent(inout) :: item integer, intent(in) :: entry_num integer, intent(out), optional :: rc diff --git a/gridcomps/ExtData2G/ExtDataOldTypesCreator.F90 b/gridcomps/ExtData2G/ExtDataOldTypesCreator.F90 index f6e4533b2f2c..1abc5720c795 100644 --- a/gridcomps/ExtData2G/ExtDataOldTypesCreator.F90 +++ b/gridcomps/ExtData2G/ExtDataOldTypesCreator.F90 @@ -21,7 +21,9 @@ module MAPL_ExtDataOldTypesCreator use MAPL_ExtDataTimeSample use MAPL_ExtDataTimeSampleMap implicit none + public :: ExtDataOldTypesCreator + public :: new_ExtDataOldTypesCreator type, extends(ExtDataConfig) :: ExtDataOldTypesCreator private @@ -30,32 +32,31 @@ module MAPL_ExtDataOldTypesCreator procedure :: fillin_derived end type ExtDataOldTypesCreator - interface ExtDataOldTypesCreator - module procedure :: new_ExtDataOldTypesCreator - end interface +!# interface ExtDataOldTypesCreator +!# module procedure :: new_ExtDataOldTypesCreator +!# end interface contains - function new_ExtDataOldTypesCreator(config_file,current_time,unusable,rc ) result(ExtDataObj) - character(len=*), intent(in) :: config_file - type(ESMF_Time), intent(in) :: current_time - class(KeywordEnforcer), optional, intent(in) :: unusable - integer, optional, intent(out) :: rc + subroutine new_ExtDataOldTypesCreator(extdataobj, config_file,current_time,unusable,rc ) + type(ExtDataOldTypesCreator), target, intent(out) :: ExtDataObj + character(len=*), intent(in) :: config_file + type(ESMF_Time), intent(in) :: current_time + class(KeywordEnforcer), optional, intent(in) :: unusable + integer, optional, intent(out) :: rc - type(ExtDataOldTypesCreator) :: ExtDataObj - integer :: status - - _UNUSED_DUMMY(unusable) - call ExtDataObj%ExtDataConfig%new_ExtDataConfig_from_yaml(config_file,current_time,rc=status) - _VERIFY(status) - - _RETURN(_SUCCESS) - end function new_ExtDataOldTypesCreator + integer :: status + + call ExtDataObj%ExtDataConfig%new_ExtDataConfig_from_yaml(config_file,current_time,_RC) + + _RETURN(_SUCCESS) + _UNUSED_DUMMY(unusable) + end subroutine new_ExtDataOldTypesCreator subroutine fillin_primary(this,item_name,base_name,primary_item,time,clock,unusable,rc) - class(ExtDataOldTypesCreator), intent(inout) :: this + class(ExtDataOldTypesCreator), target, intent(inout) :: this character(len=*), intent(in) :: item_name character(len=*), intent(in) :: base_name type(PrimaryExport), intent(inout) :: primary_item diff --git a/gridcomps/History/CMakeLists.txt b/gridcomps/History/CMakeLists.txt index c00678537245..25ba48139cfe 100644 --- a/gridcomps/History/CMakeLists.txt +++ b/gridcomps/History/CMakeLists.txt @@ -5,6 +5,7 @@ set (srcs MAPL_HistoryTrajectoryMod_smod.F90 MAPL_HistoryCollection.F90 MAPL_HistoryGridComp.F90 + MAPL_EpochSwathMod.F90 MAPL_StationSamplerMod.F90 ) diff --git a/gridcomps/History/MAPL_EpochSwathMod.F90 b/gridcomps/History/MAPL_EpochSwathMod.F90 new file mode 100644 index 000000000000..62b94145df5f --- /dev/null +++ b/gridcomps/History/MAPL_EpochSwathMod.F90 @@ -0,0 +1,1234 @@ +! +! __ Analogy to GriddedIO.F90 with a twist for Epoch Swath grid +! +#include "MAPL_Generic.h" + +module MAPL_EpochSwathMod + use ESMF + use ESMFL_Mod + use MAPL_AbstractGridFactoryMod + use MAPL_AbstractRegridderMod + use MAPL_GridManagerMod + use MAPL_BaseMod + use MAPL_NewRegridderManager + use MAPL_RegridMethods + use MAPL_TimeDataMod + use MAPL_VerticalDataMod + use MAPL_Constants + use pFIO + use MAPL_GriddedIOItemVectorMod + use MAPL_GriddedIOItemMod + use MAPL_ExceptionHandling + use pFIO_ClientManagerMod + use MAPL_DataCollectionMod + use MAPL_DataCollectionManagerMod + use gFTL_StringVector + use gFTL_StringStringMap + use MAPL_StringGridMapMod + use MAPL_FileMetadataUtilsMod + use MAPL_DownbitMod + use Plain_netCDF_Time + use, intrinsic :: ISO_C_BINDING + use, intrinsic :: iso_fortran_env, only: REAL64 + use ieee_arithmetic, only: isnan => ieee_is_nan + implicit none + + private + + type, public :: samplerHQ + type(ESMF_Clock) :: clock + type(ESMF_Alarm) :: alarm + type(ESMF_Time) :: RingTime + type(ESMF_TimeInterval) :: Frequency_epoch + type(ESMF_config) :: config_grid_save + type(ESMF_grid) :: ogrid + character(len=ESMF_MAXSTR) :: grid_type + real*8 :: arr(2) + + contains + procedure :: create_grid + procedure :: regrid_accumulate => regrid_accumulate_on_xysubset + procedure :: destroy_rh_regen_ogrid + procedure :: fill_time_in_bundle + end type samplerHQ + + interface samplerHQ + module procedure new_samplerHQ + end interface samplerHQ + + type, public :: sampler + type(FileMetaData), allocatable :: metadata + type(fileMetadataUtils), pointer :: current_file_metadata + integer :: write_collection_id + integer :: read_collection_id + integer :: metadata_collection_id + class (AbstractRegridder), pointer :: regrid_handle => null() + type(ESMF_Grid) :: output_grid + logical :: doVertRegrid = .false. + type(ESMF_FieldBundle) :: output_bundle + type(ESMF_FieldBundle) :: input_bundle + type(ESMF_FieldBundle) :: acc_bundle + type(ESMF_Time) :: startTime + integer :: regrid_method = REGRID_METHOD_BILINEAR + integer :: nbits_to_keep = MAPL_NBITS_NOT_SET + real, allocatable :: lons(:,:),lats(:,:) + real, allocatable :: corner_lons(:,:),corner_lats(:,:) + real, allocatable :: times(:) + type(TimeData) :: timeInfo + type(VerticalData) :: vdata + type(GriddedIOitemVector) :: items + integer :: deflateLevel = 0 + integer :: quantizeAlgorithm = 1 + integer :: quantizeLevel = 0 + integer, allocatable :: chunking(:) + logical :: itemOrderAlphabetical = .true. + integer :: fraction + logical :: have_initalized + contains +!! procedure :: CreateFileMetaData + procedure :: Create_bundle_RH + procedure :: CreateVariable + procedure :: regridScalar + procedure :: regridVector + procedure :: set_param + procedure :: set_default_chunking + procedure :: check_chunking + procedure :: alphabatize_variables + procedure :: addVariable_to_acc_bundle + procedure :: addVariable_to_output_bundle + procedure :: interp_accumulate_fields + end type sampler + + interface sampler + module procedure new_sampler + end interface sampler + +contains + + function new_samplerHQ(clock, config, key, rc) result(hq) + implicit none + type(samplerHQ) :: hq + type(ESMF_Clock), intent(in) :: clock + type(ESMF_Config), intent(inout) :: config + character(len=*), intent(in) :: key + integer, optional, intent(out) :: rc + + character(len=ESMF_MAXSTR) :: time_string + integer :: status + integer :: time_integer + type(ESMF_Time) :: RingTime_epoch + type(ESMF_Time) :: startTime + type(ESMF_Time) :: currTime + type(ESMF_TimeInterval) :: timeStep + type(ESMF_TimeInterval) :: Frequency_epoch + + integer :: sec, second + integer :: n1 + type(ESMF_Config) :: cf + + + hq%clock= clock + hq%config_grid_save= config + + hq%arr(1:2) = -2.d0 + call ESMF_ClockGet ( clock, CurrTime=currTime, _RC ) + call ESMF_ClockGet ( clock, timestep=timestep, _RC ) + call ESMF_ClockGet ( clock, startTime=startTime, _RC ) + call ESMF_ConfigGetAttribute(config, value=time_integer, label=trim(key)//'.Epoch:', default=0, _RC) + _ASSERT(time_integer /= 0, 'Epoch value in config wrong') + second = hms_2_s (time_integer) + call ESMF_TimeIntervalSet(frequency_epoch, s=second, _RC) + hq%frequency_epoch = frequency_epoch + hq%RingTime = currTime + hq%alarm = ESMF_AlarmCreate( clock=clock, RingInterval=Frequency_epoch, & + RingTime=hq%RingTime, sticky=.false., _RC ) + + _RETURN(_SUCCESS) + + end function new_samplerHQ + + + !--------------------------------------------------! + ! __ set + ! - ogrid via grid_manager%make_grid + ! using currTime and HQ%config_grid_save + !--------------------------------------------------! + function create_grid(this, key, currTime, grid_type, rc) result(ogrid) + type (ESMF_Grid) :: ogrid + class(samplerHQ) :: this + character(len=*), intent(in) :: key + type(ESMF_Time), intent(inout) :: currTime + character(len=*), optional, intent(in) :: grid_type + integer, intent(out), optional :: rc + integer :: status + + type(ESMF_Config) :: config_grid + character(len=ESMF_MAXSTR) :: time_string + + logical :: ispresent + + if (present(grid_type)) this%grid_type = trim(grid_type) + config_grid = this%config_grid_save + call ESMF_TimeGet(currTime, timeString=time_string, _RC) + ! + ! -- the `ESMF_ConfigSetAttribute` shows a risk + ! to overwrite the nextline in config + ! + call ESMF_ConfigSetAttribute( config_grid, trim(time_string), label=trim(key)//'.Epoch_init:', _RC) + ogrid = grid_manager%make_grid(config_grid, prefix=trim(key)//'.', _RC ) + this%ogrid = ogrid + _RETURN(_SUCCESS) + + end function create_grid + + + subroutine regrid_accumulate_on_xysubset (this, sp, rc) + class(samplerHQ) :: this + class(sampler), intent(inout) :: sp + integer, intent(out), optional :: rc + integer :: status + + class(AbstractGridFactory), pointer :: factory + integer :: xy_subset(2,2) + type(ESMF_Time) :: timeset(2) + type(ESMF_Time) :: current_time + type(ESMF_TimeInterval) :: dur + character(len=ESMF_MAXSTR) :: time_string + + integer, allocatable :: global_xy_mask(:,:) + integer, allocatable :: local_xy_mask(:,:) + + integer :: counts(5) + integer :: dims(3) + integer :: m1, m2 + + ! __ s1. get xy_subset + + factory => grid_manager%get_factory(this%ogrid,_RC) + call ESMF_ClockGet(this%clock,currTime=current_time,_RC) + call ESMF_ClockGet(this%clock,timeStep=dur, _RC ) + timeset(1) = current_time - dur + timeset(2) = current_time + call factory%get_xy_subset( timeset, xy_subset, _RC) + + ! __ s2. interpolate then save data using xy_mask + + call sp%interp_accumulate_fields (xy_subset, _RC) + + _RETURN(ESMF_SUCCESS) + + end subroutine regrid_accumulate_on_xysubset + + + subroutine destroy_rh_regen_ogrid (this, key_grid_label, output_grids, sp, rc) + implicit none + class(samplerHQ) :: this + class(sampler) :: sp + type (StringGridMap), target, intent(inout) :: output_grids + character(len=*), intent(in) :: key_grid_label + integer, intent(out), optional :: rc + integer :: status + + class(AbstractGridFactory), pointer :: factory + type(ESMF_Time) :: currTime + type(ESMF_TimeInterval) :: dur + character(len=ESMF_MAXSTR) :: time_string + + type(ESMF_Grid), pointer :: pgrid + type(ESMF_Grid) :: ogrid + type(ESMF_Grid) :: input_grid + character(len=ESMF_MAXSTR) :: key_str + type (StringGridMapIterator) :: iter + character(len=:), pointer :: key + type (ESMF_Config) :: config_grid + + integer :: i, numVars + character(len=ESMF_MAXSTR), allocatable :: names(:) + type(ESMF_Field) :: field + + if ( .NOT. ESMF_AlarmIsRinging(this%alarm) ) then + write(6,*) 'ck: regen, not in alarming' + rc=0 + return + endif + + + !__ s1. destroy ogrid + regen ogrid + + key_str=trim(key_grid_label) + pgrid => output_grids%at(trim(key_grid_label)) + call grid_manager%destroy(pgrid,_RC) + + call ESMF_ClockGet (this%clock, CurrTime=currTime, _RC ) + iter = output_grids%begin() + do while (iter /= output_grids%end()) + key => iter%key() + if (trim(key)==trim(key_str)) then + ogrid = this%create_grid (key_str, currTime, _RC) + call output_grids%set(key, ogrid) + this%ogrid = ogrid + endif + call iter%next() + enddo + + + !__ s2. destroy RH + + call sp%regrid_handle%destroy(_RC) + + + !__ s3. destroy acc_bundle / output_bundle + + call ESMF_FieldBundleGet(sp%acc_bundle,fieldCount=numVars,_RC) + allocate(names(numVars),stat=status) + call ESMF_FieldBundleGet(sp%acc_bundle,fieldNameList=names,_RC) + do i=1,numVars + call ESMF_FieldBundleGet(sp%acc_bundle,trim(names(i)),field=field,_RC) + call ESMF_FieldDestroy(field,noGarbage=.true., _RC) + enddo + call ESMF_FieldBundleDestroy(sp%acc_bundle,noGarbage=.true.,_RC) + + call ESMF_FieldBundleGet(sp%output_bundle,fieldCount=numVars,_RC) + allocate(names(numVars),stat=status) + call ESMF_FieldBundleGet(sp%output_bundle,fieldNameList=names,_RC) + do i=1,numVars + call ESMF_FieldBundleGet(sp%output_bundle,trim(names(i)),field=field,_RC) + call ESMF_FieldDestroy(field,noGarbage=.true., _RC) + enddo + call ESMF_FieldBundleDestroy(sp%output_bundle,noGarbage=.true.,_RC) + + _RETURN(ESMF_SUCCESS) + + end subroutine destroy_rh_regen_ogrid + + + subroutine fill_time_in_bundle (this, xname, bundle, rc) + implicit none + class(samplerHQ) :: this + character(len=*), intent(in) :: xname + type(ESMF_FieldBundle), intent(inout) :: bundle + integer, optional, intent(out) :: rc + integer :: status + + class(AbstractGridFactory), pointer :: factory + type(ESMF_Field) :: field + real(kind=ESMF_KIND_R4), pointer :: ptr2d(:,:) + + ! __ get field xname='time' + call ESMF_FieldBundleGet (bundle, xname, field=field, _RC) + call ESMF_FieldGet (field, farrayptr=ptr2d, _RC) + + ! __ obs_time from swath factory + factory => grid_manager%get_factory(this%ogrid,_RC) + call factory%get_obs_time (this%ogrid, ptr2d, _RC) + + _RETURN(ESMF_SUCCESS) + + end subroutine fill_time_in_bundle + + + function new_sampler(metadata,input_bundle,output_bundle,write_collection_id,read_collection_id, & + metadata_collection_id,regrid_method,fraction,items,rc) result(GriddedIO) + type(sampler) :: GriddedIO + type(Filemetadata), intent(in), optional :: metadata + type(ESMF_FieldBundle), intent(in), optional :: input_bundle + type(ESMF_FieldBundle), intent(in), optional :: output_bundle + integer, intent(in), optional :: write_collection_id + integer, intent(in), optional :: read_collection_id + integer, intent(in), optional :: metadata_collection_id + integer, intent(in), optional :: regrid_method + integer, intent(in), optional :: fraction + type(GriddedIOitemVector), intent(in), optional :: items + integer, intent(out), optional :: rc + + if (present(metadata)) GriddedIO%metadata=metadata + if (present(input_bundle)) GriddedIO%input_bundle=input_bundle + if (present(output_bundle)) GriddedIO%output_bundle=output_bundle + if (present(regrid_method)) GriddedIO%regrid_method=regrid_method + if (present(write_collection_id)) GriddedIO%write_collection_id=write_collection_id + if (present(read_collection_id)) GriddedIO%read_collection_id=read_collection_id + if (present(metadata_collection_id)) GriddedIO%metadata_collection_id=metadata_collection_id + if (present(items)) GriddedIO%items=items + if (present(fraction)) GriddedIO%fraction=fraction + _RETURN(ESMF_SUCCESS) + end function new_sampler + + + subroutine Create_bundle_RH(this,items,bundle,timeInfo,vdata,ogrid,global_attributes,rc) + class (sampler), intent(inout) :: this + type(GriddedIOitemVector), target, intent(inout) :: items + type(ESMF_FieldBundle), intent(inout) :: bundle + type(TimeData), optional, intent(inout) :: timeInfo + type(VerticalData), intent(inout), optional :: vdata + type (ESMF_Grid), intent(inout), pointer, optional :: ogrid + type(StringStringMap), target, intent(in), optional :: global_attributes + integer, intent(out), optional :: rc + + type(ESMF_Grid) :: input_grid + class (AbstractGridFactory), pointer :: factory + + type(ESMF_Field) :: new_field + type(GriddedIOitemVectorIterator) :: iter + type(GriddedIOitem), pointer :: item + type(stringVector) :: order + integer :: metadataVarsSize + type(StringStringMapIterator) :: s_iter + character(len=:), pointer :: attr_name, attr_val + integer :: status + + this%items = items + this%input_bundle = bundle + this%output_bundle = ESMF_FieldBundleCreate(rc=status) + _VERIFY(status) + if(present(timeInfo)) this%timeInfo = timeInfo + call ESMF_FieldBundleGet(this%input_bundle,grid=input_grid,rc=status) + _VERIFY(status) + if (present(ogrid)) then + this%output_grid=ogrid + else + call ESMF_FieldBundleGet(this%input_bundle,grid=this%output_grid,rc=status) + _VERIFY(status) + end if + this%regrid_handle => new_regridder_manager%make_regridder(input_grid,this%output_grid,this%regrid_method,rc=status) + _VERIFY(status) + + ! We get the regrid_method here because in the case of Identity, we set it to + ! REGRID_METHOD_IDENTITY in the regridder constructor if identity. Now we need + ! to change the regrid_method in the GriddedIO object to be the same as the + ! the regridder object. + this%regrid_method = this%regrid_handle%get_regrid_method() + + call ESMF_FieldBundleSet(this%output_bundle,grid=this%output_grid,rc=status) + _VERIFY(status) + factory => get_factory(this%output_grid,rc=status) + _VERIFY(status) + + ! __ please note, metadata in this section is not used in put_var to netCDF + ! the design used mGriddedIO%metadata in MAPL_HistoryGridComp.F90 + ! In other words, factory%append_metadata appeared here and in GriddedIO.F90 + ! + if (allocated(this%metadata)) then + deallocate (this%metadata) + end if + allocate(this%metadata) + call factory%append_metadata(this%metadata) + if (present(vdata)) then + this%vdata=vdata + else + this%vdata=VerticalData(rc=status) + _VERIFY(status) + end if + + call this%vdata%append_vertical_metadata(this%metadata,this%input_bundle,rc=status) + _VERIFY(status) + this%doVertRegrid = (this%vdata%regrid_type /= VERTICAL_METHOD_NONE) + if (this%vdata%regrid_type == VERTICAL_METHOD_ETA2LEV) call this%vdata%get_interpolating_variable(this%input_bundle,rc=status) + _VERIFY(status) + + iter = this%items%begin() + do while (iter /= this%items%end()) + item => iter%get() + if (item%itemType == ItemTypeScalar) then + call this%CreateVariable(item%xname,rc=status) + _VERIFY(status) + else if (item%itemType == ItemTypeVector) then + call this%CreateVariable(item%xname,rc=status) + _VERIFY(status) + call this%CreateVariable(item%yname,rc=status) + _VERIFY(status) + end if + call iter%next() + enddo + + + ! __ add acc_bundle and output_bundle + ! + this%acc_bundle = ESMF_FieldBundleCreate(_RC) + call ESMF_FieldBundleSet(this%acc_bundle,grid=this%output_grid,_RC) + iter = this%items%begin() + do while (iter /= this%items%end()) + item => iter%get() + call this%addVariable_to_acc_bundle(item%xname,_RC) + if (item%itemType == ItemTypeVector) then + call this%addVariable_to_acc_bundle(item%yname,_RC) + end if + call iter%next() + enddo + + + ! __ add time to acc_bundle + ! + new_field = ESMF_FieldCreate(this%output_grid ,name='time', & + typekind=ESMF_TYPEKIND_R4,_RC) + call MAPL_FieldBundleAdd( this%acc_bundle, new_field, _RC ) + + + _RETURN(_SUCCESS) + end subroutine Create_Bundle_RH + + + subroutine set_param(this,deflation,quantize_algorithm,quantize_level,chunking,nbits_to_keep,regrid_method,itemOrder,write_collection_id,rc) + class (sampler), intent(inout) :: this + integer, optional, intent(in) :: deflation + integer, optional, intent(in) :: quantize_algorithm + integer, optional, intent(in) :: quantize_level + integer, optional, intent(in) :: chunking(:) + integer, optional, intent(in) :: nbits_to_keep + integer, optional, intent(in) :: regrid_method + logical, optional, intent(in) :: itemOrder + integer, optional, intent(in) :: write_collection_id + integer, optional, intent(out) :: rc + + integer :: status + + if (present(regrid_method)) this%regrid_method=regrid_method + if (present(nbits_to_keep)) this%nbits_to_keep=nbits_to_keep + if (present(deflation)) this%deflateLevel = deflation + if (present(quantize_algorithm)) this%quantizeAlgorithm = quantize_algorithm + if (present(quantize_level)) this%quantizeLevel = quantize_level + if (present(chunking)) then + allocate(this%chunking,source=chunking,stat=status) + _VERIFY(status) + end if + if (present(itemOrder)) this%itemOrderAlphabetical = itemOrder + if (present(write_collection_id)) this%write_collection_id=write_collection_id + _RETURN(ESMF_SUCCESS) + + end subroutine set_param + + subroutine set_default_chunking(this,rc) + class (sampler), intent(inout) :: this + integer, optional, intent(out) :: rc + + integer :: global_dim(3) + integer :: status + + call MAPL_GridGet(this%output_grid,globalCellCountPerDim=global_dim,rc=status) + _VERIFY(status) + if (global_dim(1)*6 == global_dim(2)) then + allocate(this%chunking(5)) + this%chunking(1) = global_dim(1) + this%chunking(2) = global_dim(1) + this%chunking(3) = 1 + this%chunking(4) = 1 + this%chunking(5) = 1 + else + allocate(this%chunking(4)) + this%chunking(1) = global_dim(1) + this%chunking(2) = global_dim(2) + this%chunking(3) = 1 + this%chunking(4) = 1 + endif + _RETURN(ESMF_SUCCESS) + + end subroutine set_default_chunking + + subroutine check_chunking(this,lev_size,rc) + class (sampler), intent(inout) :: this + integer, intent(in) :: lev_size + integer, optional, intent(out) :: rc + + integer :: global_dim(3) + integer :: status + character(len=5) :: c1,c2 + + call MAPL_GridGet(this%output_grid,globalCellCountPerDim=global_dim,rc=status) + _VERIFY(status) + if (global_dim(1)*6 == global_dim(2)) then + write(c2,'(I5)')global_dim(1) + write(c1,'(I5)')this%chunking(1) + _ASSERT(this%chunking(1) <= global_dim(1), "Chunk for Xdim "//c1//" must be less than or equal to "//c2) + write(c1,'(I5)')this%chunking(2) + _ASSERT(this%chunking(2) <= global_dim(1), "Chunk for Ydim "//c1//" must be less than or equal to "//c2) + _ASSERT(this%chunking(3) <= 6, "Chunksize for face dimension must be 6 or less") + if (lev_size > 0) then + write(c2,'(I5)')lev_size + write(c1,'(I5)')this%chunking(4) + _ASSERT(this%chunking(4) <= lev_size, "Chunk for level size "//c1//" must be less than or equal to "//c2) + end if + _ASSERT(this%chunking(5) == 1, "Time must have chunk size of 1") + else + write(c2,'(I5)')global_dim(1) + write(c1,'(I5)')this%chunking(1) + _ASSERT(this%chunking(1) <= global_dim(1), "Chunk for lon "//c1//" must be less than or equal to "//c2) + write(c2,'(I5)')global_dim(2) + write(c1,'(I5)')this%chunking(2) + _ASSERT(this%chunking(2) <= global_dim(2), "Chunk for lat "//c1//" must be less than or equal to "//c2) + if (lev_size > 0) then + write(c2,'(I5)')lev_size + write(c1,'(I5)')this%chunking(3) + _ASSERT(this%chunking(3) <= lev_size, "Chunk for level size "//c1//" must be less than or equal to "//c2) + end if + _ASSERT(this%chunking(4) == 1, "Time must have chunk size of 1") + endif + _RETURN(ESMF_SUCCESS) + + end subroutine check_chunking + + subroutine CreateVariable(this,itemName,rc) + class (sampler), intent(inout) :: this + character(len=*), intent(in) :: itemName + integer, optional, intent(out) :: rc + + integer :: status + + type(ESMF_Field) :: field,newField + class (AbstractGridFactory), pointer :: factory + integer :: fieldRank + logical :: isPresent + character(len=ESMF_MAXSTR) :: varName,longName,units + character(len=:), allocatable :: grid_dims + character(len=:), allocatable :: vdims + type(Variable) :: v + + call ESMF_FieldBundleGet(this%input_bundle,itemName,field=field,rc=status) + _VERIFY(status) + factory => get_factory(this%output_grid,rc=status) + _VERIFY(status) + + + call ESMF_FieldGet(field,rank=fieldRank,rc=status) + _VERIFY(status) + call ESMF_FieldGet(field,name=varName,rc=status) + _VERIFY(status) + call ESMF_AttributeGet(field,name="LONG_NAME",isPresent=isPresent,rc=status) + _VERIFY(status) + if ( isPresent ) then + call ESMF_AttributeGet (FIELD, NAME="LONG_NAME",VALUE=LongName, RC=STATUS) + _VERIFY(STATUS) + else + LongName = varName + endif + call ESMF_AttributeGet(field,name="UNITS",isPresent=isPresent,rc=status) + _VERIFY(status) + if ( isPresent ) then + call ESMF_AttributeGet (FIELD, NAME="UNITS",VALUE=units, RC=STATUS) + _VERIFY(STATUS) + else + units = 'unknown' + endif + + + ! finally make a new field if neccessary + if (this%doVertRegrid .and. (fieldRank ==3) ) then + newField = MAPL_FieldCreate(field,this%output_grid,lm=this%vData%lm,rc=status) + _VERIFY(status) + call MAPL_FieldBundleAdd(this%output_bundle,newField,rc=status) + _VERIFY(status) + else + newField = MAPL_FieldCreate(field,this%output_grid,rc=status) + _VERIFY(status) + call MAPL_FieldBundleAdd(this%output_bundle,newField,rc=status) + _VERIFY(status) + end if + _RETURN(_SUCCESS) + + end subroutine CreateVariable + + + subroutine RegridScalar(this,itemName,rc) + class (sampler), intent(inout) :: this + character(len=*), intent(in) :: itemName + integer, optional, intent(out) :: rc + + integer :: status + + type(ESMF_Field) :: field,outField + integer :: fieldRank + real, pointer :: ptr3d(:,:,:),outptr3d(:,:,:) + real, pointer :: ptr2d(:,:), outptr2d(:,:) + real, allocatable, target :: ptr3d_inter(:,:,:) + type(ESMF_Grid) :: gridIn,gridOut + logical :: hasDE_in, hasDE_out + logical :: first_entry + + call ESMF_FieldBundleGet(this%output_bundle,itemName,field=outField,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%input_bundle,grid=gridIn,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%output_bundle,grid=gridOut,rc=status) + _VERIFY(status) + hasDE_in = MAPL_GridHasDE(gridIn,rc=status) + _VERIFY(status) + hasDE_out = MAPL_GridHasDE(gridOut,rc=status) + _VERIFY(status) + first_entry = .true. + if (this%doVertRegrid) then + call ESMF_FieldBundleGet(this%input_bundle,itemName,field=field,rc=status) + _VERIFY(status) + call ESMF_FieldGet(Field,rank=fieldRank,rc=status) + _VERIFY(status) + if (fieldRank==3) then + if (hasDE_in) then + call ESMF_FieldGet(field,farrayPtr=ptr3d,rc=status) + _VERIFY(status) + else + allocate(ptr3d(0,0,0)) + end if + allocate(ptr3d_inter(size(ptr3d,1),size(ptr3d,2),this%vdata%lm),stat=status) + _VERIFY(status) + if (this%vdata%regrid_type==VERTICAL_METHOD_SELECT) then + call this%vdata%regrid_select_level(ptr3d,ptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then + call this%vdata%regrid_eta_to_pressure(ptr3d,ptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_FLIP) then + call this%vdata%flip_levels(ptr3d,ptr3d_inter,rc=status) + _VERIFY(status) + end if + ptr3d => ptr3d_inter + end if + else + if (first_entry) then + nullify(ptr3d) + first_entry = .false. + end if + end if + + call ESMF_FieldBundleGet(this%input_bundle,itemName,field=field,rc=status) + _VERIFY(status) + call ESMF_FieldGet(field,rank=fieldRank,rc=status) + _VERIFY(status) + if (fieldRank==2) then + if (hasDE_in) then + call MAPL_FieldGetPointer(field,ptr2d,rc=status) + _VERIFY(status) + else + allocate(ptr2d(0,0)) + end if + if (hasDE_out) then + call MAPL_FieldGetPointer(OutField,outptr2d,rc=status) + _VERIFY(status) + else + allocate(outptr2d(0,0)) + end if + if (gridIn==gridOut) then + outPtr2d=ptr2d + else + if (this%regrid_method==REGRID_METHOD_FRACTION) ptr2d=ptr2d-this%fraction + call this%regrid_handle%regrid(ptr2d,outPtr2d,rc=status) + _VERIFY(status) + end if + +!! print *, maxval(ptr2d) +!! print *, minval(ptr2d) +!! print *, maxval(outptr2d) +!! print *, minval(outptr2d) + + else if (fieldRank==3) then + if (.not.associated(ptr3d)) then + if (hasDE_in) then + call ESMF_FieldGet(field,farrayPtr=ptr3d,rc=status) + _VERIFY(status) + else + allocate(ptr3d(0,0,0)) + end if + end if + if (hasDE_out) then + call MAPL_FieldGetPointer(OutField,outptr3d,rc=status) + _VERIFY(status) + else + allocate(outptr3d(0,0,0)) + end if + if (gridIn==gridOut) then + outPtr3d=Ptr3d + else + if (this%regrid_method==REGRID_METHOD_FRACTION) ptr3d=ptr3d-this%fraction + call this%regrid_handle%regrid(ptr3d,outPtr3d,rc=status) + _VERIFY(status) + end if + else + _FAIL('rank not supported') + end if + + if (allocated(ptr3d_inter)) deallocate(ptr3d_inter) + _RETURN(_SUCCESS) + + end subroutine RegridScalar + + subroutine RegridVector(this,xName,yName,rc) + class (sampler), intent(inout) :: this + character(len=*), intent(in) :: xName + character(len=*), intent(in) :: yName + integer, optional, intent(out) :: rc + + integer :: status + + type(ESMF_Field) :: xfield,xoutField + type(ESMF_Field) :: yfield,youtField + integer :: fieldRank + real, pointer :: xptr3d(:,:,:),xoutptr3d(:,:,:) + real, pointer :: xptr2d(:,:), xoutptr2d(:,:) + real, allocatable, target :: xptr3d_inter(:,:,:) + real, pointer :: yptr3d(:,:,:),youtptr3d(:,:,:) + real, pointer :: yptr2d(:,:), youtptr2d(:,:) + real, allocatable, target :: yptr3d_inter(:,:,:) + type(ESMF_Grid) :: gridIn, gridOut + logical :: hasDE_in, hasDE_out + + call ESMF_FieldBundleGet(this%output_bundle,xName,field=xoutField,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%output_bundle,yName,field=youtField,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%input_bundle,grid=gridIn,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%output_bundle,grid=gridOut,rc=status) + _VERIFY(status) + hasDE_in = MAPL_GridHasDE(gridIn,rc=status) + _VERIFY(status) + hasDE_out = MAPL_GridHasDE(gridOut,rc=status) + _VERIFY(status) + + if (this%doVertRegrid) then + call ESMF_FieldBundleGet(this%input_bundle,xName,field=xfield,rc=status) + _VERIFY(status) + call ESMF_FieldGet(xField,rank=fieldRank,rc=status) + _VERIFY(status) + if (fieldRank==3) then + if (hasDE_in) then + call ESMF_FieldGet(xfield,farrayPtr=xptr3d,rc=status) + _VERIFY(status) + else + allocate(xptr3d(0,0,0)) + end if + allocate(xptr3d_inter(size(xptr3d,1),size(xptr3d,2),this%vdata%lm),stat=status) + _VERIFY(status) + if (this%vdata%regrid_type==VERTICAL_METHOD_SELECT) then + call this%vdata%regrid_select_level(xptr3d,xptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then + call this%vdata%regrid_eta_to_pressure(xptr3d,xptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_FLIP) then + call this%vdata%flip_levels(xptr3d,xptr3d_inter,rc=status) + _VERIFY(status) + end if + xptr3d => xptr3d_inter + end if + call ESMF_FieldBundleGet(this%input_bundle,yName,field=yfield,rc=status) + _VERIFY(status) + call ESMF_FieldGet(yField,rank=fieldRank,rc=status) + _VERIFY(status) + if (fieldRank==3) then + if (hasDE_in) then + call ESMF_FieldGet(yfield,farrayPtr=yptr3d,rc=status) + _VERIFY(status) + else + allocate(yptr3d(0,0,0)) + end if + allocate(yptr3d_inter(size(yptr3d,1),size(yptr3d,2),this%vdata%lm),stat=status) + _VERIFY(status) + if (this%vdata%regrid_type==VERTICAL_METHOD_SELECT) then + call this%vdata%regrid_select_level(yptr3d,yptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then + call this%vdata%regrid_eta_to_pressure(yptr3d,yptr3d_inter,rc=status) + _VERIFY(status) + else if (this%vdata%regrid_type==VERTICAL_METHOD_FLIP) then + call this%vdata%flip_levels(yptr3d,yptr3d_inter,rc=status) + _VERIFY(status) + end if + yptr3d => yptr3d_inter + end if + else + if (associated(xptr3d)) nullify(xptr3d) + if (associated(yptr3d)) nullify(yptr3d) + end if + + call ESMF_FieldBundleGet(this%input_bundle,xname,field=xfield,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%input_bundle,yname,field=yfield,rc=status) + _VERIFY(status) + call ESMF_FieldGet(xfield,rank=fieldRank,rc=status) + _VERIFY(status) + if (fieldRank==2) then + if (hasDE_in) then + call MAPL_FieldGetPointer(xfield,xptr2d,rc=status) + _VERIFY(status) + call MAPL_FieldGetPointer(yfield,yptr2d,rc=status) + _VERIFY(status) + else + allocate(xptr2d(0,0)) + allocate(yptr2d(0,0)) + end if + + if (hasDE_in) then + call MAPL_FieldGetPointer(xOutField,xoutptr2d,rc=status) + _VERIFY(status) + call MAPL_FieldGetPointer(yOutField,youtptr2d,rc=status) + _VERIFY(status) + else + allocate(xoutptr2d(0,0)) + allocate(youtptr2d(0,0)) + end if + + + if (gridIn==gridOut) then + xoutPtr2d=xptr2d + youtPtr2d=yptr2d + else + call this%regrid_handle%regrid(xptr2d,yptr2d,xoutPtr2d,youtPtr2d,rc=status) + _VERIFY(status) + end if + else if (fieldRank==3) then + if (.not.associated(xptr3d)) then + if (hasDE_in) then + call MAPL_FieldGetPointer(xfield,xptr3d,rc=status) + _VERIFY(status) + else + allocate(xptr3d(0,0,0)) + end if + end if + if (.not.associated(yptr3d)) then + if (hasDE_in) then + call MAPL_FieldGetPointer(yfield,yptr3d,rc=status) + _VERIFY(status) + else + allocate(yptr3d(0,0,0)) + end if + end if + + if (hasDE_out) then + call MAPL_FieldGetPointer(xOutField,xoutptr3d,rc=status) + _VERIFY(status) + call MAPL_FieldGetPointer(yOutField,youtptr3d,rc=status) + _VERIFY(status) + else + allocate(xoutptr3d(0,0,0)) + allocate(youtptr3d(0,0,0)) + end if + + if (gridIn==gridOut) then + xoutPtr3d=xptr3d + youtPtr3d=yptr3d + else + call this%regrid_handle%regrid(xptr3d,yptr3d,xoutPtr3d,youtPtr3d,rc=status) + _VERIFY(status) + end if + end if + + if (allocated(xptr3d_inter)) deallocate(xptr3d_inter) + if (allocated(yptr3d_inter)) deallocate(yptr3d_inter) + _RETURN(_SUCCESS) + + end subroutine RegridVector + + + subroutine alphabatize_variables(this,nfixedVars,rc) + class (sampler), intent(inout) :: this + integer, intent(in) :: nFixedVars + integer, optional, intent(out) :: rc + + type(StringVector) :: order + type(StringVector) :: newOrder + character(len=:), pointer :: v1 + character(len=ESMF_MAXSTR) :: c1,c2 + character(len=ESMF_MAXSTR), allocatable :: temp(:) + logical :: swapped + integer :: n,i + integer :: status + + order = this%metadata%get_order(rc=status) + _VERIFY(status) + n = Order%size() + allocate(temp(nFixedVars+1:n)) + do i=1,n + v1 => order%at(i) + if ( i > nFixedVars) temp(i)=trim(v1) + enddo + + swapped = .true. + do while(swapped) + swapped = .false. + do i=nFixedVars+1,n-1 + c1 = temp(i) + c2 = temp(i+1) + if (c1 > c2) then + temp(i+1)=c1 + temp(i)=c2 + swapped =.true. + end if + enddo + enddo + + do i=1,nFixedVars + v1 => Order%at(i) + call newOrder%push_back(v1) + enddo + do i=nFixedVars+1,n + call newOrder%push_back(trim(temp(i))) + enddo + call this%metadata%set_order(newOrder,rc=status) + _VERIFY(status) + deallocate(temp) + + _RETURN(_SUCCESS) + + end subroutine alphabatize_variables + + + subroutine addVariable_to_acc_bundle(this,itemName,rc) + class (sampler), intent(inout) :: this + character(len=*), intent(in) :: itemName + integer, optional, intent(out) :: rc + + type(ESMF_Field) :: field,newField + type(ESMF_Array) :: array1 + real(KIND=ESMF_KIND_R4), pointer :: pt2d(:,:) + class (AbstractGridFactory), pointer :: factory + integer :: fieldRank + logical :: isPresent + integer :: status + + call ESMF_FieldBundleGet(this%input_bundle,itemName,field=field,_RC) + call ESMF_FieldGet(field,rank=fieldRank,rc=status) + if (this%doVertRegrid .and. (fieldRank ==3) ) then + newField = MAPL_FieldCreate(field,this%output_grid,lm=this%vData%lm,_RC) + else + newField = MAPL_FieldCreate(field,this%output_grid,_RC) + end if + call MAPL_FieldBundleAdd(this%acc_bundle,newField,_RC) + + _RETURN(_SUCCESS) + + end subroutine addVariable_to_acc_bundle + + + subroutine addVariable_to_output_bundle(this,itemName,rc) + class (sampler), intent(inout) :: this + character(len=*), intent(in) :: itemName + integer, optional, intent(out) :: rc + + type(ESMF_Field) :: field,newField + class (AbstractGridFactory), pointer :: factory + integer :: fieldRank + logical :: isPresent + integer :: status + + call ESMF_FieldBundleGet(this%input_bundle,itemName,field=field,_RC) + call ESMF_FieldGet(field,rank=fieldRank,rc=status) + if (this%doVertRegrid .and. (fieldRank ==3) ) then + newField = MAPL_FieldCreate(field,this%output_grid,lm=this%vData%lm,_RC) + else + newField = MAPL_FieldCreate(field,this%output_grid,_RC) + end if + call MAPL_FieldBundleAdd(this%output_bundle,newField,_RC) + + _RETURN(_SUCCESS) + end subroutine addVariable_to_output_bundle + + + + !! -- based on subroutine bundlepost(this,filename,oClients,rc) + !! + subroutine interp_accumulate_fields (this,xy_subset,rc) + implicit none + class (sampler) :: this + integer, intent(in) :: xy_subset(2,2) + !!integer, intent(in) :: xy_mask(:,:) + integer, optional, intent(out) :: rc + + integer :: status + type(ESMF_Field) :: outField + type(ESMF_Field) :: new_outField + type(ESMF_Grid) :: grid + integer :: tindex + type(ArrayReference) :: ref + + type(GriddedIOitemVectorIterator) :: iter + type(GriddedIOitem), pointer :: item + logical :: have_time + + type(ESMF_Array) :: array1, array2 + integer :: is,ie,js,je + + integer :: rank, rank1, rank2 + real(KIND=ESMF_KIND_R4), pointer :: pt2d(:,:), pt2d_(:,:) + real(KIND=ESMF_KIND_R4), pointer :: pt3d(:,:,:), pt3d_(:,:,:) + + integer :: localDe, localDECount + integer, dimension(:), allocatable :: LB, UB, exclusiveCount + integer, dimension(:), allocatable :: compLB, compUB, compCount + integer :: dimCount + integer :: y1, y2 + integer :: j, jj + integer :: ii1, iin, jj1, jjn + integer, dimension(:), allocatable :: j1, j2 + + is=xy_subset(1,1); ie=xy_subset(2,1) + js=xy_subset(1,2); je=xy_subset(2,2) + + if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then + call this%vdata%setup_eta_to_pressure(regrid_handle=this%regrid_handle,output_grid=this%output_grid,rc=status) + _VERIFY(status) + end if + + call ESMF_FieldBundleGet(this%output_bundle, grid=grid, _RC) + call ESMF_GridGet(grid, localDECount=localDECount, dimCount=dimCount, _RC) + allocate ( LB(dimCount), UB(dimCount), exclusiveCount(dimCount) ) + allocate ( compLB(dimCount), compUB(dimCount), compCount(dimCount) ) + + allocate ( j1(0:localDEcount-1) ) ! start + allocate ( j2(0:localDEcount-1) ) ! end + + _ASSERT ( localDEcount == 1, 'failed, due to localDEcount > 1') + call MAPL_GridGetInterior(grid,ii1,iin,jj1,jjn) +!! write(6,*) 'MAPL_GridGetInterior, ii1,iin,jj1,jjn', ii1,iin,jj1,jjn +!! print*, 'js,je ', js, je + + LB(1)=ii1; LB(2)=jj1 + UB(1)=iin; UB(2)=jjn + + do localDe=0, localDEcount-1 + ! + ! is/ie, js/je, [LB, UB] + ! + ! + y1=jj1; y2=jjn + if (y1 < js) then + if (y2 < js) then + j1(localDe)=-1 + j2(localDe)=-1 + elseif (y2 < je) then + j1(localDe)=js + j2(localDe)=y2 + else + j1(localDe)=js + j2(localDe)=je + endif + elseif (y1 <= je) then + j1(localDe)=y1 + if (y2 < je) then + j2(localDe)=y2 + else + j2(localDe)=je + endif + else + j1(localDe)=-1 + j2(localDe)=-1 + endif + enddo + +!! write(6,*) 'ck bundlepost_acc' +!! write(6,*) 'j1(localDe)', j1(0:localDeCount-1) +!! write(6,*) 'j2(localDe)', j2(0:localDeCount-1) + + + iter = this%items%begin() + do while (iter /= this%items%end()) + item => iter%get() + if (item%itemType == ItemTypeScalar) then + call this%RegridScalar(item%xname,rc=status) + _VERIFY(status) + call ESMF_FieldBundleGet(this%output_bundle,item%xname,field=outField, _RC) + _VERIFY(status) + if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then + call this%vdata%correct_topo(outField,rc=status) + _VERIFY(status) + end if + + ! -- mask the time interval + ! store the time interval fields into new bundle + call ESMF_FieldGet(outField, Array=array1, _RC) + call ESMF_FieldBundleGet(this%acc_bundle,item%xname,field=new_outField,_RC) + call ESMF_FieldGet(new_outField, Array=array2, _RC) + call ESMF_ArrayGet(array1, rank=rank, _RC) + if (rank==2) then + call ESMF_ArrayGet(array1, farrayptr=pt2d, _RC) + call ESMF_ArrayGet(array2, farrayptr=pt2d_, _RC) + localDe=0 + if (j1(localDe)>0) then + do j= j1(localDe), j2(localDe) + jj= j-jj1+1 ! j_local +!! write(6,*) 'j, jj', j, jj + pt2d_(:,jj) = pt2d(:,jj) + enddo + endif + elseif (rank==3) then + call ESMF_ArrayGet(array1, farrayptr=pt3d, _RC) + call ESMF_ArrayGet(array2, farrayptr=pt3d_, _RC) + do localDe=0, localDEcount-1 + if (j1(localDe)>0) then + do j= j1(localDe), j2(localDe) + jj= j-jj1+1 + pt3d_(:,jj,:) = pt3d(:,jj,:) + enddo + endif + enddo + else + _FAIL('failed interp_accumulate_fields') + endif + + else if (item%itemType == ItemTypeVector) then + _FAIL('ItemTypeVector not implemented') + end if + call iter%next() + enddo + + _RETURN(ESMF_SUCCESS) + + end subroutine interp_accumulate_fields + + + subroutine get_xy_mask(grid, xy_subset, xy_mask, rc) + implicit none + type(ESMF_Grid), intent(in) :: grid + integer, intent(in) :: xy_subset(2,2) + integer, intent(out) :: xy_mask(:,:) + integer, optional, intent(out) :: rc + + integer :: status + integer :: ii1, iin, jj1, jjn ! local box for localDE + integer :: is, ie, js, je ! global box for each time-interval + integer :: j1p, jnp ! local y-index for each time-interval + + integer :: dimCount + integer :: y1, y2 + integer :: j, jj + integer :: j1, j2 + + is=xy_subset(1,1); ie=xy_subset(2,1) + js=xy_subset(1,2); je=xy_subset(2,2) + + call MAPL_GridGetInterior(grid,ii1,iin,jj1,jjn) + write(6,*) 'MAPL_GridGetInterior, ii1,iin,jj1,jjn', ii1,iin,jj1,jjn + + y1=jj1; y2=jjn + if (y1 < js) then + if (y2 < js) then + j1=-1 + j2=-1 + elseif (y2 < je) then + j1=js + j2=y2 + else + j1=js + j2=je + endif + elseif (y1 <= je) then + j1=y1 + if (y2 < je) then + j2=y2 + else + j2=je + endif + else + j1=-1 + j2=-1 + endif + +!! write(6,*) 'get_xy_mask: j1,j2=', j1, j2 + xy_mask(:,:) = 0 + if (j1 > 0) then + do jj = j1, j2 + xy_mask(:, jj) = 1 + enddo + end if + + if(present(rc)) rc=0 + + end subroutine get_xy_mask + + +end module MAPL_EpochSwathMod diff --git a/gridcomps/History/MAPL_HistoryCollection.F90 b/gridcomps/History/MAPL_HistoryCollection.F90 index b5c0c3d35f19..98d434e78416 100644 --- a/gridcomps/History/MAPL_HistoryCollection.F90 +++ b/gridcomps/History/MAPL_HistoryCollection.F90 @@ -11,6 +11,7 @@ module MAPL_HistoryCollectionMod use HistoryTrajectoryMod use StationSamplerMod use gFTL_StringStringMap + use MAPL_EpochSwathMod implicit none private @@ -55,6 +56,7 @@ module MAPL_HistoryCollectionMod integer,pointer :: expSTATE (:) integer :: unit type(ESMF_FieldBundle) :: bundle + type(sampler) :: xsampler type(MAPL_CFIO) :: MCFIO type(MAPL_GriddedIO) :: mGriddedIO type(VerticalData) :: vdata diff --git a/gridcomps/History/MAPL_HistoryGridComp.F90 b/gridcomps/History/MAPL_HistoryGridComp.F90 index d9356e9bb70c..aca6a74fec8c 100644 --- a/gridcomps/History/MAPL_HistoryGridComp.F90 +++ b/gridcomps/History/MAPL_HistoryGridComp.F90 @@ -58,6 +58,8 @@ module MAPL_HistoryGridCompMod use MAPL_TimeUtilsMod, only: is_valid_time, is_valid_date use gFTL_StringStringMap !use ESMF_CFIOMOD + use MAPL_EpochSwathMod + use pflogger, only: Logger, logging use mpi @@ -143,6 +145,8 @@ module MAPL_HistoryGridCompMod public HISTORY_ExchangeListWrap + type(samplerHQ) :: Hsampler + contains !===================================================================== @@ -484,7 +488,6 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) ! Read User-Supplied History Lists from Config File ! ------------------------------------------------- call ESMF_GridCompGet( gc, config=config, _RC ) - call ESMF_ConfigGetAttribute ( config, value=INTSTATE%expsrc, & label ='EXPSRC:', default='', _RC ) call ESMF_ConfigGetAttribute ( config, value=INTSTATE%expid, & @@ -577,7 +580,6 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) end if call ESMF_ConfigNextLine ( config,tableEnd=tend,_RC ) enddo - if (nlist == 0) then _RETURN(ESMF_SUCCESS) end if @@ -614,13 +616,18 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) call MAPL_ConfigSetAttribute(config, value=nx,label=trim(key)//".NX:",_RC) call MAPL_ConfigSetAttribute(config, value=ny,label=trim(key)//".NY:",_RC) end if - output_grid = grid_manager%make_grid(config, prefix=key//'.', _RC) + + if (trim(grid_type)/='Swath') then + output_grid = grid_manager%make_grid(config, prefix=key//'.', _RC) + else + Hsampler = samplerHQ(clock, config, key, _RC) + output_grid = Hsampler%create_grid(key, currTime, grid_type=grid_type, _RC) + end if call IntState%output_grids%set(key, output_grid) call iter%next() end do end block OUTPUT_GRIDS end if - if (intstate%version >= 2) then call ESMF_ConfigFindLabel(config, 'FIELD_SETS:', _RC) table_end = .false. @@ -645,7 +652,6 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) end if - allocate(IntState%Regrid(nlist), _STAT) allocate( Vvarn(nlist), _STAT) allocate(INTSTATE%STAMPOFFSET(nlist), _STAT) @@ -654,17 +660,14 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) ! ---------------------------------------------------------------------------- if( MAPL_AM_I_ROOT(vm) ) then - call ESMF_ConfigGetAttribute(config, value=HIST_CF, & label="HIST_CF:", default="HIST.rc", _RC ) unitr = GETFILE(HIST_CF, FORM='formatted', _RC) - ! for each collection do n = 1, nlist rewind(unitr) string = trim( list(n)%collection ) // '.' unitw = GETFILE(trim(string)//'rcx', FORM='formatted', _RC) - match = .false. contLine = .false. con3 = .false. @@ -698,7 +701,6 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) end if - call ESMF_VMbarrier(vm, _RC) ! Initialize History Lists @@ -1379,7 +1381,6 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) end if enddo enddo - ! Get Output Export States ! ------------------------ @@ -2364,6 +2365,16 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) else list(n)%vdata = VerticalData(positive=list(n)%positive,_RC) end if + if (trim(list(n)%output_grid_label)=='SwathGrid') then + call list(n)%xsampler%set_param(deflation=list(n)%deflate,_RC) + call list(n)%xsampler%set_param(quantize_algorithm=list(n)%quantize_algorithm,_RC) + call list(n)%xsampler%set_param(quantize_level=list(n)%quantize_level,_RC) + call list(n)%xsampler%set_param(chunking=list(n)%chunkSize,_RC) + call list(n)%xsampler%set_param(nbits_to_keep=list(n)%nbits_to_keep,_RC) + call list(n)%xsampler%set_param(regrid_method=list(n)%regrid_method,_RC) + call list(n)%xsampler%set_param(itemOrder=intState%fileOrderAlphabetical,_RC) + endif + call list(n)%mGriddedIO%set_param(deflation=list(n)%deflate,_RC) call list(n)%mGriddedIO%set_param(quantize_algorithm=list(n)%quantize_algorithm,_RC) call list(n)%mGriddedIO%set_param(quantize_level=list(n)%quantize_level,_RC) @@ -2371,10 +2382,11 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) call list(n)%mGriddedIO%set_param(nbits_to_keep=list(n)%nbits_to_keep,_RC) call list(n)%mGriddedIO%set_param(regrid_method=list(n)%regrid_method,_RC) call list(n)%mGriddedIO%set_param(itemOrder=intState%fileOrderAlphabetical,_RC) + if (list(n)%monthly) then - nextMonth = currTime - oneMonth - dur = nextMonth - currTime - call ESMF_TimeIntervalGet(dur, s=sec, _RC) + nextMonth = currTime - oneMonth + dur = nextMonth - currTime + call ESMF_TimeIntervalGet(dur, s=sec, _RC) list(n)%timeInfo = TimeData(clock,tm,sec,IntState%stampoffset(n),funits='days') else list(n)%timeInfo = TimeData(clock,tm,MAPL_nsecf(list(n)%frequency),IntState%stampoffset(n),integer_time=intstate%integer_time) @@ -2387,14 +2399,19 @@ subroutine Initialize ( gc, import, dumexport, clock, rc ) call list(n)%station_sampler%add_metadata_route_handle(list(n)%bundle,list(n)%timeInfo,vdata=list(n)%vdata,_RC) else global_attributes = list(n)%global_atts%define_collection_attributes(_RC) - if (trim(list(n)%output_grid_label)/='') then + if (trim(list(n)%output_grid_label)=='SwathGrid') then pgrid => IntState%output_grids%at(trim(list(n)%output_grid_label)) - call list(n)%mGriddedIO%CreateFileMetaData(list(n)%items,list(n)%bundle,list(n)%timeInfo,ogrid=pgrid,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) + call list(n)%xsampler%Create_bundle_RH(list(n)%items,list(n)%bundle,ogrid=pgrid,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) else - call list(n)%mGriddedIO%CreateFileMetaData(list(n)%items,list(n)%bundle,list(n)%timeInfo,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) - end if - collection_id = o_Clients%add_hist_collection(list(n)%mGriddedIO%metadata, mode = create_mode) - call list(n)%mGriddedIO%set_param(write_collection_id=collection_id) + if (trim(list(n)%output_grid_label)/='') then + pgrid => IntState%output_grids%at(trim(list(n)%output_grid_label)) + call list(n)%mGriddedIO%CreateFileMetaData(list(n)%items,list(n)%bundle,list(n)%timeInfo,ogrid=pgrid,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) + else + call list(n)%mGriddedIO%CreateFileMetaData(list(n)%items,list(n)%bundle,list(n)%timeInfo,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) + end if + collection_id = o_Clients%add_hist_collection(list(n)%mGriddedIO%metadata, mode = create_mode) + call list(n)%mGriddedIO%set_param(write_collection_id=collection_id) + endif end if end if call ESMF_ConfigDestroy(cfg, _RC) @@ -3222,7 +3239,15 @@ subroutine Run ( gc, import, export, clock, rc ) type(ESMF_Time) :: lastMonth type(ESMF_TimeInterval) :: dur, oneMonth integer :: sec + type (StringGridMap) :: pt_output_grids + character(len=ESMF_MAXSTR) :: key_grid_label + type (ESMF_Grid), pointer :: pgrid + integer :: collection_id + integer :: create_mode + type(StringStringMap) :: global_attributes + type(timeData) :: timeinfo_uninit + type(ESMF_Grid) :: new_grid ! variables for "backwards" mode logical :: fwd logical, allocatable :: Ignore(:) @@ -3230,12 +3255,13 @@ subroutine Run ( gc, import, export, clock, rc ) ! ErrLog vars integer :: status logical :: file_exists + type(GriddedIOitem) :: item + type(Logger), pointer :: lgr !============================================================================= ! Begin... - _UNUSED_DUMMY(import) _UNUSED_DUMMY(export) @@ -3339,6 +3365,8 @@ subroutine Run ( gc, import, export, clock, rc ) Writing(n) = .false. else if (list(n)%timeseries_output) then Writing(n) = ESMF_AlarmIsRinging ( list(n)%trajectory%alarm ) + else if (trim(list(n)%output_grid_label)=='SwathGrid') then + Writing(n) = ESMF_AlarmIsRinging ( Hsampler%alarm ) else Writing(n) = ESMF_AlarmIsRinging ( list(n)%his_alarm ) endif @@ -3379,8 +3407,41 @@ subroutine Run ( gc, import, export, clock, rc ) end do + if(any(Writing)) call WRITE_PARALLEL("") + + ! swath only + epoch_swath_grid_case: do n=1,nlist + if (trim(list(n)%output_grid_label)=='SwathGrid') then + call Hsampler%regrid_accumulate(list(n)%xsampler,_RC) + + if( ESMF_AlarmIsRinging ( Hsampler%alarm ) ) then + create_mode = PFIO_NOCLOBBER ! defaut no overwrite + if (intState%allow_overwrite) create_mode = PFIO_CLOBBER + + ! add time to items + ! true metadata comes here from mGriddedIO%metadata + ! the mGriddedIO below only touches metadata, collection_id etc., it is safe. + ! + if (.NOT. list(n)%xsampler%have_initalized) then + list(n)%xsampler%have_initalized = .true. + global_attributes = list(n)%global_atts%define_collection_attributes(_RC) + endif + item%itemType = ItemTypeScalar + item%xname = 'time' + call list(n)%items%push_back(item) + call Hsampler%fill_time_in_bundle ('time', list(n)%xsampler%acc_bundle, _RC) + call list(n)%mGriddedIO%destroy(_RC) + call list(n)%mGriddedIO%CreateFileMetaData(list(n)%items,list(n)%xsampler%acc_bundle,timeinfo_uninit,vdata=list(n)%vdata,global_attributes=global_attributes,_RC) + call list(n)%items%pop_back() + + collection_id = o_Clients%add_hist_collection(list(n)%mGriddedIO%metadata, mode = create_mode) + call list(n)%mGriddedIO%set_param(write_collection_id=collection_id) + endif + end if + end do epoch_swath_grid_case + ! Write Id and time ! ----------------- @@ -3457,7 +3518,7 @@ subroutine Run ( gc, import, export, clock, rc ) end if elseif (list(n)%sampler_spec == 'station') then if (list(n)%unit.eq.0) then - if (mapl_am_i_root()) call lgr%debug('%a %a',& + call lgr%debug('%a %a',& "Station_data output to new file:",trim(filename(n))) call list(n)%station_sampler%close_file_handle(_RC) call list(n)%station_sampler%create_file_handle(filename(n),_RC) @@ -3471,7 +3532,9 @@ subroutine Run ( gc, import, export, clock, rc ) inquire (file=trim(filename(n)),exist=file_exists) _ASSERT(.not.file_exists,trim(filename(n))//" being created for History output already exists") end if - call list(n)%mGriddedIO%modifyTime(oClients=o_Clients,_RC) + if (trim(list(n)%output_grid_label)/='SwathGrid') then + call list(n)%mGriddedIO%modifyTime(oClients=o_Clients,_RC) + endif list(n)%currentFile = filename(n) list(n)%unit = -1 else @@ -3491,6 +3554,7 @@ subroutine Run ( gc, import, export, clock, rc ) enddo OPENLOOP call MAPL_TimerOff(GENSTATE,"----IO Create") + call MAPL_TimerOn(GENSTATE,"----IO Write") call MAPL_TimerOn(GENSTATE,"-----IO Post") POSTLOOP: do n=1,nlist @@ -3544,9 +3608,7 @@ subroutine Run ( gc, import, export, clock, rc ) if (.not.list(n)%timeseries_output) then IOTYPE: if (list(n)%unit < 0) then ! CFIO - call list(n)%mGriddedIO%bundlepost(list(n)%currentFile,oClients=o_Clients,_RC) - else if( INTSTATE%LCTL(n) ) then @@ -3579,6 +3641,7 @@ subroutine Run ( gc, import, export, clock, rc ) enddo POSTLOOP + if (any(writing)) then call o_Clients%done_collective_stage(_RC) call o_Clients%post_wait() @@ -3589,6 +3652,23 @@ subroutine Run ( gc, import, export, clock, rc ) call MAPL_TimerOn(GENSTATE,"----IO Write") call MAPL_TimerOn(GENSTATE,"-----IO Wait") + + ! destroy ogrid/RH/acc_bundle, regenerate them + ! swath only + epoch_swath_regen_grid: do n=1,nlist + if (trim(list(n)%output_grid_label)=='SwathGrid') then + if( ESMF_AlarmIsRinging ( Hsampler%alarm ) ) then + key_grid_label = list(n)%output_grid_label + call Hsampler%destroy_rh_regen_ogrid ( key_grid_label, IntState%output_grids, list(n)%xsampler, _RC ) + pgrid => IntState%output_grids%at(trim(list(n)%output_grid_label)) + call list(n)%xsampler%Create_bundle_RH(list(n)%items,list(n)%bundle,ogrid=pgrid,& + vdata=list(n)%vdata,global_attributes=global_attributes,_RC) + if( MAPL_AM_I_ROOT() ) write(6,'(//)') + endif + end if + end do epoch_swath_regen_grid + + WAITLOOP: do n=1,nlist if( Writing(n) .and. list(n)%unit < 0) then diff --git a/gridcomps/History/MAPL_HistoryTrajectoryMod_smod.F90 b/gridcomps/History/MAPL_HistoryTrajectoryMod_smod.F90 index 012c3ba6b48d..3e43e074578b 100644 --- a/gridcomps/History/MAPL_HistoryTrajectoryMod_smod.F90 +++ b/gridcomps/History/MAPL_HistoryTrajectoryMod_smod.F90 @@ -494,9 +494,9 @@ allocate(this%lons(0),this%lats(0),_STAT) allocate(this%times_R8(0),_STAT) allocate(this%obstype_id(0),_STAT) - this%epoch_index(1:2)=0 + this%epoch_index(1:2) = 0 this%nobs_epoch = 0 - rc=0 + rc = 0 return end if diff --git a/griddedio/GriddedIO.F90 b/griddedio/GriddedIO.F90 index 7b62d2e0dacf..2fa6bb522d89 100644 --- a/griddedio/GriddedIO.F90 +++ b/griddedio/GriddedIO.F90 @@ -31,7 +31,7 @@ module MAPL_GriddedIOMod private type, public :: MAPL_GriddedIO - type(FileMetaData) :: metadata + type(FileMetaData), allocatable :: metadata type(fileMetadataUtils), pointer :: current_file_metadata integer :: write_collection_id integer :: read_collection_id @@ -73,6 +73,7 @@ module MAPL_GriddedIOMod procedure :: request_data_from_file procedure :: process_data_from_file procedure :: swap_undef_value + procedure :: destroy end type MAPL_GriddedIO interface MAPL_GriddedIO @@ -108,13 +109,13 @@ function new_MAPL_GriddedIO(metadata,input_bundle,output_bundle,write_collection end function new_MAPL_GriddedIO subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attributes,rc) - class (MAPL_GriddedIO), intent(inout) :: this + class (MAPL_GriddedIO), target, intent(inout) :: this type(GriddedIOitemVector), target, intent(inout) :: items type(ESMF_FieldBundle), intent(inout) :: bundle type(TimeData), intent(inout) :: timeInfo type(VerticalData), intent(inout), optional :: vdata type (ESMF_Grid), intent(inout), pointer, optional :: ogrid - type(StringStringMap), intent(in), optional :: global_attributes + type(StringStringMap), target, intent(in), optional :: global_attributes integer, intent(out), optional :: rc type(ESMF_Grid) :: input_grid @@ -128,6 +129,11 @@ subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attr character(len=:), pointer :: attr_name, attr_val integer :: status + if ( allocated (this%metadata) ) deallocate(this%metadata) + allocate(this%metadata) + + call MAPL_FieldBundleDestroy(this%output_bundle, _RC) + this%items = items this%input_bundle = bundle this%output_bundle = ESMF_FieldBundleCreate(rc=status) @@ -141,9 +147,11 @@ subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attr call ESMF_FieldBundleGet(this%input_bundle,grid=this%output_grid,rc=status) _VERIFY(status) end if + this%regrid_handle => new_regridder_manager%make_regridder(input_grid,this%output_grid,this%regrid_method,rc=status) _VERIFY(status) + ! We get the regrid_method here because in the case of Identity, we set it to ! REGRID_METHOD_IDENTITY in the regridder constructor if identity. Now we need ! to change the regrid_method in the GriddedIO object to be the same as the @@ -156,6 +164,7 @@ subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attr _VERIFY(status) call factory%append_metadata(this%metadata) + if (present(vdata)) then this%vdata=vdata else @@ -179,6 +188,7 @@ subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attr call this%check_chunking(this%vdata%lm,_RC) end if + order = this%metadata%get_order(rc=status) _VERIFY(status) metadataVarsSize = order%size() @@ -213,7 +223,16 @@ subroutine CreateFileMetaData(this,items,bundle,timeInfo,vdata,ogrid,global_attr end if _RETURN(_SUCCESS) - end subroutine CreateFileMetaData + end subroutine CreateFileMetaData + + + subroutine destroy(this, rc) + class (MAPL_GriddedIO), intent(inout) :: this + integer, intent(out), optional :: rc + if(allocated(this%chunking)) deallocate(this%chunking) + _RETURN(_SUCCESS) + end subroutine destroy + subroutine set_param(this,deflation,quantize_algorithm,quantize_level,chunking,nbits_to_keep,regrid_method,itemOrder,write_collection_id,rc) class (MAPL_GriddedIO), intent(inout) :: this @@ -483,6 +502,7 @@ subroutine bundlepost(this,filename,oClients,rc) end if else tindex = -1 + call this%stage2DLatLon(filename,oClients=oClients,_RC) end if if (this%vdata%regrid_type==VERTICAL_METHOD_ETA2LEV) then @@ -729,8 +749,8 @@ subroutine RegridVector(this,xName,yName,rc) yptr3d => yptr3d_inter end if else - if (associated(xptr3d)) nullify(xptr3d) - if (associated(yptr3d)) nullify(yptr3d) + nullify(xptr3d) + nullify(yptr3d) end if call ESMF_FieldBundleGet(this%input_bundle,xname,field=xfield,rc=status) @@ -812,14 +832,14 @@ subroutine RegridVector(this,xName,yName,rc) end subroutine RegridVector subroutine stage2DLatLon(this, fileName, oClients, rc) - class (MAPL_GriddedIO), intent(inout) :: this + class (MAPL_GriddedIO), target, intent(inout) :: this character(len=*), intent(in) :: fileName - type (ClientManager), optional, intent(inout) :: oClients + type (ClientManager), optional, target, intent(inout) :: oClients integer, optional, intent(out) :: rc integer :: status real(REAL64), pointer :: ptr2d(:,:) - type(ArrayReference) :: ref + type(ArrayReference), target :: ref class (AbstractGridFactory), pointer :: factory integer, allocatable :: localStart(:),globalStart(:),globalCount(:) logical :: hasll @@ -840,9 +860,11 @@ subroutine stage2DLatLon(this, fileName, oClients, rc) farrayPtr=ptr2d, rc=status) _VERIFY(STATUS) this%lons=ptr2d*MAPL_RADIANS_TO_DEGREES + ref = ArrayReference(this%lons) - call oClients%collective_stage_data(this%write_collection_id,trim(filename),'lons', & - ref,start=localStart, global_start=GlobalStart, global_count=GlobalCount) + call oClients%collective_stage_data(this%write_collection_id,trim(filename),'lons', & + ref,start=localStart, global_start=GlobalStart, global_count=GlobalCount) + call ESMF_GridGetCoord(this%output_grid, localDE=0, coordDim=2, & staggerloc=ESMF_STAGGERLOC_CENTER, & farrayPtr=ptr2d, rc=status) @@ -883,6 +905,7 @@ subroutine stage2DLatLon(this, fileName, oClients, rc) call oClients%collective_stage_data(this%write_collection_id,trim(filename),'corner_lats', & ref,start=localStart, global_start=GlobalStart, global_count=GlobalCount) end if + _RETURN(_SUCCESS) end subroutine stage2DLatLon @@ -936,11 +959,12 @@ subroutine stageData(this, field, fileName, tIndex, oClients, rc) allocate(ptr2d(0,0)) end if ref = factory%generate_file_reference2D(Ptr2D) - allocate(localStart,source=[gridLocalStart,1]) if (tindex > -1) then + allocate(localStart,source=[gridLocalStart,1]) allocate(globalStart,source=[gridGlobalStart,tindex]) allocate(globalCount,source=[gridGlobalCount,1]) else + allocate(localStart,source=[gridLocalStart]) allocate(globalStart,source=gridGlobalStart) allocate(globalCount,source=gridGlobalCount) end if @@ -957,17 +981,19 @@ subroutine stageData(this, field, fileName, tIndex, oClients, rc) allocate(ptr3d(0,0,0)) end if ref = factory%generate_file_reference3D(Ptr3D) - allocate(localStart,source=[gridLocalStart,1,1]) if (tindex > -1) then + allocate(localStart,source=[gridLocalStart,1,1]) allocate(globalStart,source=[gridGlobalStart,1,tindex]) allocate(globalCount,source=[gridGlobalCount,lm,1]) else + allocate(localStart,source=[gridLocalStart,1]) allocate(globalStart,source=[gridGlobalStart,1]) allocate(globalCount,source=[gridGlobalCount,lm]) end if else _FAIL( "Rank not supported") end if + call oClients%collective_stage_data(this%write_collection_id,trim(filename),trim(fieldName), & ref,start=localStart, global_start=GlobalStart, global_count=GlobalCount) _RETURN(_SUCCESS) diff --git a/pfio/AbstractDirectoryService.F90 b/pfio/AbstractDirectoryService.F90 index 2eceb9d4c121..86088b98c0cd 100644 --- a/pfio/AbstractDirectoryService.F90 +++ b/pfio/AbstractDirectoryService.F90 @@ -53,7 +53,7 @@ subroutine connect_to_client(this, port_name, server, rc) import AbstractSocketVector class (AbstractDirectoryService), target, intent(inout) :: this character(*), intent(in) :: port_name - class (BaseServer), intent(inout) :: server + class (BaseServer), target, intent(inout) :: server integer, optional, intent(out) :: rc end subroutine connect_to_client @@ -61,9 +61,9 @@ subroutine publish(this, port, server, rc) use pFIO_BaseServerMod import AbstractDirectoryService import PortInfo - class (AbstractDirectoryService), intent(inout) :: this + class (AbstractDirectoryService), target, intent(inout) :: this type(PortInfo), target, intent(in) :: port - class (BaseServer), intent(inout) :: server + class (BaseServer), intent(in) :: server integer, optional, intent(out) :: rc end subroutine diff --git a/pfio/DirectoryService.F90 b/pfio/DirectoryService.F90 index 325beb523acb..47a4cde92ad6 100644 --- a/pfio/DirectoryService.F90 +++ b/pfio/DirectoryService.F90 @@ -286,7 +286,7 @@ end subroutine connect_to_server subroutine connect_to_client(this, port_name, server, rc) class (DirectoryService), target, intent(inout) :: this character(*), intent(in) :: port_name - class (BaseServer), intent(inout) :: server + class (BaseServer), target, intent(inout) :: server integer, optional, intent(out) :: rc class (AbstractSocket), pointer :: sckt @@ -422,9 +422,9 @@ end subroutine connect_to_client ! But it would allow future implementation to query for servers ! or possibly to allow servers to satisfy multiple clients. subroutine publish(this, port, server, rc) - class (DirectoryService), intent(inout) :: this + class (DirectoryService), target,intent(inout) :: this type(PortInfo),target, intent(in) :: port - class (BaseServer), intent(inout) :: server + class (BaseServer), intent(in) :: server integer, optional, intent(out) :: rc character(len=MAX_LEN_PORT_NAME) :: port_name integer :: ierror diff --git a/pfio/MultiGroupServer.F90 b/pfio/MultiGroupServer.F90 index 9fe61430bb41..abbee7fd73dd 100644 --- a/pfio/MultiGroupServer.F90 +++ b/pfio/MultiGroupServer.F90 @@ -440,7 +440,7 @@ subroutine start_back(this, rc) integer, parameter :: stag = 6782 integer :: status - type (StringSet) :: FilesBeingWritten + type (StringSet), target :: FilesBeingWritten allocate(this%serverthread_done_msgs(1)) this%serverthread_done_msgs(:) = .false. @@ -625,10 +625,10 @@ subroutine start_back_writers(rc) integer, pointer :: g_4d(:,:,:,:), l_4d(:,:,:,:), g_5d(:,:,:,:,:), l_5d(:,:,:,:,:) integer :: d_rank, request_id integer(kind=INT64) :: msize_word, s0, e0, s1, e1, s2, e2, s3, e3, s4, e4, s5, e5 - type (StringAttributeMap) :: vars_map + type (StringAttributeMap), target :: vars_map type (StringAttributeMapIterator) :: var_iter - type (IntegerMessageMap) :: msg_map - type (IntegerMessageMapIterator) :: msg_iter + type (IntegerMessageMap), target :: msg_map + type (IntegerMessageMapIterator) :: msg_iter class (*), pointer :: x_ptr(:) integer , allocatable :: buffer_v(:) diff --git a/pfio/StringVariableMap.F90 b/pfio/StringVariableMap.F90 index bc41c318131e..cf3a67c332d3 100644 --- a/pfio/StringVariableMap.F90 +++ b/pfio/StringVariableMap.F90 @@ -55,8 +55,8 @@ integer function StringVariableMap_get_length(this) result(length) end function StringVariableMap_get_length subroutine StringVariableMap_serialize(map, buffer, rc) - type (StringVariableMap) ,intent(in):: map - integer, allocatable,intent(inout) :: buffer(:) + type (StringVariableMap), target, intent(in):: map + integer, allocatable, intent(inout) :: buffer(:) integer, optional, intent(out) :: rc type (StringVariableMapIterator) :: iter diff --git a/pfio/tests/pfio_ctest_io.F90 b/pfio/tests/pfio_ctest_io.F90 index ff00a0b4795c..df8da3c881eb 100644 --- a/pfio/tests/pfio_ctest_io.F90 +++ b/pfio/tests/pfio_ctest_io.F90 @@ -457,7 +457,7 @@ program main integer :: rank, npes, ierror, provided,required integer :: status, color, key - class(BaseServer),allocatable :: iserver,oserver + class(BaseServer), target, allocatable :: iserver,oserver class(AbstractDirectoryService), allocatable, target :: directory_service type (CommandLineOptions0) :: options diff --git a/pfunit/ESMF_TestCase.F90 b/pfunit/ESMF_TestCase.F90 index 22ea62fa4a8d..c618f2fc99a8 100644 --- a/pfunit/ESMF_TestCase.F90 +++ b/pfunit/ESMF_TestCase.F90 @@ -42,21 +42,31 @@ module ESMF_TestCase_mod recursive subroutine runBare(this) class (ESMF_TestCase), intent(inout) :: this + ! We need an inner procedure to get the TARGET attribute + ! added to the TestCase object so that it can be called back from inside the ESMF + ! gridcomp. Inelegant but it works around the issue where NAG debug flags do + ! a copy-in/copy-out which leaves a dangling pointer in the self reference. + call runbare_inner(this) + end subroutine runBare + + subroutine runbare_inner(this) + class (ESMF_TestCase), target, intent(inout) :: this + logical :: discard type (ESMF_GridComp), target :: gc integer :: rc, userRc integer :: pet - ! Gridded component gc = ESMF_GridCompCreate(petList=[(pet,pet=0,this%getNumPETsRequested()-1)], rc=rc) if (rc /= ESMF_SUCCESS) call throw('Insufficient PETs for request') this%gc => gc this%val = 4 - + call this%setInternalState(gc,rc=rc) if (rc /= ESMF_SUCCESS) call throw('Insufficient PETs for request') + ! create subcommunicator this%context = this%parentContext%makeSubcontext(this%getNumPETsRequested()) @@ -86,9 +96,9 @@ recursive subroutine runBare(this) call gatherExceptions(this%parentContext) call this%clearInternalState(gc, rc=rc) - if (rc /= ESMF_SUCCESS) call throw('Failure in ESMF_GridCompFinalize()') + if (rc /= ESMF_SUCCESS) call throw('Failure clearing internal state') - end subroutine runBare + end subroutine runbare_inner subroutine setInternalState(this, gc, rc) class (ESMF_TestCase), target, intent(inout) :: this @@ -127,11 +137,11 @@ subroutine clearInternalState(this, gc, rc) deallocate(this%wrapped%wrapped) deallocate(this%wrapped) - call ESMF_GridCompDestroy(gc, rc=status) - if (status /= ESMF_SUCCESS) then - rc = status - return - end if +!!$ call ESMF_GridCompDestroy(gc, rc=status) +!!$ if (status /= ESMF_SUCCESS) then +!!$ rc = status +!!$ return +!!$ end if rc = ESMF_SUCCESS end subroutine clearInternalState @@ -161,7 +171,8 @@ subroutine initialize(comp, importState, exportState, clock, rc) end if ! Access private data block and verify data - testPtr => wrap%wrapped%testPtr + testPtr => wrap%wrapped%testPtr + call testPtr%setUp() rc = finalrc @@ -236,7 +247,7 @@ subroutine finalize(comp, importState, exportState, clock, rc) end subroutine finalize - subroutine setServices(comp, rc) + subroutine setServices(comp, rc) type(ESMF_GridComp) :: comp ! must not be optional integer, intent(out) :: rc ! must not be optional diff --git a/pfunit/ESMF_TestMethod.F90 b/pfunit/ESMF_TestMethod.F90 index 499e39d5d72a..2869bb9876fc 100644 --- a/pfunit/ESMF_TestMethod.F90 +++ b/pfunit/ESMF_TestMethod.F90 @@ -7,7 +7,6 @@ module ESMF_TestMethod_mod private public :: ESMF_TestMethod - public :: newESMF_TestMethod type, extends(ESMF_TestCase) :: ESMF_TestMethod procedure(esmfMethod), pointer :: userMethod => null() @@ -26,10 +25,10 @@ subroutine esmfMethod(this) end subroutine esmfMethod end interface - interface newEsmf_TestMethod + interface Esmf_TestMethod module procedure newEsmf_TestMethod_basic module procedure newEsmf_TestMethod_setUpTearDown - end interface newEsmf_TestMethod + end interface Esmf_TestMethod contains diff --git a/shared/MAPL_ErrorHandling.F90 b/shared/MAPL_ErrorHandling.F90 index f2eef1f3447c..1c1707d89a43 100644 --- a/shared/MAPL_ErrorHandling.F90 +++ b/shared/MAPL_ErrorHandling.F90 @@ -240,9 +240,9 @@ logical function MAPL_VRFYt(A,text,iam,line,rc) !$omp end critical (MAPL_ErrorHandling10) end function MAPL_VRFYT - subroutine MAPL_abort + subroutine MAPL_abort() integer :: status - integer :: error_code + integer :: error_code = -1 call MPI_Abort(MPI_COMM_WORLD,error_code,status) end subroutine MAPL_abort