2015-05-27 18:06:26 +03:00
|
|
|
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
|
|
* University Research and Technology
|
|
|
|
* Corporation. All rights reserved.
|
|
|
|
* Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
|
|
* of Tennessee Research Foundation. All rights
|
|
|
|
* reserved.
|
|
|
|
* 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.
|
2016-02-18 00:58:05 +03:00
|
|
|
* Copyright (c) 2015-2016 Los Alamos National Security, LLC. All rights
|
2015-05-27 18:06:26 +03:00
|
|
|
* reserved.
|
2018-11-29 01:52:52 +03:00
|
|
|
* Copyright (c) 2018 Triad National Security, LLC. All rights
|
|
|
|
* reserved.
|
2015-05-27 18:06:26 +03:00
|
|
|
* $COPYRIGHT$
|
|
|
|
*
|
|
|
|
* Additional copyrights may follow
|
|
|
|
*
|
|
|
|
* $HEADER$
|
|
|
|
*/
|
2006-01-16 04:48:03 +03:00
|
|
|
|
2006-02-12 04:33:29 +03:00
|
|
|
#include "opal_config.h"
|
2006-01-16 04:48:03 +03:00
|
|
|
|
2006-02-12 04:33:29 +03:00
|
|
|
#include "opal/constants.h"
|
2018-11-29 01:52:52 +03:00
|
|
|
#include "opal/runtime/opal.h"
|
2006-01-16 04:48:03 +03:00
|
|
|
#include "opal/util/keyval_parse.h"
|
|
|
|
#include "opal/util/keyval/keyval_lex.h"
|
|
|
|
#include "opal/util/output.h"
|
2018-09-27 20:31:12 +03:00
|
|
|
#include "opal/util/string_copy.h"
|
2019-02-05 06:20:37 +03:00
|
|
|
#include "opal/mca/threads/mutex.h"
|
2006-01-17 16:22:08 +03:00
|
|
|
#include <string.h>
|
2016-02-18 00:58:05 +03:00
|
|
|
#include <ctype.h>
|
2006-01-16 04:48:03 +03:00
|
|
|
|
2015-05-12 17:55:47 +03:00
|
|
|
int opal_util_keyval_parse_lineno = 0;
|
|
|
|
|
2006-01-16 04:48:03 +03:00
|
|
|
static const char *keyval_filename;
|
|
|
|
static opal_keyval_parse_fn_t keyval_callback;
|
|
|
|
static char *key_buffer = NULL;
|
|
|
|
static size_t key_buffer_len = 0;
|
|
|
|
static opal_mutex_t keyval_mutex;
|
|
|
|
|
|
|
|
static int parse_line(void);
|
2015-05-27 18:06:26 +03:00
|
|
|
static int parse_line_new(opal_keyval_parse_state_t first_val);
|
2006-01-16 04:48:03 +03:00
|
|
|
static void parse_error(int num);
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
static char *env_str = NULL;
|
|
|
|
static int envsize = 1024;
|
|
|
|
|
2018-11-29 01:52:52 +03:00
|
|
|
static void opal_util_keyval_parse_finalize (void)
|
2006-01-16 04:48:03 +03:00
|
|
|
{
|
2018-11-29 01:52:52 +03:00
|
|
|
free(key_buffer);
|
|
|
|
key_buffer = NULL;
|
|
|
|
key_buffer_len = 0;
|
2006-01-16 04:48:03 +03:00
|
|
|
|
2018-11-29 01:52:52 +03:00
|
|
|
OBJ_DESTRUCT(&keyval_mutex);
|
2006-01-16 04:48:03 +03:00
|
|
|
}
|
|
|
|
|
2018-11-29 01:52:52 +03:00
|
|
|
int opal_util_keyval_parse_init(void)
|
2006-01-16 04:48:03 +03:00
|
|
|
{
|
2018-11-29 01:52:52 +03:00
|
|
|
OBJ_CONSTRUCT(&keyval_mutex, opal_mutex_t);
|
2006-01-16 04:48:03 +03:00
|
|
|
|
2018-11-29 01:52:52 +03:00
|
|
|
opal_finalize_register_cleanup (opal_util_keyval_parse_finalize);
|
2006-01-16 04:48:03 +03:00
|
|
|
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
opal_util_keyval_parse(const char *filename,
|
|
|
|
opal_keyval_parse_fn_t callback)
|
|
|
|
{
|
|
|
|
int val;
|
|
|
|
int ret = OPAL_SUCCESS;;
|
|
|
|
|
2006-01-19 00:48:40 +03:00
|
|
|
OPAL_THREAD_LOCK(&keyval_mutex);
|
2006-01-16 04:48:03 +03:00
|
|
|
|
|
|
|
keyval_filename = filename;
|
|
|
|
keyval_callback = callback;
|
|
|
|
|
|
|
|
/* Open the opal */
|
|
|
|
opal_util_keyval_yyin = fopen(keyval_filename, "r");
|
|
|
|
if (NULL == opal_util_keyval_yyin) {
|
|
|
|
ret = OPAL_ERR_NOT_FOUND;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
opal_util_keyval_parse_done = false;
|
|
|
|
opal_util_keyval_yynewlines = 1;
|
|
|
|
opal_util_keyval_init_buffer(opal_util_keyval_yyin);
|
|
|
|
while (!opal_util_keyval_parse_done) {
|
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
switch (val) {
|
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_DONE:
|
|
|
|
/* This will also set opal_util_keyval_parse_done to true, so just
|
|
|
|
break here */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_NEWLINE:
|
|
|
|
/* blank line! ignore it */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_SINGLE_WORD:
|
|
|
|
parse_line();
|
|
|
|
break;
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_MCAVAR:
|
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_ENVVAR:
|
|
|
|
case OPAL_UTIL_KEYVAL_PARSE_ENVEQL:
|
|
|
|
parse_line_new(val);
|
|
|
|
break;
|
|
|
|
|
2006-01-16 04:48:03 +03:00
|
|
|
default:
|
|
|
|
/* anything else is an error */
|
|
|
|
parse_error(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(opal_util_keyval_yyin);
|
2012-11-09 20:10:55 +04:00
|
|
|
opal_util_keyval_yylex_destroy ();
|
2006-01-16 04:48:03 +03:00
|
|
|
|
|
|
|
cleanup:
|
2006-01-19 00:48:40 +03:00
|
|
|
OPAL_THREAD_UNLOCK(&keyval_mutex);
|
2006-01-16 04:48:03 +03:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int parse_line(void)
|
|
|
|
{
|
|
|
|
int val;
|
|
|
|
|
2015-05-12 17:55:47 +03:00
|
|
|
opal_util_keyval_parse_lineno = opal_util_keyval_yylineno;
|
|
|
|
|
2006-01-16 04:48:03 +03:00
|
|
|
/* Save the name name */
|
|
|
|
if (key_buffer_len < strlen(opal_util_keyval_yytext) + 1) {
|
|
|
|
char *tmp;
|
|
|
|
key_buffer_len = strlen(opal_util_keyval_yytext) + 1;
|
2006-08-23 00:07:42 +04:00
|
|
|
tmp = (char*)realloc(key_buffer, key_buffer_len);
|
2006-01-16 04:48:03 +03:00
|
|
|
if (NULL == tmp) {
|
|
|
|
free(key_buffer);
|
|
|
|
key_buffer_len = 0;
|
|
|
|
key_buffer = NULL;
|
|
|
|
return OPAL_ERR_TEMP_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
key_buffer = tmp;
|
|
|
|
}
|
2006-01-17 16:22:08 +03:00
|
|
|
|
2018-09-27 20:31:12 +03:00
|
|
|
opal_string_copy(key_buffer, opal_util_keyval_yytext, key_buffer_len);
|
2006-01-16 04:48:03 +03:00
|
|
|
|
|
|
|
/* The first thing we have to see is an "=" */
|
|
|
|
|
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
if (opal_util_keyval_parse_done || OPAL_UTIL_KEYVAL_PARSE_EQUAL != val) {
|
|
|
|
parse_error(2);
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Next we get the value */
|
|
|
|
|
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
if (OPAL_UTIL_KEYVAL_PARSE_SINGLE_WORD == val ||
|
|
|
|
OPAL_UTIL_KEYVAL_PARSE_VALUE == val) {
|
|
|
|
keyval_callback(key_buffer, opal_util_keyval_yytext);
|
|
|
|
|
|
|
|
/* Now we need to see the newline */
|
|
|
|
|
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
if (OPAL_UTIL_KEYVAL_PARSE_NEWLINE == val ||
|
|
|
|
OPAL_UTIL_KEYVAL_PARSE_DONE == val) {
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Did we get an EOL or EOF? */
|
|
|
|
|
|
|
|
else if (OPAL_UTIL_KEYVAL_PARSE_DONE == val ||
|
|
|
|
OPAL_UTIL_KEYVAL_PARSE_NEWLINE == val) {
|
|
|
|
keyval_callback(key_buffer, NULL);
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Nope -- we got something unexpected. Bonk! */
|
|
|
|
parse_error(3);
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void parse_error(int num)
|
|
|
|
{
|
|
|
|
/* JMS need better error/warning message here */
|
|
|
|
opal_output(0, "keyval parser: error %d reading file %s at line %d:\n %s\n",
|
|
|
|
num, keyval_filename, opal_util_keyval_yynewlines, opal_util_keyval_yytext);
|
|
|
|
}
|
2015-03-16 19:03:55 +03:00
|
|
|
|
|
|
|
int opal_util_keyval_save_internal_envars(opal_keyval_parse_fn_t callback)
|
|
|
|
{
|
|
|
|
if (NULL != env_str && 0 < strlen(env_str)) {
|
|
|
|
callback("mca_base_env_list_internal", env_str);
|
|
|
|
free(env_str);
|
|
|
|
env_str = NULL;
|
|
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2016-02-18 00:58:05 +03:00
|
|
|
static void trim_name(char *buffer, const char* prefix, const char* suffix)
|
2015-03-16 19:03:55 +03:00
|
|
|
{
|
2016-02-18 00:58:05 +03:00
|
|
|
char *pchr, *echr;
|
|
|
|
size_t buffer_len;
|
|
|
|
|
|
|
|
if (NULL == buffer) {
|
|
|
|
return;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
2016-02-18 00:58:05 +03:00
|
|
|
|
|
|
|
buffer_len = strlen (buffer);
|
|
|
|
|
|
|
|
pchr = buffer;
|
2015-03-16 19:03:55 +03:00
|
|
|
if (NULL != prefix) {
|
2016-02-18 00:58:05 +03:00
|
|
|
size_t prefix_len = strlen (prefix);
|
|
|
|
|
|
|
|
if (0 == strncmp (buffer, prefix, prefix_len)) {
|
|
|
|
pchr += prefix_len;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
}
|
2016-02-18 00:58:05 +03:00
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
/* trim spaces at the beginning */
|
2016-02-18 00:58:05 +03:00
|
|
|
while (isspace (*pchr)) {
|
2015-03-16 19:03:55 +03:00
|
|
|
pchr++;
|
|
|
|
}
|
2016-02-18 00:58:05 +03:00
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
/* trim spaces at the end */
|
2016-03-08 00:51:29 +03:00
|
|
|
echr = buffer + buffer_len;
|
2016-03-08 20:06:14 +03:00
|
|
|
while (echr > buffer && isspace (*(echr - 1))) {
|
2015-03-16 19:03:55 +03:00
|
|
|
echr--;
|
|
|
|
}
|
2016-03-08 00:51:29 +03:00
|
|
|
echr[0] = '\0';
|
2016-02-18 00:58:05 +03:00
|
|
|
|
2016-03-08 00:51:29 +03:00
|
|
|
if (NULL != suffix && (uintptr_t) (echr - buffer) > strlen (suffix)) {
|
2016-02-18 00:58:05 +03:00
|
|
|
size_t suffix_len = strlen (suffix);
|
|
|
|
|
2016-03-08 00:51:29 +03:00
|
|
|
echr -= suffix_len;
|
2016-02-18 00:58:05 +03:00
|
|
|
|
|
|
|
if (0 == strncmp (echr, suffix, strlen(suffix))) {
|
|
|
|
do {
|
2015-03-16 19:03:55 +03:00
|
|
|
echr--;
|
2016-02-18 00:58:05 +03:00
|
|
|
} while (isspace (*echr));
|
|
|
|
echr[1] = '\0';
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
}
|
2016-02-18 00:58:05 +03:00
|
|
|
|
|
|
|
if (buffer != pchr) {
|
|
|
|
/* move the trimmed string to the beginning of the buffer */
|
|
|
|
memmove (buffer, pchr, strlen (pchr) + 1);
|
|
|
|
}
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
|
2016-02-18 00:58:05 +03:00
|
|
|
static int save_param_name (void)
|
2015-03-16 19:03:55 +03:00
|
|
|
{
|
|
|
|
if (key_buffer_len < strlen(opal_util_keyval_yytext) + 1) {
|
|
|
|
char *tmp;
|
|
|
|
key_buffer_len = strlen(opal_util_keyval_yytext) + 1;
|
|
|
|
tmp = (char*)realloc(key_buffer, key_buffer_len);
|
|
|
|
if (NULL == tmp) {
|
|
|
|
free(key_buffer);
|
|
|
|
key_buffer_len = 0;
|
|
|
|
key_buffer = NULL;
|
|
|
|
return OPAL_ERR_TEMP_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
key_buffer = tmp;
|
|
|
|
}
|
|
|
|
|
2018-09-27 20:31:12 +03:00
|
|
|
opal_string_copy (key_buffer, opal_util_keyval_yytext, key_buffer_len);
|
2016-02-18 00:58:05 +03:00
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int add_to_env_str(char *var, char *val)
|
|
|
|
{
|
|
|
|
int sz, varsz, valsz;
|
2015-05-27 18:06:26 +03:00
|
|
|
void *tmp;
|
|
|
|
|
|
|
|
if (NULL == var) {
|
|
|
|
return OPAL_ERR_BAD_PARAM;
|
|
|
|
}
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
if (NULL != env_str) {
|
2015-05-27 18:06:26 +03:00
|
|
|
varsz = strlen(var);
|
2015-03-16 19:03:55 +03:00
|
|
|
valsz = (NULL != val) ? strlen(val) : 0;
|
|
|
|
sz = strlen(env_str)+varsz+valsz+2;
|
|
|
|
if (envsize <= sz) {
|
|
|
|
envsize *=2;
|
2015-05-27 18:06:26 +03:00
|
|
|
|
|
|
|
tmp = realloc(env_str, envsize);
|
|
|
|
if (NULL == tmp) {
|
|
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
|
|
|
env_str = tmp;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
strcat(env_str, ";");
|
|
|
|
} else {
|
2015-05-27 18:06:26 +03:00
|
|
|
env_str = calloc(1, envsize);
|
|
|
|
if (NULL == env_str) {
|
|
|
|
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
|
|
}
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
2015-05-27 18:06:26 +03:00
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
strcat(env_str, var);
|
|
|
|
if (NULL != val) {
|
|
|
|
strcat(env_str, "=");
|
|
|
|
strcat(env_str, val);
|
|
|
|
}
|
2015-05-27 18:06:26 +03:00
|
|
|
|
|
|
|
return OPAL_SUCCESS;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
|
2015-05-27 18:06:26 +03:00
|
|
|
static int parse_line_new(opal_keyval_parse_state_t first_val)
|
2015-03-16 19:03:55 +03:00
|
|
|
{
|
2015-05-27 18:06:26 +03:00
|
|
|
opal_keyval_parse_state_t val;
|
2015-03-16 19:03:55 +03:00
|
|
|
char *tmp;
|
2016-02-18 00:58:05 +03:00
|
|
|
int rc;
|
2015-03-16 19:03:55 +03:00
|
|
|
|
|
|
|
val = first_val;
|
|
|
|
while (OPAL_UTIL_KEYVAL_PARSE_NEWLINE != val && OPAL_UTIL_KEYVAL_PARSE_DONE != val) {
|
2016-02-18 00:58:05 +03:00
|
|
|
rc = save_param_name ();
|
|
|
|
if (OPAL_SUCCESS != rc) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
if (OPAL_UTIL_KEYVAL_PARSE_MCAVAR == val) {
|
2016-02-18 00:58:05 +03:00
|
|
|
trim_name (key_buffer, "-mca", NULL);
|
|
|
|
trim_name (key_buffer, "--mca", NULL);
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
if (OPAL_UTIL_KEYVAL_PARSE_VALUE == val) {
|
|
|
|
if (NULL != opal_util_keyval_yytext) {
|
|
|
|
tmp = strdup(opal_util_keyval_yytext);
|
|
|
|
if ('\'' == tmp[0] || '\"' == tmp[0]) {
|
2016-02-18 00:58:05 +03:00
|
|
|
trim_name (tmp, "\'", "\'");
|
|
|
|
trim_name (tmp, "\"", "\"");
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
keyval_callback(key_buffer, tmp);
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
parse_error(4);
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
} else if (OPAL_UTIL_KEYVAL_PARSE_ENVEQL == val) {
|
2016-02-18 00:58:05 +03:00
|
|
|
trim_name (key_buffer, "-x", "=");
|
|
|
|
trim_name (key_buffer, "--x", NULL);
|
|
|
|
|
2015-03-16 19:03:55 +03:00
|
|
|
val = opal_util_keyval_yylex();
|
|
|
|
if (OPAL_UTIL_KEYVAL_PARSE_VALUE == val) {
|
|
|
|
add_to_env_str(key_buffer, opal_util_keyval_yytext);
|
|
|
|
} else {
|
|
|
|
parse_error(5);
|
|
|
|
return OPAL_ERROR;
|
|
|
|
}
|
|
|
|
} else if (OPAL_UTIL_KEYVAL_PARSE_ENVVAR == val) {
|
2016-02-18 00:58:05 +03:00
|
|
|
trim_name (key_buffer, "-x", "=");
|
|
|
|
trim_name (key_buffer, "--x", NULL);
|
2015-03-16 19:03:55 +03:00
|
|
|
add_to_env_str(key_buffer, NULL);
|
2015-05-27 18:06:26 +03:00
|
|
|
} else {
|
|
|
|
/* we got something unexpected. Bonk! */
|
|
|
|
parse_error(6);
|
|
|
|
return OPAL_ERROR;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
|
2015-05-27 18:06:26 +03:00
|
|
|
val = opal_util_keyval_yylex();
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|
|
|
|
|
2015-05-27 18:06:26 +03:00
|
|
|
return OPAL_SUCCESS;
|
2015-03-16 19:03:55 +03:00
|
|
|
}
|