31 void* memptr = malloc(size);
39 int block_lengths[2] = {1, 1};
40 MPI_Aint addresses[3];
42 MPI_Get_address(&s, &addresses[0]);
43 MPI_Get_address(&(s.
gid), &addresses[1]);
44 MPI_Get_address(&(s.
spiketime), &addresses[2]);
46 MPI_Aint displacements[2] = {addresses[1] - addresses[0], addresses[2] - addresses[0]};
48 MPI_Datatype typelist[2] = {MPI_INT, MPI_DOUBLE};
49 MPI_Type_create_struct(2, block_lengths, displacements, typelist, &
spike_type);
53 #if nrn_spikebuf_size > 0
55 static MPI_Datatype spikebuf_type;
58 static void make_spikebuf_type() {
61 MPI_Datatype typelist[3] = {MPI_INT, MPI_INT, MPI_DOUBLE};
63 MPI_Aint addresses[4];
64 MPI_Get_address(&s, &addresses[0]);
65 MPI_Get_address(&(s.nspike), &addresses[1]);
66 MPI_Get_address(&(s.gid[0]), &addresses[2]);
67 MPI_Get_address(&(s.spiketime[0]), &addresses[3]);
69 MPI_Aint displacements[3] = {addresses[1] - addresses[0],
70 addresses[2] - addresses[0],
71 addresses[3] - addresses[0]};
73 MPI_Type_create_struct(3, block_lengths, displacements, typelist, &spikebuf_type);
74 MPI_Type_commit(&spikebuf_type);
103 #if nrn_spikebuf_size > 0
104 make_spikebuf_type();
107 #if nrn_spikebuf_size == 0
108 MPI_Allgather(&nout, 1, MPI_INT, nin, 1, MPI_INT,
nrnmpi_comm);
110 for (
int i = 1;
i <
np; ++
i) {
123 MPI_Allgather(spbufout, 1, spikebuf_type, spbufin, 1, spikebuf_type,
nrnmpi_comm);
125 int n = spbufin[0].
nspike;
132 for (
int i = 1;
i <
np; ++
i) {
144 if (icapacity < novfl) {
145 icapacity = novfl + 10;
178 unsigned char*& spfixin_ovfl,
182 unsigned char* spikeout_fixed,
184 unsigned char* spikein_fixed,
195 spikeout_fixed, ag_send_size, MPI_BYTE, spikein_fixed, ag_send_size, MPI_BYTE,
nrnmpi_comm);
199 for (
int i = 0;
i <
np; ++
i) {
201 int idx =
i * ag_send_size;
202 int n = spikein_fixed[idx++] * 256;
203 n += spikein_fixed[idx++];
206 if (n > send_nspike) {
207 int bs = 2 + n * (1 + localgid_size) - ag_send_size;
210 novfl += n - send_nspike;
216 if (ovfl_capacity < novfl) {
217 ovfl_capacity = novfl + 10;
219 spfixin_ovfl = (
unsigned char*)
emalloc(ovfl_capacity * (1 + localgid_size) *
220 sizeof(
unsigned char));
229 MPI_Allgatherv(spikeout_fixed + ag_send_size,
244 MPI_Allreduce(&x, &result, 1, MPI_INT, MPI_MAX,
nrnmpi_comm);
249 MPI_Alltoall(s, n, MPI_INT, r, n, MPI_INT,
nrnmpi_comm);
258 MPI_Alltoallv(s, scnt, sdispl, MPI_INT, r, rcnt, rdispl, MPI_INT,
nrnmpi_comm);
267 MPI_Alltoallv(s, scnt, sdispl, MPI_DOUBLE, r, rcnt, rdispl, MPI_DOUBLE,
nrnmpi_comm);
273 MPI_Allgather(s, n, MPI_INT, r, n, MPI_INT,
nrnmpi_comm);
278 MPI_Allreduce(&x, &result, 1, MPI_DOUBLE, MPI_MIN,
nrnmpi_comm);
284 MPI_Allreduce(&x, &result, 1, MPI_DOUBLE, MPI_MAX,
nrnmpi_comm);
297 }
else if (type == 2) {
302 MPI_Allreduce(&x, &result, 1, MPI_DOUBLE, tt,
nrnmpi_comm);
311 }
else if (type == 2) {
325 }
else if (type == 2) {
336 static MPI_Comm multisend_comm;
338 void nrnmpi_multisend_comm_impl() {
339 if (!multisend_comm) {
340 MPI_Comm_dup(MPI_COMM_WORLD, &multisend_comm);
344 void nrnmpi_multisend_impl(NRNMPI_Spike* spk,
int n,
int* hosts) {
346 for (
int i = 0;
i < n; ++
i) {
347 MPI_Isend(spk, 1,
spike_type, hosts[
i], 1, multisend_comm, &r);
348 MPI_Request_free(&r);
352 int nrnmpi_multisend_single_advance_impl(NRNMPI_Spike* spk) {
355 MPI_Iprobe(MPI_ANY_SOURCE, 1, multisend_comm, &flag, &status);
357 MPI_Recv(spk, 1,
spike_type, MPI_ANY_SOURCE, 1, multisend_comm, &status);
362 int nrnmpi_multisend_conserve_impl(
int nsend,
int nrecv) {
364 tcnts[0] = nsend - nrecv;
365 MPI_Allreduce(tcnts, tcnts + 1, 1, MPI_INT, MPI_SUM, multisend_comm);