implement unset semantics as required by POSIX

This commit is contained in:
Mike Frysinger 2009-03-30 06:50:54 +00:00
parent 5f9f1506ad
commit d690f68554
3 changed files with 92 additions and 7 deletions

View File

@ -1000,21 +1000,21 @@ static int set_local_var(char *str, int flg_export)
return 0; return 0;
} }
static void unset_local_var(const char *name) static int unset_local_var(const char *name)
{ {
struct variable *cur; struct variable *cur;
struct variable *prev = prev; /* for gcc */ struct variable *prev = prev; /* for gcc */
int name_len; int name_len;
if (!name) if (!name)
return; return EXIT_SUCCESS;
name_len = strlen(name); name_len = strlen(name);
cur = G.top_var; cur = G.top_var;
while (cur) { while (cur) {
if (strncmp(cur->varstr, name, name_len) == 0 && cur->varstr[name_len] == '=') { if (strncmp(cur->varstr, name, name_len) == 0 && cur->varstr[name_len] == '=') {
if (cur->flg_read_only) { if (cur->flg_read_only) {
bb_error_msg("%s: readonly variable", name); bb_error_msg("%s: readonly variable", name);
return; return EXIT_FAILURE;
} }
/* prev is ok to use here because 1st variable, HUSH_VERSION, /* prev is ok to use here because 1st variable, HUSH_VERSION,
* is ro, and we cannot reach this code on the 1st pass */ * is ro, and we cannot reach this code on the 1st pass */
@ -1024,11 +1024,12 @@ static void unset_local_var(const char *name)
if (!cur->max_len) if (!cur->max_len)
free(cur->varstr); free(cur->varstr);
free(cur); free(cur);
return; return EXIT_SUCCESS;
} }
prev = cur; prev = cur;
cur = cur->next; cur = cur->next;
} }
return EXIT_SUCCESS;
} }
@ -5025,11 +5026,40 @@ static int builtin_umask(char **argv)
return EXIT_SUCCESS; return EXIT_SUCCESS;
} }
/* http://www.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html#unset */
static int builtin_unset(char **argv) static int builtin_unset(char **argv)
{ {
/* bash always returns true */ size_t i;
unset_local_var(argv[1]); int ret;
bool var = true;
if (!argv[1])
return EXIT_SUCCESS; return EXIT_SUCCESS;
i = 0;
if (argv[1][0] == '-') {
switch (argv[1][1]) {
case 'v': break;
case 'f': if (ENABLE_HUSH_FUNCTIONS) { var = false; break; }
default:
bb_error_msg("unset: %s: invalid option", argv[1]);
return EXIT_FAILURE;
}
++i;
}
ret = EXIT_SUCCESS;
while (argv[++i]) {
if (var) {
if (unset_local_var(argv[i]))
ret = EXIT_FAILURE;
}
#if ENABLE_HUSH_FUNCTIONS
else
unset_local_func(argv[i]);
#endif
}
return ret;
} }
/* http://www.opengroup.org/onlinepubs/9699919799/utilities/wait.html */ /* http://www.opengroup.org/onlinepubs/9699919799/utilities/wait.html */

View File

@ -0,0 +1,19 @@
hush: unset: -: invalid option
1
hush: unset: -m: invalid option
1
0
___
0 f g
0 g
0
___
0 f g
0
0 f g
0
___
hush: HUSH_VERSION: readonly variable
1 f g
hush: HUSH_VERSION: readonly variable
1

View File

@ -0,0 +1,36 @@
# check invalid options are rejected
unset -
echo $?
unset -m a b c
echo $?
# check funky usage
unset
echo $?
# check normal usage
echo ___
f=f g=g
echo $? $f $g
unset f
echo $? $f $g
unset g
echo $? $f $g
echo ___
f=f g=g
echo $? $f $g
unset f g
echo $? $f $g
f=f g=g
echo $? $f $g
unset -v f g
echo $? $f $g
# check read only vars
echo ___
f=f g=g
unset HUSH_VERSION
echo $? $f $g
unset f HUSH_VERSION g
echo $? $f $g