1 /* chown-core.c -- core functions for changing ownership.
2 Copyright (C) 2000-2015 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>. */
17 /* Extracted from chown.c/chgrp.c and librarified by Jim Meyering. */
21 #include <sys/types.h>
26 #include "chown-core.h"
28 #include "ignore-value.h"
30 #include "root-dev-ino.h"
33 #define FTSENT_IS_DIRECTORY(E) \
34 ((E)->fts_info == FTS_D \
35 || (E)->fts_info == FTS_DC \
36 || (E)->fts_info == FTS_DP \
37 || (E)->fts_info == FTS_DNR)
41 /* we called fchown and close, and both succeeded */
44 /* required_uid and/or required_gid are specified, but don't match */
47 /* SAME_INODE check failed */
50 /* open/fchown isn't needed, isn't safe, or doesn't work due to
51 permissions problems; fall back on chown */
54 /* open, fstat, fchown, or close failed */
59 chopt_init (struct Chown_option
*chopt
)
61 chopt
->verbosity
= V_off
;
62 chopt
->root_dev_ino
= NULL
;
63 chopt
->affect_symlink_referent
= true;
64 chopt
->recurse
= false;
65 chopt
->force_silent
= false;
66 chopt
->user_name
= NULL
;
67 chopt
->group_name
= NULL
;
71 chopt_free (struct Chown_option
*chopt _GL_UNUSED
)
73 /* Deliberately do not free chopt->user_name or ->group_name.
74 They're not always allocated. */
77 /* Convert the numeric group-id, GID, to a string stored in xmalloc'd memory,
78 and return it. If there's no corresponding group name, use the decimal
79 representation of the ID. */
82 gid_to_name (gid_t gid
)
84 char buf
[INT_BUFSIZE_BOUND (intmax_t)];
85 struct group
*grp
= getgrgid (gid
);
86 return xstrdup (grp
? grp
->gr_name
87 : TYPE_SIGNED (gid_t
) ? imaxtostr (gid
, buf
)
88 : umaxtostr (gid
, buf
));
91 /* Convert the numeric user-id, UID, to a string stored in xmalloc'd memory,
92 and return it. If there's no corresponding user name, use the decimal
93 representation of the ID. */
96 uid_to_name (uid_t uid
)
98 char buf
[INT_BUFSIZE_BOUND (intmax_t)];
99 struct passwd
*pwd
= getpwuid (uid
);
100 return xstrdup (pwd
? pwd
->pw_name
101 : TYPE_SIGNED (uid_t
) ? imaxtostr (uid
, buf
)
102 : umaxtostr (uid
, buf
));
105 /* Allocate a string representing USER and GROUP. */
108 user_group_str (char const *user
, char const *group
)
116 spec
= xmalloc (strlen (user
) + 1 + strlen (group
) + 1);
117 stpcpy (stpcpy (stpcpy (spec
, user
), ":"), group
);
121 spec
= xstrdup (user
);
126 spec
= xstrdup (group
);
132 /* Tell the user how/if the user and group of FILE have been changed.
133 If USER is NULL, give the group-oriented messages.
134 CHANGED describes what (if anything) has happened. */
137 describe_change (const char *file
, enum Change_status changed
,
138 char const *old_user
, char const *old_group
,
139 char const *user
, char const *group
)
145 if (changed
== CH_NOT_APPLIED
)
147 printf (_("neither symbolic link %s nor referent has been changed\n"),
152 spec
= user_group_str (user
, group
);
153 old_spec
= user_group_str (user
? old_user
: NULL
, group
? old_group
: NULL
);
158 fmt
= (user
? _("changed ownership of %s from %s to %s\n")
159 : group
? _("changed group of %s from %s to %s\n")
160 : _("no change to ownership of %s\n"));
165 fmt
= (user
? _("failed to change ownership of %s from %s to %s\n")
166 : group
? _("failed to change group of %s from %s to %s\n")
167 : _("failed to change ownership of %s\n"));
171 fmt
= (user
? _("failed to change ownership of %s to %s\n")
172 : group
? _("failed to change group of %s to %s\n")
173 : _("failed to change ownership of %s\n"));
179 case CH_NO_CHANGE_REQUESTED
:
180 fmt
= (user
? _("ownership of %s retained as %s\n")
181 : group
? _("group of %s retained as %s\n")
182 : _("ownership of %s retained\n"));
188 printf (fmt
, quote (file
), old_spec
, spec
);
194 /* Change the owner and/or group of the FILE to UID and/or GID (safely)
195 only if REQUIRED_UID and REQUIRED_GID match the owner and group IDs
196 of FILE. ORIG_ST must be the result of 'stat'ing FILE.
198 The 'safely' part above means that we can't simply use chown(2),
199 since FILE might be replaced with some other file between the time
200 of the preceding stat/lstat and this chown call. So here we open
201 FILE and do everything else via the resulting file descriptor.
202 We first call fstat and verify that the dev/inode match those from
203 the preceding stat call, and only then, if appropriate (given the
204 required_uid and required_gid constraints) do we call fchown.
206 Return RC_do_ordinary_chown if we can't open FILE, or if FILE is a
207 special file that might have undesirable side effects when opening.
208 In this case the caller can use the less-safe ordinary chown.
210 Return one of the RCH_status values. */
212 static enum RCH_status
213 restricted_chown (int cwd_fd
, char const *file
,
214 struct stat
const *orig_st
,
215 uid_t uid
, gid_t gid
,
216 uid_t required_uid
, gid_t required_gid
)
218 enum RCH_status status
= RC_ok
;
220 int open_flags
= O_NONBLOCK
| O_NOCTTY
;
223 if (required_uid
== (uid_t
) -1 && required_gid
== (gid_t
) -1)
224 return RC_do_ordinary_chown
;
226 if (! S_ISREG (orig_st
->st_mode
))
228 if (S_ISDIR (orig_st
->st_mode
))
229 open_flags
|= O_DIRECTORY
;
231 return RC_do_ordinary_chown
;
234 fd
= openat (cwd_fd
, file
, O_RDONLY
| open_flags
);
236 || (errno
== EACCES
&& S_ISREG (orig_st
->st_mode
)
237 && 0 <= (fd
= openat (cwd_fd
, file
, O_WRONLY
| open_flags
)))))
238 return (errno
== EACCES
? RC_do_ordinary_chown
: RC_error
);
240 if (fstat (fd
, &st
) != 0)
242 else if (! SAME_INODE (*orig_st
, st
))
243 status
= RC_inode_changed
;
244 else if ((required_uid
== (uid_t
) -1 || required_uid
== st
.st_uid
)
245 && (required_gid
== (gid_t
) -1 || required_gid
== st
.st_gid
))
247 if (fchown (fd
, uid
, gid
) == 0)
249 status
= (close (fd
) == 0
259 int saved_errno
= errno
;
265 /* Change the owner and/or group of the file specified by FTS and ENT
266 to UID and/or GID as appropriate.
267 If REQUIRED_UID is not -1, then skip files with any other user ID.
268 If REQUIRED_GID is not -1, then skip files with any other group ID.
269 CHOPT specifies additional options.
270 Return true if successful. */
272 change_file_owner (FTS
*fts
, FTSENT
*ent
,
273 uid_t uid
, gid_t gid
,
274 uid_t required_uid
, gid_t required_gid
,
275 struct Chown_option
const *chopt
)
277 char const *file_full_name
= ent
->fts_path
;
278 char const *file
= ent
->fts_accpath
;
279 struct stat
const *file_stats
;
280 struct stat stat_buf
;
283 bool symlink_changed
= true;
285 switch (ent
->fts_info
)
290 if (ROOT_DEV_INO_CHECK (chopt
->root_dev_ino
, ent
->fts_statp
))
292 /* This happens e.g., with "chown -R --preserve-root 0 /"
293 and with "chown -RH --preserve-root 0 symlink-to-root". */
294 ROOT_DEV_INO_WARN (file_full_name
);
295 /* Tell fts not to traverse into this hierarchy. */
296 fts_set (fts
, ent
, FTS_SKIP
);
297 /* Ensure that we do not process "/" on the second visit. */
298 ignore_value (fts_read (fts
));
306 if (! chopt
->recurse
)
311 /* For a top-level file or directory, this FTS_NS (stat failed)
312 indicator is determined at the time of the initial fts_open call.
313 With programs like chmod, chown, and chgrp, that modify
314 permissions, it is possible that the file in question is
315 accessible when control reaches this point. So, if this is
316 the first time we've seen the FTS_NS for this file, tell
317 fts_read to stat it "again". */
318 if (ent
->fts_level
== 0 && ent
->fts_number
== 0)
321 fts_set (fts
, ent
, FTS_AGAIN
);
324 if (! chopt
->force_silent
)
325 error (0, ent
->fts_errno
, _("cannot access %s"),
326 quote (file_full_name
));
331 if (! chopt
->force_silent
)
332 error (0, ent
->fts_errno
, "%s", quote (file_full_name
));
337 if (! chopt
->force_silent
)
338 error (0, ent
->fts_errno
, _("cannot read directory %s"),
339 quote (file_full_name
));
343 case FTS_DC
: /* directory that causes cycles */
344 if (cycle_warning_required (fts
, ent
))
346 emit_cycle_warning (file_full_name
);
360 else if (required_uid
== (uid_t
) -1 && required_gid
== (gid_t
) -1
361 && chopt
->verbosity
== V_off
362 && ! chopt
->root_dev_ino
363 && ! chopt
->affect_symlink_referent
)
366 file_stats
= ent
->fts_statp
;
370 file_stats
= ent
->fts_statp
;
372 /* If this is a symlink and we're dereferencing them,
373 stat it to get info on the referent. */
374 if (chopt
->affect_symlink_referent
&& S_ISLNK (file_stats
->st_mode
))
376 if (fstatat (fts
->fts_cwd_fd
, file
, &stat_buf
, 0) != 0)
378 if (! chopt
->force_silent
)
379 error (0, errno
, _("cannot dereference %s"),
380 quote (file_full_name
));
384 file_stats
= &stat_buf
;
388 && (required_uid
== (uid_t
) -1
389 || required_uid
== file_stats
->st_uid
)
390 && (required_gid
== (gid_t
) -1
391 || required_gid
== file_stats
->st_gid
));
394 /* This happens when chown -LR --preserve-root encounters a symlink-to-/. */
396 && FTSENT_IS_DIRECTORY (ent
)
397 && ROOT_DEV_INO_CHECK (chopt
->root_dev_ino
, file_stats
))
399 ROOT_DEV_INO_WARN (file_full_name
);
405 if ( ! chopt
->affect_symlink_referent
)
407 ok
= (lchownat (fts
->fts_cwd_fd
, file
, uid
, gid
) == 0);
409 /* Ignore any error due to lack of support; POSIX requires
410 this behavior for top-level symbolic links with -h, and
411 implies that it's required for all symbolic links. */
412 if (!ok
&& errno
== EOPNOTSUPP
)
415 symlink_changed
= false;
420 /* If possible, avoid a race condition with --from=O:G and without the
421 (-h) --no-dereference option. If fts's stat call determined
422 that the uid/gid of FILE matched the --from=O:G-selected
423 owner and group IDs, blindly using chown(2) here could lead
424 chown(1) or chgrp(1) mistakenly to dereference a *symlink*
425 to an arbitrary file that an attacker had moved into the
426 place of FILE during the window between the stat and
427 chown(2) calls. If FILE is a regular file or a directory
428 that can be opened, this race condition can be avoided safely. */
431 = restricted_chown (fts
->fts_cwd_fd
, file
, file_stats
, uid
, gid
,
432 required_uid
, required_gid
);
438 case RC_do_ordinary_chown
:
439 ok
= (chownat (fts
->fts_cwd_fd
, file
, uid
, gid
) == 0);
446 case RC_inode_changed
:
447 /* FIXME: give a diagnostic in this case? */
458 /* On some systems (e.g., GNU/Linux 2.4.x),
459 the chown function resets the 'special' permission bits.
460 Do *not* restore those bits; doing so would open a window in
461 which a malicious user, M, could subvert a chown command run
462 by some other user and operating on files in a directory
463 where M has write access. */
465 if (do_chown
&& !ok
&& ! chopt
->force_silent
)
466 error (0, errno
, (uid
!= (uid_t
) -1
467 ? _("changing ownership of %s")
468 : _("changing group of %s")),
469 quote (file_full_name
));
472 if (chopt
->verbosity
!= V_off
)
475 ((do_chown
&& ok
&& symlink_changed
)
476 && ! ((uid
== (uid_t
) -1 || uid
== file_stats
->st_uid
)
477 && (gid
== (gid_t
) -1 || gid
== file_stats
->st_gid
)));
479 if (changed
|| chopt
->verbosity
== V_high
)
481 enum Change_status ch_status
=
483 : !symlink_changed
? CH_NOT_APPLIED
484 : !changed
? CH_NO_CHANGE_REQUESTED
486 char *old_usr
= file_stats
? uid_to_name (file_stats
->st_uid
) : NULL
;
487 char *old_grp
= file_stats
? gid_to_name (file_stats
->st_gid
) : NULL
;
488 describe_change (file_full_name
, ch_status
,
490 chopt
->user_name
, chopt
->group_name
);
496 if ( ! chopt
->recurse
)
497 fts_set (fts
, ent
, FTS_SKIP
);
502 /* Change the owner and/or group of the specified FILES.
503 BIT_FLAGS specifies how to treat each symlink-to-directory
504 that is encountered during a recursive traversal.
505 CHOPT specifies additional options.
506 If UID is not -1, then change the owner id of each file to UID.
507 If GID is not -1, then change the group id of each file to GID.
508 If REQUIRED_UID and/or REQUIRED_GID is not -1, then change only
509 files with user ID and group ID that match the non-(-1) value(s).
510 Return true if successful. */
512 chown_files (char **files
, int bit_flags
,
513 uid_t uid
, gid_t gid
,
514 uid_t required_uid
, gid_t required_gid
,
515 struct Chown_option
const *chopt
)
519 /* Use lstat and stat only if they're needed. */
520 int stat_flags
= ((required_uid
!= (uid_t
) -1 || required_gid
!= (gid_t
) -1
521 || chopt
->affect_symlink_referent
522 || chopt
->verbosity
!= V_off
)
526 FTS
*fts
= xfts_open (files
, bit_flags
| stat_flags
, NULL
);
532 ent
= fts_read (fts
);
537 /* FIXME: try to give a better message */
538 if (! chopt
->force_silent
)
539 error (0, errno
, _("fts_read failed"));
545 ok
&= change_file_owner (fts
, ent
, uid
, gid
,
546 required_uid
, required_gid
, chopt
);
549 if (fts_close (fts
) != 0)
551 error (0, errno
, _("fts_close failed"));