-
Notifications
You must be signed in to change notification settings - Fork 118
/
Copy pathreflink.c
152 lines (119 loc) · 4.54 KB
/
reflink.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
/* SPDX-License-Identifier: LGPL-2.1+ */
#include <fcntl.h>
#include <linux/fs.h>
#include <stddef.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include "reflink.h"
#include "util.h"
#define FS_BLOCK_SIZE 4096U
#define VALIDATE 1
#if VALIDATE
static ssize_t pread_try_harder(int fd, void *p, size_t s, off_t o) {
char path[sizeof("/proc/self/fd/") + DECIMAL_STR_MAX(fd)];
ssize_t n;
int fd_read, r;
n = pread(fd, p, s, o);
if (n >= 0)
return n;
r = -errno;
sprintf(path, "/proc/self/fd/%i", fd);
fd_read = open(path, O_CLOEXEC|O_RDONLY|O_NOCTTY);
if (fd_read < 0) {
errno = -r;
return r;
}
n = pread(fd_read, p, s, o);
safe_close(fd_read);
if (n < 0) {
errno = -r;
return r;
}
return n;
}
#endif
static void validate(int source_fd, uint64_t source_offset, int destination_fd, uint64_t destination_offset, uint64_t size) {
#if VALIDATE
ssize_t x, y;
uint8_t *buffer1, *buffer2;
buffer1 = new(uint8_t, size);
assert_se(buffer1);
buffer2 = new(uint8_t, size);
assert_se(buffer2);
x = pread_try_harder(source_fd, buffer1, size, source_offset);
y = pread_try_harder(destination_fd, buffer2, size, destination_offset);
assert_se(x == (ssize_t) size);
assert_se(y == (ssize_t) size);
assert_se(memcmp(buffer1, buffer2, size) == 0);
free(buffer1);
free(buffer2);
#endif
}
int reflink_fd(
int source_fd,
uint64_t source_offset,
int destination_fd,
uint64_t destination_offset,
uint64_t size,
uint64_t *ret_reflinked) {
struct stat a, b;
uint64_t add, reflinked;
/* Creates a reflink on btrfs and other file systems that know the concept. The input parameters are aligned to
* match the fundamental block size (for now assumed to be 4K), and possibly to EOF. */
if (source_fd < 0)
return -EBADF;
if (destination_fd < 0)
return -EBADF;
/* Can only merge blocks starting at a block size boundary */
if (source_offset % FS_BLOCK_SIZE != destination_offset % FS_BLOCK_SIZE)
return -EBADR;
/* Overflow checks */
if (source_offset + size < source_offset)
return -ERANGE;
if (destination_offset + size < destination_offset)
return -ERANGE;
/* First step, round up start offsets to multiple of 4096 */
if (source_offset % FS_BLOCK_SIZE > 0) {
add = FS_BLOCK_SIZE - (source_offset % FS_BLOCK_SIZE);
if (add >= size)
return -EBADR;
source_offset += add;
destination_offset += add;
size -= add;
}
if (fstat(source_fd, &a) < 0)
return -errno;
if (fstat(destination_fd, &b) < 0)
return -errno;
/* Never call the ioctls on something that isn't a regular file, as that's not safe (for example, if the fd
* refers to a block or char device of some kind, which overloads the same ioctl numbers) */
if (S_ISDIR(a.st_mode) || S_ISDIR(b.st_mode))
return -EISDIR;
if (!S_ISREG(a.st_mode) || !S_ISREG(b.st_mode))
return -ENOTTY;
/* Extend to EOF if we can */
if (source_offset + size >= (uint64_t) a.st_size &&
destination_offset + size >= (uint64_t) b.st_size) {
reflinked = size;
size = 0;
} else {
/* Round down size to multiple of 4096 */
size = (size / FS_BLOCK_SIZE) * FS_BLOCK_SIZE;
if (size <= 0)
return -EBADR;
reflinked = size;
}
validate(source_fd, source_offset, destination_fd, destination_offset, reflinked);
if (ioctl(destination_fd, FICLONERANGE,
&(struct file_clone_range) {
.src_fd = source_fd,
.src_offset = source_offset,
.src_length = size,
.dest_offset = destination_offset,
}) < 0)
return -errno;
validate(source_fd, source_offset, destination_fd, destination_offset, reflinked);
if (ret_reflinked)
*ret_reflinked = reflinked;
return 0;
}