! Copyright Université Reims Champagnne-Ardenne (2010-2015) ! contributor: Jérémie Burgalat ! ! jeremie.burgalat@univ-reims.fr ! ! This software is a computer program whose purpose is to compute multi-variate ! linear interpolation. ! ! This software is governed by the CeCILL-B license under French law and ! abiding by the rules of distribution of free software. You can use, ! modify and/ or redistribute the software under the terms of the CeCILL-B ! license as circulated by CEA, CNRS and INRIA at the following URL ! "http://www.cecill.info". ! ! As a counterpart to the access to the source code and rights to copy, ! modify and redistribute granted by the license, users are provided only ! with a limited warranty and the software's author, the holder of the ! economic rights, and the successive licensors have only limited ! liability. ! ! In this respect, the user's attention is drawn to the risks associated ! with loading, using, modifying and/or developing or reproducing the ! software by the user in light of its specific status of free software, ! that may mean that it is complicated to manipulate, and that also ! therefore means that it is reserved for developers and experienced ! professionals having in-depth computer knowledge. Users are therefore ! encouraged to load and test the software's suitability as regards their ! requirements in conditions enabling the security of their systems and/or ! data to be ensured and, more generally, to use and operate it in the ! same conditions as regards security. ! ! The fact that you are presently reading this means that you have had ! knowledge of the CeCILL-B license and that you accept its terms. !! file: datasets.F90 !! summary: Dataset module definition file !! author: J. Burgalat !! date: 2014 #if HAVE_CONFIG_H #include "config.h" #endif MODULE DATASETS !! dataset definitions module !! !! This module defines simple derived types that encapsulate data set variables. For a N-dimension !! dataset, N vectors of \(n_{i}\) elements and a single N-dimensional array of !! \(\prod_{i}^{N} n_{i}\) elements are defined. !! !! If the data set is to be used within [[lintdset(module)]] module then each coordinate values of the !! dataset must be sorted either in ascending or descending order with no duplicates. The module does !! not provide functionnality to sort and to check such requirements. !! !! The module also provides two interfaces to initialize data sets from input data file which can be !! a NetCDF file (if NetCDF support is enabled at compilation time) or an ASCII file. In the latter !! case, the file must contain a header which is formatted as follows: !! !! - The first line must contain only one value which is the number of coordinates (__N__) !! - The second line must contain all the dimension sizes (that is __N__ values) !! - Each other lines should contain __N+1__ columns with, for the __N__ first columns, the !! coordinates values and finally the point value in the last column. !! !! @note !! Note that for ASCII file, data must be ordered so first dimensions vary first. Same requirement !! is needed for NetCDF file but in most cases, it is implicitly done (if dimensions are ordered). USE LINT_PREC #if HAVE_NC_FTN USE NETCDF #endif IMPLICIT NONE PRIVATE PUBLIC :: read_dset, clear_dset, is_in, debug LOGICAL :: debug = .false. !! A control flag to enable verbose mode #if HAVE_NC_FTN LOGICAL, PUBLIC, PARAMETER ::nc_supported = .true. !! NetCDF input files are supported #else LOGICAL, PUBLIC, PARAMETER ::nc_supported = .false. !! NetCDF input files are not supported #endif !> Initialize a data set from either an ASCII or a NetCDF file !! !! Whatever the kind of file, this interface assumes that you know the dimensionnality of the !! extracted variable. If file content is not compatible with the kind of data set given as !! output argument, or if some I/O error occured, the dataset is cleared. !! @note !! Netcdf reader interface is available only if the library has been compiled with NetCDF support. INTERFACE read_dset #if HAVE_NC_FTN MODULE PROCEDURE ncdf_rd_1d,ncdf_rd_2d,ncdf_rd_3d,ncdf_rd_4d,ncdf_rd_5d #if HAVE_NC4_FTN MODULE PROCEDURE ncdf4_rd_1d,ncdf4_rd_2d,ncdf4_rd_3d,ncdf4_rd_4d,ncdf4_rd_5d #endif #endif MODULE PROCEDURE ascii_rd_1d,ascii_rd_2d,ascii_rd_3d,ascii_rd_4d,ascii_rd_5d END INTERFACE !> Clear the given data set INTERFACE clear_dset MODULE PROCEDURE clr_1d_set,clr_2d_set,clr_3d_set,clr_4d_set,clr_5d_set END INTERFACE !> Check if given point is within the data set INTERFACE is_in MODULE PROCEDURE is_in_1d,is_in_2d,is_in_3d,is_in_4d,is_in_5d END INTERFACE !> Private interface to netcdf informations getters #if HAVE_NC_FTN INTERFACE get_nc_info MODULE PROCEDURE get_nc3_info #if HAVE_NC4_FTN MODULE PROCEDURE get_nc4_info #endif END INTERFACE #endif TYPE, PUBLIC :: DSET1D !! A 1D data set REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: x !! X coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: data !! Tabulated function's value at each coordinate END TYPE DSET1D TYPE, PUBLIC :: DSET2D !! A 2D data set REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: x !! X coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: y !! Y coordinate tabulated values REAL(kind=wp), DIMENSION(:,:), ALLOCATABLE :: data !! Tabulated function's value at each coordinate END TYPE DSET2D TYPE, PUBLIC :: DSET3D !! A 3D data set REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: x !! X coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: y !! Y coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: z !! Z coordinate tabulated values REAL(kind=wp), DIMENSION(:,:,:), ALLOCATABLE :: data !! Tabulated function's value at each coordinate END TYPE DSET3D TYPE, PUBLIC :: DSET4D !! A 4D data set REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: x !! X coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: y !! Y coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: z !! Z coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: t !! T coordinate tabulated values REAL(kind=wp), DIMENSION(:,:,:,:), ALLOCATABLE :: data !! Tabulated function's value at each coordinate END TYPE DSET4D TYPE, PUBLIC :: DSET5D !! A 5D data set REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: x !! X coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: y !! Y coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: z !! Z coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: t !! T coordinate tabulated values REAL(kind=wp), DIMENSION(:), ALLOCATABLE :: w !! W coordinate tabulated values REAL(kind=wp), DIMENSION(:,:,:,:,:), ALLOCATABLE :: data !! Tabulated function's value at each coordinate END TYPE DSET5D CONTAINS FUNCTION is_in_1d(set,x) RESULT(ret) !! Check if point is in the 1D data set TYPE(DSET1D), INTENT(in) :: set !! Dataset object to search in REAL(kind=wp), INTENT(in) :: x !! coordinate of the point to check LOGICAL :: ret !! .true. if the point is in the data set, .false. otherwise REAL(kind=wp) :: l,u ret=.true. l = set%x(1) ; u= set%x(size(set%x)) IF ((x>=l.EQV.x<=u).OR.(x<=l.EQV.x>=u)) ret=.false. RETURN END FUNCTION is_in_1d FUNCTION is_in_2d(set,x,y) RESULT(ret) !! Check if point is in the 2D data set TYPE(DSET2D), INTENT(in) :: set !! Dataset object to search in REAL(kind=wp), INTENT(in) :: x !! X coordinate of the point to check REAL(kind=wp), INTENT(in) :: y !! Y coordinate of the point to check LOGICAL :: ret !! .true. if the point is in the data set, .false. otherwise REAL(kind=wp) :: l,u ret=.false. l = set%x(1) ; u= set%x(size(set%x)) IF ((x>l.AND.x>u).OR.(xl.AND.x>u).OR.(xl.AND.y>u).OR.(yl.AND.x>u).OR.(xl.AND.y>u).OR.(yl.AND.z>u).OR.(zl.AND.x>u).OR.(xl.AND.y>u).OR.(yl.AND.z>u).OR.(zl.AND.t>u).OR.(tl.AND.x>u).OR.(xl.AND.y>u).OR.(yl.AND.z>u).OR.(zl.AND.t>u).OR.(tl.AND.w>u).OR.(w ds(1)) cc(1)=1 IF (mod(i,dp(2))==0) cc(2) = cc(2)+1 ; IF (cc(2) > ds(2)) cc(2)=1 ENDDO READ(666,*,iostat=e) vl(1) IF (e == 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Extra value found' ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF ret = .true. CLOSE(666) RETURN END FUNCTION ascii_rd_2d FUNCTION ascii_rd_3d(path,set) RESULT(ret) !! Read a 3D data set from a ASCII file CHARACTER(len=*), INTENT(in) :: path !! Path of the ASCII file to read TYPE(dset3d), INTENT(out) :: set !! output 3D dataset LOGICAL :: ret !! .true. if no error occured, .false. otherwise INTEGER :: i,e REAL(kind=wp), DIMENSION(4) :: vl INTEGER, DIMENSION(3) :: cc,ds,dp ret = .false. CALL clear_dset(set) IF (.NOT.ascii_header(path,ds,dp)) RETURN ALLOCATE(set%x(ds(1)),set%y(ds(2)),set%z(ds(3)), & set%data(ds(1),ds(2),ds(3))) cc(:) = 1 DO i=1,PRODUCT(ds) READ(666,*,iostat=e) vl ! Reads the line IF (e /= 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Malformed file, line: ',i+2 ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF ! --- X coordinate set%x(cc(1)) = vl(1) ! --- Y coordinate set%y(cc(2)) = vl(2) ! --- Z coordinate set%z(cc(3)) = vl(3) ! --- Data set%data(cc(1),cc(2),cc(3)) = vl(4) ! - Update counters IF (mod(i,dp(1))==0) cc(1) = cc(1)+1 ; IF (cc(1) > ds(1)) cc(1)=1 IF (mod(i,dp(2))==0) cc(2) = cc(2)+1 ; IF (cc(2) > ds(2)) cc(2)=1 IF (mod(i,dp(3))==0) cc(3) = cc(3)+1 ; IF (cc(3) > ds(3)) cc(3)=1 ENDDO READ(666,*,iostat=e) vl(1) IF (e == 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Extra value found' ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF ret = .true. CLOSE(666) RETURN END FUNCTION ascii_rd_3d FUNCTION ascii_rd_4d(path,set) RESULT(ret) !! Read a 4D data set from a ASCII file CHARACTER(len=*), INTENT(in) :: path !! Path of the ASCII file to read TYPE(dset4d), INTENT(out) :: set !! output 4D dataset LOGICAL :: ret !! .true. if no error occured, .false. otherwise INTEGER :: i,e REAL(kind=wp), DIMENSION(5) :: vl INTEGER, DIMENSION(4) :: cc,ds,dp ret = .false. CALL clear_dset(set) IF (.NOT.ascii_header(path,ds,dp)) RETURN ALLOCATE(set%x(ds(1)),set%y(ds(2)),set%z(ds(3)),set%t(ds(4)), & set%data(ds(1),ds(2),ds(3),ds(4))) cc(:) = 1 DO i=1,PRODUCT(ds) READ(666,*,iostat=e) vl ! Reads the line IF (e /= 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Malformed file, line: ',i+2 ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF ! --- X coordinate set%x(cc(1)) = vl(1) ! --- Y coordinate set%y(cc(2)) = vl(2) ! --- Z coordinate set%z(cc(3)) = vl(3) ! --- T coordinate set%t(cc(4)) = vl(4) ! --- Data set%data(cc(1),cc(2),cc(3),cc(4)) = vl(5) ! - Update counters IF (mod(i,dp(1))==0) cc(1) = cc(1)+1 ; IF (cc(1) > ds(1)) cc(1)=1 IF (mod(i,dp(2))==0) cc(2) = cc(2)+1 ; IF (cc(2) > ds(2)) cc(2)=1 IF (mod(i,dp(3))==0) cc(3) = cc(3)+1 ; IF (cc(3) > ds(3)) cc(3)=1 IF (mod(i,dp(4))==0) cc(4) = cc(4)+1 ; IF (cc(4) > ds(4)) cc(4)=1 ENDDO READ(666,*,iostat=e) vl(1) IF (e == 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Extra value found' ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF CLOSE(666) ret = .true. RETURN END FUNCTION ascii_rd_4d FUNCTION ascii_rd_5d(path,set) RESULT(ret) !! Read a 5D data set from a ASCII file CHARACTER(len=*), INTENT(in) :: path !! Path of the ASCII file to read TYPE(dset5d), INTENT(out) :: set !! output 5D dataset LOGICAL :: ret !! .true. if no error occured, .false. otherwise INTEGER :: i,e REAL(kind=wp), DIMENSION(6) :: vl INTEGER, DIMENSION(5) :: cc,ds,dp ret = .false. CALL clear_dset(set) IF (.NOT.ascii_header(path,ds,dp)) RETURN ALLOCATE(set%x(ds(1)),set%y(ds(2)),set%z(ds(3)),set%t(ds(4)),set%w(ds(5)), & set%data(ds(1),ds(2),ds(3),ds(4),ds(5))) cc(:) = 1 DO i=1,PRODUCT(ds) READ(666,*,iostat=e) vl ! Reads the line IF (e /= 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Malformed file, line: ',i+2 ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF ! --- X coordinate set%x(cc(1)) = vl(1) ! --- Y coordinate set%y(cc(2)) = vl(2) ! --- Z coordinate set%z(cc(3)) = vl(3) ! --- T coordinate set%t(cc(4)) = vl(4) ! --- W coordinate set%w(cc(5)) = vl(5) ! --- Data set%data(cc(1),cc(2),cc(3),cc(4),cc(5)) = vl(6) ! - Update counters IF (mod(i,dp(1)) == 0) cc(1)=cc(1)+1 ; IF (cc(1) > ds(1)) cc(1)=1 IF (mod(i,dp(2)) == 0) cc(2)=cc(2)+1 ; IF (cc(2) > ds(2)) cc(2)=1 IF (mod(i,dp(3)) == 0) cc(3)=cc(3)+1 ; IF (cc(3) > ds(3)) cc(3)=1 IF (mod(i,dp(4)) == 0) cc(4)=cc(4)+1 ; IF (cc(4) > ds(4)) cc(4)=1 IF (mod(i,dp(5)) == 0) cc(5)=cc(5)+1 ; IF (cc(5) > ds(5)) cc(5)=1 ENDDO READ(666,*,iostat=e) vl(1) IF (e == 0) THEN IF (debug) WRITE(*,'(a)') 'ERROR: Extra value found' ret = .false. ; call clear_dset(set) ; CLOSE(666) ; RETURN ENDIF CLOSE(666) ret = .true. RETURN END FUNCTION ascii_rd_5d SUBROUTINE clr_1d_set(set) !! Clear the given 1D data set TYPE(DSET1D), INTENT(inout) :: set !! dataset object to clear IF (ALLOCATED(set%data)) DEALLOCATE(set%data) IF (ALLOCATED(set%x)) DEALLOCATE(set%x) RETURN END SUBROUTINE clr_1d_set SUBROUTINE clr_2d_set(set) !! Clear the given 2D data set TYPE(DSET2D), INTENT(inout) :: set !! dataset object to clear IF (ALLOCATED(set%data)) DEALLOCATE(set%data) IF (ALLOCATED(set%x)) DEALLOCATE(set%x) IF (ALLOCATED(set%y)) DEALLOCATE(set%y) RETURN END SUBROUTINE clr_2d_set SUBROUTINE clr_3d_set(set) !! Clear the given 3D data set TYPE(DSET3D), INTENT(inout) :: set !! dataset object to clear IF (ALLOCATED(set%data)) DEALLOCATE(set%data) IF (ALLOCATED(set%x)) DEALLOCATE(set%x) IF (ALLOCATED(set%y)) DEALLOCATE(set%y) IF (ALLOCATED(set%z)) DEALLOCATE(set%z) RETURN END SUBROUTINE clr_3d_set SUBROUTINE clr_4d_set(set) !! Clear the given 4D data set TYPE(DSET4D), INTENT(inout) :: set !! dataset object to clear IF (ALLOCATED(set%data)) DEALLOCATE(set%data) IF (ALLOCATED(set%x)) DEALLOCATE(set%x) IF (ALLOCATED(set%y)) DEALLOCATE(set%y) IF (ALLOCATED(set%z)) DEALLOCATE(set%z) IF (ALLOCATED(set%t)) DEALLOCATE(set%t) RETURN END SUBROUTINE clr_4d_set SUBROUTINE clr_5d_set(set) !! Clear the given 5D data set TYPE(DSET5D), INTENT(inout) :: set !! dataset object to clear IF (ALLOCATED(set%data)) DEALLOCATE(set%data) IF (ALLOCATED(set%x)) DEALLOCATE(set%x) IF (ALLOCATED(set%y)) DEALLOCATE(set%y) IF (ALLOCATED(set%z)) DEALLOCATE(set%z) IF (ALLOCATED(set%t)) DEALLOCATE(set%t) IF (ALLOCATED(set%w)) DEALLOCATE(set%w) RETURN END SUBROUTINE clr_5d_set END MODULE DATASETS