summaryrefslogtreecommitdiff
path: root/lib/copy-acl.c
diff options
context:
space:
mode:
authorBruno Haible <bruno@clisp.org>2008-06-02 13:03:21 +0200
committerBruno Haible <bruno@clisp.org>2008-06-02 13:03:21 +0200
commit3299e556f8a7a87a14d96a7ad93d58312e3bfc2e (patch)
treec26c747514eccdf2a13b27a53c979da8645225c4 /lib/copy-acl.c
parente1ee8ceea44fb66849207b47020547ecf492efe2 (diff)
downloadgnulib-3299e556f8a7a87a14d96a7ad93d58312e3bfc2e.tar.gz
Separate the error handling into a different function.
Diffstat (limited to 'lib/copy-acl.c')
-rw-r--r--lib/copy-acl.c84
1 files changed, 55 insertions, 29 deletions
diff --git a/lib/copy-acl.c b/lib/copy-acl.c
index 97d2087dfe..c9b8935761 100644
--- a/lib/copy-acl.c
+++ b/lib/copy-acl.c
@@ -23,6 +23,7 @@
#include "acl-internal.h"
+
/* Copy access control lists from one file to another. If SOURCE_DESC is
a valid file descriptor, use file descriptor operations, else use
filename based operations on SRC_NAME. Likewise for DEST_DESC and
@@ -30,11 +31,13 @@
If access control lists are not available, fchmod the target file to
MODE. Also sets the non-permission bits of the destination file
(S_ISUID, S_ISGID, S_ISVTX) to those from MODE if any are set.
- Return 0 if successful, otherwise output a diagnostic and return -1. */
+ Return 0 if successful.
+ Return -2 and set errno for an error relating to the source file.
+ Return -1 and set errno for an error relating to the destination file. */
-int
-copy_acl (const char *src_name, int source_desc, const char *dst_name,
- int dest_desc, mode_t mode)
+static int
+qcopy_acl (const char *src_name, int source_desc, const char *dst_name,
+ int dest_desc, mode_t mode)
{
int ret;
@@ -52,10 +55,7 @@ copy_acl (const char *src_name, int source_desc, const char *dst_name,
if (ACL_NOT_WELL_SUPPORTED (errno))
return set_acl (dst_name, dest_desc, mode);
else
- {
- error (0, errno, "%s", quote (src_name));
- return -1;
- }
+ return -2;
}
if (HAVE_ACL_SET_FD && dest_desc != -1)
@@ -92,8 +92,7 @@ copy_acl (const char *src_name, int source_desc, const char *dst_name,
acl_free (acl);
chmod_or_fchmod (dst_name, dest_desc, mode);
}
- error (0, saved_errno, _("preserving permissions for %s"),
- quote (dst_name));
+ errno = saved_errno;
return -1;
}
else
@@ -105,27 +104,21 @@ copy_acl (const char *src_name, int source_desc, const char *dst_name,
separate or special bits are to be set which don't fit into ACLs. */
if (chmod_or_fchmod (dst_name, dest_desc, mode) != 0)
- {
- error (0, errno, _("preserving permissions for %s"),
- quote (dst_name));
- return -1;
- }
+ return -1;
}
if (S_ISDIR (mode))
{
acl = acl_get_file (src_name, ACL_TYPE_DEFAULT);
if (acl == NULL)
- {
- error (0, errno, "%s", quote (src_name));
- return -1;
- }
+ return -2;
if (acl_set_file (dst_name, ACL_TYPE_DEFAULT, acl))
{
- error (0, errno, _("preserving permissions for %s"),
- quote (dst_name));
+ int saved_errno = errno;
+
acl_free (acl);
+ errno = saved_errno;
return -1;
}
else
@@ -142,28 +135,61 @@ copy_acl (const char *src_name, int source_desc, const char *dst_name,
? acl_get (src_name, ACL_NO_TRIVIAL, &aclp)
: facl_get (source_desc, ACL_NO_TRIVIAL, &aclp));
if (ret != 0 && errno != ENOSYS)
- {
- error (0, errno, "%s", quote (src_name));
- return ret;
- }
+ return -2;
# endif
ret = qset_acl (dst_name, dest_desc, mode);
if (ret != 0)
- error (0, errno, _("preserving permissions for %s"), quote (dst_name));
+ return -1;
# if USE_ACL && defined ACL_NO_TRIVIAL
- if (ret == 0 && aclp)
+ if (aclp)
{
ret = (dest_desc < 0
? acl_set (dst_name, aclp)
: facl_set (dest_desc, aclp));
if (ret != 0)
- error (0, errno, _("preserving permissions for %s"), quote (dst_name));
+ {
+ int saved_errno = errno;
+
+ acl_free (aclp);
+ errno = saved_errno;
+ return -1;
+ }
acl_free (aclp);
}
# endif
- return ret;
+ return 0;
#endif
}
+
+
+/* Copy access control lists from one file to another. If SOURCE_DESC is
+ a valid file descriptor, use file descriptor operations, else use
+ filename based operations on SRC_NAME. Likewise for DEST_DESC and
+ DST_NAME.
+ If access control lists are not available, fchmod the target file to
+ MODE. Also sets the non-permission bits of the destination file
+ (S_ISUID, S_ISGID, S_ISVTX) to those from MODE if any are set.
+ Return 0 if successful, otherwise output a diagnostic and return -1. */
+
+int
+copy_acl (const char *src_name, int source_desc, const char *dst_name,
+ int dest_desc, mode_t mode)
+{
+ int ret = qcopy_acl (src_name, source_desc, dst_name, dest_desc, mode);
+ switch (ret)
+ {
+ case -2:
+ error (0, errno, "%s", quote (src_name));
+ return -1;
+
+ case -1:
+ error (0, errno, _("preserving permissions for %s"), quote (dst_name));
+ return -1;
+
+ default:
+ return 0;
+ }
+}