#include <ctype.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <dirent.h>
#include <locale.h>
#include <string.h>
#include <stdarg.h>
#define USER 05700
#define GROUP 02070
#define OTHER 00007
#define ALL 07777
#define READ 00444
#define WRITE 00222
#define EXEC 00111
#define SETID 06000
#define LOCK 02000
#define STICKY 01000
#define GROUP_RWX (GROUP & (READ | WRITE | EXEC))
#define WHO_EMPTY 0
static char *msp;
extern void
errmsg(int severity, int code, char *format, ...);
static int
what(void);
static mode_t
abs(mode_t mode, o_mode_t *group_clear_bits, o_mode_t *group_set_bits),
who(void);
mode_t
newmode_common(char *ms, mode_t new_mode, mode_t umsk, char *file, char *path,
o_mode_t *group_clear_bits, o_mode_t *group_set_bits);
mode_t
newmode(char *ms, mode_t new_mode, mode_t umsk, char *file, char *path)
{
o_mode_t tmp1, tmp2;
return (newmode_common(ms, new_mode, umsk, file, path, &tmp1, &tmp2));
}
mode_t
newmode_common(char *ms, mode_t new_mode, mode_t umsk, char *file, char *path,
o_mode_t *group_clear_bits, o_mode_t *group_set_bits)
{
int oper;
int lcheck;
int scheck;
int xcheck;
int goon;
int operand_empty = 0;
int who_empty;
mode_t who_msk;
mode_t perms_msk;
mode_t old_mode = new_mode;
mode_t grp_change;
msp = ms;
*group_clear_bits = 0;
*group_set_bits = 0;
if (isdigit(*msp))
return (abs(old_mode, group_clear_bits, group_set_bits));
do {
if ((who_msk = who()) == WHO_EMPTY) {
who_empty = 1;
who_msk = ALL;
} else {
who_empty = 0;
}
while (oper = what()) {
operand_empty++;
perms_msk = 0;
goon = 0;
lcheck = scheck = xcheck = 0;
switch (*msp) {
case 'u':
perms_msk = (new_mode & USER) >> 6;
goto dup;
case 'g':
perms_msk = (new_mode & GROUP) >> 3;
goto dup;
case 'o':
perms_msk = (new_mode & OTHER);
dup:
perms_msk &= (READ|WRITE|EXEC);
perms_msk |= (perms_msk << 3) |
(perms_msk << 6);
msp++;
goon = 1;
}
while (goon == 0) {
switch (*msp++) {
case 'r':
perms_msk |= READ;
continue;
case 'w':
perms_msk |= WRITE;
continue;
case 'x':
perms_msk |= EXEC;
xcheck = 1;
continue;
case 'X':
if (((old_mode & S_IFMT) == S_IFDIR) ||
(old_mode & EXEC)) {
perms_msk |= EXEC;
xcheck = 1;
}
continue;
case 'l':
perms_msk |= LOCK;
who_msk |= LOCK;
lcheck = 1;
continue;
case 's':
perms_msk |= SETID;
scheck = 1;
continue;
case 't':
perms_msk |= STICKY;
continue;
default:
msp--;
goon = 1;
}
}
perms_msk &= who_msk;
switch (oper) {
case '+':
if (who_empty) {
perms_msk &= ~umsk;
}
if (xcheck == 1 &&
(perms_msk & GROUP & EXEC) ==
(GROUP & EXEC)) {
if (lcheck == 1 && !S_ISDIR(new_mode)) {
errmsg(1, 3,
gettext("Group execution "
"and locking not permitted "
"together\n"));
}
if (((new_mode & GROUP &
(LOCK | EXEC)) == LOCK) &&
!S_ISDIR(new_mode)) {
errmsg(2, 0,
gettext("%s: Group "
"execution not permitted "
"on a lockable file\n"),
path);
return (old_mode);
}
}
if (scheck == 1 && (perms_msk & GROUP & SETID)
== (GROUP & SETID)) {
if (lcheck == 1 &&
((perms_msk & GROUP & EXEC) ==
(GROUP & EXEC)) &&
!S_ISDIR(new_mode)) {
errmsg(1, 4,
gettext("Set-group-ID and "
"locking not permitted "
"together\n"));
}
if (((new_mode & GROUP &
(LOCK | EXEC)) == LOCK) &&
!S_ISDIR(new_mode)) {
errmsg(2, 0,
gettext("%s: Set-group-ID "
"not permitted on a "
"lockable file\n"), path);
return (old_mode);
}
}
if ((scheck == 1) &&
((new_mode & S_IFMT) != S_IFDIR)) {
if (((new_mode | perms_msk) &
who_msk & EXEC & (USER | GROUP)) !=
(who_msk & EXEC & (USER | GROUP))) {
errmsg(2, 0,
gettext("%s: Execute "
"permission required "
"for set-ID on "
"execution \n"),
path);
return (old_mode);
}
}
if (lcheck == 1) {
if ((new_mode & GROUP & EXEC) ==
(GROUP & EXEC) &&
!S_ISDIR(new_mode)) {
errmsg(2, 0,
gettext("%s: Locking not "
"permitted on "
"a group executable "
"file\n"),
path);
return (old_mode);
}
}
if ((grp_change = (perms_msk & GROUP_RWX) >> 3)
!= 0) {
*group_clear_bits &= ~grp_change;
*group_set_bits |= grp_change;
}
new_mode |= perms_msk;
break;
case '-':
if (who_empty) {
perms_msk &= ~umsk;
}
if (lcheck == 1 && scheck == 0 &&
(new_mode & GROUP & (LOCK | EXEC)) !=
LOCK) {
perms_msk &= ~LOCK;
}
if (scheck == 1 &&
((new_mode & S_IFMT) != S_IFDIR) &&
lcheck == 0 &&
(new_mode & GROUP & (LOCK | EXEC)) ==
LOCK) {
perms_msk &= ~(GROUP & SETID);
}
if (xcheck == 1 && scheck == 0 &&
((who_msk & GROUP) == GROUP ||
(who_msk & USER) == USER) &&
(new_mode & who_msk & (SETID | EXEC)) ==
(who_msk & (SETID | EXEC)) &&
!S_ISDIR(new_mode)) {
errmsg(2, 0,
gettext("%s: Corresponding set-ID "
"also disabled on file since "
"set-ID requires execute "
"permission\n"),
path);
if ((perms_msk & USER & SETID) !=
(USER & SETID) && (new_mode &
USER & (SETID | EXEC)) ==
(who_msk & USER &
(SETID | EXEC))) {
perms_msk |= USER & SETID;
}
if ((perms_msk & GROUP & SETID) !=
(GROUP & SETID) &&
(new_mode & GROUP &
(SETID | EXEC)) ==
(who_msk & GROUP &
(SETID | EXEC))) {
perms_msk |= GROUP & SETID;
}
}
if ((grp_change = (perms_msk & GROUP_RWX) >> 3)
!= 0) {
*group_clear_bits |= grp_change;
*group_set_bits &= ~grp_change;
}
new_mode &= ~perms_msk;
break;
case '=':
if (who_empty) {
perms_msk &= ~umsk;
}
if (lcheck == 1) {
if ((perms_msk & GROUP & EXEC) ==
(GROUP & EXEC) &&
!S_ISDIR(new_mode)) {
errmsg(1, 3,
gettext("Group execution "
"and locking not "
"permitted together\n"));
}
if ((who_msk & GROUP) != GROUP) {
new_mode &= ~(GROUP & EXEC);
}
}
if (scheck == 1 &&
(perms_msk & EXEC & (USER | GROUP)) !=
(who_msk & EXEC & (USER | GROUP)) &&
!S_ISDIR(new_mode)) {
errmsg(1, 2,
gettext("Execute permission "
"required for set-ID on "
"execution\n"));
}
if ((old_mode & S_IFMT) == S_IFDIR)
perms_msk = (perms_msk &
~S_ISGID) | (old_mode & S_ISGID);
if ((grp_change = (perms_msk & GROUP_RWX) >> 3)
!= 0) {
*group_clear_bits = GROUP_RWX >> 3;
*group_set_bits = grp_change;
}
new_mode &= ~who_msk;
new_mode |= perms_msk;
break;
}
}
} while (*msp++ == ',');
if (*--msp || operand_empty == 0) {
errmsg(1, 5, gettext("invalid mode\n"));
}
return (new_mode);
}
mode_t
abs(mode_t mode, o_mode_t *group_clear_bits, o_mode_t *group_set_bits)
{
int c;
mode_t i;
for (i = 0; (c = *msp) >= '0' && c <= '7'; msp++)
i = (mode_t)((i << 3) + (c - '0'));
if (*msp)
errmsg(1, 6, gettext("invalid mode\n"));
*group_clear_bits = GROUP_RWX >> 3;
*group_set_bits = (i & GROUP_RWX) >> 3;
if ((mode & S_IFMT) == S_IFDIR)
return ((i & ~S_ISGID) | (mode & S_ISGID));
return (i);
}
static mode_t
who(void)
{
mode_t m;
m = WHO_EMPTY;
for (; ; msp++) {
switch (*msp) {
case 'u':
m |= USER;
continue;
case 'g':
m |= GROUP;
continue;
case 'o':
m |= OTHER;
continue;
case 'a':
m |= ALL;
continue;
default:
return (m);
}
}
}
static int
what(void)
{
switch (*msp) {
case '+':
case '-':
case '=':
return (*msp++);
}
return (0);
}