Branch data Line data Source code
1 : : /*
2 : : * Copyright (c) 2016 Jeff Mahoney <jeffm@suse.com>
3 : : * All rights reserved.
4 : : *
5 : : * Redistribution and use in source and binary forms, with or without
6 : : * modification, are permitted provided that the following conditions
7 : : * are met:
8 : : * 1. Redistributions of source code must retain the above copyright
9 : : * notice, this list of conditions and the following disclaimer.
10 : : * 2. Redistributions in binary form must reproduce the above copyright
11 : : * notice, this list of conditions and the following disclaimer in the
12 : : * documentation and/or other materials provided with the distribution.
13 : : * 3. The name of the author may not be used to endorse or promote products
14 : : * derived from this software without specific prior written permission.
15 : : *
16 : : * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 : : * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 : : * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 : : * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 : : * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 : : * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 : : * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 : : * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 : : * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 : : * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 : : */
27 : :
28 : : #include "defs.h"
29 : : #include <linux/ioctl.h>
30 : : #include <linux/fs.h>
31 : :
32 : : #ifdef HAVE_LINUX_FIEMAP_H
33 : : # include <linux/fiemap.h>
34 : : # include "xlat/fiemap_flags.h"
35 : : # include "xlat/fiemap_extent_flags.h"
36 : : #endif
37 : :
38 : : #ifndef FICLONE
39 : : # define FICLONE _IOW(0x94, 9, int)
40 : : #endif
41 : :
42 : : #ifndef FICLONERANGE
43 : : # define FICLONERANGE _IOW(0x94, 13, struct file_clone_range)
44 : : struct file_clone_range {
45 : : int64_t src_fd;
46 : : uint64_t src_offset;
47 : : uint64_t src_length;
48 : : uint64_t dest_offset;
49 : : };
50 : : #endif
51 : :
52 : : #ifndef FIDEDUPERANGE
53 : : # define FIDEDUPERANGE _IOWR(0x94, 54, struct file_dedupe_range)
54 : : struct file_dedupe_range_info {
55 : : int64_t dest_fd; /* in - destination file */
56 : : uint64_t dest_offset; /* in - start of extent in destination */
57 : : uint64_t bytes_deduped; /* out - total # of bytes we were able
58 : : * to dedupe from this file. */
59 : : /* status of this dedupe operation:
60 : : * < 0 for error
61 : : * == FILE_DEDUPE_RANGE_SAME if dedupe succeeds
62 : : * == FILE_DEDUPE_RANGE_DIFFERS if data differs
63 : : */
64 : : int32_t status; /* out - see above description */
65 : : uint32_t reserved; /* must be zero */
66 : : };
67 : :
68 : : struct file_dedupe_range {
69 : : uint64_t src_offset; /* in - start of extent in source */
70 : : uint64_t src_length; /* in - length of extent */
71 : : uint16_t dest_count; /* in - total elements in info array */
72 : : uint16_t reserved1; /* must be zero */
73 : : uint32_t reserved2; /* must be zero */
74 : : struct file_dedupe_range_info info[0];
75 : : };
76 : : #endif
77 : :
78 : : static bool
79 : 30 : print_file_dedupe_range_info(struct tcb *tcp, void *elem_buf,
80 : : size_t elem_size, void *data)
81 : : {
82 : 30 : const struct file_dedupe_range_info *info = elem_buf;
83 : 30 : unsigned int *count = data;
84 : :
85 [ + + ]: 30 : if (count) {
86 [ + + ]: 15 : if (*count == 0) {
87 : 5 : tprints("...");
88 : 5 : return false;
89 : : }
90 : 10 : --*count;
91 : : }
92 : :
93 [ + - ]: 25 : if (entering(tcp)) {
94 : 25 : tprints("{dest_fd=");
95 : 25 : printfd(tcp, info->dest_fd);
96 : 25 : tprintf(", dest_offset=%" PRIu64 "}",
97 : 25 : (uint64_t) info->dest_offset);
98 : : } else {
99 : 0 : tprintf("{bytes_deduped=%" PRIu64 ", status=%d}",
100 : 0 : (uint64_t) info->bytes_deduped, info->status);
101 : : }
102 : :
103 : 25 : return true;
104 : : }
105 : :
106 : : #ifdef HAVE_LINUX_FIEMAP_H
107 : : static bool
108 : 0 : print_fiemap_extent(struct tcb *tcp, void *elem_buf, size_t elem_size, void *data)
109 : : {
110 : 0 : const struct fiemap_extent *fe = elem_buf;
111 : :
112 : 0 : tprintf("{fe_logical=%" PRI__u64
113 : : ", fe_physical=%" PRI__u64
114 : : ", fe_length=%" PRI__u64 ", ",
115 : : fe->fe_logical, fe->fe_physical, fe->fe_length);
116 : :
117 : 0 : printflags64(fiemap_extent_flags, fe->fe_flags,
118 : : "FIEMAP_EXTENT_???");
119 : 0 : tprints("}");
120 : :
121 : 0 : return true;
122 : : }
123 : : #endif /* HAVE_LINUX_FIEMAP_H */
124 : :
125 : : int
126 : 80 : file_ioctl(struct tcb *const tcp, const unsigned int code,
127 : : const kernel_ulong_t arg)
128 : : {
129 [ + + + + : 80 : switch (code) {
- ]
130 : : case FICLONE: /* W */
131 : 10 : tprintf(", %d", (int) arg);
132 : 10 : break;
133 : :
134 : : case FICLONERANGE: { /* W */
135 : : struct file_clone_range args;
136 : :
137 : 20 : tprints(", ");
138 [ + + ]: 20 : if (umove_or_printaddr(tcp, arg, &args))
139 : 10 : break;
140 : :
141 : 10 : tprints("{src_fd=");
142 : 10 : printfd(tcp, args.src_fd);
143 : 10 : tprintf(", src_offset=%" PRIu64
144 : : ", src_length=%" PRIu64
145 : : ", dest_offset=%" PRIu64 "}",
146 : 10 : (uint64_t) args.src_offset,
147 : 10 : (uint64_t) args.src_length,
148 : 10 : (uint64_t) args.dest_offset);
149 : 20 : break;
150 : : }
151 : :
152 : : case FIDEDUPERANGE: { /* RW */
153 : : struct file_dedupe_range args;
154 : : struct file_dedupe_range_info info;
155 : 40 : unsigned int *limit = NULL;
156 : 40 : unsigned int count = 2;
157 : : bool rc;
158 : :
159 [ + + ]: 40 : if (entering(tcp))
160 : 30 : tprints(", ");
161 [ + - ]: 10 : else if (syserror(tcp))
162 : 30 : break;
163 : : else
164 : 0 : tprints(" => ");
165 : :
166 [ + + ]: 30 : if (umove_or_printaddr(tcp, arg, &args))
167 : 10 : break;
168 : :
169 : 20 : tprints("{");
170 [ + - ]: 20 : if (entering(tcp)) {
171 : 20 : tprintf("src_offset=%" PRIu64
172 : : ", src_length=%" PRIu64
173 : : ", dest_count=%hu, ",
174 : 20 : (uint64_t) args.src_offset,
175 : 20 : (uint64_t) args.src_length,
176 : 20 : (uint16_t) args.dest_count);
177 : : }
178 : :
179 : 20 : tprints("info=");
180 : :
181 : : /* Limit how many elements we print in abbrev mode. */
182 [ + + ][ + + ]: 20 : if (abbrev(tcp) && args.dest_count > count)
183 : 5 : limit = &count;
184 : :
185 : 20 : rc = print_array(tcp, arg + offsetof(typeof(args), info),
186 : 20 : args.dest_count, &info, sizeof(info),
187 : : umoven_or_printaddr,
188 : : print_file_dedupe_range_info, limit);
189 : :
190 : 20 : tprints("}");
191 [ + + ][ + - ]: 20 : if (!rc || exiting(tcp))
192 : : break;
193 : :
194 : 10 : return 0;
195 : : }
196 : :
197 : : #ifdef HAVE_LINUX_FIEMAP_H
198 : : case FS_IOC_FIEMAP: {
199 : : struct fiemap args;
200 : :
201 [ + + ]: 10 : if (entering(tcp))
202 : 5 : tprints(", ");
203 [ + - ]: 5 : else if (syserror(tcp))
204 : 5 : break;
205 : : else
206 : 0 : tprints(" => ");
207 : :
208 [ - + ]: 5 : if (umove_or_printaddr(tcp, arg, &args))
209 : 0 : break;
210 : :
211 [ + - ]: 5 : if (entering(tcp)) {
212 : 5 : tprintf("{fm_start=%" PRI__u64 ", "
213 : : "fm_length=%" PRI__u64 ", "
214 : : "fm_flags=",
215 : : args.fm_start, args.fm_length);
216 : 5 : printflags64(fiemap_flags, args.fm_flags,
217 : : "FIEMAP_FLAG_???");
218 : 5 : tprintf(", fm_extent_count=%u}", args.fm_extent_count);
219 : 5 : return 0;
220 : : }
221 : :
222 : 0 : tprints("{fm_flags=");
223 : 0 : printflags64(fiemap_flags, args.fm_flags,
224 : : "FIEMAP_FLAG_???");
225 : 0 : tprintf(", fm_mapped_extents=%u",
226 : : args.fm_mapped_extents);
227 : 0 : tprints(", fm_extents=");
228 [ # # ]: 0 : if (abbrev(tcp)) {
229 : 0 : tprints("...");
230 : : } else {
231 : : struct fiemap_extent fe;
232 : 0 : print_array(tcp,
233 : : arg + offsetof(typeof(args), fm_extents),
234 : 0 : args.fm_mapped_extents, &fe, sizeof(fe),
235 : : umoven_or_printaddr,
236 : : print_fiemap_extent, 0);
237 : : }
238 : 0 : tprints("}");
239 : :
240 : 0 : break;
241 : : }
242 : : #endif /* HAVE_LINUX_FIEMAP_H */
243 : :
244 : : default:
245 : 0 : return RVAL_DECODED;
246 : : };
247 : :
248 : 65 : return RVAL_DECODED | 1;
249 : : }
|