Support disabling pipe and redirect support

This commit is contained in:
Eric Andersen 2004-02-10 01:07:45 +00:00
parent a0e4c3f119
commit ff9ad47d79

View File

@ -56,14 +56,18 @@
#include <glob.h> #include <glob.h>
#define expand_t glob_t #define expand_t glob_t
/* Always enable for the moment... */
#define CONFIG_LASH_PIPE_N_REDIRECTS
static const int MAX_READ = 128; /* size of input buffer for `read' builtin */ static const int MAX_READ = 128; /* size of input buffer for `read' builtin */
#define JOB_STATUS_FORMAT "[%d] %-22s %.40s\n" #define JOB_STATUS_FORMAT "[%d] %-22s %.40s\n"
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
enum redir_type { REDIRECT_INPUT, REDIRECT_OVERWRITE, enum redir_type { REDIRECT_INPUT, REDIRECT_OVERWRITE,
REDIRECT_APPEND REDIRECT_APPEND
}; };
#endif
static const unsigned int DEFAULT_CONTEXT=0x1; static const unsigned int DEFAULT_CONTEXT=0x1;
static const unsigned int IF_TRUE_CONTEXT=0x2; static const unsigned int IF_TRUE_CONTEXT=0x2;
@ -71,25 +75,28 @@ static const unsigned int IF_FALSE_CONTEXT=0x4;
static const unsigned int THEN_EXP_CONTEXT=0x8; static const unsigned int THEN_EXP_CONTEXT=0x8;
static const unsigned int ELSE_EXP_CONTEXT=0x10; static const unsigned int ELSE_EXP_CONTEXT=0x10;
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
struct jobset {
struct job *head; /* head of list of running jobs */
struct job *fg; /* current foreground job */
};
struct redir_struct { struct redir_struct {
enum redir_type type; /* type of redirection */ enum redir_type type; /* type of redirection */
int fd; /* file descriptor being redirected */ int fd; /* file descriptor being redirected */
char *filename; /* file to redirect fd to */ char *filename; /* file to redirect fd to */
}; };
#endif
struct child_prog { struct child_prog {
pid_t pid; /* 0 if exited */ pid_t pid; /* 0 if exited */
char **argv; /* program name and arguments */ char **argv; /* program name and arguments */
int num_redirects; /* elements in redirection array */ int num_redirects; /* elements in redirection array */
struct redir_struct *redirects; /* I/O redirects */
int is_stopped; /* is the program currently running? */ int is_stopped; /* is the program currently running? */
struct job *family; /* pointer back to the child's parent job */ struct job *family; /* pointer back to the child's parent job */
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
struct redir_struct *redirects; /* I/O redirects */
#endif
};
struct jobset {
struct job *head; /* head of list of running jobs */
struct job *fg; /* current foreground job */
}; };
struct job { struct job {
@ -514,8 +521,10 @@ static void free_job(struct job *cmd)
for (i = 0; i < cmd->num_progs; i++) { for (i = 0; i < cmd->num_progs; i++) {
free(cmd->progs[i].argv); free(cmd->progs[i].argv);
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
if (cmd->progs[i].redirects) if (cmd->progs[i].redirects)
free(cmd->progs[i].redirects); free(cmd->progs[i].redirects);
#endif
} }
free(cmd->progs); free(cmd->progs);
free(cmd->text); free(cmd->text);
@ -548,7 +557,7 @@ static void remove_job(struct jobset *j_list, struct job *job)
free(job); free(job);
} }
/* Checks to see if any background processes have exited -- if they /* Checks to see if any background processes have exited -- if they
have, figure out why and see if a job has completed */ have, figure out why and see if a job has completed */
static void checkjobs(struct jobset *j_list) static void checkjobs(struct jobset *j_list)
{ {
@ -593,7 +602,7 @@ static void checkjobs(struct jobset *j_list)
printf(JOB_STATUS_FORMAT, job->jobid, "Stopped", printf(JOB_STATUS_FORMAT, job->jobid, "Stopped",
job->text); job->text);
} }
#endif #endif
} }
} }
@ -601,6 +610,7 @@ static void checkjobs(struct jobset *j_list)
bb_perror_msg("waitpid"); bb_perror_msg("waitpid");
} }
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
/* squirrel != NULL means we squirrel away copies of stdin, stdout, /* squirrel != NULL means we squirrel away copies of stdin, stdout,
* and stderr if they are redirected. */ * and stderr if they are redirected. */
static int setup_redirects(struct child_prog *prog, int squirrel[]) static int setup_redirects(struct child_prog *prog, int squirrel[])
@ -656,6 +666,15 @@ static void restore_redirects(int squirrel[])
} }
} }
} }
#else
static inline int setup_redirects(struct child_prog *prog, int squirrel[])
{
return 0;
}
static inline void restore_redirects(int squirrel[])
{
}
#endif
static inline void cmdedit_set_initial_prompt(void) static inline void cmdedit_set_initial_prompt(void)
{ {
@ -665,7 +684,7 @@ static inline void cmdedit_set_initial_prompt(void)
PS1 = getenv("PS1"); PS1 = getenv("PS1");
if(PS1==0) if(PS1==0)
PS1 = "\\w \\$ "; PS1 = "\\w \\$ ";
#endif #endif
} }
static inline void setup_prompt_string(char **prompt_str) static inline void setup_prompt_string(char **prompt_str)
@ -682,7 +701,7 @@ static inline void setup_prompt_string(char **prompt_str)
} }
#else #else
*prompt_str = (shell_context==0)? PS1 : PS2; *prompt_str = (shell_context==0)? PS1 : PS2;
#endif #endif
} }
static int get_command(FILE * source, char *command) static int get_command(FILE * source, char *command)
@ -761,7 +780,7 @@ char * strsep_space( char *string, int * ix)
(*ix)++; (*ix)++;
} }
/* Find the end of any whitespace trailing behind /* Find the end of any whitespace trailing behind
* the token and let that be part of the token */ * the token and let that be part of the token */
while( string && string[*ix] && isspace(string[*ix]) ) { while( string && string[*ix] && isspace(string[*ix]) ) {
(*ix)++; (*ix)++;
@ -774,7 +793,7 @@ char * strsep_space( char *string, int * ix)
token = xmalloc(*ix+1); token = xmalloc(*ix+1);
token[*ix] = '\0'; token[*ix] = '\0';
strncpy(token, string, *ix); strncpy(token, string, *ix);
return token; return token;
} }
@ -789,10 +808,10 @@ static int expand_arguments(char *command)
int flags = GLOB_NOCHECK int flags = GLOB_NOCHECK
#ifdef GLOB_BRACE #ifdef GLOB_BRACE
| GLOB_BRACE | GLOB_BRACE
#endif #endif
#ifdef GLOB_TILDE #ifdef GLOB_TILDE
| GLOB_TILDE | GLOB_TILDE
#endif #endif
; ;
/* get rid of the terminating \n */ /* get rid of the terminating \n */
@ -817,7 +836,7 @@ static int expand_arguments(char *command)
* we write stuff into the original (in a minute) */ * we write stuff into the original (in a minute) */
cmd = cmd_copy = bb_xstrdup(command); cmd = cmd_copy = bb_xstrdup(command);
*command = '\0'; *command = '\0';
for (ix = 0, tmpcmd = cmd; for (ix = 0, tmpcmd = cmd;
(tmpcmd = strsep_space(cmd, &ix)) != NULL; cmd += ix, ix=0) { (tmpcmd = strsep_space(cmd, &ix)) != NULL; cmd += ix, ix=0) {
if (*tmpcmd == '\0') if (*tmpcmd == '\0')
break; break;
@ -832,7 +851,7 @@ static int expand_arguments(char *command)
return FALSE; return FALSE;
} else if (retval != 0) { } else if (retval != 0) {
/* Some other error. GLOB_NOMATCH shouldn't /* Some other error. GLOB_NOMATCH shouldn't
* happen because of the GLOB_NOCHECK flag in * happen because of the GLOB_NOCHECK flag in
* the glob call. */ * the glob call. */
bb_error_msg("syntax error"); bb_error_msg("syntax error");
return FALSE; return FALSE;
@ -856,7 +875,7 @@ static int expand_arguments(char *command)
free(cmd_copy); free(cmd_copy);
trim(command); trim(command);
/* Now do the shell variable substitutions which /* Now do the shell variable substitutions which
* wordexp can't do for us, namely $? and $! */ * wordexp can't do for us, namely $? and $! */
src = command; src = command;
while((dst = strchr(src,'$')) != NULL){ while((dst = strchr(src,'$')) != NULL){
@ -946,21 +965,25 @@ static int expand_arguments(char *command)
/* Return cmd->num_progs as 0 if no command is present (e.g. an empty /* Return cmd->num_progs as 0 if no command is present (e.g. an empty
line). If a valid command is found, command_ptr is set to point to line). If a valid command is found, command_ptr is set to point to
the beginning of the next command (if the original command had more the beginning of the next command (if the original command had more
then one job associated with it) or NULL if no more commands are then one job associated with it) or NULL if no more commands are
present. */ present. */
static int parse_command(char **command_ptr, struct job *job, int *inbg) static int parse_command(char **command_ptr, struct job *job, int *inbg)
{ {
char *command; char *command;
char *return_command = NULL; char *return_command = NULL;
char *src, *buf, *chptr; char *src, *buf;
int argc_l = 0; int argc_l = 0;
int done = 0; int done = 0;
int argv_alloced; int argv_alloced;
int i, saw_quote = 0; int saw_quote = 0;
char quote = '\0'; char quote = '\0';
int count; int count;
struct child_prog *prog; struct child_prog *prog;
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
int i;
char *chptr;
#endif
/* skip leading white space */ /* skip leading white space */
while (**command_ptr && isspace(**command_ptr)) while (**command_ptr && isspace(**command_ptr))
@ -976,21 +999,23 @@ static int parse_command(char **command_ptr, struct job *job, int *inbg)
job->num_progs = 1; job->num_progs = 1;
job->progs = xmalloc(sizeof(*job->progs)); job->progs = xmalloc(sizeof(*job->progs));
/* We set the argv elements to point inside of this string. The /* We set the argv elements to point inside of this string. The
memory is freed by free_job(). Allocate twice the original memory is freed by free_job(). Allocate twice the original
length in case we need to quote every single character. length in case we need to quote every single character.
Getting clean memory relieves us of the task of NULL Getting clean memory relieves us of the task of NULL
terminating things and makes the rest of this look a bit terminating things and makes the rest of this look a bit
cleaner (though it is, admittedly, a tad less efficient) */ cleaner (though it is, admittedly, a tad less efficient) */
job->cmdbuf = command = xcalloc(2*strlen(*command_ptr) + 1, sizeof(char)); job->cmdbuf = command = xcalloc(2*strlen(*command_ptr) + 1, sizeof(char));
job->text = NULL; job->text = NULL;
prog = job->progs; prog = job->progs;
prog->num_redirects = 0; prog->num_redirects = 0;
prog->redirects = NULL;
prog->is_stopped = 0; prog->is_stopped = 0;
prog->family = job; prog->family = job;
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
prog->redirects = NULL;
#endif
argv_alloced = 5; argv_alloced = 5;
prog->argv = xmalloc(sizeof(*prog->argv) * argv_alloced); prog->argv = xmalloc(sizeof(*prog->argv) * argv_alloced);
@ -1046,6 +1071,7 @@ static int parse_command(char **command_ptr, struct job *job, int *inbg)
done = 1; done = 1;
break; break;
#ifdef CONFIG_LASH_PIPE_N_REDIRECTS
case '>': /* redirects */ case '>': /* redirects */
case '<': case '<':
i = prog->num_redirects++; i = prog->num_redirects++;
@ -1143,6 +1169,7 @@ static int parse_command(char **command_ptr, struct job *job, int *inbg)
src--; /* we'll ++ it at the end of the loop */ src--; /* we'll ++ it at the end of the loop */
break; break;
#endif
case '&': /* background */ case '&': /* background */
*inbg = 1; *inbg = 1;
@ -1189,7 +1216,7 @@ static int parse_command(char **command_ptr, struct job *job, int *inbg)
} }
*command_ptr = return_command; *command_ptr = return_command;
return 0; return 0;
} }
@ -1236,7 +1263,7 @@ static int pseudo_exec(struct child_prog *child)
#ifdef CONFIG_FEATURE_SH_APPLETS_ALWAYS_WIN #ifdef CONFIG_FEATURE_SH_APPLETS_ALWAYS_WIN
/* If you enable CONFIG_FEATURE_SH_APPLETS_ALWAYS_WIN, then /* If you enable CONFIG_FEATURE_SH_APPLETS_ALWAYS_WIN, then
* if you run /bin/cat, it will use BusyBox cat even if * if you run /bin/cat, it will use BusyBox cat even if
* /bin/cat exists on the filesystem and is _not_ busybox. * /bin/cat exists on the filesystem and is _not_ busybox.
* Some systems want this, others do not. Choose wisely. :-) * Some systems want this, others do not. Choose wisely. :-)
*/ */
@ -1254,7 +1281,7 @@ static int pseudo_exec(struct child_prog *child)
execvp(child->argv[0], child->argv); execvp(child->argv[0], child->argv);
/* Do not use bb_perror_msg_and_die() here, since we must not /* Do not use bb_perror_msg_and_die() here, since we must not
* call exit() but should call _exit() instead */ * call exit() but should call _exit() instead */
fprintf(stderr, "%s: %m\n", child->argv[0]); fprintf(stderr, "%s: %m\n", child->argv[0]);
_exit(EXIT_FAILURE); _exit(EXIT_FAILURE);
@ -1286,7 +1313,7 @@ static void insert_job(struct job *newjob, int inbg)
thejob->stopped_progs = 0; thejob->stopped_progs = 0;
if (inbg) { if (inbg) {
/* we don't wait for background thejobs to return -- append it /* we don't wait for background thejobs to return -- append it
to the list of backgrounded thejobs and leave it alone */ to the list of backgrounded thejobs and leave it alone */
printf("[%d] %d\n", thejob->jobid, printf("[%d] %d\n", thejob->jobid,
newjob->progs[newjob->num_progs - 1].pid); newjob->progs[newjob->num_progs - 1].pid);
@ -1336,8 +1363,8 @@ static int run_command(struct job *newjob, int inbg, int outpipe[2])
if (newjob->num_progs == 1) { if (newjob->num_progs == 1) {
for (x = bltins; x->cmd; x++) { for (x = bltins; x->cmd; x++) {
if (strcmp(child->argv[0], x->cmd) == 0 ) { if (strcmp(child->argv[0], x->cmd) == 0 ) {
int squirrel[] = {-1, -1, -1};
int rcode; int rcode;
int squirrel[] = {-1, -1, -1};
setup_redirects(child, squirrel); setup_redirects(child, squirrel);
rcode = x->function(child); rcode = x->function(child);
restore_redirects(squirrel); restore_redirects(squirrel);
@ -1347,9 +1374,9 @@ static int run_command(struct job *newjob, int inbg, int outpipe[2])
} }
#if !defined(__UCLIBC__) || defined(__UCLIBC_HAS_MMU__) #if !defined(__UCLIBC__) || defined(__UCLIBC_HAS_MMU__)
if (!(child->pid = fork())) if (!(child->pid = fork()))
#else #else
if (!(child->pid = vfork())) if (!(child->pid = vfork()))
#endif #endif
{ {
/* Set the handling for job control signals back to the default. */ /* Set the handling for job control signals back to the default. */
@ -1394,7 +1421,7 @@ static int run_command(struct job *newjob, int inbg, int outpipe[2])
if (nextout != 1) if (nextout != 1)
close(nextout); close(nextout);
/* If there isn't another process, nextin is garbage /* If there isn't another process, nextin is garbage
but it doesn't matter */ but it doesn't matter */
nextin = pipefds[0]; nextin = pipefds[0];
} }
@ -1446,7 +1473,7 @@ static int busy_loop(FILE * input)
if (!parse_command(&next_command, &newjob, &inbg) && if (!parse_command(&next_command, &newjob, &inbg) &&
newjob.num_progs) { newjob.num_progs) {
int pipefds[2] = {-1,-1}; int pipefds[2] = {-1,-1};
debug_printf( "job=%p fed to run_command by busy_loop()'\n", debug_printf( "job=%p fed to run_command by busy_loop()'\n",
&newjob); &newjob);
run_command(&newjob, inbg, pipefds); run_command(&newjob, inbg, pipefds);
} }
@ -1495,7 +1522,7 @@ static int busy_loop(FILE * input)
/* move the shell to the foreground */ /* move the shell to the foreground */
/* suppress messages when run from /linuxrc mag@sysgo.de */ /* suppress messages when run from /linuxrc mag@sysgo.de */
if (tcsetpgrp(shell_terminal, getpgrp()) && errno != ENOTTY) if (tcsetpgrp(shell_terminal, getpgrp()) && errno != ENOTTY)
bb_perror_msg("tcsetpgrp"); bb_perror_msg("tcsetpgrp");
} }
} }
} }
@ -1508,7 +1535,7 @@ static int busy_loop(FILE * input)
/* return exit status if called with "-c" */ /* return exit status if called with "-c" */
if (input == NULL && WIFEXITED(status)) if (input == NULL && WIFEXITED(status))
return WEXITSTATUS(status); return WEXITSTATUS(status);
return 0; return 0;
} }
@ -1534,7 +1561,7 @@ void free_memory(void)
static void setup_job_control(void) static void setup_job_control(void)
{ {
int status; int status;
/* Loop until we are in the foreground. */ /* Loop until we are in the foreground. */
while ((status = tcgetpgrp (shell_terminal)) >= 0) { while ((status = tcgetpgrp (shell_terminal)) >= 0) {
if (status == (shell_pgrp = getpgrp ())) { if (status == (shell_pgrp = getpgrp ())) {
@ -1580,7 +1607,7 @@ int lash_main(int argc_l, char **argv_l)
prof_input = fopen("/etc/profile", "r"); prof_input = fopen("/etc/profile", "r");
if (prof_input) { if (prof_input) {
int tmp_fd = fileno(prof_input); int tmp_fd = fileno(prof_input);
mark_open(tmp_fd); mark_open(tmp_fd);
/* Now run the file */ /* Now run the file */
busy_loop(prof_input); busy_loop(prof_input);
fclose(prof_input); fclose(prof_input);
@ -1644,6 +1671,6 @@ int lash_main(int argc_l, char **argv_l)
#else #else
PS1 = NULL; PS1 = NULL;
#endif #endif
return (busy_loop(input)); return (busy_loop(input));
} }