2006-12-30 23:41:42 +00:00
|
|
|
// -*- c++ -*-
|
|
|
|
//
|
|
|
|
// Copyright (c) 2006 Los Alamos National Security, LLC. All rights
|
|
|
|
// reserved.
|
2009-10-22 16:50:45 +00:00
|
|
|
// Copyright (c) 2007-2009 Cisco Systems, Inc. All rights reserved.
|
2006-12-30 23:41:42 +00:00
|
|
|
// $COPYRIGHT$
|
|
|
|
//
|
|
|
|
// Additional copyrights may follow
|
|
|
|
//
|
|
|
|
// $HEADER$
|
|
|
|
//
|
|
|
|
|
2007-12-07 13:09:07 +00:00
|
|
|
// Do not include ompi_config.h before mpi.h because it causes
|
|
|
|
// malloc/free problems due to setting OMPI_BUILDING to 1
|
2006-12-30 23:41:42 +00:00
|
|
|
#include "mpi.h"
|
2007-12-07 13:09:07 +00:00
|
|
|
|
|
|
|
#include "ompi/constants.h"
|
2006-12-30 23:41:42 +00:00
|
|
|
#include "ompi/mpi/cxx/mpicxx.h"
|
2007-12-07 13:09:07 +00:00
|
|
|
#include "opal/class/opal_list.h"
|
|
|
|
#include "ompi/file/file.h"
|
|
|
|
#include "ompi/errhandler/errhandler.h"
|
|
|
|
#include "ompi/runtime/mpiruntime.h"
|
2006-12-30 23:41:42 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
MPI::File::Close()
|
|
|
|
{
|
|
|
|
(void) MPI_File_close(&mpi_file);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-10 19:29:25 +00:00
|
|
|
MPI::Errhandler
|
2009-10-22 16:50:45 +00:00
|
|
|
MPI::File::Create_errhandler(MPI::File::Errhandler_function* function)
|
2006-12-30 23:41:42 +00:00
|
|
|
{
|
2008-02-10 19:29:25 +00:00
|
|
|
MPI_Errhandler c_errhandler =
|
|
|
|
ompi_errhandler_create(OMPI_ERRHANDLER_TYPE_FILE,
|
|
|
|
(ompi_errhandler_generic_handler_fn_t*) function,
|
|
|
|
OMPI_ERRHANDLER_LANG_CXX);
|
|
|
|
c_errhandler->eh_cxx_dispatch_fn =
|
|
|
|
(ompi_errhandler_cxx_dispatch_fn_t*)
|
|
|
|
ompi_mpi_cxx_file_errhandler_invoke;
|
|
|
|
return c_errhandler;
|
2006-12-30 23:41:42 +00:00
|
|
|
}
|
2007-12-07 13:09:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Infrastructure for MPI_REGISTER_DATAREP
|
|
|
|
//
|
|
|
|
// Similar to what we have to do in the F77 bindings: call the C
|
|
|
|
// MPI_Register_datarep function with "intercept" callback functions
|
|
|
|
// that conform to the C bindings. In these intercepts, convert the
|
|
|
|
// arguments to C++ calling convertions, and then invoke the actual
|
|
|
|
// C++ callbacks.
|
|
|
|
|
|
|
|
// Data structure passed to the intercepts (see below). It is an OPAL
|
|
|
|
// list_item_t so that we can clean this memory up during
|
|
|
|
// MPI_FINALIZE.
|
|
|
|
typedef struct intercept_extra_state {
|
|
|
|
opal_list_item_t base;
|
|
|
|
MPI::Datarep_conversion_function *read_fn_cxx;
|
|
|
|
MPI::Datarep_conversion_function *write_fn_cxx;
|
|
|
|
MPI::Datarep_extent_function *extent_fn_cxx;
|
|
|
|
void *extra_state_cxx;
|
|
|
|
} intercept_extra_state_t;
|
|
|
|
|
|
|
|
static void intercept_extra_state_constructor(intercept_extra_state_t *obj)
|
|
|
|
{
|
|
|
|
obj->read_fn_cxx = NULL;
|
|
|
|
obj->write_fn_cxx = NULL;
|
|
|
|
obj->extent_fn_cxx = NULL;
|
|
|
|
obj->extra_state_cxx = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
OBJ_CLASS_DECLARATION(intercept_extra_state_t);
|
|
|
|
OBJ_CLASS_INSTANCE(intercept_extra_state_t,
|
|
|
|
opal_list_item_t,
|
|
|
|
intercept_extra_state_constructor, NULL);
|
|
|
|
|
|
|
|
// Intercept function for read conversions
|
|
|
|
static int read_intercept_fn(void *userbuf, MPI_Datatype type_c, int count_c,
|
|
|
|
void *filebuf, MPI_Offset position_c,
|
|
|
|
void *extra_state)
|
|
|
|
{
|
|
|
|
MPI::Datatype type_cxx(type_c);
|
|
|
|
MPI::Offset position_cxx(position_c);
|
|
|
|
intercept_extra_state_t *intercept_data =
|
|
|
|
(intercept_extra_state_t*) extra_state;
|
|
|
|
|
|
|
|
intercept_data->read_fn_cxx(userbuf, type_cxx, count_c, filebuf,
|
|
|
|
position_cxx, intercept_data->extra_state_cxx);
|
|
|
|
return MPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Intercept function for write conversions
|
|
|
|
static int write_intercept_fn(void *userbuf, MPI_Datatype type_c, int count_c,
|
|
|
|
void *filebuf, MPI_Offset position_c,
|
|
|
|
void *extra_state)
|
|
|
|
{
|
|
|
|
MPI::Datatype type_cxx(type_c);
|
|
|
|
MPI::Offset position_cxx(position_c);
|
|
|
|
intercept_extra_state_t *intercept_data =
|
|
|
|
(intercept_extra_state_t*) extra_state;
|
|
|
|
|
|
|
|
intercept_data->write_fn_cxx(userbuf, type_cxx, count_c, filebuf,
|
|
|
|
position_cxx, intercept_data->extra_state_cxx);
|
|
|
|
return MPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Intercept function for extent calculations
|
|
|
|
static int extent_intercept_fn(MPI_Datatype type_c, MPI_Aint *file_extent_c,
|
|
|
|
void *extra_state)
|
|
|
|
{
|
|
|
|
MPI::Datatype type_cxx(type_c);
|
|
|
|
MPI::Aint file_extent_cxx(*file_extent_c);
|
|
|
|
intercept_extra_state_t *intercept_data =
|
|
|
|
(intercept_extra_state_t*) extra_state;
|
|
|
|
|
|
|
|
intercept_data->extent_fn_cxx(type_cxx, file_extent_cxx,
|
|
|
|
intercept_data->extra_state_cxx);
|
|
|
|
*file_extent_c = file_extent_cxx;
|
|
|
|
return MPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// C++ bindings for MPI::Register_datarep
|
|
|
|
void
|
|
|
|
MPI::Register_datarep(const char* datarep,
|
|
|
|
Datarep_conversion_function* read_fn_cxx,
|
|
|
|
Datarep_conversion_function* write_fn_cxx,
|
|
|
|
Datarep_extent_function* extent_fn_cxx,
|
|
|
|
void* extra_state_cxx)
|
|
|
|
{
|
|
|
|
intercept_extra_state_t *intercept;
|
|
|
|
|
|
|
|
intercept = OBJ_NEW(intercept_extra_state_t);
|
|
|
|
if (NULL == intercept) {
|
|
|
|
OMPI_ERRHANDLER_INVOKE(MPI_FILE_NULL, OMPI_ERR_OUT_OF_RESOURCE,
|
|
|
|
"MPI::Register_datarep");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
opal_list_append(&ompi_registered_datareps, &(intercept->base));
|
|
|
|
intercept->read_fn_cxx = read_fn_cxx;
|
|
|
|
intercept->write_fn_cxx = write_fn_cxx;
|
|
|
|
intercept->extent_fn_cxx = extent_fn_cxx;
|
|
|
|
intercept->extra_state_cxx = extra_state_cxx;
|
|
|
|
|
|
|
|
(void)MPI_Register_datarep(const_cast<char*>(datarep), read_intercept_fn,
|
|
|
|
write_intercept_fn,
|
|
|
|
extent_intercept_fn, intercept);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
MPI::Register_datarep(const char* datarep,
|
|
|
|
MPI_Datarep_conversion_function* read_fn_c,
|
|
|
|
Datarep_conversion_function* write_fn_cxx,
|
|
|
|
Datarep_extent_function* extent_fn_cxx,
|
|
|
|
void* extra_state_cxx)
|
|
|
|
{
|
|
|
|
intercept_extra_state_t *intercept;
|
|
|
|
|
|
|
|
intercept = OBJ_NEW(intercept_extra_state_t);
|
|
|
|
if (NULL == intercept) {
|
|
|
|
OMPI_ERRHANDLER_INVOKE(MPI_FILE_NULL, OMPI_ERR_OUT_OF_RESOURCE,
|
|
|
|
"MPI::Register_datarep");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
opal_list_append(&ompi_registered_datareps, &(intercept->base));
|
|
|
|
intercept->write_fn_cxx = write_fn_cxx;
|
|
|
|
intercept->extent_fn_cxx = extent_fn_cxx;
|
|
|
|
intercept->extra_state_cxx = extra_state_cxx;
|
|
|
|
|
|
|
|
(void)MPI_Register_datarep(const_cast<char*>(datarep), read_fn_c,
|
|
|
|
write_intercept_fn,
|
|
|
|
extent_intercept_fn, intercept);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
MPI::Register_datarep(const char* datarep,
|
|
|
|
Datarep_conversion_function* read_fn_cxx,
|
|
|
|
MPI_Datarep_conversion_function* write_fn_c,
|
|
|
|
Datarep_extent_function* extent_fn_cxx,
|
|
|
|
void* extra_state_cxx)
|
|
|
|
{
|
|
|
|
intercept_extra_state_t *intercept;
|
|
|
|
|
|
|
|
intercept = OBJ_NEW(intercept_extra_state_t);
|
|
|
|
if (NULL == intercept) {
|
|
|
|
OMPI_ERRHANDLER_INVOKE(MPI_FILE_NULL, OMPI_ERR_OUT_OF_RESOURCE,
|
|
|
|
"MPI::Register_datarep");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
opal_list_append(&ompi_registered_datareps, &(intercept->base));
|
|
|
|
intercept->read_fn_cxx = read_fn_cxx;
|
|
|
|
intercept->extent_fn_cxx = extent_fn_cxx;
|
|
|
|
intercept->extra_state_cxx = extra_state_cxx;
|
|
|
|
|
|
|
|
(void)MPI_Register_datarep(const_cast<char*>(datarep), read_intercept_fn,
|
|
|
|
write_fn_c,
|
|
|
|
extent_intercept_fn, intercept);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
MPI::Register_datarep(const char* datarep,
|
|
|
|
MPI_Datarep_conversion_function* read_fn_c,
|
|
|
|
MPI_Datarep_conversion_function* write_fn_c,
|
|
|
|
Datarep_extent_function* extent_fn_cxx,
|
|
|
|
void* extra_state_cxx)
|
|
|
|
{
|
|
|
|
intercept_extra_state_t *intercept;
|
|
|
|
|
|
|
|
intercept = OBJ_NEW(intercept_extra_state_t);
|
|
|
|
if (NULL == intercept) {
|
|
|
|
OMPI_ERRHANDLER_INVOKE(MPI_FILE_NULL, OMPI_ERR_OUT_OF_RESOURCE,
|
|
|
|
"MPI::Register_datarep");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
opal_list_append(&ompi_registered_datareps, &(intercept->base));
|
|
|
|
intercept->extent_fn_cxx = extent_fn_cxx;
|
|
|
|
intercept->extra_state_cxx = extra_state_cxx;
|
|
|
|
|
|
|
|
(void)MPI_Register_datarep(const_cast<char*>(datarep), read_fn_c,
|
|
|
|
write_fn_c,
|
|
|
|
extent_intercept_fn, intercept);
|
|
|
|
}
|
|
|
|
|
|
|
|
|