xref: /aosp_15_r20/external/ltp/lib/tst_device.c (revision 49cdfc7efb34551c7342be41a7384b9c40d7cab7)
1*49cdfc7eSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*49cdfc7eSAndroid Build Coastguard Worker /*
3*49cdfc7eSAndroid Build Coastguard Worker  * Copyright (C) 2014 Cyril Hrubis [email protected]
4*49cdfc7eSAndroid Build Coastguard Worker  */
5*49cdfc7eSAndroid Build Coastguard Worker 
6*49cdfc7eSAndroid Build Coastguard Worker #include <sys/types.h>
7*49cdfc7eSAndroid Build Coastguard Worker #include <sys/stat.h>
8*49cdfc7eSAndroid Build Coastguard Worker #include <sys/ioctl.h>
9*49cdfc7eSAndroid Build Coastguard Worker #include <sys/mount.h>
10*49cdfc7eSAndroid Build Coastguard Worker #include <mntent.h>
11*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
12*49cdfc7eSAndroid Build Coastguard Worker #include <unistd.h>
13*49cdfc7eSAndroid Build Coastguard Worker #include <stdlib.h>
14*49cdfc7eSAndroid Build Coastguard Worker #include <linux/loop.h>
15*49cdfc7eSAndroid Build Coastguard Worker #include <stdint.h>
16*49cdfc7eSAndroid Build Coastguard Worker #include <inttypes.h>
17*49cdfc7eSAndroid Build Coastguard Worker #include <sys/sysmacros.h>
18*49cdfc7eSAndroid Build Coastguard Worker #include <linux/btrfs.h>
19*49cdfc7eSAndroid Build Coastguard Worker #include <linux/limits.h>
20*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/syscalls.h"
21*49cdfc7eSAndroid Build Coastguard Worker #include "test.h"
22*49cdfc7eSAndroid Build Coastguard Worker #include "safe_macros.h"
23*49cdfc7eSAndroid Build Coastguard Worker #include "tst_device.h"
24*49cdfc7eSAndroid Build Coastguard Worker 
25*49cdfc7eSAndroid Build Coastguard Worker #ifndef LOOP_CTL_GET_FREE
26*49cdfc7eSAndroid Build Coastguard Worker # define LOOP_CTL_GET_FREE 0x4C82
27*49cdfc7eSAndroid Build Coastguard Worker #endif
28*49cdfc7eSAndroid Build Coastguard Worker 
29*49cdfc7eSAndroid Build Coastguard Worker #define LOOP_CONTROL_FILE "/dev/loop-control"
30*49cdfc7eSAndroid Build Coastguard Worker 
31*49cdfc7eSAndroid Build Coastguard Worker #define DEV_FILE "test_dev.img"
32*49cdfc7eSAndroid Build Coastguard Worker #define DEV_SIZE_MB 300u
33*49cdfc7eSAndroid Build Coastguard Worker #define UUID_STR_SZ 37
34*49cdfc7eSAndroid Build Coastguard Worker #define UUID_FMT "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x"
35*49cdfc7eSAndroid Build Coastguard Worker 
36*49cdfc7eSAndroid Build Coastguard Worker static char dev_path[PATH_MAX];
37*49cdfc7eSAndroid Build Coastguard Worker static int device_acquired;
38*49cdfc7eSAndroid Build Coastguard Worker static unsigned long prev_dev_sec_write;
39*49cdfc7eSAndroid Build Coastguard Worker 
40*49cdfc7eSAndroid Build Coastguard Worker static const char * const dev_loop_variants[] = {
41*49cdfc7eSAndroid Build Coastguard Worker 	"/dev/loop%i",
42*49cdfc7eSAndroid Build Coastguard Worker 	"/dev/loop/%i",
43*49cdfc7eSAndroid Build Coastguard Worker 	"/dev/block/loop%i"
44*49cdfc7eSAndroid Build Coastguard Worker };
45*49cdfc7eSAndroid Build Coastguard Worker 
46*49cdfc7eSAndroid Build Coastguard Worker static const char * const dev_variants[] = {
47*49cdfc7eSAndroid Build Coastguard Worker 	"/dev/%s",
48*49cdfc7eSAndroid Build Coastguard Worker 	"/dev/block/%s"
49*49cdfc7eSAndroid Build Coastguard Worker };
50*49cdfc7eSAndroid Build Coastguard Worker 
set_dev_loop_path(int dev,char * path,size_t path_len)51*49cdfc7eSAndroid Build Coastguard Worker static int set_dev_loop_path(int dev, char *path, size_t path_len)
52*49cdfc7eSAndroid Build Coastguard Worker {
53*49cdfc7eSAndroid Build Coastguard Worker 	unsigned int i;
54*49cdfc7eSAndroid Build Coastguard Worker 	struct stat st;
55*49cdfc7eSAndroid Build Coastguard Worker 
56*49cdfc7eSAndroid Build Coastguard Worker 	for (i = 0; i < ARRAY_SIZE(dev_loop_variants); i++) {
57*49cdfc7eSAndroid Build Coastguard Worker 		snprintf(path, path_len, dev_loop_variants[i], dev);
58*49cdfc7eSAndroid Build Coastguard Worker 
59*49cdfc7eSAndroid Build Coastguard Worker 		if (stat(path, &st) == 0 && S_ISBLK(st.st_mode))
60*49cdfc7eSAndroid Build Coastguard Worker 			return 0;
61*49cdfc7eSAndroid Build Coastguard Worker 	}
62*49cdfc7eSAndroid Build Coastguard Worker 
63*49cdfc7eSAndroid Build Coastguard Worker 	return 1;
64*49cdfc7eSAndroid Build Coastguard Worker }
65*49cdfc7eSAndroid Build Coastguard Worker 
set_dev_path(char * dev,char * path,size_t path_len)66*49cdfc7eSAndroid Build Coastguard Worker static int set_dev_path(char *dev, char *path, size_t path_len)
67*49cdfc7eSAndroid Build Coastguard Worker {
68*49cdfc7eSAndroid Build Coastguard Worker 	unsigned int i;
69*49cdfc7eSAndroid Build Coastguard Worker 	struct stat st;
70*49cdfc7eSAndroid Build Coastguard Worker 
71*49cdfc7eSAndroid Build Coastguard Worker 	for (i = 0; i < ARRAY_SIZE(dev_variants); i++) {
72*49cdfc7eSAndroid Build Coastguard Worker 		snprintf(path, path_len, dev_variants[i], dev);
73*49cdfc7eSAndroid Build Coastguard Worker 
74*49cdfc7eSAndroid Build Coastguard Worker 		if (stat(path, &st) == 0 && S_ISBLK(st.st_mode))
75*49cdfc7eSAndroid Build Coastguard Worker 			return 0;
76*49cdfc7eSAndroid Build Coastguard Worker 	}
77*49cdfc7eSAndroid Build Coastguard Worker 
78*49cdfc7eSAndroid Build Coastguard Worker 	return 1;
79*49cdfc7eSAndroid Build Coastguard Worker }
80*49cdfc7eSAndroid Build Coastguard Worker 
tst_find_free_loopdev(char * path,size_t path_len)81*49cdfc7eSAndroid Build Coastguard Worker int tst_find_free_loopdev(char *path, size_t path_len)
82*49cdfc7eSAndroid Build Coastguard Worker {
83*49cdfc7eSAndroid Build Coastguard Worker 	int ctl_fd, dev_fd, rc, i, path_set;
84*49cdfc7eSAndroid Build Coastguard Worker 	struct loop_info loopinfo;
85*49cdfc7eSAndroid Build Coastguard Worker 	char buf[PATH_MAX];
86*49cdfc7eSAndroid Build Coastguard Worker 
87*49cdfc7eSAndroid Build Coastguard Worker 	/* since Linux 3.1 */
88*49cdfc7eSAndroid Build Coastguard Worker 	ctl_fd = open(LOOP_CONTROL_FILE, O_RDWR);
89*49cdfc7eSAndroid Build Coastguard Worker 
90*49cdfc7eSAndroid Build Coastguard Worker 	if (ctl_fd > 0) {
91*49cdfc7eSAndroid Build Coastguard Worker 		rc = ioctl(ctl_fd, LOOP_CTL_GET_FREE);
92*49cdfc7eSAndroid Build Coastguard Worker 		close(ctl_fd);
93*49cdfc7eSAndroid Build Coastguard Worker 		if (rc >= 0) {
94*49cdfc7eSAndroid Build Coastguard Worker 			if (path) {
95*49cdfc7eSAndroid Build Coastguard Worker 				// b/148978487 retry to allow time for device creation
96*49cdfc7eSAndroid Build Coastguard Worker 				for (i = 0; i < 50; i++) {
97*49cdfc7eSAndroid Build Coastguard Worker 					path_set = set_dev_loop_path(rc, path, path_len);
98*49cdfc7eSAndroid Build Coastguard Worker 					// set_dev_path returns 1 on success
99*49cdfc7eSAndroid Build Coastguard Worker 					if (!path_set)
100*49cdfc7eSAndroid Build Coastguard Worker 						break;
101*49cdfc7eSAndroid Build Coastguard Worker 					usleep(50000);
102*49cdfc7eSAndroid Build Coastguard Worker 				}
103*49cdfc7eSAndroid Build Coastguard Worker 				if (path_set)
104*49cdfc7eSAndroid Build Coastguard Worker 					tst_brkm(TBROK, NULL, "Could not stat loop device %i", rc);
105*49cdfc7eSAndroid Build Coastguard Worker 			}
106*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TINFO, "Found free device %d '%s'",
107*49cdfc7eSAndroid Build Coastguard Worker 				rc, path ?: "");
108*49cdfc7eSAndroid Build Coastguard Worker 			return rc;
109*49cdfc7eSAndroid Build Coastguard Worker 		}
110*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "Couldn't find free loop device");
111*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
112*49cdfc7eSAndroid Build Coastguard Worker 	}
113*49cdfc7eSAndroid Build Coastguard Worker 
114*49cdfc7eSAndroid Build Coastguard Worker 	switch (errno) {
115*49cdfc7eSAndroid Build Coastguard Worker 	case ENOENT:
116*49cdfc7eSAndroid Build Coastguard Worker 	break;
117*49cdfc7eSAndroid Build Coastguard Worker 	case EACCES:
118*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO | TERRNO,
119*49cdfc7eSAndroid Build Coastguard Worker 			 "Not allowed to open " LOOP_CONTROL_FILE ". "
120*49cdfc7eSAndroid Build Coastguard Worker 			 "Are you root?");
121*49cdfc7eSAndroid Build Coastguard Worker 	break;
122*49cdfc7eSAndroid Build Coastguard Worker 	default:
123*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TBROK | TERRNO, "Failed to open " LOOP_CONTROL_FILE);
124*49cdfc7eSAndroid Build Coastguard Worker 	}
125*49cdfc7eSAndroid Build Coastguard Worker 
126*49cdfc7eSAndroid Build Coastguard Worker 	/*
127*49cdfc7eSAndroid Build Coastguard Worker 	 * Older way is to iterate over /dev/loop%i and /dev/loop/%i and try
128*49cdfc7eSAndroid Build Coastguard Worker 	 * LOOP_GET_STATUS ioctl() which fails for free loop devices.
129*49cdfc7eSAndroid Build Coastguard Worker 	 */
130*49cdfc7eSAndroid Build Coastguard Worker 	for (i = 0; i < 256; i++) {
131*49cdfc7eSAndroid Build Coastguard Worker 
132*49cdfc7eSAndroid Build Coastguard Worker 		if (set_dev_loop_path(i, buf, sizeof(buf)))
133*49cdfc7eSAndroid Build Coastguard Worker 			continue;
134*49cdfc7eSAndroid Build Coastguard Worker 
135*49cdfc7eSAndroid Build Coastguard Worker 		dev_fd = open(buf, O_RDONLY);
136*49cdfc7eSAndroid Build Coastguard Worker 
137*49cdfc7eSAndroid Build Coastguard Worker 		if (dev_fd < 0)
138*49cdfc7eSAndroid Build Coastguard Worker 			continue;
139*49cdfc7eSAndroid Build Coastguard Worker 
140*49cdfc7eSAndroid Build Coastguard Worker 		if (ioctl(dev_fd, LOOP_GET_STATUS, &loopinfo) == 0) {
141*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TINFO, "Device '%s' in use", buf);
142*49cdfc7eSAndroid Build Coastguard Worker 		} else {
143*49cdfc7eSAndroid Build Coastguard Worker 			if (errno != ENXIO)
144*49cdfc7eSAndroid Build Coastguard Worker 				continue;
145*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TINFO, "Found free device '%s'", buf);
146*49cdfc7eSAndroid Build Coastguard Worker 			close(dev_fd);
147*49cdfc7eSAndroid Build Coastguard Worker 			if (path != NULL) {
148*49cdfc7eSAndroid Build Coastguard Worker 				strncpy(path, buf, path_len);
149*49cdfc7eSAndroid Build Coastguard Worker 				path[path_len-1] = '\0';
150*49cdfc7eSAndroid Build Coastguard Worker 			}
151*49cdfc7eSAndroid Build Coastguard Worker 			return i;
152*49cdfc7eSAndroid Build Coastguard Worker 		}
153*49cdfc7eSAndroid Build Coastguard Worker 
154*49cdfc7eSAndroid Build Coastguard Worker 		close(dev_fd);
155*49cdfc7eSAndroid Build Coastguard Worker 	}
156*49cdfc7eSAndroid Build Coastguard Worker 
157*49cdfc7eSAndroid Build Coastguard Worker 	tst_resm(TINFO, "No free devices found");
158*49cdfc7eSAndroid Build Coastguard Worker 
159*49cdfc7eSAndroid Build Coastguard Worker 	return -1;
160*49cdfc7eSAndroid Build Coastguard Worker }
161*49cdfc7eSAndroid Build Coastguard Worker 
tst_attach_device(const char * dev,const char * file)162*49cdfc7eSAndroid Build Coastguard Worker int tst_attach_device(const char *dev, const char *file)
163*49cdfc7eSAndroid Build Coastguard Worker {
164*49cdfc7eSAndroid Build Coastguard Worker 	int dev_fd, file_fd;
165*49cdfc7eSAndroid Build Coastguard Worker 	struct loop_info loopinfo;
166*49cdfc7eSAndroid Build Coastguard Worker 
167*49cdfc7eSAndroid Build Coastguard Worker 	// b/148978487 retry to allow time for device creation
168*49cdfc7eSAndroid Build Coastguard Worker 	int attach_tries = 20;
169*49cdfc7eSAndroid Build Coastguard Worker 	while (attach_tries--) {
170*49cdfc7eSAndroid Build Coastguard Worker 		dev_fd = open(dev, O_RDWR);
171*49cdfc7eSAndroid Build Coastguard Worker 		if (dev_fd >= 0)
172*49cdfc7eSAndroid Build Coastguard Worker 			break;
173*49cdfc7eSAndroid Build Coastguard Worker 		usleep(50000);
174*49cdfc7eSAndroid Build Coastguard Worker 	}
175*49cdfc7eSAndroid Build Coastguard Worker 	if (dev_fd < 0) {
176*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "open('%s', O_RDWR) failed", dev);
177*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
178*49cdfc7eSAndroid Build Coastguard Worker 	}
179*49cdfc7eSAndroid Build Coastguard Worker 
180*49cdfc7eSAndroid Build Coastguard Worker 	file_fd = open(file, O_RDWR);
181*49cdfc7eSAndroid Build Coastguard Worker 	if (file_fd < 0) {
182*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "open('%s', O_RDWR) failed", file);
183*49cdfc7eSAndroid Build Coastguard Worker 		close(dev_fd);
184*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
185*49cdfc7eSAndroid Build Coastguard Worker 	}
186*49cdfc7eSAndroid Build Coastguard Worker 
187*49cdfc7eSAndroid Build Coastguard Worker 	if (ioctl(dev_fd, LOOP_SET_FD, file_fd) < 0) {
188*49cdfc7eSAndroid Build Coastguard Worker 		close(dev_fd);
189*49cdfc7eSAndroid Build Coastguard Worker 		close(file_fd);
190*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "ioctl(%s, LOOP_SET_FD, %s) failed",
191*49cdfc7eSAndroid Build Coastguard Worker 			 dev, file);
192*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
193*49cdfc7eSAndroid Build Coastguard Worker 	}
194*49cdfc7eSAndroid Build Coastguard Worker 
195*49cdfc7eSAndroid Build Coastguard Worker 	/* Old mkfs.btrfs use LOOP_GET_STATUS instead of backing_file to get
196*49cdfc7eSAndroid Build Coastguard Worker 	 * associated filename, so we need to set up the device by calling
197*49cdfc7eSAndroid Build Coastguard Worker 	 * LOOP_SET_FD and LOOP_SET_STATUS.
198*49cdfc7eSAndroid Build Coastguard Worker 	 */
199*49cdfc7eSAndroid Build Coastguard Worker 	memset(&loopinfo, 0, sizeof(loopinfo));
200*49cdfc7eSAndroid Build Coastguard Worker 	strcpy(loopinfo.lo_name, file);
201*49cdfc7eSAndroid Build Coastguard Worker 
202*49cdfc7eSAndroid Build Coastguard Worker 	if (ioctl(dev_fd, LOOP_SET_STATUS, &loopinfo)) {
203*49cdfc7eSAndroid Build Coastguard Worker 		close(dev_fd);
204*49cdfc7eSAndroid Build Coastguard Worker 		close(file_fd);
205*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO,
206*49cdfc7eSAndroid Build Coastguard Worker 			 "ioctl(%s, LOOP_SET_STATUS, %s) failed", dev, file);
207*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
208*49cdfc7eSAndroid Build Coastguard Worker 	}
209*49cdfc7eSAndroid Build Coastguard Worker 
210*49cdfc7eSAndroid Build Coastguard Worker 	close(dev_fd);
211*49cdfc7eSAndroid Build Coastguard Worker 	close(file_fd);
212*49cdfc7eSAndroid Build Coastguard Worker 	return 0;
213*49cdfc7eSAndroid Build Coastguard Worker }
214*49cdfc7eSAndroid Build Coastguard Worker 
tst_get_device_size(const char * dev_path)215*49cdfc7eSAndroid Build Coastguard Worker uint64_t tst_get_device_size(const char *dev_path)
216*49cdfc7eSAndroid Build Coastguard Worker {
217*49cdfc7eSAndroid Build Coastguard Worker 	int fd;
218*49cdfc7eSAndroid Build Coastguard Worker 	uint64_t size;
219*49cdfc7eSAndroid Build Coastguard Worker 	struct stat st;
220*49cdfc7eSAndroid Build Coastguard Worker 
221*49cdfc7eSAndroid Build Coastguard Worker 	if (!dev_path)
222*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK, NULL, "No block device path");
223*49cdfc7eSAndroid Build Coastguard Worker 
224*49cdfc7eSAndroid Build Coastguard Worker 	if (stat(dev_path, &st)) {
225*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "stat() failed");
226*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
227*49cdfc7eSAndroid Build Coastguard Worker 	}
228*49cdfc7eSAndroid Build Coastguard Worker 
229*49cdfc7eSAndroid Build Coastguard Worker 	if (!S_ISBLK(st.st_mode)) {
230*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN, "%s is not a block device", dev_path);
231*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
232*49cdfc7eSAndroid Build Coastguard Worker 	}
233*49cdfc7eSAndroid Build Coastguard Worker 
234*49cdfc7eSAndroid Build Coastguard Worker 	fd = open(dev_path, O_RDONLY);
235*49cdfc7eSAndroid Build Coastguard Worker 	if (fd < 0) {
236*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO,
237*49cdfc7eSAndroid Build Coastguard Worker 				"open(%s, O_RDONLY) failed", dev_path);
238*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
239*49cdfc7eSAndroid Build Coastguard Worker 	}
240*49cdfc7eSAndroid Build Coastguard Worker 
241*49cdfc7eSAndroid Build Coastguard Worker 	if (ioctl(fd, BLKGETSIZE64, &size)) {
242*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO,
243*49cdfc7eSAndroid Build Coastguard Worker 				"ioctl(fd, BLKGETSIZE64, ...) failed");
244*49cdfc7eSAndroid Build Coastguard Worker 		close(fd);
245*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
246*49cdfc7eSAndroid Build Coastguard Worker 	}
247*49cdfc7eSAndroid Build Coastguard Worker 
248*49cdfc7eSAndroid Build Coastguard Worker 	if (close(fd)) {
249*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO,
250*49cdfc7eSAndroid Build Coastguard Worker 				"close(fd) failed");
251*49cdfc7eSAndroid Build Coastguard Worker 		return -1;
252*49cdfc7eSAndroid Build Coastguard Worker 	}
253*49cdfc7eSAndroid Build Coastguard Worker 
254*49cdfc7eSAndroid Build Coastguard Worker 	return size/1024/1024;
255*49cdfc7eSAndroid Build Coastguard Worker }
256*49cdfc7eSAndroid Build Coastguard Worker 
tst_detach_device_by_fd(const char * dev,int dev_fd)257*49cdfc7eSAndroid Build Coastguard Worker int tst_detach_device_by_fd(const char *dev, int dev_fd)
258*49cdfc7eSAndroid Build Coastguard Worker {
259*49cdfc7eSAndroid Build Coastguard Worker 	int ret, i;
260*49cdfc7eSAndroid Build Coastguard Worker 
261*49cdfc7eSAndroid Build Coastguard Worker 	/* keep trying to clear LOOPDEV until we get ENXIO, a quick succession
262*49cdfc7eSAndroid Build Coastguard Worker 	 * of attach/detach might not give udev enough time to complete
263*49cdfc7eSAndroid Build Coastguard Worker 	 *
264*49cdfc7eSAndroid Build Coastguard Worker 	 * Since 18048c1af783 ("loop: Fix a race between loop detach and loop open")
265*49cdfc7eSAndroid Build Coastguard Worker 	 * device is detached only after last close.
266*49cdfc7eSAndroid Build Coastguard Worker 	 */
267*49cdfc7eSAndroid Build Coastguard Worker 	for (i = 0; i < 40; i++) {
268*49cdfc7eSAndroid Build Coastguard Worker 		ret = ioctl(dev_fd, LOOP_CLR_FD, 0);
269*49cdfc7eSAndroid Build Coastguard Worker 
270*49cdfc7eSAndroid Build Coastguard Worker 		if (ret && (errno == ENXIO)) {
271*49cdfc7eSAndroid Build Coastguard Worker 			SAFE_CLOSE(NULL, dev_fd);
272*49cdfc7eSAndroid Build Coastguard Worker 			return 0;
273*49cdfc7eSAndroid Build Coastguard Worker 		}
274*49cdfc7eSAndroid Build Coastguard Worker 
275*49cdfc7eSAndroid Build Coastguard Worker 		if (ret && (errno != EBUSY)) {
276*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TWARN,
277*49cdfc7eSAndroid Build Coastguard Worker 				 "ioctl(%s, LOOP_CLR_FD, 0) unexpectedly failed with: %s",
278*49cdfc7eSAndroid Build Coastguard Worker 				 dev, tst_strerrno(errno));
279*49cdfc7eSAndroid Build Coastguard Worker 			SAFE_CLOSE(NULL, dev_fd);
280*49cdfc7eSAndroid Build Coastguard Worker 			return 1;
281*49cdfc7eSAndroid Build Coastguard Worker 		}
282*49cdfc7eSAndroid Build Coastguard Worker 
283*49cdfc7eSAndroid Build Coastguard Worker 		usleep(50000);
284*49cdfc7eSAndroid Build Coastguard Worker 	}
285*49cdfc7eSAndroid Build Coastguard Worker 
286*49cdfc7eSAndroid Build Coastguard Worker 	tst_resm(TWARN,
287*49cdfc7eSAndroid Build Coastguard Worker 		"ioctl(%s, LOOP_CLR_FD, 0) no ENXIO for too long", dev);
288*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_CLOSE(NULL, dev_fd);
289*49cdfc7eSAndroid Build Coastguard Worker 	return 1;
290*49cdfc7eSAndroid Build Coastguard Worker }
291*49cdfc7eSAndroid Build Coastguard Worker 
tst_detach_device(const char * dev)292*49cdfc7eSAndroid Build Coastguard Worker int tst_detach_device(const char *dev)
293*49cdfc7eSAndroid Build Coastguard Worker {
294*49cdfc7eSAndroid Build Coastguard Worker 	int dev_fd, ret;
295*49cdfc7eSAndroid Build Coastguard Worker 
296*49cdfc7eSAndroid Build Coastguard Worker 	dev_fd = open(dev, O_RDONLY);
297*49cdfc7eSAndroid Build Coastguard Worker 	if (dev_fd < 0) {
298*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "open(%s) failed", dev);
299*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
300*49cdfc7eSAndroid Build Coastguard Worker 	}
301*49cdfc7eSAndroid Build Coastguard Worker 
302*49cdfc7eSAndroid Build Coastguard Worker 	ret = tst_detach_device_by_fd(dev, dev_fd);
303*49cdfc7eSAndroid Build Coastguard Worker 	return ret;
304*49cdfc7eSAndroid Build Coastguard Worker }
305*49cdfc7eSAndroid Build Coastguard Worker 
tst_dev_sync(int fd)306*49cdfc7eSAndroid Build Coastguard Worker int tst_dev_sync(int fd)
307*49cdfc7eSAndroid Build Coastguard Worker {
308*49cdfc7eSAndroid Build Coastguard Worker 	return syscall(__NR_syncfs, fd);
309*49cdfc7eSAndroid Build Coastguard Worker }
310*49cdfc7eSAndroid Build Coastguard Worker 
tst_acquire_loop_device(unsigned int size,const char * filename)311*49cdfc7eSAndroid Build Coastguard Worker const char *tst_acquire_loop_device(unsigned int size, const char *filename)
312*49cdfc7eSAndroid Build Coastguard Worker {
313*49cdfc7eSAndroid Build Coastguard Worker 	unsigned int acq_dev_size = size ? size : DEV_SIZE_MB;
314*49cdfc7eSAndroid Build Coastguard Worker 
315*49cdfc7eSAndroid Build Coastguard Worker 	if (tst_prealloc_file(filename, 1024 * 1024, acq_dev_size)) {
316*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "Failed to create %s", filename);
317*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
318*49cdfc7eSAndroid Build Coastguard Worker 	}
319*49cdfc7eSAndroid Build Coastguard Worker 
320*49cdfc7eSAndroid Build Coastguard Worker 	if (tst_find_free_loopdev(dev_path, sizeof(dev_path)) == -1)
321*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
322*49cdfc7eSAndroid Build Coastguard Worker 
323*49cdfc7eSAndroid Build Coastguard Worker 	if (tst_attach_device(dev_path, filename))
324*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
325*49cdfc7eSAndroid Build Coastguard Worker 
326*49cdfc7eSAndroid Build Coastguard Worker 	return dev_path;
327*49cdfc7eSAndroid Build Coastguard Worker }
328*49cdfc7eSAndroid Build Coastguard Worker 
tst_acquire_device__(unsigned int size)329*49cdfc7eSAndroid Build Coastguard Worker const char *tst_acquire_device__(unsigned int size)
330*49cdfc7eSAndroid Build Coastguard Worker {
331*49cdfc7eSAndroid Build Coastguard Worker 	const char *dev;
332*49cdfc7eSAndroid Build Coastguard Worker 	unsigned int acq_dev_size;
333*49cdfc7eSAndroid Build Coastguard Worker 	uint64_t ltp_dev_size;
334*49cdfc7eSAndroid Build Coastguard Worker 
335*49cdfc7eSAndroid Build Coastguard Worker 	acq_dev_size = size ? size : DEV_SIZE_MB;
336*49cdfc7eSAndroid Build Coastguard Worker 
337*49cdfc7eSAndroid Build Coastguard Worker 	dev = getenv("LTP_DEV");
338*49cdfc7eSAndroid Build Coastguard Worker 
339*49cdfc7eSAndroid Build Coastguard Worker 	if (dev) {
340*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "Using test device LTP_DEV='%s'", dev);
341*49cdfc7eSAndroid Build Coastguard Worker 
342*49cdfc7eSAndroid Build Coastguard Worker 		ltp_dev_size = tst_get_device_size(dev);
343*49cdfc7eSAndroid Build Coastguard Worker 
344*49cdfc7eSAndroid Build Coastguard Worker 		if (acq_dev_size <= ltp_dev_size)
345*49cdfc7eSAndroid Build Coastguard Worker 			return dev;
346*49cdfc7eSAndroid Build Coastguard Worker 
347*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "Skipping $LTP_DEV size %"PRIu64"MB, requested size %uMB",
348*49cdfc7eSAndroid Build Coastguard Worker 				ltp_dev_size, acq_dev_size);
349*49cdfc7eSAndroid Build Coastguard Worker 	}
350*49cdfc7eSAndroid Build Coastguard Worker 
351*49cdfc7eSAndroid Build Coastguard Worker 	dev = tst_acquire_loop_device(acq_dev_size, DEV_FILE);
352*49cdfc7eSAndroid Build Coastguard Worker 
353*49cdfc7eSAndroid Build Coastguard Worker 	if (dev)
354*49cdfc7eSAndroid Build Coastguard Worker 		device_acquired = 1;
355*49cdfc7eSAndroid Build Coastguard Worker 
356*49cdfc7eSAndroid Build Coastguard Worker 	return dev;
357*49cdfc7eSAndroid Build Coastguard Worker }
358*49cdfc7eSAndroid Build Coastguard Worker 
tst_acquire_device_(void (cleanup_fn)(void),unsigned int size)359*49cdfc7eSAndroid Build Coastguard Worker const char *tst_acquire_device_(void (cleanup_fn)(void), unsigned int size)
360*49cdfc7eSAndroid Build Coastguard Worker {
361*49cdfc7eSAndroid Build Coastguard Worker 	const char *device;
362*49cdfc7eSAndroid Build Coastguard Worker 
363*49cdfc7eSAndroid Build Coastguard Worker 	if (device_acquired) {
364*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK, cleanup_fn, "Device already acquired");
365*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
366*49cdfc7eSAndroid Build Coastguard Worker 	}
367*49cdfc7eSAndroid Build Coastguard Worker 
368*49cdfc7eSAndroid Build Coastguard Worker 	if (!tst_tmpdir_created()) {
369*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK, cleanup_fn,
370*49cdfc7eSAndroid Build Coastguard Worker 			 "Cannot acquire device without tmpdir() created");
371*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
372*49cdfc7eSAndroid Build Coastguard Worker 	}
373*49cdfc7eSAndroid Build Coastguard Worker 
374*49cdfc7eSAndroid Build Coastguard Worker 	device = tst_acquire_device__(size);
375*49cdfc7eSAndroid Build Coastguard Worker 
376*49cdfc7eSAndroid Build Coastguard Worker 	if (!device) {
377*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK, cleanup_fn, "Failed to acquire device");
378*49cdfc7eSAndroid Build Coastguard Worker 		return NULL;
379*49cdfc7eSAndroid Build Coastguard Worker 	}
380*49cdfc7eSAndroid Build Coastguard Worker 
381*49cdfc7eSAndroid Build Coastguard Worker 	return device;
382*49cdfc7eSAndroid Build Coastguard Worker }
383*49cdfc7eSAndroid Build Coastguard Worker 
tst_release_device(const char * dev)384*49cdfc7eSAndroid Build Coastguard Worker int tst_release_device(const char *dev)
385*49cdfc7eSAndroid Build Coastguard Worker {
386*49cdfc7eSAndroid Build Coastguard Worker 	int ret;
387*49cdfc7eSAndroid Build Coastguard Worker 
388*49cdfc7eSAndroid Build Coastguard Worker 	if (!device_acquired)
389*49cdfc7eSAndroid Build Coastguard Worker 		return 0;
390*49cdfc7eSAndroid Build Coastguard Worker 
391*49cdfc7eSAndroid Build Coastguard Worker 	/*
392*49cdfc7eSAndroid Build Coastguard Worker 	 * Loop device was created -> we need to detach it.
393*49cdfc7eSAndroid Build Coastguard Worker 	 *
394*49cdfc7eSAndroid Build Coastguard Worker 	 * The file image is deleted in tst_rmdir();
395*49cdfc7eSAndroid Build Coastguard Worker 	 */
396*49cdfc7eSAndroid Build Coastguard Worker 	ret = tst_detach_device(dev);
397*49cdfc7eSAndroid Build Coastguard Worker 
398*49cdfc7eSAndroid Build Coastguard Worker 	device_acquired = 0;
399*49cdfc7eSAndroid Build Coastguard Worker 
400*49cdfc7eSAndroid Build Coastguard Worker 	return ret;
401*49cdfc7eSAndroid Build Coastguard Worker }
402*49cdfc7eSAndroid Build Coastguard Worker 
tst_clear_device(const char * dev)403*49cdfc7eSAndroid Build Coastguard Worker int tst_clear_device(const char *dev)
404*49cdfc7eSAndroid Build Coastguard Worker {
405*49cdfc7eSAndroid Build Coastguard Worker 	if (tst_fill_file(dev, 0, 1024, 512)) {
406*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN, "Failed to clear 512k block on %s", dev);
407*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
408*49cdfc7eSAndroid Build Coastguard Worker 	}
409*49cdfc7eSAndroid Build Coastguard Worker 
410*49cdfc7eSAndroid Build Coastguard Worker 	return 0;
411*49cdfc7eSAndroid Build Coastguard Worker }
412*49cdfc7eSAndroid Build Coastguard Worker 
tst_umount(const char * path)413*49cdfc7eSAndroid Build Coastguard Worker int tst_umount(const char *path)
414*49cdfc7eSAndroid Build Coastguard Worker {
415*49cdfc7eSAndroid Build Coastguard Worker 	int err, ret, i;
416*49cdfc7eSAndroid Build Coastguard Worker 
417*49cdfc7eSAndroid Build Coastguard Worker 	for (i = 0; i < 50; i++) {
418*49cdfc7eSAndroid Build Coastguard Worker 		ret = umount(path);
419*49cdfc7eSAndroid Build Coastguard Worker 		err = errno;
420*49cdfc7eSAndroid Build Coastguard Worker 
421*49cdfc7eSAndroid Build Coastguard Worker 		if (!ret)
422*49cdfc7eSAndroid Build Coastguard Worker 			return 0;
423*49cdfc7eSAndroid Build Coastguard Worker 
424*49cdfc7eSAndroid Build Coastguard Worker 		if (err != EBUSY) {
425*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TWARN, "umount('%s') failed with %s",
426*49cdfc7eSAndroid Build Coastguard Worker 				 path, tst_strerrno(err));
427*49cdfc7eSAndroid Build Coastguard Worker 			errno = err;
428*49cdfc7eSAndroid Build Coastguard Worker 			return ret;
429*49cdfc7eSAndroid Build Coastguard Worker 		}
430*49cdfc7eSAndroid Build Coastguard Worker 
431*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "umount('%s') failed with %s, try %2i...",
432*49cdfc7eSAndroid Build Coastguard Worker 			 path, tst_strerrno(err), i+1);
433*49cdfc7eSAndroid Build Coastguard Worker 
434*49cdfc7eSAndroid Build Coastguard Worker 		if (i == 0) {
435*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TINFO, "Likely gvfsd-trash is probing newly "
436*49cdfc7eSAndroid Build Coastguard Worker 				 "mounted fs, kill it to speed up tests.");
437*49cdfc7eSAndroid Build Coastguard Worker 		}
438*49cdfc7eSAndroid Build Coastguard Worker 
439*49cdfc7eSAndroid Build Coastguard Worker 		usleep(100000);
440*49cdfc7eSAndroid Build Coastguard Worker 	}
441*49cdfc7eSAndroid Build Coastguard Worker 
442*49cdfc7eSAndroid Build Coastguard Worker 	tst_resm(TWARN, "Failed to umount('%s') after 50 retries", path);
443*49cdfc7eSAndroid Build Coastguard Worker 	errno = err;
444*49cdfc7eSAndroid Build Coastguard Worker 	return -1;
445*49cdfc7eSAndroid Build Coastguard Worker }
446*49cdfc7eSAndroid Build Coastguard Worker 
tst_is_mounted(const char * path)447*49cdfc7eSAndroid Build Coastguard Worker int tst_is_mounted(const char *path)
448*49cdfc7eSAndroid Build Coastguard Worker {
449*49cdfc7eSAndroid Build Coastguard Worker 	char line[PATH_MAX];
450*49cdfc7eSAndroid Build Coastguard Worker 	FILE *file;
451*49cdfc7eSAndroid Build Coastguard Worker 	int ret = 0;
452*49cdfc7eSAndroid Build Coastguard Worker 
453*49cdfc7eSAndroid Build Coastguard Worker 	file = SAFE_FOPEN(NULL, "/proc/mounts", "r");
454*49cdfc7eSAndroid Build Coastguard Worker 
455*49cdfc7eSAndroid Build Coastguard Worker 	while (fgets(line, sizeof(line), file)) {
456*49cdfc7eSAndroid Build Coastguard Worker 		if (strstr(line, path) != NULL) {
457*49cdfc7eSAndroid Build Coastguard Worker 			ret = 1;
458*49cdfc7eSAndroid Build Coastguard Worker 			break;
459*49cdfc7eSAndroid Build Coastguard Worker 		}
460*49cdfc7eSAndroid Build Coastguard Worker 	}
461*49cdfc7eSAndroid Build Coastguard Worker 
462*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_FCLOSE(NULL, file);
463*49cdfc7eSAndroid Build Coastguard Worker 
464*49cdfc7eSAndroid Build Coastguard Worker 	if (!ret)
465*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "No device is mounted at %s", path);
466*49cdfc7eSAndroid Build Coastguard Worker 
467*49cdfc7eSAndroid Build Coastguard Worker 	return ret;
468*49cdfc7eSAndroid Build Coastguard Worker }
469*49cdfc7eSAndroid Build Coastguard Worker 
tst_is_mounted_at_tmpdir(const char * path)470*49cdfc7eSAndroid Build Coastguard Worker int tst_is_mounted_at_tmpdir(const char *path)
471*49cdfc7eSAndroid Build Coastguard Worker {
472*49cdfc7eSAndroid Build Coastguard Worker 	char cdir[PATH_MAX], mpath[PATH_MAX];
473*49cdfc7eSAndroid Build Coastguard Worker 	int ret;
474*49cdfc7eSAndroid Build Coastguard Worker 
475*49cdfc7eSAndroid Build Coastguard Worker 	if (!getcwd(cdir, PATH_MAX)) {
476*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO, "Failed to find current directory");
477*49cdfc7eSAndroid Build Coastguard Worker 		return 0;
478*49cdfc7eSAndroid Build Coastguard Worker 	}
479*49cdfc7eSAndroid Build Coastguard Worker 
480*49cdfc7eSAndroid Build Coastguard Worker 	ret = snprintf(mpath, PATH_MAX, "%s/%s", cdir, path);
481*49cdfc7eSAndroid Build Coastguard Worker 	if (ret < 0 || ret >= PATH_MAX) {
482*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TWARN | TERRNO,
483*49cdfc7eSAndroid Build Coastguard Worker 			 "snprintf() should have returned %d instead of %d",
484*49cdfc7eSAndroid Build Coastguard Worker 			 PATH_MAX, ret);
485*49cdfc7eSAndroid Build Coastguard Worker 		return 0;
486*49cdfc7eSAndroid Build Coastguard Worker 	}
487*49cdfc7eSAndroid Build Coastguard Worker 
488*49cdfc7eSAndroid Build Coastguard Worker 	return tst_is_mounted(mpath);
489*49cdfc7eSAndroid Build Coastguard Worker }
490*49cdfc7eSAndroid Build Coastguard Worker 
find_stat_file(const char * dev,char * path,size_t path_len)491*49cdfc7eSAndroid Build Coastguard Worker static int find_stat_file(const char *dev, char *path, size_t path_len)
492*49cdfc7eSAndroid Build Coastguard Worker {
493*49cdfc7eSAndroid Build Coastguard Worker 	const char *devname = strrchr(dev, '/') + 1;
494*49cdfc7eSAndroid Build Coastguard Worker 
495*49cdfc7eSAndroid Build Coastguard Worker 	snprintf(path, path_len, "/sys/block/%s/stat", devname);
496*49cdfc7eSAndroid Build Coastguard Worker 
497*49cdfc7eSAndroid Build Coastguard Worker 	if (!access(path, F_OK))
498*49cdfc7eSAndroid Build Coastguard Worker 		return 1;
499*49cdfc7eSAndroid Build Coastguard Worker 
500*49cdfc7eSAndroid Build Coastguard Worker 	DIR *dir = SAFE_OPENDIR(NULL, "/sys/block/");
501*49cdfc7eSAndroid Build Coastguard Worker 	struct dirent *ent;
502*49cdfc7eSAndroid Build Coastguard Worker 
503*49cdfc7eSAndroid Build Coastguard Worker 	while ((ent = readdir(dir))) {
504*49cdfc7eSAndroid Build Coastguard Worker 		snprintf(path, path_len, "/sys/block/%s/%s/stat", ent->d_name, devname);
505*49cdfc7eSAndroid Build Coastguard Worker 
506*49cdfc7eSAndroid Build Coastguard Worker 		if (!access(path, F_OK)) {
507*49cdfc7eSAndroid Build Coastguard Worker 			SAFE_CLOSEDIR(NULL, dir);
508*49cdfc7eSAndroid Build Coastguard Worker 			return 1;
509*49cdfc7eSAndroid Build Coastguard Worker 		}
510*49cdfc7eSAndroid Build Coastguard Worker 	}
511*49cdfc7eSAndroid Build Coastguard Worker 
512*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_CLOSEDIR(NULL, dir);
513*49cdfc7eSAndroid Build Coastguard Worker 	return 0;
514*49cdfc7eSAndroid Build Coastguard Worker }
515*49cdfc7eSAndroid Build Coastguard Worker 
tst_dev_bytes_written(const char * dev)516*49cdfc7eSAndroid Build Coastguard Worker unsigned long tst_dev_bytes_written(const char *dev)
517*49cdfc7eSAndroid Build Coastguard Worker {
518*49cdfc7eSAndroid Build Coastguard Worker 	unsigned long dev_sec_write = 0, dev_bytes_written, io_ticks = 0;
519*49cdfc7eSAndroid Build Coastguard Worker 	char dev_stat_path[PATH_MAX];
520*49cdfc7eSAndroid Build Coastguard Worker 
521*49cdfc7eSAndroid Build Coastguard Worker 	if (!find_stat_file(dev, dev_stat_path, sizeof(dev_stat_path)))
522*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TCONF, NULL, "Test device stat file: %s not found",
523*49cdfc7eSAndroid Build Coastguard Worker 			 dev_stat_path);
524*49cdfc7eSAndroid Build Coastguard Worker 
525*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_FILE_SCANF(NULL, dev_stat_path,
526*49cdfc7eSAndroid Build Coastguard Worker 			"%*s %*s %*s %*s %*s %*s %lu %*s %*s %lu",
527*49cdfc7eSAndroid Build Coastguard Worker 			&dev_sec_write, &io_ticks);
528*49cdfc7eSAndroid Build Coastguard Worker 
529*49cdfc7eSAndroid Build Coastguard Worker 	if (!io_ticks)
530*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TCONF, NULL, "Test device stat file: %s broken",
531*49cdfc7eSAndroid Build Coastguard Worker 			 dev_stat_path);
532*49cdfc7eSAndroid Build Coastguard Worker 
533*49cdfc7eSAndroid Build Coastguard Worker 	dev_bytes_written = (dev_sec_write - prev_dev_sec_write) * 512;
534*49cdfc7eSAndroid Build Coastguard Worker 
535*49cdfc7eSAndroid Build Coastguard Worker 	prev_dev_sec_write = dev_sec_write;
536*49cdfc7eSAndroid Build Coastguard Worker 
537*49cdfc7eSAndroid Build Coastguard Worker 	return dev_bytes_written;
538*49cdfc7eSAndroid Build Coastguard Worker }
539*49cdfc7eSAndroid Build Coastguard Worker 
540*49cdfc7eSAndroid Build Coastguard Worker __attribute__((nonnull))
tst_find_backing_dev(const char * path,char * dev,size_t dev_size)541*49cdfc7eSAndroid Build Coastguard Worker void tst_find_backing_dev(const char *path, char *dev, size_t dev_size)
542*49cdfc7eSAndroid Build Coastguard Worker {
543*49cdfc7eSAndroid Build Coastguard Worker 	struct stat buf;
544*49cdfc7eSAndroid Build Coastguard Worker 	struct btrfs_ioctl_fs_info_args args = {0};
545*49cdfc7eSAndroid Build Coastguard Worker 	struct dirent *d;
546*49cdfc7eSAndroid Build Coastguard Worker 	char uevent_path[PATH_MAX+PATH_MAX+10]; //10 is for the static uevent path
547*49cdfc7eSAndroid Build Coastguard Worker 	char dev_name[NAME_MAX];
548*49cdfc7eSAndroid Build Coastguard Worker 	char bdev_path[PATH_MAX];
549*49cdfc7eSAndroid Build Coastguard Worker 	char tmp_path[PATH_MAX];
550*49cdfc7eSAndroid Build Coastguard Worker 	char btrfs_uuid_str[UUID_STR_SZ];
551*49cdfc7eSAndroid Build Coastguard Worker 	DIR *dir;
552*49cdfc7eSAndroid Build Coastguard Worker 	unsigned int dev_major, dev_minor;
553*49cdfc7eSAndroid Build Coastguard Worker 	int fd;
554*49cdfc7eSAndroid Build Coastguard Worker 
555*49cdfc7eSAndroid Build Coastguard Worker 	if (stat(path, &buf) < 0)
556*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TWARN | TERRNO, NULL, "stat() failed");
557*49cdfc7eSAndroid Build Coastguard Worker 
558*49cdfc7eSAndroid Build Coastguard Worker 	strncpy(tmp_path, path, PATH_MAX-1);
559*49cdfc7eSAndroid Build Coastguard Worker 	tmp_path[PATH_MAX-1] = '\0';
560*49cdfc7eSAndroid Build Coastguard Worker 	if (S_ISREG(buf.st_mode))
561*49cdfc7eSAndroid Build Coastguard Worker 		dirname(tmp_path);
562*49cdfc7eSAndroid Build Coastguard Worker 
563*49cdfc7eSAndroid Build Coastguard Worker 	dev_major = major(buf.st_dev);
564*49cdfc7eSAndroid Build Coastguard Worker 	dev_minor = minor(buf.st_dev);
565*49cdfc7eSAndroid Build Coastguard Worker 	*dev = '\0';
566*49cdfc7eSAndroid Build Coastguard Worker 
567*49cdfc7eSAndroid Build Coastguard Worker 	if (dev_major == 0) {
568*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "Use BTRFS specific strategy");
569*49cdfc7eSAndroid Build Coastguard Worker 
570*49cdfc7eSAndroid Build Coastguard Worker 		fd = SAFE_OPEN(NULL, tmp_path, O_DIRECTORY);
571*49cdfc7eSAndroid Build Coastguard Worker 		if (!ioctl(fd, BTRFS_IOC_FS_INFO, &args)) {
572*49cdfc7eSAndroid Build Coastguard Worker 			sprintf(btrfs_uuid_str,
573*49cdfc7eSAndroid Build Coastguard Worker 				UUID_FMT,
574*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[0], args.fsid[1],
575*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[2], args.fsid[3],
576*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[4], args.fsid[5],
577*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[6], args.fsid[7],
578*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[8], args.fsid[9],
579*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[10], args.fsid[11],
580*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[12], args.fsid[13],
581*49cdfc7eSAndroid Build Coastguard Worker 				args.fsid[14], args.fsid[15]);
582*49cdfc7eSAndroid Build Coastguard Worker 			sprintf(bdev_path,
583*49cdfc7eSAndroid Build Coastguard Worker 				"/sys/fs/btrfs/%s/devices", btrfs_uuid_str);
584*49cdfc7eSAndroid Build Coastguard Worker 		} else {
585*49cdfc7eSAndroid Build Coastguard Worker 			if (errno == ENOTTY)
586*49cdfc7eSAndroid Build Coastguard Worker 				tst_brkm(TBROK | TERRNO, NULL, "BTRFS ioctl failed. Is %s on a tmpfs?", path);
587*49cdfc7eSAndroid Build Coastguard Worker 
588*49cdfc7eSAndroid Build Coastguard Worker 			tst_brkm(TBROK | TERRNO, NULL, "BTRFS ioctl on %s failed.", tmp_path);
589*49cdfc7eSAndroid Build Coastguard Worker 		}
590*49cdfc7eSAndroid Build Coastguard Worker 		SAFE_CLOSE(NULL, fd);
591*49cdfc7eSAndroid Build Coastguard Worker 
592*49cdfc7eSAndroid Build Coastguard Worker 		dir = SAFE_OPENDIR(NULL, bdev_path);
593*49cdfc7eSAndroid Build Coastguard Worker 		while ((d = SAFE_READDIR(NULL, dir))) {
594*49cdfc7eSAndroid Build Coastguard Worker 			if (d->d_name[0] != '.')
595*49cdfc7eSAndroid Build Coastguard Worker 				break;
596*49cdfc7eSAndroid Build Coastguard Worker 		}
597*49cdfc7eSAndroid Build Coastguard Worker 
598*49cdfc7eSAndroid Build Coastguard Worker 		uevent_path[0] = '\0';
599*49cdfc7eSAndroid Build Coastguard Worker 
600*49cdfc7eSAndroid Build Coastguard Worker 		if (d) {
601*49cdfc7eSAndroid Build Coastguard Worker 			sprintf(uevent_path, "%s/%s/uevent",
602*49cdfc7eSAndroid Build Coastguard Worker 				bdev_path, d->d_name);
603*49cdfc7eSAndroid Build Coastguard Worker 		} else {
604*49cdfc7eSAndroid Build Coastguard Worker 			tst_brkm(TBROK | TERRNO, NULL, "No backing device found while looking in %s.", bdev_path);
605*49cdfc7eSAndroid Build Coastguard Worker 		}
606*49cdfc7eSAndroid Build Coastguard Worker 
607*49cdfc7eSAndroid Build Coastguard Worker 		if (SAFE_READDIR(NULL, dir))
608*49cdfc7eSAndroid Build Coastguard Worker 			tst_resm(TINFO, "Warning: used first of multiple backing device.");
609*49cdfc7eSAndroid Build Coastguard Worker 
610*49cdfc7eSAndroid Build Coastguard Worker 		SAFE_CLOSEDIR(NULL, dir);
611*49cdfc7eSAndroid Build Coastguard Worker 	} else {
612*49cdfc7eSAndroid Build Coastguard Worker 		tst_resm(TINFO, "Use uevent strategy");
613*49cdfc7eSAndroid Build Coastguard Worker 		sprintf(uevent_path,
614*49cdfc7eSAndroid Build Coastguard Worker 			"/sys/dev/block/%d:%d/uevent", dev_major, dev_minor);
615*49cdfc7eSAndroid Build Coastguard Worker 	}
616*49cdfc7eSAndroid Build Coastguard Worker 
617*49cdfc7eSAndroid Build Coastguard Worker 	if (!access(uevent_path, R_OK)) {
618*49cdfc7eSAndroid Build Coastguard Worker 		FILE_LINES_SCANF(NULL, uevent_path, "DEVNAME=%s", dev_name);
619*49cdfc7eSAndroid Build Coastguard Worker 
620*49cdfc7eSAndroid Build Coastguard Worker 		if (!dev_name[0] || set_dev_path(dev_name, dev, dev_size))
621*49cdfc7eSAndroid Build Coastguard Worker 			tst_brkm(TBROK, NULL, "Could not stat backing device %s", dev);
622*49cdfc7eSAndroid Build Coastguard Worker 
623*49cdfc7eSAndroid Build Coastguard Worker 	} else {
624*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK, NULL, "uevent file (%s) access failed", uevent_path);
625*49cdfc7eSAndroid Build Coastguard Worker 	}
626*49cdfc7eSAndroid Build Coastguard Worker }
627*49cdfc7eSAndroid Build Coastguard Worker 
tst_stat_mount_dev(const char * const mnt_path,struct stat * const st)628*49cdfc7eSAndroid Build Coastguard Worker void tst_stat_mount_dev(const char *const mnt_path, struct stat *const st)
629*49cdfc7eSAndroid Build Coastguard Worker {
630*49cdfc7eSAndroid Build Coastguard Worker 	struct mntent *mnt;
631*49cdfc7eSAndroid Build Coastguard Worker 	FILE *mntf = setmntent("/proc/self/mounts", "r");
632*49cdfc7eSAndroid Build Coastguard Worker 
633*49cdfc7eSAndroid Build Coastguard Worker 	if (!mntf) {
634*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK | TERRNO, NULL, "Can't open /proc/self/mounts");
635*49cdfc7eSAndroid Build Coastguard Worker 		return;
636*49cdfc7eSAndroid Build Coastguard Worker 	}
637*49cdfc7eSAndroid Build Coastguard Worker 
638*49cdfc7eSAndroid Build Coastguard Worker 	mnt = getmntent(mntf);
639*49cdfc7eSAndroid Build Coastguard Worker 	if (!mnt) {
640*49cdfc7eSAndroid Build Coastguard Worker 		tst_brkm(TBROK | TERRNO, NULL, "Can't read mounts or no mounts?");
641*49cdfc7eSAndroid Build Coastguard Worker 		return;
642*49cdfc7eSAndroid Build Coastguard Worker 	}
643*49cdfc7eSAndroid Build Coastguard Worker 
644*49cdfc7eSAndroid Build Coastguard Worker 	do {
645*49cdfc7eSAndroid Build Coastguard Worker 		if (strcmp(mnt->mnt_dir, mnt_path)) {
646*49cdfc7eSAndroid Build Coastguard Worker 			mnt = getmntent(mntf);
647*49cdfc7eSAndroid Build Coastguard Worker 			continue;
648*49cdfc7eSAndroid Build Coastguard Worker 		}
649*49cdfc7eSAndroid Build Coastguard Worker 
650*49cdfc7eSAndroid Build Coastguard Worker 		if (stat(mnt->mnt_fsname, st)) {
651*49cdfc7eSAndroid Build Coastguard Worker 			tst_brkm(TBROK | TERRNO, NULL,
652*49cdfc7eSAndroid Build Coastguard Worker 				 "Can't stat '%s', mounted at '%s'",
653*49cdfc7eSAndroid Build Coastguard Worker 				 mnt->mnt_fsname, mnt_path);
654*49cdfc7eSAndroid Build Coastguard Worker 		}
655*49cdfc7eSAndroid Build Coastguard Worker 
656*49cdfc7eSAndroid Build Coastguard Worker 		return;
657*49cdfc7eSAndroid Build Coastguard Worker 	} while (mnt);
658*49cdfc7eSAndroid Build Coastguard Worker 
659*49cdfc7eSAndroid Build Coastguard Worker 	tst_brkm(TBROK, NULL, "Could not find mount device");
660*49cdfc7eSAndroid Build Coastguard Worker }
661*49cdfc7eSAndroid Build Coastguard Worker 
tst_dev_block_size(const char * path)662*49cdfc7eSAndroid Build Coastguard Worker int tst_dev_block_size(const char *path)
663*49cdfc7eSAndroid Build Coastguard Worker {
664*49cdfc7eSAndroid Build Coastguard Worker 	int fd;
665*49cdfc7eSAndroid Build Coastguard Worker 	int size;
666*49cdfc7eSAndroid Build Coastguard Worker 	char dev_name[PATH_MAX];
667*49cdfc7eSAndroid Build Coastguard Worker 
668*49cdfc7eSAndroid Build Coastguard Worker 	tst_find_backing_dev(path, dev_name, sizeof(dev_name));
669*49cdfc7eSAndroid Build Coastguard Worker 
670*49cdfc7eSAndroid Build Coastguard Worker 	fd = SAFE_OPEN(NULL, dev_name, O_RDONLY);
671*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_IOCTL(NULL, fd, BLKSSZGET, &size);
672*49cdfc7eSAndroid Build Coastguard Worker 	SAFE_CLOSE(NULL, fd);
673*49cdfc7eSAndroid Build Coastguard Worker 
674*49cdfc7eSAndroid Build Coastguard Worker 	return size;
675*49cdfc7eSAndroid Build Coastguard Worker }
676