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"
26 /* Copy access control lists from one file to another. If SOURCE_DESC is
27 a valid file descriptor, use file descriptor operations, else use
28 filename based operations on SRC_NAME. Likewise for DEST_DESC and
30 If access control lists are not available, fchmod the target file to
31 MODE. Also sets the non-permission bits of the destination file
32 (S_ISUID, S_ISGID, S_ISVTX) to those from MODE if any are set.
33 Return 0 if successful, otherwise output a diagnostic and return -1. */
36 copy_acl (const char *src_name, int source_desc, const char *dst_name,
37 int dest_desc, mode_t mode)
41 #if USE_ACL && HAVE_ACL_GET_FILE && HAVE_ACL_SET_FILE && HAVE_ACL_FREE
42 /* POSIX 1003.1e (draft 17 -- abandoned) specific version. */
45 if (HAVE_ACL_GET_FD && source_desc != -1)
46 acl = acl_get_fd (source_desc);
48 acl = acl_get_file (src_name, ACL_TYPE_ACCESS);
51 if (ACL_NOT_WELL_SUPPORTED (errno))
52 return set_acl (dst_name, dest_desc, mode);
55 error (0, errno, "%s", quote (src_name));
60 if (HAVE_ACL_SET_FD && dest_desc != -1)
61 ret = acl_set_fd (dest_desc, acl);
63 ret = acl_set_file (dst_name, ACL_TYPE_ACCESS, acl);
66 int saved_errno = errno;
68 if (ACL_NOT_WELL_SUPPORTED (errno))
70 int n = acl_entries (acl);
73 /* On most hosts an ACL is trivial if n == 3, and it cannot be
74 less than 3. On IRIX 6.5 it is also trivial if n == -1.
75 For simplicity and safety, assume the ACL is trivial if n <= 3.
76 Also see file-has-acl.c for some of the other possibilities;
77 it's not clear whether that complexity is needed here. */
80 if (chmod_or_fchmod (dst_name, dest_desc, mode) != 0)
86 chmod_or_fchmod (dst_name, dest_desc, mode);
91 chmod_or_fchmod (dst_name, dest_desc, mode);
93 error (0, saved_errno, _("preserving permissions for %s"),
100 if (mode & (S_ISUID | S_ISGID | S_ISVTX))
102 /* We did not call chmod so far, so the special bits have not yet
105 if (chmod_or_fchmod (dst_name, dest_desc, mode) != 0)
107 error (0, errno, _("preserving permissions for %s"),
115 acl = acl_get_file (src_name, ACL_TYPE_DEFAULT);
118 error (0, errno, "%s", quote (src_name));
122 if (acl_set_file (dst_name, ACL_TYPE_DEFAULT, acl))
124 error (0, errno, _("preserving permissions for %s"),
136 # if USE_ACL && defined ACL_NO_TRIVIAL
137 /* Solaris 10 NFSv4 ACLs. */
139 ret = (source_desc < 0
140 ? acl_get (src_name, ACL_NO_TRIVIAL, &aclp)
141 : facl_get (source_desc, ACL_NO_TRIVIAL, &aclp));
142 if (ret != 0 && errno != ENOSYS)
144 error (0, errno, "%s", quote (src_name));
149 ret = qset_acl (dst_name, dest_desc, mode);
151 error (0, errno, _("preserving permissions for %s"), quote (dst_name));
153 # if USE_ACL && defined ACL_NO_TRIVIAL
154 if (ret == 0 && aclp)
157 ? acl_set (dst_name, aclp)
158 : facl_set (dest_desc, aclp));
160 error (0, errno, _("preserving permissions for %s"), quote (dst_name));