CoreNEURON
nrnmpi.cpp
Go to the documentation of this file.
1 /*
2 # =============================================================================
3 # Copyright (c) 2016 - 2021 Blue Brain Project/EPFL
4 #
5 # See top-level LICENSE file for details.
6 # =============================================================================.
7 */
8 
9 #include <iostream>
10 #include <string>
11 #include <tuple>
12 
13 #include "coreneuron/nrnconf.h"
14 #include "coreneuron/mpi/nrnmpi.h"
16 #include "nrnmpi.hpp"
17 #if _OPENMP
18 #include <omp.h>
19 #endif
20 #include <mpi.h>
21 namespace coreneuron {
22 
24 MPI_Comm nrnmpi_comm;
27 
28 static bool nrnmpi_under_nrncontrol_{false};
29 
30 static void nrn_fatal_error(const char* msg) {
31  if (nrnmpi_myid_ == 0) {
32  printf("%s\n", msg);
33  }
35 }
36 
37 nrnmpi_init_ret_t nrnmpi_init_impl(int* pargc, char*** pargv, bool is_quiet) {
38  // Execute at most once per launch. Avoid memory leak.
39  static bool executed = false;
40  if (executed) {
42  }
43 
45 
46  if (!nrnmpi_initialized_impl()) {
47 #if defined(_OPENMP)
48  int required = MPI_THREAD_FUNNELED;
49  int provided;
50  nrn_assert(MPI_Init_thread(pargc, pargv, required, &provided) == MPI_SUCCESS);
51 
52  nrn_assert(required <= provided);
53 #else
54  nrn_assert(MPI_Init(pargc, pargv) == MPI_SUCCESS);
55 #endif
56  }
57  nrn_assert(MPI_Comm_dup(MPI_COMM_WORLD, &nrnmpi_world_comm) == MPI_SUCCESS);
58  nrn_assert(MPI_Comm_dup(nrnmpi_world_comm, &nrnmpi_comm) == MPI_SUCCESS);
59  nrn_assert(MPI_Comm_rank(nrnmpi_world_comm, &nrnmpi_myid_) == MPI_SUCCESS);
60  nrn_assert(MPI_Comm_size(nrnmpi_world_comm, &nrnmpi_numprocs_) == MPI_SUCCESS);
62 
63  if (nrnmpi_myid_ == 0 && !is_quiet) {
64 #if defined(_OPENMP)
65  printf(" num_mpi=%d\n num_omp_thread=%d\n\n", nrnmpi_numprocs_, omp_get_max_threads());
66 #else
67  printf(" num_mpi=%d\n\n", nrnmpi_numprocs_);
68 #endif
69  }
70 
71  executed = true;
73 }
74 
78  MPI_Comm_free(&nrnmpi_world_comm);
79  MPI_Comm_free(&nrnmpi_comm);
80  MPI_Finalize();
81  }
82  }
83 }
84 
85 // check if appropriate threading level supported (i.e. MPI_THREAD_FUNNELED)
87  int th = 0;
88  MPI_Query_thread(&th);
89  if (th < MPI_THREAD_FUNNELED) {
91  "\n Current MPI library doesn't support MPI_THREAD_FUNNELED,\
92  \n Run without enabling multi-threading!");
93  }
94 }
95 
97  int flag = 0;
98  MPI_Initialized(&flag);
99  return flag != 0;
100 }
101 
102 void nrnmpi_abort_impl(int errcode) {
103  MPI_Abort(MPI_COMM_WORLD, errcode);
104 }
105 
107  return MPI_Wtime();
108 }
109 
110 /**
111  * Return local mpi rank within a shared memory node
112  *
113  * When performing certain operations, we need to know the rank of mpi
114  * process on a given node. This function uses MPI 3 MPI_Comm_split_type
115  * function and MPI_COMM_TYPE_SHARED key to find out the local rank.
116  */
118  int local_rank = 0;
119  if (nrnmpi_initialized_impl()) {
120  MPI_Comm local_comm;
121  MPI_Comm_split_type(
122  MPI_COMM_WORLD, MPI_COMM_TYPE_SHARED, nrnmpi_myid_, MPI_INFO_NULL, &local_comm);
123  MPI_Comm_rank(local_comm, &local_rank);
124  MPI_Comm_free(&local_comm);
125  }
126  return local_rank;
127 }
128 
129 /**
130  * Return number of ranks running on single shared memory node
131  *
132  * We use MPI 3 MPI_Comm_split_type function and MPI_COMM_TYPE_SHARED key to
133  * determine number of mpi ranks within a shared memory node.
134  */
136  int local_size = 1;
137  if (nrnmpi_initialized_impl()) {
138  MPI_Comm local_comm;
139  MPI_Comm_split_type(
140  MPI_COMM_WORLD, MPI_COMM_TYPE_SHARED, nrnmpi_myid_, MPI_INFO_NULL, &local_comm);
141  MPI_Comm_size(local_comm, &local_size);
142  MPI_Comm_free(&local_comm);
143  }
144  return local_size;
145 }
146 
147 /**
148  * Write given buffer to a new file using MPI collective I/O
149  *
150  * For output like spikes, each rank has to write spike timing
151  * information to a single file. This routine writes buffers
152  * of length len1, len2, len3... at the offsets 0, 0+len1,
153  * 0+len1+len2... offsets. This write op is a collective across
154  * all ranks of the common MPI communicator used for spike exchange.
155  *
156  * @param filename Name of the file to write
157  * @param buffer Buffer to write
158  * @param length Length of the buffer to write
159  */
160 void nrnmpi_write_file_impl(const std::string& filename, const char* buffer, size_t length) {
161  MPI_File fh;
162  MPI_Status status;
163 
164  // global offset into file
165  unsigned long offset = 0;
166  MPI_Exscan(&length, &offset, 1, MPI_UNSIGNED_LONG, MPI_SUM, nrnmpi_comm);
167 
168  int op_status = MPI_File_open(
169  nrnmpi_comm, filename.c_str(), MPI_MODE_CREATE | MPI_MODE_WRONLY, MPI_INFO_NULL, &fh);
170  if (op_status != MPI_SUCCESS && nrnmpi_myid_ == 0) {
171  std::cerr << "Error while opening output file " << filename << std::endl;
172  abort();
173  }
174 
175  op_status = MPI_File_write_at_all(fh, offset, buffer, length, MPI_BYTE, &status);
176  if (op_status != MPI_SUCCESS && nrnmpi_myid_ == 0) {
177  std::cerr << "Error while writing output " << std::endl;
178  abort();
179  }
180 
181  MPI_File_close(&fh);
182 }
183 } // namespace coreneuron
coreneuron::nrnmpi_local_rank_impl
int nrnmpi_local_rank_impl()
Return local mpi rank within a shared memory node.
Definition: nrnmpi.cpp:117
coreneuron::nrnmpi_abort_impl
void nrnmpi_abort_impl(int errcode)
Definition: nrnmpi.cpp:102
coreneuron::nrnmpi_local_size_impl
int nrnmpi_local_size_impl()
Return number of ranks running on single shared memory node.
Definition: nrnmpi.cpp:135
coreneuron::nrnmpi_numprocs_
int nrnmpi_numprocs_
Definition: nrnmpi.cpp:25
coreneuron::nrnmpi_init_impl
nrnmpi_init_ret_t nrnmpi_init_impl(int *pargc, char ***pargv, bool is_quiet)
Definition: nrnmpi.cpp:37
coreneuron::nrnmpi_init_ret_t
Definition: nrnmpidec.h:20
coreneuron::nrn_fatal_error
static void nrn_fatal_error(const char *msg)
Definition: nrnmpi.cpp:30
coreneuron::nrnmpi_write_file_impl
void nrnmpi_write_file_impl(const std::string &filename, const char *buffer, size_t length)
Write given buffer to a new file using MPI collective I/O.
Definition: nrnmpi.cpp:160
coreneuron
THIS FILE IS AUTO GENERATED DONT MODIFY IT.
Definition: corenrn_parameters.cpp:12
coreneuron::nrnmpi_check_threading_support_impl
void nrnmpi_check_threading_support_impl()
Definition: nrnmpi.cpp:86
coreneuron::nrnmpi_world_comm
MPI_Comm nrnmpi_world_comm
Definition: nrnmpi.cpp:23
coreneuron::nrnmpi_spike_initialize
void nrnmpi_spike_initialize()
Definition: mpispike.cpp:37
coreneuron::nrnmpi_myid_
int nrnmpi_myid_
Definition: nrnmpi.cpp:26
coreneuron::nrnmpi_comm
MPI_Comm nrnmpi_comm
Definition: nrnmpi.cpp:24
coreneuron::nrnmpi_under_nrncontrol_
static bool nrnmpi_under_nrncontrol_
Definition: nrnmpi.cpp:28
coreneuron::nrnmpi_wtime_impl
double nrnmpi_wtime_impl()
Definition: nrnmpi.cpp:106
coreneuron::nrnmpi_finalize_impl
void nrnmpi_finalize_impl(void)
Definition: nrnmpi.cpp:75
nrnconf.h
nrnmpi.hpp
nrn_assert
#define nrn_assert(x)
assert()-like macro, independent of NDEBUG status
Definition: nrn_assert.h:33
nrn_assert.h
nrnmpi.h
coreneuron::nrnmpi_initialized_impl
bool nrnmpi_initialized_impl()
Definition: nrnmpi.cpp:96