/* -*- Mode: C; c-basic-offset:4 ; -*- */ /* * (C) 2001 by Argonne National Laboratory. * See COPYRIGHT in top-level directory. */ #include "mpiimpl.h" #include "mpi_init.h" /* -- Begin Profiling Symbol Block for routine MPI_Init */ #if defined(HAVE_PRAGMA_WEAK) #pragma weak MPI_Init = PMPI_Init #elif defined(HAVE_PRAGMA_HP_SEC_DEF) #pragma _HP_SECONDARY_DEF PMPI_Init MPI_Init #elif defined(HAVE_PRAGMA_CRI_DUP) #pragma _CRI duplicate MPI_Init as PMPI_Init #endif /* -- End Profiling Symbol Block */ /* Define MPICH_MPI_FROM_PMPI if weak symbols are not supported to build the MPI routines */ #ifndef MPICH_MPI_FROM_PMPI #undef MPI_Init #define MPI_Init PMPI_Init /* Fortran logical values. extern'd in mpiimpl.h */ /* MPI_Fint MPIR_F_TRUE, MPIR_F_FALSE; */ /* Any internal routines can go here. Make them static if possible */ /* must go inside this #ifdef block to prevent duplicate storage on darwin */ int MPIR_async_thread_initialized = 0; #endif #undef FUNCNAME #define FUNCNAME MPI_Init /*@ MPI_Init - Initialize the MPI execution environment Input Parameters: + argc - Pointer to the number of arguments - argv - Pointer to the argument vector Thread and Signal Safety: This routine must be called by one thread only. That thread is called the `main thread` and must be the thread that calls 'MPI_Finalize'. Notes: The MPI standard does not say what a program can do before an 'MPI_INIT' or after an 'MPI_FINALIZE'. In the MPICH implementation, you should do as little as possible. In particular, avoid anything that changes the external state of the program, such as opening files, reading standard input or writing to standard output. Notes for Fortran: The Fortran binding for 'MPI_Init' has only the error return .vb subroutine MPI_INIT( ierr ) integer ierr .ve .N Errors .N MPI_SUCCESS .N MPI_ERR_INIT .seealso: MPI_Init_thread, MPI_Finalize @*/ int MPI_Init( int *argc, char ***argv ) { static const char FCNAME[] = "MPI_Init"; int mpi_errno = MPI_SUCCESS; int rc; int threadLevel, provided; MPID_MPI_INIT_STATE_DECL(MPID_STATE_MPI_INIT); rc = MPID_Wtime_init(); #ifdef USE_DBG_LOGGING MPIU_DBG_PreInit( argc, argv, rc ); #endif MPID_MPI_INIT_FUNC_ENTER(MPID_STATE_MPI_INIT); # ifdef HAVE_ERROR_CHECKING { MPID_BEGIN_ERROR_CHECKS; { if (MPIR_Process.initialized != MPICH_PRE_INIT) { mpi_errno = MPIR_Err_create_code( MPI_SUCCESS, MPIR_ERR_RECOVERABLE, FCNAME, __LINE__, MPI_ERR_OTHER, "**inittwice", NULL ); } if (mpi_errno) goto fn_fail; } MPID_END_ERROR_CHECKS; } # endif /* HAVE_ERROR_CHECKING */ /* ... body of routine ... */ #if (MPICH_THREAD_LEVEL == MPI_THREAD_MULTIPLE) /* If we support all thread levels, allow the use of an environment variable to set the default thread level */ { const char *str = 0; threadLevel = MPI_THREAD_SINGLE; if (MPL_env2str( "MPICH_THREADLEVEL_DEFAULT", &str )) { if (strcmp(str,"MULTIPLE") == 0 || strcmp(str,"multiple") == 0) { threadLevel = MPI_THREAD_MULTIPLE; } else if (strcmp(str,"SERIALIZED") == 0 || strcmp(str,"serialized") == 0) { threadLevel = MPI_THREAD_SERIALIZED; } else if (strcmp(str,"FUNNELED") == 0 || strcmp(str,"funneled") == 0) { threadLevel = MPI_THREAD_FUNNELED; } else if (strcmp(str,"SINGLE") == 0 || strcmp(str,"single") == 0) { threadLevel = MPI_THREAD_SINGLE; } else { MPIU_Error_printf( "Unrecognized thread level %s\n", str ); exit(1); } } } #else threadLevel = MPI_THREAD_SINGLE; #endif /* If the user requested for asynchronous progress, request for * THREAD_MULTIPLE. */ rc = 0; MPL_env2bool("MPICH_ASYNC_PROGRESS", &rc); if (rc) threadLevel = MPI_THREAD_MULTIPLE; mpi_errno = MPIR_Init_thread( argc, argv, threadLevel, &provided ); if (mpi_errno != MPI_SUCCESS) goto fn_fail; if (rc && provided == MPI_THREAD_MULTIPLE) { mpi_errno = MPIR_Init_async_thread(); if (mpi_errno) goto fn_fail; MPIR_async_thread_initialized = 1; } /* ... end of body of routine ... */ MPID_MPI_INIT_FUNC_EXIT(MPID_STATE_MPI_INIT); return mpi_errno; fn_fail: /* --BEGIN ERROR HANDLING-- */ # ifdef HAVE_ERROR_REPORTING { mpi_errno = MPIR_Err_create_code( mpi_errno, MPIR_ERR_RECOVERABLE, FCNAME, __LINE__, MPI_ERR_OTHER, "**mpi_init", "**mpi_init %p %p", argc, argv); } # endif mpi_errno = MPIR_Err_return_comm( 0, FCNAME, mpi_errno ); return mpi_errno; /* --END ERROR HANDLING-- */ }