linux-user: Add generic env variable handling
Adds support for qemu to modify target process environment variables using -E and -U commandline switches. This replaces eventually the -drop-ld-preload flag. Signed-off-by: Aurelien Jarno <aurelien@aurel32.net> Signed-off-by: Riku Voipio <riku.voipio@iki.fi> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@6484 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
parent
e1ce5e400a
commit
04a6dfebb6
@ -320,7 +320,7 @@ CFLAGS+=-p
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
OBJS= main.o syscall.o strace.o mmap.o signal.o path.o thunk.o \
|
OBJS= main.o syscall.o strace.o mmap.o signal.o path.o thunk.o \
|
||||||
elfload.o linuxload.o uaccess.o
|
elfload.o linuxload.o uaccess.o envlist.o
|
||||||
LIBS+= $(AIOLIBS)
|
LIBS+= $(AIOLIBS)
|
||||||
ifdef TARGET_HAS_BFLT
|
ifdef TARGET_HAS_BFLT
|
||||||
OBJS+= flatload.o
|
OBJS+= flatload.o
|
||||||
|
247
linux-user/envlist.c
Normal file
247
linux-user/envlist.c
Normal file
@ -0,0 +1,247 @@
|
|||||||
|
#include <sys/queue.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "envlist.h"
|
||||||
|
|
||||||
|
struct envlist_entry {
|
||||||
|
const char *ev_var; /* actual env value */
|
||||||
|
LIST_ENTRY(envlist_entry) ev_link;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct envlist {
|
||||||
|
LIST_HEAD(, envlist_entry) el_entries; /* actual entries */
|
||||||
|
size_t el_count; /* number of entries */
|
||||||
|
};
|
||||||
|
|
||||||
|
static int envlist_parse(envlist_t *envlist,
|
||||||
|
const char *env, int (*)(envlist_t *, const char *));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Allocates new envlist and returns pointer to that or
|
||||||
|
* NULL in case of error.
|
||||||
|
*/
|
||||||
|
envlist_t *
|
||||||
|
envlist_create(void)
|
||||||
|
{
|
||||||
|
envlist_t *envlist;
|
||||||
|
|
||||||
|
if ((envlist = malloc(sizeof (*envlist))) == NULL)
|
||||||
|
return (NULL);
|
||||||
|
|
||||||
|
LIST_INIT(&envlist->el_entries);
|
||||||
|
envlist->el_count = 0;
|
||||||
|
|
||||||
|
return (envlist);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Releases given envlist and its entries.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
envlist_free(envlist_t *envlist)
|
||||||
|
{
|
||||||
|
struct envlist_entry *entry;
|
||||||
|
|
||||||
|
assert(envlist != NULL);
|
||||||
|
|
||||||
|
while (envlist->el_entries.lh_first != NULL) {
|
||||||
|
entry = envlist->el_entries.lh_first;
|
||||||
|
LIST_REMOVE(entry, ev_link);
|
||||||
|
|
||||||
|
free((char *)entry->ev_var);
|
||||||
|
free(entry);
|
||||||
|
}
|
||||||
|
free(envlist);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Parses comma separated list of set/modify environment
|
||||||
|
* variable entries and updates given enlist accordingly.
|
||||||
|
*
|
||||||
|
* For example:
|
||||||
|
* envlist_parse(el, "HOME=foo,SHELL=/bin/sh");
|
||||||
|
*
|
||||||
|
* inserts/sets environment variables HOME and SHELL.
|
||||||
|
*
|
||||||
|
* Returns 0 on success, errno otherwise.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
envlist_parse_set(envlist_t *envlist, const char *env)
|
||||||
|
{
|
||||||
|
return (envlist_parse(envlist, env, &envlist_setenv));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Parses comma separated list of unset environment variable
|
||||||
|
* entries and removes given variables from given envlist.
|
||||||
|
*
|
||||||
|
* Returns 0 on success, errno otherwise.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
envlist_parse_unset(envlist_t *envlist, const char *env)
|
||||||
|
{
|
||||||
|
return (envlist_parse(envlist, env, &envlist_unsetenv));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Parses comma separated list of set, modify or unset entries
|
||||||
|
* and calls given callback for each entry.
|
||||||
|
*
|
||||||
|
* Returns 0 in case of success, errno otherwise.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
envlist_parse(envlist_t *envlist, const char *env,
|
||||||
|
int (*callback)(envlist_t *, const char *))
|
||||||
|
{
|
||||||
|
char *tmpenv, *envvar;
|
||||||
|
char *envsave = NULL;
|
||||||
|
|
||||||
|
assert(callback != NULL);
|
||||||
|
|
||||||
|
if ((envlist == NULL) || (env == NULL))
|
||||||
|
return (EINVAL);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We need to make temporary copy of the env string
|
||||||
|
* as strtok_r(3) modifies it while it tokenizes.
|
||||||
|
*/
|
||||||
|
if ((tmpenv = strdup(env)) == NULL)
|
||||||
|
return (errno);
|
||||||
|
|
||||||
|
envvar = strtok_r(tmpenv, ",", &envsave);
|
||||||
|
while (envvar != NULL) {
|
||||||
|
if ((*callback)(envlist, envvar) != 0) {
|
||||||
|
free(tmpenv);
|
||||||
|
return (errno);
|
||||||
|
}
|
||||||
|
envvar = strtok_r(NULL, ",", &envsave);
|
||||||
|
}
|
||||||
|
|
||||||
|
free(tmpenv);
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Sets environment value to envlist in similar manner
|
||||||
|
* than putenv(3).
|
||||||
|
*
|
||||||
|
* Returns 0 in success, errno otherwise.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
envlist_setenv(envlist_t *envlist, const char *env)
|
||||||
|
{
|
||||||
|
struct envlist_entry *entry = NULL;
|
||||||
|
const char *eq_sign;
|
||||||
|
size_t envname_len;
|
||||||
|
|
||||||
|
if ((envlist == NULL) || (env == NULL))
|
||||||
|
return (EINVAL);
|
||||||
|
|
||||||
|
/* find out first equals sign in given env */
|
||||||
|
if ((eq_sign = strchr(env, '=')) == NULL)
|
||||||
|
return (EINVAL);
|
||||||
|
envname_len = eq_sign - env + 1;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there already exists variable with given name
|
||||||
|
* we remove and release it before allocating a whole
|
||||||
|
* new entry.
|
||||||
|
*/
|
||||||
|
for (entry = envlist->el_entries.lh_first; entry != NULL;
|
||||||
|
entry = entry->ev_link.le_next) {
|
||||||
|
if (strncmp(entry->ev_var, env, envname_len) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entry != NULL) {
|
||||||
|
LIST_REMOVE(entry, ev_link);
|
||||||
|
free((char *)entry->ev_var);
|
||||||
|
free(entry);
|
||||||
|
} else {
|
||||||
|
envlist->el_count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((entry = malloc(sizeof (*entry))) == NULL)
|
||||||
|
return (errno);
|
||||||
|
if ((entry->ev_var = strdup(env)) == NULL) {
|
||||||
|
free(entry);
|
||||||
|
return (errno);
|
||||||
|
}
|
||||||
|
LIST_INSERT_HEAD(&envlist->el_entries, entry, ev_link);
|
||||||
|
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Removes given env value from envlist in similar manner
|
||||||
|
* than unsetenv(3). Returns 0 in success, errno otherwise.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
envlist_unsetenv(envlist_t *envlist, const char *env)
|
||||||
|
{
|
||||||
|
struct envlist_entry *entry;
|
||||||
|
size_t envname_len;
|
||||||
|
|
||||||
|
if ((envlist == NULL) || (env == NULL))
|
||||||
|
return (EINVAL);
|
||||||
|
|
||||||
|
/* env is not allowed to contain '=' */
|
||||||
|
if (strchr(env, '=') != NULL)
|
||||||
|
return (EINVAL);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Find out the requested entry and remove
|
||||||
|
* it from the list.
|
||||||
|
*/
|
||||||
|
envname_len = strlen(env);
|
||||||
|
for (entry = envlist->el_entries.lh_first; entry != NULL;
|
||||||
|
entry = entry->ev_link.le_next) {
|
||||||
|
if (strncmp(entry->ev_var, env, envname_len) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (entry != NULL) {
|
||||||
|
LIST_REMOVE(entry, ev_link);
|
||||||
|
free((char *)entry->ev_var);
|
||||||
|
free(entry);
|
||||||
|
|
||||||
|
envlist->el_count--;
|
||||||
|
}
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns given envlist as array of strings (in same form that
|
||||||
|
* global variable environ is). Caller must free returned memory
|
||||||
|
* by calling free(3) for each element and for the array. Returned
|
||||||
|
* array and given envlist are not related (no common references).
|
||||||
|
*
|
||||||
|
* If caller provides count pointer, number of items in array is
|
||||||
|
* stored there. In case of error, NULL is returned and no memory
|
||||||
|
* is allocated.
|
||||||
|
*/
|
||||||
|
char **
|
||||||
|
envlist_to_environ(const envlist_t *envlist, size_t *count)
|
||||||
|
{
|
||||||
|
struct envlist_entry *entry;
|
||||||
|
char **env, **penv;
|
||||||
|
|
||||||
|
penv = env = malloc((envlist->el_count + 1) * sizeof (char *));
|
||||||
|
if (env == NULL)
|
||||||
|
return (NULL);
|
||||||
|
|
||||||
|
for (entry = envlist->el_entries.lh_first; entry != NULL;
|
||||||
|
entry = entry->ev_link.le_next) {
|
||||||
|
*(penv++) = strdup(entry->ev_var);
|
||||||
|
}
|
||||||
|
*penv = NULL; /* NULL terminate the list */
|
||||||
|
|
||||||
|
if (count != NULL)
|
||||||
|
*count = envlist->el_count;
|
||||||
|
|
||||||
|
return (env);
|
||||||
|
}
|
22
linux-user/envlist.h
Normal file
22
linux-user/envlist.h
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef ENVLIST_H
|
||||||
|
#define ENVLIST_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct envlist envlist_t;
|
||||||
|
|
||||||
|
extern envlist_t *envlist_create(void);
|
||||||
|
extern void envlist_free(envlist_t *);
|
||||||
|
extern int envlist_setenv(envlist_t *, const char *);
|
||||||
|
extern int envlist_unsetenv(envlist_t *, const char *);
|
||||||
|
extern int envlist_parse_set(envlist_t *, const char *);
|
||||||
|
extern int envlist_parse_unset(envlist_t *, const char *);
|
||||||
|
extern char **envlist_to_environ(const envlist_t *, size_t *);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* ENVLIST_H */
|
@ -32,6 +32,9 @@
|
|||||||
/* For tb_lock */
|
/* For tb_lock */
|
||||||
#include "exec-all.h"
|
#include "exec-all.h"
|
||||||
|
|
||||||
|
|
||||||
|
#include "envlist.h"
|
||||||
|
|
||||||
#define DEBUG_LOGFILE "/tmp/qemu.log"
|
#define DEBUG_LOGFILE "/tmp/qemu.log"
|
||||||
|
|
||||||
static const char *interp_prefix = CONFIG_QEMU_PREFIX;
|
static const char *interp_prefix = CONFIG_QEMU_PREFIX;
|
||||||
@ -2186,6 +2189,8 @@ static void usage(void)
|
|||||||
"-s size set the stack size in bytes (default=%ld)\n"
|
"-s size set the stack size in bytes (default=%ld)\n"
|
||||||
"-cpu model select CPU (-cpu ? for list)\n"
|
"-cpu model select CPU (-cpu ? for list)\n"
|
||||||
"-drop-ld-preload drop LD_PRELOAD for target process\n"
|
"-drop-ld-preload drop LD_PRELOAD for target process\n"
|
||||||
|
"-E var=value sets/modifies targets environment variable(s)\n"
|
||||||
|
"-U var unsets targets environment variable(s)\n"
|
||||||
"\n"
|
"\n"
|
||||||
"Debug options:\n"
|
"Debug options:\n"
|
||||||
"-d options activate log (logfile=%s)\n"
|
"-d options activate log (logfile=%s)\n"
|
||||||
@ -2195,6 +2200,12 @@ static void usage(void)
|
|||||||
"Environment variables:\n"
|
"Environment variables:\n"
|
||||||
"QEMU_STRACE Print system calls and arguments similar to the\n"
|
"QEMU_STRACE Print system calls and arguments similar to the\n"
|
||||||
" 'strace' program. Enable by setting to any value.\n"
|
" 'strace' program. Enable by setting to any value.\n"
|
||||||
|
"You can use -E and -U options to set/unset environment variables\n"
|
||||||
|
"for target process. It is possible to provide several variables\n"
|
||||||
|
"by repeating the option. For example:\n"
|
||||||
|
" -E var1=val2 -E var2=val2 -U LD_PRELOAD -U LD_DEBUG\n"
|
||||||
|
"Note that if you provide several changes to single variable\n"
|
||||||
|
"last change will stay in effect.\n"
|
||||||
,
|
,
|
||||||
TARGET_ARCH,
|
TARGET_ARCH,
|
||||||
interp_prefix,
|
interp_prefix,
|
||||||
@ -2229,8 +2240,8 @@ int main(int argc, char **argv, char **envp)
|
|||||||
int optind;
|
int optind;
|
||||||
const char *r;
|
const char *r;
|
||||||
int gdbstub_port = 0;
|
int gdbstub_port = 0;
|
||||||
int drop_ld_preload = 0, environ_count = 0;
|
char **target_environ, **wrk;
|
||||||
char **target_environ, **wrk, **dst;
|
envlist_t *envlist = NULL;
|
||||||
|
|
||||||
if (argc <= 1)
|
if (argc <= 1)
|
||||||
usage();
|
usage();
|
||||||
@ -2240,6 +2251,16 @@ int main(int argc, char **argv, char **envp)
|
|||||||
/* init debug */
|
/* init debug */
|
||||||
cpu_set_log_filename(DEBUG_LOGFILE);
|
cpu_set_log_filename(DEBUG_LOGFILE);
|
||||||
|
|
||||||
|
if ((envlist = envlist_create()) == NULL) {
|
||||||
|
(void) fprintf(stderr, "Unable to allocate envlist\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add current environment into the list */
|
||||||
|
for (wrk = environ; *wrk != NULL; wrk++) {
|
||||||
|
(void) envlist_setenv(envlist, *wrk);
|
||||||
|
}
|
||||||
|
|
||||||
cpu_model = NULL;
|
cpu_model = NULL;
|
||||||
optind = 1;
|
optind = 1;
|
||||||
for(;;) {
|
for(;;) {
|
||||||
@ -2269,6 +2290,14 @@ int main(int argc, char **argv, char **envp)
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
cpu_set_log(mask);
|
cpu_set_log(mask);
|
||||||
|
} else if (!strcmp(r, "E")) {
|
||||||
|
r = argv[optind++];
|
||||||
|
if (envlist_setenv(envlist, r) != 0)
|
||||||
|
usage();
|
||||||
|
} else if (!strcmp(r, "U")) {
|
||||||
|
r = argv[optind++];
|
||||||
|
if (envlist_unsetenv(envlist, r) != 0)
|
||||||
|
usage();
|
||||||
} else if (!strcmp(r, "s")) {
|
} else if (!strcmp(r, "s")) {
|
||||||
r = argv[optind++];
|
r = argv[optind++];
|
||||||
x86_stack_size = strtol(r, (char **)&r, 0);
|
x86_stack_size = strtol(r, (char **)&r, 0);
|
||||||
@ -2301,7 +2330,7 @@ int main(int argc, char **argv, char **envp)
|
|||||||
_exit(1);
|
_exit(1);
|
||||||
}
|
}
|
||||||
} else if (!strcmp(r, "drop-ld-preload")) {
|
} else if (!strcmp(r, "drop-ld-preload")) {
|
||||||
drop_ld_preload = 1;
|
(void) envlist_unsetenv(envlist, "LD_PRELOAD");
|
||||||
} else if (!strcmp(r, "strace")) {
|
} else if (!strcmp(r, "strace")) {
|
||||||
do_strace = 1;
|
do_strace = 1;
|
||||||
} else
|
} else
|
||||||
@ -2369,19 +2398,8 @@ int main(int argc, char **argv, char **envp)
|
|||||||
do_strace = 1;
|
do_strace = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
wrk = environ;
|
target_environ = envlist_to_environ(envlist, NULL);
|
||||||
while (*(wrk++))
|
envlist_free(envlist);
|
||||||
environ_count++;
|
|
||||||
|
|
||||||
target_environ = malloc((environ_count + 1) * sizeof(char *));
|
|
||||||
if (!target_environ)
|
|
||||||
abort();
|
|
||||||
for (wrk = environ, dst = target_environ; *wrk; wrk++) {
|
|
||||||
if (drop_ld_preload && !strncmp(*wrk, "LD_PRELOAD=", 11))
|
|
||||||
continue;
|
|
||||||
*(dst++) = strdup(*wrk);
|
|
||||||
}
|
|
||||||
*dst = NULL; /* NULL terminate target_environ */
|
|
||||||
|
|
||||||
if (loader_exec(filename, argv+optind, target_environ, regs, info) != 0) {
|
if (loader_exec(filename, argv+optind, target_environ, regs, info) != 0) {
|
||||||
printf("Error loading %s\n", filename);
|
printf("Error loading %s\n", filename);
|
||||||
|
Loading…
Reference in New Issue
Block a user