X-Git-Url: https://mattmccutchen.net/rsync/rsync.git/blobdiff_plain/81b096feef3773dfd17e384ffb9328287cdcea94..4fd842f98df4970f6e49a0365dbed60774e56c09:/chmod.c diff --git a/chmod.c b/chmod.c index 8a6adfd8..2015f5d0 100644 --- a/chmod.c +++ b/chmod.c @@ -1,6 +1,25 @@ +/* + * Implement the core of the --chmod option. + * + * Copyright (C) 2002 Scott Howard + * Copyright (C) 2005-2007 Wayne Davison + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 3 as + * published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along + * with this program; if not, visit the http://fsf.org website. + */ + #include "rsync.h" -extern int orig_umask; +extern mode_t orig_umask; #define FLAG_X_KEEP (1<<0) #define FLAG_DIRS_ONLY (1<<1) @@ -21,9 +40,10 @@ struct chmod_mode_struct { #define STATE_2ND_HALF 2 /* Parse a chmod-style argument, and break it down into one or more AND/OR - * pairs in a linked list. We use a state machine to walk through the - * options. */ -int parse_chmod(const char *modestr, struct chmod_mode_struct **root_mode_ptr) + * pairs in a linked list. We return a pointer to new items on succcess + * (appending the items to the specified list), or NULL on error. */ +struct chmod_mode_struct *parse_chmod(const char *modestr, + struct chmod_mode_struct **root_mode_ptr) { int state = STATE_1ST_HALF; int where = 0, what = 0, op = 0, topbits = 0, topoct = 0, flags = 0; @@ -55,15 +75,15 @@ int parse_chmod(const char *modestr, struct chmod_mode_struct **root_mode_ptr) switch (op) { case CHMOD_ADD: - curr_mode->ModeAND = 07777; + curr_mode->ModeAND = CHMOD_BITS; curr_mode->ModeOR = bits + topoct; break; case CHMOD_SUB: - curr_mode->ModeAND = 07777 - bits - topoct; + curr_mode->ModeAND = CHMOD_BITS - bits - topoct; curr_mode->ModeOR = 0; break; case CHMOD_EQ: - curr_mode->ModeAND = 07777 - (where * 7) - (topoct ? topbits : 0); + curr_mode->ModeAND = CHMOD_BITS - (where * 7) - (topoct ? topbits : 0); curr_mode->ModeOR = bits + topoct; break; } @@ -153,7 +173,7 @@ int parse_chmod(const char *modestr, struct chmod_mode_struct **root_mode_ptr) if (state == STATE_ERROR) { free_chmod_mode(first_mode); - return 0; + return NULL; } if (!(curr_mode = *root_mode_ptr)) @@ -164,7 +184,7 @@ int parse_chmod(const char *modestr, struct chmod_mode_struct **root_mode_ptr) curr_mode->next = first_mode; } - return 1; + return first_mode; } @@ -173,7 +193,7 @@ int parse_chmod(const char *modestr, struct chmod_mode_struct **root_mode_ptr) int tweak_mode(int mode, struct chmod_mode_struct *chmod_modes) { int IsX = mode & 0111; - int NonPerm = mode & ~07777; + int NonPerm = mode & ~CHMOD_BITS; for ( ; chmod_modes; chmod_modes = chmod_modes->next) { if ((chmod_modes->flags & FLAG_DIRS_ONLY) && !S_ISDIR(NonPerm))