diff options
-rw-r--r-- | src/chmod.c | 116 |
1 files changed, 75 insertions, 41 deletions
diff --git a/src/chmod.c b/src/chmod.c index 290c9d17f..d9191a30d 100644 --- a/src/chmod.c +++ b/src/chmod.c @@ -39,6 +39,7 @@ enum Change_status { + CH_NOT_APPLIED, CH_SUCCEEDED, CH_FAILED, CH_NO_CHANGE_REQUESTED @@ -100,19 +101,30 @@ static struct option const long_options[] = {0, 0, 0, 0} }; -static int -mode_changed (const char *file, mode_t old_mode) -{ - struct stat new_stats; +/* Return true if the chmodable permission bits of FILE changed. + The old mode was OLD_MODE, but it was changed to NEW_MODE. */ - if (stat (file, &new_stats)) +static bool +mode_changed (char const *file, mode_t old_mode, mode_t new_mode) +{ + if (new_mode & (S_ISUID | S_ISGID | S_ISVTX)) { - if (force_silent == 0) - error (0, errno, _("getting new attributes of %s"), quote (file)); - return 0; + /* The new mode contains unusual bits that the call to chmod may + have silently cleared. Check whether they actually changed. */ + + struct stat new_stats; + + if (stat (file, &new_stats) != 0) + { + if (!force_silent) + error (0, errno, _("getting new attributes of %s"), quote (file)); + return 0; + } + + new_mode = new_stats.st_mode; } - return old_mode != new_stats.st_mode; + return ((old_mode ^ new_mode) & CHMOD_MODE_BITS) != 0; } /* Tell the user how/if the MODE of FILE has been changed. @@ -125,6 +137,13 @@ describe_change (const char *file, mode_t mode, char perms[11]; /* "-rwxrwxrwx" ls-style modes. */ const char *fmt; + if (changed == CH_NOT_APPLIED) + { + printf (_("neither symbolic link %s nor referent has been changed\n"), + quote (file)); + return; + } + mode_string (mode, perms); perms[10] = '\0'; /* `mode_string' does not null terminate. */ switch (changed) @@ -146,70 +165,85 @@ describe_change (const char *file, mode_t mode, } /* Change the mode of FILE according to the list of operations CHANGES. - Return 0 if successful, 1 if errors occurred. This function is called + Return 0 if successful, -1 if errors occurred. This function is called once for every file system object that fts encounters. */ static int process_file (FTS *fts, FTSENT *ent, const struct mode_change *changes) { - const char *file_full_name = ent->fts_path; - const char *file = ent->fts_accpath; - const struct stat *sb = ent->fts_statp; - mode_t newmode; + char const *file_full_name = ent->fts_path; + char const *file = ent->fts_accpath; + const struct stat *file_stats = ent->fts_statp; + mode_t new_mode IF_LINT (= 0); int errors = 0; - int fail; - int saved_errno; + bool do_chmod; + bool symlink_changed = true; switch (ent->fts_info) { + case FTS_DP: + return 0; + case FTS_NS: error (0, ent->fts_errno, _("cannot access %s"), quote (file_full_name)); - return 1; + errors = -1; + break; case FTS_ERR: error (0, ent->fts_errno, _("%s"), quote (file_full_name)); - return 1; + errors = -1; + break; case FTS_DNR: error (0, ent->fts_errno, _("cannot read directory %s"), quote (file_full_name)); - return 1; + errors = -1; + break; default: break; } - /* If this is the second (post-order) encounter with a directory, - then return right away. */ - if (ent->fts_info == FTS_DP) - return 0; + do_chmod = !errors; - if (ROOT_DEV_INO_CHECK (root_dev_ino, sb)) + if (do_chmod && ROOT_DEV_INO_CHECK (root_dev_ino, file_stats)) { ROOT_DEV_INO_WARN (file_full_name); - return 1; + errors = -1; + do_chmod = false; } - if (S_ISLNK (sb->st_mode)) - return 0; - - newmode = mode_adjust (sb->st_mode, changes); + if (do_chmod) + { + new_mode = mode_adjust (file_stats->st_mode, changes); - fail = chmod (file, newmode); - saved_errno = errno; + if (S_ISLNK (file_stats->st_mode)) + symlink_changed = false; + else + { + errors = chmod (file, new_mode); - if (verbosity == V_high - || (verbosity == V_changes_only - && !fail && mode_changed (file, sb->st_mode))) - describe_change (file_full_name, newmode, - (fail ? CH_FAILED : CH_SUCCEEDED)); + if (errors && !force_silent) + error (0, errno, _("changing permissions of %s"), + quote (file_full_name)); + } + } - if (fail) + if (verbosity != V_off) { - if (force_silent == 0) - error (0, saved_errno, _("changing permissions of %s"), - quote (file_full_name)); - errors = 1; + bool changed = + (!errors && symlink_changed + && mode_changed (file, file_stats->st_mode, new_mode)); + + if (changed || verbosity == V_high) + { + enum Change_status ch_status = + (errors ? CH_FAILED + : !symlink_changed ? CH_NOT_APPLIED + : !changed ? CH_NO_CHANGE_REQUESTED + : CH_SUCCEEDED); + describe_change (file_full_name, new_mode, ch_status); + } } if ( ! recurse) |