summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/chmod.c63
1 files changed, 36 insertions, 27 deletions
diff --git a/src/chmod.c b/src/chmod.c
index 4bf9f53c9..569fb8298 100644
--- a/src/chmod.c
+++ b/src/chmod.c
@@ -60,6 +60,12 @@ enum Verbosity
/* The name the program was run with. */
char *program_name;
+/* The desired change to the mode. */
+static struct mode_change *change;
+
+/* The initial umask value, if it might be needed. */
+static mode_t umask_value;
+
/* If true, change the modes of directories recursively. */
static bool recurse;
@@ -69,10 +75,6 @@ static bool force_silent;
/* Level of verbosity. */
static enum Verbosity verbosity = V_off;
-/* The argument to the --reference option. Use the owner and group IDs
- of this file. This file must exist. */
-static char *reference_file;
-
/* Pointer to the device and inode numbers of `/', when --recursive.
Otherwise NULL. */
static struct dev_ino *root_dev_ino;
@@ -164,12 +166,12 @@ describe_change (const char *file, mode_t mode,
(unsigned long int) (mode & CHMOD_MODE_BITS), &perms[1]);
}
-/* Change the mode of FILE according to the list of operations CHANGES.
+/* Change the mode of FILE.
Return true if successful. This function is called
once for every file system object that fts encounters. */
static bool
-process_file (FTS *fts, FTSENT *ent, const struct mode_change *changes)
+process_file (FTS *fts, FTSENT *ent)
{
char const *file_full_name = ent->fts_path;
char const *file = ent->fts_accpath;
@@ -215,7 +217,7 @@ process_file (FTS *fts, FTSENT *ent, const struct mode_change *changes)
if (do_chmod)
{
- new_mode = mode_adjust (file_stats->st_mode, changes);
+ new_mode = mode_adjust (file_stats->st_mode, change, umask_value);
if (S_ISLNK (file_stats->st_mode))
symlink_changed = false;
@@ -253,12 +255,11 @@ process_file (FTS *fts, FTSENT *ent, const struct mode_change *changes)
}
/* Recursively change the modes of the specified FILES (the last entry
- of which is NULL) according to the list of operations CHANGES.
- BIT_FLAGS controls how fts works.
+ of which is NULL). BIT_FLAGS controls how fts works.
Return true if successful. */
static bool
-process_files (char **files, int bit_flags, const struct mode_change *changes)
+process_files (char **files, int bit_flags)
{
bool ok = true;
@@ -280,7 +281,7 @@ process_files (char **files, int bit_flags, const struct mode_change *changes)
break;
}
- ok &= process_file (fts, ent, changes);
+ ok &= process_file (fts, ent);
}
/* Ignore failure, since the only way it can do so is in failing to
@@ -324,8 +325,7 @@ Change the mode of each FILE to MODE.\n\
fputs (VERSION_OPTION_DESCRIPTION, stdout);
fputs (_("\
\n\
-Each MODE is one or more of the letters ugoa, one of the symbols +-= and\n\
-one or more of the letters rwxXstugo.\n\
+Each MODE is of the form `[ugoa]*([-+=]([rwxXst]*|[ugo]))+'.\n\
"), stdout);
printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
}
@@ -338,12 +338,12 @@ one or more of the letters rwxXstugo.\n\
int
main (int argc, char **argv)
{
- struct mode_change *changes;
char *mode = NULL;
size_t mode_len = 0;
size_t mode_alloc = 0;
bool ok;
bool preserve_root = false;
+ char const *reference_file = NULL;
int c;
initialize_main (&argc, &argv);
@@ -428,8 +428,10 @@ main (int argc, char **argv)
if (reference_file)
{
if (mode)
- error (EXIT_FAILURE, 0,
- _("cannot combine mode and --reference options"));
+ {
+ error (0, 0, _("cannot combine mode and --reference options"));
+ usage (EXIT_FAILURE);
+ }
}
else
{
@@ -446,16 +448,23 @@ main (int argc, char **argv)
usage (EXIT_FAILURE);
}
- changes = (reference_file ? mode_create_from_ref (reference_file)
- : mode_compile (mode, MODE_MASK_ALL));
-
- if (changes == MODE_INVALID)
- error (EXIT_FAILURE, 0, _("invalid mode: %s"), quote (mode));
- else if (changes == MODE_MEMORY_EXHAUSTED)
- xalloc_die ();
- else if (changes == MODE_BAD_REFERENCE)
- error (EXIT_FAILURE, errno, _("failed to get attributes of %s"),
- quote (reference_file));
+ if (reference_file)
+ {
+ change = mode_create_from_ref (reference_file);
+ if (!change)
+ error (EXIT_FAILURE, errno, _("failed to get attributes of %s"),
+ quote (reference_file));
+ }
+ else
+ {
+ change = mode_compile (mode);
+ if (!change)
+ {
+ error (0, 0, _("invalid mode: %s"), quote (mode));
+ usage (EXIT_FAILURE);
+ }
+ umask_value = umask (0);
+ }
if (recurse & preserve_root)
{
@@ -470,7 +479,7 @@ main (int argc, char **argv)
root_dev_ino = NULL;
}
- ok = process_files (argv + optind, FTS_COMFOLLOW, changes);
+ ok = process_files (argv + optind, FTS_COMFOLLOW);
exit (ok ? EXIT_SUCCESS : EXIT_FAILURE);
}