]>
git.ipfire.org Git - thirdparty/systemd.git/blob - src/basic/copy.c
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 "btrfs-util.h"
26 #include "string-util.h"
31 #define COPY_BUFFER_SIZE (16*1024)
33 int copy_bytes(int fdf
, int fdt
, uint64_t max_bytes
, bool try_reflink
) {
34 bool try_sendfile
= true, try_splice
= true;
40 /* Try btrfs reflinks first. */
42 max_bytes
== (uint64_t) -1 &&
43 lseek(fdf
, 0, SEEK_CUR
) == 0 &&
44 lseek(fdt
, 0, SEEK_CUR
) == 0) {
46 r
= btrfs_reflink(fdf
, fdt
);
48 return 0; /* we copied the whole thing, hence hit EOF, return 0 */
52 size_t m
= COPY_BUFFER_SIZE
;
55 if (max_bytes
!= (uint64_t) -1) {
58 return 1; /* return > 0 if we hit the max_bytes limit */
60 if ((uint64_t) m
> max_bytes
)
61 m
= (size_t) max_bytes
;
64 /* First try sendfile(), unless we already tried */
67 n
= sendfile(fdt
, fdf
, NULL
, m
);
69 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
73 /* use fallback below */
74 } else if (n
== 0) /* EOF */
81 /* The try splice, unless we already tried */
83 n
= splice(fdf
, NULL
, fdt
, NULL
, m
, 0);
85 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
89 /* use fallback below */
90 } else if (n
== 0) /* EOF */
97 /* As a fallback just copy bits by hand */
101 n
= read(fdf
, buf
, m
);
104 if (n
== 0) /* EOF */
107 r
= loop_write(fdt
, buf
, (size_t) n
, false);
113 if (max_bytes
!= (uint64_t) -1) {
114 assert(max_bytes
>= (uint64_t) n
);
119 return 0; /* return 0 if we hit EOF earlier than the size limit */
122 static int fd_copy_symlink(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
123 _cleanup_free_
char *target
= NULL
;
130 r
= readlinkat_malloc(df
, from
, &target
);
134 if (symlinkat(target
, dt
, to
) < 0)
137 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
143 static int fd_copy_regular(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
144 _cleanup_close_
int fdf
= -1, fdt
= -1;
145 struct timespec ts
[2];
152 fdf
= openat(df
, from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
156 fdt
= openat(dt
, to
, O_WRONLY
|O_CREAT
|O_EXCL
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
, st
->st_mode
& 07777);
160 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, true);
166 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
169 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
174 (void) futimens(fdt
, ts
);
176 (void) copy_xattr(fdf
, fdt
);
189 static int fd_copy_fifo(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
196 r
= mkfifoat(dt
, to
, st
->st_mode
& 07777);
200 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
203 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
209 static int fd_copy_node(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
216 r
= mknodat(dt
, to
, st
->st_mode
, st
->st_rdev
);
220 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
223 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
229 static int fd_copy_directory(
232 const struct stat
*st
,
235 dev_t original_device
,
238 _cleanup_close_
int fdf
= -1, fdt
= -1;
239 _cleanup_closedir_
DIR *d
= NULL
;
248 fdf
= openat(df
, from
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
250 fdf
= fcntl(df
, F_DUPFD_CLOEXEC
, 3);
257 r
= mkdirat(dt
, to
, st
->st_mode
& 07777);
260 else if (errno
== EEXIST
&& merge
)
265 fdt
= openat(dt
, to
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
272 struct timespec ut
[2] = {
277 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
280 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
283 (void) futimens(fdt
, ut
);
284 (void) copy_xattr(dirfd(d
), fdt
);
287 FOREACH_DIRENT_ALL(de
, d
, return -errno
) {
291 if (STR_IN_SET(de
->d_name
, ".", ".."))
294 if (fstatat(dirfd(d
), de
->d_name
, &buf
, AT_SYMLINK_NOFOLLOW
) < 0) {
299 if (buf
.st_dev
!= original_device
)
302 if (S_ISREG(buf
.st_mode
))
303 q
= fd_copy_regular(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
304 else if (S_ISDIR(buf
.st_mode
))
305 q
= fd_copy_directory(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
, original_device
, merge
);
306 else if (S_ISLNK(buf
.st_mode
))
307 q
= fd_copy_symlink(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
308 else if (S_ISFIFO(buf
.st_mode
))
309 q
= fd_copy_fifo(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
310 else if (S_ISBLK(buf
.st_mode
) || S_ISCHR(buf
.st_mode
))
311 q
= fd_copy_node(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
315 if (q
== -EEXIST
&& merge
)
325 int copy_tree_at(int fdf
, const char *from
, int fdt
, const char *to
, bool merge
) {
331 if (fstatat(fdf
, from
, &st
, AT_SYMLINK_NOFOLLOW
) < 0)
334 if (S_ISREG(st
.st_mode
))
335 return fd_copy_regular(fdf
, from
, &st
, fdt
, to
);
336 else if (S_ISDIR(st
.st_mode
))
337 return fd_copy_directory(fdf
, from
, &st
, fdt
, to
, st
.st_dev
, merge
);
338 else if (S_ISLNK(st
.st_mode
))
339 return fd_copy_symlink(fdf
, from
, &st
, fdt
, to
);
340 else if (S_ISFIFO(st
.st_mode
))
341 return fd_copy_fifo(fdf
, from
, &st
, fdt
, to
);
342 else if (S_ISBLK(st
.st_mode
) || S_ISCHR(st
.st_mode
))
343 return fd_copy_node(fdf
, from
, &st
, fdt
, to
);
348 int copy_tree(const char *from
, const char *to
, bool merge
) {
349 return copy_tree_at(AT_FDCWD
, from
, AT_FDCWD
, to
, merge
);
352 int copy_directory_fd(int dirfd
, const char *to
, bool merge
) {
359 if (fstat(dirfd
, &st
) < 0)
362 if (!S_ISDIR(st
.st_mode
))
365 return fd_copy_directory(dirfd
, NULL
, &st
, AT_FDCWD
, to
, st
.st_dev
, merge
);
368 int copy_file_fd(const char *from
, int fdt
, bool try_reflink
) {
369 _cleanup_close_
int fdf
= -1;
375 fdf
= open(from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
);
379 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, try_reflink
);
381 (void) copy_times(fdf
, fdt
);
382 (void) copy_xattr(fdf
, fdt
);
387 int copy_file(const char *from
, const char *to
, int flags
, mode_t mode
, unsigned chattr_flags
) {
393 RUN_WITH_UMASK(0000) {
394 fdt
= open(to
, flags
|O_WRONLY
|O_CREAT
|O_CLOEXEC
|O_NOCTTY
, mode
);
399 if (chattr_flags
!= 0)
400 (void) chattr_fd(fdt
, chattr_flags
, (unsigned) -1);
402 r
= copy_file_fd(from
, fdt
, true);
409 if (close(fdt
) < 0) {
417 int copy_file_atomic(const char *from
, const char *to
, mode_t mode
, bool replace
, unsigned chattr_flags
) {
418 _cleanup_free_
char *t
= NULL
;
424 r
= tempfn_random(to
, NULL
, &t
);
428 r
= copy_file(from
, t
, O_NOFOLLOW
|O_EXCL
, mode
, chattr_flags
);
433 r
= renameat(AT_FDCWD
, t
, AT_FDCWD
, to
);
437 r
= rename_noreplace(AT_FDCWD
, t
, AT_FDCWD
, to
);
439 (void) unlink_noerrno(t
);
446 int copy_times(int fdf
, int fdt
) {
447 struct timespec ut
[2];
454 if (fstat(fdf
, &st
) < 0)
460 if (futimens(fdt
, ut
) < 0)
463 if (fd_getcrtime(fdf
, &crtime
) >= 0)
464 (void) fd_setcrtime(fdt
, crtime
);
469 int copy_xattr(int fdf
, int fdt
) {
470 _cleanup_free_
char *bufa
= NULL
, *bufb
= NULL
;
471 size_t sza
= 100, szb
= 100;
481 n
= flistxattr(fdf
, bufa
, sza
);
499 assert(l
< (size_t) n
);
501 if (startswith(p
, "user.")) {
510 m
= fgetxattr(fdf
, p
, bufb
, szb
);
512 if (errno
== ERANGE
) {
521 if (fsetxattr(fdt
, p
, bufb
, m
, 0) < 0)