4173ac6dd0
Ensure that the project, framework, component, and variable names are
lower than max lengths. This is a follow-on to
992a8e8297
, per discussion on
https://github.com/open-mpi/ompi/pull/5642.
Signed-off-by: Jeff Squyres <jsquyres@cisco.com>
2285 строки
70 KiB
C
2285 строки
70 KiB
C
/* -*- 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) 2004-2012 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-2015 Cisco Systems, Inc. All rights reserved.
|
|
* Copyright (c) 2012-2018 Los Alamos National Security, LLC. All rights
|
|
* reserved.
|
|
* Copyright (c) 2014-2016 Intel, Inc. All rights reserved.
|
|
* Copyright (c) 2015 Research Organization for Information Science
|
|
* and Technology (RIST). All rights reserved.
|
|
* Copyright (c) 2017 IBM Corporation. All rights reserved.
|
|
* $COPYRIGHT$
|
|
*
|
|
* Additional copyrights may follow
|
|
*
|
|
* $HEADER$
|
|
*/
|
|
|
|
#include "opal_config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_PARAM_H
|
|
#include <sys/param.h>
|
|
#endif
|
|
#include <errno.h>
|
|
|
|
#include "opal/include/opal_stdint.h"
|
|
#include "opal/mca/installdirs/installdirs.h"
|
|
#include "opal/util/os_path.h"
|
|
#include "opal/util/path.h"
|
|
#include "opal/util/show_help.h"
|
|
#include "opal/util/printf.h"
|
|
#include "opal/util/argv.h"
|
|
#include "opal/mca/mca.h"
|
|
#include "opal/mca/base/mca_base_vari.h"
|
|
#include "opal/constants.h"
|
|
#include "opal/util/output.h"
|
|
#include "opal/util/opal_environ.h"
|
|
#include "opal/runtime/opal.h"
|
|
|
|
/*
|
|
* local variables
|
|
*/
|
|
static opal_pointer_array_t mca_base_vars;
|
|
static const char *mca_prefix = OPAL_MCA_PREFIX;
|
|
static char *home = NULL;
|
|
static char *cwd = NULL;
|
|
bool mca_base_var_initialized = false;
|
|
static char * force_agg_path = NULL;
|
|
static char *mca_base_var_files = NULL;
|
|
static char *mca_base_envar_files = NULL;
|
|
static char **mca_base_var_file_list = NULL;
|
|
static char *mca_base_var_override_file = NULL;
|
|
static char *mca_base_var_file_prefix = NULL;
|
|
static char *mca_base_envar_file_prefix = NULL;
|
|
static char *mca_base_param_file_path = NULL;
|
|
char *mca_base_env_list = NULL;
|
|
char *mca_base_env_list_sep = MCA_BASE_ENV_LIST_SEP_DEFAULT;
|
|
char *mca_base_env_list_internal = NULL;
|
|
static bool mca_base_var_suppress_override_warning = false;
|
|
static opal_list_t mca_base_var_file_values;
|
|
static opal_list_t mca_base_envar_file_values;
|
|
static opal_list_t mca_base_var_override_values;
|
|
|
|
static int mca_base_var_count = 0;
|
|
|
|
static opal_hash_table_t mca_base_var_index_hash;
|
|
|
|
const char *ompi_var_type_names[] = {
|
|
"int",
|
|
"unsigned_int",
|
|
"unsigned_long",
|
|
"unsigned_long_long",
|
|
"size_t",
|
|
"string",
|
|
"version_string",
|
|
"bool",
|
|
"double",
|
|
"long",
|
|
"int32_t",
|
|
"uint32_t",
|
|
"int64_t",
|
|
"uint64_t",
|
|
};
|
|
|
|
const size_t ompi_var_type_sizes[] = {
|
|
sizeof (int),
|
|
sizeof (unsigned),
|
|
sizeof (unsigned long),
|
|
sizeof (unsigned long long),
|
|
sizeof (size_t),
|
|
sizeof (char),
|
|
sizeof (char),
|
|
sizeof (bool),
|
|
sizeof (double),
|
|
sizeof (long),
|
|
sizeof (int32_t),
|
|
sizeof (uint32_t),
|
|
sizeof (int64_t),
|
|
sizeof (uint64_t),
|
|
};
|
|
|
|
static const char *var_source_names[] = {
|
|
"default",
|
|
"command line",
|
|
"environment",
|
|
"file",
|
|
"set",
|
|
"override"
|
|
};
|
|
|
|
|
|
static const char *info_lvl_strings[] = {
|
|
"user/basic",
|
|
"user/detail",
|
|
"user/all",
|
|
"tuner/basic",
|
|
"tuner/detail",
|
|
"tuner/all",
|
|
"dev/basic",
|
|
"dev/detail",
|
|
"dev/all"
|
|
};
|
|
|
|
/*
|
|
* local functions
|
|
*/
|
|
static int fixup_files(char **file_list, char * path, bool rel_path_search, char sep);
|
|
static int read_files (char *file_list, opal_list_t *file_values, char sep);
|
|
static int var_set_initial (mca_base_var_t *var, mca_base_var_t *original);
|
|
static int var_get (int vari, mca_base_var_t **var_out, bool original);
|
|
static int var_value_string (mca_base_var_t *var, char **value_string);
|
|
|
|
/*
|
|
* classes
|
|
*/
|
|
static void var_constructor (mca_base_var_t *p);
|
|
static void var_destructor (mca_base_var_t *p);
|
|
OBJ_CLASS_INSTANCE(mca_base_var_t, opal_object_t,
|
|
var_constructor, var_destructor);
|
|
|
|
static void fv_constructor (mca_base_var_file_value_t *p);
|
|
static void fv_destructor (mca_base_var_file_value_t *p);
|
|
OBJ_CLASS_INSTANCE(mca_base_var_file_value_t, opal_list_item_t,
|
|
fv_constructor, fv_destructor);
|
|
|
|
static const char *mca_base_var_source_file (const mca_base_var_t *var)
|
|
{
|
|
mca_base_var_file_value_t *fv = (mca_base_var_file_value_t *) var->mbv_file_value;
|
|
|
|
if (NULL != var->mbv_source_file) {
|
|
return var->mbv_source_file;
|
|
}
|
|
|
|
if (fv) {
|
|
return fv->mbvfv_file;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Generate a full name from three names
|
|
*/
|
|
int mca_base_var_generate_full_name4 (const char *project, const char *framework, const char *component,
|
|
const char *variable, char **full_name)
|
|
{
|
|
const char * const names[] = {project, framework, component, variable};
|
|
char *name, *tmp;
|
|
size_t i, len;
|
|
|
|
*full_name = NULL;
|
|
|
|
for (i = 0, len = 0 ; i < 4 ; ++i) {
|
|
if (NULL != names[i]) {
|
|
/* Add space for the string + _ or \0 */
|
|
len += strlen (names[i]) + 1;
|
|
}
|
|
}
|
|
|
|
name = calloc (1, len);
|
|
if (NULL == name) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
for (i = 0, tmp = name ; i < 4 ; ++i) {
|
|
if (NULL != names[i]) {
|
|
if (name != tmp) {
|
|
*tmp++ = '_';
|
|
}
|
|
strncat (name, names[i], len - (size_t)(uintptr_t)(tmp - name));
|
|
tmp += strlen (names[i]);
|
|
}
|
|
}
|
|
|
|
*full_name = name;
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static int compare_strings (const char *str1, const char *str2) {
|
|
if ((NULL != str1 && 0 == strcmp (str1, "*")) ||
|
|
(NULL == str1 && NULL == str2)) {
|
|
return 0;
|
|
}
|
|
|
|
if (NULL != str1 && NULL != str2) {
|
|
return strcmp (str1, str2);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Append a filename to the file list if it does not exist and return a
|
|
* pointer to the filename in the list.
|
|
*/
|
|
static char *append_filename_to_list(const char *filename)
|
|
{
|
|
int i, count;
|
|
|
|
(void) opal_argv_append_unique_nosize(&mca_base_var_file_list, filename, false);
|
|
|
|
count = opal_argv_count(mca_base_var_file_list);
|
|
|
|
for (i = count - 1; i >= 0; --i) {
|
|
if (0 == strcmp (mca_base_var_file_list[i], filename)) {
|
|
return mca_base_var_file_list[i];
|
|
}
|
|
}
|
|
|
|
/* *#@*? */
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Set it up
|
|
*/
|
|
int mca_base_var_init(void)
|
|
{
|
|
int ret;
|
|
|
|
if (!mca_base_var_initialized) {
|
|
/* Init the value array for the param storage */
|
|
|
|
OBJ_CONSTRUCT(&mca_base_vars, opal_pointer_array_t);
|
|
/* These values are arbitrary */
|
|
ret = opal_pointer_array_init (&mca_base_vars, 128, 16384, 128);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
mca_base_var_count = 0;
|
|
|
|
/* Init the file param value list */
|
|
|
|
OBJ_CONSTRUCT(&mca_base_var_file_values, opal_list_t);
|
|
OBJ_CONSTRUCT(&mca_base_envar_file_values, opal_list_t);
|
|
OBJ_CONSTRUCT(&mca_base_var_override_values, opal_list_t);
|
|
OBJ_CONSTRUCT(&mca_base_var_index_hash, opal_hash_table_t);
|
|
|
|
ret = opal_hash_table_init (&mca_base_var_index_hash, 1024);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = mca_base_var_group_init ();
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = mca_base_pvar_init ();
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
/* Set this before we register the parameter, below */
|
|
|
|
mca_base_var_initialized = true;
|
|
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static void process_env_list(char *env_list, char ***argv, char sep)
|
|
{
|
|
char** tokens;
|
|
char *ptr, *value;
|
|
|
|
tokens = opal_argv_split(env_list, (int)sep);
|
|
if (NULL == tokens) {
|
|
return;
|
|
}
|
|
|
|
for (int i = 0 ; NULL != tokens[i] ; ++i) {
|
|
if (NULL == (ptr = strchr(tokens[i], '='))) {
|
|
value = getenv(tokens[i]);
|
|
if (NULL == value) {
|
|
opal_show_help("help-mca-var.txt", "incorrect-env-list-param",
|
|
true, tokens[i], env_list);
|
|
break;
|
|
}
|
|
|
|
/* duplicate the value to silence tainted string coverity issue */
|
|
value = strdup (value);
|
|
if (NULL == value) {
|
|
/* out of memory */
|
|
break;
|
|
}
|
|
|
|
if (NULL != (ptr = strchr(value, '='))) {
|
|
*ptr = '\0';
|
|
opal_setenv(value, ptr + 1, true, argv);
|
|
} else {
|
|
opal_setenv(tokens[i], value, true, argv);
|
|
}
|
|
|
|
free (value);
|
|
} else {
|
|
*ptr = '\0';
|
|
opal_setenv(tokens[i], ptr + 1, true, argv);
|
|
/* NTH: don't bother resetting ptr to = since the string will not be used again */
|
|
}
|
|
}
|
|
|
|
opal_argv_free(tokens);
|
|
}
|
|
|
|
int mca_base_var_process_env_list(char *list, char ***argv)
|
|
{
|
|
char sep;
|
|
sep = ';';
|
|
if (NULL != mca_base_env_list_sep) {
|
|
if (1 == strlen(mca_base_env_list_sep)) {
|
|
sep = mca_base_env_list_sep[0];
|
|
} else {
|
|
opal_show_help("help-mca-var.txt", "incorrect-env-list-sep",
|
|
true, mca_base_env_list_sep);
|
|
return OPAL_SUCCESS;
|
|
}
|
|
}
|
|
if (NULL != list) {
|
|
process_env_list(list, argv, sep);
|
|
} else if (NULL != mca_base_env_list) {
|
|
process_env_list(mca_base_env_list, argv, sep);
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
int mca_base_var_process_env_list_from_file(char ***argv)
|
|
{
|
|
if (NULL != mca_base_env_list_internal) {
|
|
process_env_list(mca_base_env_list_internal, argv, ';');
|
|
}
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static void resolve_relative_paths(char **file_prefix, char *file_path, bool rel_path_search, char **files, char sep)
|
|
{
|
|
char *tmp_str;
|
|
/*
|
|
* Resolve all relative paths.
|
|
* the file list returned will contain only absolute paths
|
|
*/
|
|
if( OPAL_SUCCESS != fixup_files(file_prefix, file_path, rel_path_search, sep) ) {
|
|
#if 0
|
|
/* JJH We need to die! */
|
|
abort();
|
|
#else
|
|
;
|
|
#endif
|
|
}
|
|
else {
|
|
/* Prepend the files to the search list */
|
|
asprintf(&tmp_str, "%s%c%s", *file_prefix, sep, *files);
|
|
free (*files);
|
|
*files = tmp_str;
|
|
}
|
|
}
|
|
|
|
int mca_base_var_cache_files(bool rel_path_search)
|
|
{
|
|
char *tmp;
|
|
int ret;
|
|
|
|
/* We may need this later */
|
|
home = (char*)opal_home_directory();
|
|
|
|
if (NULL == cwd) {
|
|
cwd = (char *) malloc(sizeof(char) * MAXPATHLEN);
|
|
if( NULL == (cwd = getcwd(cwd, MAXPATHLEN) )) {
|
|
opal_output(0, "Error: Unable to get the current working directory\n");
|
|
cwd = strdup(".");
|
|
}
|
|
}
|
|
|
|
#if OPAL_WANT_HOME_CONFIG_FILES
|
|
asprintf(&mca_base_var_files, "%s"OPAL_PATH_SEP".openmpi" OPAL_PATH_SEP
|
|
"mca-params.conf%c%s" OPAL_PATH_SEP "openmpi-mca-params.conf",
|
|
home, ',', opal_install_dirs.sysconfdir);
|
|
#else
|
|
asprintf(&mca_base_var_files, "%s" OPAL_PATH_SEP "openmpi-mca-params.conf",
|
|
opal_install_dirs.sysconfdir);
|
|
#endif
|
|
|
|
/* Initialize a parameter that says where MCA param files can be found.
|
|
We may change this value so set the scope to MCA_BASE_VAR_SCOPE_READONLY */
|
|
tmp = mca_base_var_files;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "param_files", "Path for MCA "
|
|
"configuration files containing variable values",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, 0, OPAL_INFO_LVL_2,
|
|
MCA_BASE_VAR_SCOPE_READONLY, &mca_base_var_files);
|
|
free (tmp);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
mca_base_envar_files = strdup(mca_base_var_files);
|
|
|
|
(void) mca_base_var_register_synonym (ret, "opal", "mca", NULL, "param_files",
|
|
MCA_BASE_VAR_SYN_FLAG_DEPRECATED);
|
|
|
|
ret = asprintf(&mca_base_var_override_file, "%s" OPAL_PATH_SEP "openmpi-mca-params-override.conf",
|
|
opal_install_dirs.sysconfdir);
|
|
if (0 > ret) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
tmp = mca_base_var_override_file;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "override_param_file",
|
|
"Variables set in this file will override any value set in"
|
|
"the environment or another configuration file",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, MCA_BASE_VAR_FLAG_DEFAULT_ONLY,
|
|
OPAL_INFO_LVL_2, MCA_BASE_VAR_SCOPE_CONSTANT,
|
|
&mca_base_var_override_file);
|
|
free (tmp);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
/* Disable reading MCA parameter files. */
|
|
if (0 == strcmp (mca_base_var_files, "none")) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
mca_base_var_suppress_override_warning = false;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "suppress_override_warning",
|
|
"Suppress warnings when attempting to set an overridden value (default: false)",
|
|
MCA_BASE_VAR_TYPE_BOOL, NULL, 0, 0, OPAL_INFO_LVL_2,
|
|
MCA_BASE_VAR_SCOPE_LOCAL, &mca_base_var_suppress_override_warning);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
/* Aggregate MCA parameter files
|
|
* A prefix search path to look up aggregate MCA parameter file
|
|
* requests that do not specify an absolute path
|
|
*/
|
|
mca_base_var_file_prefix = NULL;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "param_file_prefix",
|
|
"Aggregate MCA parameter file sets",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, 0, OPAL_INFO_LVL_3,
|
|
MCA_BASE_VAR_SCOPE_READONLY, &mca_base_var_file_prefix);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
mca_base_envar_file_prefix = NULL;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "envar_file_prefix",
|
|
"Aggregate MCA parameter file set for env variables",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, 0, OPAL_INFO_LVL_3,
|
|
MCA_BASE_VAR_SCOPE_READONLY, &mca_base_envar_file_prefix);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = asprintf(&mca_base_param_file_path, "%s" OPAL_PATH_SEP "amca-param-sets%c%s",
|
|
opal_install_dirs.opaldatadir, OPAL_ENV_SEP, cwd);
|
|
if (0 > ret) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
tmp = mca_base_param_file_path;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "param_file_path",
|
|
"Aggregate MCA parameter Search path",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, 0, OPAL_INFO_LVL_3,
|
|
MCA_BASE_VAR_SCOPE_READONLY, &mca_base_param_file_path);
|
|
free (tmp);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
force_agg_path = NULL;
|
|
ret = mca_base_var_register ("opal", "mca", "base", "param_file_path_force",
|
|
"Forced Aggregate MCA parameter Search path",
|
|
MCA_BASE_VAR_TYPE_STRING, NULL, 0, 0, OPAL_INFO_LVL_3,
|
|
MCA_BASE_VAR_SCOPE_READONLY, &force_agg_path);
|
|
if (0 > ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (NULL != force_agg_path) {
|
|
if (NULL != mca_base_param_file_path) {
|
|
char *tmp_str = mca_base_param_file_path;
|
|
|
|
asprintf(&mca_base_param_file_path, "%s%c%s", force_agg_path, OPAL_ENV_SEP, tmp_str);
|
|
free(tmp_str);
|
|
} else {
|
|
mca_base_param_file_path = strdup(force_agg_path);
|
|
}
|
|
}
|
|
|
|
if (NULL != mca_base_var_file_prefix) {
|
|
resolve_relative_paths(&mca_base_var_file_prefix, mca_base_param_file_path, rel_path_search, &mca_base_var_files, OPAL_ENV_SEP);
|
|
}
|
|
read_files (mca_base_var_files, &mca_base_var_file_values, ',');
|
|
|
|
if (NULL != mca_base_envar_file_prefix) {
|
|
resolve_relative_paths(&mca_base_envar_file_prefix, mca_base_param_file_path, rel_path_search, &mca_base_envar_files, ',');
|
|
}
|
|
read_files (mca_base_envar_files, &mca_base_envar_file_values, ',');
|
|
|
|
if (0 == access(mca_base_var_override_file, F_OK)) {
|
|
read_files (mca_base_var_override_file, &mca_base_var_override_values, OPAL_ENV_SEP);
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Look up an integer MCA parameter.
|
|
*/
|
|
int mca_base_var_get_value (int vari, const void *value,
|
|
mca_base_var_source_t *source,
|
|
const char **source_file)
|
|
{
|
|
mca_base_var_t *var;
|
|
void **tmp = (void **) value;
|
|
int ret;
|
|
|
|
ret = var_get (vari, &var, true);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (!VAR_IS_VALID(var[0])) {
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (NULL != value) {
|
|
/* Return a poiner to our backing store (either a char **, int *,
|
|
or bool *) */
|
|
*tmp = var->mbv_storage;
|
|
}
|
|
|
|
if (NULL != source) {
|
|
*source = var->mbv_source;
|
|
}
|
|
|
|
if (NULL != source_file) {
|
|
*source_file = mca_base_var_source_file (var);
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static int var_set_string (mca_base_var_t *var, char *value)
|
|
{
|
|
char *tmp;
|
|
int ret;
|
|
|
|
if (NULL != var->mbv_storage->stringval) {
|
|
free (var->mbv_storage->stringval);
|
|
}
|
|
|
|
var->mbv_storage->stringval = NULL;
|
|
|
|
if (NULL == value || 0 == strlen (value)) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/* Replace all instances of ~/ in a path-style string with the
|
|
user's home directory. This may be handled by the enumerator
|
|
in the future. */
|
|
if (0 == strncmp (value, "~/", 2)) {
|
|
if (NULL != home) {
|
|
ret = asprintf (&value, "%s/%s", home, value + 2);
|
|
if (0 > ret) {
|
|
return OPAL_ERROR;
|
|
}
|
|
} else {
|
|
value = strdup (value + 2);
|
|
}
|
|
} else {
|
|
value = strdup (value);
|
|
}
|
|
|
|
if (NULL == value) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
while (NULL != (tmp = strstr (value, ":~/"))) {
|
|
tmp[0] = '\0';
|
|
tmp += 3;
|
|
|
|
ret = asprintf (&tmp, "%s:%s%s%s", value,
|
|
home ? home : "", home ? "/" : "", tmp);
|
|
|
|
free (value);
|
|
|
|
if (0 > ret) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
value = tmp;
|
|
}
|
|
|
|
var->mbv_storage->stringval = value;
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static int int_from_string(const char *src, mca_base_var_enum_t *enumerator, uint64_t *value_out)
|
|
{
|
|
uint64_t value;
|
|
bool is_int;
|
|
char *tmp;
|
|
|
|
if (NULL == src || 0 == strlen (src)) {
|
|
if (NULL == enumerator) {
|
|
*value_out = 0;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
if (enumerator) {
|
|
int int_val, ret;
|
|
ret = enumerator->value_from_string(enumerator, src, &int_val);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
*value_out = (uint64_t) int_val;
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/* Check for an integer value */
|
|
value = strtoull (src, &tmp, 0);
|
|
is_int = tmp[0] == '\0';
|
|
|
|
if (!is_int && tmp != src) {
|
|
switch (tmp[0]) {
|
|
case 'G':
|
|
case 'g':
|
|
value <<= 10;
|
|
case 'M':
|
|
case 'm':
|
|
value <<= 10;
|
|
case 'K':
|
|
case 'k':
|
|
value <<= 10;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
*value_out = value;
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static int var_set_from_string (mca_base_var_t *var, char *src)
|
|
{
|
|
mca_base_var_storage_t *dst = var->mbv_storage;
|
|
uint64_t int_value = 0;
|
|
int ret;
|
|
|
|
switch (var->mbv_type) {
|
|
case MCA_BASE_VAR_TYPE_INT:
|
|
case MCA_BASE_VAR_TYPE_INT32_T:
|
|
case MCA_BASE_VAR_TYPE_UINT32_T:
|
|
case MCA_BASE_VAR_TYPE_LONG:
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_INT:
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG:
|
|
case MCA_BASE_VAR_TYPE_INT64_T:
|
|
case MCA_BASE_VAR_TYPE_UINT64_T:
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG_LONG:
|
|
case MCA_BASE_VAR_TYPE_BOOL:
|
|
case MCA_BASE_VAR_TYPE_SIZE_T:
|
|
ret = int_from_string(src, var->mbv_enumerator, &int_value);
|
|
if (OPAL_SUCCESS != ret ||
|
|
(MCA_BASE_VAR_TYPE_INT == var->mbv_type && ((int) int_value != (int64_t) int_value)) ||
|
|
(MCA_BASE_VAR_TYPE_UNSIGNED_INT == var->mbv_type && ((unsigned int) int_value != int_value))) {
|
|
if (var->mbv_enumerator) {
|
|
char *valid_values;
|
|
(void) var->mbv_enumerator->dump(var->mbv_enumerator, &valid_values);
|
|
opal_show_help("help-mca-var.txt", "invalid-value-enum",
|
|
true, var->mbv_full_name, src, valid_values);
|
|
free(valid_values);
|
|
} else {
|
|
opal_show_help("help-mca-var.txt", "invalid-value",
|
|
true, var->mbv_full_name, src);
|
|
}
|
|
|
|
return OPAL_ERR_VALUE_OUT_OF_BOUNDS;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_TYPE_INT == var->mbv_type ||
|
|
MCA_BASE_VAR_TYPE_UNSIGNED_INT == var->mbv_type) {
|
|
int *castme = (int*) var->mbv_storage;
|
|
*castme = int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_INT32_T == var->mbv_type ||
|
|
MCA_BASE_VAR_TYPE_UINT32_T == var->mbv_type) {
|
|
int32_t *castme = (int32_t *) var->mbv_storage;
|
|
*castme = int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_INT64_T == var->mbv_type ||
|
|
MCA_BASE_VAR_TYPE_UINT64_T == var->mbv_type) {
|
|
int64_t *castme = (int64_t *) var->mbv_storage;
|
|
*castme = int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_LONG == var->mbv_type) {
|
|
long *castme = (long*) var->mbv_storage;
|
|
*castme = (long) int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_UNSIGNED_LONG == var->mbv_type) {
|
|
unsigned long *castme = (unsigned long*) var->mbv_storage;
|
|
*castme = (unsigned long) int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_UNSIGNED_LONG_LONG == var->mbv_type) {
|
|
unsigned long long *castme = (unsigned long long*) var->mbv_storage;
|
|
*castme = (unsigned long long) int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_SIZE_T == var->mbv_type) {
|
|
size_t *castme = (size_t*) var->mbv_storage;
|
|
*castme = (size_t) int_value;
|
|
} else if (MCA_BASE_VAR_TYPE_BOOL == var->mbv_type) {
|
|
bool *castme = (bool*) var->mbv_storage;
|
|
*castme = !!int_value;
|
|
}
|
|
|
|
return ret;
|
|
case MCA_BASE_VAR_TYPE_DOUBLE:
|
|
dst->lfval = strtod (src, NULL);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_STRING:
|
|
case MCA_BASE_VAR_TYPE_VERSION_STRING:
|
|
var_set_string (var, src);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_MAX:
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Set a variable
|
|
*/
|
|
int mca_base_var_set_value (int vari, const void *value, size_t size, mca_base_var_source_t source,
|
|
const char *source_file)
|
|
{
|
|
mca_base_var_t *var;
|
|
int ret;
|
|
|
|
ret = var_get (vari, &var, true);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (!VAR_IS_VALID(var[0])) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
if (!VAR_IS_SETTABLE(var[0])) {
|
|
return OPAL_ERR_PERM;
|
|
}
|
|
|
|
if (NULL != var->mbv_enumerator) {
|
|
/* Validate */
|
|
ret = var->mbv_enumerator->string_from_value(var->mbv_enumerator,
|
|
((int *) value)[0], NULL);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
if (MCA_BASE_VAR_TYPE_STRING != var->mbv_type && MCA_BASE_VAR_TYPE_VERSION_STRING != var->mbv_type) {
|
|
memmove (var->mbv_storage, value, ompi_var_type_sizes[var->mbv_type]);
|
|
} else {
|
|
var_set_string (var, (char *) value);
|
|
}
|
|
|
|
var->mbv_source = source;
|
|
|
|
if (MCA_BASE_VAR_SOURCE_FILE == source && NULL != source_file) {
|
|
var->mbv_file_value = NULL;
|
|
var->mbv_source_file = append_filename_to_list(source_file);
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Deregister a parameter
|
|
*/
|
|
int mca_base_var_deregister(int vari)
|
|
{
|
|
mca_base_var_t *var;
|
|
int ret;
|
|
|
|
ret = var_get (vari, &var, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (!VAR_IS_VALID(var[0])) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
/* Mark this parameter as invalid but keep its info in case this
|
|
parameter is reregistered later */
|
|
var->mbv_flags &= ~MCA_BASE_VAR_FLAG_VALID;
|
|
|
|
/* Done deregistering synonym */
|
|
if (MCA_BASE_VAR_FLAG_SYNONYM & var->mbv_flags) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/* Release the current value if it is a string. */
|
|
if ((MCA_BASE_VAR_TYPE_STRING == var->mbv_type || MCA_BASE_VAR_TYPE_VERSION_STRING == var->mbv_type) &&
|
|
var->mbv_storage->stringval) {
|
|
free (var->mbv_storage->stringval);
|
|
var->mbv_storage->stringval = NULL;
|
|
} else if (var->mbv_enumerator && !var->mbv_enumerator->enum_is_static) {
|
|
OBJ_RELEASE(var->mbv_enumerator);
|
|
}
|
|
|
|
var->mbv_enumerator = NULL;
|
|
|
|
var->mbv_storage = NULL;
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
static int var_get (int vari, mca_base_var_t **var_out, bool original)
|
|
{
|
|
mca_base_var_t *var;
|
|
|
|
if (var_out) {
|
|
*var_out = NULL;
|
|
}
|
|
|
|
/* Check for bozo cases */
|
|
if (!mca_base_var_initialized) {
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
if (vari < 0) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
var = opal_pointer_array_get_item (&mca_base_vars, vari);
|
|
if (NULL == var) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
if (VAR_IS_SYNONYM(var[0]) && original) {
|
|
return var_get(var->mbv_synonym_for, var_out, false);
|
|
}
|
|
|
|
if (var_out) {
|
|
*var_out = var;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
int mca_base_var_env_name(const char *param_name,
|
|
char **env_name)
|
|
{
|
|
int ret;
|
|
|
|
assert (NULL != env_name);
|
|
|
|
ret = asprintf(env_name, "%s%s", mca_prefix, param_name);
|
|
if (0 > ret) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Find the index for an MCA parameter based on its names.
|
|
*/
|
|
static int var_find_by_name (const char *full_name, int *vari, bool invalidok)
|
|
{
|
|
mca_base_var_t *var = NULL;
|
|
void *tmp;
|
|
int rc;
|
|
|
|
rc = opal_hash_table_get_value_ptr (&mca_base_var_index_hash, full_name, strlen (full_name),
|
|
&tmp);
|
|
if (OPAL_SUCCESS != rc) {
|
|
return rc;
|
|
}
|
|
|
|
(void) var_get ((int)(uintptr_t) tmp, &var, false);
|
|
|
|
if (invalidok || (var && VAR_IS_VALID(var[0]))) {
|
|
*vari = (int)(uintptr_t) tmp;
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
static int var_find (const char *project_name, const char *framework_name,
|
|
const char *component_name, const char *variable_name,
|
|
bool invalidok)
|
|
{
|
|
char *full_name;
|
|
int ret, vari;
|
|
|
|
ret = mca_base_var_generate_full_name4 (NULL, framework_name, component_name,
|
|
variable_name, &full_name);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
ret = var_find_by_name(full_name, &vari, invalidok);
|
|
|
|
/* NTH: should we verify the name components match? */
|
|
|
|
free (full_name);
|
|
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
return vari;
|
|
}
|
|
|
|
/*
|
|
* Find the index for an MCA parameter based on its name components.
|
|
*/
|
|
int mca_base_var_find (const char *project_name, const char *framework_name,
|
|
const char *component_name, const char *variable_name)
|
|
{
|
|
return var_find (project_name, framework_name, component_name, variable_name, false);
|
|
}
|
|
|
|
/*
|
|
* Find the index for an MCA parameter based on full name.
|
|
*/
|
|
int mca_base_var_find_by_name (const char *full_name, int *vari)
|
|
{
|
|
return var_find_by_name (full_name, vari, false);
|
|
}
|
|
|
|
int mca_base_var_set_flag (int vari, mca_base_var_flag_t flag, bool set)
|
|
{
|
|
mca_base_var_t *var;
|
|
int ret;
|
|
|
|
ret = var_get (vari, &var, true);
|
|
if (OPAL_SUCCESS != ret || VAR_IS_SYNONYM(var[0])) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
var->mbv_flags = (var->mbv_flags & ~flag) | (set ? flag : 0);
|
|
|
|
/* All done */
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Return info on a parameter at an index
|
|
*/
|
|
int mca_base_var_get (int vari, const mca_base_var_t **var)
|
|
{
|
|
int ret;
|
|
ret = var_get (vari, (mca_base_var_t **) var, false);
|
|
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (!VAR_IS_VALID(*(var[0]))) {
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Make an argv-style list of strings suitable for an environment
|
|
*/
|
|
int mca_base_var_build_env(char ***env, int *num_env, bool internal)
|
|
{
|
|
mca_base_var_t *var;
|
|
size_t i, len;
|
|
int ret;
|
|
|
|
/* Check for bozo cases */
|
|
|
|
if (!mca_base_var_initialized) {
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
/* Iterate through all the registered parameters */
|
|
|
|
len = opal_pointer_array_get_size(&mca_base_vars);
|
|
for (i = 0; i < len; ++i) {
|
|
char *value_string;
|
|
char *str = NULL;
|
|
|
|
var = opal_pointer_array_get_item (&mca_base_vars, i);
|
|
if (NULL == var) {
|
|
continue;
|
|
}
|
|
|
|
/* Don't output default values or internal variables (unless
|
|
requested) */
|
|
if (MCA_BASE_VAR_SOURCE_DEFAULT == var->mbv_source ||
|
|
(!internal && VAR_IS_INTERNAL(var[0]))) {
|
|
continue;
|
|
}
|
|
|
|
if ((MCA_BASE_VAR_TYPE_STRING == var->mbv_type || MCA_BASE_VAR_TYPE_VERSION_STRING == var->mbv_type) &&
|
|
NULL == var->mbv_storage->stringval) {
|
|
continue;
|
|
}
|
|
|
|
ret = var_value_string (var, &value_string);
|
|
if (OPAL_SUCCESS != ret) {
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = asprintf (&str, "%s%s=%s", mca_prefix, var->mbv_full_name,
|
|
value_string);
|
|
free (value_string);
|
|
if (0 > ret) {
|
|
goto cleanup;
|
|
}
|
|
|
|
opal_argv_append(num_env, env, str);
|
|
free(str);
|
|
|
|
switch (var->mbv_source) {
|
|
case MCA_BASE_VAR_SOURCE_FILE:
|
|
case MCA_BASE_VAR_SOURCE_OVERRIDE:
|
|
asprintf (&str, "%sSOURCE_%s=FILE:%s", mca_prefix, var->mbv_full_name,
|
|
mca_base_var_source_file (var));
|
|
break;
|
|
case MCA_BASE_VAR_SOURCE_COMMAND_LINE:
|
|
asprintf (&str, "%sSOURCE_%s=COMMAND_LINE", mca_prefix, var->mbv_full_name);
|
|
break;
|
|
case MCA_BASE_VAR_SOURCE_ENV:
|
|
case MCA_BASE_VAR_SOURCE_SET:
|
|
case MCA_BASE_VAR_SOURCE_DEFAULT:
|
|
str = NULL;
|
|
break;
|
|
case MCA_BASE_VAR_SOURCE_MAX:
|
|
goto cleanup;
|
|
}
|
|
|
|
if (NULL != str) {
|
|
opal_argv_append(num_env, env, str);
|
|
free(str);
|
|
}
|
|
}
|
|
|
|
/* All done */
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
/* Error condition */
|
|
|
|
cleanup:
|
|
if (*num_env > 0) {
|
|
opal_argv_free(*env);
|
|
*num_env = 0;
|
|
*env = NULL;
|
|
}
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
/*
|
|
* Shut down the MCA parameter system (normally only invoked by the
|
|
* MCA framework itself).
|
|
*/
|
|
int mca_base_var_finalize(void)
|
|
{
|
|
opal_object_t *object;
|
|
opal_list_item_t *item;
|
|
int size, i;
|
|
|
|
if (mca_base_var_initialized) {
|
|
size = opal_pointer_array_get_size(&mca_base_vars);
|
|
for (i = 0 ; i < size ; ++i) {
|
|
object = opal_pointer_array_get_item (&mca_base_vars, i);
|
|
if (NULL != object) {
|
|
OBJ_RELEASE(object);
|
|
}
|
|
}
|
|
OBJ_DESTRUCT(&mca_base_vars);
|
|
|
|
while (NULL !=
|
|
(item = opal_list_remove_first(&mca_base_var_file_values))) {
|
|
OBJ_RELEASE(item);
|
|
}
|
|
OBJ_DESTRUCT(&mca_base_var_file_values);
|
|
|
|
while (NULL !=
|
|
(item = opal_list_remove_first(&mca_base_envar_file_values))) {
|
|
OBJ_RELEASE(item);
|
|
}
|
|
OBJ_DESTRUCT(&mca_base_envar_file_values);
|
|
|
|
while (NULL !=
|
|
(item = opal_list_remove_first(&mca_base_var_override_values))) {
|
|
OBJ_RELEASE(item);
|
|
}
|
|
OBJ_DESTRUCT(&mca_base_var_override_values);
|
|
|
|
if( NULL != cwd ) {
|
|
free(cwd);
|
|
cwd = NULL;
|
|
}
|
|
|
|
mca_base_var_initialized = false;
|
|
mca_base_var_count = 0;
|
|
|
|
if (NULL != mca_base_var_file_list) {
|
|
opal_argv_free(mca_base_var_file_list);
|
|
}
|
|
mca_base_var_file_list = NULL;
|
|
|
|
(void) mca_base_var_group_finalize ();
|
|
(void) mca_base_pvar_finalize ();
|
|
|
|
OBJ_DESTRUCT(&mca_base_var_index_hash);
|
|
|
|
free (mca_base_envar_files);
|
|
mca_base_envar_files = NULL;
|
|
}
|
|
|
|
/* All done */
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
|
|
/*************************************************************************/
|
|
static int fixup_files(char **file_list, char * path, bool rel_path_search, char sep) {
|
|
int exit_status = OPAL_SUCCESS;
|
|
char **files = NULL;
|
|
char **search_path = NULL;
|
|
char * tmp_file = NULL;
|
|
char **argv = NULL;
|
|
char *rel_path;
|
|
int mode = R_OK; /* The file exists, and we can read it */
|
|
int count, i, argc = 0;
|
|
|
|
search_path = opal_argv_split(path, OPAL_ENV_SEP);
|
|
files = opal_argv_split(*file_list, sep);
|
|
count = opal_argv_count(files);
|
|
|
|
rel_path = force_agg_path ? force_agg_path : cwd;
|
|
|
|
/* Read in reverse order, so we can preserve the original ordering */
|
|
for (i = 0 ; i < count; ++i) {
|
|
char *msg_path = path;
|
|
if (opal_path_is_absolute(files[i])) {
|
|
/* Absolute paths preserved */
|
|
tmp_file = opal_path_access(files[i], NULL, mode);
|
|
} else if (!rel_path_search && NULL != strchr(files[i], OPAL_PATH_SEP[0])) {
|
|
/* Resolve all relative paths:
|
|
* - If filename contains a "/" (e.g., "./foo" or "foo/bar")
|
|
* - look for it relative to cwd
|
|
* - if exists, use it
|
|
* - ow warn/error
|
|
*/
|
|
msg_path = rel_path;
|
|
tmp_file = opal_path_access(files[i], rel_path, mode);
|
|
} else {
|
|
/* Resolve all relative paths:
|
|
* - Use path resolution
|
|
* - if found and readable, use it
|
|
* - otherwise, warn/error
|
|
*/
|
|
tmp_file = opal_path_find (files[i], search_path, mode, NULL);
|
|
}
|
|
|
|
if (NULL == tmp_file) {
|
|
opal_show_help("help-mca-var.txt", "missing-param-file",
|
|
true, getpid(), files[i], msg_path);
|
|
exit_status = OPAL_ERROR;
|
|
break;
|
|
}
|
|
|
|
opal_argv_append(&argc, &argv, tmp_file);
|
|
|
|
free(tmp_file);
|
|
tmp_file = NULL;
|
|
}
|
|
|
|
if (OPAL_SUCCESS == exit_status) {
|
|
free(*file_list);
|
|
*file_list = opal_argv_join(argv, sep);
|
|
}
|
|
|
|
if( NULL != files ) {
|
|
opal_argv_free(files);
|
|
files = NULL;
|
|
}
|
|
|
|
if( NULL != argv ) {
|
|
opal_argv_free(argv);
|
|
argv = NULL;
|
|
}
|
|
|
|
if( NULL != search_path ) {
|
|
opal_argv_free(search_path);
|
|
search_path = NULL;
|
|
}
|
|
|
|
return exit_status;
|
|
}
|
|
|
|
static int read_files(char *file_list, opal_list_t *file_values, char sep)
|
|
{
|
|
char **tmp = opal_argv_split(file_list, sep);
|
|
int i, count;
|
|
|
|
if (!tmp) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
count = opal_argv_count(tmp);
|
|
|
|
/* Iterate through all the files passed in -- read them in reverse
|
|
order so that we preserve unix/shell path-like semantics (i.e.,
|
|
the entries farthest to the left get precedence) */
|
|
|
|
for (i = count - 1; i >= 0; --i) {
|
|
char *file_name = append_filename_to_list (tmp[i]);
|
|
mca_base_parse_paramfile(file_name, file_values);
|
|
}
|
|
|
|
opal_argv_free (tmp);
|
|
|
|
mca_base_internal_env_store();
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int register_variable (const char *project_name, const char *framework_name,
|
|
const char *component_name, const char *variable_name,
|
|
const char *description, mca_base_var_type_t type,
|
|
mca_base_var_enum_t *enumerator, int bind,
|
|
mca_base_var_flag_t flags, mca_base_var_info_lvl_t info_lvl,
|
|
mca_base_var_scope_t scope, int synonym_for,
|
|
void *storage)
|
|
{
|
|
int ret, var_index, group_index, tmp;
|
|
mca_base_var_group_t *group;
|
|
mca_base_var_t *var, *original = NULL;
|
|
|
|
/* Developer error. Storage can not be NULL and type must exist */
|
|
assert (((flags & MCA_BASE_VAR_FLAG_SYNONYM) || NULL != storage) && type >= 0 && type < MCA_BASE_VAR_TYPE_MAX);
|
|
|
|
/* Developer error: check max length of strings */
|
|
if (NULL != project_name &&
|
|
strlen(project_name) > MCA_BASE_MAX_PROJECT_NAME_LEN) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
if (NULL != framework_name &&
|
|
strlen(framework_name) > MCA_BASE_MAX_TYPE_NAME_LEN) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
if (NULL != component_name &&
|
|
strlen(component_name) > MCA_BASE_MAX_COMPONENT_NAME_LEN) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
if (NULL != variable_name &&
|
|
strlen(variable_name) > MCA_BASE_MAX_VARIABLE_NAME_LEN) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
#if OPAL_ENABLE_DEBUG
|
|
/* Developer error: check for alignments */
|
|
uintptr_t align = 0;
|
|
switch (type) {
|
|
case MCA_BASE_VAR_TYPE_INT:
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_INT:
|
|
align = OPAL_ALIGNMENT_INT;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_INT32_T:
|
|
case MCA_BASE_VAR_TYPE_UINT32_T:
|
|
align = OPAL_ALIGNMENT_INT32;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_INT64_T:
|
|
case MCA_BASE_VAR_TYPE_UINT64_T:
|
|
align = OPAL_ALIGNMENT_INT64;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_LONG:
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG:
|
|
align = OPAL_ALIGNMENT_LONG;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG_LONG:
|
|
align = OPAL_ALIGNMENT_LONG_LONG;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_SIZE_T:
|
|
align = OPAL_ALIGNMENT_SIZE_T;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_BOOL:
|
|
align = OPAL_ALIGNMENT_BOOL;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_DOUBLE:
|
|
align = OPAL_ALIGNMENT_DOUBLE;
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_VERSION_STRING:
|
|
case MCA_BASE_VAR_TYPE_STRING:
|
|
default:
|
|
align = 0;
|
|
break;
|
|
}
|
|
|
|
if (0 != align) {
|
|
assert(((uintptr_t) storage) % align == 0);
|
|
}
|
|
|
|
/* Also check to ensure that synonym_for>=0 when
|
|
MCA_BCASE_VAR_FLAG_SYNONYM is specified */
|
|
if (flags & MCA_BASE_VAR_FLAG_SYNONYM && synonym_for < 0) {
|
|
assert((flags & MCA_BASE_VAR_FLAG_SYNONYM) && synonym_for >= 0);
|
|
}
|
|
#endif
|
|
|
|
if (flags & MCA_BASE_VAR_FLAG_SYNONYM) {
|
|
original = opal_pointer_array_get_item (&mca_base_vars, synonym_for);
|
|
if (NULL == original) {
|
|
/* Attempting to create a synonym for a non-existent variable. probably a
|
|
* developer error. */
|
|
assert (NULL != original);
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
}
|
|
|
|
/* There are data holes in the var struct */
|
|
OPAL_DEBUG_ZERO(var);
|
|
|
|
/* Initialize the array if it has never been initialized */
|
|
if (!mca_base_var_initialized) {
|
|
mca_base_var_init();
|
|
}
|
|
|
|
/* See if this entry is already in the array */
|
|
var_index = var_find (project_name, framework_name, component_name, variable_name,
|
|
true);
|
|
|
|
if (0 > var_index) {
|
|
/* Create a new parameter entry */
|
|
group_index = mca_base_var_group_register (project_name, framework_name, component_name,
|
|
NULL);
|
|
if (-1 > group_index) {
|
|
return group_index;
|
|
}
|
|
|
|
/* Read-only and constant variables can't be settable */
|
|
if (scope < MCA_BASE_VAR_SCOPE_LOCAL || (flags & MCA_BASE_VAR_FLAG_DEFAULT_ONLY)) {
|
|
if ((flags & MCA_BASE_VAR_FLAG_DEFAULT_ONLY) && (flags & MCA_BASE_VAR_FLAG_SETTABLE)) {
|
|
opal_show_help("help-mca-var.txt", "invalid-flag-combination",
|
|
true, "MCA_BASE_VAR_FLAG_DEFAULT_ONLY", "MCA_BASE_VAR_FLAG_SETTABLE");
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
/* Should we print a warning for other cases? */
|
|
flags &= ~MCA_BASE_VAR_FLAG_SETTABLE;
|
|
}
|
|
|
|
var = OBJ_NEW(mca_base_var_t);
|
|
|
|
var->mbv_type = type;
|
|
var->mbv_flags = flags;
|
|
var->mbv_group_index = group_index;
|
|
var->mbv_info_lvl = info_lvl;
|
|
var->mbv_scope = scope;
|
|
var->mbv_synonym_for = synonym_for;
|
|
var->mbv_bind = bind;
|
|
|
|
if (NULL != description) {
|
|
var->mbv_description = strdup(description);
|
|
}
|
|
|
|
ret = mca_base_var_generate_full_name4 (project_name, framework_name, component_name,
|
|
variable_name, &var->mbv_long_name);
|
|
if (OPAL_SUCCESS != ret) {
|
|
OBJ_RELEASE(var);
|
|
return OPAL_ERROR;
|
|
}
|
|
/* The mbv_full_name and the variable name are subset of the mbv_long_name
|
|
* so instead of allocating them we can just point into the var mbv_long_name
|
|
* at the right location.
|
|
*/
|
|
var->mbv_full_name = var->mbv_long_name +
|
|
(NULL == project_name ? 0 : (strlen(project_name)+1)); /* 1 for _ */
|
|
if( NULL != variable_name ) {
|
|
var->mbv_variable_name = var->mbv_full_name +
|
|
(NULL == framework_name ? 0 : (strlen(framework_name)+1)) +
|
|
(NULL == component_name ? 0 : (strlen(component_name)+1));
|
|
}
|
|
|
|
/* Add it to the array. Note that we copy the mca_var_t by value,
|
|
so the entire contents of the struct is copied. The synonym list
|
|
will always be empty at this point, so there's no need for an
|
|
extra RETAIN or RELEASE. */
|
|
var_index = opal_pointer_array_add (&mca_base_vars, var);
|
|
if (0 > var_index) {
|
|
OBJ_RELEASE(var);
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
var->mbv_index = var_index;
|
|
|
|
if (0 <= group_index) {
|
|
mca_base_var_group_add_var (group_index, var_index);
|
|
}
|
|
|
|
mca_base_var_count++;
|
|
if (0 <= var_find_by_name (var->mbv_full_name, &tmp, 0)) {
|
|
/* XXX --- FIXME: variable overshadows an existing variable. this is difficult to support */
|
|
assert (0);
|
|
}
|
|
|
|
opal_hash_table_set_value_ptr (&mca_base_var_index_hash, var->mbv_full_name, strlen (var->mbv_full_name),
|
|
(void *)(uintptr_t) var_index);
|
|
} else {
|
|
ret = var_get (var_index, &var, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
/* Shouldn't ever happen */
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
ret = mca_base_var_group_get_internal (var->mbv_group_index, &group, true);
|
|
if (OPAL_SUCCESS != ret) {
|
|
/* Shouldn't ever happen */
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
if (!group->group_isvalid) {
|
|
group->group_isvalid = true;
|
|
}
|
|
|
|
/* Verify the name components match */
|
|
if (0 != compare_strings(framework_name, group->group_framework) ||
|
|
0 != compare_strings(component_name, group->group_component) ||
|
|
0 != compare_strings(variable_name, var->mbv_variable_name)) {
|
|
opal_show_help("help-mca-var.txt", "var-name-conflict",
|
|
true, var->mbv_full_name, framework_name,
|
|
component_name, variable_name,
|
|
group->group_framework, group->group_component,
|
|
var->mbv_variable_name);
|
|
/* This is developer error. abort! */
|
|
assert (0);
|
|
return OPAL_ERROR;
|
|
}
|
|
|
|
if (var->mbv_type != type) {
|
|
#if OPAL_ENABLE_DEBUG
|
|
opal_show_help("help-mca-var.txt",
|
|
"re-register-with-different-type",
|
|
true, var->mbv_full_name);
|
|
#endif
|
|
return OPAL_ERR_VALUE_OUT_OF_BOUNDS;
|
|
}
|
|
}
|
|
|
|
if (MCA_BASE_VAR_TYPE_BOOL == var->mbv_type) {
|
|
enumerator = &mca_base_var_enum_bool;
|
|
} else if (NULL != enumerator) {
|
|
if (var->mbv_enumerator) {
|
|
OBJ_RELEASE (var->mbv_enumerator);
|
|
}
|
|
|
|
if (!enumerator->enum_is_static) {
|
|
OBJ_RETAIN(enumerator);
|
|
}
|
|
}
|
|
|
|
var->mbv_enumerator = enumerator;
|
|
|
|
if (!original) {
|
|
var->mbv_storage = storage;
|
|
|
|
/* make a copy of the default string value */
|
|
if ((MCA_BASE_VAR_TYPE_STRING == type || MCA_BASE_VAR_TYPE_VERSION_STRING == type) && NULL != ((char **)storage)[0]) {
|
|
((char **)storage)[0] = strdup (((char **)storage)[0]);
|
|
}
|
|
} else {
|
|
/* synonym variable */
|
|
opal_value_array_append_item(&original->mbv_synonyms, &var_index);
|
|
}
|
|
|
|
/* go ahead and mark this variable as valid */
|
|
var->mbv_flags |= MCA_BASE_VAR_FLAG_VALID;
|
|
|
|
ret = var_set_initial (var, original);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
/* All done */
|
|
return var_index;
|
|
}
|
|
|
|
int mca_base_var_register (const char *project_name, const char *framework_name,
|
|
const char *component_name, const char *variable_name,
|
|
const char *description, mca_base_var_type_t type,
|
|
mca_base_var_enum_t *enumerator, int bind,
|
|
mca_base_var_flag_t flags,
|
|
mca_base_var_info_lvl_t info_lvl,
|
|
mca_base_var_scope_t scope, void *storage)
|
|
{
|
|
/* Only integer variables can have enumerator */
|
|
assert (NULL == enumerator || (MCA_BASE_VAR_TYPE_INT == type || MCA_BASE_VAR_TYPE_UNSIGNED_INT == type));
|
|
|
|
return register_variable (project_name, framework_name, component_name,
|
|
variable_name, description, type, enumerator,
|
|
bind, flags, info_lvl, scope, -1, storage);
|
|
}
|
|
|
|
int mca_base_component_var_register (const mca_base_component_t *component,
|
|
const char *variable_name, const char *description,
|
|
mca_base_var_type_t type, mca_base_var_enum_t *enumerator,
|
|
int bind, mca_base_var_flag_t flags,
|
|
mca_base_var_info_lvl_t info_lvl,
|
|
mca_base_var_scope_t scope, void *storage)
|
|
{
|
|
return mca_base_var_register (component->mca_project_name, component->mca_type_name,
|
|
component->mca_component_name,
|
|
variable_name, description, type, enumerator,
|
|
bind, flags | MCA_BASE_VAR_FLAG_DWG,
|
|
info_lvl, scope, storage);
|
|
}
|
|
|
|
int mca_base_framework_var_register (const mca_base_framework_t *framework,
|
|
const char *variable_name,
|
|
const char *help_msg, mca_base_var_type_t type,
|
|
mca_base_var_enum_t *enumerator, int bind,
|
|
mca_base_var_flag_t flags,
|
|
mca_base_var_info_lvl_t info_level,
|
|
mca_base_var_scope_t scope, void *storage)
|
|
{
|
|
return mca_base_var_register (framework->framework_project, framework->framework_name,
|
|
"base", variable_name, help_msg, type, enumerator, bind,
|
|
flags | MCA_BASE_VAR_FLAG_DWG, info_level, scope, storage);
|
|
}
|
|
|
|
int mca_base_var_register_synonym (int synonym_for, const char *project_name,
|
|
const char *framework_name,
|
|
const char *component_name,
|
|
const char *synonym_name,
|
|
mca_base_var_syn_flag_t flags)
|
|
{
|
|
mca_base_var_flag_t var_flags = (mca_base_var_flag_t) MCA_BASE_VAR_FLAG_SYNONYM;
|
|
mca_base_var_t *var;
|
|
int ret;
|
|
|
|
ret = var_get (synonym_for, &var, false);
|
|
if (OPAL_SUCCESS != ret || VAR_IS_SYNONYM(var[0])) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
if (flags & MCA_BASE_VAR_SYN_FLAG_DEPRECATED) {
|
|
var_flags |= MCA_BASE_VAR_FLAG_DEPRECATED;
|
|
}
|
|
if (flags & MCA_BASE_VAR_SYN_FLAG_INTERNAL) {
|
|
var_flags |= MCA_BASE_VAR_FLAG_INTERNAL;
|
|
}
|
|
|
|
return register_variable (project_name, framework_name, component_name,
|
|
synonym_name, var->mbv_description, var->mbv_type, var->mbv_enumerator,
|
|
var->mbv_bind, var_flags, var->mbv_info_lvl, var->mbv_scope,
|
|
synonym_for, NULL);
|
|
}
|
|
|
|
static int var_get_env (mca_base_var_t *var, const char *name, char **source, char **value)
|
|
{
|
|
const char source_prefix[] = "SOURCE_";
|
|
const int max_len = strlen(mca_prefix) + strlen(source_prefix) +
|
|
strlen(name) + 1;
|
|
char *envvar = alloca(max_len);
|
|
if (NULL == envvar) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
int ret;
|
|
ret = snprintf(envvar, max_len, "%s%s", mca_prefix, name);
|
|
if (0 > ret) {
|
|
return OPAL_ERROR;
|
|
}
|
|
*value = getenv(envvar);
|
|
if( NULL == *value ) {
|
|
*source = NULL;
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
ret = snprintf(envvar, max_len, "%s%s%s", mca_prefix,
|
|
source_prefix, name);
|
|
if( 0 > ret ) {
|
|
return OPAL_ERROR;
|
|
}
|
|
*source = getenv(envvar);
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Lookup a param in the environment
|
|
*/
|
|
static int var_set_from_env (mca_base_var_t *var, mca_base_var_t *original)
|
|
{
|
|
const char *var_full_name = var->mbv_full_name;
|
|
const char *var_long_name = var->mbv_long_name;
|
|
bool deprecated = VAR_IS_DEPRECATED(var[0]);
|
|
bool is_synonym = VAR_IS_SYNONYM(var[0]);
|
|
char *source_env, *value_env;
|
|
int ret;
|
|
|
|
ret = var_get_env (var, var_long_name, &source_env, &value_env);
|
|
if (OPAL_SUCCESS != ret) {
|
|
ret = var_get_env (var, var_full_name, &source_env, &value_env);
|
|
}
|
|
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
/* we found an environment variable but this variable is default-only. print
|
|
a warning. */
|
|
if (VAR_IS_DEFAULT_ONLY(original[0])) {
|
|
opal_show_help("help-mca-var.txt", "default-only-param-set",
|
|
true, var_full_name);
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_SOURCE_OVERRIDE == original->mbv_source) {
|
|
if (!mca_base_var_suppress_override_warning) {
|
|
opal_show_help("help-mca-var.txt", "overridden-param-set",
|
|
true, var_full_name);
|
|
}
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
original->mbv_source = MCA_BASE_VAR_SOURCE_ENV;
|
|
|
|
if (NULL != source_env) {
|
|
if (0 == strncasecmp (source_env, "file:", 5)) {
|
|
original->mbv_source_file = append_filename_to_list(source_env + 5);
|
|
if (0 == strcmp (var->mbv_source_file, mca_base_var_override_file)) {
|
|
original->mbv_source = MCA_BASE_VAR_SOURCE_OVERRIDE;
|
|
} else {
|
|
original->mbv_source = MCA_BASE_VAR_SOURCE_FILE;
|
|
}
|
|
} else if (0 == strcasecmp (source_env, "command")) {
|
|
var->mbv_source = MCA_BASE_VAR_SOURCE_COMMAND_LINE;
|
|
}
|
|
}
|
|
|
|
if (deprecated) {
|
|
const char *new_variable = "None (going away)";
|
|
|
|
if (is_synonym) {
|
|
new_variable = original->mbv_full_name;
|
|
}
|
|
|
|
switch (var->mbv_source) {
|
|
case MCA_BASE_VAR_SOURCE_ENV:
|
|
opal_show_help("help-mca-var.txt", "deprecated-mca-env",
|
|
true, var_full_name, new_variable);
|
|
break;
|
|
case MCA_BASE_VAR_SOURCE_COMMAND_LINE:
|
|
opal_show_help("help-mca-var.txt", "deprecated-mca-cli",
|
|
true, var_full_name, new_variable);
|
|
break;
|
|
case MCA_BASE_VAR_SOURCE_FILE:
|
|
case MCA_BASE_VAR_SOURCE_OVERRIDE:
|
|
opal_show_help("help-mca-var.txt", "deprecated-mca-file",
|
|
true, var_full_name, mca_base_var_source_file (var),
|
|
new_variable);
|
|
break;
|
|
|
|
case MCA_BASE_VAR_SOURCE_DEFAULT:
|
|
case MCA_BASE_VAR_SOURCE_MAX:
|
|
case MCA_BASE_VAR_SOURCE_SET:
|
|
/* silence compiler warnings about unhandled enumerations */
|
|
break;
|
|
}
|
|
}
|
|
|
|
return var_set_from_string (original, value_env);
|
|
}
|
|
|
|
/*
|
|
* Lookup a param in the files
|
|
*/
|
|
static int var_set_from_file (mca_base_var_t *var, mca_base_var_t *original, opal_list_t *file_values)
|
|
{
|
|
const char *var_full_name = var->mbv_full_name;
|
|
const char *var_long_name = var->mbv_long_name;
|
|
bool deprecated = VAR_IS_DEPRECATED(var[0]);
|
|
bool is_synonym = VAR_IS_SYNONYM(var[0]);
|
|
mca_base_var_file_value_t *fv;
|
|
|
|
/* Scan through the list of values read in from files and try to
|
|
find a match. If we do, cache it on the param (for future
|
|
lookups) and save it in the storage. */
|
|
|
|
OPAL_LIST_FOREACH(fv, file_values, mca_base_var_file_value_t) {
|
|
if (0 != strcmp(fv->mbvfv_var, var_full_name) &&
|
|
0 != strcmp(fv->mbvfv_var, var_long_name)) {
|
|
continue;
|
|
}
|
|
|
|
/* found it */
|
|
if (VAR_IS_DEFAULT_ONLY(var[0])) {
|
|
opal_show_help("help-mca-var.txt", "default-only-param-set",
|
|
true, var_full_name);
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_FLAG_ENVIRONMENT_ONLY & original->mbv_flags) {
|
|
opal_show_help("help-mca-var.txt", "environment-only-param",
|
|
true, var_full_name, fv->mbvfv_value,
|
|
fv->mbvfv_file);
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_SOURCE_OVERRIDE == original->mbv_source) {
|
|
if (!mca_base_var_suppress_override_warning) {
|
|
opal_show_help("help-mca-var.txt", "overridden-param-set",
|
|
true, var_full_name);
|
|
}
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (deprecated) {
|
|
const char *new_variable = "None (going away)";
|
|
|
|
if (is_synonym) {
|
|
new_variable = original->mbv_full_name;
|
|
}
|
|
|
|
opal_show_help("help-mca-var.txt", "deprecated-mca-file",
|
|
true, var_full_name, fv->mbvfv_file,
|
|
new_variable);
|
|
}
|
|
|
|
original->mbv_file_value = (void *) fv;
|
|
original->mbv_source = MCA_BASE_VAR_SOURCE_FILE;
|
|
if (is_synonym) {
|
|
var->mbv_file_value = (void *) fv;
|
|
var->mbv_source = MCA_BASE_VAR_SOURCE_FILE;
|
|
}
|
|
|
|
return var_set_from_string (original, fv->mbvfv_value);
|
|
}
|
|
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
/*
|
|
* Lookup the initial value for a parameter
|
|
*/
|
|
static int var_set_initial (mca_base_var_t *var, mca_base_var_t *original)
|
|
{
|
|
int ret;
|
|
|
|
if (original) {
|
|
/* synonym */
|
|
var->mbv_source = original->mbv_source;
|
|
var->mbv_file_value = original->mbv_file_value;
|
|
var->mbv_source_file = original->mbv_source_file;
|
|
} else {
|
|
var->mbv_source = MCA_BASE_VAR_SOURCE_DEFAULT;
|
|
original = var;
|
|
}
|
|
|
|
/* Check all the places that the param may be hiding, in priority
|
|
order. If the default only flag is set the user will get a
|
|
warning if they try to set a value from the environment or a
|
|
file. */
|
|
ret = var_set_from_file (var, original, &mca_base_var_override_values);
|
|
if (OPAL_SUCCESS == ret) {
|
|
var->mbv_flags = ~MCA_BASE_VAR_FLAG_SETTABLE & (var->mbv_flags | MCA_BASE_VAR_FLAG_OVERRIDE);
|
|
var->mbv_source = MCA_BASE_VAR_SOURCE_OVERRIDE;
|
|
}
|
|
|
|
ret = var_set_from_env (var, original);
|
|
if (OPAL_ERR_NOT_FOUND != ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = var_set_from_file (var, original, &mca_base_envar_file_values);
|
|
if (OPAL_ERR_NOT_FOUND != ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = var_set_from_file (var, original, &mca_base_var_file_values);
|
|
if (OPAL_ERR_NOT_FOUND != ret) {
|
|
return ret;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* Create an empty param container
|
|
*/
|
|
static void var_constructor(mca_base_var_t *var)
|
|
{
|
|
memset ((char *) var + sizeof (var->super), 0, sizeof (*var) - sizeof (var->super));
|
|
|
|
var->mbv_type = MCA_BASE_VAR_TYPE_MAX;
|
|
OBJ_CONSTRUCT(&var->mbv_synonyms, opal_value_array_t);
|
|
opal_value_array_init (&var->mbv_synonyms, sizeof (int));
|
|
}
|
|
|
|
|
|
/*
|
|
* Free all the contents of a param container
|
|
*/
|
|
static void var_destructor(mca_base_var_t *var)
|
|
{
|
|
if ((MCA_BASE_VAR_TYPE_STRING == var->mbv_type ||
|
|
MCA_BASE_VAR_TYPE_VERSION_STRING == var->mbv_type) &&
|
|
NULL != var->mbv_storage &&
|
|
NULL != var->mbv_storage->stringval) {
|
|
free (var->mbv_storage->stringval);
|
|
var->mbv_storage->stringval = NULL;
|
|
}
|
|
|
|
/* don't release the boolean enumerator */
|
|
if (var->mbv_enumerator && !var->mbv_enumerator->enum_is_static) {
|
|
OBJ_RELEASE(var->mbv_enumerator);
|
|
}
|
|
|
|
if (NULL != var->mbv_long_name) {
|
|
free(var->mbv_long_name);
|
|
}
|
|
var->mbv_full_name = NULL;
|
|
var->mbv_variable_name = NULL;
|
|
|
|
if (NULL != var->mbv_description) {
|
|
free(var->mbv_description);
|
|
}
|
|
|
|
/* Destroy the synonym array */
|
|
OBJ_DESTRUCT(&var->mbv_synonyms);
|
|
|
|
/* mark this parameter as invalid */
|
|
var->mbv_type = MCA_BASE_VAR_TYPE_MAX;
|
|
|
|
#if OPAL_ENABLE_DEBUG
|
|
/* Cheap trick to reset everything to NULL */
|
|
memset ((char *) var + sizeof (var->super), 0, sizeof (*var) - sizeof (var->super));
|
|
#endif
|
|
}
|
|
|
|
|
|
static void fv_constructor(mca_base_var_file_value_t *f)
|
|
{
|
|
memset ((char *) f + sizeof (f->super), 0, sizeof (*f) - sizeof (f->super));
|
|
}
|
|
|
|
|
|
static void fv_destructor(mca_base_var_file_value_t *f)
|
|
{
|
|
if (NULL != f->mbvfv_var) {
|
|
free(f->mbvfv_var);
|
|
}
|
|
if (NULL != f->mbvfv_value) {
|
|
free(f->mbvfv_value);
|
|
}
|
|
/* the file name is stored in mca_*/
|
|
fv_constructor(f);
|
|
}
|
|
|
|
static char *source_name(mca_base_var_t *var)
|
|
{
|
|
char *ret;
|
|
|
|
if (MCA_BASE_VAR_SOURCE_FILE == var->mbv_source || MCA_BASE_VAR_SOURCE_OVERRIDE == var->mbv_source) {
|
|
struct mca_base_var_file_value_t *fv = var->mbv_file_value;
|
|
int rc;
|
|
|
|
if (fv) {
|
|
rc = asprintf(&ret, "file (%s:%d)", fv->mbvfv_file, fv->mbvfv_lineno);
|
|
} else {
|
|
rc = asprintf(&ret, "file (%s)", var->mbv_source_file);
|
|
}
|
|
|
|
/* some compilers will warn if the return code of asprintf is not checked (even if it is cast to void) */
|
|
if (0 > rc) {
|
|
return NULL;
|
|
}
|
|
return ret;
|
|
} else if (MCA_BASE_VAR_SOURCE_MAX <= var->mbv_source) {
|
|
return strdup ("unknown(!!)");
|
|
}
|
|
|
|
return strdup (var_source_names[var->mbv_source]);
|
|
}
|
|
|
|
static int var_value_string (mca_base_var_t *var, char **value_string)
|
|
{
|
|
const mca_base_var_storage_t *value=NULL;
|
|
int ret;
|
|
|
|
assert (MCA_BASE_VAR_TYPE_MAX > var->mbv_type);
|
|
|
|
/** Parameters with MCA_BASE_VAR_FLAG_DEF_UNSET flag should be shown
|
|
* as "unset" by default. */
|
|
if ((var->mbv_flags & MCA_BASE_VAR_FLAG_DEF_UNSET) &&
|
|
(MCA_BASE_VAR_SOURCE_DEFAULT == var->mbv_source)){
|
|
asprintf (value_string, "%s", "unset");
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
ret = mca_base_var_get_value(var->mbv_index, &value, NULL, NULL);
|
|
if (OPAL_SUCCESS != ret || NULL == value) {
|
|
return ret;
|
|
}
|
|
|
|
if (NULL == var->mbv_enumerator) {
|
|
switch (var->mbv_type) {
|
|
case MCA_BASE_VAR_TYPE_INT:
|
|
ret = asprintf (value_string, "%d", value->intval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_INT32_T:
|
|
ret = asprintf (value_string, "%" PRId32, value->int32tval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UINT32_T:
|
|
ret = asprintf (value_string, "%" PRIu32, value->uint32tval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_INT64_T:
|
|
ret = asprintf (value_string, "%" PRId64, value->int64tval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UINT64_T:
|
|
ret = asprintf (value_string, "%" PRIu64, value->uint64tval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_LONG:
|
|
ret = asprintf (value_string, "%ld", value->longval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_INT:
|
|
ret = asprintf (value_string, "%u", value->uintval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG:
|
|
ret = asprintf (value_string, "%lu", value->ulval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_UNSIGNED_LONG_LONG:
|
|
ret = asprintf (value_string, "%llu", value->ullval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_SIZE_T:
|
|
ret = asprintf (value_string, "%" PRIsize_t, value->sizetval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_STRING:
|
|
case MCA_BASE_VAR_TYPE_VERSION_STRING:
|
|
ret = asprintf (value_string, "%s",
|
|
value->stringval ? value->stringval : "");
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_BOOL:
|
|
ret = asprintf (value_string, "%d", value->boolval);
|
|
break;
|
|
case MCA_BASE_VAR_TYPE_DOUBLE:
|
|
ret = asprintf (value_string, "%lf", value->lfval);
|
|
break;
|
|
default:
|
|
ret = -1;
|
|
break;
|
|
}
|
|
|
|
ret = (0 > ret) ? OPAL_ERR_OUT_OF_RESOURCE : OPAL_SUCCESS;
|
|
} else {
|
|
/* we use an enumerator to handle string->bool and bool->string conversion */
|
|
if (MCA_BASE_VAR_TYPE_BOOL == var->mbv_type) {
|
|
ret = var->mbv_enumerator->string_from_value(var->mbv_enumerator, value->boolval, value_string);
|
|
} else {
|
|
ret = var->mbv_enumerator->string_from_value(var->mbv_enumerator, value->intval, value_string);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int mca_base_var_check_exclusive (const char *project,
|
|
const char *type_a,
|
|
const char *component_a,
|
|
const char *param_a,
|
|
const char *type_b,
|
|
const char *component_b,
|
|
const char *param_b)
|
|
{
|
|
mca_base_var_t *var_a = NULL, *var_b = NULL;
|
|
int var_ai, var_bi;
|
|
|
|
/* XXX -- Remove me once the project name is in the componennt */
|
|
project = NULL;
|
|
|
|
var_ai = mca_base_var_find (project, type_a, component_a, param_a);
|
|
var_bi = mca_base_var_find (project, type_b, component_b, param_b);
|
|
if (var_bi < 0 || var_ai < 0) {
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
(void) var_get (var_ai, &var_a, true);
|
|
(void) var_get (var_bi, &var_b, true);
|
|
if (NULL == var_a || NULL == var_b) {
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_SOURCE_DEFAULT != var_a->mbv_source &&
|
|
MCA_BASE_VAR_SOURCE_DEFAULT != var_b->mbv_source) {
|
|
char *str_a, *str_b;
|
|
|
|
/* Form cosmetic string names for A */
|
|
str_a = source_name(var_a);
|
|
|
|
/* Form cosmetic string names for B */
|
|
str_b = source_name(var_b);
|
|
|
|
/* Print it all out */
|
|
opal_show_help("help-mca-var.txt",
|
|
"mutually-exclusive-vars",
|
|
true, var_a->mbv_full_name,
|
|
str_a, var_b->mbv_full_name,
|
|
str_b);
|
|
|
|
/* Free the temp strings */
|
|
free(str_a);
|
|
free(str_b);
|
|
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
int mca_base_var_get_count (void)
|
|
{
|
|
return mca_base_var_count;
|
|
}
|
|
|
|
int mca_base_var_dump(int vari, char ***out, mca_base_var_dump_type_t output_type)
|
|
{
|
|
const char *framework, *component, *full_name;
|
|
int i, line_count, line = 0, enum_count = 0;
|
|
char *value_string, *source_string, *tmp;
|
|
int synonym_count, ret, *synonyms = NULL;
|
|
mca_base_var_t *var, *original=NULL;
|
|
mca_base_var_group_t *group;
|
|
|
|
ret = var_get(vari, &var, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = mca_base_var_group_get_internal(var->mbv_group_index, &group, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
if (VAR_IS_SYNONYM(var[0])) {
|
|
ret = var_get(var->mbv_synonym_for, &original, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
/* just for protection... */
|
|
if (NULL == original) {
|
|
return OPAL_ERR_NOT_FOUND;
|
|
}
|
|
}
|
|
|
|
framework = group->group_framework;
|
|
component = group->group_component ? group->group_component : "base";
|
|
full_name = var->mbv_full_name;
|
|
|
|
synonym_count = opal_value_array_get_size(&var->mbv_synonyms);
|
|
if (synonym_count) {
|
|
synonyms = OPAL_VALUE_ARRAY_GET_BASE(&var->mbv_synonyms, int);
|
|
}
|
|
|
|
ret = var_value_string (var, &value_string);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
|
|
source_string = source_name(var);
|
|
if (NULL == source_string) {
|
|
free (value_string);
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
if (MCA_BASE_VAR_DUMP_PARSABLE == output_type) {
|
|
if (NULL != var->mbv_enumerator) {
|
|
(void) var->mbv_enumerator->get_count(var->mbv_enumerator, &enum_count);
|
|
}
|
|
|
|
line_count = 8 + (var->mbv_description ? 1 : 0) + (VAR_IS_SYNONYM(var[0]) ? 1 : synonym_count) +
|
|
enum_count;
|
|
|
|
*out = (char **) calloc (line_count + 1, sizeof (char *));
|
|
if (NULL == *out) {
|
|
free (value_string);
|
|
free (source_string);
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
/* build the message*/
|
|
asprintf(&tmp, "mca:%s:%s:param:%s:", framework, component,
|
|
full_name);
|
|
|
|
/* Output the value */
|
|
char *colon = strchr(value_string, ':');
|
|
if (NULL != colon) {
|
|
asprintf(out[0] + line++, "%svalue:\"%s\"", tmp, value_string);
|
|
} else {
|
|
asprintf(out[0] + line++, "%svalue:%s", tmp, value_string);
|
|
}
|
|
|
|
/* Output the source */
|
|
asprintf(out[0] + line++, "%ssource:%s", tmp, source_string);
|
|
|
|
/* Output whether it's read only or writable */
|
|
asprintf(out[0] + line++, "%sstatus:%s", tmp, VAR_IS_DEFAULT_ONLY(var[0]) ? "read-only" : "writeable");
|
|
|
|
/* Output the info level of this parametere */
|
|
asprintf(out[0] + line++, "%slevel:%d", tmp, var->mbv_info_lvl + 1);
|
|
|
|
/* If it has a help message, output the help message */
|
|
if (var->mbv_description) {
|
|
asprintf(out[0] + line++, "%shelp:%s", tmp, var->mbv_description);
|
|
}
|
|
|
|
if (NULL != var->mbv_enumerator) {
|
|
for (i = 0 ; i < enum_count ; ++i) {
|
|
const char *enum_string = NULL;
|
|
int enum_value;
|
|
|
|
ret = var->mbv_enumerator->get_value(var->mbv_enumerator, i, &enum_value,
|
|
&enum_string);
|
|
if (OPAL_SUCCESS != ret) {
|
|
continue;
|
|
}
|
|
|
|
asprintf(out[0] + line++, "%senumerator:value:%d:%s", tmp, enum_value, enum_string);
|
|
}
|
|
}
|
|
|
|
/* Is this variable deprecated? */
|
|
asprintf(out[0] + line++, "%sdeprecated:%s", tmp, VAR_IS_DEPRECATED(var[0]) ? "yes" : "no");
|
|
|
|
asprintf(out[0] + line++, "%stype:%s", tmp, ompi_var_type_names[var->mbv_type]);
|
|
|
|
/* Does this parameter have any synonyms or is it a synonym? */
|
|
if (VAR_IS_SYNONYM(var[0])) {
|
|
asprintf(out[0] + line++, "%ssynonym_of:name:%s", tmp, original->mbv_full_name);
|
|
} else if (opal_value_array_get_size(&var->mbv_synonyms)) {
|
|
for (i = 0 ; i < synonym_count ; ++i) {
|
|
mca_base_var_t *synonym;
|
|
|
|
ret = var_get(synonyms[i], &synonym, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
continue;
|
|
}
|
|
|
|
asprintf(out[0] + line++, "%ssynonym:name:%s", tmp, synonym->mbv_full_name);
|
|
}
|
|
}
|
|
|
|
free (tmp);
|
|
} else if (MCA_BASE_VAR_DUMP_READABLE == output_type) {
|
|
/* There will be at most three lines in the pretty print case */
|
|
*out = (char **) calloc (4, sizeof (char *));
|
|
if (NULL == *out) {
|
|
free (value_string);
|
|
free (source_string);
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
asprintf (out[0], "%s \"%s\" (current value: \"%s\", data source: %s, level: %d %s, type: %s",
|
|
VAR_IS_DEFAULT_ONLY(var[0]) ? "informational" : "parameter",
|
|
full_name, value_string, source_string, var->mbv_info_lvl + 1,
|
|
info_lvl_strings[var->mbv_info_lvl], ompi_var_type_names[var->mbv_type]);
|
|
|
|
tmp = out[0][0];
|
|
if (VAR_IS_DEPRECATED(var[0])) {
|
|
asprintf (out[0], "%s, deprecated", tmp);
|
|
free (tmp);
|
|
tmp = out[0][0];
|
|
}
|
|
|
|
/* Does this parameter have any synonyms or is it a synonym? */
|
|
if (VAR_IS_SYNONYM(var[0])) {
|
|
asprintf(out[0], "%s, synonym of: %s)", tmp, original->mbv_full_name);
|
|
free (tmp);
|
|
} else if (synonym_count) {
|
|
asprintf(out[0], "%s, synonyms: ", tmp);
|
|
free (tmp);
|
|
|
|
for (i = 0 ; i < synonym_count ; ++i) {
|
|
mca_base_var_t *synonym;
|
|
|
|
ret = var_get(synonyms[i], &synonym, false);
|
|
if (OPAL_SUCCESS != ret) {
|
|
continue;
|
|
}
|
|
|
|
tmp = out[0][0];
|
|
if (synonym_count == i+1) {
|
|
asprintf(out[0], "%s%s)", tmp, synonym->mbv_full_name);
|
|
} else {
|
|
asprintf(out[0], "%s%s, ", tmp, synonym->mbv_full_name);
|
|
}
|
|
free(tmp);
|
|
}
|
|
} else {
|
|
asprintf(out[0], "%s)", tmp);
|
|
free(tmp);
|
|
}
|
|
|
|
line++;
|
|
|
|
if (var->mbv_description) {
|
|
asprintf(out[0] + line++, "%s", var->mbv_description);
|
|
}
|
|
|
|
if (NULL != var->mbv_enumerator) {
|
|
char *values;
|
|
|
|
ret = var->mbv_enumerator->dump(var->mbv_enumerator, &values);
|
|
if (OPAL_SUCCESS == ret) {
|
|
asprintf (out[0] + line++, "Valid values: %s", values);
|
|
free (values);
|
|
}
|
|
}
|
|
} else if (MCA_BASE_VAR_DUMP_SIMPLE == output_type) {
|
|
*out = (char **) calloc (2, sizeof (char *));
|
|
if (NULL == *out) {
|
|
free (value_string);
|
|
free (source_string);
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
asprintf(out[0], "%s=%s (%s)", var->mbv_full_name, value_string, source_string);
|
|
}
|
|
|
|
free (value_string);
|
|
free (source_string);
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|