1 /* copy-acl.c - copy access control list from one file to another file
3 Copyright (C) 2002-2003, 2005-2008 Free Software Foundation, Inc.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
18 Written by Paul Eggert and Andreas Gruenbacher. */
24 #include "acl-internal.h"
27 /* Copy access control lists from one file to another. If SOURCE_DESC is
28 a valid file descriptor, use file descriptor operations, else use
29 filename based operations on SRC_NAME. Likewise for DEST_DESC and
31 If access control lists are not available, fchmod the target file to
32 MODE. Also sets the non-permission bits of the destination file
33 (S_ISUID, S_ISGID, S_ISVTX) to those from MODE if any are set.
34 Return 0 if successful.
35 Return -2 and set errno for an error relating to the source file.
36 Return -1 and set errno for an error relating to the destination file. */
39 qcopy_acl (const char *src_name, int source_desc, const char *dst_name,
40 int dest_desc, mode_t mode)
44 #if USE_ACL && HAVE_ACL_GET_FILE
45 /* POSIX 1003.1e (draft 17 -- abandoned) specific version. */
46 /* Linux, FreeBSD, MacOS X, IRIX, Tru64 */
49 if (HAVE_ACL_GET_FD && source_desc != -1)
50 acl = acl_get_fd (source_desc);
52 acl = acl_get_file (src_name, ACL_TYPE_ACCESS);
55 if (ACL_NOT_WELL_SUPPORTED (errno))
56 return set_acl (dst_name, dest_desc, mode);
61 if (HAVE_ACL_SET_FD && dest_desc != -1)
62 ret = acl_set_fd (dest_desc, acl);
64 ret = acl_set_file (dst_name, ACL_TYPE_ACCESS, acl);
67 int saved_errno = errno;
69 if (ACL_NOT_WELL_SUPPORTED (errno))
71 int nontrivial = acl_access_nontrivial (acl);
77 if (chmod_or_fchmod (dst_name, dest_desc, mode) != 0)
83 chmod_or_fchmod (dst_name, dest_desc, mode);
88 chmod_or_fchmod (dst_name, dest_desc, mode);
96 if (!MODE_INSIDE_ACL || (mode & (S_ISUID | S_ISGID | S_ISVTX)))
98 /* We did not call chmod so far, and either the mode and the ACL are
99 separate or special bits are to be set which don't fit into ACLs. */
101 if (chmod_or_fchmod (dst_name, dest_desc, mode) != 0)
107 acl = acl_get_file (src_name, ACL_TYPE_DEFAULT);
111 if (acl_set_file (dst_name, ACL_TYPE_DEFAULT, acl))
113 int saved_errno = errno;
124 #elif USE_ACL && defined ACL_NO_TRIVIAL
125 /* Solaris 10 NFSv4 ACLs. */
128 ret = (source_desc < 0
129 ? acl_get (src_name, ACL_NO_TRIVIAL, &aclp)
130 : facl_get (source_desc, ACL_NO_TRIVIAL, &aclp));
131 if (ret != 0 && errno != ENOSYS)
134 ret = qset_acl (dst_name, dest_desc, mode);
141 ? acl_set (dst_name, aclp)
142 : facl_set (dest_desc, aclp));
145 int saved_errno = errno;
158 return qset_acl (dst_name, dest_desc, mode);
164 /* Copy access control lists from one file to another. If SOURCE_DESC is
165 a valid file descriptor, use file descriptor operations, else use
166 filename based operations on SRC_NAME. Likewise for DEST_DESC and
168 If access control lists are not available, fchmod the target file to
169 MODE. Also sets the non-permission bits of the destination file
170 (S_ISUID, S_ISGID, S_ISVTX) to those from MODE if any are set.
171 Return 0 if successful, otherwise output a diagnostic and return -1. */
174 copy_acl (const char *src_name, int source_desc, const char *dst_name,
175 int dest_desc, mode_t mode)
177 int ret = qcopy_acl (src_name, source_desc, dst_name, dest_desc, mode);
181 error (0, errno, "%s", quote (src_name));
185 error (0, errno, _("preserving permissions for %s"), quote (dst_name));