/* libguestfs - the guestfsd daemon * Copyright (C) 2011 Red Hat Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include #include #include #include #include "guestfs_protocol.h" #include "daemon.h" #include "actions.h" /* wrflags */ #define DEST_DEVICE_FLAGS O_WRONLY|O_CLOEXEC, 0 /* flags */ #define COPY_UNLINK_DEST_ON_FAILURE 1 /* NB: We cheat slightly by assuming that optargs_bitmask is * compatible for all four of the calls. This is true provided they * all take the same set of optional arguments. */ /* Takes optional arguments, consult optargs_bitmask. */ static int copy (const char *src, const char *src_display, const char *dest, const char *dest_display, int wrflags, int wrmode, int flags, int64_t srcoffset, int64_t destoffset, int64_t size, int sparse) { const int64_t saved_size = size; int src_fd, dest_fd; CLEANUP_FREE char *buf = NULL; size_t n; ssize_t r; int err; buf = malloc (BUFSIZ); if (buf == NULL) { reply_with_perror ("malloc"); return -1; } if ((optargs_bitmask & GUESTFS_COPY_DEVICE_TO_DEVICE_SRCOFFSET_BITMASK)) { if (srcoffset < 0) { reply_with_error ("srcoffset is negative"); return -1; } } else srcoffset = 0; if ((optargs_bitmask & GUESTFS_COPY_DEVICE_TO_DEVICE_DESTOFFSET_BITMASK)) { if (destoffset < 0) { reply_with_error ("destoffset is negative"); return -1; } } else destoffset = 0; if ((optargs_bitmask & GUESTFS_COPY_DEVICE_TO_DEVICE_SIZE_BITMASK)) { if (size < 0) { reply_with_error ("size is negative"); return -1; } } else size = -1; if (! (optargs_bitmask & GUESTFS_COPY_DEVICE_TO_DEVICE_SPARSE_BITMASK)) sparse = 0; /* Open source and destination. */ src_fd = open (src, O_RDONLY|O_CLOEXEC); if (src_fd == -1) { reply_with_perror ("%s", src_display); return -1; } if (srcoffset > 0 && lseek (src_fd, srcoffset, SEEK_SET) == (off_t) -1) { reply_with_perror ("lseek: %s", src_display); close (src_fd); return -1; } dest_fd = open (dest, wrflags, wrmode); if (dest_fd == -1) { reply_with_perror ("%s", dest_display); close (src_fd); return -1; } if (destoffset > 0 && lseek (dest_fd, destoffset, SEEK_SET) == (off_t) -1) { reply_with_perror ("lseek: %s", dest_display); close (src_fd); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } if (size == -1) pulse_mode_start (); while (size != 0) { /* Calculate bytes to copy. */ if (size == -1 || size > BUFSIZ) n = BUFSIZ; else n = size; r = read (src_fd, buf, n); if (r == -1) { err = errno; if (size == -1) pulse_mode_cancel (); errno = err; reply_with_perror ("read: %s", src_display); close (src_fd); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } if (r == 0) { if (size == -1) /* if size == -1, this is normal end of loop */ break; reply_with_error ("%s: input too short", src_display); close (src_fd); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } if (sparse && is_zero (buf, r)) { if (lseek (dest_fd, r, SEEK_CUR) == -1) { err = errno; if (size == -1) pulse_mode_cancel (); errno = err; reply_with_perror ("%s: seek (because of sparse flag)", dest_display); close (src_fd); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } goto sparse_skip; } if (xwrite (dest_fd, buf, r) == -1) { err = errno; if (size == -1) pulse_mode_cancel (); errno = err; reply_with_perror ("%s: write", dest_display); close (src_fd); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } sparse_skip: if (size != -1) { size -= r; notify_progress ((uint64_t) (saved_size - size), (uint64_t) saved_size); } } if (size == -1) pulse_mode_end (); if (close (src_fd) == -1) { reply_with_perror ("close: %s", src_display); close (dest_fd); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } if (close (dest_fd) == -1) { reply_with_perror ("close: %s", dest_display); if (flags & COPY_UNLINK_DEST_ON_FAILURE) unlink (dest); return -1; } return 0; } int do_copy_device_to_device (const char *src, const char *dest, int64_t srcoffset, int64_t destoffset, int64_t size, int sparse, int append) { if ((optargs_bitmask & GUESTFS_COPY_DEVICE_TO_DEVICE_APPEND_BITMASK) && append) { reply_with_error ("the append flag cannot be set for this call"); return -1; } return copy (src, src, dest, dest, DEST_DEVICE_FLAGS, 0, srcoffset, destoffset, size, sparse); } int do_copy_device_to_file (const char *src, const char *dest, int64_t srcoffset, int64_t destoffset, int64_t size, int sparse, int append) { CLEANUP_FREE char *dest_buf = sysroot_path (dest); int wrflags = O_WRONLY|O_CREAT|O_NOCTTY|O_CLOEXEC; if (!dest_buf) { reply_with_perror ("malloc"); return -1; } if ((optargs_bitmask & GUESTFS_COPY_DEVICE_TO_FILE_APPEND_BITMASK) && append) wrflags |= O_APPEND; else wrflags |= O_TRUNC; return copy (src, src, dest_buf, dest, wrflags, 0666, 0, srcoffset, destoffset, size, sparse); } int do_copy_file_to_device (const char *src, const char *dest, int64_t srcoffset, int64_t destoffset, int64_t size, int sparse, int append) { CLEANUP_FREE char *src_buf = sysroot_path (src); if (!src_buf) { reply_with_perror ("malloc"); return -1; } if ((optargs_bitmask & GUESTFS_COPY_FILE_TO_DEVICE_APPEND_BITMASK) && append) { reply_with_error ("the append flag cannot be set for this call"); return -1; } return copy (src_buf, src, dest, dest, DEST_DEVICE_FLAGS, 0, srcoffset, destoffset, size, sparse); } int do_copy_file_to_file (const char *src, const char *dest, int64_t srcoffset, int64_t destoffset, int64_t size, int sparse, int append) { CLEANUP_FREE char *src_buf = NULL, *dest_buf = NULL; int wrflags = O_WRONLY|O_CREAT|O_NOCTTY|O_CLOEXEC; src_buf = sysroot_path (src); if (!src_buf) { reply_with_perror ("malloc"); return -1; } dest_buf = sysroot_path (dest); if (!dest_buf) { reply_with_perror ("malloc"); return -1; } if ((optargs_bitmask & GUESTFS_COPY_FILE_TO_FILE_APPEND_BITMASK) && append) wrflags |= O_APPEND; else wrflags |= O_TRUNC; return copy (src_buf, src, dest_buf, dest, wrflags, 0666, COPY_UNLINK_DEST_ON_FAILURE, srcoffset, destoffset, size, sparse); }