- Timestamp:
- Aug 7, 2019 9:42:31 AM (5 years ago)
- Location:
- palm/trunk/SOURCE
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
palm/trunk/SOURCE/data_output_binary_module.f90
r4141 r4147 25 25 ! ----------------- 26 26 ! $Id$ 27 ! corrected indentation according to coding standard 28 ! 29 ! 4141 2019-08-05 12:24:51Z gronemeier 27 30 ! Initial revision 28 31 ! … … 38 41 !> @todo Get iostat value of write statements. 39 42 !--------------------------------------------------------------------------------------------------! 40 MODULE data_output_binary_module41 42 USE kinds43 MODULE data_output_binary_module 44 45 USE kinds 43 46 44 47 #if defined( __parallel ) 45 48 #if defined( __mpifh ) 46 INCLUDE "mpif.h"49 INCLUDE "mpif.h" 47 50 #else 48 USE MPI51 USE MPI 49 52 #endif 50 53 #endif 51 54 52 IMPLICIT NONE53 54 INTEGER, PARAMETER :: charlen = 100 !< maximum length of character variables55 56 CHARACTER(LEN=*), PARAMETER :: config_file_name = 'BINARY_TO_NETCDF_CONFIG' !< name of config file57 CHARACTER(LEN=*), PARAMETER :: mode_binary = 'binary' !< string to select operation mode of module58 CHARACTER(LEN=*), PARAMETER :: file_prefix = 'BIN_' !< file prefix for binary files59 60 CHARACTER(LEN=charlen) :: file_suffix = '' !< file suffix added to each file name61 CHARACTER(LEN=800) :: internal_error_message = '' !< string containing the last error message62 CHARACTER(LEN=800) :: temp_string !< dummy string63 64 INTEGER :: binary_file_lowest_unit = 1000 !< lowest unit number of all binary files created by this module65 INTEGER :: config_file_unit !< unit number of config file66 INTEGER :: debug_output_unit !< Fortran Unit Number of the debug-output file67 INTEGER :: global_id_in_file = -1 !< value of global ID within a file68 INTEGER :: master_rank !< master rank for tasks to be executed by single PE only69 INTEGER :: next_available_unit !< next unit number available for new file70 INTEGER :: output_group_comm !< MPI communicator addressing all MPI ranks which participate in output71 72 INTEGER, DIMENSION(:), ALLOCATABLE :: files_highest_variable_id !< highest assigned ID of variable or dimension in a file73 74 LOGICAL :: binary_open_file_first_call = .TRUE. !< true if binary_open_file routine was not called yet75 LOGICAL :: config_file_open = .FALSE. !< true if config file is opened and not closed76 LOGICAL :: print_debug_output = .FALSE. !< if true, debug output is printed77 78 SAVE79 80 PRIVATE81 82 INTERFACE binary_init_module83 MODULE PROCEDURE binary_init_module84 END INTERFACE binary_init_module85 86 INTERFACE binary_open_file87 MODULE PROCEDURE binary_open_file88 END INTERFACE binary_open_file89 90 INTERFACE binary_init_dimension91 MODULE PROCEDURE binary_init_dimension92 END INTERFACE binary_init_dimension93 94 INTERFACE binary_init_variable95 MODULE PROCEDURE binary_init_variable96 END INTERFACE binary_init_variable97 98 INTERFACE binary_write_attribute99 MODULE PROCEDURE binary_write_attribute100 END INTERFACE binary_write_attribute101 102 INTERFACE binary_stop_file_header_definition103 MODULE PROCEDURE binary_stop_file_header_definition104 END INTERFACE binary_stop_file_header_definition105 106 INTERFACE binary_write_variable107 MODULE PROCEDURE binary_write_variable108 END INTERFACE binary_write_variable109 110 INTERFACE binary_finalize111 MODULE PROCEDURE binary_finalize112 END INTERFACE binary_finalize113 114 INTERFACE binary_get_error_message115 MODULE PROCEDURE binary_get_error_message116 END INTERFACE binary_get_error_message117 118 PUBLIC &119 binary_finalize, &120 binary_get_error_message, &121 binary_init_dimension, &122 binary_stop_file_header_definition, &123 binary_init_module, &124 binary_init_variable, &125 binary_open_file, &126 binary_write_attribute, &127 binary_write_variable128 129 130 CONTAINS55 IMPLICIT NONE 56 57 INTEGER, PARAMETER :: charlen = 100 !< maximum length of character variables 58 59 CHARACTER(LEN=*), PARAMETER :: config_file_name = 'BINARY_TO_NETCDF_CONFIG' !< name of config file 60 CHARACTER(LEN=*), PARAMETER :: mode_binary = 'binary' !< string to select operation mode of module 61 CHARACTER(LEN=*), PARAMETER :: file_prefix = 'BIN_' !< file prefix for binary files 62 63 CHARACTER(LEN=charlen) :: file_suffix = '' !< file suffix added to each file name 64 CHARACTER(LEN=800) :: internal_error_message = '' !< string containing the last error message 65 CHARACTER(LEN=800) :: temp_string !< dummy string 66 67 INTEGER :: binary_file_lowest_unit = 1000 !< lowest unit number of all binary files created by this module 68 INTEGER :: config_file_unit !< unit number of config file 69 INTEGER :: debug_output_unit !< Fortran Unit Number of the debug-output file 70 INTEGER :: global_id_in_file = -1 !< value of global ID within a file 71 INTEGER :: master_rank !< master rank for tasks to be executed by single PE only 72 INTEGER :: next_available_unit !< next unit number available for new file 73 INTEGER :: output_group_comm !< MPI communicator addressing all MPI ranks which participate in output 74 75 INTEGER, DIMENSION(:), ALLOCATABLE :: files_highest_variable_id !< highest assigned ID of variable or dimension in a file 76 77 LOGICAL :: binary_open_file_first_call = .TRUE. !< true if binary_open_file routine was not called yet 78 LOGICAL :: config_file_open = .FALSE. !< true if config file is opened and not closed 79 LOGICAL :: print_debug_output = .FALSE. !< if true, debug output is printed 80 81 SAVE 82 83 PRIVATE 84 85 INTERFACE binary_init_module 86 MODULE PROCEDURE binary_init_module 87 END INTERFACE binary_init_module 88 89 INTERFACE binary_open_file 90 MODULE PROCEDURE binary_open_file 91 END INTERFACE binary_open_file 92 93 INTERFACE binary_init_dimension 94 MODULE PROCEDURE binary_init_dimension 95 END INTERFACE binary_init_dimension 96 97 INTERFACE binary_init_variable 98 MODULE PROCEDURE binary_init_variable 99 END INTERFACE binary_init_variable 100 101 INTERFACE binary_write_attribute 102 MODULE PROCEDURE binary_write_attribute 103 END INTERFACE binary_write_attribute 104 105 INTERFACE binary_stop_file_header_definition 106 MODULE PROCEDURE binary_stop_file_header_definition 107 END INTERFACE binary_stop_file_header_definition 108 109 INTERFACE binary_write_variable 110 MODULE PROCEDURE binary_write_variable 111 END INTERFACE binary_write_variable 112 113 INTERFACE binary_finalize 114 MODULE PROCEDURE binary_finalize 115 END INTERFACE binary_finalize 116 117 INTERFACE binary_get_error_message 118 MODULE PROCEDURE binary_get_error_message 119 END INTERFACE binary_get_error_message 120 121 PUBLIC & 122 binary_finalize, & 123 binary_get_error_message, & 124 binary_init_dimension, & 125 binary_stop_file_header_definition, & 126 binary_init_module, & 127 binary_init_variable, & 128 binary_open_file, & 129 binary_write_attribute, & 130 binary_write_variable 131 132 133 CONTAINS 131 134 132 135 … … 136 139 !> Initialize data-output module. 137 140 !--------------------------------------------------------------------------------------------------! 138 SUBROUTINE binary_init_module( file_suffix_of_output_group, mpi_comm_of_output_group, &139 master_output_rank, &140 program_debug_output_unit, debug_output, dom_global_id )141 142 CHARACTER(LEN=*), INTENT(IN) :: file_suffix_of_output_group !> file-name suffix added to each file;143 !> must be unique for each output group144 145 INTEGER, INTENT(IN) :: dom_global_id !< global id within a file defined by DOM146 INTEGER, INTENT(IN) :: master_output_rank !< MPI rank executing tasks which must be executed by a single PE147 INTEGER, INTENT(IN) :: mpi_comm_of_output_group !< MPI communicator specifying the rank group participating in output148 INTEGER, INTENT(IN) :: program_debug_output_unit !< file unit number for debug output149 150 LOGICAL, INTENT(IN) :: debug_output !< if true, debug output is printed151 152 153 file_suffix = file_suffix_of_output_group154 output_group_comm = mpi_comm_of_output_group155 master_rank = master_output_rank156 157 debug_output_unit = program_debug_output_unit158 print_debug_output = debug_output159 160 global_id_in_file = dom_global_id161 162 END SUBROUTINE binary_init_module141 SUBROUTINE binary_init_module( file_suffix_of_output_group, mpi_comm_of_output_group, & 142 master_output_rank, & 143 program_debug_output_unit, debug_output, dom_global_id ) 144 145 CHARACTER(LEN=*), INTENT(IN) :: file_suffix_of_output_group !> file-name suffix added to each file; 146 !> must be unique for each output group 147 148 INTEGER, INTENT(IN) :: dom_global_id !< global id within a file defined by DOM 149 INTEGER, INTENT(IN) :: master_output_rank !< MPI rank executing tasks which must be executed by a single PE 150 INTEGER, INTENT(IN) :: mpi_comm_of_output_group !< MPI communicator specifying the rank group participating in output 151 INTEGER, INTENT(IN) :: program_debug_output_unit !< file unit number for debug output 152 153 LOGICAL, INTENT(IN) :: debug_output !< if true, debug output is printed 154 155 156 file_suffix = file_suffix_of_output_group 157 output_group_comm = mpi_comm_of_output_group 158 master_rank = master_output_rank 159 160 debug_output_unit = program_debug_output_unit 161 print_debug_output = debug_output 162 163 global_id_in_file = dom_global_id 164 165 END SUBROUTINE binary_init_module 163 166 164 167 !--------------------------------------------------------------------------------------------------! … … 167 170 !> Open binary file. 168 171 !--------------------------------------------------------------------------------------------------! 169 SUBROUTINE binary_open_file( mode, file_name, file_id, return_value )170 171 CHARACTER(LEN=charlen) :: bin_filename = '' !< actual name of binary file172 CHARACTER(LEN=charlen), INTENT(IN) :: file_name !< name of file173 CHARACTER(LEN=7) :: my_rank_char !< string containing value of my_rank with leading zeros174 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode175 176 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_open_file' !< name of this routine177 178 INTEGER, INTENT(OUT) :: file_id !< file ID179 INTEGER :: my_rank !< MPI rank of local processor180 INTEGER :: nranks !< number of MPI ranks participating in output181 INTEGER, INTENT(OUT) :: return_value !< return value182 183 INTEGER, DIMENSION(:), ALLOCATABLE :: files_highest_variable_id_tmp !< temporary list of given variable IDs in file184 185 LOGICAL :: file_exists !< true if file to be opened already exists186 187 188 return_value = 0172 SUBROUTINE binary_open_file( mode, file_name, file_id, return_value ) 173 174 CHARACTER(LEN=charlen) :: bin_filename = '' !< actual name of binary file 175 CHARACTER(LEN=charlen), INTENT(IN) :: file_name !< name of file 176 CHARACTER(LEN=7) :: my_rank_char !< string containing value of my_rank with leading zeros 177 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode 178 179 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_open_file' !< name of this routine 180 181 INTEGER, INTENT(OUT) :: file_id !< file ID 182 INTEGER :: my_rank !< MPI rank of local processor 183 INTEGER :: nranks !< number of MPI ranks participating in output 184 INTEGER, INTENT(OUT) :: return_value !< return value 185 186 INTEGER, DIMENSION(:), ALLOCATABLE :: files_highest_variable_id_tmp !< temporary list of given variable IDs in file 187 188 LOGICAL :: file_exists !< true if file to be opened already exists 189 190 191 return_value = 0 189 192 190 193 #if defined( __parallel ) 191 CALL MPI_COMM_SIZE( output_group_comm, nranks, return_value )192 IF ( return_value == 0 ) CALL MPI_COMM_RANK( output_group_comm, my_rank, return_value )193 IF ( return_value == 0 ) THEN194 WRITE( my_rank_char, '("_",I6.6)' ) my_rank195 ELSE196 CALL internal_message( 'error', routine_name // ': MPI error' )197 ENDIF194 CALL MPI_COMM_SIZE( output_group_comm, nranks, return_value ) 195 IF ( return_value == 0 ) CALL MPI_COMM_RANK( output_group_comm, my_rank, return_value ) 196 IF ( return_value == 0 ) THEN 197 WRITE( my_rank_char, '("_",I6.6)' ) my_rank 198 ELSE 199 CALL internal_message( 'error', routine_name // ': MPI error' ) 200 ENDIF 198 201 #else 199 nranks = 1200 my_rank = master_rank201 WRITE( my_rank_char, '("_",I6.6)' ) my_rank202 nranks = 1 203 my_rank = master_rank 204 WRITE( my_rank_char, '("_",I6.6)' ) my_rank 202 205 #endif 203 204 205 IF ( TRIM( mode ) == mode_binary ) CONTINUE206 207 208 IF ( return_value == 0 .AND. binary_open_file_first_call ) THEN209 210 binary_open_file_first_call = .FALSE.211 config_file_unit = binary_file_lowest_unit212 213 IF ( my_rank == master_rank ) THEN214 215 !--Remove any pre-existing file216 INQUIRE( FILE=TRIM( config_file_name ) // TRIM( file_suffix ), &217 EXIST=file_exists )218 219 IF ( file_exists ) THEN220 CALL internal_message( 'debug', routine_name // &221 ': Remove existing file ' // &222 TRIM( config_file_name ) // TRIM( file_suffix ) )223 !> @note Fortran2008 feature 'EXECUTE_COMMAND_LINE' not yet supported by224 !> PGI 18.10 compiler. Hence, non-standard 'SYSTEM' call must be used225 ! CALL EXECUTE_COMMAND_LINE( &226 ! COMMAND='rm ' // TRIM( config_file_name ) // TRIM( file_suffix ), &227 ! WAIT=.TRUE., EXITSTAT=return_value )228 CALL SYSTEM( 'rm ' // TRIM( config_file_name ) // TRIM( file_suffix ) )229 ENDIF230 231 OPEN( config_file_unit, FILE=TRIM( config_file_name ) // TRIM( file_suffix ), &232 FORM='UNFORMATTED', STATUS='NEW', IOSTAT=return_value )233 234 IF ( return_value == 0 ) THEN235 236 config_file_open = .TRUE.237 238 !--Write some general information to config file239 WRITE( config_file_unit ) nranks240 WRITE( config_file_unit ) master_rank241 WRITE( config_file_unit ) LEN( file_prefix )242 WRITE( config_file_unit ) file_prefix243 WRITE( config_file_unit ) charlen244 WRITE( config_file_unit ) global_id_in_file245 246 ELSE247 248 return_value = 1249 CALL internal_message( 'error', routine_name // ': could not create config' )250 251 ENDIF252 253 ENDIF254 255 next_available_unit = binary_file_lowest_unit + 1256 257 ENDIF258 259 260 IF ( return_value == 0 ) THEN261 262 bin_filename = file_prefix // TRIM( file_name ) // TRIM( file_suffix ) // my_rank_char263 264 !--Remove any pre-existing file265 INQUIRE( FILE=TRIM( bin_filename ), EXIST=file_exists )266 267 IF ( file_exists ) THEN268 CALL internal_message( 'debug', routine_name // &269 ': remove existing file ' // TRIM( bin_filename ) )270 !> @note Fortran2008 feature 'EXECUTE_COMMAND_LINE' not yet supported by271 !> PGI 18.10 compiler. Hence, non-standard 'SYSTEM' call must be used272 ! CALL EXECUTE_COMMAND_LINE( COMMAND='rm ' // TRIM( bin_filename ), &273 ! WAIT=.TRUE., EXITSTAT=return_value )274 CALL SYSTEM( 'rm ' // TRIM( bin_filename ) )275 ENDIF276 277 !--Open binary file278 CALL internal_message( 'debug', routine_name // ': open file ' // TRIM( bin_filename ) )279 OPEN ( next_available_unit, FILE=TRIM( bin_filename ), &280 FORM='UNFORMATTED', STATUS='NEW', IOSTAT=return_value )281 282 IF ( return_value == 0 ) THEN283 284 !--Add file_name to config file285 IF ( my_rank == master_rank ) THEN286 WRITE( config_file_unit ) file_name287 ENDIF288 289 !--Save file ID and increase next file unit number290 file_id = next_available_unit291 next_available_unit = next_available_unit + 1292 293 !--Write some meta data to file294 WRITE ( file_id ) charlen295 WRITE ( file_id ) file_id296 WRITE ( file_id ) file_name297 298 !--Extend file-variable/dimension-ID list by 1 and set it to 0 for new file.299 IF ( ALLOCATED( files_highest_variable_id ) ) THEN300 ALLOCATE( files_highest_variable_id_tmp(SIZE( files_highest_variable_id )) )301 files_highest_variable_id_tmp = files_highest_variable_id302 DEALLOCATE( files_highest_variable_id )303 ALLOCATE( files_highest_variable_id(binary_file_lowest_unit+1:file_id) )304 files_highest_variable_id(:file_id-1) = files_highest_variable_id_tmp305 DEALLOCATE( files_highest_variable_id_tmp )306 ELSE307 ALLOCATE( files_highest_variable_id(binary_file_lowest_unit+1:file_id) )308 ENDIF309 files_highest_variable_id(file_id) = 0310 311 ELSE312 return_value = 1313 CALL internal_message( 'error', routine_name // &314 ': could not open file "' // TRIM( file_name ) // '"')315 ENDIF316 317 ENDIF318 319 END SUBROUTINE binary_open_file206 ! 207 !-- Check mode (not required, added for compatibility reasons) 208 IF ( TRIM( mode ) == mode_binary ) CONTINUE 209 ! 210 !-- Open binary config file for combining script 211 IF ( return_value == 0 .AND. binary_open_file_first_call ) THEN 212 213 binary_open_file_first_call = .FALSE. 214 config_file_unit = binary_file_lowest_unit 215 216 IF ( my_rank == master_rank ) THEN 217 ! 218 !-- Remove any pre-existing file 219 INQUIRE( FILE=TRIM( config_file_name ) // TRIM( file_suffix ), & 220 EXIST=file_exists ) 221 222 IF ( file_exists ) THEN 223 CALL internal_message( 'debug', routine_name // & 224 ': Remove existing file ' // & 225 TRIM( config_file_name ) // TRIM( file_suffix ) ) 226 !> @note Fortran2008 feature 'EXECUTE_COMMAND_LINE' not yet supported by 227 !> PGI 18.10 compiler. Hence, non-standard 'SYSTEM' call must be used 228 ! CALL EXECUTE_COMMAND_LINE( & 229 ! COMMAND='rm ' // TRIM( config_file_name ) // TRIM( file_suffix ), & 230 ! WAIT=.TRUE., EXITSTAT=return_value ) 231 CALL SYSTEM( 'rm ' // TRIM( config_file_name ) // TRIM( file_suffix ) ) 232 ENDIF 233 234 OPEN( config_file_unit, FILE=TRIM( config_file_name ) // TRIM( file_suffix ), & 235 FORM='UNFORMATTED', STATUS='NEW', IOSTAT=return_value ) 236 237 IF ( return_value == 0 ) THEN 238 239 config_file_open = .TRUE. 240 ! 241 !-- Write some general information to config file 242 WRITE( config_file_unit ) nranks 243 WRITE( config_file_unit ) master_rank 244 WRITE( config_file_unit ) LEN( file_prefix ) 245 WRITE( config_file_unit ) file_prefix 246 WRITE( config_file_unit ) charlen 247 WRITE( config_file_unit ) global_id_in_file 248 249 ELSE 250 251 return_value = 1 252 CALL internal_message( 'error', routine_name // ': could not create config' ) 253 254 ENDIF 255 256 ENDIF 257 258 next_available_unit = binary_file_lowest_unit + 1 259 260 ENDIF 261 ! 262 !-- Initialize output file: open, write header, initialize variable/dimension IDs 263 IF ( return_value == 0 ) THEN 264 265 bin_filename = file_prefix // TRIM( file_name ) // TRIM( file_suffix ) // my_rank_char 266 ! 267 !-- Remove any pre-existing file 268 INQUIRE( FILE=TRIM( bin_filename ), EXIST=file_exists ) 269 270 IF ( file_exists ) THEN 271 CALL internal_message( 'debug', routine_name // & 272 ': remove existing file ' // TRIM( bin_filename ) ) 273 !> @note Fortran2008 feature 'EXECUTE_COMMAND_LINE' not yet supported by 274 !> PGI 18.10 compiler. Hence, non-standard 'SYSTEM' call must be used 275 ! CALL EXECUTE_COMMAND_LINE( COMMAND='rm ' // TRIM( bin_filename ), & 276 ! WAIT=.TRUE., EXITSTAT=return_value ) 277 CALL SYSTEM( 'rm ' // TRIM( bin_filename ) ) 278 ENDIF 279 ! 280 !-- Open binary file 281 CALL internal_message( 'debug', routine_name // ': open file ' // TRIM( bin_filename ) ) 282 OPEN ( next_available_unit, FILE=TRIM( bin_filename ), & 283 FORM='UNFORMATTED', STATUS='NEW', IOSTAT=return_value ) 284 285 IF ( return_value == 0 ) THEN 286 ! 287 !-- Add file_name to config file 288 IF ( my_rank == master_rank ) THEN 289 WRITE( config_file_unit ) file_name 290 ENDIF 291 ! 292 !-- Save file ID and increase next file unit number 293 file_id = next_available_unit 294 next_available_unit = next_available_unit + 1 295 ! 296 !-- Write some meta data to file 297 WRITE ( file_id ) charlen 298 WRITE ( file_id ) file_id 299 WRITE ( file_id ) file_name 300 ! 301 !-- Extend file-variable/dimension-ID list by 1 and set it to 0 for new file. 302 IF ( ALLOCATED( files_highest_variable_id ) ) THEN 303 ALLOCATE( files_highest_variable_id_tmp(SIZE( files_highest_variable_id )) ) 304 files_highest_variable_id_tmp = files_highest_variable_id 305 DEALLOCATE( files_highest_variable_id ) 306 ALLOCATE( files_highest_variable_id(binary_file_lowest_unit+1:file_id) ) 307 files_highest_variable_id(:file_id-1) = files_highest_variable_id_tmp 308 DEALLOCATE( files_highest_variable_id_tmp ) 309 ELSE 310 ALLOCATE( files_highest_variable_id(binary_file_lowest_unit+1:file_id) ) 311 ENDIF 312 files_highest_variable_id(file_id) = 0 313 314 ELSE 315 return_value = 1 316 CALL internal_message( 'error', routine_name // & 317 ': could not open file "' // TRIM( file_name ) // '"') 318 ENDIF 319 320 ENDIF 321 322 END SUBROUTINE binary_open_file 320 323 321 324 !--------------------------------------------------------------------------------------------------! … … 324 327 !> Write attribute to file. 325 328 !--------------------------------------------------------------------------------------------------! 326 SUBROUTINE binary_write_attribute( file_id, variable_id, attribute_name, &327 value_char, value_int8, value_int16, value_int32, &328 value_real32, value_real64, return_value )329 330 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_write_attribute' !< name of this routine331 332 CHARACTER(LEN=charlen), INTENT(IN) :: attribute_name !< name of attribute333 CHARACTER(LEN=charlen), INTENT(IN), OPTIONAL :: value_char !< value of attribute334 CHARACTER(LEN=charlen) :: attribute_type !< data type of attribute335 CHARACTER(LEN=charlen) :: output_string !< output string336 337 INTEGER(KIND=1), INTENT(IN), OPTIONAL :: value_int8 !< value of attribute338 INTEGER(KIND=2), INTENT(IN), OPTIONAL :: value_int16 !< value of attribute339 INTEGER(KIND=4), INTENT(IN), OPTIONAL :: value_int32 !< value of attribute340 341 INTEGER, INTENT(IN) :: file_id !< file ID342 INTEGER, INTENT(IN) :: variable_id !< variable ID343 INTEGER, INTENT(OUT) :: return_value !< return value344 345 REAL(KIND=4), INTENT(IN), OPTIONAL :: value_real32 !< value of attribute346 REAL(KIND=8), INTENT(IN), OPTIONAL :: value_real64 !< value of attribute347 348 349 return_value = 0350 351 CALL internal_message( 'debug', TRIM( routine_name ) // &352 ': write attribute ' // TRIM( attribute_name ) )353 354 355 output_string = 'attribute'356 WRITE( file_id ) output_string357 358 WRITE( file_id ) variable_id359 WRITE( file_id ) attribute_name360 361 IF ( PRESENT( value_char ) ) THEN362 attribute_type = 'char'363 WRITE( file_id ) attribute_type364 WRITE( file_id ) value_char365 ELSEIF ( PRESENT( value_int8 ) ) THEN366 attribute_type = 'int8'367 WRITE( file_id ) attribute_type368 WRITE( file_id ) value_int8369 ELSEIF ( PRESENT( value_int16 ) ) THEN370 attribute_type = 'int16'371 WRITE( file_id ) attribute_type372 WRITE( file_id ) value_int16373 ELSEIF ( PRESENT( value_int32 ) ) THEN374 attribute_type = 'int32'375 WRITE( file_id ) attribute_type376 WRITE( file_id ) value_int32377 ELSEIF ( PRESENT( value_real32 ) ) THEN378 attribute_type = 'real32'379 WRITE( file_id ) attribute_type380 WRITE( file_id ) value_real32381 ELSEIF ( PRESENT( value_real64 ) ) THEN382 attribute_type = 'real64'383 WRITE( file_id ) attribute_type384 WRITE( file_id ) value_real64385 ELSE386 return_value = 1387 CALL internal_message( 'error', TRIM( routine_name ) // &388 ': no value given for attribute "' // TRIM( attribute_name ) // '"' )389 ENDIF390 391 END SUBROUTINE binary_write_attribute329 SUBROUTINE binary_write_attribute( file_id, variable_id, attribute_name, & 330 value_char, value_int8, value_int16, value_int32, & 331 value_real32, value_real64, return_value ) 332 333 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_write_attribute' !< name of this routine 334 335 CHARACTER(LEN=charlen), INTENT(IN) :: attribute_name !< name of attribute 336 CHARACTER(LEN=charlen), INTENT(IN), OPTIONAL :: value_char !< value of attribute 337 CHARACTER(LEN=charlen) :: attribute_type !< data type of attribute 338 CHARACTER(LEN=charlen) :: output_string !< output string 339 340 INTEGER(KIND=1), INTENT(IN), OPTIONAL :: value_int8 !< value of attribute 341 INTEGER(KIND=2), INTENT(IN), OPTIONAL :: value_int16 !< value of attribute 342 INTEGER(KIND=4), INTENT(IN), OPTIONAL :: value_int32 !< value of attribute 343 344 INTEGER, INTENT(IN) :: file_id !< file ID 345 INTEGER, INTENT(IN) :: variable_id !< variable ID 346 INTEGER, INTENT(OUT) :: return_value !< return value 347 348 REAL(KIND=4), INTENT(IN), OPTIONAL :: value_real32 !< value of attribute 349 REAL(KIND=8), INTENT(IN), OPTIONAL :: value_real64 !< value of attribute 350 351 352 return_value = 0 353 354 CALL internal_message( 'debug', TRIM( routine_name ) // & 355 ': write attribute ' // TRIM( attribute_name ) ) 356 ! 357 !-- Write attribute to file 358 output_string = 'attribute' 359 WRITE( file_id ) output_string 360 361 WRITE( file_id ) variable_id 362 WRITE( file_id ) attribute_name 363 364 IF ( PRESENT( value_char ) ) THEN 365 attribute_type = 'char' 366 WRITE( file_id ) attribute_type 367 WRITE( file_id ) value_char 368 ELSEIF ( PRESENT( value_int8 ) ) THEN 369 attribute_type = 'int8' 370 WRITE( file_id ) attribute_type 371 WRITE( file_id ) value_int8 372 ELSEIF ( PRESENT( value_int16 ) ) THEN 373 attribute_type = 'int16' 374 WRITE( file_id ) attribute_type 375 WRITE( file_id ) value_int16 376 ELSEIF ( PRESENT( value_int32 ) ) THEN 377 attribute_type = 'int32' 378 WRITE( file_id ) attribute_type 379 WRITE( file_id ) value_int32 380 ELSEIF ( PRESENT( value_real32 ) ) THEN 381 attribute_type = 'real32' 382 WRITE( file_id ) attribute_type 383 WRITE( file_id ) value_real32 384 ELSEIF ( PRESENT( value_real64 ) ) THEN 385 attribute_type = 'real64' 386 WRITE( file_id ) attribute_type 387 WRITE( file_id ) value_real64 388 ELSE 389 return_value = 1 390 CALL internal_message( 'error', TRIM( routine_name ) // & 391 ': no value given for attribute "' // TRIM( attribute_name ) // '"' ) 392 ENDIF 393 394 END SUBROUTINE binary_write_attribute 392 395 393 396 !--------------------------------------------------------------------------------------------------! … … 397 400 !> and save dimension values to be later written to file. 398 401 !--------------------------------------------------------------------------------------------------! 399 SUBROUTINE binary_init_dimension( mode, file_id, dimension_id, variable_id, &400 dimension_name, dimension_type, dimension_length, return_value )401 402 CHARACTER(LEN=charlen), INTENT(IN) :: dimension_name !< name of dimension403 CHARACTER(LEN=charlen), INTENT(IN) :: dimension_type !< data type of dimension404 CHARACTER(LEN=charlen) :: output_string !< output string405 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode406 407 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_init_dimension' !< name of this routine408 409 INTEGER, INTENT(OUT) :: dimension_id !< dimension ID410 INTEGER, INTENT(IN) :: dimension_length !< length of dimension411 INTEGER, INTENT(IN) :: file_id !< file ID412 INTEGER, INTENT(OUT) :: return_value !< return value413 INTEGER, INTENT(OUT) :: variable_id !< variable ID414 415 416 return_value = 0417 418 CALL internal_message( 'debug', routine_name // ': init dimension ' // TRIM( dimension_name ) )419 420 421 IF ( TRIM( mode ) == mode_binary ) CONTINUE422 423 424 dimension_id = files_highest_variable_id( file_id ) + 1425 files_highest_variable_id( file_id ) = dimension_id426 427 428 output_string = 'dimension'429 WRITE( file_id ) output_string430 WRITE( file_id ) dimension_name431 WRITE( file_id ) dimension_id432 WRITE( file_id ) dimension_type433 WRITE( file_id ) dimension_length434 435 436 CALL binary_init_variable( mode, file_id, variable_id, dimension_name, dimension_type, &437 (/ dimension_id /), is_global=.TRUE., return_value=return_value )438 IF ( return_value /= 0 ) THEN439 CALL internal_message( 'error', routine_name // &440 ': init dimension "' // TRIM( dimension_name ) // '"' )441 ENDIF442 443 END SUBROUTINE binary_init_dimension402 SUBROUTINE binary_init_dimension( mode, file_id, dimension_id, variable_id, & 403 dimension_name, dimension_type, dimension_length, return_value ) 404 405 CHARACTER(LEN=charlen), INTENT(IN) :: dimension_name !< name of dimension 406 CHARACTER(LEN=charlen), INTENT(IN) :: dimension_type !< data type of dimension 407 CHARACTER(LEN=charlen) :: output_string !< output string 408 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode 409 410 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_init_dimension' !< name of this routine 411 412 INTEGER, INTENT(OUT) :: dimension_id !< dimension ID 413 INTEGER, INTENT(IN) :: dimension_length !< length of dimension 414 INTEGER, INTENT(IN) :: file_id !< file ID 415 INTEGER, INTENT(OUT) :: return_value !< return value 416 INTEGER, INTENT(OUT) :: variable_id !< variable ID 417 418 419 return_value = 0 420 421 CALL internal_message( 'debug', routine_name // ': init dimension ' // TRIM( dimension_name ) ) 422 ! 423 !-- Check mode (not required, added for compatibility reasons only) 424 IF ( TRIM( mode ) == mode_binary ) CONTINUE 425 ! 426 !-- Assign dimension ID 427 dimension_id = files_highest_variable_id( file_id ) + 1 428 files_highest_variable_id( file_id ) = dimension_id 429 ! 430 !-- Define dimension in file 431 output_string = 'dimension' 432 WRITE( file_id ) output_string 433 WRITE( file_id ) dimension_name 434 WRITE( file_id ) dimension_id 435 WRITE( file_id ) dimension_type 436 WRITE( file_id ) dimension_length 437 ! 438 !-- Define variable associated with dimension 439 CALL binary_init_variable( mode, file_id, variable_id, dimension_name, dimension_type, & 440 (/ dimension_id /), is_global=.TRUE., return_value=return_value ) 441 IF ( return_value /= 0 ) THEN 442 CALL internal_message( 'error', routine_name // & 443 ': init dimension "' // TRIM( dimension_name ) // '"' ) 444 ENDIF 445 446 END SUBROUTINE binary_init_dimension 444 447 445 448 !--------------------------------------------------------------------------------------------------! … … 448 451 !> Initialize variable. Write information of variable into file header. 449 452 !--------------------------------------------------------------------------------------------------! 450 SUBROUTINE binary_init_variable( mode, file_id, variable_id, variable_name, variable_type, &451 dimension_ids, is_global, return_value )452 453 CHARACTER(LEN=charlen) :: output_string !< output string454 CHARACTER(LEN=charlen), INTENT(IN) :: variable_name !< name of variable455 CHARACTER(LEN=charlen), INTENT(IN) :: variable_type !< data type of variable456 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode457 458 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_init_variable' !< name of this routine459 460 INTEGER, INTENT(IN) :: file_id !< file ID461 INTEGER, INTENT(OUT) :: variable_id !< variable ID462 INTEGER, INTENT(OUT) :: return_value !< return value463 464 INTEGER, DIMENSION(:), INTENT(IN) :: dimension_ids !< list of dimension IDs used by variable465 466 LOGICAL, INTENT(IN) :: is_global !< true if variable is global (same on all PE)467 468 469 return_value = 0470 471 CALL internal_message( 'debug', routine_name // ': init variable ' // TRIM( variable_name ) )472 473 474 IF ( TRIM( mode ) == mode_binary ) CONTINUE475 476 477 IF ( is_global ) CONTINUE478 479 480 variable_id = files_highest_variable_id( file_id ) + 1481 files_highest_variable_id( file_id ) = variable_id482 483 484 output_string = 'variable'485 WRITE( file_id ) output_string486 WRITE( file_id ) variable_name487 WRITE( file_id ) variable_id488 WRITE( file_id ) variable_type489 WRITE( file_id ) SIZE( dimension_ids )490 WRITE( file_id ) dimension_ids491 492 END SUBROUTINE binary_init_variable453 SUBROUTINE binary_init_variable( mode, file_id, variable_id, variable_name, variable_type, & 454 dimension_ids, is_global, return_value ) 455 456 CHARACTER(LEN=charlen) :: output_string !< output string 457 CHARACTER(LEN=charlen), INTENT(IN) :: variable_name !< name of variable 458 CHARACTER(LEN=charlen), INTENT(IN) :: variable_type !< data type of variable 459 CHARACTER(LEN=*), INTENT(IN) :: mode !< operation mode 460 461 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_init_variable' !< name of this routine 462 463 INTEGER, INTENT(IN) :: file_id !< file ID 464 INTEGER, INTENT(OUT) :: variable_id !< variable ID 465 INTEGER, INTENT(OUT) :: return_value !< return value 466 467 INTEGER, DIMENSION(:), INTENT(IN) :: dimension_ids !< list of dimension IDs used by variable 468 469 LOGICAL, INTENT(IN) :: is_global !< true if variable is global (same on all PE) 470 471 472 return_value = 0 473 474 CALL internal_message( 'debug', routine_name // ': init variable ' // TRIM( variable_name ) ) 475 ! 476 !-- Check mode (not required, added for compatibility reasons only) 477 IF ( TRIM( mode ) == mode_binary ) CONTINUE 478 ! 479 !-- Check if variable is global (not required, added for compatibility reasons only) 480 IF ( is_global ) CONTINUE 481 ! 482 !-- Assign variable ID 483 variable_id = files_highest_variable_id( file_id ) + 1 484 files_highest_variable_id( file_id ) = variable_id 485 ! 486 !-- Write variable information in file 487 output_string = 'variable' 488 WRITE( file_id ) output_string 489 WRITE( file_id ) variable_name 490 WRITE( file_id ) variable_id 491 WRITE( file_id ) variable_type 492 WRITE( file_id ) SIZE( dimension_ids ) 493 WRITE( file_id ) dimension_ids 494 495 END SUBROUTINE binary_init_variable 493 496 494 497 !--------------------------------------------------------------------------------------------------! … … 497 500 !> Leave file definition state. 498 501 !--------------------------------------------------------------------------------------------------! 499 SUBROUTINE binary_stop_file_header_definition( file_id, return_value )500 501 CHARACTER(LEN=charlen) :: output_string !< output string502 503 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_stop_file_header_definition' !< name of this routine504 505 INTEGER, INTENT(IN) :: file_id !< file ID506 INTEGER, INTENT(OUT) :: return_value !< return value507 508 509 return_value = 0510 511 WRITE( temp_string, * ) file_id512 CALL internal_message( 'debug', routine_name // &513 ': finalize file definition (file_id=' // TRIM( temp_string ) // ')' )514 515 output_string = '*** end file header ***'516 WRITE( file_id ) output_string517 518 END SUBROUTINE binary_stop_file_header_definition502 SUBROUTINE binary_stop_file_header_definition( file_id, return_value ) 503 504 CHARACTER(LEN=charlen) :: output_string !< output string 505 506 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_stop_file_header_definition' !< name of this routine 507 508 INTEGER, INTENT(IN) :: file_id !< file ID 509 INTEGER, INTENT(OUT) :: return_value !< return value 510 511 512 return_value = 0 513 514 WRITE( temp_string, * ) file_id 515 CALL internal_message( 'debug', routine_name // & 516 ': finalize file definition (file_id=' // TRIM( temp_string ) // ')' ) 517 518 output_string = '*** end file header ***' 519 WRITE( file_id ) output_string 520 521 END SUBROUTINE binary_stop_file_header_definition 519 522 520 523 !--------------------------------------------------------------------------------------------------! … … 523 526 !> Write variable to file. 524 527 !--------------------------------------------------------------------------------------------------! 525 SUBROUTINE binary_write_variable( & 526 file_id, variable_id, bounds_start, value_counts, bounds_origin, & 527 is_global, & 528 values_int8_0d, values_int8_1d, values_int8_2d, values_int8_3d, & 529 values_int16_0d, values_int16_1d, values_int16_2d, values_int16_3d, & 530 values_int32_0d, values_int32_1d, values_int32_2d, values_int32_3d, & 531 values_intwp_0d, values_intwp_1d, values_intwp_2d, values_intwp_3d, & 532 values_real32_0d, values_real32_1d, values_real32_2d, values_real32_3d, & 533 values_real64_0d, values_real64_1d, values_real64_2d, values_real64_3d, & 534 values_realwp_0d, values_realwp_1d, values_realwp_2d, values_realwp_3d, & 535 return_value ) 536 537 CHARACTER(LEN=charlen) :: output_string !< output string 538 539 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_write_variable' !< name of this routine 540 541 INTEGER, INTENT(IN) :: file_id !< file ID 542 INTEGER, INTENT(OUT) :: return_value !< return value 543 INTEGER, INTENT(IN) :: variable_id !< variable ID 544 545 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_origin !< starting index of each dimension 546 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_start !< starting index of variable 547 INTEGER, DIMENSION(:), INTENT(IN) :: value_counts !< count of values along each dimension to be written 548 549 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL :: values_int8_0d !< output variable 550 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL :: values_int16_0d !< output variable 551 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_int32_0d !< output variable 552 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL :: values_intwp_0d !< output variable 553 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int8_1d !< output variable 554 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int16_1d !< output variable 555 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int32_1d !< output variable 556 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_intwp_1d !< output variable 557 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int8_2d !< output variable 558 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int16_2d !< output variable 559 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int32_2d !< output variable 560 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_intwp_2d !< output variable 561 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int8_3d !< output variable 562 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int16_3d !< output variable 563 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int32_3d !< output variable 564 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_intwp_3d !< output variable 565 566 LOGICAL, INTENT(IN) :: is_global !< true if variable is global (same on all PE) 567 568 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_real32_0d !< output variable 569 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL :: values_real64_0d !< output variable 570 REAL(wp), POINTER, INTENT(IN), OPTIONAL :: values_realwp_0d !< output variable 571 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real32_1d !< output variable 572 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real64_1d !< output variable 573 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_realwp_1d !< output variable 574 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real32_2d !< output variable 575 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real64_2d !< output variable 576 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_realwp_2d !< output variable 577 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real32_3d !< output variable 578 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real64_3d !< output variable 579 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_realwp_3d !< output variable 580 581 582 return_value = 0 583 584 WRITE( temp_string, '(": write variable ",I6," into file ",I6)' ) variable_id, file_id 585 CALL internal_message( 'debug', routine_name // TRIM( temp_string ) ) 586 587 IF ( is_global ) CONTINUE ! reqired to prevent compiler warning 588 589 IF ( .NOT. ANY( value_counts == 0 ) ) THEN 590 WRITE( file_id ) variable_id 591 WRITE( file_id ) bounds_start 592 WRITE( file_id ) value_counts 593 WRITE( file_id ) bounds_origin 594 !-- 8bit integer output 595 IF ( PRESENT( values_int8_0d ) ) THEN 596 output_string = 'int8' 597 WRITE( file_id ) output_string 598 WRITE( file_id ) values_int8_0d 599 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 600 output_string = 'int8' 601 WRITE( file_id ) output_string 602 WRITE( file_id ) values_int8_1d 603 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 604 output_string = 'int8' 605 WRITE( file_id ) output_string 606 WRITE( file_id ) values_int8_2d 607 ELSEIF ( PRESENT( values_int8_3d ) ) THEN 608 output_string = 'int8' 609 WRITE( file_id ) output_string 610 WRITE( file_id ) values_int8_3d 611 !-- 16bit integer output 612 ELSEIF ( PRESENT( values_int16_0d ) ) THEN 613 output_string = 'int16' 614 WRITE( file_id ) output_string 615 WRITE( file_id ) values_int16_0d 616 ELSEIF ( PRESENT( values_int16_1d ) ) THEN 617 output_string = 'int16' 618 WRITE( file_id ) output_string 619 WRITE( file_id ) values_int16_1d 620 ELSEIF ( PRESENT( values_int16_2d ) ) THEN 621 output_string = 'int16' 622 WRITE( file_id ) output_string 623 WRITE( file_id ) values_int16_2d 624 ELSEIF ( PRESENT( values_int16_3d ) ) THEN 625 output_string = 'int16' 626 WRITE( file_id ) output_string 627 WRITE( file_id ) values_int16_3d 628 !-- 32bit integer output 629 ELSEIF ( PRESENT( values_int32_0d ) ) THEN 630 output_string = 'int32' 631 WRITE( file_id ) output_string 632 WRITE( file_id ) values_int32_0d 633 ELSEIF ( PRESENT( values_int32_1d ) ) THEN 634 output_string = 'int32' 635 WRITE( file_id ) output_string 636 WRITE( file_id ) values_int32_1d 637 ELSEIF ( PRESENT( values_int32_2d ) ) THEN 638 output_string = 'int32' 639 WRITE( file_id ) output_string 640 WRITE( file_id ) values_int32_2d 641 ELSEIF ( PRESENT( values_int32_3d ) ) THEN 642 output_string = 'int32' 643 WRITE( file_id ) output_string 644 WRITE( file_id ) values_int32_3d 645 !-- working-precision integer output 646 ELSEIF ( PRESENT( values_intwp_0d ) ) THEN 647 output_string = 'intwp' 648 WRITE( file_id ) output_string 649 WRITE( file_id ) values_intwp_0d 650 ELSEIF ( PRESENT( values_intwp_1d ) ) THEN 651 output_string = 'intwp' 652 WRITE( file_id ) output_string 653 WRITE( file_id ) values_intwp_1d 654 ELSEIF ( PRESENT( values_intwp_2d ) ) THEN 655 output_string = 'intwp' 656 WRITE( file_id ) output_string 657 WRITE( file_id ) values_intwp_2d 658 ELSEIF ( PRESENT( values_intwp_3d ) ) THEN 659 output_string = 'intwp' 660 WRITE( file_id ) output_string 661 WRITE( file_id ) values_intwp_3d 662 !-- 32bit real output 663 ELSEIF ( PRESENT( values_real32_0d ) ) THEN 664 output_string = 'real32' 665 WRITE( file_id ) output_string 666 WRITE( file_id ) values_real32_0d 667 ELSEIF ( PRESENT( values_real32_1d ) ) THEN 668 output_string = 'real32' 669 WRITE( file_id ) output_string 670 WRITE( file_id ) values_real32_1d 671 ELSEIF ( PRESENT( values_real32_2d ) ) THEN 672 output_string = 'real32' 673 WRITE( file_id ) output_string 674 WRITE( file_id ) values_real32_2d 675 ELSEIF ( PRESENT( values_real32_3d ) ) THEN 676 output_string = 'real32' 677 WRITE( file_id ) output_string 678 WRITE( file_id ) values_real32_3d 679 !-- 64bit real output 680 ELSEIF ( PRESENT( values_real64_0d ) ) THEN 681 output_string = 'real64' 682 WRITE( file_id ) output_string 683 WRITE( file_id ) values_real64_0d 684 ELSEIF ( PRESENT( values_real64_1d ) ) THEN 685 output_string = 'real64' 686 WRITE( file_id ) output_string 687 WRITE( file_id ) values_real64_1d 688 ELSEIF ( PRESENT( values_real64_2d ) ) THEN 689 output_string = 'real64' 690 WRITE( file_id ) output_string 691 WRITE( file_id ) values_real64_2d 692 ELSEIF ( PRESENT( values_real64_3d ) ) THEN 693 output_string = 'real64' 694 WRITE( file_id ) output_string 695 WRITE( file_id ) values_real64_3d 696 !-- working-precision real output 697 ELSEIF ( PRESENT( values_realwp_0d ) ) THEN 698 output_string = 'realwp' 699 WRITE( file_id ) output_string 700 WRITE( file_id ) values_realwp_0d 701 ELSEIF ( PRESENT( values_realwp_1d ) ) THEN 702 output_string = 'realwp' 703 WRITE( file_id ) output_string 704 WRITE( file_id ) values_realwp_1d 705 ELSEIF ( PRESENT( values_realwp_2d ) ) THEN 706 output_string = 'realwp' 707 WRITE( file_id ) output_string 708 WRITE( file_id ) values_realwp_2d 709 ELSEIF ( PRESENT( values_realwp_3d ) ) THEN 710 output_string = 'realwp' 711 WRITE( file_id ) output_string 712 WRITE( file_id ) values_realwp_3d 713 ELSE 714 return_value = 1 715 CALL internal_message( 'error', routine_name // ': no values given' ) 716 ENDIF 717 718 ENDIF 719 720 END SUBROUTINE binary_write_variable 528 SUBROUTINE binary_write_variable( & 529 file_id, variable_id, bounds_start, value_counts, bounds_origin, & 530 is_global, & 531 values_int8_0d, values_int8_1d, values_int8_2d, values_int8_3d, & 532 values_int16_0d, values_int16_1d, values_int16_2d, values_int16_3d, & 533 values_int32_0d, values_int32_1d, values_int32_2d, values_int32_3d, & 534 values_intwp_0d, values_intwp_1d, values_intwp_2d, values_intwp_3d, & 535 values_real32_0d, values_real32_1d, values_real32_2d, values_real32_3d, & 536 values_real64_0d, values_real64_1d, values_real64_2d, values_real64_3d, & 537 values_realwp_0d, values_realwp_1d, values_realwp_2d, values_realwp_3d, & 538 return_value ) 539 540 CHARACTER(LEN=charlen) :: output_string !< output string 541 542 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_write_variable' !< name of this routine 543 544 INTEGER, INTENT(IN) :: file_id !< file ID 545 INTEGER, INTENT(OUT) :: return_value !< return value 546 INTEGER, INTENT(IN) :: variable_id !< variable ID 547 548 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_origin !< starting index of each dimension 549 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_start !< starting index of variable 550 INTEGER, DIMENSION(:), INTENT(IN) :: value_counts !< count of values along each dimension to be written 551 552 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL :: values_int8_0d !< output variable 553 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL :: values_int16_0d !< output variable 554 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_int32_0d !< output variable 555 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL :: values_intwp_0d !< output variable 556 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int8_1d !< output variable 557 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int16_1d !< output variable 558 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int32_1d !< output variable 559 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_intwp_1d !< output variable 560 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int8_2d !< output variable 561 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int16_2d !< output variable 562 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int32_2d !< output variable 563 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_intwp_2d !< output variable 564 INTEGER(KIND=1), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int8_3d !< output variable 565 INTEGER(KIND=2), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int16_3d !< output variable 566 INTEGER(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int32_3d !< output variable 567 INTEGER(iwp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_intwp_3d !< output variable 568 569 LOGICAL, INTENT(IN) :: is_global !< true if variable is global (same on all PE) 570 571 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_real32_0d !< output variable 572 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL :: values_real64_0d !< output variable 573 REAL(wp), POINTER, INTENT(IN), OPTIONAL :: values_realwp_0d !< output variable 574 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real32_1d !< output variable 575 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real64_1d !< output variable 576 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_realwp_1d !< output variable 577 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real32_2d !< output variable 578 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real64_2d !< output variable 579 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_realwp_2d !< output variable 580 REAL(KIND=4), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real32_3d !< output variable 581 REAL(KIND=8), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real64_3d !< output variable 582 REAL(wp), POINTER, CONTIGUOUS, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_realwp_3d !< output variable 583 584 585 return_value = 0 586 587 WRITE( temp_string, '(": write variable ",I6," into file ",I6)' ) variable_id, file_id 588 CALL internal_message( 'debug', routine_name // TRIM( temp_string ) ) 589 590 IF ( is_global ) CONTINUE ! reqired to prevent compiler warning 591 592 IF ( .NOT. ANY( value_counts == 0 ) ) THEN 593 WRITE( file_id ) variable_id 594 WRITE( file_id ) bounds_start 595 WRITE( file_id ) value_counts 596 WRITE( file_id ) bounds_origin 597 ! 598 !-- 8bit integer output 599 IF ( PRESENT( values_int8_0d ) ) THEN 600 output_string = 'int8' 601 WRITE( file_id ) output_string 602 WRITE( file_id ) values_int8_0d 603 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 604 output_string = 'int8' 605 WRITE( file_id ) output_string 606 WRITE( file_id ) values_int8_1d 607 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 608 output_string = 'int8' 609 WRITE( file_id ) output_string 610 WRITE( file_id ) values_int8_2d 611 ELSEIF ( PRESENT( values_int8_3d ) ) THEN 612 output_string = 'int8' 613 WRITE( file_id ) output_string 614 WRITE( file_id ) values_int8_3d 615 ! 616 !-- 16bit integer output 617 ELSEIF ( PRESENT( values_int16_0d ) ) THEN 618 output_string = 'int16' 619 WRITE( file_id ) output_string 620 WRITE( file_id ) values_int16_0d 621 ELSEIF ( PRESENT( values_int16_1d ) ) THEN 622 output_string = 'int16' 623 WRITE( file_id ) output_string 624 WRITE( file_id ) values_int16_1d 625 ELSEIF ( PRESENT( values_int16_2d ) ) THEN 626 output_string = 'int16' 627 WRITE( file_id ) output_string 628 WRITE( file_id ) values_int16_2d 629 ELSEIF ( PRESENT( values_int16_3d ) ) THEN 630 output_string = 'int16' 631 WRITE( file_id ) output_string 632 WRITE( file_id ) values_int16_3d 633 ! 634 !-- 32bit integer output 635 ELSEIF ( PRESENT( values_int32_0d ) ) THEN 636 output_string = 'int32' 637 WRITE( file_id ) output_string 638 WRITE( file_id ) values_int32_0d 639 ELSEIF ( PRESENT( values_int32_1d ) ) THEN 640 output_string = 'int32' 641 WRITE( file_id ) output_string 642 WRITE( file_id ) values_int32_1d 643 ELSEIF ( PRESENT( values_int32_2d ) ) THEN 644 output_string = 'int32' 645 WRITE( file_id ) output_string 646 WRITE( file_id ) values_int32_2d 647 ELSEIF ( PRESENT( values_int32_3d ) ) THEN 648 output_string = 'int32' 649 WRITE( file_id ) output_string 650 WRITE( file_id ) values_int32_3d 651 ! 652 !-- working-precision integer output 653 ELSEIF ( PRESENT( values_intwp_0d ) ) THEN 654 output_string = 'intwp' 655 WRITE( file_id ) output_string 656 WRITE( file_id ) values_intwp_0d 657 ELSEIF ( PRESENT( values_intwp_1d ) ) THEN 658 output_string = 'intwp' 659 WRITE( file_id ) output_string 660 WRITE( file_id ) values_intwp_1d 661 ELSEIF ( PRESENT( values_intwp_2d ) ) THEN 662 output_string = 'intwp' 663 WRITE( file_id ) output_string 664 WRITE( file_id ) values_intwp_2d 665 ELSEIF ( PRESENT( values_intwp_3d ) ) THEN 666 output_string = 'intwp' 667 WRITE( file_id ) output_string 668 WRITE( file_id ) values_intwp_3d 669 ! 670 !-- 32bit real output 671 ELSEIF ( PRESENT( values_real32_0d ) ) THEN 672 output_string = 'real32' 673 WRITE( file_id ) output_string 674 WRITE( file_id ) values_real32_0d 675 ELSEIF ( PRESENT( values_real32_1d ) ) THEN 676 output_string = 'real32' 677 WRITE( file_id ) output_string 678 WRITE( file_id ) values_real32_1d 679 ELSEIF ( PRESENT( values_real32_2d ) ) THEN 680 output_string = 'real32' 681 WRITE( file_id ) output_string 682 WRITE( file_id ) values_real32_2d 683 ELSEIF ( PRESENT( values_real32_3d ) ) THEN 684 output_string = 'real32' 685 WRITE( file_id ) output_string 686 WRITE( file_id ) values_real32_3d 687 ! 688 !-- 64bit real output 689 ELSEIF ( PRESENT( values_real64_0d ) ) THEN 690 output_string = 'real64' 691 WRITE( file_id ) output_string 692 WRITE( file_id ) values_real64_0d 693 ELSEIF ( PRESENT( values_real64_1d ) ) THEN 694 output_string = 'real64' 695 WRITE( file_id ) output_string 696 WRITE( file_id ) values_real64_1d 697 ELSEIF ( PRESENT( values_real64_2d ) ) THEN 698 output_string = 'real64' 699 WRITE( file_id ) output_string 700 WRITE( file_id ) values_real64_2d 701 ELSEIF ( PRESENT( values_real64_3d ) ) THEN 702 output_string = 'real64' 703 WRITE( file_id ) output_string 704 WRITE( file_id ) values_real64_3d 705 ! 706 !-- working-precision real output 707 ELSEIF ( PRESENT( values_realwp_0d ) ) THEN 708 output_string = 'realwp' 709 WRITE( file_id ) output_string 710 WRITE( file_id ) values_realwp_0d 711 ELSEIF ( PRESENT( values_realwp_1d ) ) THEN 712 output_string = 'realwp' 713 WRITE( file_id ) output_string 714 WRITE( file_id ) values_realwp_1d 715 ELSEIF ( PRESENT( values_realwp_2d ) ) THEN 716 output_string = 'realwp' 717 WRITE( file_id ) output_string 718 WRITE( file_id ) values_realwp_2d 719 ELSEIF ( PRESENT( values_realwp_3d ) ) THEN 720 output_string = 'realwp' 721 WRITE( file_id ) output_string 722 WRITE( file_id ) values_realwp_3d 723 ELSE 724 return_value = 1 725 CALL internal_message( 'error', routine_name // ': no values given' ) 726 ENDIF 727 728 ENDIF 729 730 END SUBROUTINE binary_write_variable 721 731 722 732 !--------------------------------------------------------------------------------------------------! … … 725 735 !> Close opened files. 726 736 !--------------------------------------------------------------------------------------------------! 727 SUBROUTINE binary_finalize( file_id, return_value )728 729 CHARACTER(LEN=charlen) :: output_string !< output string730 731 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_finalize' !< name of this routine732 733 INTEGER, INTENT(IN) :: file_id !< file ID734 INTEGER, INTENT(OUT) :: return_value !< return value735 736 737 IF ( config_file_open ) THEN738 739 output_string = '*** end config file ***'740 WRITE( config_file_unit ) output_string741 742 CLOSE( config_file_unit, IOSTAT=return_value )743 744 IF ( return_value /= 0 ) THEN745 CALL internal_message( 'error', routine_name // ': cannot close configuration file' )746 ELSE747 config_file_open = .FALSE.748 ENDIF749 750 ELSE751 752 return_value = 0753 754 ENDIF755 756 IF ( return_value == 0 ) THEN757 758 WRITE( temp_string, * ) file_id759 CALL internal_message( 'debug', routine_name // &760 ': close file (file_id=' // TRIM( temp_string ) // ')' )761 762 CLOSE( file_id, IOSTAT=return_value )763 IF ( return_value /= 0 ) THEN764 WRITE( temp_string, * ) file_id765 CALL internal_message( 'error', routine_name // &766 ': cannot close file (file_id=' // TRIM( temp_string ) // ')' )767 ENDIF768 769 ENDIF770 771 END SUBROUTINE binary_finalize737 SUBROUTINE binary_finalize( file_id, return_value ) 738 739 CHARACTER(LEN=charlen) :: output_string !< output string 740 741 CHARACTER(LEN=*), PARAMETER :: routine_name = 'binary_finalize' !< name of this routine 742 743 INTEGER, INTENT(IN) :: file_id !< file ID 744 INTEGER, INTENT(OUT) :: return_value !< return value 745 746 747 IF ( config_file_open ) THEN 748 749 output_string = '*** end config file ***' 750 WRITE( config_file_unit ) output_string 751 752 CLOSE( config_file_unit, IOSTAT=return_value ) 753 754 IF ( return_value /= 0 ) THEN 755 CALL internal_message( 'error', routine_name // ': cannot close configuration file' ) 756 ELSE 757 config_file_open = .FALSE. 758 ENDIF 759 760 ELSE 761 762 return_value = 0 763 764 ENDIF 765 766 IF ( return_value == 0 ) THEN 767 768 WRITE( temp_string, * ) file_id 769 CALL internal_message( 'debug', routine_name // & 770 ': close file (file_id=' // TRIM( temp_string ) // ')' ) 771 772 CLOSE( file_id, IOSTAT=return_value ) 773 IF ( return_value /= 0 ) THEN 774 WRITE( temp_string, * ) file_id 775 CALL internal_message( 'error', routine_name // & 776 ': cannot close file (file_id=' // TRIM( temp_string ) // ')' ) 777 ENDIF 778 779 ENDIF 780 781 END SUBROUTINE binary_finalize 772 782 773 783 !--------------------------------------------------------------------------------------------------! … … 777 787 !> or creating the error message string. 778 788 !--------------------------------------------------------------------------------------------------! 779 SUBROUTINE internal_message( level, string )780 781 CHARACTER(LEN=*), INTENT(IN) :: level !< message importance level782 CHARACTER(LEN=*), INTENT(IN) :: string !< message string783 784 785 IF ( TRIM( level ) == 'error' ) THEN786 787 WRITE( internal_error_message, '(A,A)' ) ': ', string788 789 ELSEIF ( TRIM( level ) == 'debug' .AND. print_debug_output ) THEN790 791 WRITE( debug_output_unit, '(A,A)' ) 'DOM DEBUG: ', string792 FLUSH( debug_output_unit )793 794 ENDIF795 796 END SUBROUTINE internal_message789 SUBROUTINE internal_message( level, string ) 790 791 CHARACTER(LEN=*), INTENT(IN) :: level !< message importance level 792 CHARACTER(LEN=*), INTENT(IN) :: string !< message string 793 794 795 IF ( TRIM( level ) == 'error' ) THEN 796 797 WRITE( internal_error_message, '(A,A)' ) ': ', string 798 799 ELSEIF ( TRIM( level ) == 'debug' .AND. print_debug_output ) THEN 800 801 WRITE( debug_output_unit, '(A,A)' ) 'DOM DEBUG: ', string 802 FLUSH( debug_output_unit ) 803 804 ENDIF 805 806 END SUBROUTINE internal_message 797 807 798 808 !--------------------------------------------------------------------------------------------------! … … 801 811 !> Return the last created error message. 802 812 !--------------------------------------------------------------------------------------------------! 803 FUNCTION binary_get_error_message() RESULT( error_message )804 805 CHARACTER(LEN=800) :: error_message !< return error message to main program806 807 808 error_message = TRIM( internal_error_message )809 810 internal_error_message = ''811 812 END FUNCTION binary_get_error_message813 814 END MODULE data_output_binary_module813 FUNCTION binary_get_error_message() RESULT( error_message ) 814 815 CHARACTER(LEN=800) :: error_message !< return error message to main program 816 817 818 error_message = TRIM( internal_error_message ) 819 820 internal_error_message = '' 821 822 END FUNCTION binary_get_error_message 823 824 END MODULE data_output_binary_module -
palm/trunk/SOURCE/data_output_module.f90
r4141 r4147 25 25 ! ----------------- 26 26 ! $Id$ 27 ! corrected indentation according to coding standard 28 ! 29 ! 4141 2019-08-05 12:24:51Z gronemeier 27 30 ! Initial revision 28 31 ! … … 59 62 !> @todo Convert variable if type of given values do not fit specified type. 60 63 !--------------------------------------------------------------------------------------------------! 61 MODULE data_output_module62 63 USE kinds64 65 USE data_output_netcdf4_module, &66 ONLY: netcdf4_init_dimension, &67 netcdf4_get_error_message, &68 netcdf4_stop_file_header_definition, &69 netcdf4_init_module, &70 netcdf4_init_variable, &71 netcdf4_finalize, &72 netcdf4_open_file, &73 netcdf4_write_attribute, &74 netcdf4_write_variable75 76 USE data_output_binary_module, &77 ONLY: binary_finalize, &78 binary_get_error_message, &79 binary_init_dimension, &80 binary_stop_file_header_definition, &81 binary_init_module, &82 binary_init_variable, &83 binary_open_file, &84 binary_write_attribute, &85 binary_write_variable86 87 IMPLICIT NONE88 89 INTEGER, PARAMETER :: charlen = 100 !< maximum length of character variables90 INTEGER, PARAMETER :: no_id = -1 !< default ID if no ID was assigned91 92 TYPE attribute_type93 CHARACTER(LEN=charlen) :: data_type = '' !< data type94 CHARACTER(LEN=charlen) :: name !< attribute name95 CHARACTER(LEN=charlen) :: value_char !< attribute value if character96 INTEGER(KIND=1) :: value_int8 !< attribute value if 8bit integer97 INTEGER(KIND=2) :: value_int16 !< attribute value if 16bit integer98 INTEGER(KIND=4) :: value_int32 !< attribute value if 32bit integer99 REAL(KIND=4) :: value_real32 !< attribute value if 32bit real100 REAL(KIND=8) :: value_real64 !< attribute value if 64bit real101 END TYPE attribute_type102 103 TYPE variable_type104 CHARACTER(LEN=charlen) :: data_type = '' !< data type105 CHARACTER(LEN=charlen) :: name !< variable name106 INTEGER :: id = no_id !< id within file107 LOGICAL :: is_global = .FALSE. !< true if global variable108 CHARACTER(LEN=charlen), DIMENSION(:), ALLOCATABLE :: dimension_names !< list of dimension names used by variable109 INTEGER, DIMENSION(:), ALLOCATABLE :: dimension_ids !< list of dimension ids used by variable110 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes111 END TYPE variable_type112 113 TYPE dimension_type114 CHARACTER(LEN=charlen) :: data_type = '' !< data type115 CHARACTER(LEN=charlen) :: name !< dimension name116 INTEGER :: id = no_id !< dimension id within file117 INTEGER :: length !< length of dimension118 INTEGER :: length_mask !< length of masked dimension119 INTEGER :: variable_id = no_id !< associated variable id within file120 LOGICAL :: is_masked = .FALSE. !< true if masked121 INTEGER, DIMENSION(2) :: bounds !< lower and upper bound of dimension122 INTEGER, DIMENSION(:), ALLOCATABLE :: masked_indices !< list of masked indices of dimension123 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE :: masked_values_int8 !< masked dimension values if 16bit integer124 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE :: masked_values_int16 !< masked dimension values if 16bit integer125 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE :: masked_values_int32 !< masked dimension values if 32bit integer126 INTEGER(iwp), DIMENSION(:), ALLOCATABLE :: masked_values_intwp !< masked dimension values if working-precision int127 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE :: values_int8 !< dimension values if 16bit integer128 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE :: values_int16 !< dimension values if 16bit integer129 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE :: values_int32 !< dimension values if 32bit integer130 INTEGER(iwp), DIMENSION(:), ALLOCATABLE :: values_intwp !< dimension values if working-precision integer131 LOGICAL, DIMENSION(:), ALLOCATABLE :: mask !< mask132 REAL(KIND=4), DIMENSION(:), ALLOCATABLE :: masked_values_real32 !< masked dimension values if 32bit real133 REAL(KIND=8), DIMENSION(:), ALLOCATABLE :: masked_values_real64 !< masked dimension values if 64bit real134 REAL(wp), DIMENSION(:), ALLOCATABLE :: masked_values_realwp !< masked dimension values if working-precision real135 REAL(KIND=4), DIMENSION(:), ALLOCATABLE :: values_real32 !< dimension values if 32bit real136 REAL(KIND=8), DIMENSION(:), ALLOCATABLE :: values_real64 !< dimension values if 64bit real137 REAL(wp), DIMENSION(:), ALLOCATABLE :: values_realwp !< dimension values if working-precision real138 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes139 END TYPE dimension_type140 141 TYPE file_type142 CHARACTER(LEN=charlen) :: format = '' !< file format143 CHARACTER(LEN=charlen) :: name = '' !< file name144 INTEGER :: id = no_id !< id of file145 LOGICAL :: is_init = .FALSE. !< true if initialized146 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes147 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimensions !< list of dimensions148 TYPE(variable_type), DIMENSION(:), ALLOCATABLE :: variables !< list of variables149 END TYPE file_type150 151 152 CHARACTER(LEN=charlen) :: output_file_suffix = '' !< file suffix added to each file name153 CHARACTER(LEN=800) :: internal_error_message = '' !< string containing the last error message154 CHARACTER(LEN=800) :: temp_string !< dummy string155 156 INTEGER :: debug_output_unit !< Fortran Unit Number of the debug-output file157 INTEGER :: nfiles = 0 !< number of files158 INTEGER :: master_rank = 0 !< master rank for tasks to be executed by single PE only159 INTEGER :: output_group_comm !< MPI communicator addressing all MPI ranks which participate in output160 161 LOGICAL :: print_debug_output = .FALSE. !< if true, debug output is printed162 163 TYPE(file_type), DIMENSION(:), ALLOCATABLE :: files !< file list164 165 SAVE166 167 PRIVATE168 169 !> Initialize the data-output module170 INTERFACE dom_init171 MODULE PROCEDURE dom_init172 END INTERFACE dom_init173 174 !> Add files to database175 INTERFACE dom_def_file176 MODULE PROCEDURE dom_def_file177 END INTERFACE dom_def_file178 179 !> Add dimensions to database180 INTERFACE dom_def_dim181 MODULE PROCEDURE dom_def_dim182 END INTERFACE dom_def_dim183 184 !> Add variables to database185 INTERFACE dom_def_var186 MODULE PROCEDURE dom_def_var187 END INTERFACE dom_def_var188 189 !> Add attributes to database190 INTERFACE dom_def_att191 MODULE PROCEDURE dom_def_att_char192 MODULE PROCEDURE dom_def_att_int8193 MODULE PROCEDURE dom_def_att_int16194 MODULE PROCEDURE dom_def_att_int32195 MODULE PROCEDURE dom_def_att_real32196 MODULE PROCEDURE dom_def_att_real64197 END INTERFACE dom_def_att198 199 !> Prepare for output: evaluate database and create files200 INTERFACE dom_def_end201 MODULE PROCEDURE dom_def_end202 END INTERFACE dom_def_end203 204 !> Write variables to file205 INTERFACE dom_write_var206 MODULE PROCEDURE dom_write_var207 END INTERFACE dom_write_var208 209 !> Last actions required for output befor termination210 INTERFACE dom_finalize_output211 MODULE PROCEDURE dom_finalize_output212 END INTERFACE dom_finalize_output213 214 !> Return error message215 INTERFACE dom_get_error_message216 MODULE PROCEDURE dom_get_error_message217 END INTERFACE dom_get_error_message218 219 !> Write database to debug output220 INTERFACE dom_database_debug_output221 MODULE PROCEDURE dom_database_debug_output222 END INTERFACE dom_database_debug_output223 224 PUBLIC &225 dom_init, &226 dom_def_file, &227 dom_def_dim, &228 dom_def_var, &229 dom_def_att, &230 dom_def_end, &231 dom_write_var, &232 dom_finalize_output, &233 dom_get_error_message, &234 dom_database_debug_output235 236 CONTAINS64 MODULE data_output_module 65 66 USE kinds 67 68 USE data_output_netcdf4_module, & 69 ONLY: netcdf4_init_dimension, & 70 netcdf4_get_error_message, & 71 netcdf4_stop_file_header_definition, & 72 netcdf4_init_module, & 73 netcdf4_init_variable, & 74 netcdf4_finalize, & 75 netcdf4_open_file, & 76 netcdf4_write_attribute, & 77 netcdf4_write_variable 78 79 USE data_output_binary_module, & 80 ONLY: binary_finalize, & 81 binary_get_error_message, & 82 binary_init_dimension, & 83 binary_stop_file_header_definition, & 84 binary_init_module, & 85 binary_init_variable, & 86 binary_open_file, & 87 binary_write_attribute, & 88 binary_write_variable 89 90 IMPLICIT NONE 91 92 INTEGER, PARAMETER :: charlen = 100 !< maximum length of character variables 93 INTEGER, PARAMETER :: no_id = -1 !< default ID if no ID was assigned 94 95 TYPE attribute_type 96 CHARACTER(LEN=charlen) :: data_type = '' !< data type 97 CHARACTER(LEN=charlen) :: name !< attribute name 98 CHARACTER(LEN=charlen) :: value_char !< attribute value if character 99 INTEGER(KIND=1) :: value_int8 !< attribute value if 8bit integer 100 INTEGER(KIND=2) :: value_int16 !< attribute value if 16bit integer 101 INTEGER(KIND=4) :: value_int32 !< attribute value if 32bit integer 102 REAL(KIND=4) :: value_real32 !< attribute value if 32bit real 103 REAL(KIND=8) :: value_real64 !< attribute value if 64bit real 104 END TYPE attribute_type 105 106 TYPE variable_type 107 CHARACTER(LEN=charlen) :: data_type = '' !< data type 108 CHARACTER(LEN=charlen) :: name !< variable name 109 INTEGER :: id = no_id !< id within file 110 LOGICAL :: is_global = .FALSE. !< true if global variable 111 CHARACTER(LEN=charlen), DIMENSION(:), ALLOCATABLE :: dimension_names !< list of dimension names used by variable 112 INTEGER, DIMENSION(:), ALLOCATABLE :: dimension_ids !< list of dimension ids used by variable 113 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes 114 END TYPE variable_type 115 116 TYPE dimension_type 117 CHARACTER(LEN=charlen) :: data_type = '' !< data type 118 CHARACTER(LEN=charlen) :: name !< dimension name 119 INTEGER :: id = no_id !< dimension id within file 120 INTEGER :: length !< length of dimension 121 INTEGER :: length_mask !< length of masked dimension 122 INTEGER :: variable_id = no_id !< associated variable id within file 123 LOGICAL :: is_masked = .FALSE. !< true if masked 124 INTEGER, DIMENSION(2) :: bounds !< lower and upper bound of dimension 125 INTEGER, DIMENSION(:), ALLOCATABLE :: masked_indices !< list of masked indices of dimension 126 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE :: masked_values_int8 !< masked dimension values if 16bit integer 127 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE :: masked_values_int16 !< masked dimension values if 16bit integer 128 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE :: masked_values_int32 !< masked dimension values if 32bit integer 129 INTEGER(iwp), DIMENSION(:), ALLOCATABLE :: masked_values_intwp !< masked dimension values if working-precision int 130 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE :: values_int8 !< dimension values if 16bit integer 131 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE :: values_int16 !< dimension values if 16bit integer 132 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE :: values_int32 !< dimension values if 32bit integer 133 INTEGER(iwp), DIMENSION(:), ALLOCATABLE :: values_intwp !< dimension values if working-precision integer 134 LOGICAL, DIMENSION(:), ALLOCATABLE :: mask !< mask 135 REAL(KIND=4), DIMENSION(:), ALLOCATABLE :: masked_values_real32 !< masked dimension values if 32bit real 136 REAL(KIND=8), DIMENSION(:), ALLOCATABLE :: masked_values_real64 !< masked dimension values if 64bit real 137 REAL(wp), DIMENSION(:), ALLOCATABLE :: masked_values_realwp !< masked dimension values if working-precision real 138 REAL(KIND=4), DIMENSION(:), ALLOCATABLE :: values_real32 !< dimension values if 32bit real 139 REAL(KIND=8), DIMENSION(:), ALLOCATABLE :: values_real64 !< dimension values if 64bit real 140 REAL(wp), DIMENSION(:), ALLOCATABLE :: values_realwp !< dimension values if working-precision real 141 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes 142 END TYPE dimension_type 143 144 TYPE file_type 145 CHARACTER(LEN=charlen) :: format = '' !< file format 146 CHARACTER(LEN=charlen) :: name = '' !< file name 147 INTEGER :: id = no_id !< id of file 148 LOGICAL :: is_init = .FALSE. !< true if initialized 149 TYPE(attribute_type), DIMENSION(:), ALLOCATABLE :: attributes !< list of attributes 150 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimensions !< list of dimensions 151 TYPE(variable_type), DIMENSION(:), ALLOCATABLE :: variables !< list of variables 152 END TYPE file_type 153 154 155 CHARACTER(LEN=charlen) :: output_file_suffix = '' !< file suffix added to each file name 156 CHARACTER(LEN=800) :: internal_error_message = '' !< string containing the last error message 157 CHARACTER(LEN=800) :: temp_string !< dummy string 158 159 INTEGER :: debug_output_unit !< Fortran Unit Number of the debug-output file 160 INTEGER :: nfiles = 0 !< number of files 161 INTEGER :: master_rank = 0 !< master rank for tasks to be executed by single PE only 162 INTEGER :: output_group_comm !< MPI communicator addressing all MPI ranks which participate in output 163 164 LOGICAL :: print_debug_output = .FALSE. !< if true, debug output is printed 165 166 TYPE(file_type), DIMENSION(:), ALLOCATABLE :: files !< file list 167 168 SAVE 169 170 PRIVATE 171 172 !> Initialize the data-output module 173 INTERFACE dom_init 174 MODULE PROCEDURE dom_init 175 END INTERFACE dom_init 176 177 !> Add files to database 178 INTERFACE dom_def_file 179 MODULE PROCEDURE dom_def_file 180 END INTERFACE dom_def_file 181 182 !> Add dimensions to database 183 INTERFACE dom_def_dim 184 MODULE PROCEDURE dom_def_dim 185 END INTERFACE dom_def_dim 186 187 !> Add variables to database 188 INTERFACE dom_def_var 189 MODULE PROCEDURE dom_def_var 190 END INTERFACE dom_def_var 191 192 !> Add attributes to database 193 INTERFACE dom_def_att 194 MODULE PROCEDURE dom_def_att_char 195 MODULE PROCEDURE dom_def_att_int8 196 MODULE PROCEDURE dom_def_att_int16 197 MODULE PROCEDURE dom_def_att_int32 198 MODULE PROCEDURE dom_def_att_real32 199 MODULE PROCEDURE dom_def_att_real64 200 END INTERFACE dom_def_att 201 202 !> Prepare for output: evaluate database and create files 203 INTERFACE dom_def_end 204 MODULE PROCEDURE dom_def_end 205 END INTERFACE dom_def_end 206 207 !> Write variables to file 208 INTERFACE dom_write_var 209 MODULE PROCEDURE dom_write_var 210 END INTERFACE dom_write_var 211 212 !> Last actions required for output befor termination 213 INTERFACE dom_finalize_output 214 MODULE PROCEDURE dom_finalize_output 215 END INTERFACE dom_finalize_output 216 217 !> Return error message 218 INTERFACE dom_get_error_message 219 MODULE PROCEDURE dom_get_error_message 220 END INTERFACE dom_get_error_message 221 222 !> Write database to debug output 223 INTERFACE dom_database_debug_output 224 MODULE PROCEDURE dom_database_debug_output 225 END INTERFACE dom_database_debug_output 226 227 PUBLIC & 228 dom_init, & 229 dom_def_file, & 230 dom_def_dim, & 231 dom_def_var, & 232 dom_def_att, & 233 dom_def_end, & 234 dom_write_var, & 235 dom_finalize_output, & 236 dom_get_error_message, & 237 dom_database_debug_output 238 239 CONTAINS 237 240 238 241 … … 247 250 !> prevents that multiple groups try to open and write to the same output file. 248 251 !--------------------------------------------------------------------------------------------------! 249 SUBROUTINE dom_init( file_suffix_of_output_group, mpi_comm_of_output_group, master_output_rank, &250 program_debug_output_unit, debug_output )251 252 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: file_suffix_of_output_group !< file-name suffix added to each file;253 !> must be unique for each output group254 255 INTEGER, INTENT(IN), OPTIONAL :: master_output_rank !< MPI rank executing tasks which must256 !> be executed by a single PE only257 INTEGER, INTENT(IN) :: mpi_comm_of_output_group !< MPI communicator specifying the MPI group258 !> which participate in the output259 INTEGER, INTENT(IN) :: program_debug_output_unit !< file unit number for debug output260 261 LOGICAL, INTENT(IN) :: debug_output !< if true, debug output is printed262 263 264 IF ( PRESENT( file_suffix_of_output_group ) ) output_file_suffix = file_suffix_of_output_group265 IF ( PRESENT( master_output_rank ) ) master_rank = master_output_rank266 267 output_group_comm = mpi_comm_of_output_group268 269 debug_output_unit = program_debug_output_unit270 print_debug_output = debug_output271 272 CALL binary_init_module( output_file_suffix, output_group_comm, master_rank, &273 debug_output_unit, debug_output, no_id )274 275 CALL netcdf4_init_module( output_file_suffix, output_group_comm, master_rank, &276 debug_output_unit, debug_output, no_id )277 278 END SUBROUTINE dom_init252 SUBROUTINE dom_init( file_suffix_of_output_group, mpi_comm_of_output_group, master_output_rank, & 253 program_debug_output_unit, debug_output ) 254 255 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: file_suffix_of_output_group !< file-name suffix added to each file; 256 !> must be unique for each output group 257 258 INTEGER, INTENT(IN), OPTIONAL :: master_output_rank !< MPI rank executing tasks which must 259 !> be executed by a single PE only 260 INTEGER, INTENT(IN) :: mpi_comm_of_output_group !< MPI communicator specifying the MPI group 261 !> which participate in the output 262 INTEGER, INTENT(IN) :: program_debug_output_unit !< file unit number for debug output 263 264 LOGICAL, INTENT(IN) :: debug_output !< if true, debug output is printed 265 266 267 IF ( PRESENT( file_suffix_of_output_group ) ) output_file_suffix = file_suffix_of_output_group 268 IF ( PRESENT( master_output_rank ) ) master_rank = master_output_rank 269 270 output_group_comm = mpi_comm_of_output_group 271 272 debug_output_unit = program_debug_output_unit 273 print_debug_output = debug_output 274 275 CALL binary_init_module( output_file_suffix, output_group_comm, master_rank, & 276 debug_output_unit, debug_output, no_id ) 277 278 CALL netcdf4_init_module( output_file_suffix, output_group_comm, master_rank, & 279 debug_output_unit, debug_output, no_id ) 280 281 END SUBROUTINE dom_init 279 282 280 283 !--------------------------------------------------------------------------------------------------! … … 285 288 !> status = dom_def_file( 'my_output_file_name', 'binary' ) 286 289 !--------------------------------------------------------------------------------------------------! 287 FUNCTION dom_def_file( file_name, file_format ) RESULT( return_value ) 288 289 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file to be created 290 CHARACTER(LEN=*), INTENT(IN) :: file_format !< format of file to be created 291 292 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_file' !< name of this routine 293 294 INTEGER :: f !< loop index 295 INTEGER :: return_value !< return value 296 297 TYPE(file_type), DIMENSION(:), ALLOCATABLE :: files_tmp !< temporary file list 298 299 300 return_value = 0 301 302 CALL internal_message( 'debug', routine_name // ': define file "' // TRIM( file_name ) // '"' ) 303 304 !-- Allocate file list or extend it by 1 305 IF ( .NOT. ALLOCATED( files ) ) THEN 306 307 nfiles = 1 308 ALLOCATE( files(nfiles) ) 309 310 ELSE 311 312 nfiles = SIZE( files ) 313 !-- Check if file already exists 314 DO f = 1, nfiles 315 IF ( files(f)%name == TRIM( file_name ) ) THEN 316 return_value = 1 317 CALL internal_message( 'error', routine_name // & 318 ': file "' // TRIM( file_name ) // '" already exists' ) 319 EXIT 320 ENDIF 321 ENDDO 322 323 !-- Extend file list 324 IF ( return_value == 0 ) THEN 325 ALLOCATE( files_tmp(nfiles) ) 326 files_tmp = files 327 DEALLOCATE( files ) 328 nfiles = nfiles + 1 329 ALLOCATE( files(nfiles) ) 330 files(:nfiles-1) = files_tmp 331 DEALLOCATE( files_tmp ) 332 ENDIF 333 334 ENDIF 335 336 !-- Add new file to database 337 IF ( return_value == 0 ) THEN 338 files(nfiles)%name = TRIM( file_name ) 339 files(nfiles)%format = TRIM( file_format ) 340 ENDIF 341 342 END FUNCTION dom_def_file 290 FUNCTION dom_def_file( file_name, file_format ) RESULT( return_value ) 291 292 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file to be created 293 CHARACTER(LEN=*), INTENT(IN) :: file_format !< format of file to be created 294 295 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_file' !< name of this routine 296 297 INTEGER :: f !< loop index 298 INTEGER :: return_value !< return value 299 300 TYPE(file_type), DIMENSION(:), ALLOCATABLE :: files_tmp !< temporary file list 301 302 303 return_value = 0 304 305 CALL internal_message( 'debug', routine_name // ': define file "' // TRIM( file_name ) // '"' ) 306 ! 307 !-- Allocate file list or extend it by 1 308 IF ( .NOT. ALLOCATED( files ) ) THEN 309 310 nfiles = 1 311 ALLOCATE( files(nfiles) ) 312 313 ELSE 314 315 nfiles = SIZE( files ) 316 ! 317 !-- Check if file already exists 318 DO f = 1, nfiles 319 IF ( files(f)%name == TRIM( file_name ) ) THEN 320 return_value = 1 321 CALL internal_message( 'error', routine_name // & 322 ': file "' // TRIM( file_name ) // '" already exists' ) 323 EXIT 324 ENDIF 325 ENDDO 326 ! 327 !-- Extend file list 328 IF ( return_value == 0 ) THEN 329 ALLOCATE( files_tmp(nfiles) ) 330 files_tmp = files 331 DEALLOCATE( files ) 332 nfiles = nfiles + 1 333 ALLOCATE( files(nfiles) ) 334 files(:nfiles-1) = files_tmp 335 DEALLOCATE( files_tmp ) 336 ENDIF 337 338 ENDIF 339 ! 340 !-- Add new file to database 341 IF ( return_value == 0 ) THEN 342 files(nfiles)%name = TRIM( file_name ) 343 files(nfiles)%format = TRIM( file_format ) 344 ENDIF 345 346 END FUNCTION dom_def_file 343 347 344 348 !--------------------------------------------------------------------------------------------------! … … 370 374 !> @todo Convert given values into selected output_type. 371 375 !--------------------------------------------------------------------------------------------------! 372 FUNCTION dom_def_dim( file_name, dimension_name, output_type, bounds, & 373 values_int8, values_int16, values_int32, values_intwp, & 374 values_real32, values_real64, values_realwp, & 375 mask ) RESULT( return_value ) 376 377 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 378 CHARACTER(LEN=*), INTENT(IN) :: dimension_name !< name of dimension 379 CHARACTER(LEN=*), INTENT(IN) :: output_type !< data type of dimension variable in output file 380 381 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_dim' !< name of this routine 382 383 INTEGER :: d !< loop index 384 INTEGER :: f !< loop index 385 INTEGER :: i !< loop index 386 INTEGER :: j !< loop index 387 INTEGER :: ndims !< number of dimensions in file 388 INTEGER :: return_value !< return value 389 390 INTEGER, DIMENSION(:), INTENT(IN) :: bounds !< lower and upper bound of dimension variable 391 INTEGER(KIND=1), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int8 !< values of dimension 392 INTEGER(KIND=2), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int16 !< values of dimension 393 INTEGER(KIND=4), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int32 !< values of dimension 394 INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: values_intwp !< values of dimension 395 396 LOGICAL, DIMENSION(:), INTENT(IN), OPTIONAL :: mask !< mask of dimesion 397 398 REAL(KIND=4), DIMENSION(:), INTENT(IN), OPTIONAL :: values_real32 !< values of dimension 399 REAL(KIND=8), DIMENSION(:), INTENT(IN), OPTIONAL :: values_real64 !< values of dimension 400 REAL(wp), DIMENSION(:), INTENT(IN), OPTIONAL :: values_realwp !< values of dimension 401 402 TYPE(dimension_type) :: dimension !< new dimension 403 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimensions_tmp !< temporary dimension list 404 405 406 return_value = 0 407 408 CALL internal_message( 'debug', routine_name // & 409 ': define dimension ' // & 410 '(dimension "' // TRIM( dimension_name ) // & 411 '", file "' // TRIM( file_name ) // '")' ) 412 413 dimension%name = TRIM( dimension_name ) 414 dimension%data_type = TRIM( output_type ) 415 416 !-- Check dimension bounds and allocate dimension according to bounds 417 IF ( SIZE( bounds ) == 1 ) THEN 418 419 !-- Dimension has only lower bound, which means it changes its size 420 !-- during simulation. 421 !-- Set length to -1 as indicator. 422 dimension%bounds(:) = bounds(1) 423 dimension%length = -1 424 425 IF ( PRESENT( mask ) ) THEN 426 return_value = 1 427 CALL internal_message( 'error', routine_name // & 428 ': unlimited dimensions cannot be masked ' // & 429 '(dimension "' // TRIM( dimension_name ) // & 430 '", file "' // TRIM( file_name ) // '")!' ) 431 ENDIF 432 433 ELSEIF ( SIZE( bounds ) == 2 ) THEN 434 435 dimension%bounds = bounds 436 dimension%length = bounds(2) - bounds(1) + 1 437 438 !-- Save dimension values 439 IF ( PRESENT( values_int8 ) ) THEN 440 ALLOCATE( dimension%values_int8(dimension%bounds(1):dimension%bounds(2)) ) 441 IF ( SIZE( values_int8 ) == dimension%length ) THEN 442 dimension%values_int8 = values_int8 443 ELSEIF ( SIZE( values_int8 ) == 1 ) THEN 444 dimension%values_int8(:) = values_int8(1) 445 ELSE 446 return_value = 2 447 ENDIF 448 ELSEIF( PRESENT( values_int16 ) ) THEN 449 ALLOCATE( dimension%values_int16(dimension%bounds(1):dimension%bounds(2)) ) 450 IF ( SIZE( values_int16 ) == dimension%length ) THEN 451 dimension%values_int16 = values_int16 452 ELSEIF ( SIZE( values_int16 ) == 1 ) THEN 453 dimension%values_int16(:) = values_int16(1) 454 ELSE 455 return_value = 2 456 ENDIF 457 ELSEIF( PRESENT( values_int32 ) ) THEN 458 ALLOCATE( dimension%values_int32(dimension%bounds(1):dimension%bounds(2)) ) 459 IF ( SIZE( values_int32 ) == dimension%length ) THEN 460 dimension%values_int32 = values_int32 461 ELSEIF ( SIZE( values_int32 ) == 1 ) THEN 462 dimension%values_int32(:) = values_int32(1) 463 ELSE 464 return_value = 2 465 ENDIF 466 ELSEIF( PRESENT( values_intwp ) ) THEN 467 ALLOCATE( dimension%values_intwp(dimension%bounds(1):dimension%bounds(2)) ) 468 IF ( SIZE( values_intwp ) == dimension%length ) THEN 469 dimension%values_intwp = values_intwp 470 ELSEIF ( SIZE( values_intwp ) == 1 ) THEN 471 dimension%values_intwp(:) = values_intwp(1) 472 ELSE 473 return_value = 2 474 ENDIF 475 ELSEIF( PRESENT( values_real32 ) ) THEN 476 ALLOCATE( dimension%values_real32(dimension%bounds(1):dimension%bounds(2)) ) 477 IF ( SIZE( values_real32 ) == dimension%length ) THEN 478 dimension%values_real32 = values_real32 479 ELSEIF ( SIZE( values_real32 ) == 1 ) THEN 480 dimension%values_real32(:) = values_real32(1) 481 ELSE 482 return_value = 2 483 ENDIF 484 ELSEIF( PRESENT( values_real64 ) ) THEN 485 ALLOCATE( dimension%values_real64(dimension%bounds(1):dimension%bounds(2)) ) 486 IF ( SIZE( values_real64 ) == dimension%length ) THEN 487 dimension%values_real64 = values_real64 488 ELSEIF ( SIZE( values_real64 ) == 1 ) THEN 489 dimension%values_real64(:) = values_real64(1) 490 ELSE 491 return_value = 2 492 ENDIF 493 ELSEIF( PRESENT( values_realwp ) ) THEN 494 ALLOCATE( dimension%values_realwp(dimension%bounds(1):dimension%bounds(2)) ) 495 IF ( SIZE( values_realwp ) == dimension%length ) THEN 496 dimension%values_realwp = values_realwp 497 ELSEIF ( SIZE( values_realwp ) == 1 ) THEN 498 dimension%values_realwp(:) = values_realwp(1) 499 ELSE 500 return_value = 2 501 ENDIF 502 ELSE 503 return_value = 1 504 CALL internal_message( 'error', routine_name // & 505 ': no values given ' // & 506 '(dimension "' // TRIM( dimension_name ) // & 507 '", file "' // TRIM( file_name ) // '")!' ) 508 ENDIF 509 510 IF ( return_value == 2 ) THEN 511 return_value = 1 512 CALL internal_message( 'error', routine_name // & 513 ': number of values and given bounds do not match ' // & 514 '(dimension "' // TRIM( dimension_name ) // & 515 '", file "' // TRIM( file_name ) // '")!' ) 516 ENDIF 517 518 !-- Initialize mask 519 IF ( PRESENT( mask ) .AND. return_value == 0 ) THEN 520 521 IF ( dimension%length == SIZE( mask ) ) THEN 522 523 IF ( ALL( mask ) ) THEN 524 525 CALL internal_message( 'debug', routine_name // & 526 ': mask contains only TRUE values. Ignoring mask ' // & 527 '(dimension "' // TRIM( dimension_name ) // & 528 '", file "' // TRIM( file_name ) // '")!' ) 529 530 ELSE 531 532 dimension%is_masked = .TRUE. 533 dimension%length_mask = COUNT( mask ) 534 535 ALLOCATE( dimension%mask(dimension%bounds(1):dimension%bounds(2)) ) 536 ALLOCATE( dimension%masked_indices(0:dimension%length_mask-1) ) 537 538 dimension%mask = mask 539 540 !-- Save masked positions and masked values 541 IF ( ALLOCATED( dimension%values_int8 ) ) THEN 542 543 ALLOCATE( dimension%masked_values_int8(0:dimension%length_mask-1) ) 544 j = 0 545 DO i = dimension%bounds(1), dimension%bounds(2) 546 IF ( dimension%mask(i) ) THEN 547 dimension%masked_values_int8(j) = dimension%values_int8(i) 548 dimension%masked_indices(j) = i 549 j = j + 1 550 ENDIF 551 ENDDO 552 553 ELSEIF ( ALLOCATED( dimension%values_int16 ) ) THEN 554 555 ALLOCATE( dimension%masked_values_int16(0:dimension%length_mask-1) ) 556 j = 0 557 DO i = dimension%bounds(1), dimension%bounds(2) 558 IF ( dimension%mask(i) ) THEN 559 dimension%masked_values_int16(j) = dimension%values_int16(i) 560 dimension%masked_indices(j) = i 561 j = j + 1 562 ENDIF 563 ENDDO 564 565 ELSEIF ( ALLOCATED( dimension%values_int32 ) ) THEN 566 567 ALLOCATE( dimension%masked_values_int32(0:dimension%length_mask-1) ) 568 j = 0 569 DO i =dimension%bounds(1), dimension%bounds(2) 570 IF ( dimension%mask(i) ) THEN 571 dimension%masked_values_int32(j) = dimension%values_int32(i) 572 dimension%masked_indices(j) = i 573 j = j + 1 574 ENDIF 575 ENDDO 576 577 ELSEIF ( ALLOCATED( dimension%values_intwp ) ) THEN 578 579 ALLOCATE( dimension%masked_values_intwp(0:dimension%length_mask-1) ) 580 j = 0 581 DO i = dimension%bounds(1), dimension%bounds(2) 582 IF ( dimension%mask(i) ) THEN 583 dimension%masked_values_intwp(j) = dimension%values_intwp(i) 584 dimension%masked_indices(j) = i 585 j = j + 1 586 ENDIF 587 ENDDO 588 589 ELSEIF ( ALLOCATED( dimension%values_real32 ) ) THEN 590 591 ALLOCATE( dimension%masked_values_real32(0:dimension%length_mask-1) ) 592 j = 0 593 DO i = dimension%bounds(1), dimension%bounds(2) 594 IF ( dimension%mask(i) ) THEN 595 dimension%masked_values_real32(j) = dimension%values_real32(i) 596 dimension%masked_indices(j) = i 597 j = j + 1 598 ENDIF 599 ENDDO 600 601 ELSEIF ( ALLOCATED(dimension%values_real64) ) THEN 602 603 ALLOCATE( dimension%masked_values_real64(0:dimension%length_mask-1) ) 604 j = 0 605 DO i = dimension%bounds(1), dimension%bounds(2) 606 IF ( dimension%mask(i) ) THEN 607 dimension%masked_values_real64(j) = dimension%values_real64(i) 608 dimension%masked_indices(j) = i 609 j = j + 1 610 ENDIF 611 ENDDO 612 613 ELSEIF ( ALLOCATED(dimension%values_realwp) ) THEN 614 615 ALLOCATE( dimension%masked_values_realwp(0:dimension%length_mask-1) ) 616 j = 0 617 DO i = dimension%bounds(1), dimension%bounds(2) 618 IF ( dimension%mask(i) ) THEN 619 dimension%masked_values_realwp(j) = dimension%values_realwp(i) 620 dimension%masked_indices(j) = i 621 j = j + 1 622 ENDIF 623 ENDDO 624 625 ENDIF 626 627 ENDIF ! if not all mask = true 628 629 ELSE 630 return_value = 1 631 CALL internal_message( 'error', routine_name // & 632 ': size of mask and given bounds do not match ' // & 633 '(dimension "' // TRIM( dimension_name ) // & 634 '", file "' // TRIM( file_name ) // '")!' ) 635 ENDIF 636 637 ENDIF 638 639 ELSE 640 641 return_value = 1 642 CALL internal_message( 'error', routine_name // & 643 ': at least one but no more than two bounds must be given ' // & 644 '(dimension "' // TRIM( dimension_name ) // & 645 '", file "' // TRIM( file_name ) // '")!' ) 646 647 ENDIF 648 649 !-- Add dimension to database 650 IF ( return_value == 0 ) THEN 651 652 DO f = 1, nfiles 653 654 IF ( TRIM( file_name ) == files(f)%name ) THEN 655 656 IF ( files(f)%is_init ) THEN 657 658 return_value = 1 659 CALL internal_message( 'error', routine_name // & 660 ': file already initialized. ' // & 661 'No further dimension definition allowed ' // & 662 '(dimension "' // TRIM( dimension_name ) // & 663 '", file "' // TRIM( file_name ) // '")!' ) 664 EXIT 665 666 ELSEIF ( .NOT. ALLOCATED( files(f)%dimensions ) ) THEN 667 668 ndims = 1 669 ALLOCATE( files(f)%dimensions(ndims) ) 670 671 ELSE 672 673 !-- Check if any variable of the same name as the new dimension is already defined 674 IF ( ALLOCATED( files(f)%variables ) ) THEN 675 DO i = 1, SIZE( files(f)%variables ) 676 IF ( files(f)%variables(i)%name == dimension%name ) THEN 677 return_value = 1 678 CALL internal_message( 'error', routine_name // & 679 ': file already has a variable of this name defined. ' // & 680 'Defining a dimension of the same name is not allowed ' // & 681 '(dimension "' // TRIM( dimension_name ) // & 682 '", file "' // TRIM( file_name ) // '")!' ) 683 EXIT 684 ENDIF 685 ENDDO 686 ENDIF 687 688 IF ( return_value == 0 ) THEN 689 !-- Check if dimension already exists in file 690 ndims = SIZE( files(f)%dimensions ) 691 692 DO d = 1, ndims 693 IF ( files(f)%dimensions(d)%name == dimension%name ) THEN 694 return_value = 1 695 CALL internal_message( 'error', routine_name // & 696 ': dimension already exists in file ' // & 697 '(dimension "' // TRIM( dimension_name ) // & 698 '", file "' // TRIM( file_name ) // '")!' ) 699 EXIT 700 ENDIF 701 ENDDO 702 703 !-- Extend dimension list 704 IF ( return_value == 0 ) THEN 705 ALLOCATE( dimensions_tmp(ndims) ) 706 dimensions_tmp = files(f)%dimensions 707 DEALLOCATE( files(f)%dimensions ) 708 ndims = ndims + 1 709 ALLOCATE( files(f)%dimensions(ndims) ) 710 files(f)%dimensions(:ndims-1) = dimensions_tmp 711 DEALLOCATE( dimensions_tmp ) 712 ENDIF 713 ENDIF 714 715 ENDIF 716 717 !-- Add new dimension to database 718 IF ( return_value == 0 ) files(f)%dimensions(ndims) = dimension 719 720 EXIT 721 722 ENDIF 723 ENDDO 724 725 IF ( f > nfiles ) THEN 726 return_value = 1 727 CALL internal_message( 'error', routine_name // & 728 ': file not found (dimension "' // TRIM( dimension_name ) // & 729 '", file "' // TRIM( file_name ) // '")!' ) 730 ENDIF 731 732 ENDIF 733 734 END FUNCTION dom_def_dim 376 FUNCTION dom_def_dim( file_name, dimension_name, output_type, bounds, & 377 values_int8, values_int16, values_int32, values_intwp, & 378 values_real32, values_real64, values_realwp, & 379 mask ) RESULT( return_value ) 380 381 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 382 CHARACTER(LEN=*), INTENT(IN) :: dimension_name !< name of dimension 383 CHARACTER(LEN=*), INTENT(IN) :: output_type !< data type of dimension variable in output file 384 385 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_dim' !< name of this routine 386 387 INTEGER :: d !< loop index 388 INTEGER :: f !< loop index 389 INTEGER :: i !< loop index 390 INTEGER :: j !< loop index 391 INTEGER :: ndims !< number of dimensions in file 392 INTEGER :: return_value !< return value 393 394 INTEGER, DIMENSION(:), INTENT(IN) :: bounds !< lower and upper bound of dimension variable 395 INTEGER(KIND=1), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int8 !< values of dimension 396 INTEGER(KIND=2), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int16 !< values of dimension 397 INTEGER(KIND=4), DIMENSION(:), INTENT(IN), OPTIONAL :: values_int32 !< values of dimension 398 INTEGER(iwp), DIMENSION(:), INTENT(IN), OPTIONAL :: values_intwp !< values of dimension 399 400 LOGICAL, DIMENSION(:), INTENT(IN), OPTIONAL :: mask !< mask of dimesion 401 402 REAL(KIND=4), DIMENSION(:), INTENT(IN), OPTIONAL :: values_real32 !< values of dimension 403 REAL(KIND=8), DIMENSION(:), INTENT(IN), OPTIONAL :: values_real64 !< values of dimension 404 REAL(wp), DIMENSION(:), INTENT(IN), OPTIONAL :: values_realwp !< values of dimension 405 406 TYPE(dimension_type) :: dimension !< new dimension 407 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimensions_tmp !< temporary dimension list 408 409 410 return_value = 0 411 412 CALL internal_message( 'debug', routine_name // & 413 ': define dimension ' // & 414 '(dimension "' // TRIM( dimension_name ) // & 415 '", file "' // TRIM( file_name ) // '")' ) 416 417 dimension%name = TRIM( dimension_name ) 418 dimension%data_type = TRIM( output_type ) 419 ! 420 !-- Check dimension bounds and allocate dimension according to bounds 421 IF ( SIZE( bounds ) == 1 ) THEN 422 ! 423 !-- Dimension has only lower bound, which means it changes its size 424 !-- during simulation. 425 !-- Set length to -1 as indicator. 426 dimension%bounds(:) = bounds(1) 427 dimension%length = -1 428 429 IF ( PRESENT( mask ) ) THEN 430 return_value = 1 431 CALL internal_message( 'error', routine_name // & 432 ': unlimited dimensions cannot be masked ' // & 433 '(dimension "' // TRIM( dimension_name ) // & 434 '", file "' // TRIM( file_name ) // '")!' ) 435 ENDIF 436 437 ELSEIF ( SIZE( bounds ) == 2 ) THEN 438 439 dimension%bounds = bounds 440 dimension%length = bounds(2) - bounds(1) + 1 441 ! 442 !-- Save dimension values 443 IF ( PRESENT( values_int8 ) ) THEN 444 ALLOCATE( dimension%values_int8(dimension%bounds(1):dimension%bounds(2)) ) 445 IF ( SIZE( values_int8 ) == dimension%length ) THEN 446 dimension%values_int8 = values_int8 447 ELSEIF ( SIZE( values_int8 ) == 1 ) THEN 448 dimension%values_int8(:) = values_int8(1) 449 ELSE 450 return_value = 2 451 ENDIF 452 ELSEIF( PRESENT( values_int16 ) ) THEN 453 ALLOCATE( dimension%values_int16(dimension%bounds(1):dimension%bounds(2)) ) 454 IF ( SIZE( values_int16 ) == dimension%length ) THEN 455 dimension%values_int16 = values_int16 456 ELSEIF ( SIZE( values_int16 ) == 1 ) THEN 457 dimension%values_int16(:) = values_int16(1) 458 ELSE 459 return_value = 2 460 ENDIF 461 ELSEIF( PRESENT( values_int32 ) ) THEN 462 ALLOCATE( dimension%values_int32(dimension%bounds(1):dimension%bounds(2)) ) 463 IF ( SIZE( values_int32 ) == dimension%length ) THEN 464 dimension%values_int32 = values_int32 465 ELSEIF ( SIZE( values_int32 ) == 1 ) THEN 466 dimension%values_int32(:) = values_int32(1) 467 ELSE 468 return_value = 2 469 ENDIF 470 ELSEIF( PRESENT( values_intwp ) ) THEN 471 ALLOCATE( dimension%values_intwp(dimension%bounds(1):dimension%bounds(2)) ) 472 IF ( SIZE( values_intwp ) == dimension%length ) THEN 473 dimension%values_intwp = values_intwp 474 ELSEIF ( SIZE( values_intwp ) == 1 ) THEN 475 dimension%values_intwp(:) = values_intwp(1) 476 ELSE 477 return_value = 2 478 ENDIF 479 ELSEIF( PRESENT( values_real32 ) ) THEN 480 ALLOCATE( dimension%values_real32(dimension%bounds(1):dimension%bounds(2)) ) 481 IF ( SIZE( values_real32 ) == dimension%length ) THEN 482 dimension%values_real32 = values_real32 483 ELSEIF ( SIZE( values_real32 ) == 1 ) THEN 484 dimension%values_real32(:) = values_real32(1) 485 ELSE 486 return_value = 2 487 ENDIF 488 ELSEIF( PRESENT( values_real64 ) ) THEN 489 ALLOCATE( dimension%values_real64(dimension%bounds(1):dimension%bounds(2)) ) 490 IF ( SIZE( values_real64 ) == dimension%length ) THEN 491 dimension%values_real64 = values_real64 492 ELSEIF ( SIZE( values_real64 ) == 1 ) THEN 493 dimension%values_real64(:) = values_real64(1) 494 ELSE 495 return_value = 2 496 ENDIF 497 ELSEIF( PRESENT( values_realwp ) ) THEN 498 ALLOCATE( dimension%values_realwp(dimension%bounds(1):dimension%bounds(2)) ) 499 IF ( SIZE( values_realwp ) == dimension%length ) THEN 500 dimension%values_realwp = values_realwp 501 ELSEIF ( SIZE( values_realwp ) == 1 ) THEN 502 dimension%values_realwp(:) = values_realwp(1) 503 ELSE 504 return_value = 2 505 ENDIF 506 ELSE 507 return_value = 1 508 CALL internal_message( 'error', routine_name // & 509 ': no values given ' // & 510 '(dimension "' // TRIM( dimension_name ) // & 511 '", file "' // TRIM( file_name ) // '")!' ) 512 ENDIF 513 514 IF ( return_value == 2 ) THEN 515 return_value = 1 516 CALL internal_message( 'error', routine_name // & 517 ': number of values and given bounds do not match ' // & 518 '(dimension "' // TRIM( dimension_name ) // & 519 '", file "' // TRIM( file_name ) // '")!' ) 520 ENDIF 521 ! 522 !-- Initialize mask 523 IF ( PRESENT( mask ) .AND. return_value == 0 ) THEN 524 525 IF ( dimension%length == SIZE( mask ) ) THEN 526 527 IF ( ALL( mask ) ) THEN 528 529 CALL internal_message( 'debug', routine_name // & 530 ': mask contains only TRUE values. Ignoring mask ' // & 531 '(dimension "' // TRIM( dimension_name ) // & 532 '", file "' // TRIM( file_name ) // '")!' ) 533 534 ELSE 535 536 dimension%is_masked = .TRUE. 537 dimension%length_mask = COUNT( mask ) 538 539 ALLOCATE( dimension%mask(dimension%bounds(1):dimension%bounds(2)) ) 540 ALLOCATE( dimension%masked_indices(0:dimension%length_mask-1) ) 541 542 dimension%mask = mask 543 ! 544 !-- Save masked positions and masked values 545 IF ( ALLOCATED( dimension%values_int8 ) ) THEN 546 547 ALLOCATE( dimension%masked_values_int8(0:dimension%length_mask-1) ) 548 j = 0 549 DO i = dimension%bounds(1), dimension%bounds(2) 550 IF ( dimension%mask(i) ) THEN 551 dimension%masked_values_int8(j) = dimension%values_int8(i) 552 dimension%masked_indices(j) = i 553 j = j + 1 554 ENDIF 555 ENDDO 556 557 ELSEIF ( ALLOCATED( dimension%values_int16 ) ) THEN 558 559 ALLOCATE( dimension%masked_values_int16(0:dimension%length_mask-1) ) 560 j = 0 561 DO i = dimension%bounds(1), dimension%bounds(2) 562 IF ( dimension%mask(i) ) THEN 563 dimension%masked_values_int16(j) = dimension%values_int16(i) 564 dimension%masked_indices(j) = i 565 j = j + 1 566 ENDIF 567 ENDDO 568 569 ELSEIF ( ALLOCATED( dimension%values_int32 ) ) THEN 570 571 ALLOCATE( dimension%masked_values_int32(0:dimension%length_mask-1) ) 572 j = 0 573 DO i =dimension%bounds(1), dimension%bounds(2) 574 IF ( dimension%mask(i) ) THEN 575 dimension%masked_values_int32(j) = dimension%values_int32(i) 576 dimension%masked_indices(j) = i 577 j = j + 1 578 ENDIF 579 ENDDO 580 581 ELSEIF ( ALLOCATED( dimension%values_intwp ) ) THEN 582 583 ALLOCATE( dimension%masked_values_intwp(0:dimension%length_mask-1) ) 584 j = 0 585 DO i = dimension%bounds(1), dimension%bounds(2) 586 IF ( dimension%mask(i) ) THEN 587 dimension%masked_values_intwp(j) = dimension%values_intwp(i) 588 dimension%masked_indices(j) = i 589 j = j + 1 590 ENDIF 591 ENDDO 592 593 ELSEIF ( ALLOCATED( dimension%values_real32 ) ) THEN 594 595 ALLOCATE( dimension%masked_values_real32(0:dimension%length_mask-1) ) 596 j = 0 597 DO i = dimension%bounds(1), dimension%bounds(2) 598 IF ( dimension%mask(i) ) THEN 599 dimension%masked_values_real32(j) = dimension%values_real32(i) 600 dimension%masked_indices(j) = i 601 j = j + 1 602 ENDIF 603 ENDDO 604 605 ELSEIF ( ALLOCATED(dimension%values_real64) ) THEN 606 607 ALLOCATE( dimension%masked_values_real64(0:dimension%length_mask-1) ) 608 j = 0 609 DO i = dimension%bounds(1), dimension%bounds(2) 610 IF ( dimension%mask(i) ) THEN 611 dimension%masked_values_real64(j) = dimension%values_real64(i) 612 dimension%masked_indices(j) = i 613 j = j + 1 614 ENDIF 615 ENDDO 616 617 ELSEIF ( ALLOCATED(dimension%values_realwp) ) THEN 618 619 ALLOCATE( dimension%masked_values_realwp(0:dimension%length_mask-1) ) 620 j = 0 621 DO i = dimension%bounds(1), dimension%bounds(2) 622 IF ( dimension%mask(i) ) THEN 623 dimension%masked_values_realwp(j) = dimension%values_realwp(i) 624 dimension%masked_indices(j) = i 625 j = j + 1 626 ENDIF 627 ENDDO 628 629 ENDIF 630 631 ENDIF ! if not all mask = true 632 633 ELSE 634 return_value = 1 635 CALL internal_message( 'error', routine_name // & 636 ': size of mask and given bounds do not match ' // & 637 '(dimension "' // TRIM( dimension_name ) // & 638 '", file "' // TRIM( file_name ) // '")!' ) 639 ENDIF 640 641 ENDIF 642 643 ELSE 644 645 return_value = 1 646 CALL internal_message( 'error', routine_name // & 647 ': at least one but no more than two bounds must be given ' // & 648 '(dimension "' // TRIM( dimension_name ) // & 649 '", file "' // TRIM( file_name ) // '")!' ) 650 651 ENDIF 652 ! 653 !-- Add dimension to database 654 IF ( return_value == 0 ) THEN 655 656 DO f = 1, nfiles 657 658 IF ( TRIM( file_name ) == files(f)%name ) THEN 659 660 IF ( files(f)%is_init ) THEN 661 662 return_value = 1 663 CALL internal_message( 'error', routine_name // & 664 ': file already initialized. ' // & 665 'No further dimension definition allowed ' // & 666 '(dimension "' // TRIM( dimension_name ) // & 667 '", file "' // TRIM( file_name ) // '")!' ) 668 EXIT 669 670 ELSEIF ( .NOT. ALLOCATED( files(f)%dimensions ) ) THEN 671 672 ndims = 1 673 ALLOCATE( files(f)%dimensions(ndims) ) 674 675 ELSE 676 ! 677 !-- Check if any variable of the same name as the new dimension is already defined 678 IF ( ALLOCATED( files(f)%variables ) ) THEN 679 DO i = 1, SIZE( files(f)%variables ) 680 IF ( files(f)%variables(i)%name == dimension%name ) THEN 681 return_value = 1 682 CALL internal_message( 'error', routine_name // & 683 ': file already has a variable of this name defined. ' // & 684 'Defining a dimension of the same name is not allowed ' // & 685 '(dimension "' // TRIM( dimension_name ) // & 686 '", file "' // TRIM( file_name ) // '")!' ) 687 EXIT 688 ENDIF 689 ENDDO 690 ENDIF 691 692 IF ( return_value == 0 ) THEN 693 ! 694 !-- Check if dimension already exists in file 695 ndims = SIZE( files(f)%dimensions ) 696 697 DO d = 1, ndims 698 IF ( files(f)%dimensions(d)%name == dimension%name ) THEN 699 return_value = 1 700 CALL internal_message( 'error', routine_name // & 701 ': dimension already exists in file ' // & 702 '(dimension "' // TRIM( dimension_name ) // & 703 '", file "' // TRIM( file_name ) // '")!' ) 704 EXIT 705 ENDIF 706 ENDDO 707 ! 708 !-- Extend dimension list 709 IF ( return_value == 0 ) THEN 710 ALLOCATE( dimensions_tmp(ndims) ) 711 dimensions_tmp = files(f)%dimensions 712 DEALLOCATE( files(f)%dimensions ) 713 ndims = ndims + 1 714 ALLOCATE( files(f)%dimensions(ndims) ) 715 files(f)%dimensions(:ndims-1) = dimensions_tmp 716 DEALLOCATE( dimensions_tmp ) 717 ENDIF 718 ENDIF 719 720 ENDIF 721 ! 722 !-- Add new dimension to database 723 IF ( return_value == 0 ) files(f)%dimensions(ndims) = dimension 724 725 EXIT 726 727 ENDIF 728 ENDDO 729 730 IF ( f > nfiles ) THEN 731 return_value = 1 732 CALL internal_message( 'error', routine_name // & 733 ': file not found (dimension "' // TRIM( dimension_name ) // & 734 '", file "' // TRIM( file_name ) // '")!' ) 735 ENDIF 736 737 ENDIF 738 739 END FUNCTION dom_def_dim 735 740 736 741 !--------------------------------------------------------------------------------------------------! … … 759 764 !> ALLOCATE( u(<z>,<y>,<x>) ) 760 765 !--------------------------------------------------------------------------------------------------! 761 FUNCTION dom_def_var( file_name, variable_name, dimension_names, output_type, is_global ) & 762 RESULT( return_value ) 763 764 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 765 CHARACTER(LEN=*), INTENT(IN) :: variable_name !< name of variable 766 CHARACTER(LEN=*), INTENT(IN) :: output_type !< data type of variable 767 768 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_var' !< name of this routine 769 770 CHARACTER(LEN=*), DIMENSION(:), INTENT(IN) :: dimension_names !< list of dimension names 771 772 INTEGER :: d !< loop index 773 INTEGER :: f !< loop index 774 INTEGER :: i !< loop index 775 INTEGER :: nvars !< number of variables in file 776 INTEGER :: return_value !< return value 777 778 LOGICAL :: found !< true if requested dimension is defined in file 779 LOGICAL, INTENT(IN), OPTIONAL :: is_global !< true if variable is global (same on all PE) 780 781 TYPE(variable_type) :: variable !< new variable 782 TYPE(variable_type), DIMENSION(:), ALLOCATABLE :: variables_tmp !< temporary variable list 783 784 785 return_value = 0 786 found = .FALSE. 787 788 CALL internal_message( 'debug', routine_name // & 789 ': define variable (variable "' // TRIM( variable_name ) // & 790 '", file "' // TRIM( file_name ) // '")' ) 791 792 variable%name = TRIM( variable_name ) 793 794 ALLOCATE( variable%dimension_names(SIZE( dimension_names )) ) 795 ALLOCATE( variable%dimension_ids(SIZE( dimension_names )) ) 796 797 variable%dimension_names = dimension_names 798 variable%dimension_ids = -1 799 variable%data_type = TRIM( output_type ) 800 801 IF ( PRESENT( is_global ) ) THEN 802 variable%is_global = is_global 803 ELSE 804 variable%is_global = .FALSE. 805 ENDIF 806 807 !-- Add variable to database 808 DO f = 1, nfiles 809 810 IF ( TRIM( file_name ) == files(f)%name ) THEN 811 812 IF ( files(f)%is_init ) THEN 813 814 return_value = 1 815 CALL internal_message( 'error', routine_name // & 816 ': file already initialized. No further variable definition allowed ' // & 817 '(variable "' // TRIM( variable_name ) // & 818 '", file "' // TRIM( file_name ) // '")!' ) 819 EXIT 820 821 ELSEIF ( ALLOCATED( files(f)%dimensions ) ) THEN 822 823 !-- Check if any dimension of the same name as the new variable is already defined 824 DO d = 1, SIZE( files(f)%dimensions ) 825 IF ( files(f)%dimensions(d)%name == variable%name ) THEN 826 return_value = 1 827 CALL internal_message( 'error', routine_name // & 828 ': file already has a dimension of this name defined. ' // & 829 'Defining a variable of the same name is not allowed ' // & 830 '(variable "' // TRIM( variable_name ) // & 831 '", file "' // TRIM( file_name ) // '")!' ) 832 EXIT 833 ENDIF 834 ENDDO 835 836 !-- Check if dimensions assigned to variable are defined within file 837 IF ( return_value == 0 ) THEN 838 DO i = 1, SIZE( variable%dimension_names ) 839 found = .FALSE. 840 DO d = 1, SIZE( files(f)%dimensions ) 841 IF ( files(f)%dimensions(d)%name == variable%dimension_names(i) ) THEN 842 found = .TRUE. 843 EXIT 844 ENDIF 845 ENDDO 846 IF ( .NOT. found ) THEN 847 return_value = 1 848 CALL internal_message( 'error', routine_name // & 849 ': required dimension "'// TRIM( variable%dimension_names(i) ) // & 850 '" for variable is not defined ' // & 851 '(variable "' // TRIM( variable_name ) // & 852 '", file "' // TRIM( file_name ) // '")!' ) 853 EXIT 854 ENDIF 855 ENDDO 856 ENDIF 857 858 ELSE 859 860 return_value = 1 861 CALL internal_message( 'error', routine_name // & 862 ': no dimensions defined in file. Cannot define variable '// & 863 '(variable "' // TRIM( variable_name ) // & 864 '", file "' // TRIM( file_name ) // '")!' ) 865 866 ENDIF 867 868 IF ( return_value == 0 ) THEN 869 870 !-- Check if variable already exists 871 IF ( .NOT. ALLOCATED( files(f)%variables ) ) THEN 872 873 nvars = 1 874 ALLOCATE( files(f)%variables(nvars) ) 875 876 ELSE 877 878 nvars = SIZE( files(f)%variables ) 879 DO i = 1, nvars 880 IF ( files(f)%variables(i)%name == variable%name ) THEN 881 return_value = 1 882 CALL internal_message( 'error', routine_name // & 883 ': variable already exists '// & 884 '(variable "' // TRIM( variable_name ) // & 885 '", file "' // TRIM( file_name ) // '")!' ) 886 EXIT 887 ENDIF 888 ENDDO 889 890 IF ( return_value == 0 ) THEN 891 !-- Extend variable list 892 ALLOCATE( variables_tmp(nvars) ) 893 variables_tmp = files(f)%variables 894 DEALLOCATE( files(f)%variables ) 895 nvars = nvars + 1 896 ALLOCATE( files(f)%variables(nvars) ) 897 files(f)%variables(:nvars-1) = variables_tmp 898 DEALLOCATE( variables_tmp ) 899 ENDIF 900 901 ENDIF 902 903 !-- Add new variable to database 904 IF ( return_value == 0 ) files(f)%variables(nvars) = variable 905 906 ENDIF 907 908 EXIT 909 910 ENDIF 911 912 ENDDO 913 914 IF ( f > nfiles ) THEN 915 return_value = 1 916 CALL internal_message( 'error', routine_name // & 917 ': file not found (variable "' // TRIM( variable_name ) // & 918 '", file "' // TRIM( file_name ) // '")!' ) 919 ENDIF 920 921 END FUNCTION dom_def_var 766 FUNCTION dom_def_var( file_name, variable_name, dimension_names, output_type, is_global ) & 767 RESULT( return_value ) 768 769 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 770 CHARACTER(LEN=*), INTENT(IN) :: variable_name !< name of variable 771 CHARACTER(LEN=*), INTENT(IN) :: output_type !< data type of variable 772 773 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_var' !< name of this routine 774 775 CHARACTER(LEN=*), DIMENSION(:), INTENT(IN) :: dimension_names !< list of dimension names 776 777 INTEGER :: d !< loop index 778 INTEGER :: f !< loop index 779 INTEGER :: i !< loop index 780 INTEGER :: nvars !< number of variables in file 781 INTEGER :: return_value !< return value 782 783 LOGICAL :: found !< true if requested dimension is defined in file 784 LOGICAL, INTENT(IN), OPTIONAL :: is_global !< true if variable is global (same on all PE) 785 786 TYPE(variable_type) :: variable !< new variable 787 TYPE(variable_type), DIMENSION(:), ALLOCATABLE :: variables_tmp !< temporary variable list 788 789 790 return_value = 0 791 found = .FALSE. 792 793 CALL internal_message( 'debug', routine_name // & 794 ': define variable (variable "' // TRIM( variable_name ) // & 795 '", file "' // TRIM( file_name ) // '")' ) 796 797 variable%name = TRIM( variable_name ) 798 799 ALLOCATE( variable%dimension_names(SIZE( dimension_names )) ) 800 ALLOCATE( variable%dimension_ids(SIZE( dimension_names )) ) 801 802 variable%dimension_names = dimension_names 803 variable%dimension_ids = -1 804 variable%data_type = TRIM( output_type ) 805 806 IF ( PRESENT( is_global ) ) THEN 807 variable%is_global = is_global 808 ELSE 809 variable%is_global = .FALSE. 810 ENDIF 811 ! 812 !-- Add variable to database 813 DO f = 1, nfiles 814 815 IF ( TRIM( file_name ) == files(f)%name ) THEN 816 817 IF ( files(f)%is_init ) THEN 818 819 return_value = 1 820 CALL internal_message( 'error', routine_name // & 821 ': file already initialized. No further variable definition allowed ' // & 822 '(variable "' // TRIM( variable_name ) // & 823 '", file "' // TRIM( file_name ) // '")!' ) 824 EXIT 825 826 ELSEIF ( ALLOCATED( files(f)%dimensions ) ) THEN 827 ! 828 !-- Check if any dimension of the same name as the new variable is already defined 829 DO d = 1, SIZE( files(f)%dimensions ) 830 IF ( files(f)%dimensions(d)%name == variable%name ) THEN 831 return_value = 1 832 CALL internal_message( 'error', routine_name // & 833 ': file already has a dimension of this name defined. ' // & 834 'Defining a variable of the same name is not allowed ' // & 835 '(variable "' // TRIM( variable_name ) // & 836 '", file "' // TRIM( file_name ) // '")!' ) 837 EXIT 838 ENDIF 839 ENDDO 840 ! 841 !-- Check if dimensions assigned to variable are defined within file 842 IF ( return_value == 0 ) THEN 843 DO i = 1, SIZE( variable%dimension_names ) 844 found = .FALSE. 845 DO d = 1, SIZE( files(f)%dimensions ) 846 IF ( files(f)%dimensions(d)%name == variable%dimension_names(i) ) THEN 847 found = .TRUE. 848 EXIT 849 ENDIF 850 ENDDO 851 IF ( .NOT. found ) THEN 852 return_value = 1 853 CALL internal_message( 'error', routine_name // & 854 ': required dimension "'// TRIM( variable%dimension_names(i) ) // & 855 '" for variable is not defined ' // & 856 '(variable "' // TRIM( variable_name ) // & 857 '", file "' // TRIM( file_name ) // '")!' ) 858 EXIT 859 ENDIF 860 ENDDO 861 ENDIF 862 863 ELSE 864 865 return_value = 1 866 CALL internal_message( 'error', routine_name // & 867 ': no dimensions defined in file. Cannot define variable '// & 868 '(variable "' // TRIM( variable_name ) // & 869 '", file "' // TRIM( file_name ) // '")!' ) 870 871 ENDIF 872 873 IF ( return_value == 0 ) THEN 874 ! 875 !-- Check if variable already exists 876 IF ( .NOT. ALLOCATED( files(f)%variables ) ) THEN 877 878 nvars = 1 879 ALLOCATE( files(f)%variables(nvars) ) 880 881 ELSE 882 883 nvars = SIZE( files(f)%variables ) 884 DO i = 1, nvars 885 IF ( files(f)%variables(i)%name == variable%name ) THEN 886 return_value = 1 887 CALL internal_message( 'error', routine_name // & 888 ': variable already exists '// & 889 '(variable "' // TRIM( variable_name ) // & 890 '", file "' // TRIM( file_name ) // '")!' ) 891 EXIT 892 ENDIF 893 ENDDO 894 895 IF ( return_value == 0 ) THEN 896 ! 897 !-- Extend variable list 898 ALLOCATE( variables_tmp(nvars) ) 899 variables_tmp = files(f)%variables 900 DEALLOCATE( files(f)%variables ) 901 nvars = nvars + 1 902 ALLOCATE( files(f)%variables(nvars) ) 903 files(f)%variables(:nvars-1) = variables_tmp 904 DEALLOCATE( variables_tmp ) 905 ENDIF 906 907 ENDIF 908 ! 909 !-- Add new variable to database 910 IF ( return_value == 0 ) files(f)%variables(nvars) = variable 911 912 ENDIF 913 914 EXIT 915 916 ENDIF 917 918 ENDDO 919 920 IF ( f > nfiles ) THEN 921 return_value = 1 922 CALL internal_message( 'error', routine_name // & 923 ': file not found (variable "' // TRIM( variable_name ) // & 924 '", file "' // TRIM( file_name ) // '")!' ) 925 ENDIF 926 927 END FUNCTION dom_def_var 922 928 923 929 !--------------------------------------------------------------------------------------------------! … … 946 952 !> value=' and this part was appended', append=.TRUE. ) 947 953 !--------------------------------------------------------------------------------------------------! 948 FUNCTION dom_def_att_char( file_name, variable_name, attribute_name, value, append ) &949 RESULT( return_value )950 951 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file952 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute953 CHARACTER(LEN=*), INTENT(IN) :: value !< attribute value954 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable955 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name956 957 ! CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_char' !< name of routine958 959 INTEGER :: return_value !< return value960 961 LOGICAL :: append_internal !< same as 'append'962 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value963 964 TYPE(attribute_type) :: attribute !< new attribute965 966 967 return_value = 0968 969 IF ( PRESENT( append ) ) THEN970 append_internal = append971 ELSE972 append_internal = .FALSE.973 ENDIF974 975 attribute%name = TRIM( attribute_name )976 attribute%data_type = 'char'977 attribute%value_char = TRIM( value )978 979 IF ( PRESENT( variable_name ) ) THEN980 variable_name_internal = TRIM( variable_name )981 ELSE982 variable_name_internal = ''983 ENDIF984 985 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &986 variable_name=TRIM( variable_name_internal ), &987 attribute=attribute, append=append_internal )988 989 END FUNCTION dom_def_att_char954 FUNCTION dom_def_att_char( file_name, variable_name, attribute_name, value, append ) & 955 RESULT( return_value ) 956 957 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 958 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 959 CHARACTER(LEN=*), INTENT(IN) :: value !< attribute value 960 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 961 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 962 963 ! CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_char' !< name of routine 964 965 INTEGER :: return_value !< return value 966 967 LOGICAL :: append_internal !< same as 'append' 968 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 969 970 TYPE(attribute_type) :: attribute !< new attribute 971 972 973 return_value = 0 974 975 IF ( PRESENT( append ) ) THEN 976 append_internal = append 977 ELSE 978 append_internal = .FALSE. 979 ENDIF 980 981 attribute%name = TRIM( attribute_name ) 982 attribute%data_type = 'char' 983 attribute%value_char = TRIM( value ) 984 985 IF ( PRESENT( variable_name ) ) THEN 986 variable_name_internal = TRIM( variable_name ) 987 ELSE 988 variable_name_internal = '' 989 ENDIF 990 991 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 992 variable_name=TRIM( variable_name_internal ), & 993 attribute=attribute, append=append_internal ) 994 995 END FUNCTION dom_def_att_char 990 996 991 997 !--------------------------------------------------------------------------------------------------! … … 1008 1014 !> value=1_1 ) 1009 1015 !--------------------------------------------------------------------------------------------------! 1010 FUNCTION dom_def_att_int8( file_name, variable_name, attribute_name, value, append ) &1011 RESULT( return_value )1012 1013 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file1014 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute1015 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable1016 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name1017 1018 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int8' !< name of routine1019 1020 INTEGER(KIND=1), INTENT(IN) :: value !< attribute value1021 1022 INTEGER :: return_value !< return value1023 1024 LOGICAL :: append_internal !< same as 'append'1025 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value1026 1027 TYPE(attribute_type) :: attribute !< new attribute1028 1029 1030 return_value = 01031 1032 IF ( PRESENT( variable_name ) ) THEN1033 variable_name_internal = TRIM( variable_name )1034 ELSE1035 variable_name_internal = ''1036 ENDIF1037 1038 IF ( PRESENT( append ) ) THEN1039 IF ( append ) THEN1040 return_value = 11041 CALL internal_message( 'error', routine_name // &1042 ': numeric attribute cannot be appended ' // &1043 '(attribute "' // TRIM( attribute_name ) // &1044 '", variable "' // TRIM( variable_name_internal ) // &1045 '", file "' // TRIM( file_name ) // '")!' )1046 ENDIF1047 ENDIF1048 1049 IF ( return_value == 0 ) THEN1050 append_internal = .FALSE.1051 1052 attribute%name = TRIM( attribute_name )1053 attribute%data_type = 'int8'1054 attribute%value_int8 = value1055 1056 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &1057 variable_name=TRIM( variable_name_internal ), &1058 attribute=attribute, append=append_internal )1059 ENDIF1060 1061 END FUNCTION dom_def_att_int81016 FUNCTION dom_def_att_int8( file_name, variable_name, attribute_name, value, append ) & 1017 RESULT( return_value ) 1018 1019 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1020 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 1021 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 1022 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 1023 1024 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int8' !< name of routine 1025 1026 INTEGER(KIND=1), INTENT(IN) :: value !< attribute value 1027 1028 INTEGER :: return_value !< return value 1029 1030 LOGICAL :: append_internal !< same as 'append' 1031 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 1032 1033 TYPE(attribute_type) :: attribute !< new attribute 1034 1035 1036 return_value = 0 1037 1038 IF ( PRESENT( variable_name ) ) THEN 1039 variable_name_internal = TRIM( variable_name ) 1040 ELSE 1041 variable_name_internal = '' 1042 ENDIF 1043 1044 IF ( PRESENT( append ) ) THEN 1045 IF ( append ) THEN 1046 return_value = 1 1047 CALL internal_message( 'error', routine_name // & 1048 ': numeric attribute cannot be appended ' // & 1049 '(attribute "' // TRIM( attribute_name ) // & 1050 '", variable "' // TRIM( variable_name_internal ) // & 1051 '", file "' // TRIM( file_name ) // '")!' ) 1052 ENDIF 1053 ENDIF 1054 1055 IF ( return_value == 0 ) THEN 1056 append_internal = .FALSE. 1057 1058 attribute%name = TRIM( attribute_name ) 1059 attribute%data_type = 'int8' 1060 attribute%value_int8 = value 1061 1062 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 1063 variable_name=TRIM( variable_name_internal ), & 1064 attribute=attribute, append=append_internal ) 1065 ENDIF 1066 1067 END FUNCTION dom_def_att_int8 1062 1068 1063 1069 !--------------------------------------------------------------------------------------------------! … … 1080 1086 !> value=1_2 ) 1081 1087 !--------------------------------------------------------------------------------------------------! 1082 FUNCTION dom_def_att_int16( file_name, variable_name, attribute_name, value, append ) &1083 RESULT( return_value )1084 1085 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file1086 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute1087 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable1088 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name1089 1090 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int16' !< name of routine1091 1092 INTEGER(KIND=2), INTENT(IN) :: value !< attribute value1093 1094 INTEGER :: return_value !< return value1095 1096 LOGICAL :: append_internal !< same as 'append'1097 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value1098 1099 TYPE(attribute_type) :: attribute !< new attribute1100 1101 1102 return_value = 01103 1104 IF ( PRESENT( variable_name ) ) THEN1105 variable_name_internal = TRIM( variable_name )1106 ELSE1107 variable_name_internal = ''1108 ENDIF1109 1110 IF ( PRESENT( append ) ) THEN1111 IF ( append ) THEN1112 return_value = 11113 CALL internal_message( 'error', routine_name // &1114 ': numeric attribute cannot be appended ' // &1115 '(attribute "' // TRIM( attribute_name ) // &1116 '", variable "' // TRIM( variable_name_internal ) // &1117 '", file "' // TRIM( file_name ) // '")!' )1118 ENDIF1119 ENDIF1120 1121 IF ( return_value == 0 ) THEN1122 append_internal = .FALSE.1123 1124 attribute%name = TRIM( attribute_name )1125 attribute%data_type = 'int16'1126 attribute%value_int16 = value1127 1128 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &1129 variable_name=TRIM( variable_name_internal ), &1130 attribute=attribute, append=append_internal )1131 ENDIF1132 1133 END FUNCTION dom_def_att_int161088 FUNCTION dom_def_att_int16( file_name, variable_name, attribute_name, value, append ) & 1089 RESULT( return_value ) 1090 1091 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1092 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 1093 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 1094 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 1095 1096 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int16' !< name of routine 1097 1098 INTEGER(KIND=2), INTENT(IN) :: value !< attribute value 1099 1100 INTEGER :: return_value !< return value 1101 1102 LOGICAL :: append_internal !< same as 'append' 1103 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 1104 1105 TYPE(attribute_type) :: attribute !< new attribute 1106 1107 1108 return_value = 0 1109 1110 IF ( PRESENT( variable_name ) ) THEN 1111 variable_name_internal = TRIM( variable_name ) 1112 ELSE 1113 variable_name_internal = '' 1114 ENDIF 1115 1116 IF ( PRESENT( append ) ) THEN 1117 IF ( append ) THEN 1118 return_value = 1 1119 CALL internal_message( 'error', routine_name // & 1120 ': numeric attribute cannot be appended ' // & 1121 '(attribute "' // TRIM( attribute_name ) // & 1122 '", variable "' // TRIM( variable_name_internal ) // & 1123 '", file "' // TRIM( file_name ) // '")!' ) 1124 ENDIF 1125 ENDIF 1126 1127 IF ( return_value == 0 ) THEN 1128 append_internal = .FALSE. 1129 1130 attribute%name = TRIM( attribute_name ) 1131 attribute%data_type = 'int16' 1132 attribute%value_int16 = value 1133 1134 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 1135 variable_name=TRIM( variable_name_internal ), & 1136 attribute=attribute, append=append_internal ) 1137 ENDIF 1138 1139 END FUNCTION dom_def_att_int16 1134 1140 1135 1141 !--------------------------------------------------------------------------------------------------! … … 1152 1158 !> value=1_4 ) 1153 1159 !--------------------------------------------------------------------------------------------------! 1154 FUNCTION dom_def_att_int32( file_name, variable_name, attribute_name, value, append ) &1155 RESULT( return_value )1156 1157 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file1158 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute1159 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable1160 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name1161 1162 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int32' !< name of routine1163 1164 INTEGER(KIND=4), INTENT(IN) :: value !< attribute value1165 1166 INTEGER :: return_value !< return value1167 1168 LOGICAL :: append_internal !< same as 'append'1169 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value1170 1171 TYPE(attribute_type) :: attribute !< new attribute1172 1173 1174 return_value = 01175 1176 IF ( PRESENT( variable_name ) ) THEN1177 variable_name_internal = TRIM( variable_name )1178 ELSE1179 variable_name_internal = ''1180 ENDIF1181 1182 IF ( PRESENT( append ) ) THEN1183 IF ( append ) THEN1184 return_value = 11185 CALL internal_message( 'error', routine_name // &1186 ': numeric attribute cannot be appended ' // &1187 '(attribute "' // TRIM( attribute_name ) // &1188 '", variable "' // TRIM( variable_name_internal ) // &1189 '", file "' // TRIM( file_name ) // '")!' )1190 ENDIF1191 ENDIF1192 1193 IF ( return_value == 0 ) THEN1194 append_internal = .FALSE.1195 1196 attribute%name = TRIM( attribute_name )1197 attribute%data_type = 'int32'1198 attribute%value_int32 = value1199 1200 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &1201 variable_name=TRIM( variable_name_internal ), &1202 attribute=attribute, append=append_internal )1203 ENDIF1204 1205 END FUNCTION dom_def_att_int321160 FUNCTION dom_def_att_int32( file_name, variable_name, attribute_name, value, append ) & 1161 RESULT( return_value ) 1162 1163 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1164 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 1165 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 1166 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 1167 1168 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_int32' !< name of routine 1169 1170 INTEGER(KIND=4), INTENT(IN) :: value !< attribute value 1171 1172 INTEGER :: return_value !< return value 1173 1174 LOGICAL :: append_internal !< same as 'append' 1175 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 1176 1177 TYPE(attribute_type) :: attribute !< new attribute 1178 1179 1180 return_value = 0 1181 1182 IF ( PRESENT( variable_name ) ) THEN 1183 variable_name_internal = TRIM( variable_name ) 1184 ELSE 1185 variable_name_internal = '' 1186 ENDIF 1187 1188 IF ( PRESENT( append ) ) THEN 1189 IF ( append ) THEN 1190 return_value = 1 1191 CALL internal_message( 'error', routine_name // & 1192 ': numeric attribute cannot be appended ' // & 1193 '(attribute "' // TRIM( attribute_name ) // & 1194 '", variable "' // TRIM( variable_name_internal ) // & 1195 '", file "' // TRIM( file_name ) // '")!' ) 1196 ENDIF 1197 ENDIF 1198 1199 IF ( return_value == 0 ) THEN 1200 append_internal = .FALSE. 1201 1202 attribute%name = TRIM( attribute_name ) 1203 attribute%data_type = 'int32' 1204 attribute%value_int32 = value 1205 1206 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 1207 variable_name=TRIM( variable_name_internal ), & 1208 attribute=attribute, append=append_internal ) 1209 ENDIF 1210 1211 END FUNCTION dom_def_att_int32 1206 1212 1207 1213 !--------------------------------------------------------------------------------------------------! … … 1224 1230 !> value=1.0_4 ) 1225 1231 !--------------------------------------------------------------------------------------------------! 1226 FUNCTION dom_def_att_real32( file_name, variable_name, attribute_name, value, append ) &1227 RESULT( return_value )1228 1229 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file1230 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute1231 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable1232 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name1233 1234 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_real32' !< name of routine1235 1236 INTEGER :: return_value !< return value1237 1238 LOGICAL :: append_internal !< same as 'append'1239 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value1240 1241 REAL(KIND=4), INTENT(IN) :: value !< attribute value1242 1243 TYPE(attribute_type) :: attribute !< new attribute1244 1245 1246 return_value = 01247 1248 IF ( PRESENT( variable_name ) ) THEN1249 variable_name_internal = TRIM( variable_name )1250 ELSE1251 variable_name_internal = ''1252 ENDIF1253 1254 IF ( PRESENT( append ) ) THEN1255 IF ( append ) THEN1256 return_value = 11257 CALL internal_message( 'error', routine_name // &1258 ': numeric attribute cannot be appended ' // &1259 '(attribute "' // TRIM( attribute_name ) // &1260 '", variable "' // TRIM( variable_name_internal ) // &1261 '", file "' // TRIM( file_name ) // '")!' )1262 ENDIF1263 ENDIF1264 1265 IF ( return_value == 0 ) THEN1266 append_internal = .FALSE.1267 1268 attribute%name = TRIM( attribute_name )1269 attribute%data_type = 'real32'1270 attribute%value_real32 = value1271 1272 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &1273 variable_name=TRIM( variable_name_internal ), &1274 attribute=attribute, append=append_internal )1275 ENDIF1276 1277 END FUNCTION dom_def_att_real321232 FUNCTION dom_def_att_real32( file_name, variable_name, attribute_name, value, append ) & 1233 RESULT( return_value ) 1234 1235 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1236 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 1237 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 1238 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 1239 1240 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_real32' !< name of routine 1241 1242 INTEGER :: return_value !< return value 1243 1244 LOGICAL :: append_internal !< same as 'append' 1245 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 1246 1247 REAL(KIND=4), INTENT(IN) :: value !< attribute value 1248 1249 TYPE(attribute_type) :: attribute !< new attribute 1250 1251 1252 return_value = 0 1253 1254 IF ( PRESENT( variable_name ) ) THEN 1255 variable_name_internal = TRIM( variable_name ) 1256 ELSE 1257 variable_name_internal = '' 1258 ENDIF 1259 1260 IF ( PRESENT( append ) ) THEN 1261 IF ( append ) THEN 1262 return_value = 1 1263 CALL internal_message( 'error', routine_name // & 1264 ': numeric attribute cannot be appended ' // & 1265 '(attribute "' // TRIM( attribute_name ) // & 1266 '", variable "' // TRIM( variable_name_internal ) // & 1267 '", file "' // TRIM( file_name ) // '")!' ) 1268 ENDIF 1269 ENDIF 1270 1271 IF ( return_value == 0 ) THEN 1272 append_internal = .FALSE. 1273 1274 attribute%name = TRIM( attribute_name ) 1275 attribute%data_type = 'real32' 1276 attribute%value_real32 = value 1277 1278 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 1279 variable_name=TRIM( variable_name_internal ), & 1280 attribute=attribute, append=append_internal ) 1281 ENDIF 1282 1283 END FUNCTION dom_def_att_real32 1278 1284 1279 1285 !--------------------------------------------------------------------------------------------------! … … 1296 1302 !> value=1.0_8 ) 1297 1303 !--------------------------------------------------------------------------------------------------! 1298 FUNCTION dom_def_att_real64( file_name, variable_name, attribute_name, value, append ) &1299 RESULT( return_value )1300 1301 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file1302 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute1303 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable1304 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name1305 1306 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_real64' !< name of routine1307 1308 INTEGER :: return_value !< return value1309 1310 LOGICAL :: append_internal !< same as 'append'1311 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value1312 1313 REAL(KIND=8), INTENT(IN) :: value !< attribute value1314 1315 TYPE(attribute_type) :: attribute !< new attribute1316 1317 1318 return_value = 01319 1320 IF ( PRESENT( variable_name ) ) THEN1321 variable_name_internal = TRIM( variable_name )1322 ELSE1323 variable_name_internal = ''1324 ENDIF1325 1326 IF ( PRESENT( append ) ) THEN1327 IF ( append ) THEN1328 return_value = 11329 CALL internal_message( 'error', routine_name // &1330 ': numeric attribute cannot be appended ' // &1331 '(attribute "' // TRIM( attribute_name ) // &1332 '", variable "' // TRIM( variable_name_internal ) // &1333 '", file "' // TRIM( file_name ) // '")!' )1334 ENDIF1335 ENDIF1336 1337 IF ( return_value == 0 ) THEN1338 append_internal = .FALSE.1339 1340 attribute%name = TRIM( attribute_name )1341 attribute%data_type = 'real64'1342 attribute%value_real64 = value1343 1344 return_value = save_attribute_in_database( file_name=TRIM( file_name ), &1345 variable_name=TRIM( variable_name_internal ), &1346 attribute=attribute, append=append_internal )1347 ENDIF1348 1349 END FUNCTION dom_def_att_real641304 FUNCTION dom_def_att_real64( file_name, variable_name, attribute_name, value, append ) & 1305 RESULT( return_value ) 1306 1307 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1308 CHARACTER(LEN=*), INTENT(IN) :: attribute_name !< name of attribute 1309 CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: variable_name !< name of variable 1310 CHARACTER(LEN=charlen) :: variable_name_internal !< internal copy of variable_name 1311 1312 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_att_real64' !< name of routine 1313 1314 INTEGER :: return_value !< return value 1315 1316 LOGICAL :: append_internal !< same as 'append' 1317 LOGICAL, INTENT(IN), OPTIONAL :: append !< if true, append value to existing value 1318 1319 REAL(KIND=8), INTENT(IN) :: value !< attribute value 1320 1321 TYPE(attribute_type) :: attribute !< new attribute 1322 1323 1324 return_value = 0 1325 1326 IF ( PRESENT( variable_name ) ) THEN 1327 variable_name_internal = TRIM( variable_name ) 1328 ELSE 1329 variable_name_internal = '' 1330 ENDIF 1331 1332 IF ( PRESENT( append ) ) THEN 1333 IF ( append ) THEN 1334 return_value = 1 1335 CALL internal_message( 'error', routine_name // & 1336 ': numeric attribute cannot be appended ' // & 1337 '(attribute "' // TRIM( attribute_name ) // & 1338 '", variable "' // TRIM( variable_name_internal ) // & 1339 '", file "' // TRIM( file_name ) // '")!' ) 1340 ENDIF 1341 ENDIF 1342 1343 IF ( return_value == 0 ) THEN 1344 append_internal = .FALSE. 1345 1346 attribute%name = TRIM( attribute_name ) 1347 attribute%data_type = 'real64' 1348 attribute%value_real64 = value 1349 1350 return_value = save_attribute_in_database( file_name=TRIM( file_name ), & 1351 variable_name=TRIM( variable_name_internal ), & 1352 attribute=attribute, append=append_internal ) 1353 ENDIF 1354 1355 END FUNCTION dom_def_att_real64 1350 1356 1351 1357 !--------------------------------------------------------------------------------------------------! … … 1357 1363 !> to the files. 1358 1364 !--------------------------------------------------------------------------------------------------! 1359 FUNCTION dom_def_end() RESULT( return_value ) 1360 1361 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_end' !< name of routine 1362 1363 INTEGER :: d !< loop index 1364 INTEGER :: f !< loop index 1365 INTEGER :: return_value !< return value 1366 1367 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE, TARGET :: values_int8 !< target array for dimension values 1368 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE, TARGET :: values_int16 !< target array for dimension values 1369 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE, TARGET :: values_int32 !< target array for dimension values 1370 INTEGER(iwp), DIMENSION(:), ALLOCATABLE, TARGET :: values_intwp !< target array for dimension values 1371 1372 INTEGER(KIND=1), DIMENSION(:), POINTER, CONTIGUOUS :: values_int8_pointer !< pointer to target array 1373 INTEGER(KIND=2), DIMENSION(:), POINTER, CONTIGUOUS :: values_int16_pointer !< pointer to target array 1374 INTEGER(KIND=4), DIMENSION(:), POINTER, CONTIGUOUS :: values_int32_pointer !< pointer to target array 1375 INTEGER(iwp), DIMENSION(:), POINTER, CONTIGUOUS :: values_intwp_pointer !< pointer to target array 1376 1377 REAL(KIND=4), DIMENSION(:), ALLOCATABLE, TARGET :: values_real32 !< target array for dimension values 1378 REAL(KIND=8), DIMENSION(:), ALLOCATABLE, TARGET :: values_real64 !< target array for dimension values 1379 REAL(wp), DIMENSION(:), ALLOCATABLE, TARGET :: values_realwp !< target array for dimension values 1380 1381 REAL(KIND=4), DIMENSION(:), POINTER, CONTIGUOUS :: values_real32_pointer !< pointer to target array 1382 REAL(KIND=8), DIMENSION(:), POINTER, CONTIGUOUS :: values_real64_pointer !< pointer to target array 1383 REAL(wp), DIMENSION(:), POINTER, CONTIGUOUS :: values_realwp_pointer !< pointer to target array 1384 1385 1386 return_value = 0 1387 CALL internal_message( 'debug', routine_name // ': start' ) 1388 1389 !-- Clear database from empty files and unused dimensions 1390 IF ( nfiles > 0 ) return_value = cleanup_database() 1391 1392 IF ( return_value == 0 ) THEN 1393 DO f = 1, nfiles 1394 1395 !-- Skip initialization if file is already initialized 1396 IF ( files(f)%is_init ) CYCLE 1397 1398 CALL internal_message( 'debug', routine_name // ': initialize file "' // & 1399 TRIM( files(f)%name ) // '"' ) 1400 1401 !-- Open file 1402 CALL open_output_file( files(f)%format, files(f)%name, files(f)%id, & 1403 return_value=return_value ) 1404 1405 !-- Initialize file header: 1406 !-- define dimensions and variables and write attributes 1407 IF ( return_value == 0 ) & 1408 CALL init_file_header( files(f), return_value=return_value ) 1409 1410 !-- End file definition 1411 IF ( return_value == 0 ) & 1412 CALL stop_file_header_definition( files(f)%format, files(f)%id, & 1413 files(f)%name, return_value ) 1414 1415 IF ( return_value == 0 ) THEN 1416 1417 !-- Flag file as initialized 1418 files(f)%is_init = .TRUE. 1419 1420 !-- Write dimension values into file 1421 DO d = 1, SIZE( files(f)%dimensions ) 1422 IF ( ALLOCATED( files(f)%dimensions(d)%values_int8 ) ) THEN 1423 ALLOCATE( values_int8(files(f)%dimensions(d)%bounds(1): & 1424 files(f)%dimensions(d)%bounds(2)) ) 1425 values_int8 = files(f)%dimensions(d)%values_int8 1426 values_int8_pointer => values_int8 1427 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1428 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1429 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1430 values_int8_1d=values_int8_pointer ) 1431 DEALLOCATE( values_int8 ) 1432 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_int16 ) ) THEN 1433 ALLOCATE( values_int16(files(f)%dimensions(d)%bounds(1): & 1365 FUNCTION dom_def_end() RESULT( return_value ) 1366 1367 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_def_end' !< name of routine 1368 1369 INTEGER :: d !< loop index 1370 INTEGER :: f !< loop index 1371 INTEGER :: return_value !< return value 1372 1373 INTEGER(KIND=1), DIMENSION(:), ALLOCATABLE, TARGET :: values_int8 !< target array for dimension values 1374 INTEGER(KIND=2), DIMENSION(:), ALLOCATABLE, TARGET :: values_int16 !< target array for dimension values 1375 INTEGER(KIND=4), DIMENSION(:), ALLOCATABLE, TARGET :: values_int32 !< target array for dimension values 1376 INTEGER(iwp), DIMENSION(:), ALLOCATABLE, TARGET :: values_intwp !< target array for dimension values 1377 1378 INTEGER(KIND=1), DIMENSION(:), POINTER, CONTIGUOUS :: values_int8_pointer !< pointer to target array 1379 INTEGER(KIND=2), DIMENSION(:), POINTER, CONTIGUOUS :: values_int16_pointer !< pointer to target array 1380 INTEGER(KIND=4), DIMENSION(:), POINTER, CONTIGUOUS :: values_int32_pointer !< pointer to target array 1381 INTEGER(iwp), DIMENSION(:), POINTER, CONTIGUOUS :: values_intwp_pointer !< pointer to target array 1382 1383 REAL(KIND=4), DIMENSION(:), ALLOCATABLE, TARGET :: values_real32 !< target array for dimension values 1384 REAL(KIND=8), DIMENSION(:), ALLOCATABLE, TARGET :: values_real64 !< target array for dimension values 1385 REAL(wp), DIMENSION(:), ALLOCATABLE, TARGET :: values_realwp !< target array for dimension values 1386 1387 REAL(KIND=4), DIMENSION(:), POINTER, CONTIGUOUS :: values_real32_pointer !< pointer to target array 1388 REAL(KIND=8), DIMENSION(:), POINTER, CONTIGUOUS :: values_real64_pointer !< pointer to target array 1389 REAL(wp), DIMENSION(:), POINTER, CONTIGUOUS :: values_realwp_pointer !< pointer to target array 1390 1391 1392 return_value = 0 1393 CALL internal_message( 'debug', routine_name // ': start' ) 1394 ! 1395 !-- Clear database from empty files and unused dimensions 1396 IF ( nfiles > 0 ) return_value = cleanup_database() 1397 1398 IF ( return_value == 0 ) THEN 1399 DO f = 1, nfiles 1400 ! 1401 !-- Skip initialization if file is already initialized 1402 IF ( files(f)%is_init ) CYCLE 1403 1404 CALL internal_message( 'debug', routine_name // ': initialize file "' // & 1405 TRIM( files(f)%name ) // '"' ) 1406 ! 1407 !-- Open file 1408 CALL open_output_file( files(f)%format, files(f)%name, files(f)%id, & 1409 return_value=return_value ) 1410 ! 1411 !-- Initialize file header: 1412 !-- define dimensions and variables and write attributes 1413 IF ( return_value == 0 ) & 1414 CALL init_file_header( files(f), return_value=return_value ) 1415 ! 1416 !-- End file definition 1417 IF ( return_value == 0 ) & 1418 CALL stop_file_header_definition( files(f)%format, files(f)%id, & 1419 files(f)%name, return_value ) 1420 1421 IF ( return_value == 0 ) THEN 1422 ! 1423 !-- Flag file as initialized 1424 files(f)%is_init = .TRUE. 1425 ! 1426 !-- Write dimension values into file 1427 DO d = 1, SIZE( files(f)%dimensions ) 1428 IF ( ALLOCATED( files(f)%dimensions(d)%values_int8 ) ) THEN 1429 ALLOCATE( values_int8(files(f)%dimensions(d)%bounds(1): & 1434 1430 files(f)%dimensions(d)%bounds(2)) ) 1435 values_int16 = files(f)%dimensions(d)%values_int16 1436 values_int16_pointer => values_int16 1437 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1438 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1439 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1440 values_int16_1d=values_int16_pointer ) 1441 DEALLOCATE( values_int16 ) 1442 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_int32 ) ) THEN 1443 ALLOCATE( values_int32(files(f)%dimensions(d)%bounds(1): & 1444 files(f)%dimensions(d)%bounds(2)) ) 1445 values_int32 = files(f)%dimensions(d)%values_int32 1446 values_int32_pointer => values_int32 1447 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1448 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1449 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1450 values_int32_1d=values_int32_pointer ) 1451 DEALLOCATE( values_int32 ) 1452 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_intwp ) ) THEN 1453 ALLOCATE( values_intwp(files(f)%dimensions(d)%bounds(1): & 1454 files(f)%dimensions(d)%bounds(2)) ) 1455 values_intwp = files(f)%dimensions(d)%values_intwp 1456 values_intwp_pointer => values_intwp 1457 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1458 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1459 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1460 values_intwp_1d=values_intwp_pointer ) 1461 DEALLOCATE( values_intwp ) 1462 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_real32 ) ) THEN 1463 ALLOCATE( values_real32(files(f)%dimensions(d)%bounds(1): & 1431 values_int8 = files(f)%dimensions(d)%values_int8 1432 values_int8_pointer => values_int8 1433 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1434 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1435 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1436 values_int8_1d=values_int8_pointer ) 1437 DEALLOCATE( values_int8 ) 1438 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_int16 ) ) THEN 1439 ALLOCATE( values_int16(files(f)%dimensions(d)%bounds(1): & 1464 1440 files(f)%dimensions(d)%bounds(2)) ) 1465 values_real32 = files(f)%dimensions(d)%values_real321466 values_real32_pointer => values_real321467 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, &1468 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), &1469 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), &1470 values_real32_1d=values_real32_pointer )1471 DEALLOCATE( values_real32)1472 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_real64) ) THEN1473 ALLOCATE( values_real64(files(f)%dimensions(d)%bounds(1): &1441 values_int16 = files(f)%dimensions(d)%values_int16 1442 values_int16_pointer => values_int16 1443 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1444 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1445 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1446 values_int16_1d=values_int16_pointer ) 1447 DEALLOCATE( values_int16 ) 1448 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_int32 ) ) THEN 1449 ALLOCATE( values_int32(files(f)%dimensions(d)%bounds(1): & 1474 1450 files(f)%dimensions(d)%bounds(2)) ) 1475 values_real64 = files(f)%dimensions(d)%values_real641476 values_real64_pointer => values_real641477 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, &1478 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), &1479 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), &1480 values_real64_1d=values_real64_pointer )1481 DEALLOCATE( values_real64)1482 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_realwp ) ) THEN1483 ALLOCATE( values_realwp(files(f)%dimensions(d)%bounds(1): &1451 values_int32 = files(f)%dimensions(d)%values_int32 1452 values_int32_pointer => values_int32 1453 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1454 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1455 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1456 values_int32_1d=values_int32_pointer ) 1457 DEALLOCATE( values_int32 ) 1458 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_intwp ) ) THEN 1459 ALLOCATE( values_intwp(files(f)%dimensions(d)%bounds(1): & 1484 1460 files(f)%dimensions(d)%bounds(2)) ) 1485 values_realwp = files(f)%dimensions(d)%values_realwp 1486 values_realwp_pointer => values_realwp 1487 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1488 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1489 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1490 values_realwp_1d=values_realwp_pointer ) 1491 DEALLOCATE( values_realwp ) 1492 ENDIF 1493 IF ( return_value /= 0 ) EXIT 1494 ENDDO 1495 1496 ENDIF 1497 1498 IF ( return_value /= 0 ) EXIT 1499 1500 ENDDO 1501 ENDIF 1502 1503 CALL internal_message( 'debug', routine_name // ': finished' ) 1504 1505 END FUNCTION dom_def_end 1461 values_intwp = files(f)%dimensions(d)%values_intwp 1462 values_intwp_pointer => values_intwp 1463 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1464 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1465 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1466 values_intwp_1d=values_intwp_pointer ) 1467 DEALLOCATE( values_intwp ) 1468 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_real32 ) ) THEN 1469 ALLOCATE( values_real32(files(f)%dimensions(d)%bounds(1): & 1470 files(f)%dimensions(d)%bounds(2)) ) 1471 values_real32 = files(f)%dimensions(d)%values_real32 1472 values_real32_pointer => values_real32 1473 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1474 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1475 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1476 values_real32_1d=values_real32_pointer ) 1477 DEALLOCATE( values_real32 ) 1478 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_real64 ) ) THEN 1479 ALLOCATE( values_real64(files(f)%dimensions(d)%bounds(1): & 1480 files(f)%dimensions(d)%bounds(2)) ) 1481 values_real64 = files(f)%dimensions(d)%values_real64 1482 values_real64_pointer => values_real64 1483 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1484 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1485 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1486 values_real64_1d=values_real64_pointer ) 1487 DEALLOCATE( values_real64 ) 1488 ELSEIF ( ALLOCATED( files(f)%dimensions(d)%values_realwp ) ) THEN 1489 ALLOCATE( values_realwp(files(f)%dimensions(d)%bounds(1): & 1490 files(f)%dimensions(d)%bounds(2)) ) 1491 values_realwp = files(f)%dimensions(d)%values_realwp 1492 values_realwp_pointer => values_realwp 1493 return_value = dom_write_var( files(f)%name, files(f)%dimensions(d)%name, & 1494 bounds_start=(/ files(f)%dimensions(d)%bounds(1) /), & 1495 bounds_end =(/ files(f)%dimensions(d)%bounds(2) /), & 1496 values_realwp_1d=values_realwp_pointer ) 1497 DEALLOCATE( values_realwp ) 1498 ENDIF 1499 IF ( return_value /= 0 ) EXIT 1500 ENDDO 1501 1502 ENDIF 1503 1504 IF ( return_value /= 0 ) EXIT 1505 1506 ENDDO 1507 ENDIF 1508 1509 CALL internal_message( 'debug', routine_name // ': finished' ) 1510 1511 END FUNCTION dom_def_end 1506 1512 1507 1513 !--------------------------------------------------------------------------------------------------! … … 1526 1532 !> chosen, the values are written to file as given in the 'dom_write_var' call. 1527 1533 !--------------------------------------------------------------------------------------------------! 1528 FUNCTION dom_write_var( file_name, variable_name, bounds_start, bounds_end, & 1529 values_int8_0d, values_int8_1d, values_int8_2d, values_int8_3d, & 1530 values_int16_0d, values_int16_1d, values_int16_2d, values_int16_3d, & 1531 values_int32_0d, values_int32_1d, values_int32_2d, values_int32_3d, & 1532 values_intwp_0d, values_intwp_1d, values_intwp_2d, values_intwp_3d, & 1533 values_real32_0d, values_real32_1d, values_real32_2d, values_real32_3d, & 1534 values_real64_0d, values_real64_1d, values_real64_2d, values_real64_3d, & 1535 values_realwp_0d, values_realwp_1d, values_realwp_2d, values_realwp_3d & 1536 ) RESULT( return_value ) 1537 1538 CHARACTER(LEN=charlen) :: file_format !< file format chosen for file 1539 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1540 CHARACTER(LEN=*), INTENT(IN) :: variable_name !< name of variable 1541 1542 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_write_var' !< name of routine 1543 1544 INTEGER :: file_id !< file ID 1545 INTEGER :: i !< loop index 1546 INTEGER :: j !< loop index 1547 INTEGER :: k !< loop index 1548 INTEGER :: output_return_value !< return value of a called output routine 1549 INTEGER :: return_value !< return value 1550 INTEGER :: variable_id !< variable ID 1551 1552 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_end !< end index per dimension of variable 1553 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_start !< start index per dimension of variable 1554 INTEGER, DIMENSION(:), ALLOCATABLE :: bounds_origin !< first index of each dimension 1555 INTEGER, DIMENSION(:), ALLOCATABLE :: bounds_start_internal !< start index per dim. for output after masking 1556 INTEGER, DIMENSION(:), ALLOCATABLE :: value_counts !< count of indices to be written per dimension 1557 INTEGER, DIMENSION(:,:), ALLOCATABLE :: masked_indices !< list containing all output indices along a dimension 1558 1559 LOGICAL :: do_output !< true if any data lies within given range of masked dimension 1560 LOGICAL :: is_global !< true if variable is global 1561 1562 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL :: values_int8_0d !< output variable 1563 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL :: values_int16_0d !< output variable 1564 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_int32_0d !< output variable 1565 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL :: values_intwp_0d !< output variable 1566 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int8_1d !< output variable 1567 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int16_1d !< output variable 1568 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int32_1d !< output variable 1569 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_intwp_1d !< output variable 1570 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int8_2d !< output variable 1571 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int16_2d !< output variable 1572 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int32_2d !< output variable 1573 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_intwp_2d !< output variable 1574 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int8_3d !< output variable 1575 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int16_3d !< output variable 1576 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int32_3d !< output variable 1577 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_intwp_3d !< output variable 1578 1579 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int8_1d_resorted !< resorted output variable 1580 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int16_1d_resorted !< resorted output variable 1581 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int32_1d_resorted !< resorted output variable 1582 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:) :: values_intwp_1d_resorted !< resorted output variable 1583 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int8_2d_resorted !< resorted output variable 1584 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int16_2d_resorted !< resorted output variable 1585 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int32_2d_resorted !< resorted output variable 1586 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_intwp_2d_resorted !< resorted output variable 1587 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int8_3d_resorted !< resorted output variable 1588 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int16_3d_resorted !< resorted output variable 1589 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int32_3d_resorted !< resorted output variable 1590 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_intwp_3d_resorted !< resorted output variable 1591 1592 INTEGER(KIND=1), POINTER :: values_int8_0d_pointer !< pointer to resortet array 1593 INTEGER(KIND=2), POINTER :: values_int16_0d_pointer !< pointer to resortet array 1594 INTEGER(KIND=4), POINTER :: values_int32_0d_pointer !< pointer to resortet array 1595 INTEGER(iwp), POINTER :: values_intwp_0d_pointer !< pointer to resortet array 1596 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int8_1d_pointer !< pointer to resortet array 1597 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int16_1d_pointer !< pointer to resortet array 1598 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int32_1d_pointer !< pointer to resortet array 1599 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:) :: values_intwp_1d_pointer !< pointer to resortet array 1600 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int8_2d_pointer !< pointer to resortet array 1601 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int16_2d_pointer !< pointer to resortet array 1602 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int32_2d_pointer !< pointer to resortet array 1603 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_intwp_2d_pointer !< pointer to resortet array 1604 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int8_3d_pointer !< pointer to resortet array 1605 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int16_3d_pointer !< pointer to resortet array 1606 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int32_3d_pointer !< pointer to resortet array 1607 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_intwp_3d_pointer !< pointer to resortet array 1608 1609 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_real32_0d !< output variable 1610 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL :: values_real64_0d !< output variable 1611 REAL(wp), POINTER, INTENT(IN), OPTIONAL :: values_realwp_0d !< output variable 1612 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real32_1d !< output variable 1613 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real64_1d !< output variable 1614 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_realwp_1d !< output variable 1615 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real32_2d !< output variable 1616 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real64_2d !< output variable 1617 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_realwp_2d !< output variable 1618 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real32_3d !< output variable 1619 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real64_3d !< output variable 1620 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_realwp_3d !< output variable 1621 1622 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:) :: values_real32_1d_resorted !< resorted output variable 1623 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:) :: values_real64_1d_resorted !< resorted output variable 1624 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:) :: values_realwp_1d_resorted !< resorted output variable 1625 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_real32_2d_resorted !< resorted output variable 1626 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_real64_2d_resorted !< resorted output variable 1627 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_realwp_2d_resorted !< resorted output variable 1628 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_real32_3d_resorted !< resorted output variable 1629 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_real64_3d_resorted !< resorted output variable 1630 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_realwp_3d_resorted !< resorted output variable 1631 1632 REAL(KIND=4), POINTER :: values_real32_0d_pointer !< pointer to resortet array 1633 REAL(KIND=8), POINTER :: values_real64_0d_pointer !< pointer to resortet array 1634 REAL(wp), POINTER :: values_realwp_0d_pointer !< pointer to resortet array 1635 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:) :: values_real32_1d_pointer !< pointer to resortet array 1636 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:) :: values_real64_1d_pointer !< pointer to resortet array 1637 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:) :: values_realwp_1d_pointer !< pointer to resortet array 1638 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_real32_2d_pointer !< pointer to resortet array 1639 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_real64_2d_pointer !< pointer to resortet array 1640 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_realwp_2d_pointer !< pointer to resortet array 1641 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_real32_3d_pointer !< pointer to resortet array 1642 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_real64_3d_pointer !< pointer to resortet array 1643 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_realwp_3d_pointer !< pointer to resortet array 1644 1645 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimension_list !< list of used dimensions of variable 1646 1647 1648 return_value = 0 1649 output_return_value = 0 1650 1651 CALL internal_message( 'debug', routine_name // ': write ' // TRIM( variable_name ) // & 1652 ' into file ' // TRIM( file_name ) ) 1653 1654 !-- Search for variable within file 1655 CALL find_var_in_file( file_name, variable_name, file_format, file_id, variable_id, & 1656 is_global, dimension_list, return_value=return_value ) 1657 1658 IF ( return_value == 0 ) THEN 1659 1660 !-- Check if the correct amount of variable bounds were given 1661 IF ( SIZE( bounds_start ) /= SIZE( dimension_list ) .OR. & 1662 SIZE( bounds_end ) /= SIZE( dimension_list ) ) THEN 1663 return_value = 1 1664 CALL internal_message( 'error', routine_name // & 1665 ': number bounds do not match with ' // & 1666 'number of dimensions of variable ' // & 1667 '(variable "' // TRIM( variable_name ) // & 1668 '", file "' // TRIM( file_name ) // '")!' ) 1669 ENDIF 1670 1671 ENDIF 1672 1673 IF ( return_value == 0 ) THEN 1674 1675 !-- Save starting index (lower bounds) of each dimension 1676 ALLOCATE( bounds_origin(SIZE( dimension_list )) ) 1677 ALLOCATE( bounds_start_internal(SIZE( dimension_list )) ) 1678 ALLOCATE( value_counts(SIZE( dimension_list )) ) 1679 1680 WRITE( temp_string, * ) bounds_start 1681 CALL internal_message( 'debug', routine_name // & 1682 ': file "' // TRIM( file_name ) // & 1683 '", variable "' // TRIM( variable_name ) // & 1684 '", bounds_start =' // TRIM( temp_string ) ) 1685 WRITE( temp_string, * ) bounds_end 1686 CALL internal_message( 'debug', routine_name // & 1687 ': file "' // TRIM( file_name ) // & 1688 '", variable "' // TRIM( variable_name ) // & 1689 '", bounds_end =' // TRIM( temp_string ) ) 1690 1691 !-- Get bounds for masking 1692 CALL get_masked_indices_and_masked_dimension_bounds( dimension_list, & 1693 bounds_start, bounds_end, bounds_start_internal, value_counts, bounds_origin, & 1694 masked_indices ) 1695 1696 do_output = .NOT. ANY( value_counts == 0 ) 1697 1698 WRITE( temp_string, * ) bounds_start_internal 1699 CALL internal_message( 'debug', routine_name // & 1700 ': file "' // TRIM( file_name ) // & 1701 '", variable "' // TRIM( variable_name ) // & 1702 '", bounds_start_internal =' // TRIM( temp_string ) ) 1703 WRITE( temp_string, * ) value_counts 1704 CALL internal_message( 'debug', routine_name // & 1705 ': file "' // TRIM( file_name ) // & 1706 '", variable "' // TRIM( variable_name ) // & 1707 '", value_counts =' // TRIM( temp_string ) ) 1708 1709 !-- Mask and resort variable 1710 !-- 8bit integer output 1711 IF ( PRESENT( values_int8_0d ) ) THEN 1712 values_int8_0d_pointer => values_int8_0d 1713 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 1714 IF ( do_output ) THEN 1715 ALLOCATE( values_int8_1d_resorted(0:value_counts(1)-1) ) 1716 !$OMP PARALLEL PRIVATE (i) 1717 !$OMP DO 1718 DO i = 0, value_counts(1) - 1 1719 values_int8_1d_resorted(i) = values_int8_1d(masked_indices(1,i)) 1720 ENDDO 1721 !$OMP END PARALLEL 1722 ELSE 1723 ALLOCATE( values_int8_1d_resorted(1) ) 1724 values_int8_1d_resorted = 0_1 1725 ENDIF 1726 values_int8_1d_pointer => values_int8_1d_resorted 1727 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 1728 IF ( do_output ) THEN 1729 ALLOCATE( values_int8_2d_resorted(0:value_counts(1)-1, & 1730 0:value_counts(2)-1) ) 1731 !$OMP PARALLEL PRIVATE (i,j) 1732 !$OMP DO 1733 DO i = 0, value_counts(1) - 1 1734 DO j = 0, value_counts(2) - 1 1735 values_int8_2d_resorted(i,j) = values_int8_2d(masked_indices(2,j), & 1736 masked_indices(1,i) ) 1737 ENDDO 1738 ENDDO 1739 !$OMP END PARALLEL 1740 ELSE 1741 ALLOCATE( values_int8_2d_resorted(1,1) ) 1742 values_int8_2d_resorted = 0_1 1743 ENDIF 1744 values_int8_2d_pointer => values_int8_2d_resorted 1745 ELSEIF ( PRESENT( values_int8_3d ) ) THEN 1746 IF ( do_output ) THEN 1747 ALLOCATE( values_int8_3d_resorted(0:value_counts(1)-1, & 1748 0:value_counts(2)-1, & 1749 0:value_counts(3)-1) ) 1750 !$OMP PARALLEL PRIVATE (i,j,k) 1751 !$OMP DO 1752 DO i = 0, value_counts(1) - 1 1753 DO j = 0, value_counts(2) - 1 1754 DO k = 0, value_counts(3) - 1 1755 values_int8_3d_resorted(i,j,k) = values_int8_3d(masked_indices(3,k), & 1756 masked_indices(2,j), & 1757 masked_indices(1,i) ) 1758 ENDDO 1759 ENDDO 1760 ENDDO 1761 !$OMP END PARALLEL 1762 ELSE 1763 ALLOCATE( values_int8_3d_resorted(1,1,1) ) 1764 values_int8_3d_resorted = 0_1 1765 ENDIF 1766 values_int8_3d_pointer => values_int8_3d_resorted 1767 1768 !-- 16bit integer output 1769 ELSEIF ( PRESENT( values_int16_0d ) ) THEN 1770 values_int16_0d_pointer => values_int16_0d 1771 ELSEIF ( PRESENT( values_int16_1d ) ) THEN 1772 IF ( do_output ) THEN 1773 ALLOCATE( values_int16_1d_resorted(0:value_counts(1)-1) ) 1774 !$OMP PARALLEL PRIVATE (i) 1775 !$OMP DO 1776 DO i = 0, value_counts(1) - 1 1777 values_int16_1d_resorted(i) = values_int16_1d(masked_indices(1,i)) 1778 ENDDO 1779 !$OMP END PARALLEL 1780 ELSE 1781 ALLOCATE( values_int16_1d_resorted(1) ) 1782 values_int16_1d_resorted = 0_1 1783 ENDIF 1784 values_int16_1d_pointer => values_int16_1d_resorted 1785 ELSEIF ( PRESENT( values_int16_2d ) ) THEN 1786 IF ( do_output ) THEN 1787 ALLOCATE( values_int16_2d_resorted(0:value_counts(1)-1, & 1534 FUNCTION dom_write_var( file_name, variable_name, bounds_start, bounds_end, & 1535 values_int8_0d, values_int8_1d, values_int8_2d, values_int8_3d, & 1536 values_int16_0d, values_int16_1d, values_int16_2d, values_int16_3d, & 1537 values_int32_0d, values_int32_1d, values_int32_2d, values_int32_3d, & 1538 values_intwp_0d, values_intwp_1d, values_intwp_2d, values_intwp_3d, & 1539 values_real32_0d, values_real32_1d, values_real32_2d, values_real32_3d, & 1540 values_real64_0d, values_real64_1d, values_real64_2d, values_real64_3d, & 1541 values_realwp_0d, values_realwp_1d, values_realwp_2d, values_realwp_3d & 1542 ) RESULT( return_value ) 1543 1544 CHARACTER(LEN=charlen) :: file_format !< file format chosen for file 1545 CHARACTER(LEN=*), INTENT(IN) :: file_name !< name of file 1546 CHARACTER(LEN=*), INTENT(IN) :: variable_name !< name of variable 1547 1548 CHARACTER(LEN=*), PARAMETER :: routine_name = 'dom_write_var' !< name of routine 1549 1550 INTEGER :: file_id !< file ID 1551 INTEGER :: i !< loop index 1552 INTEGER :: j !< loop index 1553 INTEGER :: k !< loop index 1554 INTEGER :: output_return_value !< return value of a called output routine 1555 INTEGER :: return_value !< return value 1556 INTEGER :: variable_id !< variable ID 1557 1558 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_end !< end index per dimension of variable 1559 INTEGER, DIMENSION(:), INTENT(IN) :: bounds_start !< start index per dimension of variable 1560 INTEGER, DIMENSION(:), ALLOCATABLE :: bounds_origin !< first index of each dimension 1561 INTEGER, DIMENSION(:), ALLOCATABLE :: bounds_start_internal !< start index per dim. for output after masking 1562 INTEGER, DIMENSION(:), ALLOCATABLE :: value_counts !< count of indices to be written per dimension 1563 INTEGER, DIMENSION(:,:), ALLOCATABLE :: masked_indices !< list containing all output indices along a dimension 1564 1565 LOGICAL :: do_output !< true if any data lies within given range of masked dimension 1566 LOGICAL :: is_global !< true if variable is global 1567 1568 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL :: values_int8_0d !< output variable 1569 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL :: values_int16_0d !< output variable 1570 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_int32_0d !< output variable 1571 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL :: values_intwp_0d !< output variable 1572 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int8_1d !< output variable 1573 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int16_1d !< output variable 1574 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_int32_1d !< output variable 1575 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_intwp_1d !< output variable 1576 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int8_2d !< output variable 1577 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int16_2d !< output variable 1578 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_int32_2d !< output variable 1579 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_intwp_2d !< output variable 1580 INTEGER(KIND=1), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int8_3d !< output variable 1581 INTEGER(KIND=2), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int16_3d !< output variable 1582 INTEGER(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_int32_3d !< output variable 1583 INTEGER(iwp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_intwp_3d !< output variable 1584 1585 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int8_1d_resorted !< resorted output variable 1586 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int16_1d_resorted !< resorted output variable 1587 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:) :: values_int32_1d_resorted !< resorted output variable 1588 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:) :: values_intwp_1d_resorted !< resorted output variable 1589 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int8_2d_resorted !< resorted output variable 1590 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int16_2d_resorted !< resorted output variable 1591 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_int32_2d_resorted !< resorted output variable 1592 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_intwp_2d_resorted !< resorted output variable 1593 INTEGER(KIND=1), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int8_3d_resorted !< resorted output variable 1594 INTEGER(KIND=2), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int16_3d_resorted !< resorted output variable 1595 INTEGER(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_int32_3d_resorted !< resorted output variable 1596 INTEGER(iwp), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_intwp_3d_resorted !< resorted output variable 1597 1598 INTEGER(KIND=1), POINTER :: values_int8_0d_pointer !< pointer to resortet array 1599 INTEGER(KIND=2), POINTER :: values_int16_0d_pointer !< pointer to resortet array 1600 INTEGER(KIND=4), POINTER :: values_int32_0d_pointer !< pointer to resortet array 1601 INTEGER(iwp), POINTER :: values_intwp_0d_pointer !< pointer to resortet array 1602 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int8_1d_pointer !< pointer to resortet array 1603 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int16_1d_pointer !< pointer to resortet array 1604 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:) :: values_int32_1d_pointer !< pointer to resortet array 1605 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:) :: values_intwp_1d_pointer !< pointer to resortet array 1606 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int8_2d_pointer !< pointer to resortet array 1607 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int16_2d_pointer !< pointer to resortet array 1608 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_int32_2d_pointer !< pointer to resortet array 1609 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_intwp_2d_pointer !< pointer to resortet array 1610 INTEGER(KIND=1), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int8_3d_pointer !< pointer to resortet array 1611 INTEGER(KIND=2), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int16_3d_pointer !< pointer to resortet array 1612 INTEGER(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_int32_3d_pointer !< pointer to resortet array 1613 INTEGER(iwp), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_intwp_3d_pointer !< pointer to resortet array 1614 1615 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL :: values_real32_0d !< output variable 1616 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL :: values_real64_0d !< output variable 1617 REAL(wp), POINTER, INTENT(IN), OPTIONAL :: values_realwp_0d !< output variable 1618 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real32_1d !< output variable 1619 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_real64_1d !< output variable 1620 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:) :: values_realwp_1d !< output variable 1621 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real32_2d !< output variable 1622 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_real64_2d !< output variable 1623 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:) :: values_realwp_2d !< output variable 1624 REAL(KIND=4), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real32_3d !< output variable 1625 REAL(KIND=8), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_real64_3d !< output variable 1626 REAL(wp), POINTER, INTENT(IN), OPTIONAL, DIMENSION(:,:,:) :: values_realwp_3d !< output variable 1627 1628 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:) :: values_real32_1d_resorted !< resorted output variable 1629 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:) :: values_real64_1d_resorted !< resorted output variable 1630 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:) :: values_realwp_1d_resorted !< resorted output variable 1631 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_real32_2d_resorted !< resorted output variable 1632 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_real64_2d_resorted !< resorted output variable 1633 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:,:) :: values_realwp_2d_resorted !< resorted output variable 1634 REAL(KIND=4), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_real32_3d_resorted !< resorted output variable 1635 REAL(KIND=8), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_real64_3d_resorted !< resorted output variable 1636 REAL(wp), TARGET, ALLOCATABLE, DIMENSION(:,:,:) :: values_realwp_3d_resorted !< resorted output variable 1637 1638 REAL(KIND=4), POINTER :: values_real32_0d_pointer !< pointer to resortet array 1639 REAL(KIND=8), POINTER :: values_real64_0d_pointer !< pointer to resortet array 1640 REAL(wp), POINTER :: values_realwp_0d_pointer !< pointer to resortet array 1641 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:) :: values_real32_1d_pointer !< pointer to resortet array 1642 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:) :: values_real64_1d_pointer !< pointer to resortet array 1643 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:) :: values_realwp_1d_pointer !< pointer to resortet array 1644 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_real32_2d_pointer !< pointer to resortet array 1645 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_real64_2d_pointer !< pointer to resortet array 1646 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:,:) :: values_realwp_2d_pointer !< pointer to resortet array 1647 REAL(KIND=4), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_real32_3d_pointer !< pointer to resortet array 1648 REAL(KIND=8), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_real64_3d_pointer !< pointer to resortet array 1649 REAL(wp), POINTER, CONTIGUOUS, DIMENSION(:,:,:) :: values_realwp_3d_pointer !< pointer to resortet array 1650 1651 TYPE(dimension_type), DIMENSION(:), ALLOCATABLE :: dimension_list !< list of used dimensions of variable 1652 1653 1654 return_value = 0 1655 output_return_value = 0 1656 1657 CALL internal_message( 'debug', routine_name // ': write ' // TRIM( variable_name ) // & 1658 ' into file ' // TRIM( file_name ) ) 1659 ! 1660 !-- Search for variable within file 1661 CALL find_var_in_file( file_name, variable_name, file_format, file_id, variable_id, & 1662 is_global, dimension_list, return_value=return_value ) 1663 1664 IF ( return_value == 0 ) THEN 1665 ! 1666 !-- Check if the correct amount of variable bounds were given 1667 IF ( SIZE( bounds_start ) /= SIZE( dimension_list ) .OR. & 1668 SIZE( bounds_end ) /= SIZE( dimension_list ) ) THEN 1669 return_value = 1 1670 CALL internal_message( 'error', routine_name // & 1671 ': number bounds do not match with ' // & 1672 'number of dimensions of variable ' // & 1673 '(variable "' // TRIM( variable_name ) // & 1674 '", file "' // TRIM( file_name ) // '")!' ) 1675 ENDIF 1676 1677 ENDIF 1678 1679 IF ( return_value == 0 ) THEN 1680 ! 1681 !-- Save starting index (lower bounds) of each dimension 1682 ALLOCATE( bounds_origin(SIZE( dimension_list )) ) 1683 ALLOCATE( bounds_start_internal(SIZE( dimension_list )) ) 1684 ALLOCATE( value_counts(SIZE( dimension_list )) ) 1685 1686 WRITE( temp_string, * ) bounds_start 1687 CALL internal_message( 'debug', routine_name // & 1688 ': file "' // TRIM( file_name ) // & 1689 '", variable "' // TRIM( variable_name ) // & 1690 '", bounds_start =' // TRIM( temp_string ) ) 1691 WRITE( temp_string, * ) bounds_end 1692 CALL internal_message( 'debug', routine_name // & 1693 ': file "' // TRIM( file_name ) // & 1694 '", variable "' // TRIM( variable_name ) // & 1695 '", bounds_end =' // TRIM( temp_string ) ) 1696 ! 1697 !-- Get bounds for masking 1698 CALL get_masked_indices_and_masked_dimension_bounds( dimension_list, & 1699 bounds_start, bounds_end, bounds_start_internal, value_counts, bounds_origin, & 1700 masked_indices ) 1701 1702 do_output = .NOT. ANY( value_counts == 0 ) 1703 1704 WRITE( temp_string, * ) bounds_start_internal 1705 CALL internal_message( 'debug', routine_name // & 1706 ': file "' // TRIM( file_name ) // & 1707 '", variable "' // TRIM( variable_name ) // & 1708 '", bounds_start_internal =' // TRIM( temp_string ) ) 1709 WRITE( temp_string, * ) value_counts 1710 CALL internal_message( 'debug', routine_name // & 1711 ': file "' // TRIM( file_name ) // & 1712 '", variable "' // TRIM( variable_name ) // & 1713 '", value_counts =' // TRIM( temp_string ) ) 1714 ! 1715 !-- Mask and resort variable 1716 !-- 8bit integer output 1717 IF ( PRESENT( values_int8_0d ) ) THEN 1718 values_int8_0d_pointer => values_int8_0d 1719 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 1720 IF ( do_output ) THEN 1721 ALLOCATE( values_int8_1d_resorted(0:value_counts(1)-1) ) 1722 !$OMP PARALLEL PRIVATE (i) 1723 !$OMP DO 1724 DO i = 0, value_counts(1) - 1 1725 values_int8_1d_resorted(i) = values_int8_1d(masked_indices(1,i)) 1726 ENDDO 1727 !$OMP END PARALLEL 1728 ELSE 1729 ALLOCATE( values_int8_1d_resorted(1) ) 1730 values_int8_1d_resorted = 0_1 1731 ENDIF 1732 values_int8_1d_pointer => values_int8_1d_resorted 1733 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 1734 IF ( do_output ) THEN 1735 ALLOCATE( values_int8_2d_resorted(0:value_counts(1)-1, & 1788 1736 0:value_counts(2)-1) ) 1789 !$OMP PARALLEL PRIVATE (i,j)1790 !$OMP DO1791 DO i = 0, value_counts(1) - 11792 DO j = 0, value_counts(2) - 11793 values_int16_2d_resorted(i,j) = values_int16_2d(masked_indices(2,j), &1794 masked_indices(1,i))1795 ENDDO1796 ENDDO1797 !$OMP END PARALLEL1798 ELSE1799 ALLOCATE( values_int16_2d_resorted(1,1) )1800 values_int16_2d_resorted = 0_11801 ENDIF1802 values_int16_2d_pointer => values_int16_2d_resorted1803 ELSEIF ( PRESENT( values_int16_3d ) ) THEN1804 IF ( do_output ) THEN1805 ALLOCATE( values_int16_3d_resorted(0:value_counts(1)-1, &1737 !$OMP PARALLEL PRIVATE (i,j) 1738 !$OMP DO 1739 DO i = 0, value_counts(1) - 1 1740 DO j = 0, value_counts(2) - 1 1741 values_int8_2d_resorted(i,j) = values_int8_2d(masked_indices(2,j), & 1742 masked_indices(1,i) ) 1743 ENDDO 1744 ENDDO 1745 !$OMP END PARALLEL 1746 ELSE 1747 ALLOCATE( values_int8_2d_resorted(1,1) ) 1748 values_int8_2d_resorted = 0_1 1749 ENDIF 1750 values_int8_2d_pointer => values_int8_2d_resorted 1751 ELSEIF ( PRESENT( values_int8_3d ) ) THEN 1752 IF ( do_output ) THEN 1753 ALLOCATE( values_int8_3d_resorted(0:value_counts(1)-1, & 1806 1754 0:value_counts(2)-1, & 1807 1755 0:value_counts(3)-1) ) 1808 !$OMP PARALLEL PRIVATE (i,j,k) 1809 !$OMP DO 1810 DO i = 0, value_counts(1) - 1 1811 DO j = 0, value_counts(2) - 1 1812 DO k = 0, value_counts(3) - 1 1813 values_int16_3d_resorted(i,j,k) = values_int16_3d(masked_indices(3,k), & 1814 masked_indices(2,j), & 1815 masked_indices(1,i) ) 1816 ENDDO 1817 ENDDO 1818 ENDDO 1819 !$OMP END PARALLEL 1820 ELSE 1821 ALLOCATE( values_int16_3d_resorted(1,1,1) ) 1822 values_int16_3d_resorted = 0_1 1823 ENDIF 1824 values_int16_3d_pointer => values_int16_3d_resorted 1825 1826 !-- 32bit integer output 1827 ELSEIF ( PRESENT( values_int32_0d ) ) THEN 1828 values_int32_0d_pointer => values_int32_0d 1829 ELSEIF ( PRESENT( values_int32_1d ) ) THEN 1830 IF ( do_output ) THEN 1831 ALLOCATE( values_int32_1d_resorted(0:value_counts(1)-1) ) 1832 !$OMP PARALLEL PRIVATE (i) 1833 !$OMP DO 1834 DO i = 0, value_counts(1) - 1 1835 values_int32_1d_resorted(i) = values_int32_1d(masked_indices(1,i)) 1836 ENDDO 1837 !$OMP END PARALLEL 1838 ELSE 1839 ALLOCATE( values_int32_1d_resorted(1) ) 1840 values_int32_1d_resorted = 0_1 1841 ENDIF 1842 values_int32_1d_pointer => values_int32_1d_resorted 1843 ELSEIF ( PRESENT( values_int32_2d ) ) THEN 1844 IF ( do_output ) THEN 1845 ALLOCATE( values_int32_2d_resorted(0:value_counts(1)-1, & 1846 0:value_counts(2)-1) ) 1847 !$OMP PARALLEL PRIVATE (i,j) 1848 !$OMP DO 1849 DO i = 0, value_counts(1) - 1 1850 DO j = 0, value_counts(2) - 1 1851 values_int32_2d_resorted(i,j) = values_int32_2d(masked_indices(2,j), & 1852 masked_indices(1,i) ) 1853 ENDDO 1854 ENDDO 1855 !$OMP END PARALLEL 1856 ELSE 1857 ALLOCATE( values_int32_2d_resorted(1,1) ) 1858 values_int32_2d_resorted = 0_1 1859 ENDIF 1860 values_int32_2d_pointer => values_int32_2d_resorted 1861 ELSEIF ( PRESENT( values_int32_3d ) ) THEN 1862 IF ( do_output ) THEN 1863 ALLOCATE( values_int32_3d_resorted(0:value_counts(1)-1, & 1864 0:value_counts(2)-1, & 1865 0:value_counts(3)-1) ) 1866 !$OMP PARALLEL PRIVATE (i,j,k) 1867 !$OMP DO 1868 DO i = 0, value_counts(1) - 1 1869 DO j = 0, value_counts(2) - 1 1870 DO k = 0, value_counts(3) - 1 1871 values_int32_3d_resorted(i,j,k) = values_int32_3d(masked_indices(3,k), & 1872 masked_indices(2,j), & 1873 masked_indices(1,i) ) 1874 ENDDO 1875 ENDDO 1876 ENDDO 1877 !$OMP END PARALLEL 1878 ELSE 1879 ALLOCATE( values_int32_3d_resorted(1,1,1) ) 1880 values_int32_3d_resorted = 0_1 1881 ENDIF 1882 values_int32_3d_pointer => values_int32_3d_resorted 1883 1884 !-- working-precision integer output 1885 ELSEIF ( PRESENT( values_intwp_0d ) ) THEN 1886 values_intwp_0d_pointer => values_intwp_0d 1887 ELSEIF ( PRESENT( values_intwp_1d ) ) THEN 1888 IF ( do_output ) THEN 1889 ALLOCATE( values_intwp_1d_resorted(0:value_counts(1)-1) ) 1890 !$OMP PARALLEL PRIVATE (i) 1891 !$OMP DO 1892 DO i = 0, value_counts(1) - 1 1893 values_intwp_1d_resorted(i) = values_intwp_1d(masked_indices(1,i)) 1894 ENDDO 1895 !$OMP END PARALLEL 1896 ELSE 1897 ALLOCATE( values_intwp_1d_resorted(1) ) 1898 values_intwp_1d_resorted = 0_1 1899 ENDIF 1900 values_intwp_1d_pointer => values_intwp_1d_resorted 1901 ELSEIF ( PRESENT( values_intwp_2d ) ) THEN 1902 IF ( do_output ) THEN 1903 ALLOCATE( values_intwp_2d_resorted(0:value_counts(1)-1, & 1904 0:value_counts(2)-1) ) 1905 !$OMP PARALLEL PRIVATE (i,j) 1906 !$OMP DO 1907 DO i = 0, value_counts(1) - 1 1908 DO j = 0, value_counts(2) - 1 1909 values_intwp_2d_resorted(i,j) = values_intwp_2d(masked_indices(2,j), & 1910 masked_indices(1,i) ) 1911 ENDDO 1912 ENDDO 1913 !$OMP END PARALLEL 1914 ELSE 1915 ALLOCATE( values_intwp_2d_resorted(1,1) ) 1916 values_intwp_2d_resorted = 0_1 1917 ENDIF 1918 values_intwp_2d_pointer => values_intwp_2d_resorted 1919 ELSEIF ( PRESENT( values_intwp_3d ) ) THEN 1920 IF ( do_output ) THEN 1921 ALLOCATE( values_intwp_3d_resorted(0:value_counts(1)-1, & 1922 0:value_counts(2)-1, & 1923 0:value_counts(3)-1) ) 1924 !$OMP PARALLEL PRIVATE (i,j,k) 1925 !$OMP DO 1926 DO i = 0, value_counts(1) - 1 1927 DO j = 0, value_counts(2) - 1 1928 DO k = 0, value_counts(3) - 1 1929 values_intwp_3d_resorted(i,j,k) = values_intwp_3d(masked_indices(3,k), & 1930 masked_indices(2,j), & 1931 masked_indices(1,i) ) 1932 ENDDO 1933 ENDDO 1934 ENDDO 1935 !$OMP END PARALLEL 1936 ELSE 1937 ALLOCATE( values_intwp_3d_resorted(1,1,1) ) 1938 values_intwp_3d_resorted = 0_1 1939 ENDIF 1940 values_intwp_3d_pointer => values_intwp_3d_resorted 1941 1942 !-- 32bit real output 1943 ELSEIF ( PRESENT( values_real32_0d ) ) THEN 1944 values_real32_0d_pointer => values_real32_0d 1945 ELSEIF ( PRESENT( values_real32_1d ) ) THEN 1946 IF ( do_output ) THEN 1947 ALLOCATE( values_real32_1d_resorted(0:value_counts(1)-1) ) 1948 !$OMP PARALLEL PRIVATE (i) 1949 !$OMP DO 1950 DO i = 0, value_counts(1) - 1 1951 values_real32_1d_resorted(i) = values_real32_1d(masked_indices(1,i)) 1952 ENDDO 1953 !$OMP END PARALLEL 1954 ELSE 1955 ALLOCATE( values_real32_1d_resorted(1) ) 1956 values_real32_1d_resorted = 0_1 1957 ENDIF 1958 values_real32_1d_pointer => values_real32_1d_resorted 1959 ELSEIF ( PRESENT( values_real32_2d ) ) THEN 1960 IF ( do_output ) THEN 1961 ALLOCATE( values_real32_2d_resorted(0:value_counts(1)-1, & 1756 !$OMP PARALLEL PRIVATE (i,j,k) 1757 !$OMP DO 1758 DO i = 0, value_counts(1) - 1 1759 DO j = 0, value_counts(2) - 1 1760 DO k = 0, value_counts(3) - 1 1761 values_int8_3d_resorted(i,j,k) = values_int8_3d(masked_indices(3,k), & 1762 masked_indices(2,j), & 1763 masked_indices(1,i) ) 1764 ENDDO 1765 ENDDO 1766 ENDDO 1767 !$OMP END PARALLEL 1768 ELSE 1769 ALLOCATE( values_int8_3d_resorted(1,1,1) ) 1770 values_int8_3d_resorted = 0_1 1771 ENDIF 1772 values_int8_3d_pointer => values_int8_3d_resorted 1773 ! 1774 !-- 16bit integer output 1775 ELSEIF ( PRESENT( values_int16_0d ) ) THEN 1776 values_int16_0d_pointer => values_int16_0d 1777 ELSEIF ( PRESENT( values_int16_1d ) ) THEN 1778 IF ( do_output ) THEN 1779 ALLOCATE( values_int16_1d_resorted(0:value_counts(1)-1) ) 1780 !$OMP PARALLEL PRIVATE (i) 1781 !$OMP DO 1782 DO i = 0, value_counts(1) - 1 1783 values_int16_1d_resorted(i) = values_int16_1d(masked_indices(1,i)) 1784 ENDDO 1785 !$OMP END PARALLEL 1786 ELSE 1787 ALLOCATE( values_int16_1d_resorted(1) ) 1788 values_int16_1d_resorted = 0_1 1789 ENDIF 1790 values_int16_1d_pointer => values_int16_1d_resorted 1791 ELSEIF ( PRESENT( values_int16_2d ) ) THEN 1792 IF ( do_output ) THEN 1793 ALLOCATE( values_int16_2d_resorted(0:value_counts(1)-1, & 1962 1794 0:value_counts(2)-1) ) 1963 !$OMP PARALLEL PRIVATE (i,j)1964 !$OMP DO1965 DO i = 0, value_counts(1) - 11966 DO j = 0, value_counts(2) - 11967 values_real32_2d_resorted(i,j) = values_real32_2d(masked_indices(2,j), &1968 masked_indices(1,i))1969 ENDDO1970 ENDDO1971 !$OMP END PARALLEL1972 ELSE1973 ALLOCATE( values_real32_2d_resorted(1,1) )1974 values_real32_2d_resorted = 0_11975 ENDIF1976 values_real32_2d_pointer => values_real32_2d_resorted1977 ELSEIF ( PRESENT( values_real32_3d ) ) THEN1978 IF ( do_output ) THEN1979 ALLOCATE( values_real32_3d_resorted(0:value_counts(1)-1, &1795 !$OMP PARALLEL PRIVATE (i,j) 1796 !$OMP DO 1797 DO i = 0, value_counts(1) - 1 1798 DO j = 0, value_counts(2) - 1 1799 values_int16_2d_resorted(i,j) = values_int16_2d(masked_indices(2,j), & 1800 masked_indices(1,i)) 1801 ENDDO 1802 ENDDO 1803 !$OMP END PARALLEL 1804 ELSE 1805 ALLOCATE( values_int16_2d_resorted(1,1) ) 1806 values_int16_2d_resorted = 0_1 1807 ENDIF 1808 values_int16_2d_pointer => values_int16_2d_resorted 1809 ELSEIF ( PRESENT( values_int16_3d ) ) THEN 1810 IF ( do_output ) THEN 1811 ALLOCATE( values_int16_3d_resorted(0:value_counts(1)-1, & 1980 1812 0:value_counts(2)-1, & 1981 1813 0:value_counts(3)-1) ) 1982 !$OMP PARALLEL PRIVATE (i,j,k)1983 !$OMP DO1984 DO i = 0, value_counts(1) - 11985 DO j = 0, value_counts(2) - 11986 DO k = 0, value_counts(3) - 11987 values_real32_3d_resorted(i,j,k) = values_real32_3d(masked_indices(3,k), &1988 1989 1990 ENDDO1991 ENDDO1992 ENDDO1993 !$OMP END PARALLEL1994 ELSE1995 ALLOCATE( values_real32_3d_resorted(1,1,1) )1996 values_real32_3d_resorted = 0_11997 ENDIF1998 values_real32_3d_pointer => values_real32_3d_resorted1999 2000 !-- 64bit realoutput2001 ELSEIF ( PRESENT( values_real64_0d ) ) THEN2002 values_real64_0d_pointer => values_real64_0d2003 ELSEIF ( PRESENT( values_real64_1d ) ) THEN2004 IF ( do_output ) THEN2005 ALLOCATE( values_real64_1d_resorted(0:value_counts(1)-1) )2006 !$OMP PARALLEL PRIVATE (i)2007 !$OMP DO2008 DO i = 0, value_counts(1) - 12009 values_real64_1d_resorted(i) = values_real64_1d(masked_indices(1,i))2010 ENDDO2011 !$OMP END PARALLEL2012 ELSE2013 ALLOCATE( values_real64_1d_resorted(1) )2014 values_real64_1d_resorted = 0_12015 ENDIF2016 values_real64_1d_pointer => values_real64_1d_resorted2017 ELSEIF ( PRESENT( values_real64_2d ) ) THEN2018 IF ( do_output ) THEN2019 ALLOCATE( values_real64_2d_resorted(0:value_counts(1)-1, &1814 !$OMP PARALLEL PRIVATE (i,j,k) 1815 !$OMP DO 1816 DO i = 0, value_counts(1) - 1 1817 DO j = 0, value_counts(2) - 1 1818 DO k = 0, value_counts(3) - 1 1819 values_int16_3d_resorted(i,j,k) = values_int16_3d(masked_indices(3,k), & 1820 masked_indices(2,j), & 1821 masked_indices(1,i) ) 1822 ENDDO 1823 ENDDO 1824 ENDDO 1825 !$OMP END PARALLEL 1826 ELSE 1827 ALLOCATE( values_int16_3d_resorted(1,1,1) ) 1828 values_int16_3d_resorted = 0_1 1829 ENDIF 1830 values_int16_3d_pointer => values_int16_3d_resorted 1831 ! 1832 !-- 32bit integer output 1833 ELSEIF ( PRESENT( values_int32_0d ) ) THEN 1834 values_int32_0d_pointer => values_int32_0d 1835 ELSEIF ( PRESENT( values_int32_1d ) ) THEN 1836 IF ( do_output ) THEN 1837 ALLOCATE( values_int32_1d_resorted(0:value_counts(1)-1) ) 1838 !$OMP PARALLEL PRIVATE (i) 1839 !$OMP DO 1840 DO i = 0, value_counts(1) - 1 1841 values_int32_1d_resorted(i) = values_int32_1d(masked_indices(1,i)) 1842 ENDDO 1843 !$OMP END PARALLEL 1844 ELSE 1845 ALLOCATE( values_int32_1d_resorted(1) ) 1846 values_int32_1d_resorted = 0_1 1847 ENDIF 1848 values_int32_1d_pointer => values_int32_1d_resorted 1849 ELSEIF ( PRESENT( values_int32_2d ) ) THEN 1850 IF ( do_output ) THEN 1851 ALLOCATE( values_int32_2d_resorted(0:value_counts(1)-1, & 2020 1852 0:value_counts(2)-1) ) 2021 !$OMP PARALLEL PRIVATE (i,j)2022 !$OMP DO2023 DO i = 0, value_counts(1) - 12024 DO j = 0, value_counts(2) - 12025 values_real64_2d_resorted(i,j) = values_real64_2d(masked_indices(2,j), &2026 2027 ENDDO2028 ENDDO2029 !$OMP END PARALLEL2030 ELSE2031 ALLOCATE( values_real64_2d_resorted(1,1) )2032 values_real64_2d_resorted = 0_12033 ENDIF2034 values_real64_2d_pointer => values_real64_2d_resorted2035 ELSEIF ( PRESENT( values_real64_3d ) ) THEN2036 IF ( do_output ) THEN2037 ALLOCATE( values_real64_3d_resorted(0:value_counts(1)-1, &1853 !$OMP PARALLEL PRIVATE (i,j) 1854 !$OMP DO 1855 DO i = 0, value_counts(1) - 1 1856 DO j = 0, value_counts(2) - 1 1857 values_int32_2d_resorted(i,j) = values_int32_2d(masked_indices(2,j), & 1858 masked_indices(1,i) ) 1859 ENDDO 1860 ENDDO 1861 !$OMP END PARALLEL 1862 ELSE 1863 ALLOCATE( values_int32_2d_resorted(1,1) ) 1864 values_int32_2d_resorted = 0_1 1865 ENDIF 1866 values_int32_2d_pointer => values_int32_2d_resorted 1867 ELSEIF ( PRESENT( values_int32_3d ) ) THEN 1868 IF ( do_output ) THEN 1869 ALLOCATE( values_int32_3d_resorted(0:value_counts(1)-1, & 2038 1870 0:value_counts(2)-1, & 2039 1871 0:value_counts(3)-1) ) 2040 !$OMP PARALLEL PRIVATE (i,j,k)2041 !$OMP DO2042 DO i = 0, value_counts(1) - 12043 DO j = 0, value_counts(2) - 12044 DO k = 0, value_counts(3) - 12045 values_real64_3d_resorted(i,j,k) = values_real64_3d(masked_indices(3,k), &2046 2047 2048 ENDDO2049 ENDDO2050 ENDDO2051 !$OMP END PARALLEL2052 ELSE2053 ALLOCATE( values_real64_3d_resorted(1,1,1) )2054 values_real64_3d_resorted = 0_12055 ENDIF2056 values_real64_3d_pointer => values_real64_3d_resorted2057 2058 !-- working-precision realoutput2059 ELSEIF ( PRESENT( values_realwp_0d ) ) THEN2060 values_realwp_0d_pointer => values_realwp_0d2061 ELSEIF ( PRESENT( values_realwp_1d ) ) THEN2062 IF ( do_output ) THEN2063 ALLOCATE( values_realwp_1d_resorted(0:value_counts(1)-1) )2064 !$OMP PARALLEL PRIVATE (i)2065 !$OMP DO2066 DO i = 0, value_counts(1) - 12067 values_realwp_1d_resorted(i) = values_realwp_1d(masked_indices(1,i))2068 ENDDO2069 !$OMP END PARALLEL2070 ELSE2071 ALLOCATE( values_realwp_1d_resorted(1) )2072 values_realwp_1d_resorted = 0_12073 ENDIF2074 values_realwp_1d_pointer => values_realwp_1d_resorted2075 ELSEIF ( PRESENT( values_realwp_2d ) ) THEN2076 IF ( do_output ) THEN2077 ALLOCATE( values_realwp_2d_resorted(0:value_counts(1)-1, &1872 !$OMP PARALLEL PRIVATE (i,j,k) 1873 !$OMP DO 1874 DO i = 0, value_counts(1) - 1 1875 DO j = 0, value_counts(2) - 1 1876 DO k = 0, value_counts(3) - 1 1877 values_int32_3d_resorted(i,j,k) = values_int32_3d(masked_indices(3,k), & 1878 masked_indices(2,j), & 1879 masked_indices(1,i) ) 1880 ENDDO 1881 ENDDO 1882 ENDDO 1883 !$OMP END PARALLEL 1884 ELSE 1885 ALLOCATE( values_int32_3d_resorted(1,1,1) ) 1886 values_int32_3d_resorted = 0_1 1887 ENDIF 1888 values_int32_3d_pointer => values_int32_3d_resorted 1889 ! 1890 !-- working-precision integer output 1891 ELSEIF ( PRESENT( values_intwp_0d ) ) THEN 1892 values_intwp_0d_pointer => values_intwp_0d 1893 ELSEIF ( PRESENT( values_intwp_1d ) ) THEN 1894 IF ( do_output ) THEN 1895 ALLOCATE( values_intwp_1d_resorted(0:value_counts(1)-1) ) 1896 !$OMP PARALLEL PRIVATE (i) 1897 !$OMP DO 1898 DO i = 0, value_counts(1) - 1 1899 values_intwp_1d_resorted(i) = values_intwp_1d(masked_indices(1,i)) 1900 ENDDO 1901 !$OMP END PARALLEL 1902 ELSE 1903 ALLOCATE( values_intwp_1d_resorted(1) ) 1904 values_intwp_1d_resorted = 0_1 1905 ENDIF 1906 values_intwp_1d_pointer => values_intwp_1d_resorted 1907 ELSEIF ( PRESENT( values_intwp_2d ) ) THEN 1908 IF ( do_output ) THEN 1909 ALLOCATE( values_intwp_2d_resorted(0:value_counts(1)-1, & 2078 1910 0:value_counts(2)-1) ) 2079 !$OMP PARALLEL PRIVATE (i,j)2080 !$OMP DO2081 DO i = 0, value_counts(1) - 12082 DO j = 0, value_counts(2) - 12083 values_realwp_2d_resorted(i,j) = values_realwp_2d(masked_indices(2,j), &2084 2085 ENDDO2086 ENDDO2087 !$OMP END PARALLEL2088 ELSE2089 ALLOCATE( values_realwp_2d_resorted(1,1) )2090 values_realwp_2d_resorted = 0_12091 ENDIF2092 values_realwp_2d_pointer => values_realwp_2d_resorted2093 ELSEIF ( PRESENT( values_realwp_3d ) ) THEN2094 IF ( do_output ) THEN2095 ALLOCATE( values_realwp_3d_resorted(0:value_counts(1)-1, &1911 !$OMP PARALLEL PRIVATE (i,j) 1912 !$OMP DO 1913 DO i = 0, value_counts(1) - 1 1914 DO j = 0, value_counts(2) - 1 1915 values_intwp_2d_resorted(i,j) = values_intwp_2d(masked_indices(2,j), & 1916 masked_indices(1,i) ) 1917 ENDDO 1918 ENDDO 1919 !$OMP END PARALLEL 1920 ELSE 1921 ALLOCATE( values_intwp_2d_resorted(1,1) ) 1922 values_intwp_2d_resorted = 0_1 1923 ENDIF 1924 values_intwp_2d_pointer => values_intwp_2d_resorted 1925 ELSEIF ( PRESENT( values_intwp_3d ) ) THEN 1926 IF ( do_output ) THEN 1927 ALLOCATE( values_intwp_3d_resorted(0:value_counts(1)-1, & 2096 1928 0:value_counts(2)-1, & 2097 1929 0:value_counts(3)-1) ) 2098 !$OMP PARALLEL PRIVATE (i,j,k) 2099 !$OMP DO 2100 DO i = 0, value_counts(1) - 1 2101 DO j = 0, value_counts(2) - 1 2102 DO k = 0, value_counts(3) - 1 2103 values_realwp_3d_resorted(i,j,k) = values_realwp_3d(masked_indices(3,k), & 2104 masked_indices(2,j), & 2105 masked_indices(1,i) ) 2106 ENDDO 2107 ENDDO 2108 ENDDO 2109 !$OMP END PARALLEL 2110 ELSE 2111 ALLOCATE( values_realwp_3d_resorted(1,1,1) ) 2112 values_realwp_3d_resorted = 0_1 2113 ENDIF 2114 values_realwp_3d_pointer => values_realwp_3d_resorted 2115 2116 ELSE 2117 return_value = 1 2118 CALL internal_message( 'error', routine_name // & 2119 ': no output values given ' // & 2120 '(variable "' // TRIM( variable_name ) // & 2121 '", file "' // TRIM( file_name ) // '")!' ) 2122 ENDIF 2123 2124 DEALLOCATE( masked_indices ) 2125 2126 ENDIF ! Check for error 2127 2128 IF ( return_value == 0 ) THEN 2129 2130 !-- Write variable into file 2131 SELECT CASE ( TRIM( file_format ) ) 2132 2133 CASE ( 'binary' ) 2134 !-- 8bit integer output 2135 IF ( PRESENT( values_int8_0d ) ) THEN 2136 CALL binary_write_variable( file_id, variable_id, & 2137 bounds_start_internal, value_counts, bounds_origin, is_global, & 2138 values_int8_0d=values_int8_0d_pointer, return_value=output_return_value ) 2139 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 2140 CALL binary_write_variable( file_id, variable_id, & 2141 bounds_start_internal, value_counts, bounds_origin, is_global, & 2142 values_int8_1d=values_int8_1d_pointer, return_value=output_return_value ) 2143 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 2144 CALL binary_write_variable( file_id, variable_id, & 2145 bounds_start_internal, value_counts, bounds_origin, is_global, & 2146 values_int8_2d=values_int8_2d_pointer, return_value=output_return_value ) 2147 ELSEIF ( PRESENT( values_int8_3d ) ) THEN 2148 CALL binary_write_variable( file_id, variable_id, & 2149 bounds_start_internal, value_counts, bounds_origin, is_global, & 2150 values_int8_3d=values_int8_3d_pointer, return_value=output_return_value ) 2151 !-- 16bit integer output 2152 ELSEIF ( PRESENT( values_int16_0d ) ) THEN 2153 CALL binary_write_variable( file_id, variable_id, & 2154 bounds_start_internal, value_counts, bounds_origin, is_global, & 2155 values_int16_0d=values_int16_0d_pointer, return_value=output_return_value ) 2156 ELSEIF ( PRESENT( values_int16_1d ) ) THEN 2157 CALL binary_write_variable( file_id, variable_id, & 2158 bounds_start_internal, value_counts, bounds_origin, is_global, & 2159 values_int16_1d=values_int16_1d_pointer, return_value=output_return_value ) 2160 ELSEIF ( PRESENT( values_int16_2d ) ) THEN 2161 CALL binary_write_variable( file_id, variable_id, & 2162 bounds_start_internal, value_counts, bounds_origin, is_global, & 2163 values_int16_2d=values_int16_2d_pointer, return_value=output_return_value ) 2164 ELSEIF ( PRESENT( values_int16_3d ) ) THEN 2165 CALL binary_write_variable( file_id, variable_id, & 2166 bounds_start_internal, value_counts, bounds_origin, is_global, & 2167 values_int16_3d=values_int16_3d_pointer, return_value=output_return_value ) 2168 !-- 32bit integer output 2169 ELSEIF ( PRESENT( values_int32_0d ) ) THEN 2170 CALL binary_write_variable( file_id, variable_id, & 2171 bounds_start_internal, value_counts, bounds_origin, is_global, & 2172 values_int32_0d=values_int32_0d_pointer, return_value=output_return_value ) 2173 ELSEIF ( PRESENT( values_int32_1d ) ) THEN 2174 CALL binary_write_variable( file_id, variable_id, & 2175 bounds_start_internal, value_counts, bounds_origin, is_global, & 2176 values_int32_1d=values_int32_1d_pointer, return_value=output_return_value ) 2177 ELSEIF ( PRESENT( values_int32_2d ) ) THEN 2178 CALL binary_write_variable( file_id, variable_id, & 2179 bounds_start_internal, value_counts, bounds_origin, is_global, & 2180 values_int32_2d=values_int32_2d_pointer, return_value=output_return_value ) 2181 ELSEIF ( PRESENT( values_int32_3d ) ) THEN 2182 CALL binary_write_variable( file_id, variable_id, & 2183 bounds_start_internal, value_counts, bounds_origin, is_global, & 2184 values_int32_3d=values_int32_3d_pointer, return_value=output_return_value ) 2185 !-- working-precision integer output 2186 ELSEIF ( PRESENT( values_intwp_0d ) ) THEN 2187 CALL binary_write_variable( file_id, variable_id, & 2188 bounds_start_internal, value_counts, bounds_origin, is_global, & 2189 values_intwp_0d=values_intwp_0d_pointer, return_value=output_return_value ) 2190 ELSEIF ( PRESENT( values_intwp_1d ) ) THEN 2191 CALL binary_write_variable( file_id, variable_id, & 2192 bounds_start_internal, value_counts, bounds_origin, is_global, & 2193 values_intwp_1d=values_intwp_1d_pointer, return_value=output_return_value ) 2194 ELSEIF ( PRESENT( values_intwp_2d ) ) THEN 2195 CALL binary_write_variable( file_id, variable_id, & 2196 bounds_start_internal, value_counts, bounds_origin, is_global, & 2197 values_intwp_2d=values_intwp_2d_pointer, return_value=output_return_value ) 2198 ELSEIF ( PRESENT( values_intwp_3d ) ) THEN 2199 CALL binary_write_variable( file_id, variable_id, & 2200 bounds_start_internal, value_counts, bounds_origin, is_global, & 2201 values_intwp_3d=values_intwp_3d_pointer, return_value=output_return_value ) 2202 !-- 32bit real output 2203 ELSEIF ( PRESENT( values_real32_0d ) ) THEN 2204 CALL binary_write_variable( file_id, variable_id, & 2205 bounds_start_internal, value_counts, bounds_origin, is_global, & 2206 values_real32_0d=values_real32_0d_pointer, return_value=output_return_value ) 2207 ELSEIF ( PRESENT( values_real32_1d ) ) THEN 2208 CALL binary_write_variable( file_id, variable_id, & 2209 bounds_start_internal, value_counts, bounds_origin, is_global, & 2210 values_real32_1d=values_real32_1d_pointer, return_value=output_return_value ) 2211 ELSEIF ( PRESENT( values_real32_2d ) ) THEN 2212 CALL binary_write_variable( file_id, variable_id, & 2213 bounds_start_internal, value_counts, bounds_origin, is_global, & 2214 values_real32_2d=values_real32_2d_pointer, return_value=output_return_value ) 2215 ELSEIF ( PRESENT( values_real32_3d ) ) THEN 2216 CALL binary_write_variable( file_id, variable_id, & 2217 bounds_start_internal, value_counts, bounds_origin, is_global, & 2218 values_real32_3d=values_real32_3d_pointer, return_value=output_return_value ) 2219 !-- 64bit real output 2220 ELSEIF ( PRESENT( values_real64_0d ) ) THEN 2221 CALL binary_write_variable( file_id, variable_id, & 2222 bounds_start_internal, value_counts, bounds_origin, is_global, & 2223 values_real64_0d=values_real64_0d_pointer, return_value=output_return_value ) 2224 ELSEIF ( PRESENT( values_real64_1d ) ) THEN 2225 CALL binary_write_variable( file_id, variable_id, & 2226 bounds_start_internal, value_counts, bounds_origin, is_global, & 2227 values_real64_1d=values_real64_1d_pointer, return_value=output_return_value ) 2228 ELSEIF ( PRESENT( values_real64_2d ) ) THEN 2229 CALL binary_write_variable( file_id, variable_id, & 2230 bounds_start_internal, value_counts, bounds_origin, is_global, & 2231 values_real64_2d=values_real64_2d_pointer, return_value=output_return_value ) 2232 ELSEIF ( PRESENT( values_real64_3d ) ) THEN 2233 CALL binary_write_variable( file_id, variable_id, & 2234 bounds_start_internal, value_counts, bounds_origin, is_global, & 2235 values_real64_3d=values_real64_3d_pointer, return_value=output_return_value ) 2236 !-- working-precision real output 2237 ELSEIF ( PRESENT( values_realwp_0d ) ) THEN 2238 CALL binary_write_variable( file_id, variable_id, & 2239 bounds_start_internal, value_counts, bounds_origin, is_global, & 2240 values_realwp_0d=values_realwp_0d_pointer, return_value=output_return_value ) 2241 ELSEIF ( PRESENT( values_realwp_1d ) ) THEN 2242 CALL binary_write_variable( file_id, variable_id, & 2243 bounds_start_internal, value_counts, bounds_origin, is_global, & 2244 values_realwp_1d=values_realwp_1d_pointer, return_value=output_return_value ) 2245 ELSEIF ( PRESENT( values_realwp_2d ) ) THEN 2246 CALL binary_write_variable( file_id, variable_id, & 2247 bounds_start_internal, value_counts, bounds_origin, is_global, & 2248 values_realwp_2d=values_realwp_2d_pointer, return_value=output_return_value ) 2249 ELSEIF ( PRESENT( values_realwp_3d ) ) THEN 2250 CALL binary_write_variable( file_id, variable_id, & 2251 bounds_start_internal, value_counts, bounds_origin, is_global, & 2252 values_realwp_3d=values_realwp_3d_pointer, return_value=output_return_value ) 2253 ELSE 2254 return_value = 1 2255 CALL internal_message( 'error', routine_name // & 2256 ': output_type not supported by file format "' // & 2257 TRIM( file_format ) // '" ' // & 2258 '(variable "' // TRIM( variable_name ) // & 2259 '", file "' // TRIM( file_name ) // '")!' ) 2260 ENDIF 2261 2262 CASE ( 'netcdf4-parallel', 'netcdf4-serial' ) 2263 !-- 8bit integer output 2264 IF ( PRESENT( values_int8_0d ) ) THEN 2265 CALL netcdf4_write_variable( file_id, variable_id, & 2266 bounds_start_internal, value_counts, bounds_origin, is_global, & 2267 values_int8_0d=values_int8_0d_pointer, return_value=output_return_value ) 2268 ELSEIF ( PRESENT( values_int8_1d ) ) THEN 2269 CALL netcdf4_write_variable( file_id, variable_id, & 2270 bounds_start_internal, value_counts, bounds_origin, is_global, & 2271 values_int8_1d=values_int8_1d_pointer, return_value=output_return_value ) 2272 ELSEIF ( PRESENT( values_int8_2d ) ) THEN 2273 CALL netcdf4_write_variable( file_id, variable_id, & 2274 bounds_start_internal, value_counts, bounds_origin, is_global, & 2275 values_int8_2d=values_int8_2d_pointer, ret