Commit of the unified component selection system. This was mostly
in-place already (turns out that I was wrong in thinking that it didn't work for static components), but the logic for excluding components was not there. This commit does a few things: - Adds "exclude" logic, so that you can do: mpirun --mca btl ^mvapi,openib ... (note the "^" character -- I tried "!" but then you have to escape it in the shell, and that was icky) which will exclude both the mvapi and openib btl components (excluding one component means that you are excluding all components in the list; it doesn't make sense to include some and exclude others -- you're entire entirely including or entirely excluding) - Simplifies the "include" logic, so the same old stuff like this still works: mpirun --mca btl tcp,self ... will only use the tcp and self btl components. - Added more verbosity statements to make this selection process clear. This commit was SVN r7509.
Этот коммит содержится в:
родитель
1d9b663b62
Коммит
5ffd95e971
@ -42,16 +42,19 @@ typedef struct component_name_t component_name_t;
|
|||||||
* Local variables
|
* Local variables
|
||||||
*/
|
*/
|
||||||
static bool show_errors = false;
|
static bool show_errors = false;
|
||||||
|
static const char negate = '^';
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Local functions
|
* Local functions
|
||||||
*/
|
*/
|
||||||
|
static int parse_requested(int mca_param, bool *include_mode,
|
||||||
|
char ***requested_component_names);
|
||||||
static int open_components(const char *type_name, int output_id,
|
static int open_components(const char *type_name, int output_id,
|
||||||
opal_list_t *components_found,
|
opal_list_t *src, opal_list_t *dest);
|
||||||
opal_list_t *components_available,
|
static int distill(bool include_mode, const char *type_name,
|
||||||
char **requested_component_names);
|
int output_id, opal_list_t *src, opal_list_t *dest,
|
||||||
static int parse_requested(int mca_param, char ***requested_component_names);
|
char **names);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -65,12 +68,14 @@ int mca_base_components_open(const char *type_name, int output_id,
|
|||||||
{
|
{
|
||||||
int ret, param;
|
int ret, param;
|
||||||
opal_list_item_t *item;
|
opal_list_item_t *item;
|
||||||
opal_list_t components_found;
|
opal_list_t components_found, components_distilled;
|
||||||
char **requested_component_names;
|
char **requested_component_names;
|
||||||
int param_verbose = -1;
|
int param_verbose = -1;
|
||||||
int param_type = -1;
|
int param_type = -1;
|
||||||
int verbose_level;
|
int verbose_level;
|
||||||
char *str;
|
char *str;
|
||||||
|
bool include_mode;
|
||||||
|
bool distilled = false;
|
||||||
|
|
||||||
/* Register MCA parameters */
|
/* Register MCA parameters */
|
||||||
|
|
||||||
@ -110,21 +115,36 @@ int mca_base_components_open(const char *type_name, int output_id,
|
|||||||
|
|
||||||
/* See if one or more specific components were requested */
|
/* See if one or more specific components were requested */
|
||||||
|
|
||||||
ret = parse_requested(param_type, &requested_component_names);
|
ret = parse_requested(param_type, &include_mode, &requested_component_names);
|
||||||
if (OMPI_SUCCESS == ret) {
|
if (OMPI_SUCCESS == ret) {
|
||||||
ret = open_components(type_name, output_id, &components_found,
|
ret = distill(include_mode, type_name, output_id, &components_found,
|
||||||
components_available,
|
&components_distilled, requested_component_names);
|
||||||
requested_component_names);
|
distilled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now open whatever we have left */
|
||||||
|
|
||||||
|
if (OMPI_SUCCESS == ret) {
|
||||||
|
ret = open_components(type_name, output_id,
|
||||||
|
&components_distilled, components_available);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free resources */
|
/* Free resources */
|
||||||
|
|
||||||
for (item = opal_list_remove_first(&components_found); NULL != item;
|
for (item = opal_list_remove_first(&components_found); NULL != item;
|
||||||
item = opal_list_remove_first(&components_found)) {
|
item = opal_list_remove_first(&components_found)) {
|
||||||
OBJ_RELEASE(item);
|
OBJ_RELEASE(item);
|
||||||
|
}
|
||||||
|
OBJ_DESTRUCT(&components_found);
|
||||||
|
if (distilled) {
|
||||||
|
for (item = opal_list_remove_first(&components_distilled); NULL != item;
|
||||||
|
item = opal_list_remove_first(&components_distilled)) {
|
||||||
|
OBJ_RELEASE(item);
|
||||||
|
}
|
||||||
|
OBJ_DESTRUCT(&components_distilled);
|
||||||
}
|
}
|
||||||
if (NULL != requested_component_names) {
|
if (NULL != requested_component_names) {
|
||||||
opal_argv_free(requested_component_names);
|
opal_argv_free(requested_component_names);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* All done */
|
/* All done */
|
||||||
@ -133,9 +153,12 @@ int mca_base_components_open(const char *type_name, int output_id,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int parse_requested(int mca_param, char ***requested_component_names)
|
static int parse_requested(int mca_param, bool *include_mode,
|
||||||
|
char ***requested_component_names)
|
||||||
{
|
{
|
||||||
|
int i;
|
||||||
char *requested;
|
char *requested;
|
||||||
|
char *tmp;
|
||||||
|
|
||||||
*requested_component_names = NULL;
|
*requested_component_names = NULL;
|
||||||
|
|
||||||
@ -149,12 +172,141 @@ static int parse_requested(int mca_param, char ***requested_component_names)
|
|||||||
}
|
}
|
||||||
*requested_component_names = opal_argv_split(requested, ',');
|
*requested_component_names = opal_argv_split(requested, ',');
|
||||||
|
|
||||||
|
/* Are we including or excluding? */
|
||||||
|
|
||||||
|
*include_mode = true;
|
||||||
|
for (i = 0; NULL != (*requested_component_names)[i]; ++i) {
|
||||||
|
if (negate == *((*requested_component_names)[i])) {
|
||||||
|
tmp = strdup((*requested_component_names)[i] + 1);
|
||||||
|
free((*requested_component_names)[i]);
|
||||||
|
(*requested_component_names)[i] = tmp;
|
||||||
|
|
||||||
|
*include_mode = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* All done */
|
/* All done */
|
||||||
|
|
||||||
return OMPI_SUCCESS;
|
return OMPI_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Parse the list of found components and factor in the included /
|
||||||
|
* excluded names to come up with a distilled list of components that
|
||||||
|
* we should try to open.
|
||||||
|
*/
|
||||||
|
static int distill(bool include_mode, const char *type_name,
|
||||||
|
int output_id, opal_list_t *src, opal_list_t *dest,
|
||||||
|
char **names)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
bool good;
|
||||||
|
opal_list_item_t *item, *next;
|
||||||
|
const mca_base_component_t *component;
|
||||||
|
mca_base_component_list_item_t *cli;
|
||||||
|
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"distilling %s components", type_name);
|
||||||
|
OBJ_CONSTRUCT(dest, opal_list_t);
|
||||||
|
|
||||||
|
/* Bozo case */
|
||||||
|
|
||||||
|
if (NULL == names) {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"accepting all %s components", type_name);
|
||||||
|
opal_list_join(dest, opal_list_get_end(dest), src);
|
||||||
|
return OMPI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Are we including components? */
|
||||||
|
|
||||||
|
if (include_mode) {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"including %s components", type_name);
|
||||||
|
|
||||||
|
/* Go through all the components and only keep the ones that
|
||||||
|
are specifically mentioned in the list */
|
||||||
|
|
||||||
|
for (i = 0; NULL != names[i]; ++i) {
|
||||||
|
good = false;
|
||||||
|
|
||||||
|
for (item = opal_list_get_first(src);
|
||||||
|
opal_list_get_end(src) != item;
|
||||||
|
item = next) {
|
||||||
|
next = opal_list_get_next(item);
|
||||||
|
cli = (mca_base_component_list_item_t *) item;
|
||||||
|
component = cli->cli_component;
|
||||||
|
if (0 == strcmp(names[i], component->mca_component_name)) {
|
||||||
|
opal_list_remove_item(src, item);
|
||||||
|
opal_list_append(dest, item);
|
||||||
|
good = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (good) {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"%s --> included", names[i]);
|
||||||
|
} else {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"%s --> not found", names[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* No, we are excluding components */
|
||||||
|
|
||||||
|
else {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"excluding %s components", type_name);
|
||||||
|
|
||||||
|
/* Go through all the components and only keep the ones that
|
||||||
|
are specifically mentioned in the list */
|
||||||
|
|
||||||
|
for (item = opal_list_get_first(src);
|
||||||
|
opal_list_get_end(src) != item;
|
||||||
|
item = next) {
|
||||||
|
next = opal_list_get_next(item);
|
||||||
|
good = true;
|
||||||
|
|
||||||
|
for (i = 0; NULL != names[i]; ++i) {
|
||||||
|
cli = (mca_base_component_list_item_t *) item;
|
||||||
|
component = cli->cli_component;
|
||||||
|
if (0 == strcmp(names[i], component->mca_component_name)) {
|
||||||
|
good = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!good) {
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"%s --> excluded",
|
||||||
|
component->mca_component_name);
|
||||||
|
} else {
|
||||||
|
opal_list_remove_item(src, item);
|
||||||
|
opal_list_append(dest, item);
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"%s --> included",
|
||||||
|
component->mca_component_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* All done */
|
||||||
|
|
||||||
|
return OMPI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Traverse the entire list of found components (a list of
|
* Traverse the entire list of found components (a list of
|
||||||
* mca_base_component_t instances). If the requested_component_names
|
* mca_base_component_t instances). If the requested_component_names
|
||||||
@ -163,183 +315,108 @@ static int parse_requested(int mca_param, char ***requested_component_names)
|
|||||||
* If it opens, add it to the components_available list.
|
* If it opens, add it to the components_available list.
|
||||||
*/
|
*/
|
||||||
static int open_components(const char *type_name, int output_id,
|
static int open_components(const char *type_name, int output_id,
|
||||||
opal_list_t *components_found,
|
opal_list_t *src, opal_list_t *dest)
|
||||||
opal_list_t *components_available,
|
|
||||||
char **requested_component_names)
|
|
||||||
{
|
{
|
||||||
int i;
|
opal_list_item_t *item;
|
||||||
opal_list_item_t *item;
|
const mca_base_component_t *component;
|
||||||
const mca_base_component_t *component;
|
mca_base_component_list_item_t *cli;
|
||||||
mca_base_component_list_item_t *cli;
|
bool called_open;
|
||||||
bool acceptable;
|
bool opened;
|
||||||
bool any_acceptable = true;
|
|
||||||
bool called_open;
|
/* Announce */
|
||||||
bool opened;
|
|
||||||
|
|
||||||
/* Announce */
|
|
||||||
|
|
||||||
if (NULL == requested_component_names) {
|
|
||||||
opal_output_verbose(10, output_id,
|
opal_output_verbose(10, output_id,
|
||||||
"mca: base: components_open: "
|
"mca: base: components_open: opening %s components",
|
||||||
"looking for any %s components", type_name);
|
|
||||||
} else {
|
|
||||||
opal_output_verbose(10, output_id,
|
|
||||||
"mca: base: components_open: looking for specific %s components:",
|
|
||||||
type_name);
|
type_name);
|
||||||
for (i = 0; NULL != requested_component_names[i]; ++i) {
|
|
||||||
opal_output_verbose(10, output_id, "mca: base: components_open: %s",
|
/* Traverse the list of found components */
|
||||||
requested_component_names[i]);
|
|
||||||
}
|
OBJ_CONSTRUCT(dest, opal_list_t);
|
||||||
}
|
for (item = opal_list_get_first(src);
|
||||||
|
opal_list_get_end(src) != item;
|
||||||
/*
|
item = opal_list_get_next(item)) {
|
||||||
* Check requested components for validity
|
cli = (mca_base_component_list_item_t *) item;
|
||||||
*/
|
component = cli->cli_component;
|
||||||
if (NULL != requested_component_names) {
|
|
||||||
any_acceptable = false;
|
|
||||||
|
|
||||||
/* For every requested component ... */
|
|
||||||
for (i = 0; NULL != requested_component_names[i]; ++i) {
|
|
||||||
acceptable = false;
|
|
||||||
|
|
||||||
/* Try to match it to an item in the list */
|
opened = called_open = false;
|
||||||
for (item = opal_list_get_first(components_found);
|
|
||||||
opal_list_get_end(components_found) != item;
|
|
||||||
item = opal_list_get_next(item)) {
|
|
||||||
|
|
||||||
cli = (mca_base_component_list_item_t *) item;
|
|
||||||
component = cli->cli_component;
|
|
||||||
|
|
||||||
if (0 == strcmp(requested_component_names[i],
|
|
||||||
component->mca_component_name)) {
|
|
||||||
acceptable = true;
|
|
||||||
any_acceptable = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Didn't find it in the list, warn the user */
|
|
||||||
if (!acceptable) {
|
|
||||||
opal_show_help("help-mca-base.txt", "find-available:not-valid", true,
|
|
||||||
requested_component_names[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* None of the listed components were acceptable :( */
|
|
||||||
if(!any_acceptable) {
|
|
||||||
opal_show_help("help-mca-base.txt", "find-available:none-found", true,
|
|
||||||
type_name);
|
|
||||||
return OMPI_ERROR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Traverse the list of found components */
|
|
||||||
|
|
||||||
OBJ_CONSTRUCT(components_available, opal_list_t);
|
|
||||||
for (item = opal_list_get_first(components_found);
|
|
||||||
opal_list_get_end(components_found) != item;
|
|
||||||
item = opal_list_get_next(item)) {
|
|
||||||
cli = (mca_base_component_list_item_t *) item;
|
|
||||||
component = cli->cli_component;
|
|
||||||
|
|
||||||
/* Do we need to check for specific components? */
|
|
||||||
|
|
||||||
if (NULL != requested_component_names) {
|
|
||||||
acceptable = false;
|
|
||||||
for (i = 0; NULL != requested_component_names[i]; ++i) {
|
|
||||||
if (0 == strcmp(requested_component_names[i],
|
|
||||||
component->mca_component_name)) {
|
|
||||||
acceptable = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
acceptable = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If this is an acceptable component, try to open it */
|
|
||||||
|
|
||||||
if (acceptable) {
|
|
||||||
opened = called_open = false;
|
|
||||||
opal_output_verbose(10, output_id,
|
|
||||||
"mca: base: components_open: found loaded component %s",
|
|
||||||
component->mca_component_name);
|
|
||||||
|
|
||||||
if (NULL == component->mca_open_component) {
|
|
||||||
opened = true;
|
|
||||||
opal_output_verbose(10, output_id,
|
opal_output_verbose(10, output_id,
|
||||||
"mca: base: components_open: "
|
"mca: base: components_open: found loaded component %s",
|
||||||
"component %s has no open function",
|
|
||||||
component->mca_component_name);
|
component->mca_component_name);
|
||||||
} else {
|
|
||||||
called_open = true;
|
if (NULL == component->mca_open_component) {
|
||||||
if (MCA_SUCCESS == component->mca_open_component()) {
|
opened = true;
|
||||||
opened = true;
|
|
||||||
opal_output_verbose(10, output_id,
|
|
||||||
"mca: base: components_open: "
|
|
||||||
"component %s open function successful",
|
|
||||||
component->mca_component_name);
|
|
||||||
} else {
|
|
||||||
/* We may end up displaying this twice, but it may go to
|
|
||||||
separate streams. So better to be redundant than to
|
|
||||||
not display the error in the stream where it was
|
|
||||||
expected. */
|
|
||||||
|
|
||||||
if (show_errors) {
|
|
||||||
opal_output(0, "mca: base: components_open: "
|
|
||||||
"component %s / %s open function failed",
|
|
||||||
component->mca_type_name,
|
|
||||||
component->mca_component_name);
|
|
||||||
}
|
|
||||||
opal_output_verbose(10, output_id,
|
opal_output_verbose(10, output_id,
|
||||||
"mca: base: components_open: "
|
"mca: base: components_open: "
|
||||||
"component %s open function failed",
|
"component %s has no open function",
|
||||||
|
component->mca_component_name);
|
||||||
|
} else {
|
||||||
|
called_open = true;
|
||||||
|
if (MCA_SUCCESS == component->mca_open_component()) {
|
||||||
|
opened = true;
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"component %s open function successful",
|
||||||
|
component->mca_component_name);
|
||||||
|
} else {
|
||||||
|
/* We may end up displaying this twice, but it may go
|
||||||
|
to separate streams. So better to be redundant
|
||||||
|
than to not display the error in the stream where
|
||||||
|
it was expected. */
|
||||||
|
|
||||||
|
if (show_errors) {
|
||||||
|
opal_output(0, "mca: base: components_open: "
|
||||||
|
"component %s / %s open function failed",
|
||||||
|
component->mca_type_name,
|
||||||
|
component->mca_component_name);
|
||||||
|
}
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: "
|
||||||
|
"component %s open function failed",
|
||||||
|
component->mca_component_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If it didn't open, close it out and get rid of it */
|
||||||
|
|
||||||
|
if (!opened) {
|
||||||
|
if (called_open) {
|
||||||
|
if (NULL != component->mca_close_component) {
|
||||||
|
component->mca_close_component();
|
||||||
|
}
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: component %s closed",
|
||||||
|
component->mca_component_name);
|
||||||
|
called_open = false;
|
||||||
|
}
|
||||||
|
mca_base_component_repository_release(component);
|
||||||
|
opal_output_verbose(10, output_id,
|
||||||
|
"mca: base: components_open: component %s unloaded",
|
||||||
component->mca_component_name);
|
component->mca_component_name);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
/* If it did open, register its "priority" MCA parameter (if
|
||||||
/* If it didn't open, close it out and get rid of it */
|
it doesn't already have one) and save it in the
|
||||||
|
opened_components list */
|
||||||
if (!opened) {
|
|
||||||
if (called_open) {
|
else {
|
||||||
if (NULL != component->mca_close_component) {
|
if (OMPI_ERROR == mca_base_param_find(type_name,
|
||||||
component->mca_close_component();
|
component->mca_component_name,
|
||||||
}
|
"priority")) {
|
||||||
opal_output_verbose(10, output_id,
|
mca_base_param_register_int(type_name,
|
||||||
"mca: base: components_open: component %s closed",
|
component->mca_component_name,
|
||||||
component->mca_component_name);
|
"priority", NULL, 0);
|
||||||
called_open = false;
|
}
|
||||||
|
|
||||||
|
cli = OBJ_NEW(mca_base_component_list_item_t);
|
||||||
|
if (NULL == cli) {
|
||||||
|
return OMPI_ERR_OUT_OF_RESOURCE;
|
||||||
|
}
|
||||||
|
cli->cli_component = component;
|
||||||
|
opal_list_append(dest, (opal_list_item_t *) cli);
|
||||||
}
|
}
|
||||||
mca_base_component_repository_release(component);
|
|
||||||
opal_output_verbose(10, output_id,
|
|
||||||
"mca: base: components_open: component %s unloaded",
|
|
||||||
component->mca_component_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If it did open, register its "priority" MCA parameter (if it
|
|
||||||
doesn't already have one) and save it in the opened_components
|
|
||||||
list */
|
|
||||||
|
|
||||||
else {
|
|
||||||
if (OMPI_ERROR == mca_base_param_find(type_name,
|
|
||||||
component->mca_component_name,
|
|
||||||
"priority")) {
|
|
||||||
mca_base_param_register_int(type_name,
|
|
||||||
component->mca_component_name,
|
|
||||||
"priority", NULL, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
cli = OBJ_NEW(mca_base_component_list_item_t);
|
|
||||||
if (NULL == cli) {
|
|
||||||
return OMPI_ERR_OUT_OF_RESOURCE;
|
|
||||||
}
|
|
||||||
cli->cli_component = component;
|
|
||||||
opal_list_append(components_available, (opal_list_item_t *) cli);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
/* All done */
|
||||||
/* All done */
|
|
||||||
|
return OMPI_SUCCESS;
|
||||||
return OMPI_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
Загрузка…
Ссылка в новой задаче
Block a user