/* ----------------------------------------------------------------------- * * * Copyright 2008-2011 Gene Cumm - All Rights Reserved * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, Inc., 53 Temple Place Ste 330, * Boston MA 02111-1307, USA; either version 2 of the License, or * (at your option) any later version; incorporated herein by reference. * * ----------------------------------------------------------------------- */ /* * rosh.c * * Read-Only shell; Simple shell system designed for SYSLINUX-derivitives. * Provides minimal commands utilizing the console via stdout/stderr as the * sole output devices. Designed to compile for Linux for testing/debugging. */ /* * ToDos: * prompt: Allow left/right arrow, home/end and more? * commands Break into argv/argc-like array * rosh_cfg: allow -s <file> to change config * rosh_ls(): sorted; then multiple columns * prompt: Possibly honor timeout on initial entry for usage as UI * Also possibly honor totaltimeout */ /*#define DO_DEBUG 1 //*/ /* Uncomment the above line for debugging output; Comment to remove */ /*#define DO_DEBUG2 1 //*/ /* Uncomment the above line for super-debugging output; Must have regular * debugging enabled; Comment to remove. */ #include "rosh.h" #include "version.h" #define APP_LONGNAME "Read-Only Shell" #define APP_NAME "rosh" #define APP_AUTHOR "Gene Cumm" #define APP_YEAR "2010" #define APP_VER "beta-b090" /* Print version information to stdout */ void rosh_version(int vtype) { char env[256]; env[0] = 0; printf("%s v %s; (c) %s %s.\n\tFrom Syslinux %s, %s\n", APP_LONGNAME, APP_VER, APP_YEAR, APP_AUTHOR, VERSION_STR, DATE); switch (vtype) { case 1: rosh_get_env_ver(env, 256); printf("\tRunning on %s\n", env); } } /* Print beta message and if DO_DEBUG/DO_DEBUG2 are active */ void print_beta(void) { puts(rosh_beta_str); ROSH_DEBUG("DO_DEBUG active\n"); ROSH_DEBUG2("DO_DEBUG2 active\n"); } /* Search a string for first non-space (' ') character, starting at ipos * istr input string to parse * ipos input position to start at */ int rosh_search_nonsp(const char *istr, const int ipos) { int curpos; char c; curpos = ipos; c = istr[curpos]; while (c && isspace(c)) c = istr[++curpos]; return curpos; } /* Search a string for space (' '), returning the position of the next space * or the '\0' at end of string * istr input string to parse * ipos input position to start at */ int rosh_search_sp(const char *istr, const int ipos) { int curpos; char c; curpos = ipos; c = istr[curpos]; while (c && !(isspace(c))) c = istr[++curpos]; return curpos; } /* Parse a string for the first non-space string, returning the end position * from src * dest string to contain the first non-space string * src string to parse * ipos Position to start in src */ int rosh_parse_sp_1(char *dest, const char *src, const int ipos) { int bpos, epos; /* beginning and ending position of source string to copy to destination string */ bpos = 0; epos = 0; /* //HERE-error condition checking */ bpos = rosh_search_nonsp(src, ipos); epos = rosh_search_sp(src, bpos); if (epos > bpos) { memcpy(dest, src + bpos, epos - bpos); if (dest[epos - bpos] != 0) dest[epos - bpos] = 0; } else { epos = strlen(src); dest[0] = 0; } return epos; } /* * parse_args1: Try 1 at parsing a string to an argc/argv pair. use free_args1 to free memory malloc'd * * Derived from com32/lib/sys/argv.c:__parse_argv() * Copyright 2004-2009 H. Peter Anvin - All Rights Reserved * Copyright 2009 Intel Corporation; author: H. Peter Anvin */ int parse_args1(char ***iargv, const char *istr) { int argc = 0; const char *p; char *q, *r, *args, **arg; int sp = 1; //, qt = 0; /* Was a space; inside a quote */ /* Scan 1: Length */ /* I could eliminate this if I knew a max length, like strncpy() */ int len = strlen(istr); /* Scan 2: Copy, nullify and make argc */ if (!(args = malloc(len + 1))) goto fail_args; q = args; for (p = istr;; p++) { if (*p <= ' ') { if (!sp) { sp = 1; *q++ = '\0'; } } else { if (sp) { argc++; sp = 0; } *q++ = *p; } if (!*p) break; } q--; /* Point q to final null */ /* Scan 3: Build array of pointers */ if (!(*iargv = malloc((argc + 1) * sizeof(char *)))) goto fail_args_ptr; arg = *iargv; arg[argc] = NULL; /* Nullify the last pointer */ if (*args != '\0') *arg++ = args; for (r = args; r < q ; r++) { if (*r == '\0') { *arg++ = r + 1; } } fail_args: return argc; fail_args_ptr: free(args); return 0; } /* Free argv created by parse_args1() * argv Argument Values */ void free_args1(char ***argv) { char *s; s = **argv; free(*argv); free(s); } /* Convert a string to an argc/argv pair * str String to parse * argv Argument Values * returns Argument Count */ int rosh_str2argv(char ***argv, const char *str) { return parse_args1(argv, str); } /* Free an argv created by rosh_str2argv() * argv Argument Values to free */ void rosh_free_argv(char ***argv) { free_args1(argv); } /* Print the contents of an argc/argv pair * argc Argument Count * argv Argument Values */ void rosh_pr_argv(int argc, char *argv[]) { int i; for (i = 0; i < argc; i++) { printf("%s%s", argv[i], (i < argc)? " " : ""); } puts(""); } /* Print the contents of an argc/argv pair verbosely * argc Argument Count * argv Argument Values */ void rosh_pr_argv_v(int argc, char *argv[]) { int i; for (i = 0; i < argc; i++) { printf("%4d '%s'\n", i, argv[i]); } } /* Reset the getopt() environment */ void rosh_getopt_reset(void) { optind = 0; optopt = 0; } /* Display help * type Help type * cmdstr Command for which help is requested */ void rosh_help(int type, const char *cmdstr) { switch (type) { case 2: if ((cmdstr == NULL) || (strcmp(cmdstr, "") == 0)) { rosh_version(0); puts(rosh_help_str2); } else { switch (cmdstr[0]) { case 'c': puts(rosh_help_cd_str); break; case 'l': puts(rosh_help_ls_str); break; default: printf(rosh_help_str_adv, cmdstr); } } break; case 1: default: if (cmdstr) printf("%s: %s: unknown command\n", APP_NAME, cmdstr); rosh_version(0); puts(rosh_help_str1); } } /* Handle most/all errors * ierrno Input Error number * cmdstr Command being executed to cause error * filestr File/parameter causing error */ void rosh_error(const int ierrno, const char *cmdstr, const char *filestr) { printf("--ERROR: %s '%s': ", cmdstr, filestr); switch (ierrno) { case 0: puts("NO ERROR"); break; case ENOENT: puts("not found"); /* SYSLinux-3.72 COM32 API returns this for a directory or empty file */ ROSH_COM32(" (COM32) could be a directory or empty file\n"); break; case EIO: puts("I/O Error"); break; case EBADF: puts("Bad File Descriptor"); break; case EACCES: puts("Access DENIED"); break; case ENOTDIR: puts("not a directory"); ROSH_COM32(" (COM32) could be directory\n"); break; case EISDIR: puts("IS a directory"); break; case ENOSYS: puts("not implemented"); break; default: printf("returns error; errno=%d\n", ierrno); } } /* rosh_error */ /* Concatenate command line arguments into one string * cmdstr Output command string * cmdlen Length of cmdstr * argc Argument Count * argv Argument Values * barg Beginning Argument */ int rosh_argcat(char *cmdstr, const int cmdlen, const int argc, char *argv[], const int barg) { int i, arglen, curpos; /* index, argument length, current position in cmdstr */ curpos = 0; cmdstr[0] = '\0'; /* Nullify string just to be sure */ for (i = barg; i < argc; i++) { arglen = strlen(argv[i]); /* Theoretically, this should never be met in SYSLINUX */ if ((curpos + arglen) > (cmdlen - 1)) arglen = (cmdlen - 1) - curpos; memcpy(cmdstr + curpos, argv[i], arglen); curpos += arglen; if (curpos >= (cmdlen - 1)) { /* Hopefully, curpos should not be greater than (cmdlen - 1) */ /* Still need a '\0' at the last character */ cmdstr[(cmdlen - 1)] = 0; break; /* Escape out of the for() loop; We can no longer process anything more */ } else { cmdstr[curpos] = ' '; curpos += 1; cmdstr[curpos] = 0; } } /* If there's a ' ' at the end, remove it. This is normal unless the maximum length is met/exceeded. */ if (cmdstr[curpos - 1] == ' ') cmdstr[--curpos] = 0; return curpos; } /* rosh_argcat */ /* * Prints a lot of the data in a struct termios */ /* void rosh_print_tc(struct termios *tio) { printf(" -- termios: "); printf(".c_iflag=%04X ", tio->c_iflag); printf(".c_oflag=%04X ", tio->c_oflag); printf(".c_cflag=%04X ", tio->c_cflag); printf(".c_lflag=%04X ", tio->c_lflag); printf(".c_cc[VTIME]='%d' ", tio->c_cc[VTIME]); printf(".c_cc[VMIN]='%d'", tio->c_cc[VMIN]); printf("\n"); } */ /* * Attempts to get a single key from the console * returns key pressed */ int rosh_getkey(void) { int inc; inc = KEY_NONE; while (inc == KEY_NONE) inc = get_key(stdin, 6000); return inc; } /* rosh_getkey */ /* * Qualifies a filename relative to the working directory * filestr Filename to qualify * pwdstr working directory * returns qualified file name string */ void rosh_qualify_filestr(char *filestr, const char *ifilstr, const char *pwdstr) { int filepos = 0; if ((filestr) && (pwdstr) && (ifilstr)) { if (ifilstr[0] != SEP) { strcpy(filestr, pwdstr); filepos = strlen(pwdstr); if (filestr[filepos - 1] != SEP) filestr[filepos++] = SEP; } strcpy(filestr + filepos, ifilstr); ROSH_DEBUG("--'%s'\n", filestr); } } /* Concatenate multiple files to stdout * argc Argument Count * argv Argument Values */ void rosh_cat(int argc, char *argv[]) { FILE *f; char buf[ROSH_BUF_SZ]; int i, numrd; for (i = 0; i < argc; i++) { printf("--File = '%s'\n", argv[i]); errno = 0; f = fopen(argv[i], "r"); if (f != NULL) { numrd = fread(buf, 1, ROSH_BUF_SZ, f); while (numrd > 0) { fwrite(buf, 1, numrd, stdout); numrd = fread(buf, 1, ROSH_BUF_SZ, f); } fclose(f); } else { rosh_error(errno, "cat", argv[i]); errno = 0; } } } /* rosh_cat */ /* Change PWD (Present Working Directory) * argc Argument count * argv Argument values * ipwdstr Initial PWD */ void rosh_cd(int argc, char *argv[], const char *ipwdstr) { int rv = 0; #ifdef DO_DEBUG char filestr[ROSH_PATH_SZ]; #endif /* DO_DEBUG */ ROSH_DEBUG("CMD: \n"); ROSH_DEBUG_ARGV_V(argc, argv); errno = 0; if (argc == 2) rv = chdir(argv[1]); else if (argc == 1) rv = chdir(ipwdstr); else rosh_help(2, argv[0]); if (rv != 0) { if (argc == 2) rosh_error(errno, "cd", argv[1]); else rosh_error(errno, "cd", ipwdstr); errno = 0; } else { #ifdef DO_DEBUG if (getcwd(filestr, ROSH_PATH_SZ)) ROSH_DEBUG(" %s\n", filestr); #endif /* DO_DEBUG */ } } /* rosh_cd */ /* Print the syslinux config file name */ void rosh_cfg(void) { printf("CFG: '%s'\n", syslinux_config_file()); } /* rosh_cfg */ /* Echo a string back to the screen * cmdstr command string to process */ void rosh_echo(const char *cmdstr) { int bpos = 0; ROSH_DEBUG("CMD: '%s'\n", cmdstr); bpos = rosh_search_nonsp(cmdstr, rosh_search_sp(cmdstr, 0)); if (bpos > 1) { ROSH_DEBUG(" bpos=%d\n", bpos); printf("'%s'\n", cmdstr + bpos); } else { puts(""); } } /* rosh_echo */ /* Process argc/argv to optarr * argc Argument count * argv Argument values * optarr option array to populate */ void rosh_ls_arg_opt(int argc, char *argv[], int optarr[]) { int rv = 0; optarr[0] = -1; optarr[1] = -1; optarr[2] = -1; rosh_getopt_reset(); while (rv != -1) { ROSH_DEBUG2("getopt optind=%d rv=%d\n", optind, rv); rv = getopt(argc, argv, rosh_ls_opt_str); switch (rv) { case 'l': case 0: optarr[0] = 1; break; case 'F': case 1: optarr[1] = 1; break; case 'i': case 2: optarr[2] = 1; break; case '?': case -1: default: ROSH_DEBUG2("getopt optind=%d rv=%d\n", optind, rv); break; } } ROSH_DEBUG2(" end getopt optind=%d rv=%d\n", optind, rv); ROSH_DEBUG2("\tIn rosh_ls_arg_opt() opt[0]=%d\topt[1]=%d\topt[2]=%d\n", optarr[0], optarr[1], optarr[2]); } /* rosh_ls_arg_opt */ /* Retrieve the size of a file argument * filestr directory name of directory entry * de directory entry */ int rosh_ls_de_size(const char *filestr, struct dirent *de) { int de_size; char filestr2[ROSH_PATH_SZ]; int fd2, file2pos; struct stat fdstat; filestr2[0] = 0; file2pos = -1; if (filestr) { file2pos = strlen(filestr); memcpy(filestr2, filestr, file2pos); filestr2[file2pos] = '/'; } strcpy(filestr2 + file2pos + 1, de->d_name); fd2 = open(filestr2, O_RDONLY); fstat(fd2, &fdstat); fd2 = close(fd2); de_size = (int)fdstat.st_size; return de_size; } /* rosh_ls_de_size */ /* Retrieve the size and mode of a file * filestr directory name of directory entry * de directory entry */ int rosh_ls_de_size_mode(const char *filestr, struct dirent *de, mode_t * st_mode) { int de_size; char filestr2[ROSH_PATH_SZ]; int file2pos; struct stat fdstat; int status; filestr2[0] = 0; file2pos = -1; memset(&fdstat, 0, sizeof fdstat); ROSH_DEBUG2("ls:dsm(%s, %s) ", filestr, de->d_name); if (filestr) { /* FIXME: prevent string overflow */ file2pos = strlen(filestr); memcpy(filestr2, filestr, file2pos); if (( filestr2[file2pos - 1] == SEP )) { file2pos--; } else { filestr2[file2pos] = SEP; } } strcpy(filestr2 + file2pos + 1, de->d_name); errno = 0; ROSH_DEBUG2("stat(%s) ", filestr2); status = stat(filestr2, &fdstat); (void)status; ROSH_DEBUG2("\t--stat()=%d\terr=%d\n", status, errno); if (errno) { rosh_error(errno, "ls:szmd.stat", de->d_name); errno = 0; } de_size = (int)fdstat.st_size; *st_mode = fdstat.st_mode; return de_size; } /* rosh_ls_de_size_mode */ /* Returns the Inode number if fdstat contains it * fdstat struct to extract inode from if not COM32, for now */ long rosh_ls_d_ino(struct stat *fdstat) { long de_ino; #ifdef __COM32__ if (fdstat) de_ino = -1; else de_ino = 0; #else /* __COM32__ */ de_ino = fdstat->st_ino; #endif /* __COM32__ */ return de_ino; } /* Convert a d_type to a single char in human readable format * d_type d_type to convert * returns human readable single character; a space if other */ char rosh_d_type2char_human(unsigned char d_type) { char ret; switch (d_type) { case DT_UNKNOWN: ret = 'U'; break; /* Unknown */ case DT_FIFO: ret = 'F'; break; /* FIFO */ case DT_CHR: ret = 'C'; break; /* Char Dev */ case DT_DIR: ret = 'D'; break; /* Directory */ case DT_BLK: ret = 'B'; break; /* Block Dev */ case DT_REG: ret = 'R'; break; /* Regular File */ case DT_LNK: ret = 'L'; break; /* Link, Symbolic */ case DT_SOCK: ret = 'S'; break; /* Socket */ case DT_WHT: ret = 'W'; break; /* UnionFS Whiteout */ default: ret = ' '; } return ret; } /* rosh_d_type2char_human */ /* Convert a d_type to a single char by ls's prefix standards for -l * d_type d_type to convert * returns ls style single character; a space if other */ char rosh_d_type2char_lspre(unsigned char d_type) { char ret; switch (d_type) { case DT_FIFO: ret = 'p'; break; case DT_CHR: ret = 'c'; break; case DT_DIR: ret = 'd'; break; case DT_BLK: ret = 'b'; break; case DT_REG: ret = '-'; break; case DT_LNK: ret = 'l'; break; case DT_SOCK: ret = 's'; break; default: ret = '?'; } return ret; } /* rosh_d_type2char_lspre */ /* Convert a d_type to a single char by ls's classify (-F) suffix standards * d_type d_type to convert * returns ls style single character; a space if other */ char rosh_d_type2char_lssuf(unsigned char d_type) { char ret; switch (d_type) { case DT_FIFO: ret = '|'; break; case DT_DIR: ret = '/'; break; case DT_LNK: ret = '@'; break; case DT_SOCK: ret = '='; break; default: ret = ' '; } return ret; } /* rosh_d_type2char_lssuf */ /* Converts data in the "other" place of st_mode to a ls-style string * st_mode Mode in other to analyze * st_mode_str string to hold converted string */ void rosh_st_mode_am2str(mode_t st_mode, char *st_mode_str) { st_mode_str[0] = ((st_mode & S_IROTH) ? 'r' : '-'); st_mode_str[1] = ((st_mode & S_IWOTH) ? 'w' : '-'); st_mode_str[2] = ((st_mode & S_IXOTH) ? 'x' : '-'); } /* Converts st_mode to an ls-style string * st_mode mode to convert * st_mode_str string to hold converted string */ void rosh_st_mode2str(mode_t st_mode, char *st_mode_str) { st_mode_str[0] = rosh_d_type2char_lspre(IFTODT(st_mode)); rosh_st_mode_am2str((st_mode & S_IRWXU) >> 6, st_mode_str + 1); rosh_st_mode_am2str((st_mode & S_IRWXG) >> 3, st_mode_str + 4); rosh_st_mode_am2str(st_mode & S_IRWXO, st_mode_str + 7); st_mode_str[10] = 0; } /* rosh_st_mode2str */ /* Output a single entry * filestr directory name to list * de directory entry * optarr Array of options */ void rosh_ls_arg_dir_de(const char *filestr, struct dirent *de, const int *optarr) { int de_size; mode_t st_mode; char st_mode_str[11]; st_mode = 0; ROSH_DEBUG2("+"); if (optarr[2] > -1) printf("%10d ", (int)(de->d_ino)); if (optarr[0] > -1) { de_size = rosh_ls_de_size_mode(filestr, de, &st_mode); rosh_st_mode2str(st_mode, st_mode_str); ROSH_DEBUG2("%04X ", st_mode); printf("%s %10d ", st_mode_str, de_size); } ROSH_DEBUG("'"); printf("%s", de->d_name); ROSH_DEBUG("'"); if (optarr[1] > -1) printf("%c", rosh_d_type2char_lssuf(de->d_type)); printf("\n"); } /* rosh_ls_arg_dir_de */ /* Output listing of a regular directory * filestr directory name to list * d the open DIR * optarr Array of options NOTE:This is where I could use qsort */ void rosh_ls_arg_dir(const char *filestr, DIR * d, const int *optarr) { struct dirent *de; int filepos; filepos = 0; errno = 0; while ((de = readdir(d))) { filepos++; rosh_ls_arg_dir_de(filestr, de, optarr); } if (errno) { rosh_error(errno, "ls:arg_dir", filestr); errno = 0; } else { if (filepos == 0) ROSH_DEBUG("0 files found"); } } /* rosh_ls_arg_dir */ /* Simple directory listing for one argument (file/directory) based on * filestr and pwdstr * ifilstr input filename/directory name to list * pwdstr Present Working Directory string * optarr Option Array */ void rosh_ls_arg(const char *filestr, const int *optarr) { struct stat fdstat; int status; // char filestr[ROSH_PATH_SZ]; // int filepos; DIR *d; struct dirent de; /* Initialization; make filestr based on leading character of ifilstr and pwdstr */ // rosh_qualify_filestr(filestr, ifilstr, pwdstr); fdstat.st_mode = 0; fdstat.st_size = 0; ROSH_DEBUG("\topt[0]=%d\topt[1]=%d\topt[2]=%d\n", optarr[0], optarr[1], optarr[2]); /* Now, the real work */ errno = 0; status = stat(filestr, &fdstat); if (status == 0) { if (S_ISDIR(fdstat.st_mode)) { ROSH_DEBUG("PATH '%s' is a directory\n", filestr); if ((d = opendir(filestr))) { rosh_ls_arg_dir(filestr, d, optarr); closedir(d); } else { rosh_error(errno, "ls", filestr); errno = 0; } } else { de.d_ino = rosh_ls_d_ino(&fdstat); de.d_type = (IFTODT(fdstat.st_mode)); strcpy(de.d_name, filestr); if (S_ISREG(fdstat.st_mode)) { ROSH_DEBUG("PATH '%s' is a regular file\n", filestr); } else { ROSH_DEBUG("PATH '%s' is some other file\n", filestr); } rosh_ls_arg_dir_de(NULL, &de, optarr); /* if (ifilstr[0] == SEP) rosh_ls_arg_dir_de(NULL, &de, optarr); else rosh_ls_arg_dir_de(pwdstr, &de, optarr);*/ } } else { rosh_error(errno, "ls", filestr); errno = 0; } return; } /* rosh_ls_arg */ /* Parse options that may be present in the cmdstr * filestr Possible option string to parse * optstr Current options * returns 1 if filestr does not begin with '-' else 0 */ int rosh_ls_parse_opt(const char *filestr, char *optstr) { int ret; if (filestr[0] == '-') { ret = 0; if (optstr) strcat(optstr, filestr + 1); } else { ret = 1; } ROSH_DEBUG("ParseOpt: '%s'\n\topt: '%s'\n\tret: %d\n", filestr, optstr, ret); return ret; } /* rosh_ls_parse_opt */ /* List Directory * argc Argument count * argv Argument values */ void rosh_ls(int argc, char *argv[]) { int optarr[3]; int i; rosh_ls_arg_opt(argc, argv, optarr); ROSH_DEBUG2("In ls()\n"); ROSH_DEBUG2_ARGV_V(argc, argv); #ifdef DO_DEBUG optarr[0] = 2; #endif /* DO_DEBUG */ ROSH_DEBUG2(" argc=%d; optind=%d\n", argc, optind); if (optind >= argc) rosh_ls_arg(".", optarr); for (i = optind; i < argc; i++) { rosh_ls_arg(argv[i], optarr); } } /* rosh_ls */ /* Simple directory listing; calls rosh_ls() * argc Argument count * argv Argument values */ void rosh_dir(int argc, char *argv[]) { ROSH_DEBUG(" dir implemented as ls\n"); rosh_ls(argc, argv); } /* rosh_dir */ /* Page through a buffer string * buf Buffer to page through */ void rosh_more_buf(char *buf, int buflen, int rows, int cols, char *scrbuf) { char *bufp, *bufeol, *bufeol2; /* Pointer to current and next end-of-line position in buffer */ int bufpos, bufcnt; /* current position, count characters */ int inc; int i, numln; /* Index, Number of lines */ int elpl; /* Extra lines per line read */ (void)cols; bufpos = 0; bufp = buf + bufpos; bufeol = bufp; numln = rows - 1; ROSH_DEBUG("--(%d)\n", buflen); while (bufpos < buflen) { for (i = 0; i < numln; i++) { bufeol2 = strchr(bufeol, '\n'); if (bufeol2 == NULL) { bufeol = buf + buflen; i = numln; } else { elpl = ((bufeol2 - bufeol - 1) / cols); if (elpl < 0) elpl = 0; i += elpl; ROSH_DEBUG2(" %d/%d ", elpl, i+1); /* If this will not push too much, use it */ /* but if it's the first line, use it */ /* //HERE: We should probably snip the line off */ if ((i < numln) || (i == elpl)) bufeol = bufeol2 + 1; } } ROSH_DEBUG2("\n"); bufcnt = bufeol - bufp; printf("--(%d/%d @%d)\n", bufcnt, buflen, bufpos); memcpy(scrbuf, bufp, bufcnt); scrbuf[bufcnt] = 0; printf("%s", scrbuf); bufp = bufeol; bufpos += bufcnt; if (bufpos == buflen) break; inc = rosh_getkey(); numln = 1; switch (inc) { case KEY_CTRL('c'): case 'q': case 'Q': bufpos = buflen; break; case ' ': numln = rows - 1; } } } /* rosh_more_buf */ /* Page through a single file using the open file stream * fd File Descriptor */ void rosh_more_fd(int fd, int rows, int cols, char *scrbuf) { struct stat fdstat; char *buf; int bufpos; int numrd; FILE *f; fstat(fd, &fdstat); if (S_ISREG(fdstat.st_mode)) { buf = malloc((int)fdstat.st_size); if (buf != NULL) { f = fdopen(fd, "r"); bufpos = 0; numrd = fread(buf, 1, (int)fdstat.st_size, f); while (numrd > 0) { bufpos += numrd; numrd = fread(buf + bufpos, 1, ((int)fdstat.st_size - bufpos), f); } fclose(f); rosh_more_buf(buf, bufpos, rows, cols, scrbuf); } } else { } } /* rosh_more_fd */ /* Page through a file like the more command * argc Argument Count * argv Argument Values */ void rosh_more(int argc, char *argv[]) { int fd, i; /* char filestr[ROSH_PATH_SZ]; int cmdpos;*/ int rows, cols; char *scrbuf; int ret; ROSH_DEBUG_ARGV_V(argc, argv); ret = getscreensize(1, &rows, &cols); if (ret) { ROSH_DEBUG("getscreensize() fail(%d); fall back\n", ret); ROSH_DEBUG("\tROWS='%d'\tCOLS='%d'\n", rows, cols); /* If either fail, go under normal size, just in case */ if (!rows) rows = 20; if (!cols) cols = 75; } ROSH_DEBUG("\tUSE ROWS='%d'\tCOLS='%d'\n", rows, cols); /* 32 bit align beginning of row and over allocate */ scrbuf = malloc(rows * ((cols+3)&(INT_MAX - 3))); if (!scrbuf) return; if (argc) { /* There is no need to mess up the console if we don't have a file */ rosh_console_raw(); for (i = 0; i < argc; i++) { printf("--File = '%s'\n", argv[i]); errno = 0; fd = open(argv[i], O_RDONLY); if (fd != -1) { rosh_more_fd(fd, rows, cols, scrbuf); close(fd); } else { rosh_error(errno, "more", argv[i]); errno = 0; } } rosh_console_std(); } free(scrbuf); } /* rosh_more */ /* Page a file with rewind * argc Argument Count * argv Argument Values */ void rosh_less(int argc, char *argv[]) { printf(" less implemented as more (for now)\n"); rosh_more(argc, argv); } /* rosh_less */ /* Show PWD */ void rosh_pwd(void) { char pwdstr[ROSH_PATH_SZ]; errno = 0; if (getcwd(pwdstr, ROSH_PATH_SZ)) { printf("%s\n", pwdstr); } else { rosh_error(errno, "pwd", ""); errno = 0; } } /* rosh_pwd */ /* Reboot; use warm reboot if one of certain options set * argc Argument count * argv Argument values */ void rosh_reboot(int argc, char *argv[]) { int rtype = 0; if (argc) { /* For now, just use the first */ switch (argv[0][0]) { case '1': case 's': case 'w': rtype = 1; break; case '-': switch (argv[0][1]) { case '1': case 's': case 'w': rtype = 1; break; } break; } } syslinux_reboot(rtype); } /* rosh_reboot */ /* Run a boot string, calling syslinux_run_command * argc Argument count * argv Argument values */ void rosh_run(int argc, char *argv[]) { char cmdstr[ROSH_CMD_SZ]; int len; len = rosh_argcat(cmdstr, ROSH_CMD_SZ, argc, argv, 0); if (len) { printf("--run: '%s'\n", cmdstr); syslinux_run_command(cmdstr); } else { printf(APP_NAME ":run: No arguments\n"); } } /* rosh_run */ /* Process an argc/argv pair and call handling function * argc Argument count * argv Argument values * ipwdstr Initial Present Working Directory string * returns Whether to exit prompt */ char rosh_command(int argc, char *argv[], const char *ipwdstr) { char do_exit = false; int tlen; tlen = strlen(argv[0]); ROSH_DEBUG_ARGV_V(argc, argv); switch (argv[0][0]) { case 'e': case 'E': case 'q': case 'Q': switch (argv[0][1]) { case 0: case 'x': case 'X': case 'u': case 'U': if ((strncasecmp("exit", argv[0], tlen) == 0) || (strncasecmp("quit", argv[0], tlen) == 0)) do_exit = true; else rosh_help(1, argv[0]); break; case 'c': case 'C': if (strncasecmp("echo", argv[0], tlen) == 0) rosh_pr_argv(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; default: rosh_help(1, argv[0]); } break; case 'c': case 'C': /* run 'cd' 'cat' 'cfg' */ switch (argv[0][1]) { case 'a': case 'A': if (strncasecmp("cat", argv[0], tlen) == 0) rosh_cat(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; case 'd': case 'D': if (strncasecmp("cd", argv[0], tlen) == 0) rosh_cd(argc, argv, ipwdstr); else rosh_help(1, argv[0]); break; case 'f': case 'F': if (strncasecmp("cfg", argv[0], tlen) == 0) rosh_cfg(); else rosh_help(1, argv[0]); break; default: rosh_help(1, argv[0]); } break; case 'd': case 'D': /* run 'dir' */ if (strncasecmp("dir", argv[0], tlen) == 0) rosh_dir(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; case 'h': case 'H': case '?': if ((strncasecmp("help", argv[0], tlen) == 0) || (tlen == 1)) rosh_help(2, argv[1]); else rosh_help(1, NULL); break; case 'l': case 'L': /* run 'ls' 'less' */ switch (argv[0][1]) { case 0: case 's': case 'S': if (strncasecmp("ls", argv[0], tlen) == 0) rosh_ls(argc, argv); else rosh_help(1, argv[0]); break; case 'e': case 'E': if (strncasecmp("less", argv[0], tlen) == 0) rosh_less(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; default: rosh_help(1, argv[0]); } break; case 'm': case 'M': switch (argv[0][1]) { case 'a': case 'A': if (strncasecmp("man", argv[0], tlen) == 0) rosh_help(2, argv[1]); else rosh_help(1, argv[0]); break; case 'o': case 'O': if (strncasecmp("more", argv[0], tlen) == 0) rosh_more(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; default: rosh_help(1, argv[0]); } break; case 'p': case 'P': /* run 'pwd' */ if (strncasecmp("pwd", argv[0], tlen) == 0) rosh_pwd(); else rosh_help(1, argv[0]); break; case 'r': case 'R': /* run 'run' */ switch (argv[0][1]) { case 0: case 'e': case 'E': if (strncasecmp("reboot", argv[0], tlen) == 0) rosh_reboot(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; case 'u': case 'U': if (strncasecmp("run", argv[0], tlen) == 0) rosh_run(argc - 1, &argv[1]); else rosh_help(1, argv[0]); break; default: rosh_help(1, argv[0]); } break; case 'v': case 'V': if (strncasecmp("version", argv[0], tlen) == 0) rosh_version(1); else rosh_help(1, argv[0]); break; case 0: case '\n': break; default: rosh_help(1, argv[0]); } /* switch(argv[0][0]) */ return do_exit; } /* rosh_command */ /* Process the prompt for commands as read from stdin and call rosh_command * to process command line string * icmdstr Initial command line string * returns Exit status */ int rosh_prompt(int iargc, char *iargv[]) { int rv; char cmdstr[ROSH_CMD_SZ]; char ipwdstr[ROSH_PATH_SZ]; char do_exit; char **argv; int argc; rv = 0; do_exit = false; if (!getcwd(ipwdstr, ROSH_PATH_SZ)) strcpy(ipwdstr, "./"); if (iargc > 1) do_exit = rosh_command(iargc - 1, &iargv[1], ipwdstr); while (!(do_exit)) { /* Extra preceeding newline */ printf("\nrosh: "); /* Read a line from console */ if (fgets(cmdstr, ROSH_CMD_SZ, stdin)) { argc = rosh_str2argv(&argv, cmdstr); do_exit = rosh_command(argc, argv, ipwdstr); rosh_free_argv(&argv); } else { do_exit = false; } } return rv; } int main(int argc, char *argv[]) { int rv; /* Initialization */ rv = 0; rosh_console_std(); if (argc == 1) { rosh_version(0); print_beta(); } else { #ifdef DO_DEBUG char cmdstr[ROSH_CMD_SZ]; rosh_argcat(cmdstr, ROSH_CMD_SZ, argc, argv, 1); ROSH_DEBUG("arg='%s'\n", cmdstr); #endif } rv = rosh_prompt(argc, argv); printf("--Exiting '" APP_NAME "'\n"); return rv; }