
The C++ bindings were removed a while ago; MPI::ERRORS_THROW_EXCEPTIONS and MPI_ERRORS_THROW_EXCEPTIONS no longer exist. Signed-off-by: Jeff Squyres <jsquyres@cisco.com>
390 строки
14 KiB
C
390 строки
14 KiB
C
/* -*- Mode: C; c-basic-offset:4 ; -*- */
|
|
/*
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
* University Research and Technology
|
|
* Corporation. All rights reserved.
|
|
* Copyright (c) 2004-2020 The University of Tennessee and The University
|
|
* of Tennessee Research Foundation. All rights
|
|
* reserved.
|
|
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
* University of Stuttgart. All rights reserved.
|
|
* Copyright (c) 2004-2005 The Regents of the University of California.
|
|
* All rights reserved.
|
|
* Copyright (c) 2008-2018 Cisco Systems, Inc. All rights reserved
|
|
* Copyright (c) 2009 Sun Microsystems, Inc. All rights reserved.
|
|
* Copyright (c) 2015 Research Organization for Information Science
|
|
* and Technology (RIST). All rights reserved.
|
|
* Copyright (c) 2015-2019 Intel, Inc. All rights reserved.
|
|
* $COPYRIGHT$
|
|
*
|
|
* Additional copyrights may follow
|
|
*
|
|
* $HEADER$
|
|
*/
|
|
|
|
#include "ompi_config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "ompi/communicator/communicator.h"
|
|
#include "ompi/win/win.h"
|
|
#include "ompi/errhandler/errhandler.h"
|
|
#include "ompi/errhandler/errhandler_predefined.h"
|
|
#include "opal/class/opal_pointer_array.h"
|
|
#include "opal/mca/pmix/pmix-internal.h"
|
|
#include "opal/util/string_copy.h"
|
|
|
|
|
|
/*
|
|
* Table for Fortran <-> C errhandler handle conversion
|
|
*/
|
|
opal_pointer_array_t ompi_errhandler_f_to_c_table = {{0}};
|
|
|
|
/*
|
|
* default errhandler id
|
|
*/
|
|
static size_t default_errhandler_id = SIZE_MAX;
|
|
|
|
/*
|
|
* Class information
|
|
*/
|
|
static void ompi_errhandler_construct(ompi_errhandler_t *eh);
|
|
static void ompi_errhandler_destruct(ompi_errhandler_t *eh);
|
|
|
|
|
|
/*
|
|
* Class instance
|
|
*/
|
|
OBJ_CLASS_INSTANCE(ompi_errhandler_t, opal_object_t, ompi_errhandler_construct,
|
|
ompi_errhandler_destruct);
|
|
|
|
|
|
/*
|
|
* _addr flavors are for F03 bindings
|
|
*/
|
|
ompi_predefined_errhandler_t ompi_mpi_errhandler_null = {{{0}}};
|
|
ompi_predefined_errhandler_t *ompi_mpi_errhandler_null_addr =
|
|
&ompi_mpi_errhandler_null;
|
|
ompi_predefined_errhandler_t ompi_mpi_errors_are_fatal = {{{0}}};
|
|
ompi_predefined_errhandler_t *ompi_mpi_errors_are_fatal_addr =
|
|
&ompi_mpi_errors_are_fatal;
|
|
ompi_predefined_errhandler_t ompi_mpi_errors_abort = {{{0}}};
|
|
ompi_predefined_errhandler_t *ompi_mpi_errors_abort_addr =
|
|
&ompi_mpi_errors_abort;
|
|
ompi_predefined_errhandler_t ompi_mpi_errors_return = {{{0}}};
|
|
ompi_predefined_errhandler_t *ompi_mpi_errors_return_addr =
|
|
&ompi_mpi_errors_return;
|
|
|
|
static opal_mutex_t errhandler_init_lock = OPAL_MUTEX_STATIC_INIT;
|
|
ompi_errhandler_t* ompi_initial_error_handler_eh = NULL;
|
|
void (*ompi_initial_error_handler)(struct ompi_communicator_t **comm, int *error_code, ...) = NULL;
|
|
|
|
/*
|
|
* Initialize the initial errhandler infrastructure only.
|
|
* This does not allocate any memory and does not require a corresponding fini.
|
|
*/
|
|
int ompi_initial_errhandler_init(void) {
|
|
opal_mutex_lock(&errhandler_init_lock);
|
|
if ( NULL != ompi_initial_error_handler ) {
|
|
/* Already initialized (presumably by an API call before MPI_init) */
|
|
opal_mutex_unlock(&errhandler_init_lock);
|
|
return OMPI_SUCCESS;
|
|
}
|
|
|
|
/* If it has been requested from the launch keys, set the initial
|
|
* error handler that will be attached by default with predefined
|
|
* communicators. We use an env because that can be obtained before
|
|
* OPAL and PMIx initialization.
|
|
*/
|
|
char *env = getenv("OMPI_MCA_mpi_initial_errhandler");
|
|
if( NULL != env ) {
|
|
if( 0 == strcasecmp(env, "mpi_errors_are_fatal") ) {
|
|
ompi_initial_error_handler = &ompi_mpi_errors_are_fatal_comm_handler;
|
|
ompi_initial_error_handler_eh = &ompi_mpi_errors_are_fatal.eh;
|
|
}
|
|
else if( 0 == strcasecmp(env, "mpi_errors_abort") ) {
|
|
ompi_initial_error_handler = &ompi_mpi_errors_abort_comm_handler;
|
|
ompi_initial_error_handler_eh = &ompi_mpi_errors_abort.eh;
|
|
}
|
|
else if( 0 == strcasecmp(env, "mpi_errors_return") ) {
|
|
ompi_initial_error_handler = &ompi_mpi_errors_return_comm_handler;
|
|
ompi_initial_error_handler_eh = &ompi_mpi_errors_return.eh;
|
|
}
|
|
else {
|
|
/* invalid entry detected, ignore it, set fatal by default */
|
|
opal_output(0, "WARNING: invalid value for launch key 'mpi_initial_errhandler'; defaulting to 'mpi_errors_are_fatal'.");
|
|
ompi_initial_error_handler = &ompi_mpi_errors_are_fatal_comm_handler;
|
|
ompi_initial_error_handler_eh = &ompi_mpi_errors_are_fatal.eh;
|
|
}
|
|
}
|
|
else {
|
|
ompi_initial_error_handler = &ompi_mpi_errors_are_fatal_comm_handler;
|
|
ompi_initial_error_handler_eh = &ompi_mpi_errors_are_fatal.eh;
|
|
}
|
|
opal_mutex_unlock(&errhandler_init_lock);
|
|
return OMPI_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Initialize OMPI errhandler infrastructure
|
|
*/
|
|
int ompi_errhandler_init(void)
|
|
{
|
|
/* initialize ompi_errhandler_f_to_c_table */
|
|
|
|
OBJ_CONSTRUCT( &ompi_errhandler_f_to_c_table, opal_pointer_array_t);
|
|
if( OPAL_SUCCESS != opal_pointer_array_init(&ompi_errhandler_f_to_c_table, 8,
|
|
OMPI_FORTRAN_HANDLE_MAX, 16) ) {
|
|
return OMPI_ERROR;
|
|
}
|
|
|
|
/* Initialize the predefined error handlers */
|
|
OBJ_CONSTRUCT( &ompi_mpi_errhandler_null.eh, ompi_errhandler_t );
|
|
if( ompi_mpi_errhandler_null.eh.eh_f_to_c_index != OMPI_ERRHANDLER_NULL_FORTRAN )
|
|
return OMPI_ERROR;
|
|
ompi_mpi_errhandler_null.eh.eh_mpi_object_type = OMPI_ERRHANDLER_TYPE_PREDEFINED;
|
|
ompi_mpi_errhandler_null.eh.eh_lang = OMPI_ERRHANDLER_LANG_C;
|
|
ompi_mpi_errhandler_null.eh.eh_comm_fn = NULL;
|
|
ompi_mpi_errhandler_null.eh.eh_file_fn = NULL;
|
|
ompi_mpi_errhandler_null.eh.eh_win_fn = NULL ;
|
|
ompi_mpi_errhandler_null.eh.eh_fort_fn = NULL;
|
|
opal_string_copy(ompi_mpi_errhandler_null.eh.eh_name, "MPI_ERRHANDLER_NULL",
|
|
sizeof(ompi_mpi_errhandler_null.eh.eh_name));
|
|
|
|
OBJ_CONSTRUCT( &ompi_mpi_errors_are_fatal.eh, ompi_errhandler_t );
|
|
if( ompi_mpi_errors_are_fatal.eh.eh_f_to_c_index != OMPI_ERRORS_ARE_FATAL_FORTRAN )
|
|
return OMPI_ERROR;
|
|
ompi_mpi_errors_are_fatal.eh.eh_mpi_object_type = OMPI_ERRHANDLER_TYPE_PREDEFINED;
|
|
ompi_mpi_errors_are_fatal.eh.eh_lang = OMPI_ERRHANDLER_LANG_C;
|
|
ompi_mpi_errors_are_fatal.eh.eh_comm_fn = ompi_mpi_errors_are_fatal_comm_handler;
|
|
ompi_mpi_errors_are_fatal.eh.eh_file_fn = ompi_mpi_errors_are_fatal_file_handler;
|
|
ompi_mpi_errors_are_fatal.eh.eh_win_fn = ompi_mpi_errors_are_fatal_win_handler ;
|
|
ompi_mpi_errors_are_fatal.eh.eh_fort_fn = NULL;
|
|
opal_string_copy(ompi_mpi_errors_are_fatal.eh.eh_name,
|
|
"MPI_ERRORS_ARE_FATAL",
|
|
sizeof(ompi_mpi_errors_are_fatal.eh.eh_name));
|
|
|
|
OBJ_CONSTRUCT( &ompi_mpi_errors_return.eh, ompi_errhandler_t );
|
|
if( ompi_mpi_errors_return.eh.eh_f_to_c_index != OMPI_ERRORS_RETURN_FORTRAN )
|
|
return OMPI_ERROR;
|
|
ompi_mpi_errors_return.eh.eh_mpi_object_type = OMPI_ERRHANDLER_TYPE_PREDEFINED;
|
|
ompi_mpi_errors_return.eh.eh_lang = OMPI_ERRHANDLER_LANG_C;
|
|
ompi_mpi_errors_return.eh.eh_comm_fn = ompi_mpi_errors_return_comm_handler;
|
|
ompi_mpi_errors_return.eh.eh_file_fn = ompi_mpi_errors_return_file_handler;
|
|
ompi_mpi_errors_return.eh.eh_win_fn = ompi_mpi_errors_return_win_handler;
|
|
ompi_mpi_errors_return.eh.eh_fort_fn = NULL;
|
|
opal_string_copy(ompi_mpi_errors_return.eh.eh_name, "MPI_ERRORS_RETURN",
|
|
sizeof(ompi_mpi_errors_return.eh.eh_name));
|
|
|
|
OBJ_CONSTRUCT( &ompi_mpi_errors_abort.eh, ompi_errhandler_t );
|
|
if( ompi_mpi_errors_abort.eh.eh_f_to_c_index != OMPI_ERRORS_ABORT_FORTRAN )
|
|
return OMPI_ERROR;
|
|
ompi_mpi_errors_abort.eh.eh_mpi_object_type = OMPI_ERRHANDLER_TYPE_PREDEFINED;
|
|
ompi_mpi_errors_abort.eh.eh_lang = OMPI_ERRHANDLER_LANG_C;
|
|
ompi_mpi_errors_abort.eh.eh_comm_fn = ompi_mpi_errors_abort_comm_handler;
|
|
ompi_mpi_errors_abort.eh.eh_file_fn = ompi_mpi_errors_abort_file_handler;
|
|
ompi_mpi_errors_abort.eh.eh_win_fn = ompi_mpi_errors_abort_win_handler ;
|
|
ompi_mpi_errors_abort.eh.eh_fort_fn = NULL;
|
|
opal_string_copy(ompi_mpi_errors_abort.eh.eh_name,
|
|
"MPI_ERRORS_ABORT",
|
|
sizeof(ompi_mpi_errors_abort.eh.eh_name));
|
|
|
|
/* Lets initialize the initial error handler if not already done */
|
|
char *env = getenv("OMPI_MCA_mpi_initial_errhandler");
|
|
if( NULL != env ) {
|
|
ompi_process_info.initial_errhandler = strndup(env, MPI_MAX_INFO_VAL);
|
|
}
|
|
return ompi_initial_errhandler_init();
|
|
}
|
|
|
|
|
|
/*
|
|
* Clean up the errorhandler resources
|
|
*/
|
|
int ompi_errhandler_finalize(void)
|
|
{
|
|
OBJ_DESTRUCT(&ompi_mpi_errhandler_null.eh);
|
|
OBJ_DESTRUCT(&ompi_mpi_errors_return.eh);
|
|
OBJ_DESTRUCT(&ompi_mpi_errors_are_fatal.eh);
|
|
|
|
/* JMS Add stuff here checking for unreleased errorhandlers,
|
|
similar to communicators, info handles, etc. */
|
|
PMIx_Deregister_event_handler(default_errhandler_id, NULL, NULL);
|
|
|
|
/* Remove errhandler F2C table */
|
|
|
|
OBJ_DESTRUCT(&ompi_errhandler_f_to_c_table);
|
|
|
|
/* All done */
|
|
|
|
return OMPI_SUCCESS;
|
|
}
|
|
|
|
|
|
ompi_errhandler_t *ompi_errhandler_create(ompi_errhandler_type_t object_type,
|
|
ompi_errhandler_generic_handler_fn_t *func,
|
|
ompi_errhandler_lang_t lang)
|
|
{
|
|
ompi_errhandler_t *new_errhandler;
|
|
|
|
/* Create a new object and ensure that it's valid */
|
|
|
|
new_errhandler = OBJ_NEW(ompi_errhandler_t);
|
|
if (NULL != new_errhandler) {
|
|
if (0 > new_errhandler->eh_f_to_c_index) {
|
|
OBJ_RELEASE(new_errhandler);
|
|
new_errhandler = NULL;
|
|
} else {
|
|
|
|
/* We cast the user's callback function to any one of the
|
|
function pointer types in the union; it doesn't matter which.
|
|
It only matters that we dereference/use the right member when
|
|
invoking the callback. */
|
|
|
|
new_errhandler->eh_mpi_object_type = object_type;
|
|
new_errhandler->eh_lang = lang;
|
|
switch (object_type ) {
|
|
case (OMPI_ERRHANDLER_TYPE_COMM):
|
|
new_errhandler->eh_comm_fn = (MPI_Comm_errhandler_function *)func;
|
|
break;
|
|
case (OMPI_ERRHANDLER_TYPE_FILE):
|
|
new_errhandler->eh_file_fn = (ompi_file_errhandler_function *)func;
|
|
break;
|
|
case (OMPI_ERRHANDLER_TYPE_WIN):
|
|
new_errhandler->eh_win_fn = (MPI_Win_errhandler_function *)func;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
new_errhandler->eh_fort_fn = (ompi_errhandler_fortran_handler_fn_t *)func;
|
|
}
|
|
}
|
|
|
|
/* All done */
|
|
|
|
return new_errhandler;
|
|
}
|
|
|
|
/* helper to move the error report back from the RTE thread to the MPI thread */
|
|
typedef struct ompi_errhandler_event_s {
|
|
opal_event_t super;
|
|
opal_process_name_t procname;
|
|
int status;
|
|
} ompi_errhandler_event_t;
|
|
|
|
static void *ompi_errhandler_event_cb(int fd, int flags, void *context) {
|
|
ompi_errhandler_event_t *event = (ompi_errhandler_event_t*) context;
|
|
int status = event->status;
|
|
opal_event_del(&event->super);
|
|
free(event);
|
|
/* our default action is to abort */
|
|
/* TODO: this error should return to the caller and invoke an error
|
|
* handler from the MPI API call.
|
|
* For now, it is fatal. */
|
|
ompi_mpi_errors_are_fatal_comm_handler(NULL, status, "PMIx Even Notification");
|
|
return NULL;
|
|
}
|
|
|
|
/* registration callback */
|
|
void ompi_errhandler_registration_callback(int status,
|
|
size_t errhandler_ref,
|
|
void *cbdata)
|
|
{
|
|
ompi_errhandler_errtrk_t *errtrk = (ompi_errhandler_errtrk_t*)cbdata;
|
|
|
|
default_errhandler_id = errhandler_ref;
|
|
errtrk->status = status;
|
|
errtrk->active = false;
|
|
}
|
|
|
|
/**
|
|
* Default errhandler callback
|
|
*/
|
|
void ompi_errhandler_callback(size_t refid, pmix_status_t status,
|
|
const pmix_proc_t *source,
|
|
pmix_info_t *info, size_t ninfo,
|
|
pmix_info_t *results, size_t nresults,
|
|
pmix_event_notification_cbfunc_fn_t cbfunc,
|
|
void *cbdata)
|
|
{
|
|
int rc;
|
|
/* an error has been found, report to the MPI layer and let it take
|
|
* further action. */
|
|
/* transition this from the RTE thread to the MPI progress engine */
|
|
ompi_errhandler_event_t *event = malloc(sizeof(*event));
|
|
if(NULL == event) {
|
|
OMPI_ERROR_LOG(OMPI_ERR_OUT_OF_RESOURCE);
|
|
goto error;
|
|
}
|
|
OPAL_PMIX_CONVERT_PROCT(rc, &event->procname, (pmix_proc_t*)source);
|
|
if(OPAL_UNLIKELY(OPAL_SUCCESS != rc)) {
|
|
OMPI_ERROR_LOG(rc);
|
|
free(event);
|
|
goto error;
|
|
}
|
|
event->status = status;
|
|
opal_event_set(opal_sync_event_base, &event->super, -1, OPAL_EV_READ,
|
|
ompi_errhandler_event_cb, event);
|
|
opal_event_active(&event->super, OPAL_EV_READ, 1);
|
|
/* tell the event chain engine to go no further - we
|
|
* will handle this */
|
|
if (NULL != cbfunc) {
|
|
cbfunc(PMIX_EVENT_ACTION_COMPLETE, NULL, 0, NULL, NULL, cbdata);
|
|
}
|
|
return;
|
|
|
|
error:
|
|
if (NULL != cbfunc) {
|
|
/* We can't handle this, let the default action abort. */
|
|
cbfunc(PMIX_EVENT_NO_ACTION_TAKEN, NULL, 0, NULL, NULL, cbdata);
|
|
}
|
|
}
|
|
|
|
/**************************************************************************
|
|
*
|
|
* Static functions
|
|
*
|
|
**************************************************************************/
|
|
|
|
/**
|
|
* Errhandler constructor
|
|
*/
|
|
static void ompi_errhandler_construct(ompi_errhandler_t *new_errhandler)
|
|
{
|
|
int ret_val;
|
|
|
|
/* assign entry in fortran <-> c translation array */
|
|
|
|
ret_val = opal_pointer_array_add(&ompi_errhandler_f_to_c_table,
|
|
new_errhandler);
|
|
new_errhandler->eh_f_to_c_index = ret_val;
|
|
|
|
new_errhandler->eh_lang = OMPI_ERRHANDLER_LANG_C;
|
|
|
|
new_errhandler->eh_comm_fn = NULL;
|
|
new_errhandler->eh_win_fn = NULL;
|
|
new_errhandler->eh_file_fn = NULL;
|
|
new_errhandler->eh_fort_fn = NULL;
|
|
|
|
new_errhandler->eh_cxx_dispatch_fn = NULL;
|
|
|
|
memset (new_errhandler->eh_name, 0, MPI_MAX_OBJECT_NAME);
|
|
}
|
|
|
|
|
|
/**
|
|
* Errhandler destructor
|
|
*/
|
|
static void ompi_errhandler_destruct(ompi_errhandler_t *errhandler)
|
|
{
|
|
/* reset the ompi_errhandler_f_to_c_table entry - make sure that the
|
|
entry is in the table */
|
|
|
|
if (NULL!= opal_pointer_array_get_item(&ompi_errhandler_f_to_c_table,
|
|
errhandler->eh_f_to_c_index)) {
|
|
opal_pointer_array_set_item(&ompi_errhandler_f_to_c_table,
|
|
errhandler->eh_f_to_c_index, NULL);
|
|
}
|
|
}
|