| /* vi: set ts=4 : |
| * |
| * bbsh - busybox shell |
| * |
| * Copyright 2006 Rob Landley <rob@landley.net> |
| * |
| * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. |
| */ |
| |
| // A section of code that gets repeatedly or conditionally executed is stored |
| // as a string and parsed each time it's run. |
| |
| |
| |
| // Wheee, debugging. |
| |
| // Terminal control |
| #define ENABLE_BBSH_TTY 0 |
| |
| // &, fg, bg, jobs. (ctrl-z with tty.) |
| #define ENABLE_BBSH_JOBCTL 0 |
| |
| // Flow control (if, while, for, functions { }) |
| #define ENABLE_BBSH_FLOWCTL 0 |
| |
| #define ENABLE_BBSH_ENVVARS 0 // Environment variable support |
| |
| // Local and synthetic variables, fancy prompts, set, $?, etc. |
| #define ENABLE_BBSH_LOCALVARS 0 |
| |
| // Pipes and redirects: | > < >> << && || & () ; |
| #define ENABLE_BBSH_PIPES 0 |
| |
| /* Fun: |
| |
| echo `echo hello#comment " woot` and more |
| */ |
| |
| #include <busybox.h> |
| |
| // A single executable, its arguments, and other information we know about it. |
| #define BBSH_FLAG_EXIT 1 |
| #define BBSH_FLAG_SUSPEND 2 |
| #define BBSH_FLAG_PIPE 4 |
| #define BBSH_FLAG_AND 8 |
| #define BBSH_FLAG_OR 16 |
| #define BBSH_FLAG_AMP 32 |
| #define BBSH_FLAG_SEMI 64 |
| #define BBSH_FLAG_PAREN 128 |
| |
| // What we know about a single process. |
| struct command { |
| struct command *next; |
| int flags; // exit, suspend, && || |
| int pid; // pid (or exit code) |
| int argc; |
| char *argv[0]; |
| }; |
| |
| // A collection of processes piped into/waiting on each other. |
| struct pipeline { |
| struct pipeline *next; |
| int job_id; |
| struct command *cmd; |
| char *cmdline; |
| int cmdlinelen; |
| }; |
| |
| static void free_list(void *list, void (*freeit)(void *data)) |
| { |
| while(list) { |
| void **next = (void **)list; |
| void *list_next = *next; |
| freeit(list); |
| free(list); |
| list = list_next; |
| } |
| } |
| |
| // Parse one word from the command line, appending one or more argv[] entries |
| // to struct command. Handles environment variable substitution and |
| // substrings. Returns pointer to next used byte, or NULL if it |
| // hit an ending token. |
| static char *parse_word(char *start, struct command **cmd) |
| { |
| char *end; |
| |
| // Detect end of line (and truncate line at comment) |
| if (ENABLE_BBSH_PIPES && strchr("><&|(;", *start)) return 0; |
| |
| // Grab next word. (Add dequote and envvar logic here) |
| end = start; |
| while (*end && !isspace(*end)) end++; |
| (*cmd)->argv[(*cmd)->argc++] = xstrndup(start, end-start); |
| |
| // Allocate more space if there's no room for NULL terminator. |
| |
| if (!((*cmd)->argc & 7)) |
| *cmd = xrealloc(*cmd, |
| sizeof(struct command) + ((*cmd)->argc+8)*sizeof(char *)); |
| (*cmd)->argv[(*cmd)->argc] = 0; |
| return end; |
| } |
| |
| // Parse a line of text into a pipeline. |
| // Returns a pointer to the next line. |
| |
| static char *parse_pipeline(char *cmdline, struct pipeline *line) |
| { |
| struct command **cmd = &(line->cmd); |
| char *start = line->cmdline = cmdline; |
| |
| if (!cmdline) return 0; |
| |
| if (ENABLE_BBSH_JOBCTL) line->cmdline = cmdline; |
| |
| // Parse command into argv[] |
| for (;;) { |
| char *end; |
| |
| // Skip leading whitespace and detect end of line. |
| while (isspace(*start)) start++; |
| if (!*start || *start=='#') { |
| if (ENABLE_BBSH_JOBCTL) line->cmdlinelen = start-cmdline; |
| return 0; |
| } |
| |
| // Allocate next command structure if necessary |
| if (!*cmd) *cmd = xzalloc(sizeof(struct command)+8*sizeof(char *)); |
| |
| // Parse next argument and add the results to argv[] |
| end = parse_word(start, cmd); |
| |
| // If we hit the end of this command, how did it end? |
| if (!end) { |
| if (ENABLE_BBSH_PIPES && *start) { |
| if (*start==';') { |
| start++; |
| break; |
| } |
| // handle | & < > >> << || && |
| } |
| break; |
| } |
| start = end; |
| } |
| |
| if (ENABLE_BBSH_JOBCTL) line->cmdlinelen = start-cmdline; |
| |
| return start; |
| } |
| |
| // Execute the commands in a pipeline |
| static int run_pipeline(struct pipeline *line) |
| { |
| struct command *cmd = line->cmd; |
| if (!cmd || !cmd->argc) return 0; |
| |
| // Handle local commands. This is totally fake and plastic. |
| if (cmd->argc==2 && !strcmp(cmd->argv[0],"cd")) |
| chdir(cmd->argv[1]); |
| else if(!strcmp(cmd->argv[0],"exit")) |
| exit(cmd->argc>1 ? atoi(cmd->argv[1]) : 0); |
| else { |
| int status; |
| pid_t pid=fork(); |
| if(!pid) { |
| run_applet_by_name(cmd->argv[0],cmd->argc,cmd->argv); |
| execvp(cmd->argv[0],cmd->argv); |
| printf("No %s",cmd->argv[0]); |
| exit(1); |
| } else waitpid(pid, &status, 0); |
| } |
| |
| return 0; |
| } |
| |
| static void free_cmd(void *data) |
| { |
| struct command *cmd=(struct command *)data; |
| |
| while(cmd->argc) free(cmd->argv[--cmd->argc]); |
| } |
| |
| |
| static void handle(char *command) |
| { |
| struct pipeline line; |
| char *start = command; |
| |
| for (;;) { |
| memset(&line,0,sizeof(struct pipeline)); |
| start = parse_pipeline(start, &line); |
| if (!line.cmd) break; |
| |
| run_pipeline(&line); |
| free_list(line.cmd, free_cmd); |
| } |
| } |
| |
| int bbsh_main(int argc, char *argv[]) |
| { |
| char *command=NULL; |
| FILE *f; |
| |
| getopt32(argc, argv, "c:", &command); |
| |
| f = argv[optind] ? xfopen(argv[optind],"r") : NULL; |
| if (command) handle(command); |
| else { |
| unsigned cmdlen=0; |
| for (;;) { |
| if(!f) putchar('$'); |
| if(1 > getline(&command, &cmdlen,f ? : stdin)) break; |
| |
| handle(command); |
| } |
| if (ENABLE_FEATURE_CLEAN_UP) free(command); |
| } |
| |
| return 1; |
| } |