2016-01-29 20:49:55 +03:00
|
|
|
#include "qemu/osdep.h"
|
2012-12-17 21:20:00 +04:00
|
|
|
#include "qemu/queue.h"
|
|
|
|
#include "qemu/envlist.h"
|
2009-01-30 22:59:17 +03:00
|
|
|
|
|
|
|
struct envlist_entry {
|
|
|
|
const char *ev_var; /* actual env value */
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_ENTRY(envlist_entry) ev_link;
|
2009-01-30 22:59:17 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
struct envlist {
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_HEAD(, envlist_entry) el_entries; /* actual entries */
|
2009-01-30 22:59:17 +03:00
|
|
|
size_t el_count; /* number of entries */
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2017-03-20 20:38:28 +03:00
|
|
|
* Allocates new envlist and returns pointer to it.
|
2009-01-30 22:59:17 +03:00
|
|
|
*/
|
|
|
|
envlist_t *
|
|
|
|
envlist_create(void)
|
|
|
|
{
|
|
|
|
envlist_t *envlist;
|
|
|
|
|
2017-03-20 20:38:28 +03:00
|
|
|
envlist = g_malloc(sizeof(*envlist));
|
2009-01-30 22:59:17 +03:00
|
|
|
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_INIT(&envlist->el_entries);
|
2009-01-30 22:59:17 +03:00
|
|
|
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;
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_REMOVE(entry, ev_link);
|
2009-01-30 22:59:17 +03:00
|
|
|
|
2017-03-20 20:38:28 +03:00
|
|
|
g_free((char *)entry->ev_var);
|
|
|
|
g_free(entry);
|
2009-01-30 22:59:17 +03:00
|
|
|
}
|
2017-03-20 20:38:28 +03:00
|
|
|
g_free(envlist);
|
2009-01-30 22:59:17 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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;
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
if ((envlist == NULL) || (env == NULL))
|
|
|
|
return (EINVAL);
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
/* find out first equals sign in given env */
|
|
|
|
if ((eq_sign = strchr(env, '=')) == NULL)
|
|
|
|
return (EINVAL);
|
|
|
|
envname_len = eq_sign - env + 1;
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
/*
|
|
|
|
* 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;
|
|
|
|
}
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
if (entry != NULL) {
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_REMOVE(entry, ev_link);
|
2017-03-20 20:38:28 +03:00
|
|
|
g_free((char *)entry->ev_var);
|
|
|
|
g_free(entry);
|
2009-01-30 22:59:17 +03:00
|
|
|
} else {
|
|
|
|
envlist->el_count++;
|
|
|
|
}
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2017-03-20 20:38:28 +03:00
|
|
|
entry = g_malloc(sizeof(*entry));
|
|
|
|
entry->ev_var = g_strdup(env);
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_INSERT_HEAD(&envlist->el_entries, entry, ev_link);
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
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;
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
if ((envlist == NULL) || (env == NULL))
|
|
|
|
return (EINVAL);
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
/* env is not allowed to contain '=' */
|
|
|
|
if (strchr(env, '=') != NULL)
|
|
|
|
return (EINVAL);
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
/*
|
|
|
|
* 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) {
|
2009-09-12 11:36:22 +04:00
|
|
|
QLIST_REMOVE(entry, ev_link);
|
2017-03-20 20:38:28 +03:00
|
|
|
g_free((char *)entry->ev_var);
|
|
|
|
g_free(entry);
|
2023-03-15 06:26:49 +03:00
|
|
|
|
2009-01-30 22:59:17 +03:00
|
|
|
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
|
2017-03-20 20:38:28 +03:00
|
|
|
* by calling g_free for each element and the array.
|
|
|
|
* Returned array and given envlist are not related (no common
|
|
|
|
* references).
|
2009-01-30 22:59:17 +03:00
|
|
|
*
|
|
|
|
* If caller provides count pointer, number of items in array is
|
2017-03-20 20:38:28 +03:00
|
|
|
* stored there.
|
2009-01-30 22:59:17 +03:00
|
|
|
*/
|
|
|
|
char **
|
|
|
|
envlist_to_environ(const envlist_t *envlist, size_t *count)
|
|
|
|
{
|
|
|
|
struct envlist_entry *entry;
|
|
|
|
char **env, **penv;
|
|
|
|
|
2022-03-15 17:41:56 +03:00
|
|
|
penv = env = g_new(char *, envlist->el_count + 1);
|
2009-01-30 22:59:17 +03:00
|
|
|
|
|
|
|
for (entry = envlist->el_entries.lh_first; entry != NULL;
|
|
|
|
entry = entry->ev_link.le_next) {
|
2017-03-20 20:38:28 +03:00
|
|
|
*(penv++) = g_strdup(entry->ev_var);
|
2009-01-30 22:59:17 +03:00
|
|
|
}
|
|
|
|
*penv = NULL; /* NULL terminate the list */
|
|
|
|
|
|
|
|
if (count != NULL)
|
|
|
|
*count = envlist->el_count;
|
|
|
|
|
|
|
|
return (env);
|
|
|
|
}
|