Line data Source code
1 : !--------------------------------------------------------------------------------------------------!
2 : ! CP2K: A general program to perform molecular dynamics simulations !
3 : ! Copyright 2000-2024 CP2K developers group <https://cp2k.org> !
4 : ! !
5 : ! SPDX-License-Identifier: GPL-2.0-or-later !
6 : !--------------------------------------------------------------------------------------------------!
7 :
8 : ! **************************************************************************************************
9 : !> \brief Interface to the message passing library MPI
10 : !> \par History
11 : !> JGH (02-Jan-2001): New error handling
12 : !> Performance tools
13 : !> JGH (14-Jan-2001): New routines mp_comm_compare, mp_cart_coords,
14 : !> mp_rank_compare, mp_alltoall
15 : !> JGH (06-Feb-2001): New routines mp_comm_free
16 : !> JGH (22-Mar-2001): New routines mp_comm_dup
17 : !> fawzi (04-NOV-2004): storable performance info (for f77 interface)
18 : !> Wrapper routine for mpi_gatherv added (22.12.2005,MK)
19 : !> JGH (13-Feb-2006): Flexible precision
20 : !> JGH (15-Feb-2006): single precision mp_alltoall
21 : !> \author JGH
22 : ! **************************************************************************************************
23 : MODULE message_passing
24 : USE ISO_C_BINDING, ONLY: C_F_POINTER, &
25 : C_PTR
26 : USE kinds, ONLY: &
27 : dp, int_4, int_4_size, int_8, int_8_size, real_4, real_4_size, real_8, &
28 : real_8_size, default_string_length
29 : USE machine, ONLY: m_abort
30 : USE mp_perf_env, ONLY: add_perf, &
31 : add_mp_perf_env, rm_mp_perf_env
32 :
33 : #include "../base/base_uses.f90"
34 :
35 : ! To simplify the transition between the old MPI module and the F08-style module, we introduce these constants to switch between the required handle types
36 : ! Unfortunately, Fortran does not offer something like typedef in C++
37 : #if defined(__parallel) && defined(__MPI_F08)
38 : #define MPI_DATA_TYPE TYPE(MPI_Datatype)
39 : #define MPI_COMM_TYPE TYPE(MPI_Comm)
40 : #define MPI_REQUEST_TYPE TYPE(MPI_Request)
41 : #define MPI_WIN_TYPE TYPE(MPI_Win)
42 : #define MPI_FILE_TYPE TYPE(MPI_File)
43 : #define MPI_INFO_TYPE TYPE(MPI_Info)
44 : #define MPI_STATUS_TYPE TYPE(MPI_Status)
45 : #define MPI_GROUP_TYPE TYPE(MPI_Group)
46 : #define MPI_STATUS_EXTRACT(X) %X
47 : #define MPI_GET_COMP %mpi_val
48 : #else
49 : #define MPI_DATA_TYPE INTEGER
50 : #define MPI_COMM_TYPE INTEGER
51 : #define MPI_REQUEST_TYPE INTEGER
52 : #define MPI_WIN_TYPE INTEGER
53 : #define MPI_FILE_TYPE INTEGER
54 : #define MPI_INFO_TYPE INTEGER
55 : #define MPI_STATUS_TYPE INTEGER, DIMENSION(MPI_STATUS_SIZE)
56 : #define MPI_GROUP_TYPE INTEGER
57 : #define MPI_STATUS_EXTRACT(X) (X)
58 : #define MPI_GET_COMP
59 : #endif
60 :
61 : #if defined(__parallel)
62 : #if defined(__MPI_F08)
63 : USE mpi_f08, ONLY: mpi_allgather, mpi_allgatherv, mpi_alloc_mem, mpi_allreduce, mpi_alltoall, mpi_alltoallv, mpi_bcast, &
64 : mpi_cart_coords, mpi_cart_create, mpi_cart_get, mpi_cart_rank, mpi_cart_sub, mpi_dims_create, mpi_file_close, &
65 : mpi_file_get_size, mpi_file_open, mpi_file_read_at_all, mpi_file_read_at, mpi_file_write_at_all, &
66 : mpi_file_write_at, mpi_free_mem, mpi_gather, mpi_gatherv, mpi_get_address, mpi_group_translate_ranks, mpi_irecv, &
67 : mpi_isend, mpi_recv, mpi_reduce, mpi_reduce_scatter, mpi_rget, mpi_scatter, mpi_send, &
68 : mpi_sendrecv, mpi_sendrecv_replace, mpi_testany, mpi_waitall, mpi_waitany, mpi_win_create, mpi_comm_get_attr, &
69 : mpi_ibcast, mpi_any_tag, mpi_any_source, mpi_address_kind, mpi_thread_serialized, mpi_errors_return, mpi_comm_world, &
70 : #if defined(__DLAF)
71 : mpi_thread_multiple, &
72 : #endif
73 : mpi_comm_self, mpi_comm_null, mpi_info_null, mpi_request_null, mpi_request, mpi_comm, mpi_group, &
74 : mpi_status_ignore, mpi_info, mpi_file, mpi_success, &
75 : mpi_tag_ub, mpi_host, mpi_io, mpi_wtime_is_global, mpi_logical, &
76 : mpi_status, mpi_lor, mpi_2real, mpi_real, mpi_maxloc, mpi_integer8, mpi_bottom, &
77 : mpi_iscatter, mpi_iscatterv, mpi_gatherv, mpi_igatherv, mpi_iallgather, &
78 : mpi_iallgatherv, mpi_status, mpi_comm_type_shared, mpi_integer, mpi_minloc, mpi_2double_precision, &
79 : mpi_file, mpi_minloc, mpi_integer, mpi_sum, mpi_scan, &
80 : mpi_2integer, mpi_in_place, mpi_max, mpi_min, mpi_prod, mpi_iallreduce, mpi_double_precision, &
81 : mpi_error_string, mpi_double_complex, mpi_complex, mpi_type_size, mpi_file_write_all, &
82 : mpi_max_error_string, mpi_datatype, mpi_offset_kind, mpi_win, mpi_mode_rdonly, mpi_mode_rdwr, &
83 : mpi_mode_wronly, mpi_mode_create, mpi_mode_append, mpi_mode_excl, mpi_max_library_version_string, &
84 : mpi_win_null, mpi_file_null, mpi_datatype_null, mpi_character, mpi_mode_nocheck, &
85 : mpi_status_size, mpi_proc_null, mpi_unequal, mpi_similar, mpi_ident, mpi_congruent
86 : #else
87 : USE mpi
88 : #endif
89 : ! subroutines: unfortunately, mpi implementations do not provide interfaces for all subroutines (problems with types and ranks explosion),
90 : ! we do not quite know what is in the module, so we can not include any....
91 : ! to nevertheless get checking for what is included, we use the mpi module without use clause, getting all there is
92 : ! USE mpi, ONLY: mpi_allgather, mpi_allgatherv, mpi_alloc_mem, mpi_allreduce, mpi_alltoall, mpi_alltoallv, mpi_bcast,&
93 : ! mpi_cart_coords, mpi_cart_create, mpi_cart_get, mpi_cart_rank, mpi_cart_sub, mpi_dims_create, mpi_file_close,&
94 : ! mpi_file_get_size, mpi_file_open, mpi_file_read_at_all, mpi_file_read_at, mpi_file_write_at_all,&
95 : ! mpi_file_write_at, mpi_free_mem, mpi_gather, mpi_gatherv, mpi_get_address, mpi_group_translate_ranks, mpi_irecv,&
96 : ! mpi_isend, mpi_recv, mpi_reduce, mpi_reduce_scatter, mpi_rget, mpi_scatter, mpi_send,&
97 : ! mpi_sendrecv, mpi_sendrecv_replace, mpi_testany, mpi_waitall, mpi_waitany, mpi_win_create
98 : ! functions
99 : ! USE mpi, ONLY: mpi_wtime
100 : ! constants
101 : ! USE mpi, ONLY: MPI_DOUBLE_PRECISION, MPI_DOUBLE_COMPLEX, MPI_REAL, MPI_COMPLEX, MPI_ANY_TAG,&
102 : ! MPI_ANY_SOURCE, MPI_COMM_NULL, MPI_REQUEST_NULL, MPI_WIN_NULL, MPI_STATUS_SIZE, MPI_STATUS_IGNORE, MPI_STATUSES_IGNORE, &
103 : ! MPI_ADDRESS_KIND, MPI_OFFSET_KIND, MPI_MODE_CREATE, MPI_MODE_RDONLY, MPI_MODE_WRONLY,&
104 : ! MPI_MODE_RDWR, MPI_MODE_EXCL, MPI_COMM_SELF, MPI_COMM_WORLD, MPI_THREAD_SERIALIZED,&
105 : ! MPI_ERRORS_RETURN, MPI_SUCCESS, MPI_MAX_PROCESSOR_NAME, MPI_MAX_ERROR_STRING, MPI_IDENT,&
106 : ! MPI_UNEQUAL, MPI_MAX, MPI_SUM, MPI_INFO_NULL, MPI_IN_PLACE, MPI_CONGRUENT, MPI_SIMILAR, MPI_MIN, MPI_SOURCE,&
107 : ! MPI_TAG, MPI_INTEGER8, MPI_INTEGER, MPI_MAXLOC, MPI_2INTEGER, MPI_MINLOC, MPI_LOGICAL, MPI_2DOUBLE_PRECISION,&
108 : ! MPI_LOR, MPI_CHARACTER, MPI_BOTTOM, MPI_MODE_NOCHECK, MPI_2REAL
109 : #endif
110 :
111 : IMPLICIT NONE
112 : PRIVATE
113 :
114 : ! parameters that might be needed
115 : #if defined(__parallel)
116 : LOGICAL, PARAMETER :: cp2k_is_parallel = .TRUE.
117 : INTEGER, PARAMETER, PUBLIC :: mp_any_tag = MPI_ANY_TAG
118 : INTEGER, PARAMETER, PUBLIC :: mp_any_source = MPI_ANY_SOURCE
119 : MPI_COMM_TYPE, PARAMETER :: mp_comm_null_handle = MPI_COMM_NULL
120 : MPI_COMM_TYPE, PARAMETER :: mp_comm_self_handle = MPI_COMM_SELF
121 : MPI_COMM_TYPE, PARAMETER :: mp_comm_world_handle = MPI_COMM_WORLD
122 : MPI_REQUEST_TYPE, PARAMETER :: mp_request_null_handle = MPI_REQUEST_NULL
123 : MPI_WIN_TYPE, PARAMETER :: mp_win_null_handle = MPI_WIN_NULL
124 : MPI_FILE_TYPE, PARAMETER :: mp_file_null_handle = MPI_FILE_NULL
125 : MPI_INFO_TYPE, PARAMETER :: mp_info_null_handle = MPI_INFO_NULL
126 : MPI_DATA_TYPE, PARAMETER :: mp_datatype_null_handle = MPI_DATATYPE_NULL
127 : INTEGER, PARAMETER, PUBLIC :: mp_status_size = MPI_STATUS_SIZE
128 : INTEGER, PARAMETER, PUBLIC :: mp_proc_null = MPI_PROC_NULL
129 : ! Set max allocatable memory by MPI to 2 GiByte
130 : INTEGER(KIND=MPI_ADDRESS_KIND), PARAMETER, PRIVATE :: mp_max_memory_size = HUGE(INT(1, KIND=int_4))
131 :
132 : INTEGER, PARAMETER, PUBLIC :: mp_max_library_version_string = MPI_MAX_LIBRARY_VERSION_STRING
133 :
134 : INTEGER, PARAMETER, PUBLIC :: file_offset = MPI_OFFSET_KIND
135 : INTEGER, PARAMETER, PUBLIC :: address_kind = MPI_ADDRESS_KIND
136 : INTEGER, PARAMETER, PUBLIC :: file_amode_create = MPI_MODE_CREATE
137 : INTEGER, PARAMETER, PUBLIC :: file_amode_rdonly = MPI_MODE_RDONLY
138 : INTEGER, PARAMETER, PUBLIC :: file_amode_wronly = MPI_MODE_WRONLY
139 : INTEGER, PARAMETER, PUBLIC :: file_amode_rdwr = MPI_MODE_RDWR
140 : INTEGER, PARAMETER, PUBLIC :: file_amode_excl = MPI_MODE_EXCL
141 : INTEGER, PARAMETER, PUBLIC :: file_amode_append = MPI_MODE_APPEND
142 : #else
143 : LOGICAL, PARAMETER :: cp2k_is_parallel = .FALSE.
144 : INTEGER, PARAMETER, PUBLIC :: mp_any_tag = -1
145 : INTEGER, PARAMETER, PUBLIC :: mp_any_source = -2
146 : MPI_COMM_TYPE, PARAMETER :: mp_comm_null_handle = -3
147 : MPI_COMM_TYPE, PARAMETER :: mp_comm_self_handle = -11
148 : MPI_COMM_TYPE, PARAMETER :: mp_comm_world_handle = -12
149 : MPI_REQUEST_TYPE, PARAMETER :: mp_request_null_handle = -4
150 : MPI_WIN_TYPE, PARAMETER :: mp_win_null_handle = -5
151 : MPI_FILE_TYPE, PARAMETER :: mp_file_null_handle = -6
152 : MPI_INFO_TYPE, PARAMETER :: mp_info_null_handle = -7
153 : MPI_DATA_TYPE, PARAMETER :: mp_datatype_null_handle = -8
154 : INTEGER, PARAMETER, PUBLIC :: mp_status_size = -9
155 : INTEGER, PARAMETER, PUBLIC :: mp_proc_null = -10
156 : INTEGER, PARAMETER, PUBLIC :: mp_max_library_version_string = 1
157 :
158 : INTEGER, PARAMETER, PUBLIC :: file_offset = int_8
159 : INTEGER, PARAMETER, PUBLIC :: address_kind = int_8
160 : INTEGER, PARAMETER, PUBLIC :: file_amode_create = 1
161 : INTEGER, PARAMETER, PUBLIC :: file_amode_rdonly = 2
162 : INTEGER, PARAMETER, PUBLIC :: file_amode_wronly = 4
163 : INTEGER, PARAMETER, PUBLIC :: file_amode_rdwr = 8
164 : INTEGER, PARAMETER, PUBLIC :: file_amode_excl = 64
165 : INTEGER, PARAMETER, PUBLIC :: file_amode_append = 128
166 : #endif
167 :
168 : ! we need to fix this to a given number (crossing fingers)
169 : ! so that the serial code using Fortran stream IO and the MPI have the same sizes.
170 : INTEGER, PARAMETER, PUBLIC :: mpi_character_size = 1
171 : INTEGER, PARAMETER, PUBLIC :: mpi_integer_size = 4
172 :
173 : CHARACTER(LEN=*), PARAMETER, PRIVATE :: moduleN = 'message_passing'
174 :
175 : ! internal reference counter used to debug communicator leaks
176 : INTEGER, PRIVATE, SAVE :: debug_comm_count
177 :
178 : PUBLIC :: mp_comm_type
179 : PUBLIC :: mp_request_type
180 : PUBLIC :: mp_win_type
181 : PUBLIC :: mp_file_type
182 : PUBLIC :: mp_info_type
183 : PUBLIC :: mp_cart_type
184 :
185 : PUBLIC :: mp_para_env_type, mp_para_env_p_type, mp_para_cart_type
186 : PUBLIC :: mp_para_env_create, mp_para_env_release, &
187 : mp_para_cart_create, mp_para_cart_release
188 :
189 : TYPE mp_comm_type
190 : PRIVATE
191 : MPI_COMM_TYPE :: handle = mp_comm_null_handle
192 : ! Number of dimensions within a Cartesian topology (useful with mp_cart_type)
193 : INTEGER :: ndims = 1
194 : ! Meta data to the communicator
195 : INTEGER, PUBLIC :: mepos = -1, source = -1, num_pe = -1
196 : CONTAINS
197 : ! Setters/Getters
198 : PROCEDURE, PASS, NON_OVERRIDABLE :: set_handle => mp_comm_type_set_handle
199 : PROCEDURE, PASS, NON_OVERRIDABLE :: get_handle => mp_comm_type_get_handle
200 : ! Comparisons
201 : PROCEDURE, PRIVATE, PASS, NON_OVERRIDABLE :: mp_comm_op_eq
202 : PROCEDURE, PRIVATE, PASS, NON_OVERRIDABLE :: mp_comm_op_neq
203 : GENERIC, PUBLIC :: operator(.EQ.) => mp_comm_op_eq
204 : GENERIC, PUBLIC :: operator(.NE.) => mp_comm_op_neq
205 : ! Communication routines
206 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: &
207 : mp_sendrecv_i, mp_sendrecv_l, mp_sendrecv_r, mp_sendrecv_d, &
208 : mp_sendrecv_c, mp_sendrecv_z, &
209 : mp_sendrecv_iv, mp_sendrecv_im2, mp_sendrecv_im3, mp_sendrecv_im4, &
210 : mp_sendrecv_lv, mp_sendrecv_lm2, mp_sendrecv_lm3, mp_sendrecv_lm4, &
211 : mp_sendrecv_rv, mp_sendrecv_rm2, mp_sendrecv_rm3, mp_sendrecv_rm4, &
212 : mp_sendrecv_dv, mp_sendrecv_dm2, mp_sendrecv_dm3, mp_sendrecv_dm4, &
213 : mp_sendrecv_cv, mp_sendrecv_cm2, mp_sendrecv_cm3, mp_sendrecv_cm4, &
214 : mp_sendrecv_zv, mp_sendrecv_zm2, mp_sendrecv_zm3, mp_sendrecv_zm4
215 : GENERIC, PUBLIC :: sendrecv => mp_sendrecv_i, mp_sendrecv_l, &
216 : mp_sendrecv_r, mp_sendrecv_d, mp_sendrecv_c, mp_sendrecv_z, &
217 : mp_sendrecv_iv, mp_sendrecv_im2, mp_sendrecv_im3, mp_sendrecv_im4, &
218 : mp_sendrecv_lv, mp_sendrecv_lm2, mp_sendrecv_lm3, mp_sendrecv_lm4, &
219 : mp_sendrecv_rv, mp_sendrecv_rm2, mp_sendrecv_rm3, mp_sendrecv_rm4, &
220 : mp_sendrecv_dv, mp_sendrecv_dm2, mp_sendrecv_dm3, mp_sendrecv_dm4, &
221 : mp_sendrecv_cv, mp_sendrecv_cm2, mp_sendrecv_cm3, mp_sendrecv_cm4, &
222 : mp_sendrecv_zv, mp_sendrecv_zm2, mp_sendrecv_zm3, mp_sendrecv_zm4
223 :
224 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_minloc_iv, &
225 : mp_minloc_lv, mp_minloc_rv, mp_minloc_dv
226 : GENERIC, PUBLIC :: minloc => mp_minloc_iv, &
227 : mp_minloc_lv, mp_minloc_rv, mp_minloc_dv
228 :
229 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_maxloc_iv, &
230 : mp_maxloc_lv, mp_maxloc_rv, mp_maxloc_dv
231 : GENERIC, PUBLIC :: maxloc => mp_maxloc_iv, &
232 : mp_maxloc_lv, mp_maxloc_rv, mp_maxloc_dv
233 :
234 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_shift_im, mp_shift_i, &
235 : mp_shift_lm, mp_shift_l, mp_shift_rm, mp_shift_r, &
236 : mp_shift_dm, mp_shift_d, mp_shift_cm, mp_shift_c, &
237 : mp_shift_zm, mp_shift_z
238 : GENERIC, PUBLIC :: shift => mp_shift_im, mp_shift_i, &
239 : mp_shift_lm, mp_shift_l, mp_shift_rm, mp_shift_r, &
240 : mp_shift_dm, mp_shift_d, mp_shift_cm, mp_shift_c, &
241 : mp_shift_zm, mp_shift_z
242 :
243 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_bcast_i, mp_bcast_iv, mp_bcast_im, mp_bcast_i3, &
244 : mp_bcast_l, mp_bcast_lv, mp_bcast_lm, mp_bcast_l3, &
245 : mp_bcast_r, mp_bcast_rv, mp_bcast_rm, mp_bcast_r3, &
246 : mp_bcast_d, mp_bcast_dv, mp_bcast_dm, mp_bcast_d3, &
247 : mp_bcast_c, mp_bcast_cv, mp_bcast_cm, mp_bcast_c3, &
248 : mp_bcast_z, mp_bcast_zv, mp_bcast_zm, mp_bcast_z3, &
249 : mp_bcast_b, mp_bcast_bv, mp_bcast_av, mp_bcast_am, &
250 : mp_bcast_i_src, mp_bcast_iv_src, mp_bcast_im_src, mp_bcast_i3_src, &
251 : mp_bcast_l_src, mp_bcast_lv_src, mp_bcast_lm_src, mp_bcast_l3_src, &
252 : mp_bcast_r_src, mp_bcast_rv_src, mp_bcast_rm_src, mp_bcast_r3_src, &
253 : mp_bcast_d_src, mp_bcast_dv_src, mp_bcast_dm_src, mp_bcast_d3_src, &
254 : mp_bcast_c_src, mp_bcast_cv_src, mp_bcast_cm_src, mp_bcast_c3_src, &
255 : mp_bcast_z_src, mp_bcast_zv_src, mp_bcast_zm_src, mp_bcast_z3_src, &
256 : mp_bcast_b_src, mp_bcast_bv_src, mp_bcast_av_src, mp_bcast_am_src
257 : GENERIC, PUBLIC :: bcast => mp_bcast_i, mp_bcast_iv, mp_bcast_im, mp_bcast_i3, &
258 : mp_bcast_l, mp_bcast_lv, mp_bcast_lm, mp_bcast_l3, &
259 : mp_bcast_r, mp_bcast_rv, mp_bcast_rm, mp_bcast_r3, &
260 : mp_bcast_d, mp_bcast_dv, mp_bcast_dm, mp_bcast_d3, &
261 : mp_bcast_c, mp_bcast_cv, mp_bcast_cm, mp_bcast_c3, &
262 : mp_bcast_z, mp_bcast_zv, mp_bcast_zm, mp_bcast_z3, &
263 : mp_bcast_b, mp_bcast_bv, mp_bcast_av, mp_bcast_am, &
264 : mp_bcast_i_src, mp_bcast_iv_src, mp_bcast_im_src, mp_bcast_i3_src, &
265 : mp_bcast_l_src, mp_bcast_lv_src, mp_bcast_lm_src, mp_bcast_l3_src, &
266 : mp_bcast_r_src, mp_bcast_rv_src, mp_bcast_rm_src, mp_bcast_r3_src, &
267 : mp_bcast_d_src, mp_bcast_dv_src, mp_bcast_dm_src, mp_bcast_d3_src, &
268 : mp_bcast_c_src, mp_bcast_cv_src, mp_bcast_cm_src, mp_bcast_c3_src, &
269 : mp_bcast_z_src, mp_bcast_zv_src, mp_bcast_zm_src, mp_bcast_z3_src, &
270 : mp_bcast_b_src, mp_bcast_bv_src, mp_bcast_av_src, mp_bcast_am_src
271 :
272 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_ibcast_i, mp_ibcast_iv, &
273 : mp_ibcast_l, mp_ibcast_lv, mp_ibcast_r, mp_ibcast_rv, &
274 : mp_ibcast_d, mp_ibcast_dv, mp_ibcast_c, mp_ibcast_cv, &
275 : mp_ibcast_z, mp_ibcast_zv
276 : GENERIC, PUBLIC :: ibcast => mp_ibcast_i, mp_ibcast_iv, &
277 : mp_ibcast_l, mp_ibcast_lv, mp_ibcast_r, mp_ibcast_rv, &
278 : mp_ibcast_d, mp_ibcast_dv, mp_ibcast_c, mp_ibcast_cv, &
279 : mp_ibcast_z, mp_ibcast_zv
280 :
281 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: &
282 : mp_sum_i, mp_sum_iv, mp_sum_im, mp_sum_im3, mp_sum_im4, &
283 : mp_sum_l, mp_sum_lv, mp_sum_lm, mp_sum_lm3, mp_sum_lm4, &
284 : mp_sum_r, mp_sum_rv, mp_sum_rm, mp_sum_rm3, mp_sum_rm4, &
285 : mp_sum_d, mp_sum_dv, mp_sum_dm, mp_sum_dm3, mp_sum_dm4, &
286 : mp_sum_c, mp_sum_cv, mp_sum_cm, mp_sum_cm3, mp_sum_cm4, &
287 : mp_sum_z, mp_sum_zv, mp_sum_zm, mp_sum_zm3, mp_sum_zm4, &
288 : mp_sum_root_iv, mp_sum_root_im, mp_sum_root_lv, mp_sum_root_lm, &
289 : mp_sum_root_rv, mp_sum_root_rm, mp_sum_root_dv, mp_sum_root_dm, &
290 : mp_sum_root_cv, mp_sum_root_cm, mp_sum_root_zv, mp_sum_root_zm, &
291 : mp_sum_b, mp_sum_bv
292 : GENERIC, PUBLIC :: sum => mp_sum_i, mp_sum_iv, mp_sum_im, mp_sum_im3, mp_sum_im4, &
293 : mp_sum_l, mp_sum_lv, mp_sum_lm, mp_sum_lm3, mp_sum_lm4, &
294 : mp_sum_r, mp_sum_rv, mp_sum_rm, mp_sum_rm3, mp_sum_rm4, &
295 : mp_sum_d, mp_sum_dv, mp_sum_dm, mp_sum_dm3, mp_sum_dm4, &
296 : mp_sum_c, mp_sum_cv, mp_sum_cm, mp_sum_cm3, mp_sum_cm4, &
297 : mp_sum_z, mp_sum_zv, mp_sum_zm, mp_sum_zm3, mp_sum_zm4, &
298 : mp_sum_root_iv, mp_sum_root_im, mp_sum_root_lv, mp_sum_root_lm, &
299 : mp_sum_root_rv, mp_sum_root_rm, mp_sum_root_dv, mp_sum_root_dm, &
300 : mp_sum_root_cv, mp_sum_root_cm, mp_sum_root_zv, mp_sum_root_zm, &
301 : mp_sum_b, mp_sum_bv
302 :
303 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_isum_iv, &
304 : mp_isum_lv, mp_isum_rv, mp_isum_dv, mp_isum_cv, &
305 : mp_isum_zv, mp_isum_bv
306 : GENERIC, PUBLIC :: isum => mp_isum_iv, &
307 : mp_isum_lv, mp_isum_rv, mp_isum_dv, mp_isum_cv, &
308 : mp_isum_zv, mp_isum_bv
309 :
310 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_sum_partial_im, &
311 : mp_sum_partial_lm, mp_sum_partial_rm, mp_sum_partial_dm, &
312 : mp_sum_partial_cm, mp_sum_partial_zm
313 : GENERIC, PUBLIC :: sum_partial => mp_sum_partial_im, &
314 : mp_sum_partial_lm, mp_sum_partial_rm, mp_sum_partial_dm, &
315 : mp_sum_partial_cm, mp_sum_partial_zm
316 :
317 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_max_i, mp_max_iv, &
318 : mp_max_l, mp_max_lv, mp_max_r, mp_max_rv, &
319 : mp_max_d, mp_max_dv, mp_max_c, mp_max_cv, &
320 : mp_max_z, mp_max_zv, mp_max_root_i, mp_max_root_l, &
321 : mp_max_root_r, mp_max_root_d, mp_max_root_c, mp_max_root_z, &
322 : mp_max_root_im, mp_max_root_lm, mp_max_root_rm, mp_max_root_dm, &
323 : mp_max_root_cm, mp_max_root_zm
324 : GENERIC, PUBLIC :: max => mp_max_i, mp_max_iv, &
325 : mp_max_l, mp_max_lv, mp_max_r, mp_max_rv, &
326 : mp_max_d, mp_max_dv, mp_max_c, mp_max_cv, &
327 : mp_max_z, mp_max_zv, mp_max_root_i, mp_max_root_l, &
328 : mp_max_root_r, mp_max_root_d, mp_max_root_c, mp_max_root_z, &
329 : mp_max_root_im, mp_max_root_lm, mp_max_root_rm, mp_max_root_dm, &
330 : mp_max_root_cm, mp_max_root_zm
331 :
332 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_min_i, mp_min_iv, &
333 : mp_min_l, mp_min_lv, mp_min_r, mp_min_rv, &
334 : mp_min_d, mp_min_dv, mp_min_c, mp_min_cv, &
335 : mp_min_z, mp_min_zv
336 : GENERIC, PUBLIC :: min => mp_min_i, mp_min_iv, &
337 : mp_min_l, mp_min_lv, mp_min_r, mp_min_rv, &
338 : mp_min_d, mp_min_dv, mp_min_c, mp_min_cv, &
339 : mp_min_z, mp_min_zv
340 :
341 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: &
342 : mp_sum_scatter_iv, mp_sum_scatter_lv, mp_sum_scatter_rv, &
343 : mp_sum_scatter_dv, mp_sum_scatter_cv, mp_sum_scatter_zv
344 : GENERIC, PUBLIC :: sum_scatter => &
345 : mp_sum_scatter_iv, mp_sum_scatter_lv, mp_sum_scatter_rv, &
346 : mp_sum_scatter_dv, mp_sum_scatter_cv, mp_sum_scatter_zv
347 :
348 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_prod_r, mp_prod_d, mp_prod_c, mp_prod_z
349 : GENERIC, PUBLIC :: prod => mp_prod_r, mp_prod_d, mp_prod_c, mp_prod_z
350 :
351 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_gather_i, mp_gather_iv, mp_gather_im, &
352 : mp_gather_l, mp_gather_lv, mp_gather_lm, &
353 : mp_gather_r, mp_gather_rv, mp_gather_rm, &
354 : mp_gather_d, mp_gather_dv, mp_gather_dm, &
355 : mp_gather_c, mp_gather_cv, mp_gather_cm, &
356 : mp_gather_z, mp_gather_zv, mp_gather_zm, &
357 : mp_gather_i_src, mp_gather_iv_src, mp_gather_im_src, &
358 : mp_gather_l_src, mp_gather_lv_src, mp_gather_lm_src, &
359 : mp_gather_r_src, mp_gather_rv_src, mp_gather_rm_src, &
360 : mp_gather_d_src, mp_gather_dv_src, mp_gather_dm_src, &
361 : mp_gather_c_src, mp_gather_cv_src, mp_gather_cm_src, &
362 : mp_gather_z_src, mp_gather_zv_src, mp_gather_zm_src
363 : GENERIC, PUBLIC :: gather => mp_gather_i, mp_gather_iv, mp_gather_im, &
364 : mp_gather_l, mp_gather_lv, mp_gather_lm, &
365 : mp_gather_r, mp_gather_rv, mp_gather_rm, &
366 : mp_gather_d, mp_gather_dv, mp_gather_dm, &
367 : mp_gather_c, mp_gather_cv, mp_gather_cm, &
368 : mp_gather_z, mp_gather_zv, mp_gather_zm, &
369 : mp_gather_i_src, mp_gather_iv_src, mp_gather_im_src, &
370 : mp_gather_l_src, mp_gather_lv_src, mp_gather_lm_src, &
371 : mp_gather_r_src, mp_gather_rv_src, mp_gather_rm_src, &
372 : mp_gather_d_src, mp_gather_dv_src, mp_gather_dm_src, &
373 : mp_gather_c_src, mp_gather_cv_src, mp_gather_cm_src, &
374 : mp_gather_z_src, mp_gather_zv_src, mp_gather_zm_src
375 :
376 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_gatherv_iv, &
377 : mp_gatherv_lv, mp_gatherv_rv, mp_gatherv_dv, &
378 : mp_gatherv_cv, mp_gatherv_zv, mp_gatherv_lm2, mp_gatherv_rm2, &
379 : mp_gatherv_dm2, mp_gatherv_cm2, mp_gatherv_zm2, mp_gatherv_iv_src, &
380 : mp_gatherv_lv_src, mp_gatherv_rv_src, mp_gatherv_dv_src, &
381 : mp_gatherv_cv_src, mp_gatherv_zv_src, mp_gatherv_lm2_src, mp_gatherv_rm2_src, &
382 : mp_gatherv_dm2_src, mp_gatherv_cm2_src, mp_gatherv_zm2_src
383 : GENERIC, PUBLIC :: gatherv => mp_gatherv_iv, &
384 : mp_gatherv_lv, mp_gatherv_rv, mp_gatherv_dv, &
385 : mp_gatherv_cv, mp_gatherv_zv, mp_gatherv_lm2, mp_gatherv_rm2, &
386 : mp_gatherv_dm2, mp_gatherv_cm2, mp_gatherv_zm2, mp_gatherv_iv_src, &
387 : mp_gatherv_lv_src, mp_gatherv_rv_src, mp_gatherv_dv_src, &
388 : mp_gatherv_cv_src, mp_gatherv_zv_src, mp_gatherv_lm2_src, mp_gatherv_rm2_src, &
389 : mp_gatherv_dm2_src, mp_gatherv_cm2_src, mp_gatherv_zm2_src
390 :
391 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_igatherv_iv, &
392 : mp_igatherv_lv, mp_igatherv_rv, mp_igatherv_dv, &
393 : mp_igatherv_cv, mp_igatherv_zv
394 : GENERIC, PUBLIC :: igatherv => mp_igatherv_iv, &
395 : mp_igatherv_lv, mp_igatherv_rv, mp_igatherv_dv, &
396 : mp_igatherv_cv, mp_igatherv_zv
397 :
398 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_allgather_i, mp_allgather_i2, &
399 : mp_allgather_i12, mp_allgather_i23, mp_allgather_i34, &
400 : mp_allgather_i22, mp_allgather_l, mp_allgather_l2, &
401 : mp_allgather_l12, mp_allgather_l23, mp_allgather_l34, &
402 : mp_allgather_l22, mp_allgather_r, mp_allgather_r2, &
403 : mp_allgather_r12, mp_allgather_r23, mp_allgather_r34, &
404 : mp_allgather_r22, mp_allgather_d, mp_allgather_d2, &
405 : mp_allgather_d12, mp_allgather_d23, mp_allgather_d34, &
406 : mp_allgather_d22, mp_allgather_c, mp_allgather_c2, &
407 : mp_allgather_c12, mp_allgather_c23, mp_allgather_c34, &
408 : mp_allgather_c22, mp_allgather_z, mp_allgather_z2, &
409 : mp_allgather_z12, mp_allgather_z23, mp_allgather_z34, &
410 : mp_allgather_z22
411 : GENERIC, PUBLIC :: allgather => mp_allgather_i, mp_allgather_i2, &
412 : mp_allgather_i12, mp_allgather_i23, mp_allgather_i34, &
413 : mp_allgather_i22, mp_allgather_l, mp_allgather_l2, &
414 : mp_allgather_l12, mp_allgather_l23, mp_allgather_l34, &
415 : mp_allgather_l22, mp_allgather_r, mp_allgather_r2, &
416 : mp_allgather_r12, mp_allgather_r23, mp_allgather_r34, &
417 : mp_allgather_r22, mp_allgather_d, mp_allgather_d2, &
418 : mp_allgather_d12, mp_allgather_d23, mp_allgather_d34, &
419 : mp_allgather_d22, mp_allgather_c, mp_allgather_c2, &
420 : mp_allgather_c12, mp_allgather_c23, mp_allgather_c34, &
421 : mp_allgather_c22, mp_allgather_z, mp_allgather_z2, &
422 : mp_allgather_z12, mp_allgather_z23, mp_allgather_z34, &
423 : mp_allgather_z22
424 :
425 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_allgatherv_iv, mp_allgatherv_lv, &
426 : mp_allgatherv_rv, mp_allgatherv_dv, mp_allgatherv_cv, mp_allgatherv_zv, &
427 : mp_allgatherv_im2, mp_allgatherv_lm2, mp_allgatherv_rm2, &
428 : mp_allgatherv_dm2, mp_allgatherv_cm2, mp_allgatherv_zm2
429 : GENERIC, PUBLIC :: allgatherv => mp_allgatherv_iv, mp_allgatherv_lv, &
430 : mp_allgatherv_rv, mp_allgatherv_dv, mp_allgatherv_cv, mp_allgatherv_zv, &
431 : mp_allgatherv_im2, mp_allgatherv_lm2, mp_allgatherv_rm2, &
432 : mp_allgatherv_dm2, mp_allgatherv_cm2, mp_allgatherv_zm2
433 :
434 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_iallgather_i, mp_iallgather_l, &
435 : mp_iallgather_r, mp_iallgather_d, mp_iallgather_c, mp_iallgather_z, &
436 : mp_iallgather_i11, mp_iallgather_l11, mp_iallgather_r11, mp_iallgather_d11, &
437 : mp_iallgather_c11, mp_iallgather_z11, mp_iallgather_i13, mp_iallgather_l13, &
438 : mp_iallgather_r13, mp_iallgather_d13, mp_iallgather_c13, mp_iallgather_z13, &
439 : mp_iallgather_i22, mp_iallgather_l22, mp_iallgather_r22, mp_iallgather_d22, &
440 : mp_iallgather_c22, mp_iallgather_z22, mp_iallgather_i24, mp_iallgather_l24, &
441 : mp_iallgather_r24, mp_iallgather_d24, mp_iallgather_c24, mp_iallgather_z24, &
442 : mp_iallgather_i33, mp_iallgather_l33, mp_iallgather_r33, mp_iallgather_d33, &
443 : mp_iallgather_c33, mp_iallgather_z33
444 : GENERIC, PUBLIC :: iallgather => mp_iallgather_i, mp_iallgather_l, &
445 : mp_iallgather_r, mp_iallgather_d, mp_iallgather_c, mp_iallgather_z, &
446 : mp_iallgather_i11, mp_iallgather_l11, mp_iallgather_r11, mp_iallgather_d11, &
447 : mp_iallgather_c11, mp_iallgather_z11, mp_iallgather_i13, mp_iallgather_l13, &
448 : mp_iallgather_r13, mp_iallgather_d13, mp_iallgather_c13, mp_iallgather_z13, &
449 : mp_iallgather_i22, mp_iallgather_l22, mp_iallgather_r22, mp_iallgather_d22, &
450 : mp_iallgather_c22, mp_iallgather_z22, mp_iallgather_i24, mp_iallgather_l24, &
451 : mp_iallgather_r24, mp_iallgather_d24, mp_iallgather_c24, mp_iallgather_z24, &
452 : mp_iallgather_i33, mp_iallgather_l33, mp_iallgather_r33, mp_iallgather_d33, &
453 : mp_iallgather_c33, mp_iallgather_z33
454 :
455 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_iallgatherv_iv, mp_iallgatherv_iv2, &
456 : mp_iallgatherv_lv, mp_iallgatherv_lv2, mp_iallgatherv_rv, mp_iallgatherv_rv2, &
457 : mp_iallgatherv_dv, mp_iallgatherv_dv2, mp_iallgatherv_cv, mp_iallgatherv_cv2, &
458 : mp_iallgatherv_zv, mp_iallgatherv_zv2
459 : GENERIC, PUBLIC :: iallgatherv => mp_iallgatherv_iv, mp_iallgatherv_iv2, &
460 : mp_iallgatherv_lv, mp_iallgatherv_lv2, mp_iallgatherv_rv, mp_iallgatherv_rv2, &
461 : mp_iallgatherv_dv, mp_iallgatherv_dv2, mp_iallgatherv_cv, mp_iallgatherv_cv2, &
462 : mp_iallgatherv_zv, mp_iallgatherv_zv2
463 :
464 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_scatter_iv, mp_scatter_lv, &
465 : mp_scatter_rv, mp_scatter_dv, mp_scatter_cv, mp_scatter_zv
466 : GENERIC, PUBLIC :: scatter => mp_scatter_iv, mp_scatter_lv, &
467 : mp_scatter_rv, mp_scatter_dv, mp_scatter_cv, mp_scatter_zv
468 :
469 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_iscatter_i, mp_iscatter_l, &
470 : mp_iscatter_r, mp_iscatter_d, mp_iscatter_c, mp_iscatter_z, &
471 : mp_iscatter_iv2, mp_iscatter_lv2, mp_iscatter_rv2, mp_iscatter_dv2, &
472 : mp_iscatter_cv2, mp_iscatter_zv2
473 : GENERIC, PUBLIC :: iscatter => mp_iscatter_i, mp_iscatter_l, &
474 : mp_iscatter_r, mp_iscatter_d, mp_iscatter_c, mp_iscatter_z, &
475 : mp_iscatter_iv2, mp_iscatter_lv2, mp_iscatter_rv2, mp_iscatter_dv2, &
476 : mp_iscatter_cv2, mp_iscatter_zv2
477 :
478 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_iscatterv_iv, mp_iscatterv_lv, &
479 : mp_iscatterv_rv, mp_iscatterv_dv, mp_iscatterv_cv, mp_iscatterv_zv
480 : GENERIC, PUBLIC :: iscatterv => mp_iscatterv_iv, mp_iscatterv_lv, &
481 : mp_iscatterv_rv, mp_iscatterv_dv, mp_iscatterv_cv, mp_iscatterv_zv
482 :
483 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_alltoall_i, mp_alltoall_i22, mp_alltoall_i33, &
484 : mp_alltoall_i44, mp_alltoall_i55, mp_alltoall_i45, mp_alltoall_i34, &
485 : mp_alltoall_i11v, mp_alltoall_i22v, mp_alltoall_i54, &
486 : mp_alltoall_l, mp_alltoall_l22, mp_alltoall_l33, &
487 : mp_alltoall_l44, mp_alltoall_l55, mp_alltoall_l45, mp_alltoall_l34, &
488 : mp_alltoall_l11v, mp_alltoall_l22v, mp_alltoall_l54, &
489 : mp_alltoall_r, mp_alltoall_r22, mp_alltoall_r33, &
490 : mp_alltoall_r44, mp_alltoall_r55, mp_alltoall_r45, mp_alltoall_r34, &
491 : mp_alltoall_r11v, mp_alltoall_r22v, mp_alltoall_r54, &
492 : mp_alltoall_d, mp_alltoall_d22, mp_alltoall_d33, &
493 : mp_alltoall_d44, mp_alltoall_d55, mp_alltoall_d45, mp_alltoall_d34, &
494 : mp_alltoall_d11v, mp_alltoall_d22v, mp_alltoall_d54, &
495 : mp_alltoall_c, mp_alltoall_c22, mp_alltoall_c33, &
496 : mp_alltoall_c44, mp_alltoall_c55, mp_alltoall_c45, mp_alltoall_c34, &
497 : mp_alltoall_c11v, mp_alltoall_c22v, mp_alltoall_c54, &
498 : mp_alltoall_z, mp_alltoall_z22, mp_alltoall_z33, &
499 : mp_alltoall_z44, mp_alltoall_z55, mp_alltoall_z45, mp_alltoall_z34, &
500 : mp_alltoall_z11v, mp_alltoall_z22v, mp_alltoall_z54
501 : GENERIC, PUBLIC :: alltoall => mp_alltoall_i, mp_alltoall_i22, mp_alltoall_i33, &
502 : mp_alltoall_i44, mp_alltoall_i55, mp_alltoall_i45, mp_alltoall_i34, &
503 : mp_alltoall_i11v, mp_alltoall_i22v, mp_alltoall_i54, &
504 : mp_alltoall_l, mp_alltoall_l22, mp_alltoall_l33, &
505 : mp_alltoall_l44, mp_alltoall_l55, mp_alltoall_l45, mp_alltoall_l34, &
506 : mp_alltoall_l11v, mp_alltoall_l22v, mp_alltoall_l54, &
507 : mp_alltoall_r, mp_alltoall_r22, mp_alltoall_r33, &
508 : mp_alltoall_r44, mp_alltoall_r55, mp_alltoall_r45, mp_alltoall_r34, &
509 : mp_alltoall_r11v, mp_alltoall_r22v, mp_alltoall_r54, &
510 : mp_alltoall_d, mp_alltoall_d22, mp_alltoall_d33, &
511 : mp_alltoall_d44, mp_alltoall_d55, mp_alltoall_d45, mp_alltoall_d34, &
512 : mp_alltoall_d11v, mp_alltoall_d22v, mp_alltoall_d54, &
513 : mp_alltoall_c, mp_alltoall_c22, mp_alltoall_c33, &
514 : mp_alltoall_c44, mp_alltoall_c55, mp_alltoall_c45, mp_alltoall_c34, &
515 : mp_alltoall_c11v, mp_alltoall_c22v, mp_alltoall_c54, &
516 : mp_alltoall_z, mp_alltoall_z22, mp_alltoall_z33, &
517 : mp_alltoall_z44, mp_alltoall_z55, mp_alltoall_z45, mp_alltoall_z34, &
518 : mp_alltoall_z11v, mp_alltoall_z22v, mp_alltoall_z54
519 :
520 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_send_i, mp_send_iv, mp_send_im2, mp_send_im3, &
521 : mp_send_l, mp_send_lv, mp_send_lm2, mp_send_lm3, &
522 : mp_send_r, mp_send_rv, mp_send_rm2, mp_send_rm3, &
523 : mp_send_d, mp_send_dv, mp_send_dm2, mp_send_dm3, &
524 : mp_send_c, mp_send_cv, mp_send_cm2, mp_send_cm3, &
525 : mp_send_z, mp_send_zv, mp_send_zm2, mp_send_zm3
526 : GENERIC, PUBLIC :: send => mp_send_i, mp_send_iv, mp_send_im2, mp_send_im3, &
527 : mp_send_l, mp_send_lv, mp_send_lm2, mp_send_lm3, &
528 : mp_send_r, mp_send_rv, mp_send_rm2, mp_send_rm3, &
529 : mp_send_d, mp_send_dv, mp_send_dm2, mp_send_dm3, &
530 : mp_send_c, mp_send_cv, mp_send_cm2, mp_send_cm3, &
531 : mp_send_z, mp_send_zv, mp_send_zm2, mp_send_zm3
532 :
533 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_recv_i, mp_recv_iv, mp_recv_im2, mp_recv_im3, &
534 : mp_recv_l, mp_recv_lv, mp_recv_lm2, mp_recv_lm3, &
535 : mp_recv_r, mp_recv_rv, mp_recv_rm2, mp_recv_rm3, &
536 : mp_recv_d, mp_recv_dv, mp_recv_dm2, mp_recv_dm3, &
537 : mp_recv_c, mp_recv_cv, mp_recv_cm2, mp_recv_cm3, &
538 : mp_recv_z, mp_recv_zv, mp_recv_zm2, mp_recv_zm3
539 : GENERIC, PUBLIC :: recv => mp_recv_i, mp_recv_iv, mp_recv_im2, mp_recv_im3, &
540 : mp_recv_l, mp_recv_lv, mp_recv_lm2, mp_recv_lm3, &
541 : mp_recv_r, mp_recv_rv, mp_recv_rm2, mp_recv_rm3, &
542 : mp_recv_d, mp_recv_dv, mp_recv_dm2, mp_recv_dm3, &
543 : mp_recv_c, mp_recv_cv, mp_recv_cm2, mp_recv_cm3, &
544 : mp_recv_z, mp_recv_zv, mp_recv_zm2, mp_recv_zm3
545 :
546 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_isendrecv_i, mp_isendrecv_iv, &
547 : mp_isendrecv_l, mp_isendrecv_lv, mp_isendrecv_r, mp_isendrecv_rv, &
548 : mp_isendrecv_d, mp_isendrecv_dv, mp_isendrecv_c, mp_isendrecv_cv, &
549 : mp_isendrecv_z, mp_isendrecv_zv
550 : GENERIC, PUBLIC :: isendrecv => mp_isendrecv_i, mp_isendrecv_iv, &
551 : mp_isendrecv_l, mp_isendrecv_lv, mp_isendrecv_r, mp_isendrecv_rv, &
552 : mp_isendrecv_d, mp_isendrecv_dv, mp_isendrecv_c, mp_isendrecv_cv, &
553 : mp_isendrecv_z, mp_isendrecv_zv
554 :
555 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_isend_iv, mp_isend_im2, mp_isend_im3, mp_isend_im4, &
556 : mp_isend_lv, mp_isend_lm2, mp_isend_lm3, mp_isend_lm4, &
557 : mp_isend_rv, mp_isend_rm2, mp_isend_rm3, mp_isend_rm4, &
558 : mp_isend_dv, mp_isend_dm2, mp_isend_dm3, mp_isend_dm4, &
559 : mp_isend_cv, mp_isend_cm2, mp_isend_cm3, mp_isend_cm4, &
560 : mp_isend_zv, mp_isend_zm2, mp_isend_zm3, mp_isend_zm4, &
561 : mp_isend_bv, mp_isend_bm3, mp_isend_custom
562 : GENERIC, PUBLIC :: isend => mp_isend_iv, mp_isend_im2, mp_isend_im3, mp_isend_im4, &
563 : mp_isend_lv, mp_isend_lm2, mp_isend_lm3, mp_isend_lm4, &
564 : mp_isend_rv, mp_isend_rm2, mp_isend_rm3, mp_isend_rm4, &
565 : mp_isend_dv, mp_isend_dm2, mp_isend_dm3, mp_isend_dm4, &
566 : mp_isend_cv, mp_isend_cm2, mp_isend_cm3, mp_isend_cm4, &
567 : mp_isend_zv, mp_isend_zm2, mp_isend_zm3, mp_isend_zm4, &
568 : mp_isend_bv, mp_isend_bm3, mp_isend_custom
569 :
570 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_irecv_iv, mp_irecv_im2, mp_irecv_im3, mp_irecv_im4, &
571 : mp_irecv_lv, mp_irecv_lm2, mp_irecv_lm3, mp_irecv_lm4, &
572 : mp_irecv_rv, mp_irecv_rm2, mp_irecv_rm3, mp_irecv_rm4, &
573 : mp_irecv_dv, mp_irecv_dm2, mp_irecv_dm3, mp_irecv_dm4, &
574 : mp_irecv_cv, mp_irecv_cm2, mp_irecv_cm3, mp_irecv_cm4, &
575 : mp_irecv_zv, mp_irecv_zm2, mp_irecv_zm3, mp_irecv_zm4, &
576 : mp_irecv_bv, mp_irecv_bm3, mp_irecv_custom
577 : GENERIC, PUBLIC :: irecv => mp_irecv_iv, mp_irecv_im2, mp_irecv_im3, mp_irecv_im4, &
578 : mp_irecv_lv, mp_irecv_lm2, mp_irecv_lm3, mp_irecv_lm4, &
579 : mp_irecv_rv, mp_irecv_rm2, mp_irecv_rm3, mp_irecv_rm4, &
580 : mp_irecv_dv, mp_irecv_dm2, mp_irecv_dm3, mp_irecv_dm4, &
581 : mp_irecv_cv, mp_irecv_cm2, mp_irecv_cm3, mp_irecv_cm4, &
582 : mp_irecv_zv, mp_irecv_zm2, mp_irecv_zm3, mp_irecv_zm4, &
583 : mp_irecv_bv, mp_irecv_bm3, mp_irecv_custom
584 :
585 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: probe => mp_probe
586 :
587 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: sync => mp_sync
588 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: isync => mp_isync
589 :
590 : PROCEDURE, PUBLIC, PASS(comm1), NON_OVERRIDABLE :: compare => mp_comm_compare
591 : PROCEDURE, PUBLIC, PASS(comm1), NON_OVERRIDABLE :: rank_compare => mp_rank_compare
592 :
593 : PROCEDURE, PUBLIC, PASS(comm2), NON_OVERRIDABLE :: from_dup => mp_comm_dup
594 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: mp_comm_free
595 : GENERIC, PUBLIC :: free => mp_comm_free
596 :
597 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: mp_comm_init
598 : GENERIC, PUBLIC :: init => mp_comm_init
599 :
600 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: get_size => mp_comm_size
601 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: get_rank => mp_comm_rank
602 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: get_ndims => mp_comm_get_ndims
603 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: is_source => mp_comm_is_source
604 :
605 : ! Creation routines
606 : PROCEDURE, PRIVATE, PASS(sub_comm), NON_OVERRIDABLE :: mp_comm_split, mp_comm_split_direct
607 : GENERIC, PUBLIC :: from_split => mp_comm_split, mp_comm_split_direct
608 : PROCEDURE, PUBLIC, PASS(mp_new_comm), NON_OVERRIDABLE :: from_reordering => mp_reordering
609 : PROCEDURE, PUBLIC, PASS(comm_new), NON_OVERRIDABLE :: mp_comm_assign
610 : GENERIC, PUBLIC :: ASSIGNMENT(=) => mp_comm_assign
611 :
612 : ! Other Getters
613 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_comm_get_tag_ub
614 : GENERIC, PUBLIC :: get_tag_ub => mp_comm_get_tag_ub
615 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_comm_get_host_rank
616 : GENERIC, PUBLIC :: get_host_rank => mp_comm_get_host_rank
617 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_comm_get_io_rank
618 : GENERIC, PUBLIC :: get_io_rank => mp_comm_get_io_rank
619 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: mp_comm_get_wtime_is_global
620 : GENERIC, PUBLIC :: get_wtime_is_global => mp_comm_get_wtime_is_global
621 : END TYPE
622 :
623 : TYPE mp_request_type
624 : PRIVATE
625 : MPI_REQUEST_TYPE :: handle = mp_request_null_handle
626 : CONTAINS
627 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: set_handle => mp_request_type_set_handle
628 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: get_handle => mp_request_type_get_handle
629 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_request_op_eq
630 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_request_op_neq
631 : GENERIC, PUBLIC :: OPERATOR(.EQ.) => mp_request_op_eq
632 : GENERIC, PUBLIC :: OPERATOR(.NE.) => mp_request_op_neq
633 :
634 : PROCEDURE, PUBLIC, PASS(request), NON_OVERRIDABLE :: test => mp_test_1
635 :
636 : PROCEDURE, PUBLIC, PASS(request), NON_OVERRIDABLE :: wait => mp_wait
637 : END TYPE
638 :
639 : TYPE mp_win_type
640 : PRIVATE
641 : MPI_WIN_TYPE :: handle = mp_win_null_handle
642 : CONTAINS
643 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: set_handle => mp_win_type_set_handle
644 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: get_handle => mp_win_type_get_handle
645 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_win_op_eq
646 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_win_op_neq
647 : GENERIC, PUBLIC :: OPERATOR(.EQ.) => mp_win_op_eq
648 : GENERIC, PUBLIC :: OPERATOR(.NE.) => mp_win_op_neq
649 :
650 : PROCEDURE, PRIVATE, PASS(win), NON_OVERRIDABLE :: mp_win_create_iv, mp_win_create_lv, &
651 : mp_win_create_rv, mp_win_create_dv, mp_win_create_cv, mp_win_create_zv
652 : GENERIC, PUBLIC :: create => mp_win_create_iv, mp_win_create_lv, &
653 : mp_win_create_rv, mp_win_create_dv, mp_win_create_cv, mp_win_create_zv
654 :
655 : PROCEDURE, PRIVATE, PASS(win), NON_OVERRIDABLE :: mp_rget_iv, mp_rget_lv, &
656 : mp_rget_rv, mp_rget_dv, mp_rget_cv, mp_rget_zv
657 : GENERIC, PUBLIC :: rget => mp_rget_iv, mp_rget_lv, &
658 : mp_rget_rv, mp_rget_dv, mp_rget_cv, mp_rget_zv
659 :
660 : PROCEDURE, PUBLIC, PASS(win), NON_OVERRIDABLE :: free => mp_win_free
661 : PROCEDURE, PUBLIC, PASS(win_new), NON_OVERRIDABLE :: mp_win_assign
662 : GENERIC, PUBLIC :: ASSIGNMENT(=) => mp_win_assign
663 :
664 : PROCEDURE, PUBLIC, PASS(win), NON_OVERRIDABLE :: lock_all => mp_win_lock_all
665 : PROCEDURE, PUBLIC, PASS(win), NON_OVERRIDABLE :: unlock_all => mp_win_unlock_all
666 : PROCEDURE, PUBLIC, PASS(win), NON_OVERRIDABLE :: flush_all => mp_win_flush_all
667 : END TYPE
668 :
669 : TYPE mp_file_type
670 : PRIVATE
671 : MPI_FILE_TYPE :: handle = mp_file_null_handle
672 : CONTAINS
673 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: set_handle => mp_file_type_set_handle
674 : PROCEDURE, PUBLIC, NON_OVERRIDABLE :: get_handle => mp_file_type_get_handle
675 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_file_op_eq
676 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_file_op_neq
677 : GENERIC, PUBLIC :: OPERATOR(.EQ.) => mp_file_op_eq
678 : GENERIC, PUBLIC :: OPERATOR(.NE.) => mp_file_op_neq
679 :
680 : PROCEDURE, PRIVATE, PASS(fh), NON_OVERRIDABLE :: mp_file_write_at_ch, mp_file_write_at_chv, &
681 : mp_file_write_at_i, mp_file_write_at_iv, mp_file_write_at_r, mp_file_write_at_rv, &
682 : mp_file_write_at_d, mp_file_write_at_dv, mp_file_write_at_c, mp_file_write_at_cv, &
683 : mp_file_write_at_z, mp_file_write_at_zv, mp_file_write_at_l, mp_file_write_at_lv
684 : GENERIC, PUBLIC :: write_at => mp_file_write_at_ch, mp_file_write_at_chv, &
685 : mp_file_write_at_i, mp_file_write_at_iv, mp_file_write_at_r, mp_file_write_at_rv, &
686 : mp_file_write_at_d, mp_file_write_at_dv, mp_file_write_at_c, mp_file_write_at_cv, &
687 : mp_file_write_at_z, mp_file_write_at_zv, mp_file_write_at_l, mp_file_write_at_lv
688 :
689 : PROCEDURE, PRIVATE, PASS(fh), NON_OVERRIDABLE :: mp_file_write_at_all_ch, mp_file_write_at_all_chv, &
690 : mp_file_write_at_all_i, mp_file_write_at_all_iv, mp_file_write_at_all_l, mp_file_write_at_all_lv, &
691 : mp_file_write_at_all_r, mp_file_write_at_all_rv, mp_file_write_at_all_d, mp_file_write_at_all_dv, &
692 : mp_file_write_at_all_c, mp_file_write_at_all_cv, mp_file_write_at_all_z, mp_file_write_at_all_zv
693 : GENERIC, PUBLIC :: write_at_all => mp_file_write_at_all_ch, mp_file_write_at_all_chv, &
694 : mp_file_write_at_all_i, mp_file_write_at_all_iv, mp_file_write_at_all_l, mp_file_write_at_all_lv, &
695 : mp_file_write_at_all_r, mp_file_write_at_all_rv, mp_file_write_at_all_d, mp_file_write_at_all_dv, &
696 : mp_file_write_at_all_c, mp_file_write_at_all_cv, mp_file_write_at_all_z, mp_file_write_at_all_zv
697 :
698 : PROCEDURE, PRIVATE, PASS(fh), NON_OVERRIDABLE :: mp_file_read_at_ch, mp_file_read_at_chv, &
699 : mp_file_read_at_i, mp_file_read_at_iv, mp_file_read_at_r, mp_file_read_at_rv, &
700 : mp_file_read_at_d, mp_file_read_at_dv, mp_file_read_at_c, mp_file_read_at_cv, &
701 : mp_file_read_at_z, mp_file_read_at_zv, mp_file_read_at_l, mp_file_read_at_lv
702 : GENERIC, PUBLIC :: read_at => mp_file_read_at_ch, mp_file_read_at_chv, &
703 : mp_file_read_at_i, mp_file_read_at_iv, mp_file_read_at_r, mp_file_read_at_rv, &
704 : mp_file_read_at_d, mp_file_read_at_dv, mp_file_read_at_c, mp_file_read_at_cv, &
705 : mp_file_read_at_z, mp_file_read_at_zv, mp_file_read_at_l, mp_file_read_at_lv
706 :
707 : PROCEDURE, PRIVATE, PASS(fh), NON_OVERRIDABLE :: mp_file_read_at_all_ch, mp_file_read_at_all_chv, &
708 : mp_file_read_at_all_i, mp_file_read_at_all_iv, mp_file_read_at_all_l, mp_file_read_at_all_lv, &
709 : mp_file_read_at_all_r, mp_file_read_at_all_rv, mp_file_read_at_all_d, mp_file_read_at_all_dv, &
710 : mp_file_read_at_all_c, mp_file_read_at_all_cv, mp_file_read_at_all_z, mp_file_read_at_all_zv
711 : GENERIC, PUBLIC :: read_at_all => mp_file_read_at_all_ch, mp_file_read_at_all_chv, &
712 : mp_file_read_at_all_i, mp_file_read_at_all_iv, mp_file_read_at_all_l, mp_file_read_at_all_lv, &
713 : mp_file_read_at_all_r, mp_file_read_at_all_rv, mp_file_read_at_all_d, mp_file_read_at_all_dv, &
714 : mp_file_read_at_all_c, mp_file_read_at_all_cv, mp_file_read_at_all_z, mp_file_read_at_all_zv
715 :
716 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: open => mp_file_open
717 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: close => mp_file_close
718 : PROCEDURE, PRIVATE, PASS(fh_new), NON_OVERRIDABLE :: mp_file_assign
719 : GENERIC, PUBLIC :: ASSIGNMENT(=) => mp_file_assign
720 :
721 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: get_size => mp_file_get_size
722 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: get_position => mp_file_get_position
723 :
724 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: read_all => mp_file_read_all_chv
725 : PROCEDURE, PUBLIC, PASS(fh), NON_OVERRIDABLE :: write_all => mp_file_write_all_chv
726 : END TYPE
727 :
728 : TYPE mp_info_type
729 : PRIVATE
730 : MPI_INFO_TYPE :: handle = mp_info_null_handle
731 : CONTAINS
732 : PROCEDURE, NON_OVERRIDABLE :: set_handle => mp_info_type_set_handle
733 : PROCEDURE, NON_OVERRIDABLE :: get_handle => mp_info_type_get_handle
734 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_info_op_eq
735 : PROCEDURE, PRIVATE, NON_OVERRIDABLE :: mp_info_op_neq
736 : GENERIC, PUBLIC :: OPERATOR(.EQ.) => mp_info_op_eq
737 : GENERIC, PUBLIC :: OPERATOR(.NE.) => mp_info_op_neq
738 : END TYPE
739 :
740 : TYPE, EXTENDS(mp_comm_type) :: mp_cart_type
741 : INTEGER, DIMENSION(:), ALLOCATABLE, PUBLIC :: mepos_cart, num_pe_cart
742 : LOGICAL, DIMENSION(:), ALLOCATABLE, PUBLIC :: periodic
743 : CONTAINS
744 : PROCEDURE, PUBLIC, PASS(comm_cart), NON_OVERRIDABLE :: create => mp_cart_create
745 : PROCEDURE, PUBLIC, PASS(sub_comm), NON_OVERRIDABLE :: from_sub => mp_cart_sub
746 :
747 : PROCEDURE, PRIVATE, PASS(comm), NON_OVERRIDABLE :: get_info_cart => mp_cart_get
748 :
749 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: coords => mp_cart_coords
750 : PROCEDURE, PUBLIC, PASS(comm), NON_OVERRIDABLE :: rank_cart => mp_cart_rank
751 : END TYPE
752 :
753 : ! **************************************************************************************************
754 : !> \brief stores all the informations relevant to an mpi environment
755 : !> \param owns_group if it owns the group (and thus should free it when
756 : !> this object is deallocated)
757 : !> \param ref_count the reference count, when it is zero this object gets
758 : !> deallocated
759 : !> \par History
760 : !> 08.2002 created [fawzi]
761 : !> \author Fawzi Mohamed
762 : ! **************************************************************************************************
763 : TYPE, EXTENDS(mp_comm_type) :: mp_para_env_type
764 : PRIVATE
765 : ! We set it to true to have less initialization steps in case we create a new communicator
766 : LOGICAL :: owns_group = .TRUE.
767 : INTEGER :: ref_count = -1
768 : CONTAINS
769 : PROCEDURE, PUBLIC, PASS(para_env), NON_OVERRIDABLE :: retain => mp_para_env_retain
770 : PROCEDURE, PUBLIC, PASS(para_env), NON_OVERRIDABLE :: is_valid => mp_para_env_is_valid
771 : END TYPE mp_para_env_type
772 :
773 : ! **************************************************************************************************
774 : !> \brief represent a pointer to a para env (to build arrays)
775 : !> \param para_env the pointer to the para_env
776 : !> \par History
777 : !> 07.2003 created [fawzi]
778 : !> \author Fawzi Mohamed
779 : ! **************************************************************************************************
780 : TYPE mp_para_env_p_type
781 : TYPE(mp_para_env_type), POINTER :: para_env => NULL()
782 : END TYPE mp_para_env_p_type
783 :
784 : ! **************************************************************************************************
785 : !> \brief represent a multidimensional parallel environment
786 : !> \param mepos_cart the position of the actual processor
787 : !> \param num_pe_cart number of processors in the group in each dimension
788 : !> \param source_cart id of a special processor (for example the one for i-o,
789 : !> or the master
790 : !> \param owns_group if it owns the group (and thus should free it when
791 : !> this object is deallocated)
792 : !> \param ref_count the reference count, when it is zero this object gets
793 : !> deallocated
794 : !> \note
795 : !> not yet implemented for mpi
796 : !> \par History
797 : !> 08.2002 created [fawzi]
798 : !> \author Fawzi Mohamed
799 : ! **************************************************************************************************
800 : TYPE, EXTENDS(mp_cart_type) :: mp_para_cart_type
801 : PRIVATE
802 : ! We set it to true to have less initialization steps in case we create a new communicator
803 : LOGICAL :: owns_group = .TRUE.
804 : INTEGER :: ref_count = -1
805 : CONTAINS
806 : PROCEDURE, PUBLIC, PASS(cart), NON_OVERRIDABLE :: retain => mp_para_cart_retain
807 : PROCEDURE, PUBLIC, PASS(cart), NON_OVERRIDABLE :: is_valid => mp_para_cart_is_valid
808 : END TYPE mp_para_cart_type
809 :
810 : ! Create the constants from the corresponding handles
811 : TYPE(mp_comm_type), PARAMETER, PUBLIC :: mp_comm_null = mp_comm_type(mp_comm_null_handle)
812 : TYPE(mp_comm_type), PARAMETER, PUBLIC :: mp_comm_self = mp_comm_type(mp_comm_self_handle)
813 : TYPE(mp_comm_type), PARAMETER, PUBLIC :: mp_comm_world = mp_comm_type(mp_comm_world_handle)
814 : TYPE(mp_request_type), PARAMETER, PUBLIC :: mp_request_null = mp_request_type(mp_request_null_handle)
815 : TYPE(mp_win_type), PARAMETER, PUBLIC :: mp_win_null = mp_win_type(mp_win_null_handle)
816 : TYPE(mp_file_type), PARAMETER, PUBLIC :: mp_file_null = mp_file_type(mp_file_null_handle)
817 : TYPE(mp_info_type), PARAMETER, PUBLIC :: mp_info_null = mp_info_type(mp_info_null_handle)
818 :
819 : #if !defined(__parallel)
820 : ! This communicator is to be used in serial mode to emulate a valid communicator which is not a compiler constant
821 : INTEGER, PARAMETER, PRIVATE :: mp_comm_default_handle = 1
822 : TYPE(mp_comm_type), PARAMETER, PRIVATE :: mp_comm_default = mp_comm_type(mp_comm_default_handle)
823 : #endif
824 :
825 : ! Constants to compare communicators
826 : INTEGER, PARAMETER, PUBLIC :: mp_comm_ident = 0
827 : INTEGER, PARAMETER, PUBLIC :: mp_comm_congruent = 1
828 : INTEGER, PARAMETER, PUBLIC :: mp_comm_similar = 2
829 : INTEGER, PARAMETER, PUBLIC :: mp_comm_unequal = 3
830 : INTEGER, PARAMETER, PUBLIC :: mp_comm_compare_default = -1
831 :
832 : ! init and error
833 : PUBLIC :: mp_world_init, mp_world_finalize
834 : PUBLIC :: mp_abort
835 :
836 : ! informational / generation of sub comms
837 : PUBLIC :: mp_dims_create
838 : PUBLIC :: cp2k_is_parallel
839 : PUBLIC :: mp_get_node_global_rank
840 :
841 : ! message passing
842 : PUBLIC :: mp_waitall, mp_waitany
843 : PUBLIC :: mp_testall, mp_testany
844 :
845 : ! Memory management
846 : PUBLIC :: mp_allocate, mp_deallocate
847 :
848 : ! I/O
849 : PUBLIC :: mp_file_delete
850 : PUBLIC :: mp_file_get_amode
851 :
852 : ! some 'advanced types' currently only used for dbcsr
853 : PUBLIC :: mp_type_descriptor_type
854 : PUBLIC :: mp_type_make
855 : PUBLIC :: mp_type_size
856 :
857 : ! vector types
858 : PUBLIC :: mp_type_indexed_make_r, mp_type_indexed_make_d, &
859 : mp_type_indexed_make_c, mp_type_indexed_make_z
860 :
861 : ! More I/O types and routines: variable spaced data using bytes for spacings
862 : PUBLIC :: mp_file_descriptor_type
863 : PUBLIC :: mp_file_type_free
864 : PUBLIC :: mp_file_type_hindexed_make_chv
865 : PUBLIC :: mp_file_type_set_view_chv
866 :
867 : PUBLIC :: mp_get_library_version
868 :
869 : ! assumed to be private
870 :
871 : INTERFACE mp_waitall
872 : MODULE PROCEDURE mp_waitall_1, mp_waitall_2
873 : END INTERFACE
874 :
875 : INTERFACE mp_testall
876 : MODULE PROCEDURE mp_testall_tv
877 : END INTERFACE
878 :
879 : INTERFACE mp_testany
880 : MODULE PROCEDURE mp_testany_1, mp_testany_2
881 : END INTERFACE
882 :
883 : INTERFACE mp_type_free
884 : MODULE PROCEDURE mp_type_free_m, mp_type_free_v
885 : END INTERFACE
886 :
887 : !
888 : ! interfaces to deal easily with scalars / vectors / matrices / ...
889 : ! of the different types (integers, doubles, logicals, characters)
890 : !
891 : INTERFACE mp_allocate
892 : MODULE PROCEDURE mp_allocate_i, &
893 : mp_allocate_l, &
894 : mp_allocate_r, &
895 : mp_allocate_d, &
896 : mp_allocate_c, &
897 : mp_allocate_z
898 : END INTERFACE
899 :
900 : INTERFACE mp_deallocate
901 : MODULE PROCEDURE mp_deallocate_i, &
902 : mp_deallocate_l, &
903 : mp_deallocate_r, &
904 : mp_deallocate_d, &
905 : mp_deallocate_c, &
906 : mp_deallocate_z
907 : END INTERFACE
908 :
909 : INTERFACE mp_type_make
910 : MODULE PROCEDURE mp_type_make_struct
911 : MODULE PROCEDURE mp_type_make_i, mp_type_make_l, &
912 : mp_type_make_r, mp_type_make_d, &
913 : mp_type_make_c, mp_type_make_z
914 : END INTERFACE
915 :
916 : INTERFACE mp_alloc_mem
917 : MODULE PROCEDURE mp_alloc_mem_i, mp_alloc_mem_l, &
918 : mp_alloc_mem_d, mp_alloc_mem_z, &
919 : mp_alloc_mem_r, mp_alloc_mem_c
920 : END INTERFACE
921 :
922 : INTERFACE mp_free_mem
923 : MODULE PROCEDURE mp_free_mem_i, mp_free_mem_l, &
924 : mp_free_mem_d, mp_free_mem_z, &
925 : mp_free_mem_r, mp_free_mem_c
926 : END INTERFACE
927 :
928 : ! Type declarations
929 : TYPE mp_indexing_meta_type
930 : INTEGER, DIMENSION(:), POINTER :: index => NULL(), chunks => NULL()
931 : END TYPE mp_indexing_meta_type
932 :
933 : TYPE mp_type_descriptor_type
934 : MPI_DATA_TYPE :: type_handle = mp_datatype_null_handle
935 : INTEGER :: length = -1
936 : #if defined(__parallel)
937 : INTEGER(kind=mpi_address_kind) :: base = -1
938 : #endif
939 : INTEGER(kind=int_4), DIMENSION(:), POINTER :: data_i => NULL()
940 : INTEGER(kind=int_8), DIMENSION(:), POINTER :: data_l => NULL()
941 : REAL(kind=real_4), DIMENSION(:), POINTER :: data_r => NULL()
942 : REAL(kind=real_8), DIMENSION(:), POINTER :: data_d => NULL()
943 : COMPLEX(kind=real_4), DIMENSION(:), POINTER :: data_c => NULL()
944 : COMPLEX(kind=real_8), DIMENSION(:), POINTER :: data_z => NULL()
945 : TYPE(mp_type_descriptor_type), DIMENSION(:), POINTER :: subtype => NULL()
946 : INTEGER :: vector_descriptor(2) = -1
947 : LOGICAL :: has_indexing = .FALSE.
948 : TYPE(mp_indexing_meta_type) :: index_descriptor = mp_indexing_meta_type()
949 : END TYPE mp_type_descriptor_type
950 :
951 : TYPE mp_file_indexing_meta_type
952 : INTEGER, DIMENSION(:), POINTER :: index => NULL()
953 : INTEGER(kind=file_offset), &
954 : DIMENSION(:), POINTER :: chunks => NULL()
955 : END TYPE mp_file_indexing_meta_type
956 :
957 : TYPE mp_file_descriptor_type
958 : MPI_DATA_TYPE :: type_handle = mp_datatype_null_handle
959 : INTEGER :: length = -1
960 : LOGICAL :: has_indexing = .FALSE.
961 : TYPE(mp_file_indexing_meta_type) :: index_descriptor = mp_file_indexing_meta_type()
962 : END TYPE
963 :
964 : ! we make some assumptions on the length of INTEGERS, REALS and LOGICALS
965 : INTEGER, PARAMETER :: intlen = BIT_SIZE(0)/8
966 : INTEGER, PARAMETER :: reallen = 8
967 : INTEGER, PARAMETER :: loglen = BIT_SIZE(0)/8
968 : INTEGER, PARAMETER :: charlen = 1
969 :
970 : LOGICAL, PUBLIC, SAVE :: mp_collect_timings = .FALSE.
971 :
972 : CONTAINS
973 :
974 : #:mute
975 : #:set types = ["comm", "request", "win", "file", "info"]
976 : #:endmute
977 : #:for type in types
978 2636880 : LOGICAL FUNCTION mp_${type}$_op_eq(${type}$1, ${type}$2)
979 : CLASS(mp_${type}$_type), INTENT(IN) :: ${type}$1, ${type}$2
980 : #if defined(__parallel) && defined(__MPI_F08)
981 : mp_${type}$_op_eq = (${type}$1%handle%mpi_val .EQ. ${type}$2%handle%mpi_val)
982 : #else
983 2636880 : mp_${type}$_op_eq = (${type}$1%handle .EQ. ${type}$2%handle)
984 : #endif
985 2636880 : END FUNCTION mp_${type}$_op_eq
986 :
987 2834258 : LOGICAL FUNCTION mp_${type}$_op_neq(${type}$1, ${type}$2)
988 : CLASS(mp_${type}$_type), INTENT(IN) :: ${type}$1, ${type}$2
989 : #if defined(__parallel) && defined(__MPI_F08)
990 : mp_${type}$_op_neq = (${type}$1%handle%mpi_val .NE. ${type}$2%handle%mpi_val)
991 : #else
992 2834258 : mp_${type}$_op_neq = (${type}$1%handle .NE. ${type}$2%handle)
993 : #endif
994 2834258 : END FUNCTION mp_${type}$_op_neq
995 :
996 6365891 : ELEMENTAL #{if type=="comm"}#IMPURE #{endif}#SUBROUTINE mp_${type}$_type_set_handle(this, handle #{if type=="comm"}#, ndims#{endif}#)
997 : CLASS(mp_${type}$_type), INTENT(INOUT) :: this
998 : INTEGER, INTENT(IN) :: handle
999 : #:if type=="comm"
1000 : INTEGER, INTENT(IN), OPTIONAL :: ndims
1001 : #:endif
1002 :
1003 : #if defined(__parallel) && defined(__MPI_F08)
1004 : this%handle%mpi_val = handle
1005 : #else
1006 6365891 : this%handle = handle
1007 : #endif
1008 :
1009 : #:if type=="comm"
1010 : SELECT TYPE (this)
1011 : CLASS IS (mp_cart_type)
1012 0 : IF (.NOT. PRESENT(ndims)) &
1013 : CALL cp_abort(__LOCATION__, &
1014 0 : "Setup of a cartesian communicator requires information on the number of dimensions!")
1015 : END SELECT
1016 6362191 : IF (PRESENT(ndims)) this%ndims = ndims
1017 6362191 : CALL this%init()
1018 : #:endif
1019 :
1020 6365891 : END SUBROUTINE mp_${type}$_type_set_handle
1021 :
1022 2137417 : ELEMENTAL FUNCTION mp_${type}$_type_get_handle(this) RESULT(handle)
1023 : CLASS(mp_${type}$_type), INTENT(IN) :: this
1024 : INTEGER :: handle
1025 :
1026 : #if defined(__parallel) && defined(__MPI_F08)
1027 : handle = this%handle%mpi_val
1028 : #else
1029 2137417 : handle = this%handle
1030 : #endif
1031 2137417 : END FUNCTION mp_${type}$_type_get_handle
1032 : #:endfor
1033 :
1034 19510 : FUNCTION mp_comm_get_tag_ub(comm) RESULT(tag_ub)
1035 : CLASS(mp_comm_type), INTENT(IN) :: comm
1036 : INTEGER :: tag_ub
1037 :
1038 : #if defined(__parallel)
1039 : INTEGER :: ierr
1040 : LOGICAL :: flag
1041 : INTEGER(KIND=MPI_ADDRESS_KIND) :: attrval
1042 :
1043 19510 : CALL MPI_COMM_GET_ATTR(comm%handle, MPI_TAG_UB, attrval, flag, ierr)
1044 19510 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_get_attr @ mp_comm_get_tag_ub")
1045 19510 : IF (.NOT. flag) CPABORT("Upper bound of tags not available!")
1046 19510 : tag_ub = INT(attrval, KIND=KIND(tag_ub))
1047 : #else
1048 : MARK_USED(comm)
1049 : tag_ub = HUGE(1)
1050 : #endif
1051 19510 : END FUNCTION mp_comm_get_tag_ub
1052 :
1053 0 : FUNCTION mp_comm_get_host_rank(comm) RESULT(host_rank)
1054 : CLASS(mp_comm_type), INTENT(IN) :: comm
1055 : INTEGER :: host_rank
1056 :
1057 : #if defined(__parallel)
1058 : INTEGER :: ierr
1059 : LOGICAL :: flag
1060 : INTEGER(KIND=MPI_ADDRESS_KIND) :: attrval
1061 :
1062 0 : CALL MPI_COMM_GET_ATTR(comm%handle, MPI_HOST, attrval, flag, ierr)
1063 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_get_attr @ mp_comm_get_host_rank")
1064 0 : IF (.NOT. flag) CPABORT("Host process rank not available!")
1065 0 : host_rank = INT(attrval, KIND=KIND(host_rank))
1066 : #else
1067 : MARK_USED(comm)
1068 : host_rank = 0
1069 : #endif
1070 0 : END FUNCTION mp_comm_get_host_rank
1071 :
1072 0 : FUNCTION mp_comm_get_io_rank(comm) RESULT(io_rank)
1073 : CLASS(mp_comm_type), INTENT(IN) :: comm
1074 : INTEGER :: io_rank
1075 :
1076 : #if defined(__parallel)
1077 : INTEGER :: ierr
1078 : LOGICAL :: flag
1079 : INTEGER(KIND=MPI_ADDRESS_KIND) :: attrval
1080 :
1081 0 : CALL MPI_COMM_GET_ATTR(comm%handle, MPI_IO, attrval, flag, ierr)
1082 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_get_attr @ mp_comm_get_io_rank")
1083 0 : IF (.NOT. flag) CPABORT("IO rank not available!")
1084 0 : io_rank = INT(attrval, KIND=KIND(io_rank))
1085 : #else
1086 : MARK_USED(comm)
1087 : io_rank = 0
1088 : #endif
1089 0 : END FUNCTION mp_comm_get_io_rank
1090 :
1091 0 : FUNCTION mp_comm_get_wtime_is_global(comm) RESULT(wtime_is_global)
1092 : CLASS(mp_comm_type), INTENT(IN) :: comm
1093 : LOGICAL :: wtime_is_global
1094 :
1095 : #if defined(__parallel)
1096 : INTEGER :: ierr
1097 : LOGICAL :: flag
1098 : INTEGER(KIND=MPI_ADDRESS_KIND) :: attrval
1099 :
1100 0 : CALL MPI_COMM_GET_ATTR(comm%handle, MPI_TAG_UB, attrval, flag, ierr)
1101 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_get_attr @ mp_comm_get_wtime_is_global")
1102 0 : IF (.NOT. flag) CPABORT("Synchronization state of WTIME not available!")
1103 0 : wtime_is_global = (attrval == 1_MPI_ADDRESS_KIND)
1104 : #else
1105 : MARK_USED(comm)
1106 : wtime_is_global = .TRUE.
1107 : #endif
1108 0 : END FUNCTION mp_comm_get_wtime_is_global
1109 :
1110 : ! **************************************************************************************************
1111 : !> \brief initializes the system default communicator
1112 : !> \param mp_comm [output] : handle of the default communicator
1113 : !> \par History
1114 : !> 2.2004 created [Joost VandeVondele ]
1115 : !> \note
1116 : !> should only be called once
1117 : ! **************************************************************************************************
1118 8536 : SUBROUTINE mp_world_init(mp_comm)
1119 : CLASS(mp_comm_type), INTENT(OUT) :: mp_comm
1120 : #if defined(__parallel)
1121 : INTEGER :: ierr
1122 : !$ INTEGER :: provided_tsl
1123 : !$ LOGICAL :: no_threading_support
1124 :
1125 : #if defined(__NO_MPI_THREAD_SUPPORT_CHECK)
1126 : ! Hack that does not request or check MPI thread support level.
1127 : ! User asserts that the MPI library will work correctly with
1128 : ! threads.
1129 : !
1130 : !$ no_threading_support = .TRUE.
1131 : #else
1132 : ! Does the right thing when using OpenMP: requests that the MPI
1133 : ! library supports serialized mode and verifies that the MPI library
1134 : ! provides that support.
1135 : !
1136 : ! Developers: Only the master thread will ever make calls to the
1137 : ! MPI library.
1138 : !
1139 8536 : !$ no_threading_support = .FALSE.
1140 : #endif
1141 : !$ IF (no_threading_support) THEN
1142 : CALL mpi_init(ierr)
1143 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_init @ mp_world_init")
1144 : !$ ELSE
1145 8536 : !$OMP MASTER
1146 : #if defined(__DLAF)
1147 : ! DLA-Future requires that the MPI library supports
1148 : ! THREAD_MULTIPLE mode
1149 : !$ CALL mpi_init_thread(MPI_THREAD_MULTIPLE, provided_tsl, ierr)
1150 : #else
1151 8536 : !$ CALL mpi_init_thread(MPI_THREAD_SERIALIZED, provided_tsl, ierr)
1152 : #endif
1153 8536 : !$ IF (ierr /= 0) CALL mp_stop(ierr, "mpi_init_thread @ mp_world_init")
1154 : #if defined(__DLAF)
1155 : !$ IF (provided_tsl .LT. MPI_THREAD_MULTIPLE) THEN
1156 : !$ CALL mp_stop(0, "MPI library does not support the requested level of threading (MPI_THREAD_MULTIPLE), required by DLA-Future. Build CP2K without DLA-Future.")
1157 : !$ END IF
1158 : #else
1159 8536 : !$ IF (provided_tsl .LT. MPI_THREAD_SERIALIZED) THEN
1160 0 : !$ CALL mp_stop(0, "MPI library does not support the requested level of threading (MPI_THREAD_SERIALIZED).")
1161 : !$ END IF
1162 : #endif
1163 : !$OMP END MASTER
1164 : !$ END IF
1165 8536 : CALL mpi_comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN, ierr)
1166 8536 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_set_errhandler @ mp_world_init")
1167 : #endif
1168 8536 : debug_comm_count = 1
1169 8536 : mp_comm = mp_comm_world
1170 8536 : CALL mp_comm%init()
1171 8536 : CALL add_mp_perf_env()
1172 8536 : END SUBROUTINE mp_world_init
1173 :
1174 : ! **************************************************************************************************
1175 : !> \brief re-create the system default communicator with a different MPI
1176 : !> rank order
1177 : !> \param mp_comm [output] : handle of the default communicator
1178 : !> \param mp_new_comm ...
1179 : !> \param ranks_order ...
1180 : !> \par History
1181 : !> 1.2012 created [ Christiane Pousa ]
1182 : !> \note
1183 : !> should only be called once, at very beginning of CP2K run
1184 : ! **************************************************************************************************
1185 744 : SUBROUTINE mp_reordering(mp_comm, mp_new_comm, ranks_order)
1186 : CLASS(mp_comm_type), INTENT(IN) :: mp_comm
1187 : CLASS(mp_comm_type), INTENT(out) :: mp_new_comm
1188 : INTEGER, DIMENSION(:), CONTIGUOUS, INTENT(IN) :: ranks_order
1189 :
1190 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_reordering'
1191 :
1192 : INTEGER :: handle, ierr
1193 : #if defined(__parallel)
1194 : MPI_GROUP_TYPE :: newgroup, oldgroup
1195 : #endif
1196 :
1197 744 : CALL mp_timeset(routineN, handle)
1198 744 : ierr = 0
1199 : #if defined(__parallel)
1200 :
1201 744 : CALL mpi_comm_group(mp_comm%handle, oldgroup, ierr)
1202 744 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_group @ mp_reordering")
1203 744 : CALL mpi_group_incl(oldgroup, SIZE(ranks_order), ranks_order, newgroup, ierr)
1204 744 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_group_incl @ mp_reordering")
1205 :
1206 744 : CALL mpi_comm_create(mp_comm%handle, newgroup, mp_new_comm%handle, ierr)
1207 744 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_create @ mp_reordering")
1208 :
1209 744 : CALL mpi_group_free(oldgroup, ierr)
1210 744 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_group_free @ mp_reordering")
1211 744 : CALL mpi_group_free(newgroup, ierr)
1212 744 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_group_free @ mp_reordering")
1213 :
1214 744 : CALL add_perf(perf_id=1, count=1)
1215 : #else
1216 : MARK_USED(mp_comm)
1217 : MARK_USED(ranks_order)
1218 : mp_new_comm%handle = mp_comm_default_handle
1219 : #endif
1220 744 : debug_comm_count = debug_comm_count + 1
1221 744 : CALL mp_new_comm%init()
1222 744 : CALL mp_timestop(handle)
1223 744 : END SUBROUTINE mp_reordering
1224 :
1225 : ! **************************************************************************************************
1226 : !> \brief finalizes the system default communicator
1227 : !> \par History
1228 : !> 2.2004 created [Joost VandeVondele]
1229 : ! **************************************************************************************************
1230 8536 : SUBROUTINE mp_world_finalize()
1231 :
1232 : CHARACTER(LEN=default_string_length) :: debug_comm_count_char
1233 : #if defined(__parallel)
1234 : INTEGER :: ierr
1235 8536 : CALL mpi_barrier(MPI_COMM_WORLD, ierr) ! call mpi directly to avoid 0 stack pointer
1236 : #endif
1237 8536 : CALL rm_mp_perf_env()
1238 :
1239 8536 : debug_comm_count = debug_comm_count - 1
1240 : #if defined(__parallel)
1241 8536 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_barrier @ mp_world_finalize")
1242 : #endif
1243 8536 : IF (debug_comm_count .NE. 0) THEN
1244 : ! A bug, we're leaking or double-freeing communicators. Needs to be fixed where the leak happens.
1245 : ! Memory leak checking might be helpful to locate the culprit
1246 0 : WRITE (unit=debug_comm_count_char, FMT='(I0)')
1247 : CALL cp_abort(__LOCATION__, "mp_world_finalize: assert failed:"// &
1248 0 : " leaking communicators "//TRIM(debug_comm_count_char))
1249 : END IF
1250 : #if defined(__parallel)
1251 8536 : CALL mpi_finalize(ierr)
1252 8536 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_finalize @ mp_world_finalize")
1253 : #endif
1254 :
1255 8536 : END SUBROUTINE mp_world_finalize
1256 :
1257 : ! all the following routines should work for a given communicator, not MPI_WORLD
1258 :
1259 : ! **************************************************************************************************
1260 : !> \brief globally stops all tasks
1261 : !> this is intended to be low level, most of CP2K should call cp_abort()
1262 : ! **************************************************************************************************
1263 0 : SUBROUTINE mp_abort()
1264 : INTEGER :: ierr
1265 :
1266 0 : ierr = 0
1267 :
1268 : #if !defined(__NO_ABORT)
1269 : #if defined(__parallel)
1270 : CALL mpi_abort(MPI_COMM_WORLD, 1, ierr)
1271 : #else
1272 : CALL m_abort()
1273 : #endif
1274 : #endif
1275 : ! this routine never returns and levels with non-zero exit code
1276 0 : STOP 1
1277 : END SUBROUTINE mp_abort
1278 :
1279 : ! **************************************************************************************************
1280 : !> \brief stops *after an mpi error* translating the error code
1281 : !> \param ierr an error code * returned by an mpi call *
1282 : !> \param prg_code ...
1283 : !> \note
1284 : !> this function is private to message_passing.F
1285 : ! **************************************************************************************************
1286 0 : SUBROUTINE mp_stop(ierr, prg_code)
1287 : INTEGER, INTENT(IN) :: ierr
1288 : CHARACTER(LEN=*), INTENT(IN) :: prg_code
1289 :
1290 : #if defined(__parallel)
1291 : INTEGER :: istat, len
1292 : CHARACTER(LEN=MPI_MAX_ERROR_STRING) :: error_string
1293 : CHARACTER(LEN=MPI_MAX_ERROR_STRING + 512) :: full_error
1294 : #else
1295 : CHARACTER(LEN=512) :: full_error
1296 : #endif
1297 :
1298 : #if defined(__parallel)
1299 0 : CALL mpi_error_string(ierr, error_string, len, istat)
1300 0 : WRITE (full_error, '(A,I0,A)') ' MPI error ', ierr, ' in '//TRIM(prg_code)//' : '//error_string(1:len)
1301 : #else
1302 : WRITE (full_error, '(A,I0,A)') ' MPI error (!?) ', ierr, ' in '//TRIM(prg_code)
1303 : #endif
1304 :
1305 0 : CPABORT(full_error)
1306 :
1307 0 : END SUBROUTINE mp_stop
1308 :
1309 : ! **************************************************************************************************
1310 : !> \brief synchronizes with a barrier a given group of mpi tasks
1311 : !> \param group mpi communicator
1312 : ! **************************************************************************************************
1313 3068892 : SUBROUTINE mp_sync(comm)
1314 : CLASS(mp_comm_type), INTENT(IN) :: comm
1315 :
1316 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_sync'
1317 :
1318 : INTEGER :: handle, ierr
1319 :
1320 3068892 : ierr = 0
1321 3068892 : CALL mp_timeset(routineN, handle)
1322 :
1323 : #if defined(__parallel)
1324 3068892 : CALL mpi_barrier(comm%handle, ierr)
1325 3068892 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_barrier @ mp_sync")
1326 3068892 : CALL add_perf(perf_id=5, count=1)
1327 : #else
1328 : MARK_USED(comm)
1329 : #endif
1330 3068892 : CALL mp_timestop(handle)
1331 :
1332 3068892 : END SUBROUTINE mp_sync
1333 :
1334 : ! **************************************************************************************************
1335 : !> \brief synchronizes with a barrier a given group of mpi tasks
1336 : !> \param comm mpi communicator
1337 : !> \param request ...
1338 : ! **************************************************************************************************
1339 0 : SUBROUTINE mp_isync(comm, request)
1340 : CLASS(mp_comm_type), INTENT(IN) :: comm
1341 : TYPE(mp_request_type), INTENT(OUT) :: request
1342 :
1343 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_isync'
1344 :
1345 : INTEGER :: handle, ierr
1346 :
1347 0 : ierr = 0
1348 0 : CALL mp_timeset(routineN, handle)
1349 :
1350 : #if defined(__parallel)
1351 0 : CALL mpi_ibarrier(comm%handle, request%handle, ierr)
1352 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_ibarrier @ mp_isync")
1353 0 : CALL add_perf(perf_id=26, count=1)
1354 : #else
1355 : MARK_USED(comm)
1356 : request = mp_request_null
1357 : #endif
1358 0 : CALL mp_timestop(handle)
1359 :
1360 0 : END SUBROUTINE mp_isync
1361 :
1362 : ! **************************************************************************************************
1363 : !> \brief returns task id for a given mpi communicator
1364 : !> \param taskid The ID of the communicator
1365 : !> \param comm mpi communicator
1366 : ! **************************************************************************************************
1367 17878668 : SUBROUTINE mp_comm_rank(taskid, comm)
1368 :
1369 : INTEGER, INTENT(OUT) :: taskid
1370 : CLASS(mp_comm_type), INTENT(IN) :: comm
1371 :
1372 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_rank'
1373 :
1374 : INTEGER :: handle
1375 : #if defined(__parallel)
1376 : INTEGER :: ierr
1377 : #endif
1378 :
1379 17878668 : CALL mp_timeset(routineN, handle)
1380 :
1381 : #if defined(__parallel)
1382 17878668 : CALL mpi_comm_rank(comm%handle, taskid, ierr)
1383 17878668 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_rank @ mp_comm_rank")
1384 : #else
1385 : MARK_USED(comm)
1386 : taskid = 0
1387 : #endif
1388 17878668 : CALL mp_timestop(handle)
1389 :
1390 17878668 : END SUBROUTINE mp_comm_rank
1391 :
1392 : ! **************************************************************************************************
1393 : !> \brief returns number of tasks for a given mpi communicator
1394 : !> \param numtask ...
1395 : !> \param comm mpi communicator
1396 : ! **************************************************************************************************
1397 17878668 : SUBROUTINE mp_comm_size(numtask, comm)
1398 :
1399 : INTEGER, INTENT(OUT) :: numtask
1400 : CLASS(mp_comm_type), INTENT(IN) :: comm
1401 :
1402 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_size'
1403 :
1404 : INTEGER :: handle
1405 : #if defined(__parallel)
1406 : INTEGER :: ierr
1407 : #endif
1408 :
1409 17878668 : CALL mp_timeset(routineN, handle)
1410 :
1411 : #if defined(__parallel)
1412 17878668 : CALL mpi_comm_size(comm%handle, numtask, ierr)
1413 17878668 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_rank @ mp_comm_size")
1414 : #else
1415 : MARK_USED(comm)
1416 : numtask = 1
1417 : #endif
1418 17878668 : CALL mp_timestop(handle)
1419 :
1420 17878668 : END SUBROUTINE mp_comm_size
1421 :
1422 : ! **************************************************************************************************
1423 : !> \brief returns info for a given Cartesian MPI communicator
1424 : !> \param comm ...
1425 : !> \param ndims ...
1426 : !> \param dims ...
1427 : !> \param task_coor ...
1428 : !> \param periods ...
1429 : ! **************************************************************************************************
1430 8279055 : SUBROUTINE mp_cart_get(comm, dims, task_coor, periods)
1431 :
1432 : CLASS(mp_cart_type), INTENT(IN) :: comm
1433 : INTEGER, INTENT(OUT), OPTIONAL :: dims(comm%ndims), task_coor(comm%ndims)
1434 : LOGICAL, INTENT(out), OPTIONAL :: periods(comm%ndims)
1435 :
1436 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_cart_get'
1437 :
1438 : INTEGER :: handle
1439 : #if defined(__parallel)
1440 : INTEGER :: ierr
1441 16558110 : INTEGER :: my_dims(comm%ndims), my_task_coor(comm%ndims)
1442 16558110 : LOGICAL :: my_periods(comm%ndims)
1443 : #endif
1444 :
1445 8279055 : CALL mp_timeset(routineN, handle)
1446 :
1447 : #if defined(__parallel)
1448 8279055 : CALL mpi_cart_get(comm%handle, comm%ndims, my_dims, my_periods, my_task_coor, ierr)
1449 8279055 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_cart_get @ mp_cart_get")
1450 33117016 : IF (PRESENT(dims)) dims = my_dims
1451 33117016 : IF (PRESENT(task_coor)) task_coor = my_task_coor
1452 33117016 : IF (PRESENT(periods)) periods = my_periods
1453 : #else
1454 : MARK_USED(comm)
1455 : IF (PRESENT(task_coor)) task_coor = 0
1456 : IF (PRESENT(dims)) dims = 1
1457 : IF (PRESENT(periods)) periods = .FALSE.
1458 : #endif
1459 8279055 : CALL mp_timestop(handle)
1460 :
1461 8279055 : END SUBROUTINE mp_cart_get
1462 :
1463 0 : INTEGER ELEMENTAL FUNCTION mp_comm_get_ndims(comm)
1464 : CLASS(mp_comm_type), INTENT(IN) :: comm
1465 :
1466 0 : mp_comm_get_ndims = comm%ndims
1467 :
1468 0 : END FUNCTION
1469 :
1470 : ! **************************************************************************************************
1471 : !> \brief creates a cartesian communicator from any communicator
1472 : !> \param comm_old ...
1473 : !> \param ndims ...
1474 : !> \param dims ...
1475 : !> \param pos ...
1476 : !> \param comm_cart ...
1477 : ! **************************************************************************************************
1478 1569303 : SUBROUTINE mp_cart_create(comm_old, ndims, dims, comm_cart)
1479 :
1480 : CLASS(mp_comm_type), INTENT(IN) :: comm_old
1481 : INTEGER, INTENT(IN) :: ndims
1482 : INTEGER, INTENT(INOUT) :: dims(ndims)
1483 : CLASS(mp_cart_type), INTENT(OUT) :: comm_cart
1484 :
1485 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_cart_create'
1486 :
1487 : INTEGER :: handle, ierr
1488 : #if defined(__parallel)
1489 1569303 : LOGICAL, DIMENSION(1:ndims) :: period
1490 : LOGICAL :: reorder
1491 : #endif
1492 :
1493 1569303 : ierr = 0
1494 1569303 : CALL mp_timeset(routineN, handle)
1495 :
1496 1569303 : comm_cart%handle = comm_old%handle
1497 : #if defined(__parallel)
1498 :
1499 4302313 : IF (ANY(dims == 0)) CALL mpi_dims_create(comm_old%num_pe, ndims, dims, ierr)
1500 1569303 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_dims_create @ mp_cart_create")
1501 :
1502 : ! FIX ME. Quick hack to avoid problems with realspace grids for compilers
1503 : ! like IBM that actually reorder the processors when creating the new
1504 : ! communicator
1505 1569303 : reorder = .FALSE.
1506 4708705 : period = .TRUE.
1507 : CALL mpi_cart_create(comm_old%handle, ndims, dims, period, reorder, comm_cart%handle, &
1508 1569303 : ierr)
1509 1569303 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_cart_create @ mp_cart_create")
1510 1569303 : CALL add_perf(perf_id=1, count=1)
1511 : #else
1512 : dims = 1
1513 : comm_cart%handle = mp_comm_default_handle
1514 : #endif
1515 1569303 : comm_cart%ndims = ndims
1516 1569303 : debug_comm_count = debug_comm_count + 1
1517 1569303 : CALL comm_cart%init()
1518 1569303 : CALL mp_timestop(handle)
1519 :
1520 1569303 : END SUBROUTINE mp_cart_create
1521 :
1522 : ! **************************************************************************************************
1523 : !> \brief wrapper to MPI_Cart_coords
1524 : !> \param comm ...
1525 : !> \param rank ...
1526 : !> \param coords ...
1527 : ! **************************************************************************************************
1528 56088 : SUBROUTINE mp_cart_coords(comm, rank, coords)
1529 :
1530 : CLASS(mp_cart_type), INTENT(IN) :: comm
1531 : INTEGER, INTENT(IN) :: rank
1532 : INTEGER, DIMENSION(:), INTENT(OUT) :: coords
1533 :
1534 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_cart_coords'
1535 :
1536 : INTEGER :: handle, ierr, m
1537 :
1538 56088 : ierr = 0
1539 56088 : CALL mp_timeset(routineN, handle)
1540 :
1541 56088 : m = SIZE(coords)
1542 : #if defined(__parallel)
1543 56088 : CALL mpi_cart_coords(comm%handle, rank, m, coords, ierr)
1544 56088 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_cart_coords @ mp_cart_coords")
1545 : #else
1546 : coords = 0
1547 : MARK_USED(rank)
1548 : MARK_USED(comm)
1549 : #endif
1550 56088 : CALL mp_timestop(handle)
1551 :
1552 56088 : END SUBROUTINE mp_cart_coords
1553 :
1554 : ! **************************************************************************************************
1555 : !> \brief wrapper to MPI_Comm_compare
1556 : !> \param comm1 ...
1557 : !> \param comm2 ...
1558 : !> \param res ...
1559 : ! **************************************************************************************************
1560 2260 : FUNCTION mp_comm_compare(comm1, comm2) RESULT(res)
1561 :
1562 : CLASS(mp_comm_type), INTENT(IN) :: comm1, comm2
1563 : INTEGER :: res
1564 :
1565 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_compare'
1566 :
1567 : INTEGER :: handle
1568 : #if defined(__parallel)
1569 : INTEGER :: ierr, iout
1570 : #endif
1571 :
1572 2260 : CALL mp_timeset(routineN, handle)
1573 :
1574 2260 : res = 0
1575 : #if defined(__parallel)
1576 2260 : CALL mpi_comm_compare(comm1%handle, comm2%handle, iout, ierr)
1577 2260 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_compare @ mp_comm_compare")
1578 : SELECT CASE (iout)
1579 : CASE (MPI_IDENT)
1580 2260 : res = mp_comm_ident
1581 : CASE (MPI_CONGRUENT)
1582 2260 : res = mp_comm_congruent
1583 : CASE (MPI_SIMILAR)
1584 0 : res = mp_comm_similar
1585 : CASE (MPI_UNEQUAL)
1586 0 : res = mp_comm_unequal
1587 : CASE default
1588 2260 : CPABORT("Unknown comparison state of the communicators!")
1589 : END SELECT
1590 : #else
1591 : MARK_USED(comm1)
1592 : MARK_USED(comm2)
1593 : #endif
1594 2260 : CALL mp_timestop(handle)
1595 :
1596 2260 : END FUNCTION mp_comm_compare
1597 :
1598 : ! **************************************************************************************************
1599 : !> \brief wrapper to MPI_Cart_sub
1600 : !> \param comm ...
1601 : !> \param rdim ...
1602 : !> \param sub_comm ...
1603 : ! **************************************************************************************************
1604 1592 : SUBROUTINE mp_cart_sub(comm, rdim, sub_comm)
1605 :
1606 : CLASS(mp_cart_type), INTENT(IN) :: comm
1607 : LOGICAL, DIMENSION(:), CONTIGUOUS, INTENT(IN) :: rdim
1608 : CLASS(mp_cart_type), INTENT(OUT) :: sub_comm
1609 :
1610 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_cart_sub'
1611 :
1612 : INTEGER :: handle
1613 : #if defined(__parallel)
1614 : INTEGER :: ierr
1615 : #endif
1616 :
1617 1592 : CALL mp_timeset(routineN, handle)
1618 :
1619 : #if defined(__parallel)
1620 1592 : CALL mpi_cart_sub(comm%handle, rdim, sub_comm%handle, ierr)
1621 1592 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_cart_sub @ mp_cart_sub")
1622 : #else
1623 : MARK_USED(comm)
1624 : MARK_USED(rdim)
1625 : sub_comm%handle = mp_comm_default_handle
1626 : #endif
1627 6368 : sub_comm%ndims = COUNT(rdim)
1628 1592 : debug_comm_count = debug_comm_count + 1
1629 1592 : CALL sub_comm%init()
1630 1592 : CALL mp_timestop(handle)
1631 :
1632 1592 : END SUBROUTINE mp_cart_sub
1633 :
1634 : ! **************************************************************************************************
1635 : !> \brief wrapper to MPI_Comm_free
1636 : !> \param comm ...
1637 : ! **************************************************************************************************
1638 3707545 : SUBROUTINE mp_comm_free(comm)
1639 :
1640 : CLASS(mp_comm_type), INTENT(INOUT) :: comm
1641 :
1642 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_free'
1643 :
1644 : INTEGER :: handle
1645 : LOGICAL :: free_comm
1646 : #if defined(__parallel)
1647 : INTEGER :: ierr
1648 : #endif
1649 :
1650 3707545 : free_comm = .TRUE.
1651 : SELECT TYPE (comm)
1652 : CLASS IS (mp_para_env_type)
1653 898237 : free_comm = .FALSE.
1654 898237 : IF (comm%ref_count <= 0) &
1655 0 : CPABORT("para_env%ref_count <= 0")
1656 898237 : comm%ref_count = comm%ref_count - 1
1657 898237 : IF (comm%ref_count <= 0) THEN
1658 188596 : free_comm = comm%owns_group
1659 : END IF
1660 : CLASS IS (mp_para_cart_type)
1661 144 : free_comm = .FALSE.
1662 144 : IF (comm%ref_count <= 0) &
1663 0 : CPABORT("para_cart%ref_count <= 0")
1664 144 : comm%ref_count = comm%ref_count - 1
1665 144 : IF (comm%ref_count <= 0) THEN
1666 144 : free_comm = comm%owns_group
1667 : END IF
1668 : END SELECT
1669 :
1670 3707545 : CALL mp_timeset(routineN, handle)
1671 :
1672 3707545 : IF (free_comm) THEN
1673 : #if defined(__parallel)
1674 2970131 : CALL mpi_comm_free(comm%handle, ierr)
1675 2970131 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_free @ mp_comm_free")
1676 : #else
1677 : comm%handle = mp_comm_null_handle
1678 : #endif
1679 2970131 : debug_comm_count = debug_comm_count - 1
1680 : END IF
1681 :
1682 : SELECT TYPE (comm)
1683 : CLASS IS (mp_cart_type)
1684 2101329 : DEALLOCATE (comm%periodic, comm%mepos_cart, comm%num_pe_cart)
1685 : END SELECT
1686 :
1687 3707545 : CALL mp_timestop(handle)
1688 :
1689 3707545 : END SUBROUTINE mp_comm_free
1690 :
1691 : ! **************************************************************************************************
1692 : !> \brief check whether the environment exists
1693 : !> \param para_env ...
1694 : !> \return ...
1695 : ! **************************************************************************************************
1696 760173 : ELEMENTAL LOGICAL FUNCTION mp_para_env_is_valid(para_env)
1697 : CLASS(mp_para_env_type), INTENT(IN) :: para_env
1698 :
1699 760173 : mp_para_env_is_valid = para_env%ref_count > 0
1700 :
1701 760173 : END FUNCTION mp_para_env_is_valid
1702 :
1703 : ! **************************************************************************************************
1704 : !> \brief increase the reference counter but ensure that you free it later
1705 : !> \param para_env ...
1706 : ! **************************************************************************************************
1707 709641 : ELEMENTAL SUBROUTINE mp_para_env_retain(para_env)
1708 : CLASS(mp_para_env_type), INTENT(INOUT) :: para_env
1709 :
1710 709641 : para_env%ref_count = para_env%ref_count + 1
1711 :
1712 709641 : END SUBROUTINE mp_para_env_retain
1713 :
1714 : ! **************************************************************************************************
1715 : !> \brief check whether the given environment is valid, i.e. existent
1716 : !> \param cart ...
1717 : !> \return ...
1718 : ! **************************************************************************************************
1719 144 : ELEMENTAL LOGICAL FUNCTION mp_para_cart_is_valid(cart)
1720 : CLASS(mp_para_cart_type), INTENT(IN) :: cart
1721 :
1722 144 : mp_para_cart_is_valid = cart%ref_count > 0
1723 :
1724 144 : END FUNCTION mp_para_cart_is_valid
1725 :
1726 : ! **************************************************************************************************
1727 : !> \brief increase the reference counter, don't forget to free it later
1728 : !> \param cart ...
1729 : ! **************************************************************************************************
1730 0 : ELEMENTAL SUBROUTINE mp_para_cart_retain(cart)
1731 : CLASS(mp_para_cart_type), INTENT(INOUT) :: cart
1732 :
1733 0 : cart%ref_count = cart%ref_count + 1
1734 :
1735 0 : END SUBROUTINE mp_para_cart_retain
1736 :
1737 : ! **************************************************************************************************
1738 : !> \brief wrapper to MPI_Comm_dup
1739 : !> \param comm1 ...
1740 : !> \param comm2 ...
1741 : ! **************************************************************************************************
1742 559760 : SUBROUTINE mp_comm_dup(comm1, comm2)
1743 :
1744 : CLASS(mp_comm_type), INTENT(IN) :: comm1
1745 : CLASS(mp_comm_type), INTENT(OUT) :: comm2
1746 :
1747 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_dup'
1748 :
1749 : INTEGER :: handle
1750 : #if defined(__parallel)
1751 : INTEGER :: ierr
1752 : #endif
1753 :
1754 559760 : CALL mp_timeset(routineN, handle)
1755 :
1756 : #if defined(__parallel)
1757 559760 : CALL mpi_comm_dup(comm1%handle, comm2%handle, ierr)
1758 559760 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_dup @ mp_comm_dup")
1759 : #else
1760 : MARK_USED(comm1)
1761 : comm2%handle = mp_comm_default_handle
1762 : #endif
1763 559760 : comm2%ndims = comm1%ndims
1764 559760 : debug_comm_count = debug_comm_count + 1
1765 559760 : CALL comm2%init()
1766 559760 : CALL mp_timestop(handle)
1767 :
1768 559760 : END SUBROUTINE mp_comm_dup
1769 :
1770 : ! **************************************************************************************************
1771 : !> \brief Implements a simple assignment function to overload the assignment operator
1772 : !> \param comm_new communicator on the r.h.s. of the assignment operator
1773 : !> \param comm_old communicator on the l.h.s. of the assignment operator
1774 : ! **************************************************************************************************
1775 8557436 : ELEMENTAL IMPURE SUBROUTINE mp_comm_assign(comm_new, comm_old)
1776 : CLASS(mp_comm_type), INTENT(IN) :: comm_old
1777 : CLASS(mp_comm_type), INTENT(OUT) :: comm_new
1778 :
1779 8557436 : comm_new%handle = comm_old%handle
1780 8557436 : comm_new%ndims = comm_old%ndims
1781 8557436 : CALL comm_new%init(.FALSE.)
1782 8557436 : END SUBROUTINE
1783 :
1784 : ! **************************************************************************************************
1785 : !> \brief check whether the local process is the source process
1786 : !> \param para_env ...
1787 : !> \return ...
1788 : ! **************************************************************************************************
1789 13559231 : ELEMENTAL LOGICAL FUNCTION mp_comm_is_source(comm)
1790 : CLASS(mp_comm_type), INTENT(IN) :: comm
1791 :
1792 13559231 : mp_comm_is_source = comm%source == comm%mepos
1793 :
1794 13559231 : END FUNCTION mp_comm_is_source
1795 :
1796 : ! **************************************************************************************************
1797 : !> \brief Initializes the communicator (mostly relevant for its derived classes)
1798 : !> \param comm ...
1799 : ! **************************************************************************************************
1800 17898294 : ELEMENTAL IMPURE SUBROUTINE mp_comm_init(comm, owns_group)
1801 : CLASS(mp_comm_type), INTENT(INOUT) :: comm
1802 : LOGICAL, INTENT(IN), OPTIONAL :: owns_group
1803 :
1804 17898294 : IF (comm%handle MPI_GET_COMP .NE. mp_comm_null_handle MPI_GET_COMP) THEN
1805 17720998 : comm%source = 0
1806 17720998 : CALL comm%get_size(comm%num_pe)
1807 17720998 : CALL comm%get_rank(comm%mepos)
1808 : END IF
1809 :
1810 : SELECT TYPE (comm)
1811 : CLASS IS (mp_cart_type)
1812 8279055 : IF (ALLOCATED(comm%periodic)) DEALLOCATE (comm%periodic)
1813 8279055 : IF (ALLOCATED(comm%mepos_cart)) DEALLOCATE (comm%mepos_cart)
1814 8279055 : IF (ALLOCATED(comm%num_pe_cart)) DEALLOCATE (comm%num_pe_cart)
1815 :
1816 : ASSOCIATE (ndims => comm%ndims)
1817 :
1818 0 : ALLOCATE (comm%periodic(ndims), comm%mepos_cart(ndims), &
1819 57953385 : comm%num_pe_cart(ndims))
1820 : END ASSOCIATE
1821 :
1822 24837961 : comm%mepos_cart = 0
1823 24837961 : comm%periodic = .FALSE.
1824 8279055 : IF (comm%handle MPI_GET_COMP .NE. mp_comm_null_handle MPI_GET_COMP) THEN
1825 : CALL comm%get_info_cart(comm%num_pe_cart, comm%mepos_cart, &
1826 8279055 : comm%periodic)
1827 : END IF
1828 : END SELECT
1829 :
1830 : SELECT TYPE (comm)
1831 : CLASS IS (mp_para_env_type)
1832 197126 : IF (PRESENT(owns_group)) comm%owns_group = owns_group
1833 197126 : comm%ref_count = 1
1834 : CLASS IS (mp_para_cart_type)
1835 144 : IF (PRESENT(owns_group)) comm%owns_group = owns_group
1836 144 : comm%ref_count = 1
1837 : END SELECT
1838 :
1839 17898294 : END SUBROUTINE
1840 :
1841 : ! **************************************************************************************************
1842 : !> \brief creates a new para environment
1843 : !> \param para_env the new parallel environment
1844 : !> \param group the id of the actual mpi_group
1845 : !> \par History
1846 : !> 08.2002 created [fawzi]
1847 : !> \author Fawzi Mohamed
1848 : ! **************************************************************************************************
1849 0 : SUBROUTINE mp_para_env_create(para_env, group)
1850 : TYPE(mp_para_env_type), POINTER :: para_env
1851 : CLASS(mp_comm_type), INTENT(in) :: group
1852 :
1853 0 : IF (ASSOCIATED(para_env)) &
1854 0 : CPABORT("The passed para_env must not be associated!")
1855 0 : ALLOCATE (para_env)
1856 0 : para_env%mp_comm_type = group
1857 0 : CALL para_env%init()
1858 0 : END SUBROUTINE mp_para_env_create
1859 :
1860 : ! **************************************************************************************************
1861 : !> \brief releases the para object (to be called when you don't want anymore
1862 : !> the shared copy of this object)
1863 : !> \param para_env the new group
1864 : !> \par History
1865 : !> 08.2002 created [fawzi]
1866 : !> \author Fawzi Mohamed
1867 : !> \note
1868 : !> to avoid circular dependencies cp_log_handling has a private copy
1869 : !> of this method (see cp_log_handling:my_mp_para_env_release)!
1870 : ! **************************************************************************************************
1871 767414 : SUBROUTINE mp_para_env_release(para_env)
1872 : TYPE(mp_para_env_type), POINTER :: para_env
1873 :
1874 767414 : IF (ASSOCIATED(para_env)) THEN
1875 740428 : CALL para_env%free()
1876 740428 : IF (.NOT. para_env%is_valid()) DEALLOCATE (para_env)
1877 : END IF
1878 767414 : NULLIFY (para_env)
1879 767414 : END SUBROUTINE mp_para_env_release
1880 :
1881 : ! **************************************************************************************************
1882 : !> \brief creates a cart (multidimensional parallel environment)
1883 : !> \param cart the cart environment to create
1884 : !> \param group the mpi communicator
1885 : !> \author fawzi
1886 : ! **************************************************************************************************
1887 0 : SUBROUTINE mp_para_cart_create(cart, group)
1888 : TYPE(mp_para_cart_type), POINTER, INTENT(OUT) :: cart
1889 : CLASS(mp_comm_type), INTENT(in) :: group
1890 :
1891 0 : IF (ASSOCIATED(cart)) &
1892 0 : CPABORT("The passed para_cart must not be associated!")
1893 0 : ALLOCATE (cart)
1894 0 : cart%mp_cart_type = group
1895 0 : CALL cart%init()
1896 :
1897 0 : END SUBROUTINE mp_para_cart_create
1898 :
1899 : ! **************************************************************************************************
1900 : !> \brief releases the given cart
1901 : !> \param cart the cart to release
1902 : !> \author fawzi
1903 : ! **************************************************************************************************
1904 144 : SUBROUTINE mp_para_cart_release(cart)
1905 : TYPE(mp_para_cart_type), POINTER :: cart
1906 :
1907 144 : IF (ASSOCIATED(cart)) THEN
1908 144 : CALL cart%free()
1909 144 : IF (.NOT. cart%is_valid()) DEALLOCATE (cart)
1910 : END IF
1911 144 : NULLIFY (cart)
1912 144 : END SUBROUTINE mp_para_cart_release
1913 :
1914 : ! **************************************************************************************************
1915 : !> \brief wrapper to MPI_Group_translate_ranks
1916 : !> \param comm1 ...
1917 : !> \param comm2 ...
1918 : !> \param rank ...
1919 : ! **************************************************************************************************
1920 2663210 : SUBROUTINE mp_rank_compare(comm1, comm2, rank)
1921 :
1922 : CLASS(mp_comm_type), INTENT(IN) :: comm1, comm2
1923 : INTEGER, DIMENSION(:), CONTIGUOUS, INTENT(OUT) :: rank
1924 :
1925 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_rank_compare'
1926 :
1927 : INTEGER :: handle
1928 : #if defined(__parallel)
1929 : INTEGER :: i, ierr, n, n1, n2
1930 2663210 : INTEGER, ALLOCATABLE, DIMENSION(:) :: rin
1931 : MPI_GROUP_TYPE :: g1, g2
1932 : #endif
1933 :
1934 2663210 : CALL mp_timeset(routineN, handle)
1935 :
1936 7989630 : rank = 0
1937 : #if defined(__parallel)
1938 2663210 : CALL mpi_comm_size(comm1%handle, n1, ierr)
1939 2663210 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_size @ mp_rank_compare")
1940 2663210 : CALL mpi_comm_size(comm2%handle, n2, ierr)
1941 2663210 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_size @ mp_rank_compare")
1942 2663210 : n = MAX(n1, n2)
1943 2663210 : CALL mpi_comm_group(comm1%handle, g1, ierr)
1944 2663210 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_group @ mp_rank_compare")
1945 2663210 : CALL mpi_comm_group(comm2%handle, g2, ierr)
1946 2663210 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_comm_group @ mp_rank_compare")
1947 7989630 : ALLOCATE (rin(0:n - 1), STAT=ierr)
1948 2663210 : IF (ierr /= 0) &
1949 0 : CPABORT("allocate @ mp_rank_compare")
1950 7989630 : DO i = 0, n - 1
1951 7989630 : rin(i) = i
1952 : END DO
1953 2663210 : CALL mpi_group_translate_ranks(g1, n, rin, g2, rank, ierr)
1954 2663210 : IF (ierr /= 0) CALL mp_stop(ierr, &
1955 0 : "mpi_group_translate_rank @ mp_rank_compare")
1956 2663210 : CALL mpi_group_free(g1, ierr)
1957 2663210 : IF (ierr /= 0) &
1958 0 : CPABORT("group_free @ mp_rank_compare")
1959 2663210 : CALL mpi_group_free(g2, ierr)
1960 2663210 : IF (ierr /= 0) &
1961 0 : CPABORT("group_free @ mp_rank_compare")
1962 2663210 : DEALLOCATE (rin)
1963 : #else
1964 : MARK_USED(comm1)
1965 : MARK_USED(comm2)
1966 : #endif
1967 2663210 : CALL mp_timestop(handle)
1968 :
1969 2663210 : END SUBROUTINE mp_rank_compare
1970 :
1971 : ! **************************************************************************************************
1972 : !> \brief wrapper to MPI_Dims_create
1973 : !> \param nodes ...
1974 : !> \param dims ...
1975 : ! **************************************************************************************************
1976 755645 : SUBROUTINE mp_dims_create(nodes, dims)
1977 :
1978 : INTEGER, INTENT(IN) :: nodes
1979 : INTEGER, DIMENSION(:), INTENT(INOUT) :: dims
1980 :
1981 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_dims_create'
1982 :
1983 : INTEGER :: handle, ndim
1984 : #if defined(__parallel)
1985 : INTEGER :: ierr
1986 : #endif
1987 :
1988 755645 : CALL mp_timeset(routineN, handle)
1989 :
1990 755645 : ndim = SIZE(dims)
1991 : #if defined(__parallel)
1992 755645 : IF (ANY(dims == 0)) CALL mpi_dims_create(nodes, ndim, dims, ierr)
1993 755645 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_dims_create @ mp_dims_create")
1994 : #else
1995 : dims = 1
1996 : MARK_USED(nodes)
1997 : #endif
1998 755645 : CALL mp_timestop(handle)
1999 :
2000 755645 : END SUBROUTINE mp_dims_create
2001 :
2002 : ! **************************************************************************************************
2003 : !> \brief wrapper to MPI_Cart_rank
2004 : !> \param comm ...
2005 : !> \param pos ...
2006 : !> \param rank ...
2007 : ! **************************************************************************************************
2008 4167870 : SUBROUTINE mp_cart_rank(comm, pos, rank)
2009 : CLASS(mp_cart_type), INTENT(IN) :: comm
2010 : INTEGER, DIMENSION(:), INTENT(IN) :: pos
2011 : INTEGER, INTENT(OUT) :: rank
2012 :
2013 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_cart_rank'
2014 :
2015 : INTEGER :: handle
2016 : #if defined(__parallel)
2017 : INTEGER :: ierr
2018 : #endif
2019 :
2020 4167870 : CALL mp_timeset(routineN, handle)
2021 :
2022 : #if defined(__parallel)
2023 4167870 : CALL mpi_cart_rank(comm%handle, pos, rank, ierr)
2024 4167870 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_cart_rank @ mp_cart_rank")
2025 : #else
2026 : rank = 0
2027 : MARK_USED(comm)
2028 : MARK_USED(pos)
2029 : #endif
2030 4167870 : CALL mp_timestop(handle)
2031 :
2032 4167870 : END SUBROUTINE mp_cart_rank
2033 :
2034 : ! **************************************************************************************************
2035 : !> \brief waits for completion of the given request
2036 : !> \param request ...
2037 : !> \par History
2038 : !> 08.2003 created [f&j]
2039 : !> \author joost & fawzi
2040 : !> \note
2041 : !> see isendrecv
2042 : ! **************************************************************************************************
2043 5948 : SUBROUTINE mp_wait(request)
2044 : CLASS(mp_request_type), INTENT(inout) :: request
2045 :
2046 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_wait'
2047 :
2048 : INTEGER :: handle
2049 : #if defined(__parallel)
2050 : INTEGER :: ierr
2051 : #endif
2052 :
2053 5948 : CALL mp_timeset(routineN, handle)
2054 :
2055 : #if defined(__parallel)
2056 :
2057 5948 : CALL mpi_wait(request%handle, MPI_STATUS_IGNORE, ierr)
2058 5948 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_wait @ mp_wait")
2059 :
2060 5948 : CALL add_perf(perf_id=9, count=1)
2061 : #else
2062 : request%handle = mp_request_null_handle
2063 : #endif
2064 5948 : CALL mp_timestop(handle)
2065 5948 : END SUBROUTINE mp_wait
2066 :
2067 : ! **************************************************************************************************
2068 : !> \brief waits for completion of the given requests
2069 : !> \param requests ...
2070 : !> \par History
2071 : !> 08.2003 created [f&j]
2072 : !> \author joost & fawzi
2073 : !> \note
2074 : !> see isendrecv
2075 : ! **************************************************************************************************
2076 1587945 : SUBROUTINE mp_waitall_1(requests)
2077 : TYPE(mp_request_type), DIMENSION(:), INTENT(inout) :: requests
2078 :
2079 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_waitall_1'
2080 :
2081 : INTEGER :: handle
2082 : #if defined(__parallel)
2083 : INTEGER :: count, ierr
2084 : #if !defined(__MPI_F08)
2085 1587945 : INTEGER, ALLOCATABLE, DIMENSION(:, :) :: status
2086 : #else
2087 : TYPE(MPI_Status), ALLOCATABLE, DIMENSION(:) :: status
2088 : #endif
2089 : #endif
2090 :
2091 1587945 : CALL mp_timeset(routineN, handle)
2092 :
2093 : #if defined(__parallel)
2094 1587945 : count = SIZE(requests)
2095 : #if !defined(__MPI_F08)
2096 4725162 : ALLOCATE (status(MPI_STATUS_SIZE, count))
2097 : #else
2098 : ALLOCATE (status(count))
2099 : #endif
2100 1587945 : CALL mpi_waitall_internal(count, requests, status, ierr) ! MPI_STATUSES_IGNORE openmpi workaround
2101 1587945 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_waitall @ mp_waitall_1")
2102 1587945 : DEALLOCATE (status)
2103 1587945 : CALL add_perf(perf_id=9, count=1)
2104 : #else
2105 : requests = mp_request_null
2106 : #endif
2107 1587945 : CALL mp_timestop(handle)
2108 1587945 : END SUBROUTINE mp_waitall_1
2109 :
2110 : ! **************************************************************************************************
2111 : !> \brief waits for completion of the given requests
2112 : !> \param requests ...
2113 : !> \par History
2114 : !> 08.2003 created [f&j]
2115 : !> \author joost & fawzi
2116 : ! **************************************************************************************************
2117 748412 : SUBROUTINE mp_waitall_2(requests)
2118 : TYPE(mp_request_type), DIMENSION(:, :), INTENT(inout) :: requests
2119 :
2120 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_waitall_2'
2121 :
2122 : INTEGER :: handle
2123 : #if defined(__parallel)
2124 : INTEGER :: count, ierr
2125 : #if !defined(__MPI_F08)
2126 748412 : INTEGER, ALLOCATABLE, DIMENSION(:, :) :: status
2127 : #else
2128 : TYPE(MPI_Status), ALLOCATABLE, DIMENSION(:) :: status
2129 : #endif
2130 : #endif
2131 :
2132 748412 : CALL mp_timeset(routineN, handle)
2133 :
2134 : #if defined(__parallel)
2135 2245236 : count = SIZE(requests)
2136 : #if !defined(__MPI_F08)
2137 2242961 : ALLOCATE (status(MPI_STATUS_SIZE, count))
2138 : #else
2139 : ALLOCATE (status(count))
2140 : #endif
2141 :
2142 4205221 : CALL mpi_waitall_internal(count, requests, status, ierr) ! MPI_STATUSES_IGNORE openmpi workaround
2143 748412 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_waitall @ mp_waitall_2")
2144 748412 : DEALLOCATE (status)
2145 :
2146 748412 : CALL add_perf(perf_id=9, count=1)
2147 : #else
2148 : requests = mp_request_null
2149 : #endif
2150 748412 : CALL mp_timestop(handle)
2151 748412 : END SUBROUTINE mp_waitall_2
2152 :
2153 : ! **************************************************************************************************
2154 : !> \brief wrapper needed to deal with interfaces as present in openmpi 1.8.1
2155 : !> the issue is with the rank or requests
2156 : !> \param count ...
2157 : !> \param array_of_requests ...
2158 : !> \param array_of_statuses ...
2159 : !> \param ierr ...
2160 : !> \author Joost VandeVondele
2161 : ! **************************************************************************************************
2162 : #if defined(__parallel)
2163 2336357 : SUBROUTINE mpi_waitall_internal(count, array_of_requests, array_of_statuses, ierr)
2164 : INTEGER, INTENT(in) :: count
2165 : TYPE(mp_request_type), DIMENSION(count), INTENT(inout) :: array_of_requests
2166 : #if !defined(__MPI_F08)
2167 : INTEGER, DIMENSION(MPI_STATUS_SIZE, count), &
2168 : INTENT(out) :: array_of_statuses
2169 : #else
2170 : TYPE(MPI_Status), DIMENSION(count), &
2171 : INTENT(out) :: array_of_statuses
2172 : #endif
2173 : INTEGER, INTENT(out) :: ierr
2174 :
2175 : INTEGER :: i
2176 2336357 : MPI_REQUEST_TYPE, ALLOCATABLE, DIMENSION(:) :: request_handles
2177 :
2178 6968123 : ALLOCATE (request_handles(count))
2179 8431771 : DO i = 1, count
2180 8431771 : request_handles(i) = array_of_requests(i)%handle
2181 : END DO
2182 :
2183 2336357 : CALL mpi_waitall(count, request_handles, array_of_statuses, ierr)
2184 :
2185 8431771 : DO i = 1, count
2186 8431771 : array_of_requests(i)%handle = request_handles(i)
2187 : END DO
2188 :
2189 2336357 : END SUBROUTINE mpi_waitall_internal
2190 : #endif
2191 :
2192 : ! **************************************************************************************************
2193 : !> \brief waits for completion of any of the given requests
2194 : !> \param requests ...
2195 : !> \param completed ...
2196 : !> \par History
2197 : !> 09.2008 created
2198 : !> \author Iain Bethune (c) The Numerical Algorithms Group (NAG) Ltd, 2008 on behalf of the HECToR project
2199 : ! **************************************************************************************************
2200 12376 : SUBROUTINE mp_waitany(requests, completed)
2201 : TYPE(mp_request_type), DIMENSION(:), INTENT(inout) :: requests
2202 : INTEGER, INTENT(out) :: completed
2203 :
2204 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_waitany'
2205 :
2206 : INTEGER :: handle
2207 : #if defined(__parallel)
2208 : INTEGER :: count, i, ierr
2209 12376 : MPI_REQUEST_TYPE, ALLOCATABLE, DIMENSION(:) :: request_handles
2210 : #endif
2211 :
2212 12376 : CALL mp_timeset(routineN, handle)
2213 :
2214 : #if defined(__parallel)
2215 12376 : count = SIZE(requests)
2216 : ! Convert CP2K's request_handles to the plane handle for the library
2217 : ! (Maybe, the compiler optimizes it away)
2218 37128 : ALLOCATE (request_handles(count))
2219 37128 : DO i = 1, count
2220 37128 : request_handles(i) = requests(i)%handle
2221 : END DO
2222 12376 : CALL mpi_waitany(count, request_handles, completed, MPI_STATUS_IGNORE, ierr)
2223 12376 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_waitany @ mp_waitany")
2224 : ! Convert the plane handles to CP2K handles
2225 37128 : DO i = 1, count
2226 37128 : requests(i)%handle = request_handles(i)
2227 : END DO
2228 12376 : CALL add_perf(perf_id=9, count=1)
2229 : #else
2230 : requests = mp_request_null
2231 : completed = 1
2232 : #endif
2233 12376 : CALL mp_timestop(handle)
2234 24752 : END SUBROUTINE mp_waitany
2235 :
2236 : ! **************************************************************************************************
2237 : !> \brief Tests for completion of the given requests.
2238 : !> \brief We use mpi_test so that we can use a single status.
2239 : !> \param requests the list of requests to test
2240 : !> \return logical which determines if requests are complete
2241 : !> \par History
2242 : !> 3.2016 adapted to any shape [Nico Holmberg]
2243 : !> \author Alfio Lazzaro
2244 : ! **************************************************************************************************
2245 6400 : FUNCTION mp_testall_tv(requests) RESULT(flag)
2246 : TYPE(mp_request_type), DIMENSION(:), INTENT(INOUT) :: requests
2247 : LOGICAL :: flag
2248 :
2249 : #if defined(__parallel)
2250 : INTEGER :: i, ierr
2251 : LOGICAL, DIMENSION(:), POINTER :: flags
2252 : #endif
2253 :
2254 6400 : flag = .TRUE.
2255 :
2256 : #if defined(__parallel)
2257 19200 : ALLOCATE (flags(SIZE(requests)))
2258 25600 : DO i = 1, SIZE(requests)
2259 19200 : CALL mpi_test(requests(i)%handle, flags(i), MPI_STATUS_IGNORE, ierr)
2260 19200 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_testall @ mp_testall_tv")
2261 26253 : flag = flag .AND. flags(i)
2262 : END DO
2263 6400 : DEALLOCATE (flags)
2264 : #else
2265 : requests = mp_request_null
2266 : #endif
2267 6400 : END FUNCTION mp_testall_tv
2268 :
2269 : ! **************************************************************************************************
2270 : !> \brief Tests for completion of the given request.
2271 : !> \param request the request
2272 : !> \param flag logical which determines if the request is completed
2273 : !> \par History
2274 : !> 3.2016 created
2275 : !> \author Nico Holmberg
2276 : ! **************************************************************************************************
2277 24 : FUNCTION mp_test_1(request) RESULT(flag)
2278 : CLASS(mp_request_type), INTENT(inout) :: request
2279 : LOGICAL :: flag
2280 :
2281 : #if defined(__parallel)
2282 : INTEGER :: ierr
2283 :
2284 24 : CALL mpi_test(request%handle, flag, MPI_STATUS_IGNORE, ierr)
2285 24 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_test @ mp_test_1")
2286 : #else
2287 : MARK_USED(request)
2288 : flag = .TRUE.
2289 : #endif
2290 24 : END FUNCTION mp_test_1
2291 :
2292 : ! **************************************************************************************************
2293 : !> \brief tests for completion of the given requests
2294 : !> \param requests ...
2295 : !> \param completed ...
2296 : !> \param flag ...
2297 : !> \par History
2298 : !> 08.2011 created
2299 : !> \author Iain Bethune
2300 : ! **************************************************************************************************
2301 0 : SUBROUTINE mp_testany_1(requests, completed, flag)
2302 : TYPE(mp_request_type), DIMENSION(:), INTENT(inout) :: requests
2303 : INTEGER, INTENT(out), OPTIONAL :: completed
2304 : LOGICAL, INTENT(out), OPTIONAL :: flag
2305 :
2306 : #if defined(__parallel)
2307 : INTEGER :: completed_l, count, ierr
2308 : LOGICAL :: flag_l
2309 :
2310 0 : count = SIZE(requests)
2311 :
2312 0 : CALL mpi_testany_internal(count, requests, completed_l, flag_l, MPI_STATUS_IGNORE, ierr)
2313 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_testany_1 @ mp_testany")
2314 :
2315 0 : IF (PRESENT(completed)) completed = completed_l
2316 0 : IF (PRESENT(flag)) flag = flag_l
2317 : #else
2318 : MARK_USED(requests)
2319 : IF (PRESENT(completed)) completed = 1
2320 : IF (PRESENT(flag)) flag = .TRUE.
2321 : #endif
2322 0 : END SUBROUTINE mp_testany_1
2323 :
2324 : ! **************************************************************************************************
2325 : !> \brief tests for completion of the given requests
2326 : !> \param requests ...
2327 : !> \param completed ...
2328 : !> \param flag ...
2329 : !> \par History
2330 : !> 08.2011 created
2331 : !> \author Iain Bethune
2332 : ! **************************************************************************************************
2333 0 : SUBROUTINE mp_testany_2(requests, completed, flag)
2334 : TYPE(mp_request_type), DIMENSION(:, :), INTENT(inout) :: requests
2335 : INTEGER, INTENT(out), OPTIONAL :: completed
2336 : LOGICAL, INTENT(out), OPTIONAL :: flag
2337 :
2338 : #if defined(__parallel)
2339 : INTEGER :: completed_l, count, ierr
2340 : LOGICAL :: flag_l
2341 :
2342 0 : count = SIZE(requests)
2343 :
2344 0 : CALL mpi_testany_internal(count, requests, completed_l, flag_l, MPI_STATUS_IGNORE, ierr)
2345 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_testany_2 @ mp_testany")
2346 :
2347 0 : IF (PRESENT(completed)) completed = completed_l
2348 0 : IF (PRESENT(flag)) flag = flag_l
2349 : #else
2350 : MARK_USED(requests)
2351 : IF (PRESENT(completed)) completed = 1
2352 : IF (PRESENT(flag)) flag = .TRUE.
2353 : #endif
2354 0 : END SUBROUTINE mp_testany_2
2355 :
2356 : ! **************************************************************************************************
2357 : !> \brief wrapper needed to deal with interfaces as present in openmpi 1.8.1
2358 : !> the issue is with the rank or requests
2359 : !> \param count ...
2360 : !> \param array_of_requests ...
2361 : !> \param index ...
2362 : !> \param flag ...
2363 : !> \param status ...
2364 : !> \param ierr ...
2365 : !> \author Joost VandeVondele
2366 : ! **************************************************************************************************
2367 : #if defined(__parallel)
2368 0 : SUBROUTINE mpi_testany_internal(count, array_of_requests, index, flag, status, ierr)
2369 : INTEGER, INTENT(in) :: count
2370 : TYPE(mp_request_type), DIMENSION(count), INTENT(inout) :: array_of_requests
2371 : INTEGER, INTENT(out) :: index
2372 : LOGICAL, INTENT(out) :: flag
2373 : MPI_STATUS_TYPE, INTENT(out) :: status
2374 : INTEGER, INTENT(out) :: ierr
2375 :
2376 : INTEGER :: i
2377 0 : MPI_REQUEST_TYPE, ALLOCATABLE, DIMENSION(:) :: request_handles
2378 :
2379 0 : ALLOCATE (request_handles(count))
2380 0 : DO i = 1, count
2381 0 : request_handles(i) = array_of_requests(i)%handle
2382 : END DO
2383 :
2384 0 : CALL mpi_testany(count, request_handles, index, flag, status, ierr)
2385 :
2386 0 : DO i = 1, count
2387 0 : array_of_requests(i)%handle = request_handles(i)
2388 : END DO
2389 :
2390 0 : END SUBROUTINE mpi_testany_internal
2391 : #endif
2392 :
2393 : ! **************************************************************************************************
2394 : !> \brief the direct way to split a communicator each color is a sub_comm,
2395 : !> the rank order is according to the order in the orig comm
2396 : !> \param comm ...
2397 : !> \param sub_comm ...
2398 : !> \param color ...
2399 : !> \param key ...
2400 : !> \author Joost VandeVondele
2401 : ! **************************************************************************************************
2402 681062 : SUBROUTINE mp_comm_split_direct(comm, sub_comm, color, key)
2403 : CLASS(mp_comm_type), INTENT(in) :: comm
2404 : CLASS(mp_comm_type), INTENT(OUT) :: sub_comm
2405 : INTEGER, INTENT(in) :: color
2406 : INTEGER, INTENT(in), OPTIONAL :: key
2407 :
2408 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_comm_split_direct'
2409 :
2410 : INTEGER :: handle
2411 : #if defined(__parallel)
2412 : INTEGER :: ierr, my_key
2413 : #endif
2414 :
2415 681062 : CALL mp_timeset(routineN, handle)
2416 :
2417 : #if defined(__parallel)
2418 681062 : my_key = 0
2419 681062 : IF (PRESENT(key)) my_key = key
2420 681062 : CALL mpi_comm_split(comm%handle, color, my_key, sub_comm%handle, ierr)
2421 681062 : IF (ierr /= mpi_success) CALL mp_stop(ierr, routineN)
2422 681062 : CALL add_perf(perf_id=10, count=1)
2423 : #else
2424 : sub_comm%handle = mp_comm_default_handle
2425 : MARK_USED(comm)
2426 : MARK_USED(color)
2427 : MARK_USED(key)
2428 : #endif
2429 681062 : debug_comm_count = debug_comm_count + 1
2430 681062 : CALL sub_comm%init()
2431 681062 : CALL mp_timestop(handle)
2432 :
2433 681062 : END SUBROUTINE mp_comm_split_direct
2434 : ! **************************************************************************************************
2435 : !> \brief splits the given communicator in group in subgroups trying to organize
2436 : !> them in a way that the communication within each subgroup is
2437 : !> efficient (but not necessarily the communication between subgroups)
2438 : !> \param comm the mpi communicator that you want to split
2439 : !> \param sub_comm the communicator for the subgroup (created, needs to be freed later)
2440 : !> \param ngroups actual number of groups
2441 : !> \param group_distribution input : allocated with array with the nprocs entries (0 .. nprocs-1)
2442 : !> \param subgroup_min_size the minimum size of the subgroup
2443 : !> \param n_subgroups the number of subgroups wanted
2444 : !> \param group_partition n_subgroups sized array containing the number of cpus wanted per group.
2445 : !> should match the total number of cpus (only used if present and associated) (0..ngroups-1)
2446 : !> \param stride create groups using a stride (default=1) through the ranks of the comm to be split.
2447 : !> \par History
2448 : !> 10.2003 created [fawzi]
2449 : !> 02.2004 modified [Joost VandeVondele]
2450 : !> \author Fawzi Mohamed
2451 : !> \note
2452 : !> at least one of subgroup_min_size and n_subgroups is needed,
2453 : !> the other default to the value needed to use most processors.
2454 : !> if less cpus are present than needed for subgroup min size, n_subgroups,
2455 : !> just one comm is created that contains all cpus
2456 : ! **************************************************************************************************
2457 157670 : SUBROUTINE mp_comm_split(comm, sub_comm, ngroups, group_distribution, &
2458 157670 : subgroup_min_size, n_subgroups, group_partition, stride)
2459 : CLASS(mp_comm_type), INTENT(in) :: comm
2460 : CLASS(mp_comm_type), INTENT(out) :: sub_comm
2461 : INTEGER, INTENT(out) :: ngroups
2462 : INTEGER, DIMENSION(0:), INTENT(INOUT) :: group_distribution
2463 : INTEGER, INTENT(in), OPTIONAL :: subgroup_min_size, n_subgroups
2464 : INTEGER, DIMENSION(0:), INTENT(IN), OPTIONAL :: group_partition
2465 : INTEGER, OPTIONAL, INTENT(IN) :: stride
2466 :
2467 : CHARACTER(LEN=*), PARAMETER :: routineN = 'mp_comm_split', &
2468 : routineP = moduleN//':'//routineN
2469 :
2470 : INTEGER :: handle, mepos, nnodes
2471 : #if defined(__parallel)
2472 : INTEGER :: color, i, ierr, j, k, &
2473 : my_subgroup_min_size, &
2474 : istride, local_stride, irank
2475 157670 : INTEGER, DIMENSION(:), ALLOCATABLE :: rank_permutation
2476 : #endif
2477 :
2478 157670 : CALL mp_timeset(routineN, handle)
2479 :
2480 : ! actual number of groups
2481 :
2482 157670 : IF (.NOT. PRESENT(subgroup_min_size) .AND. .NOT. PRESENT(n_subgroups)) THEN
2483 0 : CPABORT(routineP//" missing arguments")
2484 : END IF
2485 157670 : IF (PRESENT(subgroup_min_size) .AND. PRESENT(n_subgroups)) THEN
2486 0 : CPABORT(routineP//" too many arguments")
2487 : END IF
2488 :
2489 157670 : CALL comm%get_size(nnodes)
2490 157670 : CALL comm%get_rank(mepos)
2491 :
2492 157670 : IF (UBOUND(group_distribution, 1) .NE. nnodes - 1) THEN
2493 0 : CPABORT(routineP//" group_distribution wrong bounds")
2494 : END IF
2495 :
2496 : #if defined(__parallel)
2497 157670 : IF (PRESENT(subgroup_min_size)) THEN
2498 144 : IF (subgroup_min_size < 0 .OR. subgroup_min_size > nnodes) THEN
2499 0 : CPABORT(routineP//" subgroup_min_size too small or too large")
2500 : END IF
2501 144 : ngroups = nnodes/subgroup_min_size
2502 144 : my_subgroup_min_size = subgroup_min_size
2503 : ELSE ! n_subgroups
2504 157526 : IF (n_subgroups <= 0) THEN
2505 0 : CPABORT(routineP//" n_subgroups too small")
2506 : END IF
2507 157526 : IF (nnodes/n_subgroups > 0) THEN ! we have a least one cpu per group
2508 153663 : ngroups = n_subgroups
2509 : ELSE ! well, only one group then
2510 3863 : ngroups = 1
2511 : END IF
2512 157526 : my_subgroup_min_size = nnodes/ngroups
2513 : END IF
2514 :
2515 : ! rank_permutation: is a permutation of ranks, so that groups are not necessarily continuous in rank of the master group
2516 : ! while the order is not critical (we only color ranks), it can e.g. be used to make groups that have just 1 rank per node
2517 : ! (by setting stride equal to the number of mpi ranks per node), or by sharing a node between two groups (stride 2).
2518 473010 : ALLOCATE (rank_permutation(0:nnodes - 1))
2519 157670 : local_stride = 1
2520 157670 : IF (PRESENT(stride)) local_stride = stride
2521 157670 : k = 0
2522 315340 : DO istride = 1, local_stride
2523 315340 : DO irank = istride - 1, nnodes - 1, local_stride
2524 311476 : rank_permutation(k) = irank
2525 311476 : k = k + 1
2526 : END DO
2527 : END DO
2528 :
2529 469146 : DO i = 0, nnodes - 1
2530 469146 : group_distribution(rank_permutation(i)) = MIN(i/my_subgroup_min_size, ngroups - 1)
2531 : END DO
2532 : ! even the user gave a partition, see if we can use it to overwrite this choice
2533 157670 : IF (PRESENT(group_partition)) THEN
2534 682118 : IF (ALL(group_partition > 0) .AND. (SUM(group_partition) .EQ. nnodes) .AND. (ngroups == SIZE(group_partition))) THEN
2535 90 : k = 0
2536 90 : DO i = 0, SIZE(group_partition) - 1
2537 150 : DO j = 1, group_partition(i)
2538 60 : group_distribution(rank_permutation(k)) = i
2539 120 : k = k + 1
2540 : END DO
2541 : END DO
2542 : ELSE
2543 : ! just ignore silently as we have reasonable defaults. Probably a warning would not be to bad
2544 : END IF
2545 : END IF
2546 157670 : color = group_distribution(mepos)
2547 157670 : CALL mpi_comm_split(comm%handle, color, 0, sub_comm%handle, ierr)
2548 157670 : IF (ierr /= mpi_success) CALL mp_stop(ierr, "in "//routineP//" split")
2549 :
2550 157670 : CALL add_perf(perf_id=10, count=1)
2551 : #else
2552 : sub_comm%handle = mp_comm_default_handle
2553 : group_distribution(0) = 0
2554 : ngroups = 1
2555 : MARK_USED(comm)
2556 : MARK_USED(stride)
2557 : MARK_USED(group_partition)
2558 : #endif
2559 157670 : debug_comm_count = debug_comm_count + 1
2560 157670 : CALL sub_comm%init()
2561 157670 : CALL mp_timestop(handle)
2562 :
2563 473010 : END SUBROUTINE mp_comm_split
2564 :
2565 : ! **************************************************************************************************
2566 : !> \brief Get the local rank on the node according to the global communicator
2567 : !> \return Node Rank id
2568 : !> \author Alfio Lazzaro
2569 : ! **************************************************************************************************
2570 0 : FUNCTION mp_get_node_global_rank() &
2571 : RESULT(node_rank)
2572 :
2573 : INTEGER :: node_rank
2574 :
2575 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_get_node_global_rank'
2576 : INTEGER :: handle
2577 : #if defined(__parallel)
2578 : INTEGER :: ierr, rank
2579 : TYPE(mp_comm_type) :: comm
2580 : #endif
2581 :
2582 0 : CALL mp_timeset(routineN, handle)
2583 :
2584 : #if defined(__parallel)
2585 0 : CALL mpi_comm_rank(MPI_COMM_WORLD, rank, ierr)
2586 0 : IF (ierr /= mpi_success) CALL mp_stop(ierr, routineN)
2587 0 : CALL mpi_comm_split_type(MPI_COMM_WORLD, MPI_COMM_TYPE_SHARED, rank, MPI_INFO_NULL, comm%handle, ierr)
2588 0 : IF (ierr /= mpi_success) CALL mp_stop(ierr, routineN)
2589 0 : CALL mpi_comm_rank(comm%handle, node_rank, ierr)
2590 0 : IF (ierr /= mpi_success) CALL mp_stop(ierr, routineN)
2591 0 : CALL mpi_comm_free(comm%handle, ierr)
2592 0 : IF (ierr /= mpi_success) CALL mp_stop(ierr, routineN)
2593 : #else
2594 : node_rank = 0
2595 : #endif
2596 0 : CALL mp_timestop(handle)
2597 :
2598 0 : END FUNCTION mp_get_node_global_rank
2599 :
2600 : ! **************************************************************************************************
2601 : !> \brief probes for an incoming message with any tag
2602 : !> \param[inout] source the source of the possible incoming message,
2603 : !> if MP_ANY_SOURCE it is a blocking one and return value is the source
2604 : !> of the next incoming message
2605 : !> if source is a different value it is a non-blocking probe returning
2606 : !> MP_ANY_SOURCE if there is no incoming message
2607 : !> \param[in] comm the communicator
2608 : !> \param[out] tag the tag of the incoming message
2609 : !> \author Mandes
2610 : ! **************************************************************************************************
2611 1824199 : SUBROUTINE mp_probe(source, comm, tag)
2612 : INTEGER, INTENT(INOUT) :: source
2613 : CLASS(mp_comm_type), INTENT(IN) :: comm
2614 : INTEGER, INTENT(OUT) :: tag
2615 :
2616 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_probe'
2617 :
2618 : INTEGER :: handle
2619 : #if defined(__parallel)
2620 : INTEGER :: ierr
2621 : MPI_STATUS_TYPE :: status_single
2622 : LOGICAL :: flag
2623 : #endif
2624 :
2625 : ! ---------------------------------------------------------------------------
2626 :
2627 1824199 : CALL mp_timeset(routineN, handle)
2628 :
2629 : #if defined(__parallel)
2630 1824199 : IF (source .EQ. mp_any_source) THEN
2631 14 : CALL mpi_probe(mp_any_source, mp_any_tag, comm%handle, status_single, ierr)
2632 14 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_probe @ mp_probe")
2633 14 : source = status_single MPI_STATUS_EXTRACT(MPI_SOURCE)
2634 14 : tag = status_single MPI_STATUS_EXTRACT(MPI_TAG)
2635 : ELSE
2636 1824185 : flag = .FALSE.
2637 1824185 : CALL mpi_iprobe(source, mp_any_tag, comm%handle, flag, status_single, ierr)
2638 1824185 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_iprobe @ mp_probe")
2639 1824185 : IF (flag .EQV. .FALSE.) THEN
2640 1815041 : source = mp_any_source
2641 1815041 : tag = -1 !status_single(MPI_TAG) ! in case of flag==false status is undefined
2642 : ELSE
2643 9144 : tag = status_single MPI_STATUS_EXTRACT(MPI_TAG)
2644 : END IF
2645 : END IF
2646 : #else
2647 : tag = -1
2648 : MARK_USED(comm)
2649 : MARK_USED(source)
2650 : #endif
2651 1824199 : CALL mp_timestop(handle)
2652 1824199 : END SUBROUTINE mp_probe
2653 :
2654 : ! **************************************************************************************************
2655 : ! Here come the data routines with none of the standard data types.
2656 : ! **************************************************************************************************
2657 :
2658 : ! **************************************************************************************************
2659 : !> \brief ...
2660 : !> \param msg ...
2661 : !> \param source ...
2662 : !> \param comm ...
2663 : ! **************************************************************************************************
2664 675672 : SUBROUTINE mp_bcast_b(msg, source, comm)
2665 : LOGICAL, INTENT(INOUT) :: msg
2666 : INTEGER, INTENT(IN) :: source
2667 : CLASS(mp_comm_type), INTENT(IN) :: comm
2668 :
2669 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_b'
2670 :
2671 : INTEGER :: handle
2672 : #if defined(__parallel)
2673 : INTEGER :: ierr, msglen
2674 : #endif
2675 :
2676 675672 : CALL mp_timeset(routineN, handle)
2677 :
2678 : #if defined(__parallel)
2679 675672 : msglen = 1
2680 675672 : CALL mpi_bcast(msg, msglen, MPI_LOGICAL, source, comm%handle, ierr)
2681 675672 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
2682 675672 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*loglen)
2683 : #else
2684 : MARK_USED(msg)
2685 : MARK_USED(source)
2686 : MARK_USED(comm)
2687 : #endif
2688 675672 : CALL mp_timestop(handle)
2689 675672 : END SUBROUTINE mp_bcast_b
2690 :
2691 : ! **************************************************************************************************
2692 : !> \brief ...
2693 : !> \param msg ...
2694 : !> \param source ...
2695 : !> \param comm ...
2696 : ! **************************************************************************************************
2697 651532 : SUBROUTINE mp_bcast_b_src(msg, comm)
2698 : LOGICAL, INTENT(INOUT) :: msg
2699 : CLASS(mp_comm_type), INTENT(IN) :: comm
2700 :
2701 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_b_src'
2702 :
2703 : INTEGER :: handle
2704 : #if defined(__parallel)
2705 : INTEGER :: ierr, msglen
2706 : #endif
2707 :
2708 651532 : CALL mp_timeset(routineN, handle)
2709 :
2710 : #if defined(__parallel)
2711 651532 : msglen = 1
2712 651532 : CALL mpi_bcast(msg, msglen, MPI_LOGICAL, comm%source, comm%handle, ierr)
2713 651532 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
2714 651532 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*loglen)
2715 : #else
2716 : MARK_USED(msg)
2717 : MARK_USED(comm)
2718 : #endif
2719 651532 : CALL mp_timestop(handle)
2720 651532 : END SUBROUTINE mp_bcast_b_src
2721 :
2722 : ! **************************************************************************************************
2723 : !> \brief ...
2724 : !> \param msg ...
2725 : !> \param source ...
2726 : !> \param comm ...
2727 : ! **************************************************************************************************
2728 0 : SUBROUTINE mp_bcast_bv(msg, source, comm)
2729 : LOGICAL, CONTIGUOUS, INTENT(INOUT) :: msg(:)
2730 : INTEGER, INTENT(IN) :: source
2731 : CLASS(mp_comm_type), INTENT(IN) :: comm
2732 :
2733 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_bv'
2734 :
2735 : INTEGER :: handle
2736 : #if defined(__parallel)
2737 : INTEGER :: ierr, msglen
2738 : #endif
2739 :
2740 0 : CALL mp_timeset(routineN, handle)
2741 :
2742 : #if defined(__parallel)
2743 0 : msglen = SIZE(msg)
2744 0 : CALL mpi_bcast(msg, msglen, MPI_LOGICAL, source, comm%handle, ierr)
2745 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
2746 0 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*loglen)
2747 : #else
2748 : MARK_USED(msg)
2749 : MARK_USED(source)
2750 : MARK_USED(comm)
2751 : #endif
2752 0 : CALL mp_timestop(handle)
2753 0 : END SUBROUTINE mp_bcast_bv
2754 :
2755 : ! **************************************************************************************************
2756 : !> \brief ...
2757 : !> \param msg ...
2758 : !> \param comm ...
2759 : ! **************************************************************************************************
2760 0 : SUBROUTINE mp_bcast_bv_src(msg, comm)
2761 : LOGICAL, CONTIGUOUS, INTENT(INOUT) :: msg(:)
2762 : CLASS(mp_comm_type), INTENT(IN) :: comm
2763 :
2764 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_bv_src'
2765 :
2766 : INTEGER :: handle
2767 : #if defined(__parallel)
2768 : INTEGER :: ierr, msglen
2769 : #endif
2770 :
2771 0 : CALL mp_timeset(routineN, handle)
2772 :
2773 : #if defined(__parallel)
2774 0 : msglen = SIZE(msg)
2775 0 : CALL mpi_bcast(msg, msglen, MPI_LOGICAL, comm%source, comm%handle, ierr)
2776 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
2777 0 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*loglen)
2778 : #else
2779 : MARK_USED(msg)
2780 : MARK_USED(comm)
2781 : #endif
2782 0 : CALL mp_timestop(handle)
2783 0 : END SUBROUTINE mp_bcast_bv_src
2784 :
2785 : ! **************************************************************************************************
2786 : !> \brief Non-blocking send of logical vector data
2787 : !> \param msgin the input message
2788 : !> \param dest the destination processor
2789 : !> \param comm the communicator object
2790 : !> \param request communication request index
2791 : !> \param tag message tag
2792 : !> \par History
2793 : !> 3.2016 added _bv subroutine [Nico Holmberg]
2794 : !> \author fawzi
2795 : !> \note see mp_irecv_iv
2796 : !> \note
2797 : !> arrays can be pointers or assumed shape, but they must be contiguous!
2798 : ! **************************************************************************************************
2799 16 : SUBROUTINE mp_isend_bv(msgin, dest, comm, request, tag)
2800 : LOGICAL, DIMENSION(:), INTENT(IN) :: msgin
2801 : INTEGER, INTENT(IN) :: dest
2802 : CLASS(mp_comm_type), INTENT(IN) :: comm
2803 : TYPE(mp_request_type), INTENT(out) :: request
2804 : INTEGER, INTENT(in), OPTIONAL :: tag
2805 :
2806 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_isend_bv'
2807 :
2808 : INTEGER :: handle
2809 : #if defined(__parallel)
2810 : INTEGER :: ierr, msglen, my_tag
2811 : LOGICAL :: foo(1)
2812 : #endif
2813 :
2814 16 : CALL mp_timeset(routineN, handle)
2815 :
2816 : #if defined(__parallel)
2817 : #if !defined(__GNUC__) || __GNUC__ >= 9
2818 16 : CPASSERT(IS_CONTIGUOUS(msgin))
2819 : #endif
2820 :
2821 16 : my_tag = 0
2822 16 : IF (PRESENT(tag)) my_tag = tag
2823 :
2824 16 : msglen = SIZE(msgin, 1)
2825 16 : IF (msglen > 0) THEN
2826 : CALL mpi_isend(msgin(1), msglen, MPI_LOGICAL, dest, my_tag, &
2827 16 : comm%handle, request%handle, ierr)
2828 : ELSE
2829 : CALL mpi_isend(foo, msglen, MPI_LOGICAL, dest, my_tag, &
2830 0 : comm%handle, request%handle, ierr)
2831 : END IF
2832 16 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_isend @ "//routineN)
2833 :
2834 16 : CALL add_perf(perf_id=11, count=1, msg_size=msglen*loglen)
2835 : #else
2836 : CPABORT("mp_isend called in non parallel case")
2837 : MARK_USED(msgin)
2838 : MARK_USED(dest)
2839 : MARK_USED(comm)
2840 : MARK_USED(tag)
2841 : request = mp_request_null
2842 : #endif
2843 16 : CALL mp_timestop(handle)
2844 16 : END SUBROUTINE mp_isend_bv
2845 :
2846 : ! **************************************************************************************************
2847 : !> \brief Non-blocking receive of logical vector data
2848 : !> \param msgout the received message
2849 : !> \param source the source processor
2850 : !> \param comm the communicator object
2851 : !> \param request communication request index
2852 : !> \param tag message tag
2853 : !> \par History
2854 : !> 3.2016 added _bv subroutine [Nico Holmberg]
2855 : !> \author fawzi
2856 : !> \note see mp_irecv_iv
2857 : !> \note
2858 : !> arrays can be pointers or assumed shape, but they must be contiguous!
2859 : ! **************************************************************************************************
2860 16 : SUBROUTINE mp_irecv_bv(msgout, source, comm, request, tag)
2861 : LOGICAL, DIMENSION(:), INTENT(INOUT) :: msgout
2862 : INTEGER, INTENT(IN) :: source
2863 : CLASS(mp_comm_type), INTENT(IN) :: comm
2864 : TYPE(mp_request_type), INTENT(out) :: request
2865 : INTEGER, INTENT(in), OPTIONAL :: tag
2866 :
2867 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_irecv_bv'
2868 :
2869 : INTEGER :: handle
2870 : #if defined(__parallel)
2871 : INTEGER :: ierr, msglen, my_tag
2872 : LOGICAL :: foo(1)
2873 : #endif
2874 :
2875 16 : CALL mp_timeset(routineN, handle)
2876 :
2877 : #if defined(__parallel)
2878 : #if !defined(__GNUC__) || __GNUC__ >= 9
2879 16 : CPASSERT(IS_CONTIGUOUS(msgout))
2880 : #endif
2881 :
2882 16 : my_tag = 0
2883 16 : IF (PRESENT(tag)) my_tag = tag
2884 :
2885 16 : msglen = SIZE(msgout, 1)
2886 16 : IF (msglen > 0) THEN
2887 : CALL mpi_irecv(msgout(1), msglen, MPI_LOGICAL, source, my_tag, &
2888 16 : comm%handle, request%handle, ierr)
2889 : ELSE
2890 : CALL mpi_irecv(foo, msglen, MPI_LOGICAL, source, my_tag, &
2891 0 : comm%handle, request%handle, ierr)
2892 : END IF
2893 16 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_ircv @ "//routineN)
2894 :
2895 16 : CALL add_perf(perf_id=12, count=1, msg_size=msglen*loglen)
2896 : #else
2897 : CPABORT("mp_irecv called in non parallel case")
2898 : MARK_USED(msgout)
2899 : MARK_USED(source)
2900 : MARK_USED(comm)
2901 : MARK_USED(tag)
2902 : request = mp_request_null
2903 : #endif
2904 16 : CALL mp_timestop(handle)
2905 16 : END SUBROUTINE mp_irecv_bv
2906 :
2907 : ! **************************************************************************************************
2908 : !> \brief Non-blocking send of rank-3 logical data
2909 : !> \param msgin the input message
2910 : !> \param dest the destination processor
2911 : !> \param comm the communicator object
2912 : !> \param request communication request index
2913 : !> \param tag message tag
2914 : !> \par History
2915 : !> 2.2016 added _bm3 subroutine [Nico Holmberg]
2916 : !> \author fawzi
2917 : !> \note see mp_irecv_iv
2918 : !> \note
2919 : !> arrays can be pointers or assumed shape, but they must be contiguous!
2920 : ! **************************************************************************************************
2921 0 : SUBROUTINE mp_isend_bm3(msgin, dest, comm, request, tag)
2922 : LOGICAL, DIMENSION(:, :, :), INTENT(INOUT) :: msgin
2923 : INTEGER, INTENT(IN) :: dest
2924 : CLASS(mp_comm_type), INTENT(IN) :: comm
2925 : TYPE(mp_request_type), INTENT(out) :: request
2926 : INTEGER, INTENT(in), OPTIONAL :: tag
2927 :
2928 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_isend_bm3'
2929 :
2930 : INTEGER :: handle
2931 : #if defined(__parallel)
2932 : INTEGER :: ierr, msglen, my_tag
2933 : LOGICAL :: foo(1)
2934 : #endif
2935 :
2936 0 : CALL mp_timeset(routineN, handle)
2937 :
2938 : #if defined(__parallel)
2939 : #if !defined(__GNUC__) || __GNUC__ >= 9
2940 0 : CPASSERT(IS_CONTIGUOUS(msgin))
2941 : #endif
2942 :
2943 0 : my_tag = 0
2944 0 : IF (PRESENT(tag)) my_tag = tag
2945 :
2946 0 : msglen = SIZE(msgin, 1)*SIZE(msgin, 2)*SIZE(msgin, 3)
2947 0 : IF (msglen > 0) THEN
2948 : CALL mpi_isend(msgin(1, 1, 1), msglen, MPI_LOGICAL, dest, my_tag, &
2949 0 : comm%handle, request%handle, ierr)
2950 : ELSE
2951 : CALL mpi_isend(foo, msglen, MPI_LOGICAL, dest, my_tag, &
2952 0 : comm%handle, request%handle, ierr)
2953 : END IF
2954 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_isend @ "//routineN)
2955 :
2956 0 : CALL add_perf(perf_id=11, count=1, msg_size=msglen*loglen)
2957 : #else
2958 : CPABORT("mp_isend called in non parallel case")
2959 : MARK_USED(msgin)
2960 : MARK_USED(dest)
2961 : MARK_USED(comm)
2962 : MARK_USED(tag)
2963 : request = mp_request_null
2964 : #endif
2965 0 : CALL mp_timestop(handle)
2966 0 : END SUBROUTINE mp_isend_bm3
2967 :
2968 : ! **************************************************************************************************
2969 : !> \brief Non-blocking receive of rank-3 logical data
2970 : !> \param msgout the received message
2971 : !> \param source the source processor
2972 : !> \param comm the communicator object
2973 : !> \param request communication request index
2974 : !> \param tag message tag
2975 : !> \par History
2976 : !> 2.2016 added _bm3 subroutine [Nico Holmberg]
2977 : !> \author fawzi
2978 : !> \note see mp_irecv_iv
2979 : !> \note
2980 : !> arrays can be pointers or assumed shape, but they must be contiguous!
2981 : ! **************************************************************************************************
2982 0 : SUBROUTINE mp_irecv_bm3(msgout, source, comm, request, tag)
2983 : LOGICAL, DIMENSION(:, :, :), INTENT(INOUT) :: msgout
2984 : INTEGER, INTENT(IN) :: source
2985 : CLASS(mp_comm_type), INTENT(IN) :: comm
2986 : TYPE(mp_request_type), INTENT(out) :: request
2987 : INTEGER, INTENT(in), OPTIONAL :: tag
2988 :
2989 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_irecv_bm3'
2990 :
2991 : INTEGER :: handle
2992 : #if defined(__parallel)
2993 : INTEGER :: ierr, msglen, my_tag
2994 : LOGICAL :: foo(1)
2995 : #endif
2996 :
2997 0 : CALL mp_timeset(routineN, handle)
2998 :
2999 : #if defined(__parallel)
3000 : #if !defined(__GNUC__) || __GNUC__ >= 9
3001 0 : CPASSERT(IS_CONTIGUOUS(msgout))
3002 : #endif
3003 :
3004 0 : my_tag = 0
3005 0 : IF (PRESENT(tag)) my_tag = tag
3006 :
3007 0 : msglen = SIZE(msgout, 1)*SIZE(msgout, 2)*SIZE(msgout, 3)
3008 0 : IF (msglen > 0) THEN
3009 : CALL mpi_irecv(msgout(1, 1, 1), msglen, MPI_LOGICAL, source, my_tag, &
3010 0 : comm%handle, request%handle, ierr)
3011 : ELSE
3012 : CALL mpi_irecv(foo, msglen, MPI_LOGICAL, source, my_tag, &
3013 0 : comm%handle, request%handle, ierr)
3014 : END IF
3015 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_ircv @ "//routineN)
3016 :
3017 0 : CALL add_perf(perf_id=12, count=1, msg_size=msglen*loglen)
3018 : #else
3019 : CPABORT("mp_irecv called in non parallel case")
3020 : MARK_USED(msgout)
3021 : MARK_USED(source)
3022 : MARK_USED(comm)
3023 : MARK_USED(request)
3024 : MARK_USED(tag)
3025 : request = mp_request_null
3026 : #endif
3027 0 : CALL mp_timestop(handle)
3028 0 : END SUBROUTINE mp_irecv_bm3
3029 :
3030 : ! **************************************************************************************************
3031 : !> \brief ...
3032 : !> \param msg ...
3033 : !> \param source ...
3034 : !> \param comm ...
3035 : ! **************************************************************************************************
3036 3671001 : SUBROUTINE mp_bcast_av(msg, source, comm)
3037 : CHARACTER(LEN=*), INTENT(INOUT) :: msg
3038 : INTEGER, INTENT(IN) :: source
3039 : CLASS(mp_comm_type), INTENT(IN) :: comm
3040 :
3041 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_av'
3042 :
3043 : INTEGER :: handle
3044 : #if defined(__parallel)
3045 : INTEGER :: i, ierr, msglen
3046 3671001 : INTEGER, DIMENSION(:), ALLOCATABLE :: imsg
3047 : #endif
3048 :
3049 3671001 : CALL mp_timeset(routineN, handle)
3050 :
3051 : #if defined(__parallel)
3052 :
3053 3671001 : IF (comm%mepos == source) msglen = LEN_TRIM(msg)
3054 :
3055 3671001 : CALL comm%bcast(msglen, source)
3056 : ! this is a workaround to avoid problems on the T3E
3057 : ! at the moment we have a data alignment error when trying to
3058 : ! broadcast characters on the T3E (not always!)
3059 : ! JH 19/3/99 on galileo
3060 : ! CALL mpi_bcast(msg,msglen,MPI_CHARACTER,source,gid%handle,ierr)
3061 11003876 : ALLOCATE (imsg(1:msglen))
3062 72185033 : DO i = 1, msglen
3063 72185033 : imsg(i) = ICHAR(msg(i:i))
3064 : END DO
3065 3671001 : CALL mpi_bcast(imsg, msglen, MPI_INTEGER, source, comm%handle, ierr)
3066 3671001 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
3067 3671001 : msg = ""
3068 72185033 : DO i = 1, msglen
3069 72185033 : msg(i:i) = CHAR(imsg(i))
3070 : END DO
3071 3671001 : DEALLOCATE (imsg)
3072 3671001 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*charlen)
3073 : #else
3074 : MARK_USED(msg)
3075 : MARK_USED(source)
3076 : MARK_USED(comm)
3077 : #endif
3078 3671001 : CALL mp_timestop(handle)
3079 3671001 : END SUBROUTINE mp_bcast_av
3080 :
3081 : ! **************************************************************************************************
3082 : !> \brief ...
3083 : !> \param msg ...
3084 : !> \param comm ...
3085 : ! **************************************************************************************************
3086 718 : SUBROUTINE mp_bcast_av_src(msg, comm)
3087 : CHARACTER(LEN=*), INTENT(INOUT) :: msg
3088 : CLASS(mp_comm_type), INTENT(IN) :: comm
3089 :
3090 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_av_src'
3091 :
3092 : INTEGER :: handle
3093 : #if defined(__parallel)
3094 : INTEGER :: i, ierr, msglen
3095 718 : INTEGER, DIMENSION(:), ALLOCATABLE :: imsg
3096 : #endif
3097 :
3098 718 : CALL mp_timeset(routineN, handle)
3099 :
3100 : #if defined(__parallel)
3101 :
3102 718 : IF (comm%is_source()) msglen = LEN_TRIM(msg)
3103 :
3104 718 : CALL comm%bcast(msglen, comm%source)
3105 : ! this is a workaround to avoid problems on the T3E
3106 : ! at the moment we have a data alignment error when trying to
3107 : ! broadcast characters on the T3E (not always!)
3108 : ! JH 19/3/99 on galileo
3109 : ! CALL mpi_bcast(msg,msglen,MPI_CHARACTER,source,gid%handle,ierr)
3110 2154 : ALLOCATE (imsg(1:msglen))
3111 13874 : DO i = 1, msglen
3112 13874 : imsg(i) = ICHAR(msg(i:i))
3113 : END DO
3114 718 : CALL mpi_bcast(imsg, msglen, MPI_INTEGER, comm%source, comm%handle, ierr)
3115 718 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
3116 718 : msg = ""
3117 13874 : DO i = 1, msglen
3118 13874 : msg(i:i) = CHAR(imsg(i))
3119 : END DO
3120 718 : DEALLOCATE (imsg)
3121 718 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*charlen)
3122 : #else
3123 : MARK_USED(msg)
3124 : MARK_USED(comm)
3125 : #endif
3126 718 : CALL mp_timestop(handle)
3127 718 : END SUBROUTINE mp_bcast_av_src
3128 :
3129 : ! **************************************************************************************************
3130 : !> \brief ...
3131 : !> \param msg ...
3132 : !> \param source ...
3133 : !> \param comm ...
3134 : ! **************************************************************************************************
3135 28 : SUBROUTINE mp_bcast_am(msg, source, comm)
3136 : CHARACTER(LEN=*), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3137 : INTEGER, INTENT(IN) :: source
3138 : CLASS(mp_comm_type), INTENT(IN) :: comm
3139 :
3140 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_am'
3141 :
3142 : INTEGER :: handle
3143 : #if defined(__parallel)
3144 : INTEGER :: i, ierr, j, k, msglen, msgsiz
3145 28 : INTEGER, ALLOCATABLE :: imsg(:), imsglen(:)
3146 : #endif
3147 :
3148 28 : CALL mp_timeset(routineN, handle)
3149 :
3150 : #if defined(__parallel)
3151 28 : msgsiz = SIZE(msg)
3152 : ! Determine size of the minimum array of integers to broadcast the string
3153 84 : ALLOCATE (imsglen(1:msgsiz))
3154 28 : IF (comm%mepos == source) THEN
3155 1894 : DO j = 1, msgsiz
3156 1894 : imsglen(j) = LEN_TRIM(msg(j))
3157 : END DO
3158 : END IF
3159 28 : CALL comm%bcast(imsglen, source)
3160 3788 : msglen = SUM(imsglen)
3161 : ! this is a workaround to avoid problems on the T3E
3162 : ! at the moment we have a data alignment error when trying to
3163 : ! broadcast characters on the T3E (not always!)
3164 : ! JH 19/3/99 on galileo
3165 84 : ALLOCATE (imsg(1:msglen))
3166 3788 : k = 0
3167 3788 : DO j = 1, msgsiz
3168 7548 : DO i = 1, imsglen(j)
3169 3760 : k = k + 1
3170 7520 : imsg(k) = ICHAR(msg(j) (i:i))
3171 : END DO
3172 : END DO
3173 28 : CALL mpi_bcast(imsg, msglen, MPI_INTEGER, source, comm%handle, ierr)
3174 28 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
3175 3788 : msg = ""
3176 : k = 0
3177 3788 : DO j = 1, msgsiz
3178 7548 : DO i = 1, imsglen(j)
3179 3760 : k = k + 1
3180 7520 : msg(j) (i:i) = CHAR(imsg(k))
3181 : END DO
3182 : END DO
3183 28 : DEALLOCATE (imsg)
3184 28 : DEALLOCATE (imsglen)
3185 28 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*charlen*msgsiz)
3186 : #else
3187 : MARK_USED(msg)
3188 : MARK_USED(source)
3189 : MARK_USED(comm)
3190 : #endif
3191 28 : CALL mp_timestop(handle)
3192 56 : END SUBROUTINE mp_bcast_am
3193 :
3194 76888 : SUBROUTINE mp_bcast_am_src(msg, comm)
3195 : CHARACTER(LEN=*), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3196 : CLASS(mp_comm_type), INTENT(IN) :: comm
3197 :
3198 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_bcast_am_src'
3199 :
3200 : INTEGER :: handle
3201 : #if defined(__parallel)
3202 : INTEGER :: i, ierr, j, k, msglen, msgsiz
3203 76888 : INTEGER, ALLOCATABLE :: imsg(:), imsglen(:)
3204 : #endif
3205 :
3206 76888 : CALL mp_timeset(routineN, handle)
3207 :
3208 : #if defined(__parallel)
3209 76888 : msgsiz = SIZE(msg)
3210 : ! Determine size of the minimum array of integers to broadcast the string
3211 230664 : ALLOCATE (imsglen(1:msgsiz))
3212 76964888 : DO j = 1, msgsiz
3213 76964888 : imsglen(j) = LEN_TRIM(msg(j))
3214 : END DO
3215 76888 : CALL comm%bcast(imsglen, comm%source)
3216 76964888 : msglen = SUM(imsglen)
3217 : ! this is a workaround to avoid problems on the T3E
3218 : ! at the moment we have a data alignment error when trying to
3219 : ! broadcast characters on the T3E (not always!)
3220 : ! JH 19/3/99 on galileo
3221 230664 : ALLOCATE (imsg(1:msglen))
3222 76964888 : k = 0
3223 76964888 : DO j = 1, msgsiz
3224 2322754206 : DO i = 1, imsglen(j)
3225 2245789318 : k = k + 1
3226 2322677318 : imsg(k) = ICHAR(msg(j) (i:i))
3227 : END DO
3228 : END DO
3229 76888 : CALL mpi_bcast(imsg, msglen, MPI_INTEGER, comm%source, comm%handle, ierr)
3230 76888 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_bcast @ "//routineN)
3231 76964888 : msg = ""
3232 : k = 0
3233 76964888 : DO j = 1, msgsiz
3234 2322754206 : DO i = 1, imsglen(j)
3235 2245789318 : k = k + 1
3236 2322677318 : msg(j) (i:i) = CHAR(imsg(k))
3237 : END DO
3238 : END DO
3239 76888 : DEALLOCATE (imsg)
3240 76888 : DEALLOCATE (imsglen)
3241 76888 : CALL add_perf(perf_id=2, count=1, msg_size=msglen*charlen*msgsiz)
3242 : #else
3243 : MARK_USED(msg)
3244 : MARK_USED(comm)
3245 : #endif
3246 76888 : CALL mp_timestop(handle)
3247 153776 : END SUBROUTINE mp_bcast_am_src
3248 :
3249 : ! **************************************************************************************************
3250 : !> \brief Finds the location of the minimal element in a vector.
3251 : !> \param[in,out] msg Find location of maximum element among these
3252 : !> data (input).
3253 : !> \param[in] comm Message passing environment identifier
3254 : !> \par MPI mapping
3255 : !> mpi_allreduce with the MPI_MINLOC reduction function identifier
3256 : !> \par Invalid data types
3257 : !> This routine is invalid for (int_8) data!
3258 : ! **************************************************************************************************
3259 310 : SUBROUTINE mp_minloc_dv(msg, comm)
3260 : REAL(kind=real_8), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3261 : CLASS(mp_comm_type), INTENT(IN) :: comm
3262 :
3263 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_minloc_dv'
3264 :
3265 : INTEGER :: handle
3266 : #if defined(__parallel)
3267 : INTEGER :: ierr, msglen
3268 310 : REAL(kind=real_8), ALLOCATABLE :: res(:)
3269 : #endif
3270 :
3271 : IF ("d" .EQ. "l" .AND. real_8 .EQ. int_8) THEN
3272 : CPABORT("Minimal location not available with long integers @ "//routineN)
3273 : END IF
3274 310 : CALL mp_timeset(routineN, handle)
3275 :
3276 : #if defined(__parallel)
3277 310 : msglen = SIZE(msg)
3278 930 : ALLOCATE (res(1:msglen), STAT=ierr)
3279 310 : IF (ierr /= 0) &
3280 0 : CPABORT("allocate @ "//routineN)
3281 310 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2DOUBLE_PRECISION, MPI_MINLOC, comm%handle, ierr)
3282 310 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3283 930 : msg = res
3284 310 : DEALLOCATE (res)
3285 310 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*real_8_size)
3286 : #else
3287 : MARK_USED(msg)
3288 : MARK_USED(comm)
3289 : #endif
3290 310 : CALL mp_timestop(handle)
3291 310 : END SUBROUTINE mp_minloc_dv
3292 :
3293 : ! **************************************************************************************************
3294 : !> \brief Finds the location of the minimal element in a vector.
3295 : !> \param[in,out] msg Find location of maximum element among these
3296 : !> data (input).
3297 : !> \param[in] comm Message passing environment identifier
3298 : !> \par MPI mapping
3299 : !> mpi_allreduce with the MPI_MINLOC reduction function identifier
3300 : !> \par Invalid data types
3301 : !> This routine is invalid for (int_8) data!
3302 : ! **************************************************************************************************
3303 0 : SUBROUTINE mp_minloc_iv(msg, comm)
3304 : INTEGER(KIND=int_4), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3305 : CLASS(mp_comm_type), INTENT(IN) :: comm
3306 :
3307 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_minloc_iv'
3308 :
3309 : INTEGER :: handle
3310 : #if defined(__parallel)
3311 : INTEGER :: ierr, msglen
3312 0 : INTEGER(KIND=int_4), ALLOCATABLE :: res(:)
3313 : #endif
3314 :
3315 : IF ("i" .EQ. "l" .AND. int_4 .EQ. int_8) THEN
3316 : CPABORT("Minimal location not available with long integers @ "//routineN)
3317 : END IF
3318 0 : CALL mp_timeset(routineN, handle)
3319 :
3320 : #if defined(__parallel)
3321 0 : msglen = SIZE(msg)
3322 0 : ALLOCATE (res(1:msglen))
3323 0 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2INTEGER, MPI_MINLOC, comm%handle, ierr)
3324 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3325 0 : msg = res
3326 0 : DEALLOCATE (res)
3327 0 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*int_4_size)
3328 : #else
3329 : MARK_USED(msg)
3330 : MARK_USED(comm)
3331 : #endif
3332 0 : CALL mp_timestop(handle)
3333 0 : END SUBROUTINE mp_minloc_iv
3334 :
3335 : ! **************************************************************************************************
3336 : !> \brief Finds the location of the minimal element in a vector.
3337 : !> \param[in,out] msg Find location of maximum element among these
3338 : !> data (input).
3339 : !> \param[in] comm Message passing environment identifier
3340 : !> \par MPI mapping
3341 : !> mpi_allreduce with the MPI_MINLOC reduction function identifier
3342 : !> \par Invalid data types
3343 : !> This routine is invalid for (int_8) data!
3344 : ! **************************************************************************************************
3345 0 : SUBROUTINE mp_minloc_lv(msg, comm)
3346 : INTEGER(KIND=int_8), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3347 : CLASS(mp_comm_type), INTENT(IN) :: comm
3348 :
3349 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_minloc_lv'
3350 :
3351 : INTEGER :: handle
3352 : #if defined(__parallel)
3353 : INTEGER :: ierr, msglen
3354 0 : INTEGER(KIND=int_8), ALLOCATABLE :: res(:)
3355 : #endif
3356 :
3357 : IF ("l" .EQ. "l" .AND. int_8 .EQ. int_8) THEN
3358 0 : CPABORT("Minimal location not available with long integers @ "//routineN)
3359 : END IF
3360 0 : CALL mp_timeset(routineN, handle)
3361 :
3362 : #if defined(__parallel)
3363 0 : msglen = SIZE(msg)
3364 0 : ALLOCATE (res(1:msglen))
3365 0 : CALL mpi_allreduce(msg, res, msglen/2, MPI_INTEGER8, MPI_MINLOC, comm%handle, ierr)
3366 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3367 0 : msg = res
3368 0 : DEALLOCATE (res)
3369 0 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*int_8_size)
3370 : #else
3371 : MARK_USED(msg)
3372 : MARK_USED(comm)
3373 : #endif
3374 0 : CALL mp_timestop(handle)
3375 0 : END SUBROUTINE mp_minloc_lv
3376 :
3377 : ! **************************************************************************************************
3378 : !> \brief Finds the location of the minimal element in a vector.
3379 : !> \param[in,out] msg Find location of maximum element among these
3380 : !> data (input).
3381 : !> \param[in] comm Message passing environment identifier
3382 : !> \par MPI mapping
3383 : !> mpi_allreduce with the MPI_MINLOC reduction function identifier
3384 : !> \par Invalid data types
3385 : !> This routine is invalid for (int_8) data!
3386 : ! **************************************************************************************************
3387 0 : SUBROUTINE mp_minloc_rv(msg, comm)
3388 : REAL(kind=real_4), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3389 : CLASS(mp_comm_type), INTENT(IN) :: comm
3390 :
3391 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_minloc_rv'
3392 :
3393 : INTEGER :: handle
3394 : #if defined(__parallel)
3395 : INTEGER :: ierr, msglen
3396 0 : REAL(kind=real_4), ALLOCATABLE :: res(:)
3397 : #endif
3398 :
3399 : IF ("r" .EQ. "l" .AND. real_4 .EQ. int_8) THEN
3400 : CPABORT("Minimal location not available with long integers @ "//routineN)
3401 : END IF
3402 0 : CALL mp_timeset(routineN, handle)
3403 :
3404 : #if defined(__parallel)
3405 0 : msglen = SIZE(msg)
3406 0 : ALLOCATE (res(1:msglen))
3407 0 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2REAL, MPI_MINLOC, comm%handle, ierr)
3408 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3409 0 : msg = res
3410 0 : DEALLOCATE (res)
3411 0 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*real_4_size)
3412 : #else
3413 : MARK_USED(msg)
3414 : MARK_USED(comm)
3415 : #endif
3416 0 : CALL mp_timestop(handle)
3417 0 : END SUBROUTINE mp_minloc_rv
3418 :
3419 : ! **************************************************************************************************
3420 : !> \brief Finds the location of the maximal element in a vector.
3421 : !> \param[in,out] msg Find location of maximum element among these
3422 : !> data (input).
3423 : !> \param[in] comm Message passing environment identifier
3424 : !> \par MPI mapping
3425 : !> mpi_allreduce with the MPI_MAXLOC reduction function identifier
3426 : !> \par Invalid data types
3427 : !> This routine is invalid for (int_8) data!
3428 : ! **************************************************************************************************
3429 7330255 : SUBROUTINE mp_maxloc_dv(msg, comm)
3430 : REAL(kind=real_8), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3431 : CLASS(mp_comm_type), INTENT(IN) :: comm
3432 :
3433 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_maxloc_dv'
3434 :
3435 : INTEGER :: handle
3436 : #if defined(__parallel)
3437 : INTEGER :: ierr, msglen
3438 7330255 : REAL(kind=real_8), ALLOCATABLE :: res(:)
3439 : #endif
3440 :
3441 : IF ("d" .EQ. "l" .AND. real_8 .EQ. int_8) THEN
3442 : CPABORT("Maximal location not available with long integers @ "//routineN)
3443 : END IF
3444 7330255 : CALL mp_timeset(routineN, handle)
3445 :
3446 : #if defined(__parallel)
3447 7330255 : msglen = SIZE(msg)
3448 21990765 : ALLOCATE (res(1:msglen))
3449 7330255 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2DOUBLE_PRECISION, MPI_MAXLOC, comm%handle, ierr)
3450 7330255 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3451 21990765 : msg = res
3452 7330255 : DEALLOCATE (res)
3453 7330255 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*real_8_size)
3454 : #else
3455 : MARK_USED(msg)
3456 : MARK_USED(comm)
3457 : #endif
3458 7330255 : CALL mp_timestop(handle)
3459 7330255 : END SUBROUTINE mp_maxloc_dv
3460 :
3461 : ! **************************************************************************************************
3462 : !> \brief Finds the location of the maximal element in a vector.
3463 : !> \param[in,out] msg Find location of maximum element among these
3464 : !> data (input).
3465 : !> \param[in] comm Message passing environment identifier
3466 : !> \par MPI mapping
3467 : !> mpi_allreduce with the MPI_MAXLOC reduction function identifier
3468 : !> \par Invalid data types
3469 : !> This routine is invalid for (int_8) data!
3470 : ! **************************************************************************************************
3471 138 : SUBROUTINE mp_maxloc_iv(msg, comm)
3472 : INTEGER(KIND=int_4), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3473 : CLASS(mp_comm_type), INTENT(IN) :: comm
3474 :
3475 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_maxloc_iv'
3476 :
3477 : INTEGER :: handle
3478 : #if defined(__parallel)
3479 : INTEGER :: ierr, msglen
3480 138 : INTEGER(KIND=int_4), ALLOCATABLE :: res(:)
3481 : #endif
3482 :
3483 : IF ("i" .EQ. "l" .AND. int_4 .EQ. int_8) THEN
3484 : CPABORT("Maximal location not available with long integers @ "//routineN)
3485 : END IF
3486 138 : CALL mp_timeset(routineN, handle)
3487 :
3488 : #if defined(__parallel)
3489 138 : msglen = SIZE(msg)
3490 414 : ALLOCATE (res(1:msglen))
3491 138 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2INTEGER, MPI_MAXLOC, comm%handle, ierr)
3492 138 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3493 414 : msg = res
3494 138 : DEALLOCATE (res)
3495 138 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*int_4_size)
3496 : #else
3497 : MARK_USED(msg)
3498 : MARK_USED(comm)
3499 : #endif
3500 138 : CALL mp_timestop(handle)
3501 138 : END SUBROUTINE mp_maxloc_iv
3502 :
3503 : ! **************************************************************************************************
3504 : !> \brief Finds the location of the maximal element in a vector.
3505 : !> \param[in,out] msg Find location of maximum element among these
3506 : !> data (input).
3507 : !> \param[in] comm Message passing environment identifier
3508 : !> \par MPI mapping
3509 : !> mpi_allreduce with the MPI_MAXLOC reduction function identifier
3510 : !> \par Invalid data types
3511 : !> This routine is invalid for (int_8) data!
3512 : ! **************************************************************************************************
3513 0 : SUBROUTINE mp_maxloc_lv(msg, comm)
3514 : INTEGER(KIND=int_8), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3515 : CLASS(mp_comm_type), INTENT(IN) :: comm
3516 :
3517 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_maxloc_lv'
3518 :
3519 : INTEGER :: handle
3520 : #if defined(__parallel)
3521 : INTEGER :: ierr, msglen
3522 0 : INTEGER(KIND=int_8), ALLOCATABLE :: res(:)
3523 : #endif
3524 :
3525 : IF ("l" .EQ. "l" .AND. int_8 .EQ. int_8) THEN
3526 0 : CPABORT("Maximal location not available with long integers @ "//routineN)
3527 : END IF
3528 0 : CALL mp_timeset(routineN, handle)
3529 :
3530 : #if defined(__parallel)
3531 0 : msglen = SIZE(msg)
3532 0 : ALLOCATE (res(1:msglen))
3533 0 : CALL mpi_allreduce(msg, res, msglen/2, MPI_INTEGER8, MPI_MAXLOC, comm%handle, ierr)
3534 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3535 0 : msg = res
3536 0 : DEALLOCATE (res)
3537 0 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*int_8_size)
3538 : #else
3539 : MARK_USED(msg)
3540 : MARK_USED(comm)
3541 : #endif
3542 0 : CALL mp_timestop(handle)
3543 0 : END SUBROUTINE mp_maxloc_lv
3544 :
3545 : ! **************************************************************************************************
3546 : !> \brief Finds the location of the maximal element in a vector.
3547 : !> \param[in,out] msg Find location of maximum element among these
3548 : !> data (input).
3549 : !> \param[in] comm Message passing environment identifier
3550 : !> \par MPI mapping
3551 : !> mpi_allreduce with the MPI_MAXLOC reduction function identifier
3552 : !> \par Invalid data types
3553 : !> This routine is invalid for (int_8) data!
3554 : ! **************************************************************************************************
3555 0 : SUBROUTINE mp_maxloc_rv(msg, comm)
3556 : REAL(kind=real_4), CONTIGUOUS, INTENT(INOUT) :: msg(:)
3557 : CLASS(mp_comm_type), INTENT(IN) :: comm
3558 :
3559 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_maxloc_rv'
3560 :
3561 : INTEGER :: handle
3562 : #if defined(__parallel)
3563 : INTEGER :: ierr, msglen
3564 0 : REAL(kind=real_4), ALLOCATABLE :: res(:)
3565 : #endif
3566 :
3567 : IF ("r" .EQ. "l" .AND. real_4 .EQ. int_8) THEN
3568 : CPABORT("Maximal location not available with long integers @ "//routineN)
3569 : END IF
3570 0 : CALL mp_timeset(routineN, handle)
3571 :
3572 : #if defined(__parallel)
3573 0 : msglen = SIZE(msg)
3574 0 : ALLOCATE (res(1:msglen))
3575 0 : CALL mpi_allreduce(msg, res, msglen/2, MPI_2REAL, MPI_MAXLOC, comm%handle, ierr)
3576 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3577 0 : msg = res
3578 0 : DEALLOCATE (res)
3579 0 : CALL add_perf(perf_id=3, count=1, msg_size=msglen*real_4_size)
3580 : #else
3581 : MARK_USED(msg)
3582 : MARK_USED(comm)
3583 : #endif
3584 0 : CALL mp_timestop(handle)
3585 0 : END SUBROUTINE mp_maxloc_rv
3586 :
3587 : ! **************************************************************************************************
3588 : !> \brief Logical OR reduction
3589 : !> \param[in,out] msg Datum to perform inclusive disjunction (input)
3590 : !> and resultant inclusive disjunction (output)
3591 : !> \param[in] comm Message passing environment identifier
3592 : !> \par MPI mapping
3593 : !> mpi_allreduce
3594 : ! **************************************************************************************************
3595 60418 : SUBROUTINE mp_sum_b(msg, comm)
3596 : LOGICAL, INTENT(INOUT) :: msg
3597 : CLASS(mp_comm_type), INTENT(IN) :: comm
3598 :
3599 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_sum_b'
3600 :
3601 : INTEGER :: handle
3602 : #if defined(__parallel)
3603 : INTEGER :: ierr, msglen
3604 : #endif
3605 :
3606 60418 : CALL mp_timeset(routineN, handle)
3607 : #if defined(__parallel)
3608 60418 : msglen = 1
3609 60418 : CALL mpi_allreduce(MPI_IN_PLACE, msg, msglen, MPI_LOGICAL, MPI_LOR, comm%handle, ierr)
3610 60418 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3611 : #else
3612 : MARK_USED(msg)
3613 : MARK_USED(comm)
3614 : #endif
3615 60418 : CALL mp_timestop(handle)
3616 60418 : END SUBROUTINE mp_sum_b
3617 :
3618 : ! **************************************************************************************************
3619 : !> \brief Logical OR reduction
3620 : !> \param[in,out] msg Datum to perform inclusive disjunction (input)
3621 : !> and resultant inclusive disjunction (output)
3622 : !> \param[in] comm Message passing environment identifier
3623 : !> \par MPI mapping
3624 : !> mpi_allreduce
3625 : ! **************************************************************************************************
3626 0 : SUBROUTINE mp_sum_bv(msg, comm)
3627 : LOGICAL, DIMENSION(:), CONTIGUOUS, INTENT(INOUT) :: msg
3628 : CLASS(mp_comm_type), INTENT(IN) :: comm
3629 :
3630 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_sum_bv'
3631 :
3632 : INTEGER :: handle
3633 : #if defined(__parallel)
3634 : INTEGER :: ierr, msglen
3635 : #endif
3636 :
3637 0 : CALL mp_timeset(routineN, handle)
3638 : #if defined(__parallel)
3639 0 : msglen = SIZE(msg)
3640 0 : IF (msglen .GT. 0) THEN
3641 0 : CALL mpi_allreduce(MPI_IN_PLACE, msg, msglen, MPI_LOGICAL, MPI_LOR, comm%handle, ierr)
3642 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3643 : END IF
3644 : #else
3645 : MARK_USED(msg)
3646 : MARK_USED(comm)
3647 : #endif
3648 0 : CALL mp_timestop(handle)
3649 0 : END SUBROUTINE mp_sum_bv
3650 :
3651 : ! **************************************************************************************************
3652 : !> \brief Logical OR reduction
3653 : !> \param[in,out] msg Datum to perform inclusive disjunction (input)
3654 : !> and resultant inclusive disjunction (output)
3655 : !> \param[in] comm Message passing environment identifier
3656 : !> \param request ...
3657 : !> \par MPI mapping
3658 : !> mpi_allreduce
3659 : ! **************************************************************************************************
3660 0 : SUBROUTINE mp_isum_bv(msg, comm, request)
3661 : LOGICAL, DIMENSION(:), INTENT(INOUT) :: msg
3662 : CLASS(mp_comm_type), INTENT(IN) :: comm
3663 : TYPE(mp_request_type), INTENT(INOUT) :: request
3664 :
3665 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_isum_bv'
3666 :
3667 : INTEGER :: handle
3668 : #if defined(__parallel)
3669 : INTEGER :: ierr, msglen
3670 : #endif
3671 :
3672 0 : CALL mp_timeset(routineN, handle)
3673 : #if defined(__parallel)
3674 0 : msglen = SIZE(msg)
3675 : #if !defined(__GNUC__) || __GNUC__ >= 9
3676 0 : CPASSERT(IS_CONTIGUOUS(msg))
3677 : #endif
3678 :
3679 0 : IF (msglen .GT. 0) THEN
3680 0 : CALL mpi_iallreduce(MPI_IN_PLACE, msg, msglen, MPI_LOGICAL, MPI_LOR, comm%handle, request%handle, ierr)
3681 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_allreduce @ "//routineN)
3682 : ELSE
3683 0 : request = mp_request_null
3684 : END IF
3685 : #else
3686 : MARK_USED(msg)
3687 : MARK_USED(comm)
3688 : request = mp_request_null
3689 : #endif
3690 0 : CALL mp_timestop(handle)
3691 0 : END SUBROUTINE mp_isum_bv
3692 :
3693 : ! **************************************************************************************************
3694 : !> \brief Get Version of the MPI Library (MPI 3)
3695 : !> \param[out] version Version of the library,
3696 : !> declared as CHARACTER(LEN=mp_max_library_version_string)
3697 : !> \param[out] resultlen Length (in printable characters) of
3698 : !> the result returned in version (integer)
3699 : ! **************************************************************************************************
3700 0 : SUBROUTINE mp_get_library_version(version, resultlen)
3701 : CHARACTER(len=*), INTENT(OUT) :: version
3702 : INTEGER, INTENT(OUT) :: resultlen
3703 :
3704 : #if defined(__parallel)
3705 : INTEGER :: ierr
3706 : #endif
3707 :
3708 0 : version = ''
3709 :
3710 : #if defined(__parallel)
3711 0 : ierr = 0
3712 0 : CALL mpi_get_library_version(version, resultlen, ierr)
3713 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_get_library_version @ mp_get_library_version")
3714 : #else
3715 : resultlen = 0
3716 : #endif
3717 0 : END SUBROUTINE mp_get_library_version
3718 :
3719 : ! **************************************************************************************************
3720 : !> \brief Opens a file
3721 : !> \param[in] groupid message passing environment identifier
3722 : !> \param[out] fh file handle (file storage unit)
3723 : !> \param[in] filepath path to the file
3724 : !> \param amode_status access mode
3725 : !> \param info ...
3726 : !> \par MPI-I/O mapping mpi_file_open
3727 : !> \par STREAM-I/O mapping OPEN
3728 : !>
3729 : !> \param[in](optional) info info object
3730 : !> \par History
3731 : !> 11.2012 created [Hossein Bani-Hashemian]
3732 : ! **************************************************************************************************
3733 1888 : SUBROUTINE mp_file_open(groupid, fh, filepath, amode_status, info)
3734 : CLASS(mp_comm_type), INTENT(IN) :: groupid
3735 : CLASS(mp_file_type), INTENT(OUT) :: fh
3736 : CHARACTER(len=*), INTENT(IN) :: filepath
3737 : INTEGER, INTENT(IN) :: amode_status
3738 : TYPE(mp_info_type), INTENT(IN), OPTIONAL :: info
3739 :
3740 : #if defined(__parallel)
3741 : INTEGER :: ierr
3742 : MPI_INFO_TYPE :: my_info
3743 : #else
3744 : CHARACTER(LEN=10) :: fstatus, fposition
3745 : INTEGER :: amode, handle, istat
3746 : LOGICAL :: exists, is_open
3747 : #endif
3748 :
3749 : #if defined(__parallel)
3750 1888 : ierr = 0
3751 1888 : my_info = mpi_info_null
3752 1888 : IF (PRESENT(info)) my_info = info%handle
3753 1888 : CALL mpi_file_open(groupid%handle, filepath, amode_status, my_info, fh%handle, ierr)
3754 1888 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
3755 1888 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ mp_file_open")
3756 : #else
3757 : MARK_USED(groupid)
3758 : MARK_USED(info)
3759 : amode = amode_status
3760 : IF (amode .GT. file_amode_append) THEN
3761 : fposition = "APPEND"
3762 : amode = amode - file_amode_append
3763 : ELSE
3764 : fposition = "REWIND"
3765 : END IF
3766 : IF ((amode .EQ. file_amode_create) .OR. &
3767 : (amode .EQ. file_amode_create + file_amode_wronly) .OR. &
3768 : (amode .EQ. file_amode_create + file_amode_wronly + file_amode_excl)) THEN
3769 : fstatus = "UNKNOWN"
3770 : ELSE
3771 : fstatus = "OLD"
3772 : END IF
3773 : ! Get a new unit number
3774 : DO handle = 1, 999
3775 : INQUIRE (UNIT=handle, EXIST=exists, OPENED=is_open, IOSTAT=istat)
3776 : IF (exists .AND. (.NOT. is_open) .AND. (istat == 0)) EXIT
3777 : END DO
3778 : OPEN (UNIT=handle, FILE=filepath, STATUS=fstatus, ACCESS="STREAM", POSITION=fposition)
3779 : fh%handle = handle
3780 : #endif
3781 1888 : END SUBROUTINE mp_file_open
3782 :
3783 : ! **************************************************************************************************
3784 : !> \brief Deletes a file. Auxiliary routine to emulate 'replace' action for mp_file_open.
3785 : !> Only the master processor should call this routine.
3786 : !> \param[in] filepath path to the file
3787 : !> \param[in](optional) info info object
3788 : !> \par History
3789 : !> 11.2017 created [Nico Holmberg]
3790 : ! **************************************************************************************************
3791 162 : SUBROUTINE mp_file_delete(filepath, info)
3792 : CHARACTER(len=*), INTENT(IN) :: filepath
3793 : TYPE(mp_info_type), INTENT(IN), OPTIONAL :: info
3794 :
3795 : #if defined(__parallel)
3796 : INTEGER :: ierr
3797 : MPI_INFO_TYPE :: my_info
3798 : LOGICAL :: exists
3799 :
3800 162 : ierr = 0
3801 162 : my_info = mpi_info_null
3802 162 : IF (PRESENT(info)) my_info = info%handle
3803 162 : INQUIRE (FILE=filepath, EXIST=exists)
3804 162 : IF (exists) CALL mpi_file_delete(filepath, my_info, ierr)
3805 162 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ mp_file_delete")
3806 : #else
3807 : MARK_USED(filepath)
3808 : MARK_USED(info)
3809 : ! Explicit file delete not necessary, handled by subsequent call to open_file with action 'replace'
3810 : #endif
3811 :
3812 162 : END SUBROUTINE mp_file_delete
3813 :
3814 : ! **************************************************************************************************
3815 : !> \brief Closes a file
3816 : !> \param[in] fh file handle (file storage unit)
3817 : !> \par MPI-I/O mapping mpi_file_close
3818 : !> \par STREAM-I/O mapping CLOSE
3819 : !>
3820 : !> \par History
3821 : !> 11.2012 created [Hossein Bani-Hashemian]
3822 : ! **************************************************************************************************
3823 1888 : SUBROUTINE mp_file_close(fh)
3824 : CLASS(mp_file_type), INTENT(INOUT) :: fh
3825 :
3826 : #if defined(__parallel)
3827 : INTEGER :: ierr
3828 :
3829 1888 : ierr = 0
3830 1888 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
3831 1888 : CALL mpi_file_close(fh%handle, ierr)
3832 1888 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ mp_file_close")
3833 : #else
3834 : CLOSE (fh%handle)
3835 : fh%handle = mp_file_null_handle
3836 : #endif
3837 1888 : END SUBROUTINE mp_file_close
3838 :
3839 0 : SUBROUTINE mp_file_assign(fh_new, fh_old)
3840 : CLASS(mp_file_type), INTENT(OUT) :: fh_new
3841 : CLASS(mp_file_type), INTENT(IN) :: fh_old
3842 :
3843 0 : fh_new%handle = fh_old%handle
3844 :
3845 0 : END SUBROUTINE
3846 :
3847 : ! **************************************************************************************************
3848 : !> \brief Returns the file size
3849 : !> \param[in] fh file handle (file storage unit)
3850 : !> \param[out] file_size the file size
3851 : !> \par MPI-I/O mapping mpi_file_get_size
3852 : !> \par STREAM-I/O mapping INQUIRE
3853 : !>
3854 : !> \par History
3855 : !> 12.2012 created [Hossein Bani-Hashemian]
3856 : ! **************************************************************************************************
3857 0 : SUBROUTINE mp_file_get_size(fh, file_size)
3858 : CLASS(mp_file_type), INTENT(IN) :: fh
3859 : INTEGER(kind=file_offset), INTENT(OUT) :: file_size
3860 :
3861 : #if defined(__parallel)
3862 : INTEGER :: ierr
3863 : #endif
3864 :
3865 : #if defined(__parallel)
3866 0 : ierr = 0
3867 0 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
3868 0 : CALL mpi_file_get_size(fh%handle, file_size, ierr)
3869 0 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ mp_file_get_size")
3870 : #else
3871 : INQUIRE (UNIT=fh%handle, SIZE=file_size)
3872 : #endif
3873 0 : END SUBROUTINE mp_file_get_size
3874 :
3875 : ! **************************************************************************************************
3876 : !> \brief Returns the file position
3877 : !> \param[in] fh file handle (file storage unit)
3878 : !> \param[out] file_size the file position
3879 : !> \par MPI-I/O mapping mpi_file_get_position
3880 : !> \par STREAM-I/O mapping INQUIRE
3881 : !>
3882 : !> \par History
3883 : !> 11.2017 created [Nico Holmberg]
3884 : ! **************************************************************************************************
3885 1850 : SUBROUTINE mp_file_get_position(fh, pos)
3886 : CLASS(mp_file_type), INTENT(IN) :: fh
3887 : INTEGER(kind=file_offset), INTENT(OUT) :: pos
3888 :
3889 : #if defined(__parallel)
3890 : INTEGER :: ierr
3891 : #endif
3892 :
3893 : #if defined(__parallel)
3894 1850 : ierr = 0
3895 1850 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
3896 1850 : CALL mpi_file_get_position(fh%handle, pos, ierr)
3897 1850 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ mp_file_get_position")
3898 : #else
3899 : INQUIRE (UNIT=fh%handle, POS=pos)
3900 : #endif
3901 1850 : END SUBROUTINE mp_file_get_position
3902 :
3903 : ! **************************************************************************************************
3904 : !> \brief (parallel) Blocking individual file write using explicit offsets
3905 : !> (serial) Unformatted stream write
3906 : !> \param[in] fh file handle (file storage unit)
3907 : !> \param[in] offset file offset (position)
3908 : !> \param[in] msg data to be written to the file
3909 : !> \param msglen ...
3910 : !> \par MPI-I/O mapping mpi_file_write_at
3911 : !> \par STREAM-I/O mapping WRITE
3912 : !> \param[in](optional) msglen number of the elements of data
3913 : ! **************************************************************************************************
3914 0 : SUBROUTINE mp_file_write_at_chv(fh, offset, msg, msglen)
3915 : CHARACTER, CONTIGUOUS, INTENT(IN) :: msg(:)
3916 : CLASS(mp_file_type), INTENT(IN) :: fh
3917 : INTEGER, INTENT(IN), OPTIONAL :: msglen
3918 : INTEGER(kind=file_offset), INTENT(IN) :: offset
3919 :
3920 : #if defined(__parallel)
3921 : INTEGER :: ierr, msg_len
3922 : #endif
3923 :
3924 : #if defined(__parallel)
3925 0 : msg_len = SIZE(msg)
3926 0 : IF (PRESENT(msglen)) msg_len = msglen
3927 0 : CALL MPI_FILE_WRITE_AT(fh%handle, offset, msg, msg_len, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
3928 0 : IF (ierr .NE. 0) &
3929 0 : CPABORT("mpi_file_write_at_chv @ mp_file_write_at_chv")
3930 : #else
3931 : MARK_USED(msglen)
3932 : WRITE (UNIT=fh%handle, POS=offset + 1) msg
3933 : #endif
3934 0 : END SUBROUTINE mp_file_write_at_chv
3935 :
3936 : ! **************************************************************************************************
3937 : !> \brief ...
3938 : !> \param fh ...
3939 : !> \param offset ...
3940 : !> \param msg ...
3941 : ! **************************************************************************************************
3942 8692 : SUBROUTINE mp_file_write_at_ch(fh, offset, msg)
3943 : CHARACTER(LEN=*), INTENT(IN) :: msg
3944 : CLASS(mp_file_type), INTENT(IN) :: fh
3945 : INTEGER(kind=file_offset), INTENT(IN) :: offset
3946 :
3947 : #if defined(__parallel)
3948 : INTEGER :: ierr
3949 : #endif
3950 :
3951 : #if defined(__parallel)
3952 8692 : CALL MPI_FILE_WRITE_AT(fh%handle, offset, msg, LEN(msg), MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
3953 8692 : IF (ierr .NE. 0) &
3954 0 : CPABORT("mpi_file_write_at_ch @ mp_file_write_at_ch")
3955 : #else
3956 : WRITE (UNIT=fh%handle, POS=offset + 1) msg
3957 : #endif
3958 8692 : END SUBROUTINE mp_file_write_at_ch
3959 :
3960 : ! **************************************************************************************************
3961 : !> \brief (parallel) Blocking collective file write using explicit offsets
3962 : !> (serial) Unformatted stream write
3963 : !> \param fh ...
3964 : !> \param offset ...
3965 : !> \param msg ...
3966 : !> \param msglen ...
3967 : !> \par MPI-I/O mapping mpi_file_write_at_all
3968 : !> \par STREAM-I/O mapping WRITE
3969 : ! **************************************************************************************************
3970 0 : SUBROUTINE mp_file_write_at_all_chv(fh, offset, msg, msglen)
3971 : CHARACTER, CONTIGUOUS, INTENT(IN) :: msg(:)
3972 : CLASS(mp_file_type), INTENT(IN) :: fh
3973 : INTEGER, INTENT(IN), OPTIONAL :: msglen
3974 : INTEGER(kind=file_offset), INTENT(IN) :: offset
3975 :
3976 : #if defined(__parallel)
3977 : INTEGER :: ierr, msg_len
3978 : #endif
3979 :
3980 : #if defined(__parallel)
3981 0 : msg_len = SIZE(msg)
3982 0 : IF (PRESENT(msglen)) msg_len = msglen
3983 0 : CALL MPI_FILE_WRITE_AT_ALL(fh%handle, offset, msg, msg_len, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
3984 0 : IF (ierr .NE. 0) &
3985 0 : CPABORT("mpi_file_write_at_all_chv @ mp_file_write_at_all_chv")
3986 : #else
3987 : MARK_USED(msglen)
3988 : WRITE (UNIT=fh%handle, POS=offset + 1) msg
3989 : #endif
3990 0 : END SUBROUTINE mp_file_write_at_all_chv
3991 :
3992 : ! **************************************************************************************************
3993 : !> \brief wrapper to MPI_File_write_at_all
3994 : !> \param fh ...
3995 : !> \param offset ...
3996 : !> \param msg ...
3997 : ! **************************************************************************************************
3998 0 : SUBROUTINE mp_file_write_at_all_ch(fh, offset, msg)
3999 : CHARACTER(LEN=*), INTENT(IN) :: msg
4000 : CLASS(mp_file_type), INTENT(IN) :: fh
4001 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4002 :
4003 : #if defined(__parallel)
4004 : INTEGER :: ierr
4005 : #endif
4006 :
4007 : #if defined(__parallel)
4008 0 : CALL MPI_FILE_WRITE_AT_ALL(fh%handle, offset, msg, LEN(msg), MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4009 0 : IF (ierr .NE. 0) &
4010 0 : CPABORT("mpi_file_write_at_all_ch @ mp_file_write_at_all_ch")
4011 : #else
4012 : WRITE (UNIT=fh%handle, POS=offset + 1) msg
4013 : #endif
4014 0 : END SUBROUTINE mp_file_write_at_all_ch
4015 :
4016 : ! **************************************************************************************************
4017 : !> \brief (parallel) Blocking individual file read using explicit offsets
4018 : !> (serial) Unformatted stream read
4019 : !> \param[in] fh file handle (file storage unit)
4020 : !> \param[in] offset file offset (position)
4021 : !> \param[out] msg data to be read from the file
4022 : !> \param msglen ...
4023 : !> \par MPI-I/O mapping mpi_file_read_at
4024 : !> \par STREAM-I/O mapping READ
4025 : !> \param[in](optional) msglen number of elements of data
4026 : ! **************************************************************************************************
4027 0 : SUBROUTINE mp_file_read_at_chv(fh, offset, msg, msglen)
4028 : CHARACTER, CONTIGUOUS, INTENT(OUT) :: msg(:)
4029 : CLASS(mp_file_type), INTENT(IN) :: fh
4030 : INTEGER, INTENT(IN), OPTIONAL :: msglen
4031 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4032 :
4033 : #if defined(__parallel)
4034 : INTEGER :: ierr, msg_len
4035 : #endif
4036 :
4037 : #if defined(__parallel)
4038 0 : msg_len = SIZE(msg)
4039 0 : IF (PRESENT(msglen)) msg_len = msglen
4040 0 : CALL MPI_FILE_READ_AT(fh%handle, offset, msg, msg_len, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4041 0 : IF (ierr .NE. 0) &
4042 0 : CPABORT("mpi_file_read_at_chv @ mp_file_read_at_chv")
4043 : #else
4044 : MARK_USED(msglen)
4045 : READ (UNIT=fh%handle, POS=offset + 1) msg
4046 : #endif
4047 0 : END SUBROUTINE mp_file_read_at_chv
4048 :
4049 : ! **************************************************************************************************
4050 : !> \brief wrapper to MPI_File_read_at
4051 : !> \param fh ...
4052 : !> \param offset ...
4053 : !> \param msg ...
4054 : ! **************************************************************************************************
4055 0 : SUBROUTINE mp_file_read_at_ch(fh, offset, msg)
4056 : CHARACTER(LEN=*), INTENT(OUT) :: msg
4057 : CLASS(mp_file_type), INTENT(IN) :: fh
4058 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4059 :
4060 : #if defined(__parallel)
4061 : INTEGER :: ierr
4062 : #endif
4063 :
4064 : #if defined(__parallel)
4065 0 : CALL MPI_FILE_READ_AT(fh%handle, offset, msg, LEN(msg), MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4066 0 : IF (ierr .NE. 0) &
4067 0 : CPABORT("mpi_file_read_at_ch @ mp_file_read_at_ch")
4068 : #else
4069 : READ (UNIT=fh%handle, POS=offset + 1) msg
4070 : #endif
4071 0 : END SUBROUTINE mp_file_read_at_ch
4072 :
4073 : ! **************************************************************************************************
4074 : !> \brief (parallel) Blocking collective file read using explicit offsets
4075 : !> (serial) Unformatted stream read
4076 : !> \param fh ...
4077 : !> \param offset ...
4078 : !> \param msg ...
4079 : !> \param msglen ...
4080 : !> \par MPI-I/O mapping mpi_file_read_at_all
4081 : !> \par STREAM-I/O mapping READ
4082 : ! **************************************************************************************************
4083 0 : SUBROUTINE mp_file_read_at_all_chv(fh, offset, msg, msglen)
4084 : CHARACTER, INTENT(OUT) :: msg(:)
4085 : CLASS(mp_file_type), INTENT(IN) :: fh
4086 : INTEGER, INTENT(IN), OPTIONAL :: msglen
4087 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4088 :
4089 : #if defined(__parallel)
4090 : INTEGER :: ierr, msg_len
4091 : #endif
4092 :
4093 : #if defined(__parallel)
4094 0 : msg_len = SIZE(msg)
4095 0 : IF (PRESENT(msglen)) msg_len = msglen
4096 0 : CALL MPI_FILE_READ_AT_ALL(fh%handle, offset, msg, msg_len, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4097 0 : IF (ierr .NE. 0) &
4098 0 : CPABORT("mpi_file_read_at_all_chv @ mp_file_read_at_all_chv")
4099 : #else
4100 : MARK_USED(msglen)
4101 : READ (UNIT=fh%handle, POS=offset + 1) msg
4102 : #endif
4103 0 : END SUBROUTINE mp_file_read_at_all_chv
4104 :
4105 : ! **************************************************************************************************
4106 : !> \brief wrapper to MPI_File_read_at_all
4107 : !> \param fh ...
4108 : !> \param offset ...
4109 : !> \param msg ...
4110 : ! **************************************************************************************************
4111 0 : SUBROUTINE mp_file_read_at_all_ch(fh, offset, msg)
4112 : CHARACTER(LEN=*), INTENT(OUT) :: msg
4113 : CLASS(mp_file_type), INTENT(IN) :: fh
4114 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4115 :
4116 : #if defined(__parallel)
4117 : INTEGER :: ierr
4118 : #endif
4119 :
4120 : #if defined(__parallel)
4121 0 : CALL MPI_FILE_READ_AT_ALL(fh%handle, offset, msg, LEN(msg), MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4122 0 : IF (ierr .NE. 0) &
4123 0 : CPABORT("mpi_file_read_at_all_ch @ mp_file_read_at_all_ch")
4124 : #else
4125 : READ (UNIT=fh%handle, POS=offset + 1) msg
4126 : #endif
4127 0 : END SUBROUTINE mp_file_read_at_all_ch
4128 :
4129 : ! **************************************************************************************************
4130 : !> \brief Returns the size of a data type in bytes
4131 : !> \param[in] type_descriptor data type
4132 : !> \param[out] type_size size of the data type
4133 : !> \par MPI mapping
4134 : !> mpi_type_size
4135 : !>
4136 : ! **************************************************************************************************
4137 0 : SUBROUTINE mp_type_size(type_descriptor, type_size)
4138 : TYPE(mp_type_descriptor_type), INTENT(IN) :: type_descriptor
4139 : INTEGER, INTENT(OUT) :: type_size
4140 :
4141 : #if defined(__parallel)
4142 : INTEGER :: ierr
4143 :
4144 0 : ierr = 0
4145 0 : CALL MPI_TYPE_SIZE(type_descriptor%type_handle, type_size, ierr)
4146 0 : IF (ierr .NE. 0) &
4147 0 : CPABORT("mpi_type_size failed @ mp_type_size")
4148 : #else
4149 : SELECT CASE (type_descriptor%type_handle)
4150 : CASE (1)
4151 : type_size = real_4_size
4152 : CASE (3)
4153 : type_size = real_8_size
4154 : CASE (5)
4155 : type_size = 2*real_4_size
4156 : CASE (7)
4157 : type_size = 2*real_8_size
4158 : END SELECT
4159 : #endif
4160 0 : END SUBROUTINE mp_type_size
4161 :
4162 : ! **************************************************************************************************
4163 : !> \brief wrapper to MPI_Type_create_struct
4164 : !> \param subtypes ...
4165 : !> \param vector_descriptor ...
4166 : !> \param index_descriptor ...
4167 : !> \return ...
4168 : ! **************************************************************************************************
4169 0 : FUNCTION mp_type_make_struct(subtypes, &
4170 : vector_descriptor, index_descriptor) &
4171 : RESULT(type_descriptor)
4172 : TYPE(mp_type_descriptor_type), &
4173 : DIMENSION(:), INTENT(IN) :: subtypes
4174 : INTEGER, DIMENSION(2), INTENT(IN), &
4175 : OPTIONAL :: vector_descriptor
4176 : TYPE(mp_indexing_meta_type), &
4177 : INTENT(IN), OPTIONAL :: index_descriptor
4178 : TYPE(mp_type_descriptor_type) :: type_descriptor
4179 :
4180 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_type_make_struct'
4181 :
4182 : INTEGER :: i, n
4183 0 : INTEGER, ALLOCATABLE, DIMENSION(:) :: lengths
4184 : #if defined(__parallel)
4185 : INTEGER :: ierr
4186 : INTEGER(kind=mpi_address_kind), &
4187 0 : ALLOCATABLE, DIMENSION(:) :: displacements
4188 : #endif
4189 0 : MPI_DATA_TYPE, ALLOCATABLE, DIMENSION(:) :: old_types
4190 :
4191 0 : n = SIZE(subtypes)
4192 0 : type_descriptor%length = 1
4193 : #if defined(__parallel)
4194 0 : ierr = 0
4195 0 : CALL mpi_get_address(MPI_BOTTOM, type_descriptor%base, ierr)
4196 0 : IF (ierr /= 0) &
4197 0 : CPABORT("MPI_get_address @ "//routineN)
4198 0 : ALLOCATE (displacements(n))
4199 : #endif
4200 0 : type_descriptor%vector_descriptor(1:2) = 1
4201 0 : type_descriptor%has_indexing = .FALSE.
4202 0 : ALLOCATE (type_descriptor%subtype(n))
4203 0 : type_descriptor%subtype(:) = subtypes(:)
4204 0 : ALLOCATE (lengths(n), old_types(n))
4205 0 : DO i = 1, SIZE(subtypes)
4206 : #if defined(__parallel)
4207 0 : displacements(i) = subtypes(i)%base
4208 : #endif
4209 0 : old_types(i) = subtypes(i)%type_handle
4210 0 : lengths(i) = subtypes(i)%length
4211 : END DO
4212 : #if defined(__parallel)
4213 : CALL MPI_Type_create_struct(n, &
4214 : lengths, displacements, old_types, &
4215 0 : type_descriptor%type_handle, ierr)
4216 0 : IF (ierr /= 0) &
4217 0 : CPABORT("MPI_Type_create_struct @ "//routineN)
4218 0 : CALL MPI_Type_commit(type_descriptor%type_handle, ierr)
4219 0 : IF (ierr /= 0) &
4220 0 : CPABORT("MPI_Type_commit @ "//routineN)
4221 : #endif
4222 0 : IF (PRESENT(vector_descriptor) .OR. PRESENT(index_descriptor)) THEN
4223 0 : CPABORT(routineN//" Vectors and indices NYI")
4224 : END IF
4225 0 : END FUNCTION mp_type_make_struct
4226 :
4227 : ! **************************************************************************************************
4228 : !> \brief wrapper to MPI_Type_free
4229 : !> \param type_descriptor ...
4230 : ! **************************************************************************************************
4231 0 : RECURSIVE SUBROUTINE mp_type_free_m(type_descriptor)
4232 : TYPE(mp_type_descriptor_type), INTENT(inout) :: type_descriptor
4233 :
4234 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_type_free_m'
4235 :
4236 : INTEGER :: handle, i
4237 : #if defined(__parallel)
4238 : INTEGER :: ierr
4239 : #endif
4240 :
4241 0 : CALL mp_timeset(routineN, handle)
4242 :
4243 : ! If the subtype is associated, then it's a user-defined data type.
4244 :
4245 0 : IF (ASSOCIATED(type_descriptor%subtype)) THEN
4246 0 : DO i = 1, SIZE(type_descriptor%subtype)
4247 0 : CALL mp_type_free_m(type_descriptor%subtype(i))
4248 : END DO
4249 0 : DEALLOCATE (type_descriptor%subtype)
4250 : END IF
4251 : #if defined(__parallel)
4252 0 : ierr = 0
4253 0 : CALL MPI_Type_free(type_descriptor%type_handle, ierr)
4254 0 : IF (ierr /= 0) &
4255 0 : CPABORT("MPI_Type_free @ "//routineN)
4256 : #endif
4257 :
4258 0 : CALL mp_timestop(handle)
4259 :
4260 0 : END SUBROUTINE mp_type_free_m
4261 :
4262 : ! **************************************************************************************************
4263 : !> \brief ...
4264 : !> \param type_descriptors ...
4265 : ! **************************************************************************************************
4266 0 : SUBROUTINE mp_type_free_v(type_descriptors)
4267 : TYPE(mp_type_descriptor_type), DIMENSION(:), &
4268 : INTENT(inout) :: type_descriptors
4269 :
4270 : INTEGER :: i
4271 :
4272 0 : DO i = 1, SIZE(type_descriptors)
4273 0 : CALL mp_type_free(type_descriptors(i))
4274 : END DO
4275 :
4276 0 : END SUBROUTINE mp_type_free_v
4277 :
4278 : ! **************************************************************************************************
4279 : !> \brief Creates an indexed MPI type for arrays of strings using bytes for spacing (hindexed type)
4280 : !> \param count number of array blocks to read
4281 : !> \param lengths lengths of each array block
4282 : !> \param displs byte offsets for array blocks
4283 : !> \return container holding the created type
4284 : !> \author Nico Holmberg [05.2017]
4285 : ! **************************************************************************************************
4286 1888 : FUNCTION mp_file_type_hindexed_make_chv(count, lengths, displs) &
4287 : RESULT(type_descriptor)
4288 : INTEGER, INTENT(IN) :: count
4289 : INTEGER, DIMENSION(1:count), &
4290 : INTENT(IN), TARGET :: lengths
4291 : INTEGER(kind=file_offset), &
4292 : DIMENSION(1:count), INTENT(in), TARGET :: displs
4293 : TYPE(mp_file_descriptor_type) :: type_descriptor
4294 :
4295 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_file_hindexed_make_chv'
4296 :
4297 : INTEGER :: ierr, handle
4298 :
4299 1888 : ierr = 0
4300 1888 : CALL mp_timeset(routineN, handle)
4301 :
4302 : #if defined(__parallel)
4303 : CALL MPI_Type_create_hindexed(count, lengths, INT(displs, KIND=address_kind), MPI_CHARACTER, &
4304 406396 : type_descriptor%type_handle, ierr)
4305 1888 : IF (ierr /= 0) &
4306 0 : CPABORT("MPI_Type_create_hindexed @ "//routineN)
4307 1888 : CALL MPI_Type_commit(type_descriptor%type_handle, ierr)
4308 1888 : IF (ierr /= 0) &
4309 0 : CPABORT("MPI_Type_commit @ "//routineN)
4310 : #else
4311 : type_descriptor%type_handle = 68
4312 : #endif
4313 1888 : type_descriptor%length = count
4314 1888 : type_descriptor%has_indexing = .TRUE.
4315 1888 : type_descriptor%index_descriptor%index => lengths
4316 1888 : type_descriptor%index_descriptor%chunks => displs
4317 :
4318 1888 : CALL mp_timestop(handle)
4319 :
4320 1888 : END FUNCTION mp_file_type_hindexed_make_chv
4321 :
4322 : ! **************************************************************************************************
4323 : !> \brief Uses a previously created indexed MPI character type to tell the MPI processes
4324 : !> how to partition (set_view) an opened file
4325 : !> \param fh the file handle associated with the input file
4326 : !> \param offset global offset determining where the relevant data begins
4327 : !> \param type_descriptor container for the MPI type
4328 : !> \author Nico Holmberg [05.2017]
4329 : ! **************************************************************************************************
4330 1888 : SUBROUTINE mp_file_type_set_view_chv(fh, offset, type_descriptor)
4331 : TYPE(mp_file_type), INTENT(IN) :: fh
4332 : INTEGER(kind=file_offset), INTENT(IN) :: offset
4333 : TYPE(mp_file_descriptor_type) :: type_descriptor
4334 :
4335 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_file_set_view_chv'
4336 :
4337 : INTEGER :: handle
4338 : #if defined(__parallel)
4339 : INTEGER :: ierr
4340 : #endif
4341 :
4342 1888 : CALL mp_timeset(routineN, handle)
4343 :
4344 : #if defined(__parallel)
4345 1888 : ierr = 0
4346 1888 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
4347 : CALL MPI_File_set_view(fh%handle, offset, MPI_CHARACTER, &
4348 1888 : type_descriptor%type_handle, "native", MPI_INFO_NULL, ierr)
4349 1888 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ MPI_File_set_view")
4350 : #else
4351 : ! Uses absolute offsets stored in mp_file_descriptor_type
4352 : MARK_USED(fh)
4353 : MARK_USED(offset)
4354 : MARK_USED(type_descriptor)
4355 : #endif
4356 :
4357 1888 : CALL mp_timestop(handle)
4358 :
4359 1888 : END SUBROUTINE mp_file_type_set_view_chv
4360 :
4361 : ! **************************************************************************************************
4362 : !> \brief (parallel) Collective, blocking read of a character array from a file. File access pattern
4363 : ! determined by a previously set file view.
4364 : !> (serial) Unformatted stream read using explicit offsets
4365 : !> \param fh the file handle associated with the input file
4366 : !> \param msglen the message length of an individual vector component
4367 : !> \param ndims the number of vector components
4368 : !> \param buffer the buffer where the data is placed
4369 : !> \param type_descriptor container for the MPI type
4370 : !> \author Nico Holmberg [05.2017]
4371 : ! **************************************************************************************************
4372 38 : SUBROUTINE mp_file_read_all_chv(fh, msglen, ndims, buffer, type_descriptor)
4373 : CLASS(mp_file_type), INTENT(IN) :: fh
4374 : INTEGER, INTENT(IN) :: msglen
4375 : INTEGER, INTENT(IN) :: ndims
4376 : CHARACTER(LEN=msglen), DIMENSION(ndims), INTENT(INOUT) :: buffer
4377 : TYPE(mp_file_descriptor_type), &
4378 : INTENT(IN), OPTIONAL :: type_descriptor
4379 :
4380 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_file_read_all_chv'
4381 :
4382 : INTEGER :: handle
4383 : #if defined(__parallel)
4384 : INTEGER:: ierr
4385 : #else
4386 : INTEGER :: i
4387 : #endif
4388 :
4389 38 : CALL mp_timeset(routineN, handle)
4390 :
4391 : #if defined(__parallel)
4392 38 : ierr = 0
4393 : MARK_USED(type_descriptor)
4394 38 : CALL MPI_File_read_all(fh%handle, buffer, ndims*msglen, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4395 38 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ MPI_File_read_all")
4396 38 : CALL add_perf(perf_id=28, count=1, msg_size=ndims*msglen)
4397 : #else
4398 : MARK_USED(msglen)
4399 : MARK_USED(ndims)
4400 : IF (.NOT. PRESENT(type_descriptor)) &
4401 : CALL cp_abort(__LOCATION__, &
4402 : "Container for mp_file_descriptor_type must be present in serial call.")
4403 : IF (.NOT. type_descriptor%has_indexing) &
4404 : CALL cp_abort(__LOCATION__, &
4405 : "File view has not been set in mp_file_descriptor_type.")
4406 : ! Use explicit offsets
4407 : DO i = 1, ndims
4408 : READ (fh%handle, POS=type_descriptor%index_descriptor%chunks(i)) buffer(i)
4409 : END DO
4410 : #endif
4411 :
4412 38 : CALL mp_timestop(handle)
4413 :
4414 38 : END SUBROUTINE mp_file_read_all_chv
4415 :
4416 : ! **************************************************************************************************
4417 : !> \brief (parallel) Collective, blocking write of a character array to a file. File access pattern
4418 : ! determined by a previously set file view.
4419 : !> (serial) Unformatted stream write using explicit offsets
4420 : !> \param fh the file handle associated with the output file
4421 : !> \param msglen the message length of an individual vector component
4422 : !> \param ndims the number of vector components
4423 : !> \param buffer the buffer where the data is placed
4424 : !> \param type_descriptor container for the MPI type
4425 : !> \author Nico Holmberg [05.2017]
4426 : ! **************************************************************************************************
4427 1850 : SUBROUTINE mp_file_write_all_chv(fh, msglen, ndims, buffer, type_descriptor)
4428 : CLASS(mp_file_type), INTENT(IN) :: fh
4429 : INTEGER, INTENT(IN) :: msglen
4430 : INTEGER, INTENT(IN) :: ndims
4431 : CHARACTER(LEN=msglen), DIMENSION(ndims), INTENT(IN) :: buffer
4432 : TYPE(mp_file_descriptor_type), &
4433 : INTENT(IN), OPTIONAL :: type_descriptor
4434 :
4435 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_file_write_all_chv'
4436 :
4437 : INTEGER :: handle
4438 : #if defined(__parallel)
4439 : INTEGER :: ierr
4440 : #else
4441 : INTEGER :: i
4442 : #endif
4443 :
4444 1850 : CALL mp_timeset(routineN, handle)
4445 :
4446 : #if defined(__parallel)
4447 : MARK_USED(type_descriptor)
4448 1850 : CALL mpi_file_set_errhandler(fh%handle, MPI_ERRORS_RETURN, ierr)
4449 1850 : CALL MPI_File_write_all(fh%handle, buffer, ndims*msglen, MPI_CHARACTER, MPI_STATUS_IGNORE, ierr)
4450 1850 : IF (ierr .NE. 0) CALL mp_stop(ierr, "mpi_file_set_errhandler @ MPI_File_write_all")
4451 1850 : CALL add_perf(perf_id=28, count=1, msg_size=ndims*msglen)
4452 : #else
4453 : MARK_USED(msglen)
4454 : MARK_USED(ndims)
4455 : IF (.NOT. PRESENT(type_descriptor)) &
4456 : CALL cp_abort(__LOCATION__, &
4457 : "Container for mp_file_descriptor_type must be present in serial call.")
4458 : IF (.NOT. type_descriptor%has_indexing) &
4459 : CALL cp_abort(__LOCATION__, &
4460 : "File view has not been set in mp_file_descriptor_type.")
4461 : ! Use explicit offsets
4462 : DO i = 1, ndims
4463 : WRITE (fh%handle, POS=type_descriptor%index_descriptor%chunks(i)) buffer(i)
4464 : END DO
4465 : #endif
4466 :
4467 1850 : CALL mp_timestop(handle)
4468 :
4469 1850 : END SUBROUTINE mp_file_write_all_chv
4470 :
4471 : ! **************************************************************************************************
4472 : !> \brief Releases the type used for MPI I/O
4473 : !> \param type_descriptor the container for the MPI type
4474 : !> \author Nico Holmberg [05.2017]
4475 : ! **************************************************************************************************
4476 1888 : SUBROUTINE mp_file_type_free(type_descriptor)
4477 : TYPE(mp_file_descriptor_type) :: type_descriptor
4478 :
4479 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_file_type_free'
4480 :
4481 : INTEGER :: handle
4482 : #if defined(__parallel)
4483 : INTEGER :: ierr
4484 : #endif
4485 :
4486 1888 : CALL mp_timeset(routineN, handle)
4487 :
4488 : #if defined(__parallel)
4489 1888 : CALL MPI_Type_free(type_descriptor%type_handle, ierr)
4490 1888 : IF (ierr /= 0) &
4491 0 : CPABORT("MPI_Type_free @ "//routineN)
4492 : #endif
4493 : #if defined(__parallel) && defined(__MPI_F08)
4494 : type_descriptor%type_handle%mpi_val = -1
4495 : #else
4496 1888 : type_descriptor%type_handle = -1
4497 : #endif
4498 1888 : type_descriptor%length = -1
4499 1888 : IF (type_descriptor%has_indexing) THEN
4500 1888 : NULLIFY (type_descriptor%index_descriptor%index)
4501 1888 : NULLIFY (type_descriptor%index_descriptor%chunks)
4502 1888 : type_descriptor%has_indexing = .FALSE.
4503 : END IF
4504 :
4505 1888 : CALL mp_timestop(handle)
4506 :
4507 1888 : END SUBROUTINE mp_file_type_free
4508 :
4509 : ! **************************************************************************************************
4510 : !> \brief (parallel) Utility routine to determine MPI file access mode based on variables
4511 : ! that in the serial case would get passed to the intrinsic OPEN
4512 : !> (serial) No action
4513 : !> \param mpi_io flag that determines if MPI I/O will actually be used
4514 : !> \param replace flag that indicates whether file needs to be deleted prior to opening it
4515 : !> \param amode the MPI I/O access mode
4516 : !> \param form formatted or unformatted data?
4517 : !> \param action the variable that determines what to do with file
4518 : !> \param status the status flag:
4519 : !> \param position should the file be appended or rewound
4520 : !> \author Nico Holmberg [11.2017]
4521 : ! **************************************************************************************************
4522 1850 : SUBROUTINE mp_file_get_amode(mpi_io, replace, amode, form, action, status, position)
4523 : LOGICAL, INTENT(INOUT) :: mpi_io, replace
4524 : INTEGER, INTENT(OUT) :: amode
4525 : CHARACTER(len=*), INTENT(IN) :: form, action, status, position
4526 :
4527 1850 : amode = -1
4528 : #if defined(__parallel)
4529 : ! Disable mpi io for unformatted access
4530 0 : SELECT CASE (form)
4531 : CASE ("FORMATTED")
4532 : ! Do nothing
4533 : CASE ("UNFORMATTED")
4534 0 : mpi_io = .FALSE.
4535 : CASE DEFAULT
4536 1850 : CPABORT("Unknown MPI file form requested.")
4537 : END SELECT
4538 : ! Determine file access mode (limited set of allowed choices)
4539 1850 : SELECT CASE (action)
4540 : CASE ("WRITE")
4541 1850 : amode = file_amode_wronly
4542 0 : SELECT CASE (status)
4543 : CASE ("NEW")
4544 : ! Try to open new file for writing, crash if file already exists
4545 0 : amode = amode + file_amode_create + file_amode_excl
4546 : CASE ("UNKNOWN")
4547 : ! Open file for writing and create it if file does not exist
4548 1526 : amode = amode + file_amode_create
4549 76 : SELECT CASE (position)
4550 : CASE ("APPEND")
4551 : ! Append existing file
4552 76 : amode = amode + file_amode_append
4553 : CASE ("REWIND", "ASIS")
4554 : ! Do nothing
4555 : CASE DEFAULT
4556 1526 : CPABORT("Unknown MPI file position requested.")
4557 : END SELECT
4558 : CASE ("OLD")
4559 324 : SELECT CASE (position)
4560 : CASE ("APPEND")
4561 : ! Append existing file
4562 0 : amode = amode + file_amode_append
4563 : CASE ("REWIND", "ASIS")
4564 : ! Do nothing
4565 : CASE DEFAULT
4566 0 : CPABORT("Unknown MPI file position requested.")
4567 : END SELECT
4568 : CASE ("REPLACE")
4569 : ! Overwrite existing file. Must delete existing file first
4570 324 : amode = amode + file_amode_create
4571 324 : replace = .TRUE.
4572 : CASE ("SCRATCH")
4573 : ! Disable
4574 0 : mpi_io = .FALSE.
4575 : CASE DEFAULT
4576 1850 : CPABORT("Unknown MPI file status requested.")
4577 : END SELECT
4578 : CASE ("READ")
4579 0 : amode = file_amode_rdonly
4580 0 : SELECT CASE (status)
4581 : CASE ("NEW")
4582 0 : CPABORT("Cannot read from 'NEW' file.")
4583 : CASE ("REPLACE")
4584 0 : CPABORT("Illegal status 'REPLACE' for read.")
4585 : CASE ("UNKNOWN", "OLD")
4586 : ! Do nothing
4587 : CASE ("SCRATCH")
4588 : ! Disable
4589 0 : mpi_io = .FALSE.
4590 : CASE DEFAULT
4591 0 : CPABORT("Unknown MPI file status requested.")
4592 : END SELECT
4593 : CASE ("READWRITE")
4594 0 : amode = file_amode_rdwr
4595 0 : SELECT CASE (status)
4596 : CASE ("NEW")
4597 : ! Try to open new file, crash if file already exists
4598 0 : amode = amode + file_amode_create + file_amode_excl
4599 : CASE ("UNKNOWN")
4600 : ! Open file and create it if file does not exist
4601 0 : amode = amode + file_amode_create
4602 0 : SELECT CASE (position)
4603 : CASE ("APPEND")
4604 : ! Append existing file
4605 0 : amode = amode + file_amode_append
4606 : CASE ("REWIND", "ASIS")
4607 : ! Do nothing
4608 : CASE DEFAULT
4609 0 : CPABORT("Unknown MPI file position requested.")
4610 : END SELECT
4611 : CASE ("OLD")
4612 0 : SELECT CASE (position)
4613 : CASE ("APPEND")
4614 : ! Append existing file
4615 0 : amode = amode + file_amode_append
4616 : CASE ("REWIND", "ASIS")
4617 : ! Do nothing
4618 : CASE DEFAULT
4619 0 : CPABORT("Unknown MPI file position requested.")
4620 : END SELECT
4621 : CASE ("REPLACE")
4622 : ! Overwrite existing file. Must delete existing file first
4623 0 : amode = amode + file_amode_create
4624 0 : replace = .TRUE.
4625 : CASE ("SCRATCH")
4626 : ! Disable
4627 0 : mpi_io = .FALSE.
4628 : CASE DEFAULT
4629 0 : CPABORT("Unknown MPI file status requested.")
4630 : END SELECT
4631 : CASE DEFAULT
4632 1850 : CPABORT("Unknown MPI file action requested.")
4633 : END SELECT
4634 : #else
4635 : MARK_USED(replace)
4636 : MARK_USED(form)
4637 : MARK_USED(position)
4638 : MARK_USED(status)
4639 : MARK_USED(action)
4640 : mpi_io = .FALSE.
4641 : #endif
4642 :
4643 1850 : END SUBROUTINE mp_file_get_amode
4644 :
4645 : ! **************************************************************************************************
4646 : !> \brief Non-blocking send of custom type
4647 : !> \param msgin ...
4648 : !> \param dest ...
4649 : !> \param comm ...
4650 : !> \param request ...
4651 : !> \param tag ...
4652 : ! **************************************************************************************************
4653 0 : SUBROUTINE mp_isend_custom(msgin, dest, comm, request, tag)
4654 : TYPE(mp_type_descriptor_type), INTENT(IN) :: msgin
4655 : INTEGER, INTENT(IN) :: dest
4656 : CLASS(mp_comm_type), INTENT(IN) :: comm
4657 : TYPE(mp_request_type), INTENT(out) :: request
4658 : INTEGER, INTENT(in), OPTIONAL :: tag
4659 :
4660 : INTEGER :: ierr, my_tag
4661 :
4662 0 : ierr = 0
4663 0 : my_tag = 0
4664 :
4665 : #if defined(__parallel)
4666 0 : IF (PRESENT(tag)) my_tag = tag
4667 :
4668 : CALL mpi_isend(MPI_BOTTOM, 1, msgin%type_handle, dest, my_tag, &
4669 0 : comm%handle, request%handle, ierr)
4670 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_isend @ mp_isend_custom")
4671 : #else
4672 : MARK_USED(msgin)
4673 : MARK_USED(dest)
4674 : MARK_USED(comm)
4675 : MARK_USED(tag)
4676 : ierr = 1
4677 : request = mp_request_null
4678 : CALL mp_stop(ierr, "mp_isend called in non parallel case")
4679 : #endif
4680 0 : END SUBROUTINE mp_isend_custom
4681 :
4682 : ! **************************************************************************************************
4683 : !> \brief Non-blocking receive of vector data
4684 : !> \param msgout ...
4685 : !> \param source ...
4686 : !> \param comm ...
4687 : !> \param request ...
4688 : !> \param tag ...
4689 : ! **************************************************************************************************
4690 0 : SUBROUTINE mp_irecv_custom(msgout, source, comm, request, tag)
4691 : TYPE(mp_type_descriptor_type), INTENT(INOUT) :: msgout
4692 : INTEGER, INTENT(IN) :: source
4693 : CLASS(mp_comm_type), INTENT(IN) :: comm
4694 : TYPE(mp_request_type), INTENT(out) :: request
4695 : INTEGER, INTENT(in), OPTIONAL :: tag
4696 :
4697 : INTEGER :: ierr, my_tag
4698 :
4699 0 : ierr = 0
4700 0 : my_tag = 0
4701 :
4702 : #if defined(__parallel)
4703 0 : IF (PRESENT(tag)) my_tag = tag
4704 :
4705 : CALL mpi_irecv(MPI_BOTTOM, 1, msgout%type_handle, source, my_tag, &
4706 0 : comm%handle, request%handle, ierr)
4707 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_irecv @ mp_irecv_custom")
4708 : #else
4709 : MARK_USED(msgout)
4710 : MARK_USED(source)
4711 : MARK_USED(comm)
4712 : MARK_USED(tag)
4713 : ierr = 1
4714 : request = mp_request_null
4715 : CPABORT("mp_irecv called in non parallel case")
4716 : #endif
4717 0 : END SUBROUTINE mp_irecv_custom
4718 :
4719 : ! **************************************************************************************************
4720 : !> \brief Window free
4721 : !> \param win ...
4722 : ! **************************************************************************************************
4723 0 : SUBROUTINE mp_win_free(win)
4724 : CLASS(mp_win_type), INTENT(INOUT) :: win
4725 :
4726 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_win_free'
4727 :
4728 : INTEGER :: handle
4729 : #if defined(__parallel)
4730 : INTEGER :: ierr
4731 : #endif
4732 :
4733 0 : CALL mp_timeset(routineN, handle)
4734 :
4735 : #if defined(__parallel)
4736 0 : ierr = 0
4737 0 : CALL mpi_win_free(win%handle, ierr)
4738 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_win_free @ "//routineN)
4739 :
4740 0 : CALL add_perf(perf_id=21, count=1)
4741 : #else
4742 : win%handle = mp_win_null_handle
4743 : #endif
4744 0 : CALL mp_timestop(handle)
4745 0 : END SUBROUTINE mp_win_free
4746 :
4747 0 : SUBROUTINE mp_win_assign(win_new, win_old)
4748 : CLASS(mp_win_type), INTENT(OUT) :: win_new
4749 : CLASS(mp_win_type), INTENT(IN) :: win_old
4750 :
4751 0 : win_new%handle = win_old%handle
4752 :
4753 0 : END SUBROUTINE mp_win_assign
4754 :
4755 : ! **************************************************************************************************
4756 : !> \brief Window flush
4757 : !> \param win ...
4758 : ! **************************************************************************************************
4759 0 : SUBROUTINE mp_win_flush_all(win)
4760 : CLASS(mp_win_type), INTENT(IN) :: win
4761 :
4762 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_win_flush_all'
4763 :
4764 : INTEGER :: handle, ierr
4765 :
4766 0 : ierr = 0
4767 0 : CALL mp_timeset(routineN, handle)
4768 :
4769 : #if defined(__parallel)
4770 0 : CALL mpi_win_flush_all(win%handle, ierr)
4771 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_win_flush_all @ "//routineN)
4772 : #else
4773 : MARK_USED(win)
4774 : #endif
4775 0 : CALL mp_timestop(handle)
4776 0 : END SUBROUTINE mp_win_flush_all
4777 :
4778 : ! **************************************************************************************************
4779 : !> \brief Window lock
4780 : !> \param win ...
4781 : ! **************************************************************************************************
4782 0 : SUBROUTINE mp_win_lock_all(win)
4783 : CLASS(mp_win_type), INTENT(IN) :: win
4784 :
4785 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_win_lock_all'
4786 :
4787 : INTEGER :: handle, ierr
4788 :
4789 0 : ierr = 0
4790 0 : CALL mp_timeset(routineN, handle)
4791 :
4792 : #if defined(__parallel)
4793 :
4794 0 : CALL mpi_win_lock_all(MPI_MODE_NOCHECK, win%handle, ierr)
4795 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_win_lock_all @ "//routineN)
4796 :
4797 0 : CALL add_perf(perf_id=19, count=1)
4798 : #else
4799 : MARK_USED(win)
4800 : #endif
4801 0 : CALL mp_timestop(handle)
4802 0 : END SUBROUTINE mp_win_lock_all
4803 :
4804 : ! **************************************************************************************************
4805 : !> \brief Window lock
4806 : !> \param win ...
4807 : ! **************************************************************************************************
4808 0 : SUBROUTINE mp_win_unlock_all(win)
4809 : CLASS(mp_win_type), INTENT(IN) :: win
4810 :
4811 : CHARACTER(len=*), PARAMETER :: routineN = 'mp_win_unlock_all'
4812 :
4813 : INTEGER :: handle, ierr
4814 :
4815 0 : ierr = 0
4816 0 : CALL mp_timeset(routineN, handle)
4817 :
4818 : #if defined(__parallel)
4819 :
4820 0 : CALL mpi_win_unlock_all(win%handle, ierr)
4821 0 : IF (ierr /= 0) CALL mp_stop(ierr, "mpi_win_unlock_all @ "//routineN)
4822 :
4823 0 : CALL add_perf(perf_id=19, count=1)
4824 : #else
4825 : MARK_USED(win)
4826 : #endif
4827 0 : CALL mp_timestop(handle)
4828 0 : END SUBROUTINE mp_win_unlock_all
4829 :
4830 : ! **************************************************************************************************
4831 : !> \brief Starts a timer region
4832 : !> \param routineN ...
4833 : !> \param handle ...
4834 : ! **************************************************************************************************
4835 139613645 : SUBROUTINE mp_timeset(routineN, handle)
4836 : CHARACTER(len=*), INTENT(IN) :: routineN
4837 : INTEGER, INTENT(OUT) :: handle
4838 :
4839 139613645 : IF (mp_collect_timings) &
4840 139425515 : CALL timeset(routineN, handle)
4841 139613645 : END SUBROUTINE mp_timeset
4842 :
4843 : ! **************************************************************************************************
4844 : !> \brief Ends a timer region
4845 : !> \param handle ...
4846 : ! **************************************************************************************************
4847 139613645 : SUBROUTINE mp_timestop(handle)
4848 : INTEGER, INTENT(IN) :: handle
4849 :
4850 139613645 : IF (mp_collect_timings) &
4851 139425515 : CALL timestop(handle)
4852 139613645 : END SUBROUTINE mp_timestop
4853 :
4854 : #:include 'message_passing.fypp'
4855 :
4856 64791795 : END MODULE message_passing
|