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