2004-09-14 10:55:10 +00:00
|
|
|
/*
|
2005-11-05 19:57:48 +00:00
|
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
* University Research and Technology
|
|
|
|
* Corporation. All rights reserved.
|
|
|
|
* Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
|
|
* of Tennessee Research Foundation. All rights
|
|
|
|
* reserved.
|
2004-11-28 20:09:25 +00:00
|
|
|
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
|
|
* University of Stuttgart. All rights reserved.
|
2005-03-24 12:43:37 +00:00
|
|
|
* Copyright (c) 2004-2005 The Regents of the University of California.
|
|
|
|
* All rights reserved.
|
2004-11-22 01:38:40 +00:00
|
|
|
* $COPYRIGHT$
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
2004-09-14 10:55:10 +00:00
|
|
|
* $HEADER$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ompi_config.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "mpi.h"
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "ompi/file/file.h"
|
2005-07-03 16:22:16 +00:00
|
|
|
#include "opal/class/opal_list.h"
|
2005-07-04 00:13:44 +00:00
|
|
|
#include "opal/util/argv.h"
|
2005-07-03 23:31:27 +00:00
|
|
|
#include "opal/util/output.h"
|
2006-02-12 01:33:29 +00:00
|
|
|
#include "opal/mca/mca.h"
|
|
|
|
#include "opal/mca/base/base.h"
|
|
|
|
#include "ompi/mca/io/io.h"
|
|
|
|
#include "ompi/mca/io/base/base.h"
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Local types
|
|
|
|
*/
|
|
|
|
struct avail_io_t {
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_item_t super;
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
mca_io_base_version_t ai_version;
|
|
|
|
|
|
|
|
int ai_priority;
|
|
|
|
mca_io_base_components_t ai_component;
|
|
|
|
struct mca_io_base_delete_t *ai_private_data;
|
|
|
|
};
|
|
|
|
typedef struct avail_io_t avail_io_t;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Local functions
|
|
|
|
*/
|
2005-07-03 16:22:16 +00:00
|
|
|
static opal_list_t *check_components(opal_list_t *components,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info,
|
2004-09-14 10:55:10 +00:00
|
|
|
char **names, int num_names);
|
|
|
|
static avail_io_t *check_one_component(const mca_base_component_t *component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
static avail_io_t *query(const mca_base_component_t *component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info);
|
2004-09-14 10:55:10 +00:00
|
|
|
static avail_io_t *query_1_0_0(const mca_io_base_component_1_0_0_t *io_component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
2004-11-05 07:52:30 +00:00
|
|
|
static void unquery(avail_io_t *avail, char *filename, struct ompi_info_t *info);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
2004-11-05 07:52:30 +00:00
|
|
|
static int delete_file(avail_io_t *avail, char *filename, struct ompi_info_t *info);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stuff for the OBJ interface
|
|
|
|
*/
|
2005-07-03 16:22:16 +00:00
|
|
|
static OBJ_CLASS_INSTANCE(avail_io_t, opal_list_item_t, NULL, NULL);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
2004-11-05 07:52:30 +00:00
|
|
|
int mca_io_base_delete(char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
int err, num_names;
|
|
|
|
char *names, **name_array;
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_t *selectable;
|
|
|
|
opal_list_item_t *item;
|
2004-09-14 10:55:10 +00:00
|
|
|
avail_io_t *avail, selected;
|
|
|
|
|
|
|
|
/* Announce */
|
|
|
|
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: deleting file: %s",
|
|
|
|
filename);
|
|
|
|
|
|
|
|
/* See if a set of component was requested by the MCA parameter.
|
|
|
|
Don't check for error. */
|
|
|
|
|
|
|
|
names = NULL;
|
|
|
|
mca_base_param_lookup_string(mca_io_base_param, &names);
|
|
|
|
|
|
|
|
/* Compute the intersection of all of my available components with
|
|
|
|
the components from all the other processes in this file */
|
|
|
|
|
|
|
|
/* JMS CONTINUE HERE */
|
|
|
|
|
|
|
|
/* See if there were any listed in the MCA parameter; parse them
|
|
|
|
and check them all */
|
|
|
|
|
|
|
|
err = OMPI_ERROR;
|
|
|
|
if (NULL != names && 0 < strlen(names)) {
|
2005-07-04 00:13:44 +00:00
|
|
|
name_array = opal_argv_split(names, ',');
|
|
|
|
num_names = opal_argv_count(name_array);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: Checking specific modules: %s",
|
|
|
|
names);
|
|
|
|
selectable = check_components(&mca_io_base_components_available,
|
|
|
|
filename, info, name_array, num_names);
|
2005-07-04 00:13:44 +00:00
|
|
|
opal_argv_free(name_array);
|
2004-09-14 10:55:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Nope -- a specific [set of] component[s] was not requested. Go
|
|
|
|
check them all. */
|
|
|
|
|
|
|
|
else {
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: Checking all available modules");
|
|
|
|
selectable = check_components(&mca_io_base_components_available,
|
|
|
|
filename, info, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Upon return from the above, the modules list will contain the
|
|
|
|
list of modules that returned (priority >= 0). If we have no
|
|
|
|
io modules available, it's an error */
|
|
|
|
|
|
|
|
if (NULL == selectable) {
|
|
|
|
/* There's no modules available. Doh! */
|
|
|
|
/* show_help */
|
|
|
|
return OMPI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Do some kind of collective operation to find a module that
|
|
|
|
everyone has available */
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
/* For the moment, just take the top module off the list */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
item = opal_list_remove_first(selectable);
|
2004-09-14 10:55:10 +00:00
|
|
|
avail = (avail_io_t *) item;
|
|
|
|
selected = *avail;
|
|
|
|
OBJ_RELEASE(avail);
|
|
|
|
#else
|
|
|
|
/* JMS CONTINUE HERE */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Everything left in the selectable list is therefore unwanted,
|
|
|
|
and we call their unquery() method (because they all had
|
|
|
|
query() invoked, but will never have init() invoked in this
|
|
|
|
scope). */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
for (item = opal_list_remove_first(selectable); item != NULL;
|
|
|
|
item = opal_list_remove_first(selectable)) {
|
2004-09-14 10:55:10 +00:00
|
|
|
avail = (avail_io_t *) item;
|
|
|
|
unquery(avail, filename, info);
|
|
|
|
OBJ_RELEASE(item);
|
|
|
|
}
|
|
|
|
OBJ_RELEASE(selectable);
|
|
|
|
|
|
|
|
/* Finally -- delete the file with the selected component */
|
|
|
|
|
|
|
|
if (OMPI_SUCCESS != (err = delete_file(&selected, filename, info))) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Announce the winner */
|
|
|
|
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: Selected io component %s",
|
|
|
|
selected.ai_component.v1_0_0.io_version.mca_component_name);
|
|
|
|
|
|
|
|
return OMPI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For each module in the list, if it is in the list of names (or the
|
|
|
|
* list of names is NULL), then check and see if it wants to run, and
|
|
|
|
* do the resulting priority comparison. Make a list of components to
|
|
|
|
* be only those who returned that they want to run, and put them in
|
|
|
|
* priority order.
|
|
|
|
*/
|
2005-07-03 16:22:16 +00:00
|
|
|
static opal_list_t *check_components(opal_list_t *components,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info,
|
2004-09-14 10:55:10 +00:00
|
|
|
char **names, int num_names)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
const mca_base_component_t *component;
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_item_t *item, *item2;
|
2004-09-14 10:55:10 +00:00
|
|
|
bool want_to_check;
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_t *selectable;
|
2004-09-14 10:55:10 +00:00
|
|
|
avail_io_t *avail, *avail2;
|
|
|
|
|
|
|
|
/* Make a list of the components that query successfully */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
selectable = OBJ_NEW(opal_list_t);
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
/* Scan through the list of components. This nested loop is
|
|
|
|
O(N^2), but we should never have too many components and/or
|
|
|
|
names, so this *hopefully* shouldn't matter... */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
for (item = opal_list_get_first(components);
|
|
|
|
item != opal_list_get_end(components);
|
|
|
|
item = opal_list_get_next(item)) {
|
2004-09-14 10:55:10 +00:00
|
|
|
component = ((mca_base_component_priority_list_item_t *)
|
2004-11-12 16:55:41 +00:00
|
|
|
item)->super.cli_component;
|
2004-09-14 10:55:10 +00:00
|
|
|
|
|
|
|
/* If we have a list of names, scan through it */
|
|
|
|
|
|
|
|
if (0 == num_names) {
|
|
|
|
want_to_check = true;
|
|
|
|
} else {
|
|
|
|
want_to_check = false;
|
|
|
|
for (i = 0; i < num_names; ++i) {
|
|
|
|
if (0 == strcmp(names[i], component->mca_component_name)) {
|
|
|
|
want_to_check = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we determined that we want to check this component, then
|
|
|
|
do so */
|
|
|
|
|
|
|
|
if (want_to_check) {
|
|
|
|
avail = check_one_component(component, filename, info);
|
|
|
|
if (NULL != avail) {
|
|
|
|
|
|
|
|
/* Put this item on the list in priority order
|
|
|
|
(highest priority first). Should it go first? */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
item2 = opal_list_get_first(selectable);
|
2004-09-14 10:55:10 +00:00
|
|
|
avail2 = (avail_io_t *) item2;
|
2005-07-03 16:22:16 +00:00
|
|
|
if (opal_list_get_end(selectable) == item2 ||
|
2004-09-14 10:55:10 +00:00
|
|
|
avail->ai_priority > avail2->ai_priority) {
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_prepend(selectable, (opal_list_item_t*) avail);
|
2004-09-14 10:55:10 +00:00
|
|
|
} else {
|
2005-07-03 16:22:16 +00:00
|
|
|
for (i = 1; item2 != opal_list_get_end(selectable);
|
|
|
|
item2 = opal_list_get_next(selectable), ++i) {
|
2004-09-14 10:55:10 +00:00
|
|
|
avail2 = (avail_io_t *) item2;
|
|
|
|
if (avail->ai_priority > avail2->ai_priority) {
|
2005-07-03 16:22:16 +00:00
|
|
|
opal_list_insert(selectable,
|
|
|
|
(opal_list_item_t *) avail, i);
|
2004-09-14 10:55:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we didn't find a place to put it in the
|
|
|
|
list, then append it (because it has the lowest
|
|
|
|
priority found so far) */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
if (opal_list_get_end(selectable) == item2) {
|
|
|
|
opal_list_append(selectable,
|
|
|
|
(opal_list_item_t *) avail);
|
2004-09-14 10:55:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we didn't find any available components, return an error */
|
|
|
|
|
2005-07-03 16:22:16 +00:00
|
|
|
if (0 == opal_list_get_size(selectable)) {
|
2004-09-14 10:55:10 +00:00
|
|
|
OBJ_RELEASE(selectable);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All done */
|
|
|
|
|
|
|
|
return selectable;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check a single component
|
|
|
|
*/
|
|
|
|
static avail_io_t *check_one_component(const mca_base_component_t *component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
avail_io_t *avail;
|
|
|
|
|
|
|
|
avail = query(component, filename, info);
|
|
|
|
if (NULL != avail) {
|
|
|
|
avail->ai_priority = (avail->ai_priority < 100) ?
|
|
|
|
avail->ai_priority : 100;
|
|
|
|
avail->ai_priority = (avail->ai_priority < 0) ?
|
|
|
|
0 : avail->ai_priority;
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: component available: %s, priority: %d",
|
|
|
|
component->mca_component_name,
|
|
|
|
avail->ai_priority);
|
|
|
|
} else {
|
2005-07-03 23:31:27 +00:00
|
|
|
opal_output_verbose(10, mca_io_base_output,
|
2004-09-14 10:55:10 +00:00
|
|
|
"io:base:delete: component not available: %s",
|
|
|
|
component->mca_component_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return avail;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
* Query functions
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Take any version of a io module, query it, and return the right
|
|
|
|
* module struct
|
|
|
|
*/
|
|
|
|
static avail_io_t *query(const mca_base_component_t *component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
const mca_io_base_component_1_0_0_t *ioc_100;
|
|
|
|
|
|
|
|
/* io v1.0.0 */
|
|
|
|
|
|
|
|
if (1 == component->mca_major_version &&
|
|
|
|
0 == component->mca_minor_version &&
|
|
|
|
0 == component->mca_release_version) {
|
|
|
|
ioc_100 = (mca_io_base_component_1_0_0_t *) component;
|
|
|
|
|
|
|
|
return query_1_0_0(ioc_100, filename, info);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unknown io API version -- return error */
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static avail_io_t *query_1_0_0(const mca_io_base_component_1_0_0_t *component,
|
2004-11-05 07:52:30 +00:00
|
|
|
char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
bool usable;
|
|
|
|
int priority, ret;
|
|
|
|
avail_io_t *avail;
|
|
|
|
struct mca_io_base_delete_t *private_data;
|
|
|
|
|
|
|
|
/* Query v1.0.0 */
|
|
|
|
|
|
|
|
avail = NULL;
|
|
|
|
private_data = NULL;
|
|
|
|
usable = false;
|
|
|
|
ret = component->io_delete_query(filename, info, &private_data, &usable,
|
|
|
|
&priority);
|
|
|
|
if (OMPI_SUCCESS == ret && usable) {
|
|
|
|
avail = OBJ_NEW(avail_io_t);
|
|
|
|
avail->ai_version = MCA_IO_BASE_V_1_0_0;
|
|
|
|
avail->ai_priority = priority;
|
|
|
|
avail->ai_component.v1_0_0 = *component;
|
|
|
|
avail->ai_private_data = private_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
return avail;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
* Unquery functions
|
|
|
|
**************************************************************************/
|
|
|
|
|
2004-11-05 07:52:30 +00:00
|
|
|
static void unquery(avail_io_t *avail, char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
const mca_io_base_component_1_0_0_t *ioc_100;
|
|
|
|
|
|
|
|
switch(avail->ai_version) {
|
|
|
|
case MCA_IO_BASE_V_1_0_0:
|
|
|
|
ioc_100 = &(avail->ai_component.v1_0_0);
|
|
|
|
if (NULL != ioc_100->io_delete_unquery) {
|
|
|
|
ioc_100->io_delete_unquery(filename, info, avail->ai_private_data);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
* File delete functions
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Invoke the component's delete function
|
|
|
|
*/
|
2004-11-05 07:52:30 +00:00
|
|
|
static int delete_file(avail_io_t *avail, char *filename, struct ompi_info_t *info)
|
2004-09-14 10:55:10 +00:00
|
|
|
{
|
|
|
|
const mca_io_base_component_1_0_0_t *ioc_100;
|
|
|
|
|
|
|
|
switch(avail->ai_version) {
|
|
|
|
case MCA_IO_BASE_V_1_0_0:
|
|
|
|
ioc_100 = &(avail->ai_component.v1_0_0);
|
|
|
|
return ioc_100->io_delete_select(filename, info,
|
|
|
|
avail->ai_private_data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return OMPI_ERROR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No way to reach here */
|
|
|
|
}
|