]>
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 "chattr-util.h"
28 #include "dirent-util.h"
33 #include "string-util.h"
36 #include "xattr-util.h"
38 #define COPY_BUFFER_SIZE (16*1024)
40 int copy_bytes(int fdf
, int fdt
, uint64_t max_bytes
, bool try_reflink
) {
41 bool try_sendfile
= true, try_splice
= true;
47 /* Try btrfs reflinks first. */
49 max_bytes
== (uint64_t) -1 &&
50 lseek(fdf
, 0, SEEK_CUR
) == 0 &&
51 lseek(fdt
, 0, SEEK_CUR
) == 0) {
53 r
= btrfs_reflink(fdf
, fdt
);
55 return 0; /* we copied the whole thing, hence hit EOF, return 0 */
59 size_t m
= COPY_BUFFER_SIZE
;
62 if (max_bytes
!= (uint64_t) -1) {
65 return 1; /* return > 0 if we hit the max_bytes limit */
67 if ((uint64_t) m
> max_bytes
)
68 m
= (size_t) max_bytes
;
71 /* First try sendfile(), unless we already tried */
74 n
= sendfile(fdt
, fdf
, NULL
, m
);
76 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
80 /* use fallback below */
81 } else if (n
== 0) /* EOF */
88 /* The try splice, unless we already tried */
90 n
= splice(fdf
, NULL
, fdt
, NULL
, m
, 0);
92 if (errno
!= EINVAL
&& errno
!= ENOSYS
)
96 /* use fallback below */
97 } else if (n
== 0) /* EOF */
104 /* As a fallback just copy bits by hand */
108 n
= read(fdf
, buf
, m
);
111 if (n
== 0) /* EOF */
114 r
= loop_write(fdt
, buf
, (size_t) n
, false);
120 if (max_bytes
!= (uint64_t) -1) {
121 assert(max_bytes
>= (uint64_t) n
);
126 return 0; /* return 0 if we hit EOF earlier than the size limit */
129 static int fd_copy_symlink(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
130 _cleanup_free_
char *target
= NULL
;
137 r
= readlinkat_malloc(df
, from
, &target
);
141 if (symlinkat(target
, dt
, to
) < 0)
144 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
150 static int fd_copy_regular(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
151 _cleanup_close_
int fdf
= -1, fdt
= -1;
152 struct timespec ts
[2];
159 fdf
= openat(df
, from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
163 fdt
= openat(dt
, to
, O_WRONLY
|O_CREAT
|O_EXCL
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
, st
->st_mode
& 07777);
167 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, true);
173 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
176 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
181 (void) futimens(fdt
, ts
);
183 (void) copy_xattr(fdf
, fdt
);
196 static int fd_copy_fifo(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
203 r
= mkfifoat(dt
, to
, st
->st_mode
& 07777);
207 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
210 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
216 static int fd_copy_node(int df
, const char *from
, const struct stat
*st
, int dt
, const char *to
) {
223 r
= mknodat(dt
, to
, st
->st_mode
, st
->st_rdev
);
227 if (fchownat(dt
, to
, st
->st_uid
, st
->st_gid
, AT_SYMLINK_NOFOLLOW
) < 0)
230 if (fchmodat(dt
, to
, st
->st_mode
& 07777, 0) < 0)
236 static int fd_copy_directory(
239 const struct stat
*st
,
242 dev_t original_device
,
245 _cleanup_close_
int fdf
= -1, fdt
= -1;
246 _cleanup_closedir_
DIR *d
= NULL
;
255 fdf
= openat(df
, from
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
257 fdf
= fcntl(df
, F_DUPFD_CLOEXEC
, 3);
264 r
= mkdirat(dt
, to
, st
->st_mode
& 07777);
267 else if (errno
== EEXIST
&& merge
)
272 fdt
= openat(dt
, to
, O_RDONLY
|O_DIRECTORY
|O_CLOEXEC
|O_NOCTTY
|O_NOFOLLOW
);
279 struct timespec ut
[2] = {
284 if (fchown(fdt
, st
->st_uid
, st
->st_gid
) < 0)
287 if (fchmod(fdt
, st
->st_mode
& 07777) < 0)
290 (void) futimens(fdt
, ut
);
291 (void) copy_xattr(dirfd(d
), fdt
);
294 FOREACH_DIRENT_ALL(de
, d
, return -errno
) {
298 if (STR_IN_SET(de
->d_name
, ".", ".."))
301 if (fstatat(dirfd(d
), de
->d_name
, &buf
, AT_SYMLINK_NOFOLLOW
) < 0) {
306 if (buf
.st_dev
!= original_device
)
309 if (S_ISREG(buf
.st_mode
))
310 q
= fd_copy_regular(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
311 else if (S_ISDIR(buf
.st_mode
))
312 q
= fd_copy_directory(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
, original_device
, merge
);
313 else if (S_ISLNK(buf
.st_mode
))
314 q
= fd_copy_symlink(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
315 else if (S_ISFIFO(buf
.st_mode
))
316 q
= fd_copy_fifo(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
317 else if (S_ISBLK(buf
.st_mode
) || S_ISCHR(buf
.st_mode
))
318 q
= fd_copy_node(dirfd(d
), de
->d_name
, &buf
, fdt
, de
->d_name
);
322 if (q
== -EEXIST
&& merge
)
332 int copy_tree_at(int fdf
, const char *from
, int fdt
, const char *to
, bool merge
) {
338 if (fstatat(fdf
, from
, &st
, AT_SYMLINK_NOFOLLOW
) < 0)
341 if (S_ISREG(st
.st_mode
))
342 return fd_copy_regular(fdf
, from
, &st
, fdt
, to
);
343 else if (S_ISDIR(st
.st_mode
))
344 return fd_copy_directory(fdf
, from
, &st
, fdt
, to
, st
.st_dev
, merge
);
345 else if (S_ISLNK(st
.st_mode
))
346 return fd_copy_symlink(fdf
, from
, &st
, fdt
, to
);
347 else if (S_ISFIFO(st
.st_mode
))
348 return fd_copy_fifo(fdf
, from
, &st
, fdt
, to
);
349 else if (S_ISBLK(st
.st_mode
) || S_ISCHR(st
.st_mode
))
350 return fd_copy_node(fdf
, from
, &st
, fdt
, to
);
355 int copy_tree(const char *from
, const char *to
, bool merge
) {
356 return copy_tree_at(AT_FDCWD
, from
, AT_FDCWD
, to
, merge
);
359 int copy_directory_fd(int dirfd
, const char *to
, bool merge
) {
366 if (fstat(dirfd
, &st
) < 0)
369 if (!S_ISDIR(st
.st_mode
))
372 return fd_copy_directory(dirfd
, NULL
, &st
, AT_FDCWD
, to
, st
.st_dev
, merge
);
375 int copy_file_fd(const char *from
, int fdt
, bool try_reflink
) {
376 _cleanup_close_
int fdf
= -1;
382 fdf
= open(from
, O_RDONLY
|O_CLOEXEC
|O_NOCTTY
);
386 r
= copy_bytes(fdf
, fdt
, (uint64_t) -1, try_reflink
);
388 (void) copy_times(fdf
, fdt
);
389 (void) copy_xattr(fdf
, fdt
);
394 int copy_file(const char *from
, const char *to
, int flags
, mode_t mode
, unsigned chattr_flags
) {
400 RUN_WITH_UMASK(0000) {
401 fdt
= open(to
, flags
|O_WRONLY
|O_CREAT
|O_CLOEXEC
|O_NOCTTY
, mode
);
406 if (chattr_flags
!= 0)
407 (void) chattr_fd(fdt
, chattr_flags
, (unsigned) -1);
409 r
= copy_file_fd(from
, fdt
, true);
416 if (close(fdt
) < 0) {
424 int copy_file_atomic(const char *from
, const char *to
, mode_t mode
, bool replace
, unsigned chattr_flags
) {
425 _cleanup_free_
char *t
= NULL
;
431 r
= tempfn_random(to
, NULL
, &t
);
435 r
= copy_file(from
, t
, O_NOFOLLOW
|O_EXCL
, mode
, chattr_flags
);
440 r
= renameat(AT_FDCWD
, t
, AT_FDCWD
, to
);
444 r
= rename_noreplace(AT_FDCWD
, t
, AT_FDCWD
, to
);
446 (void) unlink_noerrno(t
);
453 int copy_times(int fdf
, int fdt
) {
454 struct timespec ut
[2];
461 if (fstat(fdf
, &st
) < 0)
467 if (futimens(fdt
, ut
) < 0)
470 if (fd_getcrtime(fdf
, &crtime
) >= 0)
471 (void) fd_setcrtime(fdt
, crtime
);
476 int copy_xattr(int fdf
, int fdt
) {
477 _cleanup_free_
char *bufa
= NULL
, *bufb
= NULL
;
478 size_t sza
= 100, szb
= 100;
488 n
= flistxattr(fdf
, bufa
, sza
);
506 assert(l
< (size_t) n
);
508 if (startswith(p
, "user.")) {
517 m
= fgetxattr(fdf
, p
, bufb
, szb
);
519 if (errno
== ERANGE
) {
528 if (fsetxattr(fdt
, p
, bufb
, m
, 0) < 0)