97b3603036
* fixed some bugs where "unknown" tokens were allowed on the command line (which should really only be used for ortertun). * if an unknown token is encountered, print a short error to stderr and quit with a nonzero exit status * if we don't find the right number of parameters to an option, print a short error to stderr and quit with a nonzero exit status * when --help is given, print the help message to stdout (not stderr) and quit with a zero exit status * added --showme:help option to the wrapper compilers * updated docs in opal/util/cmd_line.h * other small/miscellaneous CLI parsing bugs in various tools I won't bore you with what we did before. :-) Here's some examples of what the new behavior looks like: {{{ % ompi_info --bogus ompi_info: Error: unknown option "--bogus" Type 'ompi_info --help' for usage. % ompi_info --param bogus ompi_info: Error: option "--param" did not have enough parameters (2) Type 'ompi_info --help' for usage. % }}} This commit was SVN r26072.
1327 строки
40 KiB
C
1327 строки
40 KiB
C
/*
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
* University Research and Technology
|
|
* Corporation. All rights reserved.
|
|
* Copyright (c) 2004-2006 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) 2012 Los Alamos National Security, LLC.
|
|
* All rights reserved.
|
|
* Copyright (c) 2012 Cisco Systems, Inc. All rights reserved.
|
|
* $COPYRIGHT$
|
|
*
|
|
* Additional copyrights may follow
|
|
*
|
|
* $HEADER$
|
|
*/
|
|
|
|
#include "opal_config.h"
|
|
|
|
#include <stdio.h>
|
|
#ifdef HAVE_STRING_H
|
|
#include <string.h>
|
|
#endif /* HAVE_STRING_H */
|
|
#include <ctype.h>
|
|
|
|
#include "opal/class/opal_object.h"
|
|
#include "opal/class/opal_list.h"
|
|
#include "opal/threads/mutex.h"
|
|
#include "opal/util/argv.h"
|
|
#include "opal/util/cmd_line.h"
|
|
#include "opal/util/output.h"
|
|
|
|
#include "opal/mca/base/mca_base_param.h"
|
|
#include "opal/constants.h"
|
|
|
|
|
|
/*
|
|
* Some usage message constants
|
|
*
|
|
* Max width for param listings before the description will be listed
|
|
* on the next line
|
|
*/
|
|
#define PARAM_WIDTH 25
|
|
/*
|
|
* Max length of any line in the usage message
|
|
*/
|
|
#define MAX_WIDTH 76
|
|
|
|
/*
|
|
* Description of a command line option
|
|
*/
|
|
struct cmd_line_option_t {
|
|
opal_list_item_t super;
|
|
|
|
char clo_short_name;
|
|
char *clo_single_dash_name;
|
|
char *clo_long_name;
|
|
|
|
int clo_num_params;
|
|
char *clo_description;
|
|
|
|
opal_cmd_line_type_t clo_type;
|
|
char *clo_mca_param_env_var;
|
|
void *clo_variable_dest;
|
|
bool clo_variable_set;
|
|
};
|
|
typedef struct cmd_line_option_t cmd_line_option_t;
|
|
static void option_constructor(cmd_line_option_t *cmd);
|
|
static void option_destructor(cmd_line_option_t *cmd);
|
|
|
|
OBJ_CLASS_INSTANCE(cmd_line_option_t,
|
|
opal_list_item_t,
|
|
option_constructor, option_destructor);
|
|
|
|
/*
|
|
* An option that was used in the argv that was parsed
|
|
*/
|
|
struct cmd_line_param_t {
|
|
opal_list_item_t super;
|
|
|
|
/* Note that clp_arg points to storage "owned" by someone else; it
|
|
has the original option string by referene, not by value.
|
|
Hence, it should not be free()'ed. */
|
|
|
|
char *clp_arg;
|
|
|
|
/* Pointer to the existing option. This is also by reference; it
|
|
should not be free()ed. */
|
|
|
|
cmd_line_option_t *clp_option;
|
|
|
|
/* This argv array is a list of all the parameters of this option.
|
|
It is owned by this parameter, and should be freed when this
|
|
param_t is freed. */
|
|
|
|
int clp_argc;
|
|
char **clp_argv;
|
|
};
|
|
typedef struct cmd_line_param_t cmd_line_param_t;
|
|
static void param_constructor(cmd_line_param_t *cmd);
|
|
static void param_destructor(cmd_line_param_t *cmd);
|
|
OBJ_CLASS_INSTANCE(cmd_line_param_t,
|
|
opal_list_item_t,
|
|
param_constructor, param_destructor);
|
|
|
|
/*
|
|
* Instantiate the opal_cmd_line_t class
|
|
*/
|
|
static void cmd_line_constructor(opal_cmd_line_t *cmd);
|
|
static void cmd_line_destructor(opal_cmd_line_t *cmd);
|
|
OBJ_CLASS_INSTANCE(opal_cmd_line_t,
|
|
opal_object_t,
|
|
cmd_line_constructor,
|
|
cmd_line_destructor);
|
|
|
|
/*
|
|
* Private variables
|
|
*/
|
|
static char special_empty_token[] = {
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, '\0'
|
|
};
|
|
|
|
/*
|
|
* Private functions
|
|
*/
|
|
static int make_opt(opal_cmd_line_t *cmd, opal_cmd_line_init_t *e);
|
|
static void free_parse_results(opal_cmd_line_t *cmd);
|
|
static int split_shorts(opal_cmd_line_t *cmd,
|
|
char *token, char **args,
|
|
int *output_argc, char ***output_argv,
|
|
int *num_args_used, bool ignore_unknown);
|
|
static cmd_line_option_t *find_option(opal_cmd_line_t *cmd,
|
|
const char *option_name) __opal_attribute_nonnull__(1) __opal_attribute_nonnull__(2);
|
|
static int set_dest(cmd_line_option_t *option, char *sval);
|
|
static void fill(const cmd_line_option_t *a, char result[3][BUFSIZ]);
|
|
static int qsort_callback(const void *a, const void *b);
|
|
|
|
|
|
/*
|
|
* Create an entire command line handle from a table
|
|
*/
|
|
int opal_cmd_line_create(opal_cmd_line_t *cmd,
|
|
opal_cmd_line_init_t *table)
|
|
{
|
|
int i, ret = OPAL_SUCCESS;
|
|
|
|
/* Check bozo case */
|
|
|
|
if (NULL == cmd) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
OBJ_CONSTRUCT(cmd, opal_cmd_line_t);
|
|
|
|
/* Ensure we got a table */
|
|
|
|
if (NULL == table) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/* Loop through the table */
|
|
|
|
for (i = 0; ; ++i) {
|
|
|
|
/* Is this the end? */
|
|
|
|
if ('\0' == table[i].ocl_cmd_short_name &&
|
|
NULL == table[i].ocl_cmd_single_dash_name &&
|
|
NULL == table[i].ocl_cmd_long_name) {
|
|
break;
|
|
}
|
|
|
|
/* Nope -- it's an entry. Process it. */
|
|
|
|
ret = make_opt(cmd, &table[i]);
|
|
if (OPAL_SUCCESS != ret) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Append a command line entry to the previously constructed command line
|
|
*/
|
|
int opal_cmd_line_make_opt_mca(opal_cmd_line_t *cmd,
|
|
opal_cmd_line_init_t entry)
|
|
{
|
|
/* Ensure we got an entry */
|
|
if ('\0' == entry.ocl_cmd_short_name &&
|
|
NULL == entry.ocl_cmd_single_dash_name &&
|
|
NULL == entry.ocl_cmd_long_name) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
return make_opt(cmd, &entry);
|
|
}
|
|
|
|
|
|
/*
|
|
* Create a command line option, --long-name and/or -s (short name).
|
|
*/
|
|
int opal_cmd_line_make_opt(opal_cmd_line_t *cmd, char short_name,
|
|
const char *long_name, int num_params,
|
|
const char *desc)
|
|
{
|
|
opal_cmd_line_init_t e;
|
|
|
|
e.ocl_mca_type_name = NULL;
|
|
e.ocl_mca_component_name = NULL;
|
|
e.ocl_mca_param_name = NULL;
|
|
|
|
e.ocl_cmd_short_name = short_name;
|
|
e.ocl_cmd_single_dash_name = NULL;
|
|
e.ocl_cmd_long_name = long_name;
|
|
|
|
e.ocl_num_params = num_params;
|
|
|
|
e.ocl_variable_dest = NULL;
|
|
e.ocl_variable_type = OPAL_CMD_LINE_TYPE_NULL;
|
|
|
|
e.ocl_description = desc;
|
|
|
|
return make_opt(cmd, &e);
|
|
}
|
|
|
|
|
|
/*
|
|
* Create a command line option, --long-name and/or -s (short name).
|
|
*/
|
|
int opal_cmd_line_make_opt3(opal_cmd_line_t *cmd, char short_name,
|
|
const char *sd_name, const char *long_name,
|
|
int num_params, const char *desc)
|
|
{
|
|
opal_cmd_line_init_t e;
|
|
|
|
e.ocl_mca_type_name = NULL;
|
|
e.ocl_mca_component_name = NULL;
|
|
e.ocl_mca_param_name = NULL;
|
|
|
|
e.ocl_cmd_short_name = short_name;
|
|
e.ocl_cmd_single_dash_name = sd_name;
|
|
e.ocl_cmd_long_name = long_name;
|
|
|
|
e.ocl_num_params = num_params;
|
|
|
|
e.ocl_variable_dest = NULL;
|
|
e.ocl_variable_type = OPAL_CMD_LINE_TYPE_NULL;
|
|
|
|
e.ocl_description = desc;
|
|
|
|
return make_opt(cmd, &e);
|
|
}
|
|
|
|
|
|
/*
|
|
* Parse a command line according to a pre-built OPAL command line
|
|
* handle.
|
|
*/
|
|
int opal_cmd_line_parse(opal_cmd_line_t *cmd, bool ignore_unknown,
|
|
int argc, char **argv)
|
|
{
|
|
int i, j, orig, ret;
|
|
cmd_line_option_t *option;
|
|
cmd_line_param_t *param;
|
|
bool is_unknown;
|
|
bool is_option;
|
|
bool has_unknowns;
|
|
char **shortsv;
|
|
int shortsc;
|
|
int num_args_used;
|
|
bool have_help_option = false;
|
|
bool printed_error = false;
|
|
|
|
/* Bozo check */
|
|
|
|
if (0 == argc || NULL == argv) {
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
|
|
/* Free any parsed results that are already on this handle */
|
|
|
|
free_parse_results(cmd);
|
|
|
|
/* Analyze each token */
|
|
|
|
cmd->lcl_argc = argc;
|
|
cmd->lcl_argv = opal_argv_copy(argv);
|
|
|
|
/* Check up front: do we have a --help option? */
|
|
|
|
option = find_option(cmd, "help");
|
|
if (NULL != option) {
|
|
have_help_option = true;
|
|
}
|
|
|
|
/* Now traverse the easy-to-parse sequence of tokens. Note that
|
|
incrementing i must happen elsehwere; it can't be the third
|
|
clause in the "if" statement. */
|
|
|
|
param = NULL;
|
|
option = NULL;
|
|
has_unknowns = false;
|
|
for (i = 1; i < cmd->lcl_argc; ) {
|
|
is_unknown = false;
|
|
is_option = false;
|
|
|
|
/* Are we done? i.e., did we find the special "--" token? If
|
|
so, copy everying beyond it into the tail (i.e., don't
|
|
bother copying the "--" into the tail). */
|
|
|
|
if (0 == strcmp(cmd->lcl_argv[i], "--")) {
|
|
++i;
|
|
while (i < cmd->lcl_argc) {
|
|
opal_argv_append(&cmd->lcl_tail_argc, &cmd->lcl_tail_argv,
|
|
cmd->lcl_argv[i]);
|
|
++i;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
/* If it's not an option, then we've found an unrecognized
|
|
token. */
|
|
|
|
else if ('-' != cmd->lcl_argv[i][0]) {
|
|
is_unknown = true;
|
|
}
|
|
|
|
/* Nope, this is supposedly an option. Is it a long name? */
|
|
|
|
else if (0 == strncmp(cmd->lcl_argv[i], "--", 2)) {
|
|
is_option = true;
|
|
option = find_option(cmd, cmd->lcl_argv[i] + 2);
|
|
}
|
|
|
|
/* It could be a short name. Is it? */
|
|
|
|
else {
|
|
option = find_option(cmd, cmd->lcl_argv[i] + 1);
|
|
|
|
/* If we didn't find it, try to split it into shorts. If
|
|
we find the short option, replace lcl_argv[i] and
|
|
insert the rest into lcl_argv starting after position
|
|
i. If we don't find the short option, don't do
|
|
anything to lcl_argv so that it can fall through to the
|
|
error condition, below. */
|
|
|
|
if (NULL == option) {
|
|
shortsv = NULL;
|
|
shortsc = 0;
|
|
ret = split_shorts(cmd, cmd->lcl_argv[i] + 1,
|
|
&(cmd->lcl_argv[i + 1]),
|
|
&shortsc, &shortsv,
|
|
&num_args_used, ignore_unknown);
|
|
if (OPAL_SUCCESS == ret) {
|
|
option = find_option(cmd, shortsv[0] + 1);
|
|
|
|
if (NULL != option) {
|
|
opal_argv_delete(&cmd->lcl_argc,
|
|
&cmd->lcl_argv, i,
|
|
1 + num_args_used);
|
|
opal_argv_insert(&cmd->lcl_argv, i, shortsv);
|
|
cmd->lcl_argc = opal_argv_count(cmd->lcl_argv);
|
|
} else {
|
|
is_unknown = true;
|
|
}
|
|
opal_argv_free(shortsv);
|
|
} else {
|
|
is_unknown = true;
|
|
}
|
|
}
|
|
|
|
if (NULL != option) {
|
|
is_option = true;
|
|
}
|
|
}
|
|
|
|
/* If we figured out above that this is an option, handle it */
|
|
|
|
if (is_option) {
|
|
if (NULL == option) {
|
|
is_unknown = true;
|
|
} else {
|
|
is_unknown = false;
|
|
orig = i;
|
|
++i;
|
|
|
|
/* Suck down the following parameters that belong to
|
|
this option. If we run out of parameters, or find
|
|
that any of them are the special_empty_param
|
|
(insertted by split_shorts()), then print an error
|
|
and return. */
|
|
|
|
param = OBJ_NEW(cmd_line_param_t);
|
|
if (NULL == param) {
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
param->clp_arg = cmd->lcl_argv[i];
|
|
param->clp_option = option;
|
|
|
|
/* If we have any parameters to this option, suck down
|
|
tokens starting one beyond the token that we just
|
|
recognized */
|
|
|
|
for (j = 0; j < option->clo_num_params; ++j, ++i) {
|
|
|
|
/* If we run out of parameters, error */
|
|
|
|
if (i >= cmd->lcl_argc) {
|
|
fprintf(stderr, "%s: Error: option \"%s\" did not "
|
|
"have enough parameters (%d)\n",
|
|
cmd->lcl_argv[0],
|
|
cmd->lcl_argv[orig],
|
|
option->clo_num_params);
|
|
if (have_help_option) {
|
|
fprintf(stderr, "Type '%s --help' for usage.\n",
|
|
cmd->lcl_argv[0]);
|
|
}
|
|
OBJ_RELEASE(param);
|
|
printed_error = true;
|
|
goto error;
|
|
} else {
|
|
if (0 == strcmp(cmd->lcl_argv[i],
|
|
special_empty_token)) {
|
|
fprintf(stderr, "%s: Error: option \"%s\" did not "
|
|
"have enough parameters (%d)\n",
|
|
cmd->lcl_argv[0],
|
|
cmd->lcl_argv[orig],
|
|
option->clo_num_params);
|
|
if (have_help_option) {
|
|
fprintf(stderr, "Type '%s --help' for usage.\n",
|
|
cmd->lcl_argv[0]);
|
|
}
|
|
if (NULL != param->clp_argv) {
|
|
opal_argv_free(param->clp_argv);
|
|
}
|
|
OBJ_RELEASE(param);
|
|
printed_error = true;
|
|
goto error;
|
|
}
|
|
|
|
/* Otherwise, save this parameter */
|
|
|
|
else {
|
|
/* Save in the argv on the param entry */
|
|
|
|
opal_argv_append(¶m->clp_argc,
|
|
¶m->clp_argv,
|
|
cmd->lcl_argv[i]);
|
|
|
|
/* If it's the first, save it in the
|
|
variable dest and/or MCA parameter */
|
|
|
|
if (0 == j &&
|
|
(NULL != option->clo_mca_param_env_var ||
|
|
NULL != option->clo_variable_dest)) {
|
|
if (OPAL_SUCCESS != (ret = set_dest(option, cmd->lcl_argv[i]))) {
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
return ret;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If there are no options to this command, see if we
|
|
need to set a boolean value to "true". */
|
|
|
|
if (0 == option->clo_num_params) {
|
|
if (OPAL_SUCCESS != (ret = set_dest(option, "1"))) {
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
/* If we succeeded in all that, save the param to the
|
|
list on the opal_cmd_line_t handle */
|
|
|
|
if (NULL != param) {
|
|
opal_list_append(&cmd->lcl_params, ¶m->super);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If we figured out above that this was an unknown option,
|
|
handle it. Copy everything (including the current token)
|
|
into the tail. If we're not ignoring unknowns, then print
|
|
an error and return. */
|
|
if (is_unknown) {
|
|
has_unknowns = true;
|
|
if (!ignore_unknown) {
|
|
fprintf(stderr, "%s: Error: unknown option \"%s\"\n",
|
|
cmd->lcl_argv[0], cmd->lcl_argv[i]);
|
|
printed_error = true;
|
|
if (have_help_option) {
|
|
fprintf(stderr, "Type '%s --help' for usage.\n",
|
|
cmd->lcl_argv[0]);
|
|
}
|
|
}
|
|
error:
|
|
while (i < cmd->lcl_argc) {
|
|
opal_argv_append(&cmd->lcl_tail_argc, &cmd->lcl_tail_argv,
|
|
cmd->lcl_argv[i]);
|
|
++i;
|
|
}
|
|
|
|
/* Because i has advanced, we'll fall out of the loop */
|
|
}
|
|
}
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
|
|
/* All done */
|
|
if (printed_error) {
|
|
return OPAL_ERR_SILENT;
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return a consolidated "usage" message for a OPAL command line handle.
|
|
*/
|
|
char *opal_cmd_line_get_usage_msg(opal_cmd_line_t *cmd)
|
|
{
|
|
size_t i, len;
|
|
int argc;
|
|
size_t j;
|
|
char **argv;
|
|
char *ret, temp[MAX_WIDTH * 2], line[MAX_WIDTH * 2];
|
|
char *start, *desc, *ptr;
|
|
opal_list_item_t *item;
|
|
cmd_line_option_t *option, **sorted;
|
|
bool filled;
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
|
|
/* Make an argv of all the usage strings */
|
|
|
|
argc = 0;
|
|
argv = NULL;
|
|
ret = NULL;
|
|
|
|
/* First, take the original list and sort it */
|
|
|
|
sorted = (cmd_line_option_t**)malloc(sizeof(cmd_line_option_t *) *
|
|
opal_list_get_size(&cmd->lcl_options));
|
|
if (NULL == sorted) {
|
|
return NULL;
|
|
}
|
|
for (i = 0, item = opal_list_get_first(&cmd->lcl_options);
|
|
opal_list_get_end(&cmd->lcl_options) != item;
|
|
++i, item = opal_list_get_next(item)) {
|
|
sorted[i] = (cmd_line_option_t *) item;
|
|
}
|
|
qsort(sorted, i, sizeof(cmd_line_option_t*), qsort_callback);
|
|
|
|
/* Now go through the sorted array and make the strings */
|
|
|
|
for (j = 0; j < opal_list_get_size(&cmd->lcl_options); ++j) {
|
|
option = sorted[j];
|
|
if (NULL != option->clo_description) {
|
|
|
|
/* Build up the output line */
|
|
|
|
filled = false;
|
|
line[0] = '\0';
|
|
if ('\0' != option->clo_short_name) {
|
|
line[0] = '-';
|
|
line[1] = option->clo_short_name;
|
|
filled = true;
|
|
} else {
|
|
line[0] = ' ';
|
|
line[1] = ' ';
|
|
}
|
|
line[2] = '\0';
|
|
line[3] = '\0';
|
|
if (NULL != option->clo_single_dash_name) {
|
|
line[2] = (filled) ? '|' : ' ';
|
|
strcat(line, "-");
|
|
strcat(line, option->clo_single_dash_name);
|
|
filled = true;
|
|
}
|
|
if (NULL != option->clo_long_name) {
|
|
if (filled) {
|
|
strcat(line, "|");
|
|
} else {
|
|
strcat(line, " ");
|
|
}
|
|
strcat(line, "--");
|
|
strcat(line, option->clo_long_name);
|
|
filled = true;
|
|
}
|
|
strcat(line, " ");
|
|
for (i = 0; (int)i < option->clo_num_params; ++i) {
|
|
len = sizeof(temp);
|
|
snprintf(temp, len, "<arg%d> ", (int)i);
|
|
strcat(line, temp);
|
|
}
|
|
if (option->clo_num_params > 0) {
|
|
strcat(line, " ");
|
|
}
|
|
|
|
/* If we're less than param width, then start adding the
|
|
description to this line. Otherwise, finish this line
|
|
and start adding the description on the next line. */
|
|
|
|
if (strlen(line) > PARAM_WIDTH) {
|
|
opal_argv_append(&argc, &argv, line);
|
|
|
|
/* Now reset the line to be all blanks up to
|
|
PARAM_WIDTH so that we can start adding the
|
|
description */
|
|
|
|
memset(line, ' ', PARAM_WIDTH);
|
|
line[PARAM_WIDTH] = '\0';
|
|
} else {
|
|
|
|
/* Add enough blanks to the end of the line so that we
|
|
can start adding the description */
|
|
|
|
for (i = strlen(line); i < PARAM_WIDTH; ++i) {
|
|
line[i] = ' ';
|
|
}
|
|
line[i] = '\0';
|
|
}
|
|
|
|
/* Loop over adding the description to the array, breaking
|
|
the string at most at MAX_WIDTH characters. We need a
|
|
modifyable description (for simplicity), so strdup the
|
|
clo_description (because it's likely a compiler
|
|
constant, and may barf if we write temporary \0's in
|
|
the middle). */
|
|
|
|
desc = strdup(option->clo_description);
|
|
if (NULL == desc) {
|
|
free(sorted);
|
|
return strdup("");
|
|
}
|
|
start = desc;
|
|
len = strlen(desc);
|
|
do {
|
|
|
|
/* Trim off leading whitespace */
|
|
|
|
while (isspace(*start) && start < desc + len) {
|
|
++start;
|
|
}
|
|
if (start >= desc + len) {
|
|
break;
|
|
}
|
|
|
|
/* Last line */
|
|
|
|
if (strlen(start) < (MAX_WIDTH - PARAM_WIDTH)) {
|
|
strcat(line, start);
|
|
opal_argv_append(&argc, &argv, line);
|
|
break;
|
|
}
|
|
|
|
/* We have more than 1 line's worth left -- find this
|
|
line's worth and add it to the array. Then reset
|
|
and loop around to get the next line's worth. */
|
|
|
|
for (ptr = start + (MAX_WIDTH - PARAM_WIDTH);
|
|
ptr > start; --ptr) {
|
|
if (isspace(*ptr)) {
|
|
*ptr = '\0';
|
|
strcat(line, start);
|
|
opal_argv_append(&argc, &argv, line);
|
|
|
|
start = ptr + 1;
|
|
memset(line, ' ', PARAM_WIDTH);
|
|
line[PARAM_WIDTH] = '\0';
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* If we got all the way back to the beginning of the
|
|
string, then go forward looking for a whitespace
|
|
and break there. */
|
|
|
|
if (ptr == start) {
|
|
for (ptr = start + (MAX_WIDTH - PARAM_WIDTH);
|
|
ptr < start + len; ++ptr) {
|
|
if (isspace(*ptr)) {
|
|
*ptr = '\0';
|
|
|
|
strcat(line, start);
|
|
opal_argv_append(&argc, &argv, line);
|
|
|
|
start = ptr + 1;
|
|
memset(line, ' ', PARAM_WIDTH);
|
|
line[PARAM_WIDTH] = '\0';
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* If we reached the end of the string with no
|
|
whitespace, then just add it on and be done */
|
|
|
|
if (ptr >= start + len) {
|
|
strcat(line, start);
|
|
opal_argv_append(&argc, &argv, line);
|
|
start = desc + len + 1;
|
|
}
|
|
}
|
|
} while (start < desc + len);
|
|
free(desc);
|
|
}
|
|
}
|
|
if (NULL != argv) {
|
|
ret = opal_argv_join(argv, '\n');
|
|
opal_argv_free(argv);
|
|
} else {
|
|
ret = strdup("");
|
|
}
|
|
free(sorted);
|
|
|
|
/* Thread serialization */
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
|
|
/* All done */
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test if a given option was taken on the parsed command line.
|
|
*/
|
|
bool opal_cmd_line_is_taken(opal_cmd_line_t *cmd, const char *opt)
|
|
{
|
|
return (opal_cmd_line_get_ninsts(cmd, opt) > 0);
|
|
}
|
|
|
|
|
|
/*
|
|
* Return the number of instances of an option found during parsing.
|
|
*/
|
|
int opal_cmd_line_get_ninsts(opal_cmd_line_t *cmd, const char *opt)
|
|
{
|
|
int ret;
|
|
opal_list_item_t *item;
|
|
cmd_line_param_t *param;
|
|
cmd_line_option_t *option;
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
|
|
/* Find the corresponding option. If we find it, look through all
|
|
the parsed params and see if we have any matches. */
|
|
|
|
ret = 0;
|
|
option = find_option(cmd, opt);
|
|
if (NULL != option) {
|
|
for (item = opal_list_get_first(&cmd->lcl_params);
|
|
opal_list_get_end(&cmd->lcl_params) != item;
|
|
item = opal_list_get_next(item)) {
|
|
param = (cmd_line_param_t *) item;
|
|
if (param->clp_option == option) {
|
|
++ret;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
|
|
/* All done */
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return a specific parameter for a specific instance of a option
|
|
* from the parsed command line.
|
|
*/
|
|
char *opal_cmd_line_get_param(opal_cmd_line_t *cmd, const char *opt, int inst,
|
|
int idx)
|
|
{
|
|
int num_found;
|
|
opal_list_item_t *item;
|
|
cmd_line_param_t *param;
|
|
cmd_line_option_t *option;
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
|
|
/* Find the corresponding option. If we find it, look through all
|
|
the parsed params and see if we have any matches. */
|
|
|
|
num_found = 0;
|
|
option = find_option(cmd, opt);
|
|
if (NULL != option) {
|
|
|
|
/* Ensure to check for the case where the user has asked for a
|
|
parameter index greater than we will have */
|
|
|
|
if (idx < option->clo_num_params) {
|
|
for (item = opal_list_get_first(&cmd->lcl_params);
|
|
opal_list_get_end(&cmd->lcl_params) != item;
|
|
item = opal_list_get_next(item)) {
|
|
param = (cmd_line_param_t *) item;
|
|
if (param->clp_option == option) {
|
|
if (num_found == inst) {
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
return param->clp_argv[idx];
|
|
}
|
|
++num_found;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Thread serialization */
|
|
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
|
|
/* All done */
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return the number of arguments parsed on a OPAL command line handle.
|
|
*/
|
|
int opal_cmd_line_get_argc(opal_cmd_line_t *cmd)
|
|
{
|
|
return (NULL != cmd) ? cmd->lcl_argc : OPAL_ERROR;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return a string argument parsed on a OPAL command line handle.
|
|
*/
|
|
char *opal_cmd_line_get_argv(opal_cmd_line_t *cmd, int index)
|
|
{
|
|
return (NULL == cmd) ? NULL :
|
|
(index >= cmd->lcl_argc || index < 0) ? NULL : cmd->lcl_argv[index];
|
|
}
|
|
|
|
|
|
/*
|
|
* Return the entire "tail" of unprocessed argv from a OPAL command
|
|
* line handle.
|
|
*/
|
|
int opal_cmd_line_get_tail(opal_cmd_line_t *cmd, int *tailc, char ***tailv)
|
|
{
|
|
if (NULL != cmd) {
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
*tailc = cmd->lcl_tail_argc;
|
|
*tailv = opal_argv_copy(cmd->lcl_tail_argv);
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
return OPAL_SUCCESS;
|
|
} else {
|
|
return OPAL_ERROR;
|
|
}
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Static functions
|
|
**************************************************************************/
|
|
|
|
static void option_constructor(cmd_line_option_t *o)
|
|
{
|
|
o->clo_short_name = '\0';
|
|
o->clo_single_dash_name = NULL;
|
|
o->clo_long_name = NULL;
|
|
o->clo_num_params = 0;
|
|
o->clo_description = NULL;
|
|
|
|
o->clo_type = OPAL_CMD_LINE_TYPE_NULL;
|
|
o->clo_mca_param_env_var = NULL;
|
|
o->clo_variable_dest = NULL;
|
|
o->clo_variable_set = false;
|
|
}
|
|
|
|
|
|
static void option_destructor(cmd_line_option_t *o)
|
|
{
|
|
if (NULL != o->clo_single_dash_name) {
|
|
free(o->clo_single_dash_name);
|
|
}
|
|
if (NULL != o->clo_long_name) {
|
|
free(o->clo_long_name);
|
|
}
|
|
if (NULL != o->clo_description) {
|
|
free(o->clo_description);
|
|
}
|
|
if (NULL != o->clo_mca_param_env_var) {
|
|
free(o->clo_mca_param_env_var);
|
|
}
|
|
}
|
|
|
|
|
|
static void param_constructor(cmd_line_param_t *p)
|
|
{
|
|
p->clp_arg = NULL;
|
|
p->clp_option = NULL;
|
|
p->clp_argc = 0;
|
|
p->clp_argv = NULL;
|
|
}
|
|
|
|
|
|
static void param_destructor(cmd_line_param_t *p)
|
|
{
|
|
if (NULL != p->clp_argv) {
|
|
opal_argv_free(p->clp_argv);
|
|
}
|
|
}
|
|
|
|
|
|
static void cmd_line_constructor(opal_cmd_line_t *cmd)
|
|
{
|
|
/* Initialize the mutex. Since we're creating (and therefore the
|
|
only thread that has this instance), there's no need to lock it
|
|
right now. */
|
|
|
|
OBJ_CONSTRUCT(&cmd->lcl_mutex, opal_mutex_t);
|
|
|
|
/* Initialize the lists */
|
|
|
|
OBJ_CONSTRUCT(&cmd->lcl_options, opal_list_t);
|
|
OBJ_CONSTRUCT(&cmd->lcl_params, opal_list_t);
|
|
|
|
/* Initialize the argc/argv pairs */
|
|
|
|
cmd->lcl_argc = 0;
|
|
cmd->lcl_argv = NULL;
|
|
cmd->lcl_tail_argc = 0;
|
|
cmd->lcl_tail_argv = NULL;
|
|
}
|
|
|
|
|
|
static void cmd_line_destructor(opal_cmd_line_t *cmd)
|
|
{
|
|
opal_list_item_t *item;
|
|
|
|
/* Free the contents of the options list (do not free the list
|
|
itself; it was not allocated from the heap) */
|
|
|
|
for (item = opal_list_remove_first(&cmd->lcl_options);
|
|
NULL != item;
|
|
item = opal_list_remove_first(&cmd->lcl_options)) {
|
|
OBJ_RELEASE(item);
|
|
}
|
|
|
|
/* Free any parsed results */
|
|
|
|
free_parse_results(cmd);
|
|
|
|
/* Destroy the lists */
|
|
|
|
OBJ_DESTRUCT(&cmd->lcl_options);
|
|
OBJ_DESTRUCT(&cmd->lcl_params);
|
|
|
|
/* Destroy the mutex */
|
|
|
|
OBJ_DESTRUCT(&cmd->lcl_mutex);
|
|
}
|
|
|
|
|
|
static int make_opt(opal_cmd_line_t *cmd, opal_cmd_line_init_t *e)
|
|
{
|
|
cmd_line_option_t *option;
|
|
|
|
/* Bozo checks */
|
|
|
|
if (NULL == cmd) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
} else if ('\0' == e->ocl_cmd_short_name &&
|
|
NULL == e->ocl_cmd_single_dash_name &&
|
|
NULL == e->ocl_cmd_long_name) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
} else if (e->ocl_num_params < 0) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
|
|
/* Allocate and fill an option item */
|
|
|
|
option = OBJ_NEW(cmd_line_option_t);
|
|
if (NULL == option) {
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
}
|
|
|
|
option->clo_short_name = e->ocl_cmd_short_name;
|
|
if (NULL != e->ocl_cmd_single_dash_name) {
|
|
option->clo_single_dash_name = strdup(e->ocl_cmd_single_dash_name);
|
|
}
|
|
if (NULL != e->ocl_cmd_long_name) {
|
|
option->clo_long_name = strdup(e->ocl_cmd_long_name);
|
|
}
|
|
option->clo_num_params = e->ocl_num_params;
|
|
if (NULL != e->ocl_description) {
|
|
option->clo_description = strdup(e->ocl_description);
|
|
}
|
|
|
|
option->clo_type = e->ocl_variable_type;
|
|
option->clo_variable_dest = e->ocl_variable_dest;
|
|
if (NULL != e->ocl_mca_type_name) {
|
|
option->clo_mca_param_env_var =
|
|
mca_base_param_environ_variable(e->ocl_mca_type_name,
|
|
e->ocl_mca_component_name,
|
|
e->ocl_mca_param_name);
|
|
}
|
|
|
|
/* Append the item, serializing thread access */
|
|
|
|
opal_mutex_lock(&cmd->lcl_mutex);
|
|
opal_list_append(&cmd->lcl_options, &option->super);
|
|
opal_mutex_unlock(&cmd->lcl_mutex);
|
|
|
|
/* All done */
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
|
|
static void free_parse_results(opal_cmd_line_t *cmd)
|
|
{
|
|
opal_list_item_t *item;
|
|
|
|
/* Free the contents of the params list (do not free the list
|
|
itself; it was not allocated from the heap) */
|
|
|
|
for (item = opal_list_remove_first(&cmd->lcl_params);
|
|
NULL != item;
|
|
item = opal_list_remove_first(&cmd->lcl_params)) {
|
|
OBJ_RELEASE(item);
|
|
}
|
|
|
|
/* Free the argv's */
|
|
|
|
if (NULL != cmd->lcl_argv) {
|
|
opal_argv_free(cmd->lcl_argv);
|
|
}
|
|
cmd->lcl_argv = NULL;
|
|
cmd->lcl_argc = 0;
|
|
|
|
if (NULL != cmd->lcl_tail_argv) {
|
|
opal_argv_free(cmd->lcl_tail_argv);
|
|
}
|
|
cmd->lcl_tail_argv = NULL;
|
|
cmd->lcl_tail_argc = 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Traverse a token and split it into individual letter options (the
|
|
* token has already been certified to not be a long name and not be a
|
|
* short name). Ensure to differentiate the resulting options from
|
|
* "single dash" names.
|
|
*/
|
|
static int split_shorts(opal_cmd_line_t *cmd, char *token, char **args,
|
|
int *output_argc, char ***output_argv,
|
|
int *num_args_used, bool ignore_unknown)
|
|
{
|
|
int i, j, len;
|
|
cmd_line_option_t *option;
|
|
char fake_token[3];
|
|
int num_args;
|
|
|
|
/* Setup that we didn't use any of the args */
|
|
|
|
num_args = opal_argv_count(args);
|
|
*num_args_used = 0;
|
|
|
|
/* Traverse the token. If it's empty (e.g., if someone passes a
|
|
"-" token, which, since the upper level calls this function as
|
|
(argv[i] + 1), will be empty by the time it gets down here),
|
|
just return that we didn't find a short option. */
|
|
|
|
len = (int)strlen(token);
|
|
if (0 == len) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
}
|
|
fake_token[0] = '-';
|
|
fake_token[2] = '\0';
|
|
for (i = 0; i < len; ++i) {
|
|
fake_token[1] = token[i];
|
|
option = find_option(cmd, fake_token + 1);
|
|
|
|
/* If we don't find the option, either return an error or pass
|
|
it through unmodified to the new argv */
|
|
|
|
if (NULL == option) {
|
|
if (!ignore_unknown) {
|
|
return OPAL_ERR_BAD_PARAM;
|
|
} else {
|
|
opal_argv_append(output_argc, output_argv, fake_token);
|
|
}
|
|
}
|
|
|
|
/* If we do find the option, copy it and all of its parameters
|
|
to the output args. If we run out of paramters (i.e., no
|
|
more tokens in the original argv), that error will be
|
|
handled at a higher level) */
|
|
|
|
else {
|
|
opal_argv_append(output_argc, output_argv, fake_token);
|
|
for (j = 0; j < option->clo_num_params; ++j) {
|
|
if (*num_args_used < num_args) {
|
|
opal_argv_append(output_argc, output_argv,
|
|
args[*num_args_used]);
|
|
++(*num_args_used);
|
|
} else {
|
|
opal_argv_append(output_argc, output_argv,
|
|
special_empty_token);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* All done */
|
|
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
|
|
static cmd_line_option_t *find_option(opal_cmd_line_t *cmd,
|
|
const char *option_name)
|
|
{
|
|
opal_list_item_t *item;
|
|
cmd_line_option_t *option;
|
|
|
|
/* Iterate through the list of options hanging off the
|
|
opal_cmd_line_t and see if we find a match in either the short
|
|
or long names */
|
|
|
|
for (item = opal_list_get_first(&cmd->lcl_options);
|
|
opal_list_get_end(&cmd->lcl_options) != item;
|
|
item = opal_list_get_next(item)) {
|
|
option = (cmd_line_option_t *) item;
|
|
if ((NULL != option->clo_long_name &&
|
|
0 == strcmp(option_name, option->clo_long_name)) ||
|
|
(NULL != option->clo_single_dash_name &&
|
|
0 == strcmp(option_name, option->clo_single_dash_name)) ||
|
|
(strlen(option_name) == 1 &&
|
|
option_name[0] == option->clo_short_name)) {
|
|
return option;
|
|
}
|
|
}
|
|
|
|
/* Not found */
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static int set_dest(cmd_line_option_t *option, char *sval)
|
|
{
|
|
int ival = atol(sval);
|
|
long lval = strtoul(sval, NULL, 10);
|
|
char *str = NULL;
|
|
size_t i;
|
|
|
|
/* Set MCA param. We do this in the environment because the MCA
|
|
parameter may not have been registered yet -- and if it isn't
|
|
registered, we don't really want to register a dummy one
|
|
because we don't know what it's type and default value should
|
|
be. These are solvable problems (e.g., make a re-registration
|
|
overwrite everything), but it's far simpler to just leave the
|
|
registered table alone and set an environment variable with the
|
|
desired value. The environment variable will get picked up
|
|
during a nromal parameter lookup, and all will be well. */
|
|
|
|
if (NULL != option->clo_mca_param_env_var) {
|
|
switch(option->clo_type) {
|
|
case OPAL_CMD_LINE_TYPE_STRING:
|
|
case OPAL_CMD_LINE_TYPE_INT:
|
|
case OPAL_CMD_LINE_TYPE_SIZE_T:
|
|
asprintf(&str, "%s=%s", option->clo_mca_param_env_var, sval);
|
|
break;
|
|
case OPAL_CMD_LINE_TYPE_BOOL:
|
|
asprintf(&str, "%s=1", option->clo_mca_param_env_var);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (NULL != str) {
|
|
putenv(str);
|
|
}
|
|
}
|
|
|
|
/* Set variable */
|
|
|
|
if (NULL != option->clo_variable_dest) {
|
|
switch(option->clo_type) {
|
|
case OPAL_CMD_LINE_TYPE_STRING:
|
|
*((char**) option->clo_variable_dest) = strdup(sval);
|
|
break;
|
|
case OPAL_CMD_LINE_TYPE_INT:
|
|
/* check to see that the value given to us truly is an int */
|
|
for (i=0; i < strlen(sval); i++) {
|
|
if (!isdigit(sval[i]) && '-' != sval[i]) {
|
|
/* show help isn't going to be available yet, so just
|
|
* print the msg
|
|
*/
|
|
fprintf(stderr, "----------------------------------------------------------------------------\n");
|
|
fprintf(stderr, "Open MPI has detected that a parameter given to a command line\n");
|
|
fprintf(stderr, "option does not match the expected format:\n\n");
|
|
if (NULL != option->clo_long_name) {
|
|
fprintf(stderr, " Option: %s\n", option->clo_long_name);
|
|
} else if ('\0' != option->clo_short_name) {
|
|
fprintf(stderr, " Option: %c\n", option->clo_short_name);
|
|
} else {
|
|
fprintf(stderr, " Option: <unknown>\n");
|
|
}
|
|
fprintf(stderr, " Param: %s\n\n", sval);
|
|
fprintf(stderr, "This is frequently caused by omitting to provide the parameter\n");
|
|
fprintf(stderr, "to an option that requires one. Please check the command line and try again.\n");
|
|
fprintf(stderr, "----------------------------------------------------------------------------\n");
|
|
return OPAL_ERR_SILENT;
|
|
}
|
|
}
|
|
*((int*) option->clo_variable_dest) = ival;
|
|
break;
|
|
case OPAL_CMD_LINE_TYPE_SIZE_T:
|
|
/* check to see that the value given to us truly is a size_t */
|
|
for (i=0; i < strlen(sval); i++) {
|
|
if (!isdigit(sval[i]) && '-' != sval[i]) {
|
|
/* show help isn't going to be available yet, so just
|
|
* print the msg
|
|
*/
|
|
fprintf(stderr, "----------------------------------------------------------------------------\n");
|
|
fprintf(stderr, "Open MPI has detected that a parameter given to a command line\n");
|
|
fprintf(stderr, "option does not match the expected format:\n\n");
|
|
if (NULL != option->clo_long_name) {
|
|
fprintf(stderr, " Option: %s\n", option->clo_long_name);
|
|
} else if ('\0' != option->clo_short_name) {
|
|
fprintf(stderr, " Option: %c\n", option->clo_short_name);
|
|
} else {
|
|
fprintf(stderr, " Option: <unknown>\n");
|
|
}
|
|
fprintf(stderr, " Param: %s\n\n", sval);
|
|
fprintf(stderr, "This is frequently caused by omitting to provide the parameter\n");
|
|
fprintf(stderr, "to an option that requires one. Please check the command line and try again.\n");
|
|
fprintf(stderr, "----------------------------------------------------------------------------\n");
|
|
return OPAL_ERR_SILENT;
|
|
}
|
|
}
|
|
*((size_t*) option->clo_variable_dest) = lval;
|
|
break;
|
|
case OPAL_CMD_LINE_TYPE_BOOL:
|
|
*((bool*) option->clo_variable_dest) = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return OPAL_SUCCESS;
|
|
}
|
|
|
|
|
|
/*
|
|
* Helper function to qsort_callback
|
|
*/
|
|
static void fill(const cmd_line_option_t *a, char result[3][BUFSIZ])
|
|
{
|
|
int i = 0;
|
|
|
|
result[0][0] = '\0';
|
|
result[1][0] = '\0';
|
|
result[2][0] = '\0';
|
|
|
|
if ('\0' != a->clo_short_name) {
|
|
snprintf(&result[i][0], BUFSIZ, "%c", a->clo_short_name);
|
|
++i;
|
|
}
|
|
if (NULL != a->clo_single_dash_name) {
|
|
snprintf(&result[i][0], BUFSIZ, "%s", a->clo_single_dash_name);
|
|
++i;
|
|
}
|
|
if (NULL != a->clo_long_name) {
|
|
snprintf(&result[i][0], BUFSIZ, "%s", a->clo_long_name);
|
|
++i;
|
|
}
|
|
}
|
|
|
|
|
|
static int qsort_callback(const void *aa, const void *bb)
|
|
{
|
|
int ret, i;
|
|
char str1[3][BUFSIZ], str2[3][BUFSIZ];
|
|
const cmd_line_option_t *a = *((const cmd_line_option_t**) aa);
|
|
const cmd_line_option_t *b = *((const cmd_line_option_t**) bb);
|
|
|
|
/* Icky comparison of command line options. There are multiple
|
|
forms of each command line option, so we first have to check
|
|
which forms each option has. Compare, in order: short name,
|
|
single-dash name, long name. */
|
|
|
|
fill(a, str1);
|
|
fill(b, str2);
|
|
|
|
for (i = 0; i < 3; ++i) {
|
|
if (0 != (ret = strcasecmp(str1[i], str2[i]))) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
/* Shrug -- they must be equal */
|
|
|
|
return 0;
|
|
}
|