2002-02-11 20:40:09 +00:00
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* YAP Prolog *
|
|
|
|
* *
|
|
|
|
* Yap Prolog was developed at NCCUP - Universidade do Porto *
|
|
|
|
* *
|
|
|
|
* Copyright S. Konstantopoulos and Universidade do Porto 2002 *
|
|
|
|
* *
|
|
|
|
**************************************************************************
|
|
|
|
* *
|
|
|
|
* File: mpi.c *
|
2002-11-11 17:38:10 +00:00
|
|
|
* Last rev: $Date: 2002-11-11 17:38:06 $ *
|
2002-02-11 20:40:09 +00:00
|
|
|
* mods: *
|
|
|
|
* comments: Interface to an MPI library *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
|
|
|
|
#ifndef lint
|
2002-11-11 17:38:10 +00:00
|
|
|
static char *rcsid = "$Header: /Users/vitor/Yap/yap-cvsbackup/library/mpi/mpi.c,v 1.14 2002-11-11 17:38:06 vsc Exp $";
|
2002-02-11 20:40:09 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "Yap.h"
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
#if HAVE_MPI
|
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
#include "Yatom.h"
|
|
|
|
#include "yapio.h"
|
|
|
|
|
|
|
|
/* for AtomEof */
|
|
|
|
#include "Heap.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <mpi.h>
|
|
|
|
|
|
|
|
STATIC_PROTO (Int p_mpi_open, (void));
|
|
|
|
STATIC_PROTO (Int p_mpi_close, (void));
|
|
|
|
STATIC_PROTO (Int p_mpi_send, (void));
|
|
|
|
STATIC_PROTO (Int p_mpi_receive, (void));
|
2002-03-12 20:03:55 +00:00
|
|
|
STATIC_PROTO (Int p_mpi_bcast3, (void));
|
|
|
|
STATIC_PROTO (Int p_mpi_bcast2, (void));
|
2002-02-26 15:34:08 +00:00
|
|
|
STATIC_PROTO (Int p_mpi_barrier, (void));
|
2002-02-11 20:40:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Auxiliary Data and Functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
static Int rank, numprocs, namelen;
|
|
|
|
static char processor_name[MPI_MAX_PROCESSOR_NAME];
|
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
/* used by the parser */
|
|
|
|
static int StartLine;
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
static Int mpi_argc;
|
|
|
|
static char **mpi_argv;
|
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
/* mini-stream */
|
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
#define RECV_BUF_SIZE 1024*32
|
2002-02-27 13:41:24 +00:00
|
|
|
|
2002-03-13 09:01:39 +00:00
|
|
|
static size_t bufsize, bufstrlen;
|
2002-02-27 13:41:24 +00:00
|
|
|
static char *buf;
|
2002-03-13 09:01:39 +00:00
|
|
|
static int bufptr;
|
2002-02-27 13:41:24 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
expand_buffer( int space )
|
|
|
|
{
|
2002-10-03 18:28:37 +01:00
|
|
|
#if MPI_AVOID_REALLOC
|
|
|
|
/*
|
|
|
|
realloc() has been SIGSEGV'ing on HP-UX 10.20, but there is
|
|
|
|
no problem in HP-UX 11.0. We can remove this bit here as soon
|
|
|
|
as Yap stops compiling on 10.20 anyway. If removed, also remove
|
|
|
|
the MPI_AVOID_REALLOC bits from configure.in and config.h.in
|
|
|
|
*/
|
2002-02-27 13:41:24 +00:00
|
|
|
|
|
|
|
char *tmp;
|
|
|
|
|
2002-10-31 11:13:21 +00:00
|
|
|
#if 0
|
|
|
|
printf( "expanding by %d (to %d)...", space, (bufsize+space));
|
|
|
|
#endif
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
tmp = malloc( bufsize + space );
|
|
|
|
if( tmp == NULL ) {
|
|
|
|
Error(SYSTEM_ERROR, TermNil, "out of memory" );
|
|
|
|
exit_yap( EXIT_FAILURE );
|
|
|
|
}
|
|
|
|
memcpy( tmp, buf, bufsize );
|
2002-10-31 11:13:21 +00:00
|
|
|
#if 0
|
|
|
|
printf("memcpy'd...");
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
free( buf );
|
2002-10-31 11:13:21 +00:00
|
|
|
#if 0
|
|
|
|
printf("free'd...");
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
buf = tmp;
|
2002-10-03 18:28:37 +01:00
|
|
|
#else /* use realloc */
|
2002-02-27 13:41:24 +00:00
|
|
|
buf = realloc( buf, bufsize + space );
|
2002-10-31 11:13:21 +00:00
|
|
|
#if 0
|
|
|
|
printf("realloc'ed space...");
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
if( buf == NULL ) {
|
2002-10-03 18:28:37 +01:00
|
|
|
Error(SYSTEM_ERROR, TermNil, "out of memory");
|
2002-02-27 13:41:24 +00:00
|
|
|
exit_yap( EXIT_FAILURE );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bufsize += space;
|
2002-10-31 11:13:21 +00:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
printf("SUCCESS\n");
|
|
|
|
printf( "New bufsize: %d\n", bufsize );
|
|
|
|
buf[bufsize-space] = 0;
|
|
|
|
printf("Buffer contents: %s\n", buf);
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
}
|
2002-02-11 20:40:09 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
mpi_putc(Int stream, Int ch)
|
|
|
|
{
|
2002-10-03 18:28:37 +01:00
|
|
|
#if 0
|
|
|
|
printf("%d: PUTC %d a.k.a. %c at %d\n", rank, ch, (char)ch, bufptr);
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
if( ch > 0 ) {
|
|
|
|
if( bufptr >= bufsize ) expand_buffer( RECV_BUF_SIZE );
|
2002-02-11 20:40:09 +00:00
|
|
|
buf[bufptr++] = ch;
|
2002-02-27 13:41:24 +00:00
|
|
|
}
|
2002-02-11 20:40:09 +00:00
|
|
|
return ch;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Int
|
|
|
|
mpi_getc(Int stream)
|
|
|
|
{
|
2002-10-03 18:28:37 +01:00
|
|
|
#if 0
|
|
|
|
printf("%d: GETC %c at %d\n", rank, buf[bufptr], bufptr);
|
|
|
|
#endif
|
2002-02-11 20:40:09 +00:00
|
|
|
return buf[bufptr++];
|
|
|
|
}
|
|
|
|
|
|
|
|
static Int
|
|
|
|
mpi_eob(void)
|
|
|
|
{
|
2002-02-27 13:41:24 +00:00
|
|
|
return (bufptr<bufstrlen) && (buf[bufptr] != EOF);
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
|
2002-02-12 17:38:38 +00:00
|
|
|
/* Term parser */
|
|
|
|
|
|
|
|
static Term
|
|
|
|
mpi_parse(void)
|
|
|
|
{
|
2002-11-05 11:14:08 +00:00
|
|
|
Term t;
|
2002-02-12 17:38:38 +00:00
|
|
|
TokEntry *tokstart;
|
2002-10-29 14:02:43 +00:00
|
|
|
tr_fr_ptr old_TR, TR_before_parse;
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
old_TR = TR;
|
|
|
|
while( TRUE ) {
|
2002-10-29 13:58:21 +00:00
|
|
|
CELL *old_H;
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
/* Scans the term using stack space */
|
|
|
|
eot_before_eof = FALSE;
|
|
|
|
|
|
|
|
/* the first arg is the getc_for_read, diff only if CharConv is on */
|
2002-11-05 11:14:08 +00:00
|
|
|
tokstart = tokptr = toktide = tokenizer(mpi_getc, mpi_getc);
|
2002-02-12 17:38:38 +00:00
|
|
|
|
2002-10-29 13:58:21 +00:00
|
|
|
/* preserve value of H after scanning: otherwise we may lose strings
|
|
|
|
and floats */
|
|
|
|
old_H = H;
|
|
|
|
|
2002-02-12 17:38:38 +00:00
|
|
|
if ( mpi_eob() && !eot_before_eof) {
|
|
|
|
if (tokstart != NIL && tokstart->Tok != Ord (eot_tok)) {
|
|
|
|
/* we got the end of file from an abort */
|
|
|
|
if (ErrorMessage == "Abort") {
|
|
|
|
TR = old_TR;
|
2002-03-12 20:03:55 +00:00
|
|
|
return TermNil;
|
2002-02-12 17:38:38 +00:00
|
|
|
}
|
|
|
|
/* we need to force the next reading to also give end of file.*/
|
|
|
|
buf[bufptr] = EOF;
|
|
|
|
ErrorMessage = "[ Error: end of file found before end of term ]";
|
|
|
|
} else {
|
|
|
|
/* restore TR */
|
|
|
|
TR = old_TR;
|
2002-11-05 11:14:08 +00:00
|
|
|
|
|
|
|
return (unify_constant(t, MkAtomTerm(AtomEof)));
|
2002-02-12 17:38:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
repeat_cycle:
|
2002-10-29 14:02:43 +00:00
|
|
|
TR_before_parse = TR;
|
2002-11-05 11:14:08 +00:00
|
|
|
if( ErrorMessage || (t = Parse())==0 ) {
|
2002-02-12 17:38:38 +00:00
|
|
|
if (ErrorMessage && (strcmp(ErrorMessage,"Stack Overflow") == 0)) {
|
|
|
|
/* ignore term we just built */
|
2002-10-29 14:02:43 +00:00
|
|
|
TR = TR_before_parse;
|
2002-02-12 17:38:38 +00:00
|
|
|
H = old_H;
|
|
|
|
if (growstack_in_parser(&old_TR, &tokstart, &VarTable)) {
|
2002-10-29 13:58:21 +00:00
|
|
|
old_H = H;
|
2002-02-12 17:38:38 +00:00
|
|
|
tokptr = toktide = tokstart;
|
|
|
|
ErrorMessage = NULL;
|
|
|
|
goto repeat_cycle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TR = old_TR;
|
2002-11-05 11:14:08 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
behave as if ParserErrorStyle were QUIET_ON_PARSER_ERROR,
|
|
|
|
(see iopreds.c), except with bombing Yap instead of simply
|
|
|
|
failing the predicate: the parse cannot fail unless there
|
|
|
|
is a problem with MPI or the pretty printer.
|
|
|
|
*/
|
|
|
|
Error(SYSTEM_ERROR, TermNil, "Failed to parse MPI_Recv()'ed term" );
|
|
|
|
exit_yap( EXIT_FAILURE );
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
/* parsing succeeded */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
TR = old_TR;
|
2002-02-12 17:38:38 +00:00
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* C Predicates
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
static Int
|
|
|
|
p_mpi_open(void) /* mpi_open(?rank, ?num_procs, ?proc_name) */
|
|
|
|
{
|
2002-02-22 14:31:45 +00:00
|
|
|
Term t_rank = Deref(ARG1), t_numprocs = Deref(ARG2), t_procname = Deref(ARG3);
|
|
|
|
Int retv;
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-10-31 11:13:21 +00:00
|
|
|
/*
|
|
|
|
With MPICH MPI_Init() must be called during initialisation,
|
|
|
|
but with LAM it can be called from Prolog (mpi_open/3)
|
|
|
|
|
|
|
|
The symptoms match a known RedHat bug, see
|
|
|
|
http://email.osc.edu/pipermail/mpiexec/2002-July/000067.html
|
|
|
|
for a suggested workaround:
|
|
|
|
Redhat have somehow broken their sem.h and ipc.h. If you use your own
|
|
|
|
kernel, copy from ../src/kernel/include/asm & ../src/kernel/include/linux
|
|
|
|
the file ipc.h and sem.h to /usr/include/sys, recompile your mpich and
|
|
|
|
everything might start working. (it did for us)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Note that if MPI_Init() fails, Yap/MPICH and Yap/LAM bahave differently:
|
|
|
|
in Yap/MPICH we are still at the Yap initialisation phase, so we let
|
|
|
|
Yap exit(FAILURE), whereas in Yap/LAM mpi_open/3 simply fails.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if ! HAVE_LIBMPICH
|
|
|
|
retv = MPI_Init( &mpi_argc, &mpi_argv );
|
|
|
|
if( retv ) {
|
|
|
|
Term t;
|
|
|
|
|
|
|
|
t = MkIntegerTerm(retv);
|
|
|
|
Error( SYSTEM_ERROR, t, "MPI_Init() returned non-zero" );
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
MPI_Comm_size( MPI_COMM_WORLD, &numprocs );
|
|
|
|
MPI_Comm_rank( MPI_COMM_WORLD, &rank );
|
|
|
|
MPI_Get_processor_name( processor_name, &namelen );
|
|
|
|
|
2002-02-22 14:31:45 +00:00
|
|
|
retv = unify(t_rank, MkIntTerm(rank));
|
|
|
|
retv = retv && unify(t_numprocs, MkIntTerm(numprocs));
|
|
|
|
retv = retv && unify(t_procname, MkAtomTerm(LookupAtom(processor_name)));
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-02-22 14:31:45 +00:00
|
|
|
return retv;
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Int /* mpi_close */
|
|
|
|
p_mpi_close()
|
|
|
|
{
|
|
|
|
MPI_Finalize();
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Int
|
|
|
|
p_mpi_send() /* mpi_send(+data, +destination, +tag) */
|
|
|
|
{
|
|
|
|
Term t_data = Deref(ARG1), t_dest = Deref(ARG2), t_tag = Deref(ARG3);
|
|
|
|
int tag, dest, retv;
|
|
|
|
|
|
|
|
/* The first argument (data) must be bound */
|
|
|
|
if (IsVarTerm(t_data)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_send");
|
|
|
|
return (FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The second and third args must be bount to integers */
|
|
|
|
if (IsVarTerm(t_dest)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_dest, "mpi_send");
|
|
|
|
return (FALSE);
|
|
|
|
} else if( !IsIntegerTerm(t_dest) ) {
|
|
|
|
Error(TYPE_ERROR_INTEGER, t_dest, "mpi_send");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
dest = IntOfTerm( t_dest );
|
|
|
|
}
|
|
|
|
if (IsVarTerm(t_tag)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_tag, "mpi_send");
|
|
|
|
return (FALSE);
|
|
|
|
} else if( !IsIntegerTerm(t_tag) ) {
|
|
|
|
Error(TYPE_ERROR_INTEGER, t_tag, "mpi_send");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
tag = IntOfTerm( t_tag );
|
|
|
|
}
|
|
|
|
|
|
|
|
bufptr = 0;
|
|
|
|
/* Turn the term into its ASCII representation */
|
|
|
|
plwrite( t_data, mpi_putc, 5 );
|
2002-10-03 18:28:37 +01:00
|
|
|
bufstrlen = (size_t)bufptr;
|
2002-02-27 13:41:24 +00:00
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
/* The buf is not NULL-terminated and does not have the
|
|
|
|
trailing ". " required by the parser */
|
|
|
|
mpi_putc( 0, '.' );
|
|
|
|
mpi_putc( 0, ' ' );
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
buf[bufptr] = 0;
|
|
|
|
bufstrlen = bufptr + 1;
|
|
|
|
bufptr = 0;
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
FILE *debug_out;
|
|
|
|
debug_out = fopen("debug.out", "a");
|
|
|
|
fprintf(debug_out, "%d: About to send %d chars to %d\n",
|
|
|
|
rank, bufstrlen, dest);
|
|
|
|
fclose(debug_out);
|
|
|
|
}
|
|
|
|
#endif
|
2002-10-03 18:28:37 +01:00
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
/* send the data */
|
2002-10-03 18:28:37 +01:00
|
|
|
retv = MPI_Send( &buf[bufptr], bufstrlen, MPI_CHAR, dest, tag, MPI_COMM_WORLD );
|
|
|
|
if( retv != MPI_SUCCESS ) return FALSE;
|
2002-02-27 13:41:24 +00:00
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
FILE *debug_out;
|
|
|
|
debug_out = fopen("debug.out", "a");
|
|
|
|
fprintf(debug_out, "%d: Sent %s to %d\n", rank, &buf[bufptr], dest);
|
|
|
|
fclose(debug_out);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
return TRUE;
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Int
|
|
|
|
p_mpi_receive() /* mpi_receive(-data, ?orig, ?tag) */
|
|
|
|
{
|
2002-02-12 17:38:38 +00:00
|
|
|
Term t_data = Deref(ARG1), t_orig = Deref(ARG2), t_tag = Deref(ARG3);
|
2002-02-11 20:40:09 +00:00
|
|
|
int tag, orig, retv;
|
|
|
|
MPI_Status status;
|
|
|
|
|
2002-02-12 17:38:38 +00:00
|
|
|
/* The first argument (data) must be unbound */
|
|
|
|
if(!IsVarTerm(t_data)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_receive");
|
|
|
|
return FALSE;
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
2002-02-12 17:38:38 +00:00
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
/* The second argument (source) must be bound to an integer
|
|
|
|
(the rank of the source) or left unbound (i.e. any source
|
|
|
|
is OK) */
|
|
|
|
if (IsVarTerm(t_orig)) {
|
|
|
|
orig = MPI_ANY_SOURCE;
|
|
|
|
} else if( !IsIntegerTerm(t_orig) ) {
|
|
|
|
Error(TYPE_ERROR_INTEGER, t_orig, "mpi_receive");
|
|
|
|
return (FALSE);
|
|
|
|
} else {
|
|
|
|
orig = IntOfTerm( t_orig );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The third argument must be bound to an integer (the tag)
|
2002-02-12 17:38:38 +00:00
|
|
|
or left unbound (i.e. any tag is OK) */
|
2002-02-11 20:40:09 +00:00
|
|
|
if (IsVarTerm(t_tag)) {
|
|
|
|
tag = MPI_ANY_TAG;
|
|
|
|
} else if( !IsIntegerTerm(t_tag) ) {
|
|
|
|
Error(TYPE_ERROR_INTEGER, t_tag, "mpi_receive");
|
|
|
|
return (FALSE);
|
|
|
|
} else
|
|
|
|
tag = IntOfTerm( t_tag );
|
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
/* probe for the size of the term */
|
|
|
|
retv = MPI_Probe( orig, tag, MPI_COMM_WORLD, &status );
|
2002-10-03 18:28:37 +01:00
|
|
|
if( retv != MPI_SUCCESS ) {
|
|
|
|
return FALSE;
|
2002-02-27 13:41:24 +00:00
|
|
|
}
|
2002-11-05 11:14:08 +00:00
|
|
|
MPI_Get_count( &status, MPI_CHAR, &bufstrlen );
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
#if 1
|
|
|
|
{
|
|
|
|
FILE *debug_out;
|
|
|
|
debug_out = fopen("debug.out", "a");
|
|
|
|
fprintf(debug_out, "%d: About to receive %d chars from %d\n",
|
|
|
|
rank, bufstrlen, orig);
|
|
|
|
fclose(debug_out);
|
|
|
|
}
|
2002-10-03 18:28:37 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* adjust the buffer */
|
|
|
|
if( bufsize < bufstrlen ) expand_buffer(bufstrlen-bufsize);
|
2002-02-11 20:40:09 +00:00
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
/* Already know the source from MPI_Probe() */
|
2002-10-03 18:28:37 +01:00
|
|
|
if( orig == MPI_ANY_SOURCE ) {
|
|
|
|
orig = status.MPI_SOURCE;
|
|
|
|
retv = unify(t_orig, MkIntTerm(orig));
|
|
|
|
if( retv == FALSE ) puts( "PROBLEM1" );
|
|
|
|
}
|
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
/* Already know the tag from MPI_Probe() */
|
2002-10-03 18:28:37 +01:00
|
|
|
if( tag == MPI_ANY_TAG ) {
|
|
|
|
tag = status.MPI_TAG;
|
|
|
|
retv = unify(t_tag, MkIntTerm(status.MPI_TAG));
|
|
|
|
if( retv == FALSE ) puts( "PROBLEM2" );
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Receive the message as a C string */
|
|
|
|
retv = MPI_Recv( &buf[bufptr], bufstrlen, MPI_CHAR, orig, tag,
|
|
|
|
MPI_COMM_WORLD, &status );
|
|
|
|
if( retv != MPI_SUCCESS ) {
|
2002-11-05 11:14:08 +00:00
|
|
|
/* Getting in here would be weird; it means the first package
|
|
|
|
(size) was sent properly, but there was a glitch with
|
|
|
|
the actual content! */
|
2002-10-03 18:28:37 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2002-10-03 18:32:40 +01:00
|
|
|
#if 0
|
2002-10-03 18:28:37 +01:00
|
|
|
{
|
|
|
|
int aa;
|
2002-11-05 11:14:08 +00:00
|
|
|
FILE *debug_out;
|
2002-10-03 18:28:37 +01:00
|
|
|
MPI_Get_count( &status, MPI_CHAR, &aa );
|
2002-11-05 11:14:08 +00:00
|
|
|
debug_out = fopen("debug.out", "a");
|
|
|
|
fprintf(debug_out, "%d: Received %d chars from %d\n\
|
|
|
|
%d: The message was: %s\n", rank, aa, orig, rank, &buf[bufptr]);
|
|
|
|
fclose(debug_out);
|
2002-10-03 18:28:37 +01:00
|
|
|
}
|
|
|
|
#endif
|
2002-02-27 13:41:24 +00:00
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
/* parse received string into a Prolog term */
|
2002-10-03 18:28:37 +01:00
|
|
|
bufptr = 0;
|
2002-11-05 11:14:08 +00:00
|
|
|
retv = unify(ARG1, mpi_parse());
|
|
|
|
|
2002-10-03 18:32:40 +01:00
|
|
|
#if 0
|
2002-11-05 11:14:08 +00:00
|
|
|
/* check up on mpi_parse():
|
|
|
|
convert the newly-parsed term back to text and print */
|
|
|
|
bufptr = 0;
|
|
|
|
plwrite( t_data, mpi_putc, 5 );
|
|
|
|
mpi_putc( 0, '.' );
|
|
|
|
mpi_putc( 0, ' ' );
|
|
|
|
buf[bufptr] = 0;
|
|
|
|
bufptr = 0;
|
|
|
|
{
|
|
|
|
FILE *debug_out;
|
|
|
|
debug_out = fopen("debug.out", "a");
|
|
|
|
fprintf(debug_out, "%d: mpi_receive: t_data == %d, retv == %d term == %s\n",
|
|
|
|
rank, t_data, retv, buf);
|
|
|
|
fclose(debug_out);
|
|
|
|
}
|
2002-10-03 18:28:37 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return retv;
|
2002-02-12 17:38:38 +00:00
|
|
|
}
|
2002-02-11 20:40:09 +00:00
|
|
|
|
|
|
|
|
2002-02-12 17:38:38 +00:00
|
|
|
static Int
|
2002-03-12 20:03:55 +00:00
|
|
|
p_mpi_bcast3() /* mpi_bcast( ?data, +root, +max_size ) */
|
2002-02-12 17:38:38 +00:00
|
|
|
{
|
2002-02-27 13:41:24 +00:00
|
|
|
Term t_data = Deref(ARG1), t_root = Deref(ARG2), t_max_size = Deref(ARG3);
|
|
|
|
int root, retv, max_size;
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
/* The second argument must be bound to an integer (the rank of
|
|
|
|
root processor */
|
|
|
|
if (IsVarTerm(t_root)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_root, "mpi_bcast");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
root = IntOfTerm( t_root );
|
|
|
|
|
|
|
|
/* If this is the root processor, then the first argument must
|
|
|
|
be bound to the term to be sent. */
|
|
|
|
if( root == rank ) {
|
|
|
|
if( IsVarTerm(t_data) ) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_bcast");
|
|
|
|
return FALSE;
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
2002-02-12 17:38:38 +00:00
|
|
|
bufptr = 0;
|
|
|
|
/* Turn the term into its ASCII representation */
|
|
|
|
plwrite( t_data, mpi_putc, 5 );
|
|
|
|
/* NULL-terminate the string and add the ". " termination
|
|
|
|
required by the parser. */
|
|
|
|
buf[bufptr] = 0;
|
|
|
|
strcat( buf, ". " );
|
2002-03-12 20:03:55 +00:00
|
|
|
bufstrlen = bufptr + 2;
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
2002-02-12 17:38:38 +00:00
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
/* The third argument must be bound to an integer (the maximum length
|
|
|
|
of the broadcast term's ASCII representation */
|
|
|
|
if (IsVarTerm(t_max_size)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_max_size, "mpi_bcast");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/* allow for the ". " bit and the NULL at the end */
|
|
|
|
max_size = IntOfTerm( t_max_size ) + 3;
|
|
|
|
|
|
|
|
#if 0
|
2002-03-12 20:03:55 +00:00
|
|
|
if( (rank == root) && (max_size < bufstrlen) )
|
2002-02-27 13:41:24 +00:00
|
|
|
/* issue a warning? explode? bcast s'thing unparsable? */
|
2002-03-13 09:01:39 +00:00
|
|
|
printf( "MAYDAY: max_size == %d, bufstrlen == %d\n", max_size, bufstrlen );
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-02-27 13:41:24 +00:00
|
|
|
#endif
|
2002-03-13 09:01:39 +00:00
|
|
|
printf( "%d: About to Bcast(): max_size == %d, bufstrlen == %d\n",
|
|
|
|
rank, max_size, bufstrlen );
|
2002-02-27 13:41:24 +00:00
|
|
|
|
|
|
|
/* adjust the buffer size, if necessary */
|
|
|
|
if( max_size > bufsize ) {
|
|
|
|
expand_buffer( max_size - bufsize );
|
|
|
|
}
|
|
|
|
|
|
|
|
retv = MPI_Bcast( buf, max_size, MPI_CHAR, root, MPI_COMM_WORLD );
|
2002-03-13 09:01:39 +00:00
|
|
|
if( retv != MPI_SUCCESS ) {
|
|
|
|
printf( "OOOPS! MPI_Bcast() returned %d.\n", retv );
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf( "%d: I'm just after Bcast()ing. strlen(buf) == %d\n",
|
|
|
|
rank, strlen(buf) );
|
2002-02-12 17:38:38 +00:00
|
|
|
|
|
|
|
if( root == rank ) return TRUE;
|
|
|
|
else {
|
|
|
|
/* ARG1 must be unbound so that it can receive data */
|
|
|
|
if( !IsVarTerm(t_data) ) {
|
2002-03-13 09:01:39 +00:00
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_bcast");
|
2002-02-12 17:38:38 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
bufstrlen = strlen(buf);
|
2002-02-12 17:38:38 +00:00
|
|
|
bufptr = 0;
|
|
|
|
|
|
|
|
/* parse received string into a Prolog term */
|
|
|
|
return unify(mpi_parse(), ARG1);
|
|
|
|
}
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
/*
|
|
|
|
This is the same as above, but for dynamic data size.
|
|
|
|
It is implemented as two broadcasts, the first being the size
|
|
|
|
and the second the actual data.
|
2002-03-12 20:03:55 +00:00
|
|
|
*/
|
|
|
|
static Int
|
|
|
|
p_mpi_bcast2() /* mpi_bcast( ?data, +root ) */
|
|
|
|
{
|
|
|
|
Term t_data = Deref(ARG1), t_root = Deref(ARG2);
|
|
|
|
int root, retv;
|
|
|
|
|
|
|
|
/* The second argument must be bound to an integer (the rank of
|
|
|
|
root processor */
|
|
|
|
if (IsVarTerm(t_root)) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_root, "mpi_bcast");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
root = IntOfTerm( t_root );
|
|
|
|
|
2002-10-03 18:28:37 +01:00
|
|
|
|
2002-03-12 20:03:55 +00:00
|
|
|
/* If this is the root processor, then the first argument must
|
|
|
|
be bound to the term to be sent. */
|
|
|
|
if( root == rank ) {
|
|
|
|
if( IsVarTerm(t_data) ) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_bcast");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
bufptr = 0;
|
|
|
|
/* Turn the term into its ASCII representation */
|
|
|
|
plwrite( t_data, mpi_putc, 5 );
|
|
|
|
/* NULL-terminate the string and add the ". " termination
|
|
|
|
required by the parser. */
|
|
|
|
buf[bufptr] = 0;
|
|
|
|
strcat( buf, ". " );
|
|
|
|
bufstrlen = bufptr + 2;
|
|
|
|
}
|
2002-10-03 18:28:37 +01:00
|
|
|
/* Otherwise, it must a variable */
|
|
|
|
else {
|
|
|
|
if( !IsVarTerm(t_data) ) {
|
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_bcast");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-12 20:03:55 +00:00
|
|
|
|
|
|
|
/* Broadcast the data size */
|
|
|
|
retv = MPI_Bcast( &bufstrlen, sizeof bufstrlen, MPI_INT, root, MPI_COMM_WORLD );
|
2002-10-03 18:28:37 +01:00
|
|
|
if( retv != MPI_SUCCESS ) {
|
|
|
|
printf("PROBLEM: file %s, line %d\n", __FILE__, __LINE__);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-03-12 20:03:55 +00:00
|
|
|
|
|
|
|
/* adjust the buffer size, if necessary */
|
2002-03-13 09:01:39 +00:00
|
|
|
if( bufstrlen > bufsize ) {
|
|
|
|
printf("expanding by %d\n", (bufstrlen-bufsize) );
|
2002-03-12 20:03:55 +00:00
|
|
|
expand_buffer( bufstrlen - bufsize );
|
|
|
|
}
|
2002-10-03 18:28:37 +01:00
|
|
|
else {
|
|
|
|
printf("bufstrlen: %d, bufsize %d: not expanding\n",bufstrlen,bufsize);
|
|
|
|
}
|
2002-03-12 20:03:55 +00:00
|
|
|
/* Broadcast the data */
|
|
|
|
retv = MPI_Bcast( buf, bufstrlen, MPI_CHAR, root, MPI_COMM_WORLD );
|
2002-10-03 18:28:37 +01:00
|
|
|
if( retv != MPI_SUCCESS ) {
|
|
|
|
printf("PROBLEM: file %s, line %d\n", __FILE__, __LINE__);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-03-12 20:03:55 +00:00
|
|
|
|
|
|
|
if( root == rank ) return TRUE;
|
|
|
|
else {
|
|
|
|
/* ARG1 must be unbound so that it can receive data */
|
|
|
|
if( !IsVarTerm(t_data) ) {
|
2002-03-13 09:01:39 +00:00
|
|
|
Error(INSTANTIATION_ERROR, t_data, "mpi_bcast");
|
2002-03-12 20:03:55 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
bufstrlen = strlen(buf);
|
|
|
|
bufptr = 0;
|
|
|
|
|
|
|
|
/* parse received string into a Prolog term */
|
|
|
|
return unify(mpi_parse(), ARG1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-02-26 15:34:08 +00:00
|
|
|
static Int
|
|
|
|
p_mpi_barrier() /* mpi_barrier/0 */
|
|
|
|
{
|
|
|
|
int retv;
|
|
|
|
|
|
|
|
retv = MPI_Barrier( MPI_COMM_WORLD );
|
|
|
|
|
|
|
|
return (retv == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-02-12 17:38:38 +00:00
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
/*
|
|
|
|
* Init
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2002-11-11 17:38:10 +00:00
|
|
|
_YAP_InitMPI(void)
|
2002-02-11 20:40:09 +00:00
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
|
|
|
|
mpi_argv = malloc( yap_argc * sizeof(char *) );
|
|
|
|
mpi_argv[0] = strdup( yap_args[0] );
|
|
|
|
|
2002-02-27 13:41:24 +00:00
|
|
|
bufsize = RECV_BUF_SIZE;
|
|
|
|
buf = malloc(bufsize * sizeof(char));
|
|
|
|
|
2002-02-11 20:40:09 +00:00
|
|
|
for( i=1; i<yap_argc; ++i ) {
|
|
|
|
if( !strcmp(yap_args[i], "--") ) { ++i; break; }
|
|
|
|
}
|
|
|
|
for( j=1; i<yap_argc; ++i, ++j ) {
|
|
|
|
mpi_argv[j] = strdup( yap_args[i] );
|
|
|
|
}
|
|
|
|
mpi_argc = j;
|
|
|
|
|
|
|
|
mpi_argv[0] = strdup( yap_args[0] );
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* DEBUG */
|
|
|
|
printf( "yap_argc = %d\n", yap_argc );
|
|
|
|
for( i=0; i<yap_argc; ++i ) {
|
|
|
|
printf( "%d %s\n", i, yap_args[i] );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* DEBUG */
|
|
|
|
printf( "mpi_argc = %d\n", mpi_argc );
|
|
|
|
for( i=0; i<mpi_argc; ++i ) {
|
|
|
|
printf( "%d %s\n", i, mpi_argv[i] );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-11-11 17:38:10 +00:00
|
|
|
/* With MPICH MPI__YAP_Init() must be called during initialisation,
|
2002-10-31 11:13:21 +00:00
|
|
|
but with LAM it can be called from Prolog (mpi_open/3).
|
|
|
|
See also the comment at "if ! HAVE_LIBMPICH" above!
|
|
|
|
*/
|
|
|
|
#if HAVE_LIBMPICH
|
|
|
|
{
|
|
|
|
int retv;
|
|
|
|
|
|
|
|
retv = MPI_Init(&mpi_argc, &mpi_argv);
|
|
|
|
if( retv ) {
|
|
|
|
Term t;
|
|
|
|
|
2002-11-05 11:14:08 +00:00
|
|
|
t = MkIntegerTerm(retv);
|
2002-10-31 11:13:21 +00:00
|
|
|
Error(SYSTEM_ERROR, t, "MPI_Init() returned non-zero");
|
|
|
|
exit_yap( EXIT_FAILURE );
|
|
|
|
}
|
2002-11-05 11:14:08 +00:00
|
|
|
#if 0
|
2002-10-31 11:13:21 +00:00
|
|
|
/* DEBUG */
|
|
|
|
else {
|
|
|
|
puts("MPI_Init() is happy!");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-11-11 17:38:10 +00:00
|
|
|
_YAP_InitCPred( "mpi_open", 3, p_mpi_open, SafePredFlag|SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_close", 0, p_mpi_close, SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_send", 3, p_mpi_send, SafePredFlag|SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_receive", 3, p_mpi_receive, SafePredFlag|SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_bcast", 3, p_mpi_bcast3, SafePredFlag|SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_bcast", 2, p_mpi_bcast2, SafePredFlag|SyncPredFlag );
|
|
|
|
_YAP_InitCPred( "mpi_barrier", 0, p_mpi_barrier, SyncPredFlag );
|
2002-02-11 20:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* HAVE_MPI */
|