123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 |
- #include <sys/types.h>
- #include <errno.h>
- #include <pwd.h>
- #include <stdio.h>
- #include <string.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include "tmux.h"
- int cmd_string_getc(const char *, size_t *);
- void cmd_string_ungetc(size_t *);
- void cmd_string_copy(char **, char *, size_t *);
- char *cmd_string_string(const char *, size_t *, char, int);
- char *cmd_string_variable(const char *, size_t *);
- char *cmd_string_expand_tilde(const char *, size_t *);
- int
- cmd_string_getc(const char *s, size_t *p)
- {
- const u_char *ucs = s;
- if (ucs[*p] == '\0')
- return (EOF);
- return (ucs[(*p)++]);
- }
- void
- cmd_string_ungetc(size_t *p)
- {
- (*p)--;
- }
- int
- cmd_string_parse(const char *s, struct cmd_list **cmdlist, const char *file,
- u_int line, char **cause)
- {
- size_t p;
- int ch, i, argc, rval;
- char **argv, *buf, *t;
- const char *whitespace, *equals;
- size_t len;
- argv = NULL;
- argc = 0;
- buf = NULL;
- len = 0;
- *cause = NULL;
- *cmdlist = NULL;
- rval = -1;
- p = 0;
- for (;;) {
- ch = cmd_string_getc(s, &p);
- switch (ch) {
- case '\'':
- if ((t = cmd_string_string(s, &p, '\'', 0)) == NULL)
- goto error;
- cmd_string_copy(&buf, t, &len);
- break;
- case '"':
- if ((t = cmd_string_string(s, &p, '"', 1)) == NULL)
- goto error;
- cmd_string_copy(&buf, t, &len);
- break;
- case '$':
- if ((t = cmd_string_variable(s, &p)) == NULL)
- goto error;
- cmd_string_copy(&buf, t, &len);
- break;
- case '#':
-
- while ((ch = cmd_string_getc(s, &p)) != EOF)
- ;
-
- case EOF:
- case ' ':
- case '\t':
- if (buf != NULL) {
- buf = xrealloc(buf, len + 1);
- buf[len] = '\0';
- argv = xreallocarray(argv, argc + 1,
- sizeof *argv);
- argv[argc++] = buf;
- buf = NULL;
- len = 0;
- }
- if (ch != EOF)
- break;
- while (argc != 0) {
- equals = strchr(argv[0], '=');
- whitespace = argv[0] + strcspn(argv[0], " \t");
- if (equals == NULL || equals > whitespace)
- break;
- environ_put(global_environ, argv[0]);
- argc--;
- memmove(argv, argv + 1, argc * (sizeof *argv));
- }
- if (argc == 0)
- goto out;
- *cmdlist = cmd_list_parse(argc, argv, file, line, cause);
- if (*cmdlist == NULL)
- goto out;
- rval = 0;
- goto out;
- case '~':
- if (buf == NULL) {
- t = cmd_string_expand_tilde(s, &p);
- if (t == NULL)
- goto error;
- cmd_string_copy(&buf, t, &len);
- break;
- }
-
- default:
- if (len >= SIZE_MAX - 2)
- goto error;
- buf = xrealloc(buf, len + 1);
- buf[len++] = ch;
- break;
- }
- }
- error:
- xasprintf(cause, "invalid or unknown command: %s", s);
- out:
- free(buf);
- if (argv != NULL) {
- for (i = 0; i < argc; i++)
- free(argv[i]);
- free(argv);
- }
- return (rval);
- }
- void
- cmd_string_copy(char **dst, char *src, size_t *len)
- {
- size_t srclen;
- srclen = strlen(src);
- *dst = xrealloc(*dst, *len + srclen + 1);
- strlcpy(*dst + *len, src, srclen + 1);
- *len += srclen;
- free(src);
- }
- char *
- cmd_string_string(const char *s, size_t *p, char endch, int esc)
- {
- int ch;
- char *buf, *t;
- size_t len;
- buf = NULL;
- len = 0;
- while ((ch = cmd_string_getc(s, p)) != endch) {
- switch (ch) {
- case EOF:
- goto error;
- case '\\':
- if (!esc)
- break;
- switch (ch = cmd_string_getc(s, p)) {
- case EOF:
- goto error;
- case 'e':
- ch = '\033';
- break;
- case 'r':
- ch = '\r';
- break;
- case 'n':
- ch = '\n';
- break;
- case 't':
- ch = '\t';
- break;
- }
- break;
- case '$':
- if (!esc)
- break;
- if ((t = cmd_string_variable(s, p)) == NULL)
- goto error;
- cmd_string_copy(&buf, t, &len);
- continue;
- }
- if (len >= SIZE_MAX - 2)
- goto error;
- buf = xrealloc(buf, len + 1);
- buf[len++] = ch;
- }
- buf = xrealloc(buf, len + 1);
- buf[len] = '\0';
- return (buf);
- error:
- free(buf);
- return (NULL);
- }
- char *
- cmd_string_variable(const char *s, size_t *p)
- {
- int ch, fch;
- char *buf, *t;
- size_t len;
- struct environ_entry *envent;
- #define cmd_string_first(ch) ((ch) == '_' || \
- ((ch) >= 'a' && (ch) <= 'z') || ((ch) >= 'A' && (ch) <= 'Z'))
- #define cmd_string_other(ch) ((ch) == '_' || \
- ((ch) >= 'a' && (ch) <= 'z') || ((ch) >= 'A' && (ch) <= 'Z') || \
- ((ch) >= '0' && (ch) <= '9'))
- buf = NULL;
- len = 0;
- fch = EOF;
- switch (ch = cmd_string_getc(s, p)) {
- case EOF:
- goto error;
- case '{':
- fch = '{';
- ch = cmd_string_getc(s, p);
- if (!cmd_string_first(ch))
- goto error;
-
- default:
- if (!cmd_string_first(ch)) {
- xasprintf(&t, "$%c", ch);
- return (t);
- }
- buf = xrealloc(buf, len + 1);
- buf[len++] = ch;
- for (;;) {
- ch = cmd_string_getc(s, p);
- if (ch == EOF || !cmd_string_other(ch))
- break;
- else {
- if (len >= SIZE_MAX - 3)
- goto error;
- buf = xrealloc(buf, len + 1);
- buf[len++] = ch;
- }
- }
- }
- if (fch == '{' && ch != '}')
- goto error;
- if (ch != EOF && fch != '{')
- cmd_string_ungetc(p);
- buf = xrealloc(buf, len + 1);
- buf[len] = '\0';
- envent = environ_find(global_environ, buf);
- free(buf);
- if (envent == NULL)
- return (xstrdup(""));
- return (xstrdup(envent->value));
- error:
- free(buf);
- return (NULL);
- }
- char *
- cmd_string_expand_tilde(const char *s, size_t *p)
- {
- struct passwd *pw;
- struct environ_entry *envent;
- char *home, *path, *user, *cp;
- int last;
- home = NULL;
- last = cmd_string_getc(s, p);
- if (last == EOF || last == '/' || last == ' '|| last == '\t') {
- envent = environ_find(global_environ, "HOME");
- if (envent != NULL && *envent->value != '\0')
- home = envent->value;
- else if ((pw = getpwuid(getuid())) != NULL)
- home = pw->pw_dir;
- } else {
- cmd_string_ungetc(p);
- cp = user = xmalloc(strlen(s));
- for (;;) {
- last = cmd_string_getc(s, p);
- if (last == EOF || last == '/' || last == ' '|| last == '\t')
- break;
- *cp++ = last;
- }
- *cp = '\0';
- if ((pw = getpwnam(user)) != NULL)
- home = pw->pw_dir;
- free(user);
- }
- if (home == NULL)
- return (NULL);
- if (last != EOF)
- xasprintf(&path, "%s%c", home, last);
- else
- xasprintf(&path, "%s", home);
- return (path);
- }
|