123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867 |
- /* cslmpi.c */
- /*
- * Interfaces for mpi from CSL. The bulk of this code was written by
- * M O Seymour (1997-98) who has released it for inclusion as part of
- * this Lisp system.
- */
- /*
- * This code may be used and modified, and redistributed in binary
- * or source form, subject to the "CCL Public License", which should
- * accompany it. This license is a variant on the BSD license, and thus
- * permits use of code derived from this in either open and commercial
- * projects: but it does require that updates to this code be made
- * available back to the originators of the package.
- * Before merging other code in with this or linking this code
- * with other packages or libraries please check that the license terms
- * of the other material are compatible with those of this.
- */
- /* Signature: 55e2c6ac 10-Oct-2002 */
- #include <stdarg.h>
- #include <string.h>
- #include <ctype.h>
- #include "machine.h"
- #include "tags.h"
- #include "cslerror.h"
- #include "externs.h"
- #include "arith.h"
- #include "entries.h"
- #ifdef TIMEOUT
- #include "timeout.h"
- #endif
- #ifdef USE_MPI
- #include "read.h"
- #include "mpipack.c"
- #define check_fix(v) if (!is_fixnum(v)) return aerror1(fun_name, v)
- #define get_arg(v) v = va_arg(a,Lisp_Object)
- #define get_fix_arg(v) get_arg(v); check_fix(v); v=int_of_fixnum(v)
- /************************ Environmental functions *******************/
- /* Returns process rank
- * (mpi_comm_rank comm)
- */
- /* For now, I assume that comm will fit into a fixnum.
- * This appears to be the case with MPICH (values in the hundreds),
- * but assumptions like this should not be made.
- */
- static Lisp_Object Lmpi_comm_rank(Lisp_Object nil, Lisp_Object comm)
- {
- int rank;
- static char fun_name[] = "mpi_comm_rank";
- CSL_IGNORE(nil);
- check_fix(comm);
- MPI_Comm_rank(int_of_fixnum(comm),&rank);
- return onevalue(fixnum_of_int(rank));
- }
- /* returns size of communicator
- * (mpi_comm_size comm)
- */
- /* Same assumption about comm. */
- static Lisp_Object Lmpi_comm_size(Lisp_Object nil, Lisp_Object comm)
- {
- int size;
- static char fun_name[] = "mpi_comm_size";
- CSL_IGNORE(nil);
- check_fix(comm);
- MPI_Comm_size(int_of_fixnum(comm),&size);
- return onevalue(fixnum_of_int(size));
- }
- /********************** Blocking point-to-point functions *************/
- /* Standard blocking send
- * (mpi_send message dest tag comm)
- * returns nil.
- */
- /* Same assumption about comm. */
- static Lisp_Object MS_CDECL Lmpi_send(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_send";
- Lisp_Object message;
- int dest,tag,comm;
- va_list a;
- argcheck(nargs,4,fun_name);
- va_start(a,nargs);
- get_arg(message);
- get_fix_arg(dest); get_fix_arg(tag); get_fix_arg(comm);
- pack_object(message);
- MPI_Send(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- dest, tag, comm);
- free(mpi_pack_buffer);
- return onevalue(nil);
- }
- /* Standard blocking receive
- * (mpi_recv source tag comm)
- * returns (message (source tag error)).
- */
- static Lisp_Object MS_CDECL Lmpi_recv(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_recv";
- MPI_Status status;
- int source,tag,comm;
- Lisp_Object Lstatus;
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_fix_arg(source); get_fix_arg(tag); get_fix_arg(comm);
- MPI_Probe(source, tag, comm, &status);
- MPI_Get_count(&status, MPI_PACKED, &mpi_pack_size);
- mpi_pack_buffer = (char*)malloc(mpi_pack_size);
- MPI_Recv(mpi_pack_buffer, mpi_pack_size, MPI_PACKED,
- source, tag, comm, &status);
-
- /* The only relevant status things are the 3 public fields, so I'll
- * stick them in a list and return them as the 2nd value
- */
- push(unpack_object());
- free(mpi_pack_buffer);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
- return onevalue(list2(my_pop(),Lstatus));
- }
- /* Standard blocking simultaneous send and receive
- * (mpi_sendrecv send_message dest send_tag source recv_tag comm)
- * returns (recv_message (source recv_tag error))
- */
- /* THERE IS A LIMIT OF 1024 BYTES FOR THE RECEIVE BUFFER (sorry.)
- * THIS WILL BE REMOVED ASAP.
- */
- static Lisp_Object MS_CDECL Lmpi_sendrecv(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_sendrecv";
- MPI_Status status;
- Lisp_Object Lstatus;
- Lisp_Object s_mess;
- int s_tag, r_tag, dest, source, comm;
- char r_buffer[1024];
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,6,fun_name);
- va_start(a,nargs);
- get_arg(s_mess);
- get_fix_arg(dest); get_fix_arg(s_tag);
- get_fix_arg(source); get_fix_arg(r_tag); get_fix_arg(comm);
- pack_object(s_mess);
- MPI_Sendrecv(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- dest, s_tag,
- r_buffer, 1024, MPI_PACKED,
- source, r_tag, comm, &status);
- free(mpi_pack_buffer);
- mpi_pack_buffer = r_buffer;
- push(unpack_object());
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
- return onevalue(list2(my_pop(),Lstatus));
- }
- /************** Non-Blocking point-to-point functions ***********/
- /* Standard non-blocking send post
- * (mpi_isend message dest tag comm)
- * returns request handle
- */
- static Lisp_Object MS_CDECL Lmpi_isend(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_isend";
- Lisp_Object message, request;
- int dest, tag, comm;
- va_list a;
- CSL_IGNORE(nil);
-
- /* For now, we assume type MPI_Request to be 32 bits. */
- request = Lmkvect32(nil,fixnum_of_int(2));
-
- argcheck(nargs,4,fun_name);
- va_start(a,nargs);
- get_arg(message);
- get_fix_arg(dest); get_fix_arg(tag); get_fix_arg(comm);
-
- pack_object(message);
- MPI_Isend(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- dest, tag, comm, (MPI_Request*)&elt(request,0));
- elt(request,1) = (int)mpi_pack_buffer;
- return onevalue(request);
- }
- /* Standard non-blocking receive post
- * (mpi_irecv source tag comm)
- * returns request handle
- */
- /* I actually cheat horribly by not posting the request at all (at least
- * not via MPI), but rather create my own "dummy" request structure.
- * Then, to complete the request, I MPI_(I)Probe for a matching message,
- * and receive it if it is there.
- * This is unsatisfactory since the operation is only non-blocking until the
- * first lump of the message arrives; for a long message, there may by
- * a lot of latency after this.
- */
- struct dummy_request{
- int source;
- int tag;
- int comm;
- };
- static Lisp_Object MS_CDECL Lmpi_irecv(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_irecv";
- int source,tag,comm;
- Lisp_Object request;
- va_list a;
- char* buffer;
- CSL_IGNORE(nil);
- /* For now, we assume type MPI_Request to be 32 bits. */
- request = Lmkvect32(nil,fixnum_of_int(2));
-
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_fix_arg(source); get_fix_arg(tag); get_fix_arg(comm);
- elt(request,1) = 0; /* There is no buffer yet */
- elt(request,0) = (int)malloc(sizeof(struct dummy_request));
- ((struct dummy_request*)elt(request,0))->source = source;
- ((struct dummy_request*)elt(request,0))->tag = tag;
- ((struct dummy_request*)elt(request,0))->comm = comm;
-
- return onevalue(request);
- }
- /* Wait to complete operation, and deallocate buffer.
- * (mpi_wait request)
- * for send, returns nil
- * for recv, returns (message (source tag error))
- */
- static Lisp_Object Lmpi_wait(Lisp_Object nil, Lisp_Object request)
- {
- MPI_Status status;
- Lisp_Object message, Lstatus;
- if ( !(is_vector(request) && type_of_header(vechdr(request)) == TYPE_VEC32 &&
- length_of_header(vechdr(request)) == 3*CELL) )
- return aerror1("mpi_wait",request);
- if ( elt(request,1)){
- status.MPI_ERROR = MPI_UNDEFINED;
- mpi_pack_buffer = (void*)elt(request,1);
- MPI_Wait( (MPI_Request*)&elt(request,0), &status);
- if (status.MPI_ERROR == MPI_UNDEFINED){ /* i.e. send request */
- free(mpi_pack_buffer);
- return onevalue(nil);
- }
- else { /* old-style receive */
- push(unpack_object());
- free(mpi_pack_buffer);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
- return onevalue(list2(my_pop(),Lstatus));
- }
- }
- else{ /* new-style receive */
- int source = ((struct dummy_request*)elt(request,0))->source,
- tag = ((struct dummy_request*)elt(request,0))->tag,
- comm = ((struct dummy_request*)elt(request,0))->comm;
- MPI_Probe(source, tag, comm, &status);
- free((struct dummy_request*)elt(request,0));
- MPI_Get_count(&status, MPI_PACKED, &mpi_pack_size);
- mpi_pack_buffer = (char*)malloc(mpi_pack_size);
- MPI_Recv(mpi_pack_buffer, mpi_pack_size, MPI_PACKED,
- source, tag, comm, &status);
-
- /* The only relevant status things are the 3 public fields, so I'll
- * stick them in a list and return them as the 2nd value
- */
- push(unpack_object());
- free(mpi_pack_buffer);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
-
- return onevalue(list2(my_pop(),Lstatus));
- }
- }
- /* Test for completion, deallocate buffer if so
- * (mpi_test request)
- * for send, returns flag
- * for recv, returns nil or (message (source tag error))
- */
- static Lisp_Object Lmpi_test(Lisp_Object nil, Lisp_Object request)
- {
- MPI_Status status;
- Lisp_Object message, Lstatus;
- int flag;
- if ( !(is_vector(request) && type_of_header(vechdr(request)) == TYPE_VEC32 &&
- length_of_header(vechdr(request)) == 3*CELL) )
- return aerror1("mpi_wait",request);
- if (elt(request,1)){
- status.MPI_ERROR = MPI_UNDEFINED;
- mpi_pack_buffer = (void*)elt(request,1);
- MPI_Test( (MPI_Request*)&elt(request,0), &flag, &status);
- if (!flag) return onevalue(nil);
- if (status.MPI_ERROR == MPI_UNDEFINED){ /* send request */
- free(mpi_pack_buffer);
- return onevalue(Lispify_predicate(YES));
- }
- else{ /* old-style receive */
- push(unpack_object());
- free(mpi_pack_buffer);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
-
- return onevalue(list2(my_pop(),Lstatus));
- }
- }
- else { /* new-style receive */
- int source = ((struct dummy_request*)elt(request,0))->source,
- tag = ((struct dummy_request*)elt(request,0))->tag,
- comm = ((struct dummy_request*)elt(request,0))->comm, flag;
- MPI_Iprobe(source, tag, comm, &flag, &status);
-
- if (!flag) return onevalue(nil);
-
- free((struct dummy_request*)elt(request,0));
- MPI_Get_count(&status, MPI_PACKED, &mpi_pack_size);
- mpi_pack_buffer = (char*)malloc(mpi_pack_size);
- MPI_Recv(mpi_pack_buffer, mpi_pack_size, MPI_PACKED,
- source, tag, comm, &status);
-
- /* The only relevant status things are the 3 public fields, so I'll
- * stick them in a list and return them as the 2nd value
- */
- push(unpack_object());
- free(mpi_pack_buffer);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
-
- return onevalue(list2(my_pop(),Lstatus));
- }
- }
- /************** Probe functions *******************/
- /* Non-blocking probe
- * (mpi_iprobe source tag comm)
- * returns (flag (source tag error))
- */
- static Lisp_Object MS_CDECL Lmpi_iprobe(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "impi_probe";
- MPI_Status status;
- int source, tag, comm, flag;
- Lisp_Object Lstatus;
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_fix_arg(source); get_fix_arg(tag); get_fix_arg(comm);
- MPI_Iprobe(source, tag, comm, &flag, &status);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
- return onevalue(list2(Lispify_predicate(flag), Lstatus));
- }
- /* Blocking probe
- * (mpi_probe source tag comm)
- * returns (source tag error)
- */
- static Lisp_Object MS_CDECL Lmpi_probe(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_probe";
- MPI_Status status;
- int source, tag, comm;
- Lisp_Object Lstatus;
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_fix_arg(source); get_fix_arg(tag); get_fix_arg(comm);
- MPI_Probe(source, tag, comm, &status);
- Lstatus = list3(fixnum_of_int(status.MPI_SOURCE),
- fixnum_of_int(status.MPI_TAG),
- fixnum_of_int(status.MPI_ERROR));
- return onevalue(Lstatus);
- }
- /************** Collective Communications *********/
- /* Barrier; blocks until all processes have called
- * (mpi_barrier comm)
- * returns nil
- */
- static Lisp_Object Lmpi_barrier(Lisp_Object nil, Lisp_Object comm)
- {
- int rank;
- static char fun_name[] = "mpi_barrier";
- check_fix(comm);
- MPI_Barrier(int_of_fixnum(comm));
- return onevalue(nil);
- }
- /* Broadcast; sends buffer of root to buffers of others.
- * (mpi_bcast message root comm) [message ignored if not root]
- * returns message
- */
- static Lisp_Object MS_CDECL Lmpi_bcast(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_bcast";
- Lisp_Object message;
- int root,comm,rank;
- va_list a;
-
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_arg(message); get_fix_arg(root); get_fix_arg(comm);
-
- MPI_Comm_rank(comm,&rank);
- if (rank == root){
- pack_object(message);
- MPI_Bcast(&mpi_pack_position, 1, MPI_LONG, root, comm);
- MPI_Bcast(mpi_pack_buffer, mpi_pack_position, MPI_PACKED, root, comm);
- free(mpi_pack_buffer);
- }
- else {
- MPI_Bcast(&mpi_pack_size, 1, MPI_LONG, root, comm);
- mpi_pack_buffer = (char*)malloc(mpi_pack_size);
- MPI_Bcast(mpi_pack_buffer, mpi_pack_size, MPI_PACKED, root, comm);
- message = unpack_object();
- free(mpi_pack_buffer);
- }
- return onevalue(message);
- }
- /* Gather: root receives messages from others.
- * (mpi_gather message root comm)
- * returns vector of messages if root, else nil.
- */
- static Lisp_Object MS_CDECL Lmpi_gather(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_gather";
- Lisp_Object message;
- int root,comm,rank;
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_arg(message); get_fix_arg(root); get_fix_arg(comm);
-
- MPI_Comm_rank(comm,&rank);
- pack_object(message);
- if (rank == root){
- int commsize, count;
- int *recvcounts, *displs;
- char *recvbuffer;
- MPI_Comm_size(comm,&commsize);
- recvcounts = (int*)calloc(commsize, sizeof(int));
- displs = (int*)calloc(commsize+1, sizeof(int));
- MPI_Gather(&mpi_pack_position, 1, MPI_LONG,
- recvcounts, 1, MPI_LONG, root, comm);
- displs[0] = 0;
- for (count = 0; count < commsize; ++count)
- displs[count+1] = displs[count] + recvcounts[count];
- recvbuffer = (char*)malloc(displs[commsize]);
- MPI_Gatherv(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- recvbuffer, recvcounts, displs, MPI_PACKED, root, comm);
- free(mpi_pack_buffer);
- message = Lmkvect(nil, fixnum_of_int(commsize-1));
- for (count = 0; count < commsize; ++count){
- mpi_pack_buffer = recvbuffer + displs[count];
- mpi_pack_size = recvcounts[count];
- elt(message, count) = unpack_object();
- }
- free(recvbuffer); free(recvcounts); free(displs);
- }
- else {
- MPI_Gather(&mpi_pack_position, 1, MPI_LONG, 0, 0, MPI_LONG, root, comm);
- MPI_Gatherv(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- 0,0,0,MPI_PACKED, root, comm);
- free(mpi_pack_buffer);
- message = nil;
- }
- return onevalue(message);
- }
- /* Scatter: inverse of gather.
- * (mpi_scatter vector_of_messages root comm) [messages ignored if not root]
- * returns message
- */
- static Lisp_Object MS_CDECL Lmpi_scatter(Lisp_Object nil, int nargs, ...)
- {
- static char fun_name[] = "mpi_scatter";
-
- Lisp_Object messages, message;
- int root, comm, rank;
- va_list a;
- CSL_IGNORE(nil);
- argcheck(nargs,3,fun_name);
- va_start(a,nargs);
- get_arg(messages); get_fix_arg(root); get_fix_arg(comm);
- MPI_Comm_rank(comm,&rank);
- if (rank == root){
- int commsize, count, *sendcounts, *displs, recvcount;
- char* recvbuffer;
- MPI_Comm_size(comm,&commsize);
- sendcounts = (int*)calloc(commsize, sizeof(int));
- displs = (int*)calloc(commsize+1, sizeof(int));
- displs[0] = 0;
- /* Call private functions in mpi_packing for consecutive packs */
- check_buffer = scatter_check_buffer;
- mpi_pack_offset = 0;
- mpi_pack_position = 0;
- mpi_pack_size = 0;
- mpi_buffer_bottom = 0;
- mpi_real_size = 0;
- for (count = 0; count < commsize; ++count){
- pack_cell(elt(messages,count));
- sendcounts[count] = mpi_pack_position;
- mpi_pack_size -= mpi_pack_position;
- mpi_pack_offset += mpi_pack_position;
- mpi_pack_buffer += mpi_pack_position;
- displs[count+1] = mpi_pack_offset;
- mpi_pack_position = 0;
- }
- check_buffer = default_check_buffer;
- MPI_Scatter(sendcounts, 1, MPI_LONG, &recvcount, 1, MPI_LONG, root, comm);
- recvbuffer = (char*)malloc(recvcount);
- MPI_Scatterv(mpi_buffer_bottom, sendcounts, displs, MPI_PACKED,
- recvbuffer, recvcount, MPI_PACKED, root, comm);
- free(recvbuffer);
- free(sendcounts);
- free(displs);
- free(mpi_buffer_bottom);
- message = elt(messages, root);
- }
- else {
- MPI_Scatter(0,0,MPI_LONG,&mpi_pack_size,1,MPI_LONG,root,comm);
- mpi_pack_buffer = (char*)malloc(mpi_pack_size);
- MPI_Scatterv(0,0,0,MPI_PACKED,
- mpi_pack_buffer,mpi_pack_size,MPI_PACKED,root,comm);
- message = unpack_object();
- free(mpi_pack_buffer);
- }
- return onevalue(message);
- }
- /* Allgather: just like gather, only everyone gets the result.
- * (mpi_allgather message comm)
- * returns vector of messages
- */
- static Lisp_Object Lmpi_allgather(Lisp_Object nil,
- Lisp_Object message,
- Lisp_Object comm)
- {
- static char fun_name[] = "mpi_gather";
- int commsize, buffersize, count;
- int *recvcounts, *displs;
- char *recvbuffer;
- check_fix(comm);
- comm = int_of_fixnum(comm);
- CSL_IGNORE(nil);
-
- pack_object(message);
- MPI_Comm_size(comm,&commsize);
- recvcounts = (int*)calloc(commsize, sizeof(int));
- displs = (int*)calloc(commsize+1, sizeof(int));
- MPI_Allgather(&mpi_pack_position, 1, MPI_LONG, recvcounts, 1, MPI_LONG, comm);
-
- displs[0] = 0;
- for (count = 0; count < commsize; ++count)
- displs[count+1] = displs[count] + recvcounts[count];
- recvbuffer = (char*)malloc(displs[commsize]);
- MPI_Allgatherv(mpi_pack_buffer, mpi_pack_position, MPI_PACKED,
- recvbuffer, recvcounts, displs, MPI_PACKED, comm);
- free(mpi_pack_buffer); free(recvcounts); free(displs);
- message = Lmkvect(nil, fixnum_of_int(commsize-1));
- for (count = 0; count < commsize; ++count){
- mpi_pack_buffer = recvbuffer + displs[count];
- mpi_pack_size = recvcounts[count];
- elt(message, count) = unpack_object();
- }
- free(recvbuffer);
- return onevalue(message);
- }
- /* All to all scatter/gather.
- * (mpi_alltoall vector_of_messages comm)
- * returns vector of messages.
- */
- static Lisp_Object Lmpi_alltoall(Lisp_Object nil,
- Lisp_Object smessages, Lisp_Object Lcomm)
- {
- static char fun_name[] = "mpi_alltoall";
- Lisp_Object rmessages;
- int rank,comm, commsize, count;
- int *sendcounts, *recvcounts, *sdispls, *rdispls;
- char* recvbuffer;
- CSL_IGNORE(nil);
- check_fix(Lcomm);
- comm = int_of_fixnum(Lcomm);
- MPI_Comm_size(comm,&commsize);
- sendcounts = (int*)calloc(commsize, sizeof(int));
- recvcounts = (int*)calloc(commsize, sizeof(int));
- sdispls = (int*)calloc(commsize+1, sizeof(int));
- rdispls = (int*)calloc(commsize+1, sizeof(int));
-
- /* Call private functions in mpi_packing for consecutive packs */
- check_buffer = scatter_check_buffer;
- mpi_pack_offset = 0;
- mpi_pack_position = 0;
- mpi_pack_size = 0;
- mpi_buffer_bottom = 0;
- mpi_real_size = 0;
- for (count = 0; count < commsize; ++count){
- pack_cell(elt(smessages,count));
- sendcounts[count] = mpi_pack_position;
- mpi_pack_size -= mpi_pack_position;
- mpi_pack_offset += mpi_pack_position;
- mpi_pack_buffer += mpi_pack_position;
- sdispls[count+1] = mpi_pack_offset;
- mpi_pack_position = 0;
- }
- check_buffer = default_check_buffer;
-
- MPI_Comm_rank(comm,&rank);
- MPI_Alltoall(sendcounts, 1, MPI_LONG, recvcounts, 1, MPI_LONG, comm);
- rdispls[0] = 0;
- for (count = 0; count < commsize; ++count)
- rdispls[count+1] = rdispls[count] + recvcounts[count];
- recvbuffer = (char*)malloc(rdispls[commsize]);
- MPI_Alltoallv(mpi_buffer_bottom, sendcounts, sdispls, MPI_PACKED,
- recvbuffer, recvcounts, rdispls, MPI_PACKED, comm);
- free(mpi_buffer_bottom); free(sendcounts); free(sdispls);
- rmessages = Lmkvect(nil, fixnum_of_int(commsize-1));
- for (count = 0; count < commsize; ++count){
- mpi_pack_buffer = recvbuffer + rdispls[count];
- mpi_pack_size = recvcounts[count];
- elt(rmessages, count) = unpack_object();
- }
- free(recvbuffer); free(recvcounts); free(rdispls);
- return onevalue(rmessages);
- }
- #else /* USE_MPI */
- static Lisp_Object Lmpi_comm_rank(Lisp_Object nil, Lisp_Object comm)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(comm);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_comm_size(Lisp_Object nil, Lisp_Object comm)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(comm);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_send(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_recv(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_sendrecv(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_isend(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_irecv(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_wait(Lisp_Object nil, Lisp_Object request)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(request);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_test(Lisp_Object nil, Lisp_Object request)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(request);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_iprobe(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_probe(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_barrier(Lisp_Object nil, Lisp_Object comm)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(comm);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_bcast(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_gather(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object MS_CDECL Lmpi_scatter(Lisp_Object nil, int nargs, ...)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(nargs);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_allgather(Lisp_Object nil,
- Lisp_Object message,
- Lisp_Object comm)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(message);
- CSL_IGNORE(comm);
- return aerror0("mpi support not built into this version of CSL");
- }
- static Lisp_Object Lmpi_alltoall(Lisp_Object nil,
- Lisp_Object smessages, Lisp_Object Lcomm)
- {
- CSL_IGNORE(nil);
- CSL_IGNORE(smessages);
- CSL_IGNORE(Lcomm);
- return aerror0("mpi support not built into this version of CSL");
- }
- #endif /* USE_MPI */
- setup_type const mpi_setup[] =
- {
- {"mpi_comm_rank", Lmpi_comm_rank, too_many_1, wrong_no_1},
- {"mpi_comm_size", Lmpi_comm_size, too_many_1, wrong_no_1},
- {"mpi_send", wrong_no_0a, wrong_no_0b, Lmpi_send},
- {"mpi_recv", wrong_no_0a, wrong_no_0b, Lmpi_recv},
- {"mpi_sendrecv", wrong_no_0a, wrong_no_0b, Lmpi_sendrecv},
- {"mpi_isend", wrong_no_0a, wrong_no_0b, Lmpi_isend},
- {"mpi_irecv", wrong_no_0a, wrong_no_0b, Lmpi_irecv},
- {"mpi_barrier", Lmpi_barrier, too_many_1, wrong_no_1},
- {"mpi_wait", Lmpi_wait, too_many_1, wrong_no_1},
- {"mpi_test", Lmpi_test, too_many_1, wrong_no_1},
- {"mpi_probe", wrong_no_0a, wrong_no_0b, Lmpi_probe},
- {"mpi_iprobe", wrong_no_0a, wrong_no_0b, Lmpi_iprobe},
- {"mpi_bcast", wrong_no_0a, wrong_no_0b, Lmpi_bcast},
- {"mpi_gather", wrong_no_0a, wrong_no_0b, Lmpi_gather},
- {"mpi_allgather", wrong_no_0a, Lmpi_allgather, wrong_no_2},
- {"mpi_scatter", wrong_no_0a, wrong_no_0b, Lmpi_scatter},
- {"mpi_alltoall", wrong_no_0a, Lmpi_alltoall, wrong_no_2},
- {NULL, 0, 0, 0}
- };
- /* end of cslmpi.c */
|