1
1
openmpi/opal/util/opal_environ.c

212 строки
4.9 KiB
C
Исходник Обычный вид История

/*
* Copyright (c) 2004-2005 The Trustees of Indiana University.
* All rights reserved.
* Copyright (c) 2004-2005 The Trustees of the University of Tennessee.
* 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$
*
* Additional copyrights may follow
*
* $HEADER$
*/
#include "ompi_config.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "include/constants.h"
#include "opal/util/printf.h"
#include "opal/util/argv.h"
#include "opal/util/opal_environ.h"
/*
* Merge two environ-like char arrays, ensuring that there are no
* duplicate entires
*/
char **opal_environ_merge(char **minor, char **major)
{
int i;
char **ret = NULL;
char *name, *value;
/* Check for bozo cases */
if (NULL == major) {
if (NULL == minor) {
return NULL;
} else {
return opal_argv_copy(minor);
}
}
/* First, copy major */
ret = opal_argv_copy(major);
/* Do we have something in minor? */
if (NULL == minor) {
return ret;
}
/* Now go through minor and call opal_setenv(), but with overwrite
as false */
for (i = 0; NULL != minor[i]; ++i) {
value = strchr(minor[i], '=');
if (NULL == value) {
opal_setenv(minor[i], NULL, false, &ret);
} else {
/* strdup minor[i] in case it's a constat string */
name = strdup(minor[i]);
value = name + (value - minor[i]);
*value = '\0';
opal_setenv(name, value + 1, false, &ret);
free(name);
}
}
/* All done */
return ret;
}
/*
* Portable version of setenv(), allowing editing of any environ-like
* array
*/
int opal_setenv(const char *name, const char *value, bool overwrite,
char ***env)
{
int i;
char *newvalue, *compare;
size_t len;
extern char **environ;
/* Make the new value */
if (NULL == value) {
asprintf(&newvalue, "%s=", name);
} else {
asprintf(&newvalue, "%s=%s", name, value);
}
if (NULL == newvalue) {
return OMPI_ERR_OUT_OF_RESOURCE;
}
/* Check the bozo case */
if (NULL == env) {
return OMPI_ERR_BAD_PARAM;
} else if (NULL == *env) {
i = 0;
opal_argv_append(&i, env, newvalue);
return OMPI_SUCCESS;
}
/* If this is the "environ" array, use putenv */
if (*env == environ) {
/* THIS IS POTENTIALLY A MEMORY LEAK! But I am doing it
because so that we don't violate the law of least
astonishmet for OPAL developers (i.e., those that don't
check the return code of opal_setenv() and notice that we
returned an error if you passed in the real environ) */
putenv(newvalue);
return OMPI_SUCCESS;
}
/* Make something easy to compare to */
asprintf(&compare, "%s=", name);
if (NULL == compare) {
free(newvalue);
return OMPI_ERR_OUT_OF_RESOURCE;
}
len = strlen(compare);
/* Look for a duplicate that's already set in the env */
for (i = 0; (*env)[i] != NULL; ++i) {
if (0 == strncmp((*env)[i], compare, len)) {
if (overwrite) {
free((*env)[i]);
(*env)[i] = newvalue;
free(compare);
return OMPI_SUCCESS;
} else {
free(compare);
free(newvalue);
return OMPI_EXISTS;
}
}
}
/* If we found no match, append this value */
i = opal_argv_count(*env);
opal_argv_append(&i, env, newvalue);
/* All done */
free(compare);
free(newvalue);
return OMPI_SUCCESS;
}
/*
* Portable version of unsetenv(), allowing editing of any
* environ-like array
*/
int opal_unsetenv(const char *name, char ***env)
{
int i;
char *compare;
size_t len;
bool found;
/* Check for bozo case */
if (NULL == *env) {
return OMPI_SUCCESS;
}
/* Make something easy to compare to */
asprintf(&compare, "%s=", name);
if (NULL == compare) {
return OMPI_ERR_OUT_OF_RESOURCE;
}
len = strlen(compare);
/* Look for a duplicate that's already set in the env. If we find
it, free it, and then start shifting all elements down one in
the array. */
found = false;
for (i = 0; (*env)[i] != NULL; ++i) {
if (found) {
(*env)[i] = (*env)[i + 1];
} else if (0 == strncmp((*env)[i], compare, len)) {
free((*env)[i]);
(*env)[i] = (*env)[i + 1];
found = true;
}
}
/* All done */
return (found) ? OMPI_SUCCESS : OMPI_ERR_NOT_FOUND;
}