/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */ /* * Copyright (c) 2004-2008 The Trustees of Indiana University and Indiana * University Research and Technology * Corporation. All rights reserved. * Copyright (c) 2015 Los Alamos National Security, LLC. All rights * reserved. * Copyright (c) 2015-2020 Intel, Inc. All rights reserved. * Copyright (c) 2021-2022 Nanook Consulting. All rights reserved. * $COPYRIGHT$ * * Additional copyrights may follow * * $HEADER$ */ #include "src/include/pmix_config.h" #include #include #include #ifdef HAVE_SYS_TYPES_H # include #endif #include "pmix_common.h" #include "src/class/pmix_list.h" #include "src/mca/base/pmix_base.h" #include "src/mca/base/pmix_mca_base_component_repository.h" #include "src/mca/mca.h" #include "src/util/pmix_error.h" #include "src/util/pmix_output.h" int pmix_mca_base_select(const char *type_name, int output_id, pmix_list_t *components_available, pmix_mca_base_module_t **best_module, pmix_mca_base_component_t **best_component, int *priority_out) { pmix_mca_base_component_list_item_t *cli = NULL; pmix_mca_base_component_t *component = NULL; pmix_mca_base_module_t *module = NULL; int priority = 0, best_priority = INT32_MIN; int rc; *best_module = NULL; *best_component = NULL; pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select: Auto-selecting %s components", type_name); /* * Traverse the list of available components. * For each call their 'query' functions to determine relative priority. */ PMIX_LIST_FOREACH (cli, components_available, pmix_mca_base_component_list_item_t) { component = (pmix_mca_base_component_t *) cli->cli_component; /* * If there is a query function then use it. */ if (NULL == component->pmix_mca_query_component) { pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) Skipping component [%s]. It does not " "implement a query function", type_name, component->pmix_mca_component_name); continue; } /* * Query this component for the module and priority */ pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) Querying component [%s]", type_name, component->pmix_mca_component_name); rc = component->pmix_mca_query_component(&module, &priority); if (PMIX_ERR_FATAL == rc) { /* a fatal error was detected by this component - e.g., the * user specified a required element and the component could * not find it. In this case, we must not continue as we might * find some other component that could run, causing us to do * something the user didn't want */ return rc; } else if (PMIX_SUCCESS != rc) { /* silently skip this component */ continue; } /* * If no module was returned, then skip component */ if (NULL == module) { pmix_output_verbose( PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) Skipping component [%s]. Query failed to return a module", type_name, component->pmix_mca_component_name); continue; } /* * Determine if this is the best module we have seen by looking the priority */ pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) Query of component [%s] set priority to %d", type_name, component->pmix_mca_component_name, priority); if (priority > best_priority) { best_priority = priority; *best_component = component; *best_module = module; } } if (priority_out) { *priority_out = best_priority; } /* * Finished querying all components. * Make sure we found something in the process. */ if (NULL == *best_component) { pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) No component selected!", type_name); /* * Still close the non-selected components */ pmix_mca_base_components_close(0, /* Pass 0 to keep this from closing the pmix_output_handle */ components_available, NULL); return PMIX_ERR_NOT_FOUND; } pmix_output_verbose(PMIX_MCA_BASE_VERBOSE_COMPONENT, output_id, "mca:base:select:(%5s) Selected component [%s]", type_name, (*best_component)->pmix_mca_component_name); /* * Close the non-selected components */ pmix_mca_base_components_close(output_id, components_available, (pmix_mca_base_component_t *) (*best_component)); return PMIX_SUCCESS; }