]>
git.proxmox.com Git - systemd.git/blob - src/basic/copy.c
a187ae08fe6dd542168ffe57eb67b37f0d17fc0a
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2014 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
22 #include <sys/sendfile.h>
23 #include <sys/xattr.h>
25 #include "alloc-util.h"
26 #include "btrfs-util.h"
27 #include "chattr-util.h"
29 #include "dirent-util.h"
34 #include "string-util.h"
36 #include "umask-util.h"
38 #include "xattr-util.h"
40 #define COPY_BUFFER_SIZE (16*1024)
42 int copy_bytes(int fdf
, int fdt
, uint64_t max_bytes
, bool try_reflink
) {
43 bool try_sendfile
= true, try_splice
= true;
49 /* Try btrfs reflinks first. */
51 max_bytes
== (uint64_t) -1 &&
52 lseek(fdf
, 0, SEEK_CUR
) == 0 &&
53 lseek(fdt
, 0, SEEK_CUR
) == 0) {
55 r
= btrfs_reflink(fdf
, fdt
);
57 return 0; /* we copied the whole thing, hence hit EOF, return 0 */
61 size_t m
= COPY_BUFFER_SIZE
;
64 if (max_bytes
!= (uint64_t) -1) {
67 return 1; /* return > 0 if we hit the max_bytes limit */
69 if ((uint64_t) m
> max_bytes
)
70 m
= (size_t) max_bytes
;
73 /* First try sendfile(), unless we already tried */
76 n
= sendfile(fdt
, fdf
, NULL
, m
);
78 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
82 /* use fallback below */
83 } else if (n
== 0) /* EOF */
90 /* The try splice, unless we already tried */
92 n
= splice(fdf
, NULL
, fdt
, NULL
, m
, 0);
94 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
98 /* use fallback below */
99 } else if (n
== 0) /* EOF */
106 /* As a fallback just copy bits by hand */
110 n
= read(fdf
, buf
, m
);
113 if (n
== 0) /* EOF */
116 r
= loop_write(fdt
, buf
, (size_t) n
, false);
122 if (max_bytes
!= (uint64_t) -1) {
123 assert(max_bytes
>= (uint64_t) n
);
128 return 0; /* return 0 if we hit EOF earlier than the size limit */
131 static int fd_copy_symlink(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
132 _cleanup_free_
char *target
= NULL
;
139 r
= readlinkat_malloc(df
, from
, &target
);
143 if (symlinkat(target
, dt
, to
) < 0)
146 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
152 static int fd_copy_regular(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
153 _cleanup_close_
int fdf
= -1, fdt
= -1;
154 struct timespec ts
[2];
161 fdf
= openat(df
, from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
165 fdt
= openat(dt
, to
, O_WRONLY
|O_CREAT
|O_EXCL
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
, st
->st_mode
& 07777);
169 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, true);
175 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
178 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
183 (void) futimens(fdt
, ts
);
185 (void) copy_xattr(fdf
, fdt
);
198 static int fd_copy_fifo(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
205 r
= mkfifoat(dt
, to
, st
->st_mode
& 07777);
209 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
212 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
218 static int fd_copy_node(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
225 r
= mknodat(dt
, to
, st
->st_mode
, st
->st_rdev
);
229 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
232 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
238 static int fd_copy_directory(
241 const struct stat
*st
,
244 dev_t original_device
,
247 _cleanup_close_
int fdf
= -1, fdt
= -1;
248 _cleanup_closedir_
DIR *d
= NULL
;
257 fdf
= openat(df
, from
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
259 fdf
= fcntl(df
, F_DUPFD_CLOEXEC
, 3);
266 r
= mkdirat(dt
, to
, st
->st_mode
& 07777);
269 else if (errno
== EEXIST
&& merge
)
274 fdt
= openat(dt
, to
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
281 struct timespec ut
[2] = {
286 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
289 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
292 (void) futimens(fdt
, ut
);
293 (void) copy_xattr(dirfd(d
), fdt
);
296 FOREACH_DIRENT_ALL(de
, d
, return -errno
) {
300 if (STR_IN_SET(de
->d_name
, ".", ".."))
303 if (fstatat(dirfd(d
), de
->d_name
, &buf
, AT_SYMLINK_NOFOLLOW
) < 0) {
308 if (buf
.st_dev
!= original_device
)
311 if (S_ISREG(buf
.st_mode
))
312 q
= fd_copy_regular(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
313 else if (S_ISDIR(buf
.st_mode
))
314 q
= fd_copy_directory(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
, original_device
, merge
);
315 else if (S_ISLNK(buf
.st_mode
))
316 q
= fd_copy_symlink(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
317 else if (S_ISFIFO(buf
.st_mode
))
318 q
= fd_copy_fifo(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
319 else if (S_ISBLK(buf
.st_mode
) || S_ISCHR(buf
.st_mode
))
320 q
= fd_copy_node(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
324 if (q
== -EEXIST
&& merge
)
334 int copy_tree_at(int fdf
, const char *from
, int fdt
, const char *to
, bool merge
) {
340 if (fstatat(fdf
, from
, &st
, AT_SYMLINK_NOFOLLOW
) < 0)
343 if (S_ISREG(st
.st_mode
))
344 return fd_copy_regular(fdf
, from
, &st
, fdt
, to
);
345 else if (S_ISDIR(st
.st_mode
))
346 return fd_copy_directory(fdf
, from
, &st
, fdt
, to
, st
.st_dev
, merge
);
347 else if (S_ISLNK(st
.st_mode
))
348 return fd_copy_symlink(fdf
, from
, &st
, fdt
, to
);
349 else if (S_ISFIFO(st
.st_mode
))
350 return fd_copy_fifo(fdf
, from
, &st
, fdt
, to
);
351 else if (S_ISBLK(st
.st_mode
) || S_ISCHR(st
.st_mode
))
352 return fd_copy_node(fdf
, from
, &st
, fdt
, to
);
357 int copy_tree(const char *from
, const char *to
, bool merge
) {
358 return copy_tree_at(AT_FDCWD
, from
, AT_FDCWD
, to
, merge
);
361 int copy_directory_fd(int dirfd
, const char *to
, bool merge
) {
368 if (fstat(dirfd
, &st
) < 0)
371 if (!S_ISDIR(st
.st_mode
))
374 return fd_copy_directory(dirfd
, NULL
, &st
, AT_FDCWD
, to
, st
.st_dev
, merge
);
377 int copy_file_fd(const char *from
, int fdt
, bool try_reflink
) {
378 _cleanup_close_
int fdf
= -1;
384 fdf
= open(from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
);
388 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, try_reflink
);
390 (void) copy_times(fdf
, fdt
);
391 (void) copy_xattr(fdf
, fdt
);
396 int copy_file(const char *from
, const char *to
, int flags
, mode_t mode
, unsigned chattr_flags
) {
402 RUN_WITH_UMASK(0000) {
403 fdt
= open(to
, flags
|O_WRONLY
|O_CREAT
|O_CLOEXEC
|O_NOCTTY
, mode
);
408 if (chattr_flags
!= 0)
409 (void) chattr_fd(fdt
, chattr_flags
, (unsigned) -1);
411 r
= copy_file_fd(from
, fdt
, true);
418 if (close(fdt
) < 0) {
426 int copy_file_atomic(const char *from
, const char *to
, mode_t mode
, bool replace
, unsigned chattr_flags
) {
427 _cleanup_free_
char *t
= NULL
;
433 r
= tempfn_random(to
, NULL
, &t
);
437 r
= copy_file(from
, t
, O_NOFOLLOW
|O_EXCL
, mode
, chattr_flags
);
442 r
= renameat(AT_FDCWD
, t
, AT_FDCWD
, to
);
446 r
= rename_noreplace(AT_FDCWD
, t
, AT_FDCWD
, to
);
448 (void) unlink_noerrno(t
);
455 int copy_times(int fdf
, int fdt
) {
456 struct timespec ut
[2];
463 if (fstat(fdf
, &st
) < 0)
469 if (futimens(fdt
, ut
) < 0)
472 if (fd_getcrtime(fdf
, &crtime
) >= 0)
473 (void) fd_setcrtime(fdt
, crtime
);
478 int copy_xattr(int fdf
, int fdt
) {
479 _cleanup_free_
char *bufa
= NULL
, *bufb
= NULL
;
480 size_t sza
= 100, szb
= 100;
490 n
= flistxattr(fdf
, bufa
, sza
);
508 assert(l
< (size_t) n
);
510 if (startswith(p
, "user.")) {
519 m
= fgetxattr(fdf
, p
, bufb
, szb
);
521 if (errno
== ERANGE
) {
530 if (fsetxattr(fdt
, p
, bufb
, m
, 0) < 0)