xref: /aosp_15_r20/external/f2fs-tools/mkfs/f2fs_format_utils.c (revision 59bfda1f02d633cd6b8b69f31eee485d40f6eef6)
1*59bfda1fSAndroid Build Coastguard Worker /**
2*59bfda1fSAndroid Build Coastguard Worker  * f2fs_format_utils.c
3*59bfda1fSAndroid Build Coastguard Worker  *
4*59bfda1fSAndroid Build Coastguard Worker  * Copyright (c) 2014 Samsung Electronics Co., Ltd.
5*59bfda1fSAndroid Build Coastguard Worker  *             http://www.samsung.com/
6*59bfda1fSAndroid Build Coastguard Worker  *
7*59bfda1fSAndroid Build Coastguard Worker  * Dual licensed under the GPL or LGPL version 2 licenses.
8*59bfda1fSAndroid Build Coastguard Worker  */
9*59bfda1fSAndroid Build Coastguard Worker #ifndef _GNU_SOURCE
10*59bfda1fSAndroid Build Coastguard Worker #define _GNU_SOURCE
11*59bfda1fSAndroid Build Coastguard Worker #endif
12*59bfda1fSAndroid Build Coastguard Worker 
13*59bfda1fSAndroid Build Coastguard Worker #include <f2fs_fs.h>
14*59bfda1fSAndroid Build Coastguard Worker 
15*59bfda1fSAndroid Build Coastguard Worker #include <stdio.h>
16*59bfda1fSAndroid Build Coastguard Worker #include <unistd.h>
17*59bfda1fSAndroid Build Coastguard Worker #include <stdlib.h>
18*59bfda1fSAndroid Build Coastguard Worker #include <stdbool.h>
19*59bfda1fSAndroid Build Coastguard Worker #ifdef HAVE_SYS_IOCTL_H
20*59bfda1fSAndroid Build Coastguard Worker #include <sys/ioctl.h>
21*59bfda1fSAndroid Build Coastguard Worker #endif
22*59bfda1fSAndroid Build Coastguard Worker #include <sys/stat.h>
23*59bfda1fSAndroid Build Coastguard Worker #include <fcntl.h>
24*59bfda1fSAndroid Build Coastguard Worker 
25*59bfda1fSAndroid Build Coastguard Worker #ifdef HAVE_LINUX_FS_H
26*59bfda1fSAndroid Build Coastguard Worker #include <linux/fs.h>
27*59bfda1fSAndroid Build Coastguard Worker #endif
28*59bfda1fSAndroid Build Coastguard Worker #ifdef HAVE_LINUX_FALLOC_H
29*59bfda1fSAndroid Build Coastguard Worker #include <linux/falloc.h>
30*59bfda1fSAndroid Build Coastguard Worker #endif
31*59bfda1fSAndroid Build Coastguard Worker 
32*59bfda1fSAndroid Build Coastguard Worker #ifdef __linux__
33*59bfda1fSAndroid Build Coastguard Worker #ifndef BLKDISCARD
34*59bfda1fSAndroid Build Coastguard Worker #define BLKDISCARD	_IO(0x12,119)
35*59bfda1fSAndroid Build Coastguard Worker #endif
36*59bfda1fSAndroid Build Coastguard Worker #ifndef BLKSECDISCARD
37*59bfda1fSAndroid Build Coastguard Worker #define BLKSECDISCARD	_IO(0x12,125)
38*59bfda1fSAndroid Build Coastguard Worker #endif
39*59bfda1fSAndroid Build Coastguard Worker #endif
40*59bfda1fSAndroid Build Coastguard Worker 
41*59bfda1fSAndroid Build Coastguard Worker #if defined(FALLOC_FL_PUNCH_HOLE) || defined(BLKDISCARD) || \
42*59bfda1fSAndroid Build Coastguard Worker 	defined(BLKSECDISCARD)
trim_device(int i)43*59bfda1fSAndroid Build Coastguard Worker static int trim_device(int i)
44*59bfda1fSAndroid Build Coastguard Worker {
45*59bfda1fSAndroid Build Coastguard Worker 	unsigned long long range[2];
46*59bfda1fSAndroid Build Coastguard Worker 	struct stat *stat_buf;
47*59bfda1fSAndroid Build Coastguard Worker 	struct device_info *dev = c.devices + i;
48*59bfda1fSAndroid Build Coastguard Worker 	uint64_t bytes = dev->total_sectors * dev->sector_size;
49*59bfda1fSAndroid Build Coastguard Worker 	int fd = dev->fd;
50*59bfda1fSAndroid Build Coastguard Worker 
51*59bfda1fSAndroid Build Coastguard Worker 	if (dev->alias_filename) {
52*59bfda1fSAndroid Build Coastguard Worker 		MSG(0, "Info: [%s] Skip Discarding as aliased\n", dev->path);
53*59bfda1fSAndroid Build Coastguard Worker 		return 0;
54*59bfda1fSAndroid Build Coastguard Worker 	}
55*59bfda1fSAndroid Build Coastguard Worker 
56*59bfda1fSAndroid Build Coastguard Worker 	stat_buf = malloc(sizeof(struct stat));
57*59bfda1fSAndroid Build Coastguard Worker 	if (stat_buf == NULL) {
58*59bfda1fSAndroid Build Coastguard Worker 		MSG(1, "\tError: Malloc Failed for trim_stat_buf!!!\n");
59*59bfda1fSAndroid Build Coastguard Worker 		return -1;
60*59bfda1fSAndroid Build Coastguard Worker 	}
61*59bfda1fSAndroid Build Coastguard Worker 
62*59bfda1fSAndroid Build Coastguard Worker 	if (fstat(fd, stat_buf) < 0 ) {
63*59bfda1fSAndroid Build Coastguard Worker 		MSG(1, "\tError: Failed to get the device stat!!!\n");
64*59bfda1fSAndroid Build Coastguard Worker 		free(stat_buf);
65*59bfda1fSAndroid Build Coastguard Worker 		return -1;
66*59bfda1fSAndroid Build Coastguard Worker 	}
67*59bfda1fSAndroid Build Coastguard Worker 
68*59bfda1fSAndroid Build Coastguard Worker 	range[0] = 0;
69*59bfda1fSAndroid Build Coastguard Worker 	range[1] = bytes;
70*59bfda1fSAndroid Build Coastguard Worker 
71*59bfda1fSAndroid Build Coastguard Worker #if defined(WITH_BLKDISCARD) && defined(BLKDISCARD)
72*59bfda1fSAndroid Build Coastguard Worker 	MSG(0, "Info: [%s] Discarding device\n", dev->path);
73*59bfda1fSAndroid Build Coastguard Worker 	if (S_ISREG(stat_buf->st_mode)) {
74*59bfda1fSAndroid Build Coastguard Worker #if defined(HAVE_FALLOCATE) && defined(FALLOC_FL_PUNCH_HOLE)
75*59bfda1fSAndroid Build Coastguard Worker 		if (fallocate(fd, FALLOC_FL_PUNCH_HOLE | FALLOC_FL_KEEP_SIZE,
76*59bfda1fSAndroid Build Coastguard Worker 				range[0], range[1]) < 0) {
77*59bfda1fSAndroid Build Coastguard Worker 			MSG(0, "Info: fallocate(PUNCH_HOLE|KEEP_SIZE) is failed\n");
78*59bfda1fSAndroid Build Coastguard Worker 		}
79*59bfda1fSAndroid Build Coastguard Worker #endif
80*59bfda1fSAndroid Build Coastguard Worker 		free(stat_buf);
81*59bfda1fSAndroid Build Coastguard Worker 		return 0;
82*59bfda1fSAndroid Build Coastguard Worker 	} else if (S_ISBLK(stat_buf->st_mode)) {
83*59bfda1fSAndroid Build Coastguard Worker 		if (dev->zoned_model != F2FS_ZONED_NONE) {
84*59bfda1fSAndroid Build Coastguard Worker 			free(stat_buf);
85*59bfda1fSAndroid Build Coastguard Worker 			return f2fs_reset_zones(i);
86*59bfda1fSAndroid Build Coastguard Worker 		}
87*59bfda1fSAndroid Build Coastguard Worker #ifdef BLKSECDISCARD
88*59bfda1fSAndroid Build Coastguard Worker 		if (ioctl(fd, BLKSECDISCARD, &range) < 0) {
89*59bfda1fSAndroid Build Coastguard Worker 			MSG(0, "Info: This device doesn't support BLKSECDISCARD\n");
90*59bfda1fSAndroid Build Coastguard Worker 		} else {
91*59bfda1fSAndroid Build Coastguard Worker 			MSG(0, "Info: Secure Discarded %lu MB\n",
92*59bfda1fSAndroid Build Coastguard Worker 					(unsigned long)stat_buf->st_size >> 20);
93*59bfda1fSAndroid Build Coastguard Worker 			free(stat_buf);
94*59bfda1fSAndroid Build Coastguard Worker 			return 0;
95*59bfda1fSAndroid Build Coastguard Worker 		}
96*59bfda1fSAndroid Build Coastguard Worker #endif
97*59bfda1fSAndroid Build Coastguard Worker 		if (ioctl(fd, BLKDISCARD, &range) < 0) {
98*59bfda1fSAndroid Build Coastguard Worker 			MSG(0, "Info: This device doesn't support BLKDISCARD\n");
99*59bfda1fSAndroid Build Coastguard Worker 		} else {
100*59bfda1fSAndroid Build Coastguard Worker 			MSG(0, "Info: Discarded %llu MB\n", range[1] >> 20);
101*59bfda1fSAndroid Build Coastguard Worker 		}
102*59bfda1fSAndroid Build Coastguard Worker 	} else {
103*59bfda1fSAndroid Build Coastguard Worker 		free(stat_buf);
104*59bfda1fSAndroid Build Coastguard Worker 		return -1;
105*59bfda1fSAndroid Build Coastguard Worker 	}
106*59bfda1fSAndroid Build Coastguard Worker #endif
107*59bfda1fSAndroid Build Coastguard Worker 	free(stat_buf);
108*59bfda1fSAndroid Build Coastguard Worker 	return 0;
109*59bfda1fSAndroid Build Coastguard Worker }
110*59bfda1fSAndroid Build Coastguard Worker #else
trim_device(int UNUSED (i))111*59bfda1fSAndroid Build Coastguard Worker static int trim_device(int UNUSED(i))
112*59bfda1fSAndroid Build Coastguard Worker {
113*59bfda1fSAndroid Build Coastguard Worker 	return 0;
114*59bfda1fSAndroid Build Coastguard Worker }
115*59bfda1fSAndroid Build Coastguard Worker #endif
116*59bfda1fSAndroid Build Coastguard Worker 
117*59bfda1fSAndroid Build Coastguard Worker #ifdef WITH_ANDROID
is_wiped_device(int i)118*59bfda1fSAndroid Build Coastguard Worker static bool is_wiped_device(int i)
119*59bfda1fSAndroid Build Coastguard Worker {
120*59bfda1fSAndroid Build Coastguard Worker 	struct device_info *dev = c.devices + i;
121*59bfda1fSAndroid Build Coastguard Worker 	int fd = dev->fd;
122*59bfda1fSAndroid Build Coastguard Worker 	char *buf, *zero_buf;
123*59bfda1fSAndroid Build Coastguard Worker 	bool wiped = true;
124*59bfda1fSAndroid Build Coastguard Worker 	int nblocks = (4096 * 4096) / F2FS_BLKSIZE;	/* 16MB size */
125*59bfda1fSAndroid Build Coastguard Worker 	int j;
126*59bfda1fSAndroid Build Coastguard Worker 
127*59bfda1fSAndroid Build Coastguard Worker 	/* let's trim the other devices except the first device */
128*59bfda1fSAndroid Build Coastguard Worker 	if (i > 0)
129*59bfda1fSAndroid Build Coastguard Worker 		return false;
130*59bfda1fSAndroid Build Coastguard Worker 
131*59bfda1fSAndroid Build Coastguard Worker 	buf = malloc(F2FS_BLKSIZE);
132*59bfda1fSAndroid Build Coastguard Worker 	if (buf == NULL) {
133*59bfda1fSAndroid Build Coastguard Worker 		MSG(1, "\tError: Malloc Failed for buf!!!\n");
134*59bfda1fSAndroid Build Coastguard Worker 		return false;
135*59bfda1fSAndroid Build Coastguard Worker 	}
136*59bfda1fSAndroid Build Coastguard Worker 	zero_buf = calloc(1, F2FS_BLKSIZE);
137*59bfda1fSAndroid Build Coastguard Worker 	if (zero_buf == NULL) {
138*59bfda1fSAndroid Build Coastguard Worker 		MSG(1, "\tError: Calloc Failed for zero buf!!!\n");
139*59bfda1fSAndroid Build Coastguard Worker 		free(buf);
140*59bfda1fSAndroid Build Coastguard Worker 		return false;
141*59bfda1fSAndroid Build Coastguard Worker 	}
142*59bfda1fSAndroid Build Coastguard Worker 
143*59bfda1fSAndroid Build Coastguard Worker 	if (lseek(fd, 0, SEEK_SET) < 0) {
144*59bfda1fSAndroid Build Coastguard Worker 		free(zero_buf);
145*59bfda1fSAndroid Build Coastguard Worker 		free(buf);
146*59bfda1fSAndroid Build Coastguard Worker 		return false;
147*59bfda1fSAndroid Build Coastguard Worker 	}
148*59bfda1fSAndroid Build Coastguard Worker 
149*59bfda1fSAndroid Build Coastguard Worker 	/* check first n blocks */
150*59bfda1fSAndroid Build Coastguard Worker 	for (j = 0; j < nblocks; j++) {
151*59bfda1fSAndroid Build Coastguard Worker 		if (read(fd, buf, F2FS_BLKSIZE) != F2FS_BLKSIZE ||
152*59bfda1fSAndroid Build Coastguard Worker 				memcmp(buf, zero_buf, F2FS_BLKSIZE)) {
153*59bfda1fSAndroid Build Coastguard Worker 			wiped = false;
154*59bfda1fSAndroid Build Coastguard Worker 			break;
155*59bfda1fSAndroid Build Coastguard Worker 		}
156*59bfda1fSAndroid Build Coastguard Worker 	}
157*59bfda1fSAndroid Build Coastguard Worker 	free(zero_buf);
158*59bfda1fSAndroid Build Coastguard Worker 	free(buf);
159*59bfda1fSAndroid Build Coastguard Worker 
160*59bfda1fSAndroid Build Coastguard Worker 	if (wiped)
161*59bfda1fSAndroid Build Coastguard Worker 		MSG(0, "Info: Found all zeros in first %d blocks\n", nblocks);
162*59bfda1fSAndroid Build Coastguard Worker 	return wiped;
163*59bfda1fSAndroid Build Coastguard Worker }
164*59bfda1fSAndroid Build Coastguard Worker #else
is_wiped_device(int UNUSED (i))165*59bfda1fSAndroid Build Coastguard Worker static bool is_wiped_device(int UNUSED(i))
166*59bfda1fSAndroid Build Coastguard Worker {
167*59bfda1fSAndroid Build Coastguard Worker 	return false;
168*59bfda1fSAndroid Build Coastguard Worker }
169*59bfda1fSAndroid Build Coastguard Worker #endif
170*59bfda1fSAndroid Build Coastguard Worker 
f2fs_trim_devices(void)171*59bfda1fSAndroid Build Coastguard Worker int f2fs_trim_devices(void)
172*59bfda1fSAndroid Build Coastguard Worker {
173*59bfda1fSAndroid Build Coastguard Worker 	int i;
174*59bfda1fSAndroid Build Coastguard Worker 
175*59bfda1fSAndroid Build Coastguard Worker 	for (i = 0; i < c.ndevs; i++) {
176*59bfda1fSAndroid Build Coastguard Worker 		if (!is_wiped_device(i) && trim_device(i))
177*59bfda1fSAndroid Build Coastguard Worker 			return -1;
178*59bfda1fSAndroid Build Coastguard Worker 	}
179*59bfda1fSAndroid Build Coastguard Worker 	c.trimmed = 1;
180*59bfda1fSAndroid Build Coastguard Worker 	return 0;
181*59bfda1fSAndroid Build Coastguard Worker }
182