summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/chown.c32
1 files changed, 30 insertions, 2 deletions
diff --git a/src/chown.c b/src/chown.c
index fae399db0..956cd8943 100644
--- a/src/chown.c
+++ b/src/chown.c
@@ -39,6 +39,7 @@
#include "fts_.h"
#include "lchown.h"
#include "quote.h"
+#include "root-dev-ino.h"
#include "userspec.h"
/* The official name of this program (e.g., no `g' prefix). */
@@ -57,8 +58,10 @@ static char *reference_file;
non-character as a pseudo short option, starting with CHAR_MAX + 1. */
enum
{
- FROM_OPTION = CHAR_MAX + 1,
- DEREFERENCE_OPTION,
+ DEREFERENCE_OPTION = CHAR_MAX + 1,
+ FROM_OPTION,
+ NO_PRESERVE_ROOT,
+ PRESERVE_ROOT,
REFERENCE_FILE_OPTION
};
@@ -69,6 +72,8 @@ static struct option const long_options[] =
{"dereference", no_argument, 0, DEREFERENCE_OPTION},
{"from", required_argument, 0, FROM_OPTION},
{"no-dereference", no_argument, 0, 'h'},
+ {"no-preserve-root", no_argument, 0, NO_PRESERVE_ROOT},
+ {"preserve-root", no_argument, 0, PRESERVE_ROOT},
{"quiet", no_argument, 0, 'f'},
{"silent", no_argument, 0, 'f'},
{"reference", required_argument, 0, REFERENCE_FILE_OPTION},
@@ -113,6 +118,10 @@ With --reference, change the owner and group of each FILE to those of RFILE.\n\
is not required for the omitted attribute.\n\
"), stdout);
fputs (_("\
+ --no-preserve-root do not treat `/' specially (the default)\n\
+ --preserve-root fail to operate recursively on `/'\n\
+"), stdout);
+ fputs (_("\
-f, --silent, --quiet suppress most error messages\n\
--reference=RFILE use RFILE's owner and group rather than\n\
the specifying OWNER:GROUP values\n\
@@ -148,6 +157,8 @@ as symbolic.\n\
int
main (int argc, char **argv)
{
+ bool preserve_root = false;
+
uid_t uid = -1; /* Specified uid; -1 if not to be changed. */
gid_t gid = -1; /* Specified gid; -1 if not to be changed. */
@@ -202,6 +213,14 @@ main (int argc, char **argv)
chopt.affect_symlink_referent = true;
break;
+ case NO_PRESERVE_ROOT:
+ preserve_root = false;
+ break;
+
+ case PRESERVE_ROOT:
+ preserve_root = true;
+ break;
+
case REFERENCE_FILE_OPTION:
reference_file = optarg;
break;
@@ -272,6 +291,15 @@ main (int argc, char **argv)
optind++;
}
+ if (chopt.recurse && preserve_root)
+ {
+ static struct dev_ino dev_ino_buf;
+ chopt.root_dev_ino = get_root_dev_ino (&dev_ino_buf);
+ if (chopt.root_dev_ino == NULL)
+ error (EXIT_FAILURE, errno, _("failed to get attributes of %s"),
+ quote ("/"));
+ }
+
fail = chown_files (argv + optind, bit_flags,
uid, gid,
required_uid, required_gid, &chopt);