/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- * * Copyright (C) 2012 William Jon McCann * Copyright (C) 2012 Colin Walters * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #include "config.h" #ifndef _GNU_SOURCE #define _GNU_SOURCE #endif #define _GSYSTEM_NO_LOCAL_ALLOC #include "libgsystem.h" #include #include #include #include #include /* Taken from systemd/src/shared/util.h */ union dirent_storage { struct dirent dent; guint8 storage[offsetof(struct dirent, d_name) + ((NAME_MAX + 1 + sizeof(long)) & ~(sizeof(long) - 1))]; }; static gboolean cp_internal (GFile *src, GFile *dest, gboolean use_hardlinks, GCancellable *cancellable, GError **error); static gboolean cp_internal_one_item (GFile *src, GFile *dest, GFileInfo *file_info, gboolean use_hardlinks, GCancellable *cancellable, GError **error) { gboolean ret = FALSE; const char *name = g_file_info_get_name (file_info); GFile *src_child = g_file_get_child (src, name); GFile *dest_child = g_file_get_child (dest, name); if (g_file_info_get_file_type (file_info) == G_FILE_TYPE_DIRECTORY) { if (!gs_file_ensure_directory (dest_child, FALSE, cancellable, error)) goto out; /* Can't do this even though we'd like to; it fails with an error about * setting standard::type not being supported =/ * if (!g_file_set_attributes_from_info (dest_child, file_info, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, cancellable, error)) goto out; */ if (chmod (gs_file_get_path_cached (dest_child), g_file_info_get_attribute_uint32 (file_info, "unix::mode")) == -1) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } if (chown (gs_file_get_path_cached (dest_child), g_file_info_get_attribute_uint32 (file_info, "unix::uid"), g_file_info_get_attribute_uint32 (file_info, "unix::gid")) == -1) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } if (!cp_internal (src_child, dest_child, use_hardlinks, cancellable, error)) goto out; } else { gboolean did_link = FALSE; (void) unlink (gs_file_get_path_cached (dest_child)); if (use_hardlinks) { if (link (gs_file_get_path_cached (src_child), gs_file_get_path_cached (dest_child)) == -1) { if (!(errno == EMLINK || errno == EXDEV)) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } use_hardlinks = FALSE; } else did_link = TRUE; } if (!did_link) { if (!g_file_copy (src_child, dest_child, G_FILE_COPY_OVERWRITE | G_FILE_COPY_ALL_METADATA | G_FILE_COPY_NOFOLLOW_SYMLINKS, cancellable, NULL, NULL, error)) goto out; } } ret = TRUE; out: g_clear_object (&src_child); g_clear_object (&dest_child); return ret; } static gboolean cp_internal (GFile *src, GFile *dest, gboolean use_hardlinks, GCancellable *cancellable, GError **error) { gboolean ret = FALSE; GFileEnumerator *enumerator = NULL; GFileInfo *file_info = NULL; GError *temp_error = NULL; enumerator = g_file_enumerate_children (src, "standard::type,standard::name,unix::uid,unix::gid,unix::mode", G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, cancellable, error); if (!enumerator) goto out; if (!gs_file_ensure_directory (dest, FALSE, cancellable, error)) goto out; while ((file_info = g_file_enumerator_next_file (enumerator, cancellable, &temp_error)) != NULL) { if (!cp_internal_one_item (src, dest, file_info, use_hardlinks, cancellable, error)) goto out; g_clear_object (&file_info); } if (temp_error) { g_propagate_error (error, temp_error); goto out; } ret = TRUE; out: g_clear_object (&enumerator); g_clear_object (&file_info); return ret; } /** * gs_shutil_cp_al_or_fallback: * @src: Source path * @dest: Destination path * @cancellable: * @error: * * Recursively copy path @src (which must be a directory) to the * target @dest. If possible, hardlinks are used; if a hardlink is * not possible, a regular copy is created. Any existing files are * overwritten. * * Returns: %TRUE on success */ gboolean gs_shutil_cp_al_or_fallback (GFile *src, GFile *dest, GCancellable *cancellable, GError **error) { return cp_internal (src, dest, TRUE, cancellable, error); } /** * gs_shutil_cp_a: * @src: Source path * @dest: Destination path * @cancellable: * @error: * * Recursively copy path @src (which must be a directory) to the * target @dest. Any existing files are overwritten. * * Returns: %TRUE on success */ gboolean gs_shutil_cp_a (GFile *src, GFile *dest, GCancellable *cancellable, GError **error) { return cp_internal (src, dest, FALSE, cancellable, error); } static unsigned char struct_stat_to_dt (struct stat *stbuf) { if (S_ISDIR (stbuf->st_mode)) return DT_DIR; if (S_ISREG (stbuf->st_mode)) return DT_REG; if (S_ISCHR (stbuf->st_mode)) return DT_CHR; if (S_ISBLK (stbuf->st_mode)) return DT_BLK; if (S_ISFIFO (stbuf->st_mode)) return DT_FIFO; if (S_ISLNK (stbuf->st_mode)) return DT_LNK; if (S_ISSOCK (stbuf->st_mode)) return DT_SOCK; return DT_UNKNOWN; } static gboolean gs_shutil_rm_rf_children (DIR *dir, GCancellable *cancellable, GError **error) { gboolean ret = FALSE; int dfd; DIR *child_dir = NULL; struct dirent *dent; union dirent_storage buf; if (g_cancellable_set_error_if_cancelled (cancellable, error)) goto out; dfd = dirfd (dir); while (readdir_r (dir, &buf.dent, &dent) == 0) { if (dent == NULL) break; if (dent->d_type == DT_UNKNOWN) { struct stat stbuf; if (fstatat (dfd, dent->d_name, &stbuf, AT_SYMLINK_NOFOLLOW) == -1) { int errsv = errno; if (errsv == ENOENT) continue; else { g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } } dent->d_type = struct_stat_to_dt (&stbuf); /* Assume unknown types are just treated like regular files */ if (dent->d_type == DT_UNKNOWN) dent->d_type = DT_REG; } if (strcmp (dent->d_name, ".") == 0 || strcmp (dent->d_name, "..") == 0) continue; if (dent->d_type == DT_DIR) { int child_dfd = openat (dfd, dent->d_name, O_RDONLY | O_NONBLOCK | O_DIRECTORY | O_CLOEXEC | O_NOFOLLOW); if (child_dfd == -1) { if (errno == ENOENT) continue; else { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } } child_dir = fdopendir (child_dfd); if (!child_dir) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } if (!gs_shutil_rm_rf_children (child_dir, cancellable, error)) goto out; if (unlinkat (dfd, dent->d_name, AT_REMOVEDIR) == -1) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } (void) closedir (child_dir); child_dir = NULL; } else { if (unlinkat (dfd, dent->d_name, 0) == -1) { int errsv = errno; if (errno != ENOENT) { g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } } } } /* Ignore error result from readdir_r, that's what others * seem to do =( */ ret = TRUE; out: if (child_dir) (void) closedir (child_dir); return ret; } /** * gs_shutil_rm_rf: * @path: A file or directory * @cancellable: * @error: * * Recursively delete the filename referenced by @path; it may be a * file or directory. No error is thrown if @path does not exist. */ gboolean gs_shutil_rm_rf (GFile *path, GCancellable *cancellable, GError **error) { gboolean ret = FALSE; int dfd = -1; DIR *d = NULL; /* With O_NOFOLLOW first */ dfd = openat (AT_FDCWD, gs_file_get_path_cached (path), O_RDONLY | O_NONBLOCK | O_DIRECTORY | O_CLOEXEC | O_NOFOLLOW); if (dfd == -1) { int errsv = errno; if (errsv == ENOENT) { ; } else if (errsv == ENOTDIR || errsv == ELOOP) { if (!gs_file_unlink (path, cancellable, error)) goto out; } else { g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } } else { d = fdopendir (dfd); if (!d) { int errsv = errno; g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } if (!gs_shutil_rm_rf_children (d, cancellable, error)) goto out; if (rmdir (gs_file_get_path_cached (path)) == -1) { int errsv = errno; if (errsv != ENOENT) { g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); goto out; } } } ret = TRUE; out: if (d) (void) closedir (d); return ret; }