123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148 |
- /* vi: set sw=4 ts=4: */
- /*
- * parse_mode implementation for busybox
- *
- * Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org>
- *
- * Licensed under GPLv2 or later, see file LICENSE in this source tree.
- */
- #include "libbb.h"
- /* http://www.opengroup.org/onlinepubs/007904975/utilities/chmod.html */
- /* This function is used from NOFORK applets. It must not allocate anything */
- #define FILEMODEBITS (S_ISUID | S_ISGID | S_ISVTX | S_IRWXU | S_IRWXG | S_IRWXO)
- int FAST_FUNC bb_parse_mode(const char *s, unsigned current_mode)
- {
- /* should be mode_t really, but in all Unixes these constants fit into uint16 */
- static const uint16_t who_mask[] ALIGN2 = {
- S_ISUID | S_ISGID | S_ISVTX | S_IRWXU | S_IRWXG | S_IRWXO, /* a */
- S_ISUID | S_IRWXU, /* u */
- S_ISGID | S_IRWXG, /* g */
- S_IRWXO /* o */
- };
- static const uint16_t perm_mask[] ALIGN2 = {
- S_IRUSR | S_IRGRP | S_IROTH, /* r */
- S_IWUSR | S_IWGRP | S_IWOTH, /* w */
- S_IXUSR | S_IXGRP | S_IXOTH, /* x */
- S_IXUSR | S_IXGRP | S_IXOTH, /* X -- special -- see below */
- S_ISUID | S_ISGID, /* s */
- S_ISVTX /* t */
- };
- static const char who_chars[] ALIGN1 = "augo";
- static const char perm_chars[] ALIGN1 = "rwxXst";
- const char *p;
- mode_t wholist;
- mode_t permlist;
- mode_t new_mode;
- char op;
- if ((unsigned char)(*s - '0') < 8) {
- unsigned long tmp;
- char *e;
- tmp = strtoul(s, &e, 8);
- if (*e || (tmp > 07777U)) { /* Check range and trailing chars. */
- return -1;
- }
- return tmp;
- }
- new_mode = current_mode;
- /* Note: we allow empty clauses, and hence empty modes.
- * We treat an empty mode as no change to perms. */
- while (*s) { /* Process clauses. */
- if (*s == ',') { /* We allow empty clauses. */
- ++s;
- continue;
- }
- /* Get a wholist. */
- wholist = 0;
- WHO_LIST:
- p = who_chars;
- do {
- if (*p == *s) {
- wholist |= who_mask[(int)(p-who_chars)];
- if (!*++s) {
- return -1;
- }
- goto WHO_LIST;
- }
- } while (*++p);
- do { /* Process action list. */
- if ((*s != '+') && (*s != '-')) {
- if (*s != '=') {
- return -1;
- }
- /* Since op is '=', clear all bits corresponding to the
- * wholist, or all file bits if wholist is empty. */
- permlist = ~FILEMODEBITS;
- if (wholist) {
- permlist = ~wholist;
- }
- new_mode &= permlist;
- }
- op = *s++;
- /* Check for permcopy. */
- p = who_chars + 1; /* Skip 'a' entry. */
- do {
- if (*p == *s) {
- int i = 0;
- permlist = who_mask[(int)(p-who_chars)]
- & (S_IRWXU | S_IRWXG | S_IRWXO)
- & new_mode;
- do {
- if (permlist & perm_mask[i]) {
- permlist |= perm_mask[i];
- }
- } while (++i < 3);
- ++s;
- goto GOT_ACTION;
- }
- } while (*++p);
- /* It was not a permcopy, so get a permlist. */
- permlist = 0;
- PERM_LIST:
- p = perm_chars;
- do {
- if (*p == *s) {
- if ((*p != 'X')
- || (new_mode & (S_IFDIR | S_IXUSR | S_IXGRP | S_IXOTH))
- ) {
- permlist |= perm_mask[(int)(p-perm_chars)];
- }
- if (!*++s) {
- break;
- }
- goto PERM_LIST;
- }
- } while (*++p);
- GOT_ACTION:
- if (permlist) { /* The permlist was nonempty. */
- mode_t tmp = wholist;
- if (!wholist) {
- mode_t u_mask = umask(0);
- umask(u_mask);
- tmp = ~u_mask;
- }
- permlist &= tmp;
- if (op == '-') {
- new_mode &= ~permlist;
- } else {
- new_mode |= permlist;
- }
- }
- } while (*s && (*s != ','));
- }
- return new_mode;
- }
|