xref: /aosp_15_r20/external/erofs-utils/fsck/main.c (revision 33b1fccf6a0fada2c2875d400ed01119b7676ee5)
1*33b1fccfSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0+
2*33b1fccfSAndroid Build Coastguard Worker /*
3*33b1fccfSAndroid Build Coastguard Worker  * Copyright 2021 Google LLC
4*33b1fccfSAndroid Build Coastguard Worker  * Author: Daeho Jeong <[email protected]>
5*33b1fccfSAndroid Build Coastguard Worker  */
6*33b1fccfSAndroid Build Coastguard Worker #include <stdlib.h>
7*33b1fccfSAndroid Build Coastguard Worker #include <getopt.h>
8*33b1fccfSAndroid Build Coastguard Worker #include <time.h>
9*33b1fccfSAndroid Build Coastguard Worker #include <utime.h>
10*33b1fccfSAndroid Build Coastguard Worker #include <unistd.h>
11*33b1fccfSAndroid Build Coastguard Worker #include <sys/stat.h>
12*33b1fccfSAndroid Build Coastguard Worker #include "erofs/print.h"
13*33b1fccfSAndroid Build Coastguard Worker #include "erofs/compress.h"
14*33b1fccfSAndroid Build Coastguard Worker #include "erofs/decompress.h"
15*33b1fccfSAndroid Build Coastguard Worker #include "erofs/dir.h"
16*33b1fccfSAndroid Build Coastguard Worker #include "../lib/compressor.h"
17*33b1fccfSAndroid Build Coastguard Worker 
18*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_check_inode(erofs_nid_t pnid, erofs_nid_t nid);
19*33b1fccfSAndroid Build Coastguard Worker 
20*33b1fccfSAndroid Build Coastguard Worker struct erofsfsck_cfg {
21*33b1fccfSAndroid Build Coastguard Worker 	u64 physical_blocks;
22*33b1fccfSAndroid Build Coastguard Worker 	u64 logical_blocks;
23*33b1fccfSAndroid Build Coastguard Worker 	char *extract_path;
24*33b1fccfSAndroid Build Coastguard Worker 	size_t extract_pos;
25*33b1fccfSAndroid Build Coastguard Worker 	mode_t umask;
26*33b1fccfSAndroid Build Coastguard Worker 	bool superuser;
27*33b1fccfSAndroid Build Coastguard Worker 	bool corrupted;
28*33b1fccfSAndroid Build Coastguard Worker 	bool print_comp_ratio;
29*33b1fccfSAndroid Build Coastguard Worker 	bool check_decomp;
30*33b1fccfSAndroid Build Coastguard Worker 	bool force;
31*33b1fccfSAndroid Build Coastguard Worker 	bool overwrite;
32*33b1fccfSAndroid Build Coastguard Worker 	bool preserve_owner;
33*33b1fccfSAndroid Build Coastguard Worker 	bool preserve_perms;
34*33b1fccfSAndroid Build Coastguard Worker };
35*33b1fccfSAndroid Build Coastguard Worker static struct erofsfsck_cfg fsckcfg;
36*33b1fccfSAndroid Build Coastguard Worker 
37*33b1fccfSAndroid Build Coastguard Worker static struct option long_options[] = {
38*33b1fccfSAndroid Build Coastguard Worker 	{"version", no_argument, 0, 'V'},
39*33b1fccfSAndroid Build Coastguard Worker 	{"help", no_argument, 0, 'h'},
40*33b1fccfSAndroid Build Coastguard Worker 	{"extract", optional_argument, 0, 2},
41*33b1fccfSAndroid Build Coastguard Worker 	{"device", required_argument, 0, 3},
42*33b1fccfSAndroid Build Coastguard Worker 	{"force", no_argument, 0, 4},
43*33b1fccfSAndroid Build Coastguard Worker 	{"overwrite", no_argument, 0, 5},
44*33b1fccfSAndroid Build Coastguard Worker 	{"preserve", no_argument, 0, 6},
45*33b1fccfSAndroid Build Coastguard Worker 	{"preserve-owner", no_argument, 0, 7},
46*33b1fccfSAndroid Build Coastguard Worker 	{"preserve-perms", no_argument, 0, 8},
47*33b1fccfSAndroid Build Coastguard Worker 	{"no-preserve", no_argument, 0, 9},
48*33b1fccfSAndroid Build Coastguard Worker 	{"no-preserve-owner", no_argument, 0, 10},
49*33b1fccfSAndroid Build Coastguard Worker 	{"no-preserve-perms", no_argument, 0, 11},
50*33b1fccfSAndroid Build Coastguard Worker 	{"offset", required_argument, 0, 12},
51*33b1fccfSAndroid Build Coastguard Worker 	{0, 0, 0, 0},
52*33b1fccfSAndroid Build Coastguard Worker };
53*33b1fccfSAndroid Build Coastguard Worker 
54*33b1fccfSAndroid Build Coastguard Worker #define NR_HARDLINK_HASHTABLE	16384
55*33b1fccfSAndroid Build Coastguard Worker 
56*33b1fccfSAndroid Build Coastguard Worker struct erofsfsck_hardlink_entry {
57*33b1fccfSAndroid Build Coastguard Worker 	struct list_head list;
58*33b1fccfSAndroid Build Coastguard Worker 	erofs_nid_t nid;
59*33b1fccfSAndroid Build Coastguard Worker 	char *path;
60*33b1fccfSAndroid Build Coastguard Worker };
61*33b1fccfSAndroid Build Coastguard Worker 
62*33b1fccfSAndroid Build Coastguard Worker static struct list_head erofsfsck_link_hashtable[NR_HARDLINK_HASHTABLE];
63*33b1fccfSAndroid Build Coastguard Worker 
print_available_decompressors(FILE * f,const char * delim)64*33b1fccfSAndroid Build Coastguard Worker static void print_available_decompressors(FILE *f, const char *delim)
65*33b1fccfSAndroid Build Coastguard Worker {
66*33b1fccfSAndroid Build Coastguard Worker 	int i = 0;
67*33b1fccfSAndroid Build Coastguard Worker 	bool comma = false;
68*33b1fccfSAndroid Build Coastguard Worker 	const struct erofs_algorithm *s;
69*33b1fccfSAndroid Build Coastguard Worker 
70*33b1fccfSAndroid Build Coastguard Worker 	while ((s = z_erofs_list_available_compressors(&i)) != NULL) {
71*33b1fccfSAndroid Build Coastguard Worker 		if (comma)
72*33b1fccfSAndroid Build Coastguard Worker 			fputs(delim, f);
73*33b1fccfSAndroid Build Coastguard Worker 		fputs(s->name, f);
74*33b1fccfSAndroid Build Coastguard Worker 		comma = true;
75*33b1fccfSAndroid Build Coastguard Worker 	}
76*33b1fccfSAndroid Build Coastguard Worker 	fputc('\n', f);
77*33b1fccfSAndroid Build Coastguard Worker }
78*33b1fccfSAndroid Build Coastguard Worker 
usage(int argc,char ** argv)79*33b1fccfSAndroid Build Coastguard Worker static void usage(int argc, char **argv)
80*33b1fccfSAndroid Build Coastguard Worker {
81*33b1fccfSAndroid Build Coastguard Worker 	//	"         1         2         3         4         5         6         7         8  "
82*33b1fccfSAndroid Build Coastguard Worker 	//	"12345678901234567890123456789012345678901234567890123456789012345678901234567890\n"
83*33b1fccfSAndroid Build Coastguard Worker 	printf(
84*33b1fccfSAndroid Build Coastguard Worker 		"Usage: %s [OPTIONS] IMAGE\n"
85*33b1fccfSAndroid Build Coastguard Worker 		"Check erofs filesystem compatibility and integrity of IMAGE.\n"
86*33b1fccfSAndroid Build Coastguard Worker 		"\n"
87*33b1fccfSAndroid Build Coastguard Worker 		"This version of fsck.erofs is capable of checking images that use any of the\n"
88*33b1fccfSAndroid Build Coastguard Worker 		"following algorithms: ", argv[0]);
89*33b1fccfSAndroid Build Coastguard Worker 	print_available_decompressors(stdout, ", ");
90*33b1fccfSAndroid Build Coastguard Worker 	printf("\n"
91*33b1fccfSAndroid Build Coastguard Worker 		"General options:\n"
92*33b1fccfSAndroid Build Coastguard Worker 		" -V, --version          print the version number of fsck.erofs and exit\n"
93*33b1fccfSAndroid Build Coastguard Worker 		" -h, --help             display this help and exit\n"
94*33b1fccfSAndroid Build Coastguard Worker 		"\n"
95*33b1fccfSAndroid Build Coastguard Worker 		" -d<0-9>                set output verbosity; 0=quiet, 9=verbose (default=%i)\n"
96*33b1fccfSAndroid Build Coastguard Worker 		" -p                     print total compression ratio of all files\n"
97*33b1fccfSAndroid Build Coastguard Worker 		" --device=X             specify an extra device to be used together\n"
98*33b1fccfSAndroid Build Coastguard Worker 		" --extract[=X]          check if all files are well encoded, optionally\n"
99*33b1fccfSAndroid Build Coastguard Worker 		"                        extract to X\n"
100*33b1fccfSAndroid Build Coastguard Worker 		" --offset=#             skip # bytes at the beginning of IMAGE\n"
101*33b1fccfSAndroid Build Coastguard Worker 		"\n"
102*33b1fccfSAndroid Build Coastguard Worker 		" -a, -A, -y             no-op, for compatibility with fsck of other filesystems\n"
103*33b1fccfSAndroid Build Coastguard Worker 		"\n"
104*33b1fccfSAndroid Build Coastguard Worker 		"Extraction options (--extract=X is required):\n"
105*33b1fccfSAndroid Build Coastguard Worker 		" --force                allow extracting to root\n"
106*33b1fccfSAndroid Build Coastguard Worker 		" --overwrite            overwrite files that already exist\n"
107*33b1fccfSAndroid Build Coastguard Worker 		" --[no-]preserve        same as --[no-]preserve-owner --[no-]preserve-perms\n"
108*33b1fccfSAndroid Build Coastguard Worker 		" --[no-]preserve-owner  whether to preserve the ownership from the\n"
109*33b1fccfSAndroid Build Coastguard Worker 		"                        filesystem (default for superuser), or to extract as\n"
110*33b1fccfSAndroid Build Coastguard Worker 		"                        yourself (default for ordinary users)\n"
111*33b1fccfSAndroid Build Coastguard Worker 		" --[no-]preserve-perms  whether to preserve the exact permissions from the\n"
112*33b1fccfSAndroid Build Coastguard Worker 		"                        filesystem without applying umask (default for\n"
113*33b1fccfSAndroid Build Coastguard Worker 		"                        superuser), or to modify the permissions by applying\n"
114*33b1fccfSAndroid Build Coastguard Worker 		"                        umask (default for ordinary users)\n",
115*33b1fccfSAndroid Build Coastguard Worker 		EROFS_WARN);
116*33b1fccfSAndroid Build Coastguard Worker }
117*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_print_version(void)118*33b1fccfSAndroid Build Coastguard Worker static void erofsfsck_print_version(void)
119*33b1fccfSAndroid Build Coastguard Worker {
120*33b1fccfSAndroid Build Coastguard Worker 	printf("fsck.erofs (erofs-utils) %s\navailable decompressors: ",
121*33b1fccfSAndroid Build Coastguard Worker 	       cfg.c_version);
122*33b1fccfSAndroid Build Coastguard Worker 	print_available_decompressors(stdout, ", ");
123*33b1fccfSAndroid Build Coastguard Worker }
124*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_parse_options_cfg(int argc,char ** argv)125*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_parse_options_cfg(int argc, char **argv)
126*33b1fccfSAndroid Build Coastguard Worker {
127*33b1fccfSAndroid Build Coastguard Worker 	char *endptr;
128*33b1fccfSAndroid Build Coastguard Worker 	int opt, ret;
129*33b1fccfSAndroid Build Coastguard Worker 	bool has_opt_preserve = false;
130*33b1fccfSAndroid Build Coastguard Worker 
131*33b1fccfSAndroid Build Coastguard Worker 	while ((opt = getopt_long(argc, argv, "Vd:phaAy",
132*33b1fccfSAndroid Build Coastguard Worker 				  long_options, NULL)) != -1) {
133*33b1fccfSAndroid Build Coastguard Worker 		switch (opt) {
134*33b1fccfSAndroid Build Coastguard Worker 		case 'V':
135*33b1fccfSAndroid Build Coastguard Worker 			erofsfsck_print_version();
136*33b1fccfSAndroid Build Coastguard Worker 			exit(0);
137*33b1fccfSAndroid Build Coastguard Worker 		case 'd':
138*33b1fccfSAndroid Build Coastguard Worker 			ret = atoi(optarg);
139*33b1fccfSAndroid Build Coastguard Worker 			if (ret < EROFS_MSG_MIN || ret > EROFS_MSG_MAX) {
140*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("invalid debug level %d", ret);
141*33b1fccfSAndroid Build Coastguard Worker 				return -EINVAL;
142*33b1fccfSAndroid Build Coastguard Worker 			}
143*33b1fccfSAndroid Build Coastguard Worker 			cfg.c_dbg_lvl = ret;
144*33b1fccfSAndroid Build Coastguard Worker 			break;
145*33b1fccfSAndroid Build Coastguard Worker 		case 'p':
146*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.print_comp_ratio = true;
147*33b1fccfSAndroid Build Coastguard Worker 			break;
148*33b1fccfSAndroid Build Coastguard Worker 		case 'h':
149*33b1fccfSAndroid Build Coastguard Worker 			usage(argc, argv);
150*33b1fccfSAndroid Build Coastguard Worker 			exit(0);
151*33b1fccfSAndroid Build Coastguard Worker 		case 'a':
152*33b1fccfSAndroid Build Coastguard Worker 		case 'A':
153*33b1fccfSAndroid Build Coastguard Worker 		case 'y':
154*33b1fccfSAndroid Build Coastguard Worker 			break;
155*33b1fccfSAndroid Build Coastguard Worker 		case 2:
156*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.check_decomp = true;
157*33b1fccfSAndroid Build Coastguard Worker 			if (optarg) {
158*33b1fccfSAndroid Build Coastguard Worker 				size_t len = strlen(optarg);
159*33b1fccfSAndroid Build Coastguard Worker 
160*33b1fccfSAndroid Build Coastguard Worker 				if (len == 0) {
161*33b1fccfSAndroid Build Coastguard Worker 					erofs_err("empty value given for --extract=X");
162*33b1fccfSAndroid Build Coastguard Worker 					return -EINVAL;
163*33b1fccfSAndroid Build Coastguard Worker 				}
164*33b1fccfSAndroid Build Coastguard Worker 
165*33b1fccfSAndroid Build Coastguard Worker 				/* remove trailing slashes except root */
166*33b1fccfSAndroid Build Coastguard Worker 				while (len > 1 && optarg[len - 1] == '/')
167*33b1fccfSAndroid Build Coastguard Worker 					len--;
168*33b1fccfSAndroid Build Coastguard Worker 
169*33b1fccfSAndroid Build Coastguard Worker 				if (len >= PATH_MAX) {
170*33b1fccfSAndroid Build Coastguard Worker 					erofs_err("target directory name too long!");
171*33b1fccfSAndroid Build Coastguard Worker 					return -ENAMETOOLONG;
172*33b1fccfSAndroid Build Coastguard Worker 				}
173*33b1fccfSAndroid Build Coastguard Worker 
174*33b1fccfSAndroid Build Coastguard Worker 				fsckcfg.extract_path = malloc(PATH_MAX);
175*33b1fccfSAndroid Build Coastguard Worker 				if (!fsckcfg.extract_path)
176*33b1fccfSAndroid Build Coastguard Worker 					return -ENOMEM;
177*33b1fccfSAndroid Build Coastguard Worker 				strncpy(fsckcfg.extract_path, optarg, len);
178*33b1fccfSAndroid Build Coastguard Worker 				fsckcfg.extract_path[len] = '\0';
179*33b1fccfSAndroid Build Coastguard Worker 				/* if path is root, start writing from position 0 */
180*33b1fccfSAndroid Build Coastguard Worker 				if (len == 1 && fsckcfg.extract_path[0] == '/')
181*33b1fccfSAndroid Build Coastguard Worker 					len = 0;
182*33b1fccfSAndroid Build Coastguard Worker 				fsckcfg.extract_pos = len;
183*33b1fccfSAndroid Build Coastguard Worker 			}
184*33b1fccfSAndroid Build Coastguard Worker 			break;
185*33b1fccfSAndroid Build Coastguard Worker 		case 3:
186*33b1fccfSAndroid Build Coastguard Worker 			ret = erofs_blob_open_ro(&g_sbi, optarg);
187*33b1fccfSAndroid Build Coastguard Worker 			if (ret)
188*33b1fccfSAndroid Build Coastguard Worker 				return ret;
189*33b1fccfSAndroid Build Coastguard Worker 			++g_sbi.extra_devices;
190*33b1fccfSAndroid Build Coastguard Worker 			break;
191*33b1fccfSAndroid Build Coastguard Worker 		case 4:
192*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.force = true;
193*33b1fccfSAndroid Build Coastguard Worker 			break;
194*33b1fccfSAndroid Build Coastguard Worker 		case 5:
195*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.overwrite = true;
196*33b1fccfSAndroid Build Coastguard Worker 			break;
197*33b1fccfSAndroid Build Coastguard Worker 		case 6:
198*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_owner = fsckcfg.preserve_perms = true;
199*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
200*33b1fccfSAndroid Build Coastguard Worker 			break;
201*33b1fccfSAndroid Build Coastguard Worker 		case 7:
202*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_owner = true;
203*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
204*33b1fccfSAndroid Build Coastguard Worker 			break;
205*33b1fccfSAndroid Build Coastguard Worker 		case 8:
206*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_perms = true;
207*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
208*33b1fccfSAndroid Build Coastguard Worker 			break;
209*33b1fccfSAndroid Build Coastguard Worker 		case 9:
210*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_owner = fsckcfg.preserve_perms = false;
211*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
212*33b1fccfSAndroid Build Coastguard Worker 			break;
213*33b1fccfSAndroid Build Coastguard Worker 		case 10:
214*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_owner = false;
215*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
216*33b1fccfSAndroid Build Coastguard Worker 			break;
217*33b1fccfSAndroid Build Coastguard Worker 		case 11:
218*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.preserve_perms = false;
219*33b1fccfSAndroid Build Coastguard Worker 			has_opt_preserve = true;
220*33b1fccfSAndroid Build Coastguard Worker 			break;
221*33b1fccfSAndroid Build Coastguard Worker 		case 12:
222*33b1fccfSAndroid Build Coastguard Worker 			g_sbi.bdev.offset = strtoull(optarg, &endptr, 0);
223*33b1fccfSAndroid Build Coastguard Worker 			if (*endptr != '\0') {
224*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("invalid disk offset %s", optarg);
225*33b1fccfSAndroid Build Coastguard Worker 				return -EINVAL;
226*33b1fccfSAndroid Build Coastguard Worker 			}
227*33b1fccfSAndroid Build Coastguard Worker 			break;
228*33b1fccfSAndroid Build Coastguard Worker 		default:
229*33b1fccfSAndroid Build Coastguard Worker 			return -EINVAL;
230*33b1fccfSAndroid Build Coastguard Worker 		}
231*33b1fccfSAndroid Build Coastguard Worker 	}
232*33b1fccfSAndroid Build Coastguard Worker 
233*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.extract_path) {
234*33b1fccfSAndroid Build Coastguard Worker 		if (!fsckcfg.extract_pos && !fsckcfg.force) {
235*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("--extract=/ must be used together with --force");
236*33b1fccfSAndroid Build Coastguard Worker 			return -EINVAL;
237*33b1fccfSAndroid Build Coastguard Worker 		}
238*33b1fccfSAndroid Build Coastguard Worker 	} else {
239*33b1fccfSAndroid Build Coastguard Worker 		if (fsckcfg.force) {
240*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("--force must be used together with --extract=X");
241*33b1fccfSAndroid Build Coastguard Worker 			return -EINVAL;
242*33b1fccfSAndroid Build Coastguard Worker 		}
243*33b1fccfSAndroid Build Coastguard Worker 		if (fsckcfg.overwrite) {
244*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("--overwrite must be used together with --extract=X");
245*33b1fccfSAndroid Build Coastguard Worker 			return -EINVAL;
246*33b1fccfSAndroid Build Coastguard Worker 		}
247*33b1fccfSAndroid Build Coastguard Worker 		if (has_opt_preserve) {
248*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("--[no-]preserve[-owner/-perms] must be used together with --extract=X");
249*33b1fccfSAndroid Build Coastguard Worker 			return -EINVAL;
250*33b1fccfSAndroid Build Coastguard Worker 		}
251*33b1fccfSAndroid Build Coastguard Worker 	}
252*33b1fccfSAndroid Build Coastguard Worker 
253*33b1fccfSAndroid Build Coastguard Worker 	if (optind >= argc) {
254*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("missing argument: IMAGE");
255*33b1fccfSAndroid Build Coastguard Worker 		return -EINVAL;
256*33b1fccfSAndroid Build Coastguard Worker 	}
257*33b1fccfSAndroid Build Coastguard Worker 
258*33b1fccfSAndroid Build Coastguard Worker 	cfg.c_img_path = strdup(argv[optind++]);
259*33b1fccfSAndroid Build Coastguard Worker 	if (!cfg.c_img_path)
260*33b1fccfSAndroid Build Coastguard Worker 		return -ENOMEM;
261*33b1fccfSAndroid Build Coastguard Worker 
262*33b1fccfSAndroid Build Coastguard Worker 	if (optind < argc) {
263*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("unexpected argument: %s", argv[optind]);
264*33b1fccfSAndroid Build Coastguard Worker 		return -EINVAL;
265*33b1fccfSAndroid Build Coastguard Worker 	}
266*33b1fccfSAndroid Build Coastguard Worker 	return 0;
267*33b1fccfSAndroid Build Coastguard Worker }
268*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_set_attributes(struct erofs_inode * inode,char * path)269*33b1fccfSAndroid Build Coastguard Worker static void erofsfsck_set_attributes(struct erofs_inode *inode, char *path)
270*33b1fccfSAndroid Build Coastguard Worker {
271*33b1fccfSAndroid Build Coastguard Worker 	int ret;
272*33b1fccfSAndroid Build Coastguard Worker 
273*33b1fccfSAndroid Build Coastguard Worker 	/* don't apply attributes when fsck is used without extraction */
274*33b1fccfSAndroid Build Coastguard Worker 	if (!fsckcfg.extract_path)
275*33b1fccfSAndroid Build Coastguard Worker 		return;
276*33b1fccfSAndroid Build Coastguard Worker 
277*33b1fccfSAndroid Build Coastguard Worker #ifdef HAVE_UTIMENSAT
278*33b1fccfSAndroid Build Coastguard Worker 	if (utimensat(AT_FDCWD, path, (struct timespec []) {
279*33b1fccfSAndroid Build Coastguard Worker 				[0] = { .tv_sec = inode->i_mtime,
280*33b1fccfSAndroid Build Coastguard Worker 					.tv_nsec = inode->i_mtime_nsec },
281*33b1fccfSAndroid Build Coastguard Worker 				[1] = { .tv_sec = inode->i_mtime,
282*33b1fccfSAndroid Build Coastguard Worker 					.tv_nsec = inode->i_mtime_nsec },
283*33b1fccfSAndroid Build Coastguard Worker 			}, AT_SYMLINK_NOFOLLOW) < 0)
284*33b1fccfSAndroid Build Coastguard Worker #else
285*33b1fccfSAndroid Build Coastguard Worker 	if (utime(path, &((struct utimbuf){.actime = inode->i_mtime,
286*33b1fccfSAndroid Build Coastguard Worker 					   .modtime = inode->i_mtime})) < 0)
287*33b1fccfSAndroid Build Coastguard Worker #endif
288*33b1fccfSAndroid Build Coastguard Worker 		erofs_warn("failed to set times: %s", path);
289*33b1fccfSAndroid Build Coastguard Worker 
290*33b1fccfSAndroid Build Coastguard Worker 	if (!S_ISLNK(inode->i_mode)) {
291*33b1fccfSAndroid Build Coastguard Worker 		if (fsckcfg.preserve_perms)
292*33b1fccfSAndroid Build Coastguard Worker 			ret = chmod(path, inode->i_mode);
293*33b1fccfSAndroid Build Coastguard Worker 		else
294*33b1fccfSAndroid Build Coastguard Worker 			ret = chmod(path, inode->i_mode & ~fsckcfg.umask);
295*33b1fccfSAndroid Build Coastguard Worker 		if (ret < 0)
296*33b1fccfSAndroid Build Coastguard Worker 			erofs_warn("failed to set permissions: %s", path);
297*33b1fccfSAndroid Build Coastguard Worker 	}
298*33b1fccfSAndroid Build Coastguard Worker 
299*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.preserve_owner) {
300*33b1fccfSAndroid Build Coastguard Worker 		ret = lchown(path, inode->i_uid, inode->i_gid);
301*33b1fccfSAndroid Build Coastguard Worker 		if (ret < 0)
302*33b1fccfSAndroid Build Coastguard Worker 			erofs_warn("failed to change ownership: %s", path);
303*33b1fccfSAndroid Build Coastguard Worker 	}
304*33b1fccfSAndroid Build Coastguard Worker }
305*33b1fccfSAndroid Build Coastguard Worker 
erofs_check_sb_chksum(void)306*33b1fccfSAndroid Build Coastguard Worker static int erofs_check_sb_chksum(void)
307*33b1fccfSAndroid Build Coastguard Worker {
308*33b1fccfSAndroid Build Coastguard Worker #ifndef FUZZING
309*33b1fccfSAndroid Build Coastguard Worker 	u8 buf[EROFS_MAX_BLOCK_SIZE];
310*33b1fccfSAndroid Build Coastguard Worker 	u32 crc;
311*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_super_block *sb;
312*33b1fccfSAndroid Build Coastguard Worker 	int ret;
313*33b1fccfSAndroid Build Coastguard Worker 
314*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_blk_read(&g_sbi, 0, buf, 0, 1);
315*33b1fccfSAndroid Build Coastguard Worker 	if (ret) {
316*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to read superblock to check checksum: %d",
317*33b1fccfSAndroid Build Coastguard Worker 			  ret);
318*33b1fccfSAndroid Build Coastguard Worker 		return -1;
319*33b1fccfSAndroid Build Coastguard Worker 	}
320*33b1fccfSAndroid Build Coastguard Worker 
321*33b1fccfSAndroid Build Coastguard Worker 	sb = (struct erofs_super_block *)(buf + EROFS_SUPER_OFFSET);
322*33b1fccfSAndroid Build Coastguard Worker 	sb->checksum = 0;
323*33b1fccfSAndroid Build Coastguard Worker 
324*33b1fccfSAndroid Build Coastguard Worker 	crc = erofs_crc32c(~0, (u8 *)sb, erofs_blksiz(&g_sbi) - EROFS_SUPER_OFFSET);
325*33b1fccfSAndroid Build Coastguard Worker 	if (crc != g_sbi.checksum) {
326*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("superblock chksum doesn't match: saved(%08xh) calculated(%08xh)",
327*33b1fccfSAndroid Build Coastguard Worker 			  g_sbi.checksum, crc);
328*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.corrupted = true;
329*33b1fccfSAndroid Build Coastguard Worker 		return -1;
330*33b1fccfSAndroid Build Coastguard Worker 	}
331*33b1fccfSAndroid Build Coastguard Worker #endif
332*33b1fccfSAndroid Build Coastguard Worker 	return 0;
333*33b1fccfSAndroid Build Coastguard Worker }
334*33b1fccfSAndroid Build Coastguard Worker 
erofs_verify_xattr(struct erofs_inode * inode)335*33b1fccfSAndroid Build Coastguard Worker static int erofs_verify_xattr(struct erofs_inode *inode)
336*33b1fccfSAndroid Build Coastguard Worker {
337*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_sb_info *sbi = inode->sbi;
338*33b1fccfSAndroid Build Coastguard Worker 	unsigned int xattr_hdr_size = sizeof(struct erofs_xattr_ibody_header);
339*33b1fccfSAndroid Build Coastguard Worker 	unsigned int xattr_entry_size = sizeof(struct erofs_xattr_entry);
340*33b1fccfSAndroid Build Coastguard Worker 	erofs_off_t addr;
341*33b1fccfSAndroid Build Coastguard Worker 	unsigned int ofs, xattr_shared_count;
342*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_xattr_ibody_header *ih;
343*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_xattr_entry *entry;
344*33b1fccfSAndroid Build Coastguard Worker 	int i, remaining = inode->xattr_isize, ret = 0;
345*33b1fccfSAndroid Build Coastguard Worker 	char buf[EROFS_MAX_BLOCK_SIZE];
346*33b1fccfSAndroid Build Coastguard Worker 
347*33b1fccfSAndroid Build Coastguard Worker 	if (inode->xattr_isize == xattr_hdr_size) {
348*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("xattr_isize %d of nid %llu is not supported yet",
349*33b1fccfSAndroid Build Coastguard Worker 			  inode->xattr_isize, inode->nid | 0ULL);
350*33b1fccfSAndroid Build Coastguard Worker 		ret = -EFSCORRUPTED;
351*33b1fccfSAndroid Build Coastguard Worker 		goto out;
352*33b1fccfSAndroid Build Coastguard Worker 	} else if (inode->xattr_isize < xattr_hdr_size) {
353*33b1fccfSAndroid Build Coastguard Worker 		if (inode->xattr_isize) {
354*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("bogus xattr ibody @ nid %llu",
355*33b1fccfSAndroid Build Coastguard Worker 				  inode->nid | 0ULL);
356*33b1fccfSAndroid Build Coastguard Worker 			ret = -EFSCORRUPTED;
357*33b1fccfSAndroid Build Coastguard Worker 			goto out;
358*33b1fccfSAndroid Build Coastguard Worker 		}
359*33b1fccfSAndroid Build Coastguard Worker 	}
360*33b1fccfSAndroid Build Coastguard Worker 
361*33b1fccfSAndroid Build Coastguard Worker 	addr = erofs_iloc(inode) + inode->inode_isize;
362*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_dev_read(sbi, 0, buf, addr, xattr_hdr_size);
363*33b1fccfSAndroid Build Coastguard Worker 	if (ret < 0) {
364*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to read xattr header @ nid %llu: %d",
365*33b1fccfSAndroid Build Coastguard Worker 			  inode->nid | 0ULL, ret);
366*33b1fccfSAndroid Build Coastguard Worker 		goto out;
367*33b1fccfSAndroid Build Coastguard Worker 	}
368*33b1fccfSAndroid Build Coastguard Worker 	ih = (struct erofs_xattr_ibody_header *)buf;
369*33b1fccfSAndroid Build Coastguard Worker 	xattr_shared_count = ih->h_shared_count;
370*33b1fccfSAndroid Build Coastguard Worker 
371*33b1fccfSAndroid Build Coastguard Worker 	ofs = erofs_blkoff(sbi, addr) + xattr_hdr_size;
372*33b1fccfSAndroid Build Coastguard Worker 	addr += xattr_hdr_size;
373*33b1fccfSAndroid Build Coastguard Worker 	remaining -= xattr_hdr_size;
374*33b1fccfSAndroid Build Coastguard Worker 	for (i = 0; i < xattr_shared_count; ++i) {
375*33b1fccfSAndroid Build Coastguard Worker 		if (ofs >= erofs_blksiz(sbi)) {
376*33b1fccfSAndroid Build Coastguard Worker 			if (ofs != erofs_blksiz(sbi)) {
377*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("unaligned xattr entry in xattr shared area @ nid %llu",
378*33b1fccfSAndroid Build Coastguard Worker 					  inode->nid | 0ULL);
379*33b1fccfSAndroid Build Coastguard Worker 				ret = -EFSCORRUPTED;
380*33b1fccfSAndroid Build Coastguard Worker 				goto out;
381*33b1fccfSAndroid Build Coastguard Worker 			}
382*33b1fccfSAndroid Build Coastguard Worker 			ofs = 0;
383*33b1fccfSAndroid Build Coastguard Worker 		}
384*33b1fccfSAndroid Build Coastguard Worker 		ofs += xattr_entry_size;
385*33b1fccfSAndroid Build Coastguard Worker 		addr += xattr_entry_size;
386*33b1fccfSAndroid Build Coastguard Worker 		remaining -= xattr_entry_size;
387*33b1fccfSAndroid Build Coastguard Worker 	}
388*33b1fccfSAndroid Build Coastguard Worker 
389*33b1fccfSAndroid Build Coastguard Worker 	while (remaining > 0) {
390*33b1fccfSAndroid Build Coastguard Worker 		unsigned int entry_sz;
391*33b1fccfSAndroid Build Coastguard Worker 
392*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_dev_read(sbi, 0, buf, addr, xattr_entry_size);
393*33b1fccfSAndroid Build Coastguard Worker 		if (ret) {
394*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to read xattr entry @ nid %llu: %d",
395*33b1fccfSAndroid Build Coastguard Worker 				  inode->nid | 0ULL, ret);
396*33b1fccfSAndroid Build Coastguard Worker 			goto out;
397*33b1fccfSAndroid Build Coastguard Worker 		}
398*33b1fccfSAndroid Build Coastguard Worker 
399*33b1fccfSAndroid Build Coastguard Worker 		entry = (struct erofs_xattr_entry *)buf;
400*33b1fccfSAndroid Build Coastguard Worker 		entry_sz = erofs_xattr_entry_size(entry);
401*33b1fccfSAndroid Build Coastguard Worker 		if (remaining < entry_sz) {
402*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("xattr on-disk corruption: xattr entry beyond xattr_isize @ nid %llu",
403*33b1fccfSAndroid Build Coastguard Worker 				  inode->nid | 0ULL);
404*33b1fccfSAndroid Build Coastguard Worker 			ret = -EFSCORRUPTED;
405*33b1fccfSAndroid Build Coastguard Worker 			goto out;
406*33b1fccfSAndroid Build Coastguard Worker 		}
407*33b1fccfSAndroid Build Coastguard Worker 		addr += entry_sz;
408*33b1fccfSAndroid Build Coastguard Worker 		remaining -= entry_sz;
409*33b1fccfSAndroid Build Coastguard Worker 	}
410*33b1fccfSAndroid Build Coastguard Worker out:
411*33b1fccfSAndroid Build Coastguard Worker 	return ret;
412*33b1fccfSAndroid Build Coastguard Worker }
413*33b1fccfSAndroid Build Coastguard Worker 
erofs_verify_inode_data(struct erofs_inode * inode,int outfd)414*33b1fccfSAndroid Build Coastguard Worker static int erofs_verify_inode_data(struct erofs_inode *inode, int outfd)
415*33b1fccfSAndroid Build Coastguard Worker {
416*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_map_blocks map = {
417*33b1fccfSAndroid Build Coastguard Worker 		.index = UINT_MAX,
418*33b1fccfSAndroid Build Coastguard Worker 	};
419*33b1fccfSAndroid Build Coastguard Worker 	int ret = 0;
420*33b1fccfSAndroid Build Coastguard Worker 	bool compressed;
421*33b1fccfSAndroid Build Coastguard Worker 	erofs_off_t pos = 0;
422*33b1fccfSAndroid Build Coastguard Worker 	u64 pchunk_len = 0;
423*33b1fccfSAndroid Build Coastguard Worker 	unsigned int raw_size = 0, buffer_size = 0;
424*33b1fccfSAndroid Build Coastguard Worker 	char *raw = NULL, *buffer = NULL;
425*33b1fccfSAndroid Build Coastguard Worker 
426*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("verify data chunk of nid(%llu): type(%d)",
427*33b1fccfSAndroid Build Coastguard Worker 		  inode->nid | 0ULL, inode->datalayout);
428*33b1fccfSAndroid Build Coastguard Worker 
429*33b1fccfSAndroid Build Coastguard Worker 	switch (inode->datalayout) {
430*33b1fccfSAndroid Build Coastguard Worker 	case EROFS_INODE_FLAT_PLAIN:
431*33b1fccfSAndroid Build Coastguard Worker 	case EROFS_INODE_FLAT_INLINE:
432*33b1fccfSAndroid Build Coastguard Worker 	case EROFS_INODE_CHUNK_BASED:
433*33b1fccfSAndroid Build Coastguard Worker 		compressed = false;
434*33b1fccfSAndroid Build Coastguard Worker 		break;
435*33b1fccfSAndroid Build Coastguard Worker 	case EROFS_INODE_COMPRESSED_FULL:
436*33b1fccfSAndroid Build Coastguard Worker 	case EROFS_INODE_COMPRESSED_COMPACT:
437*33b1fccfSAndroid Build Coastguard Worker 		compressed = true;
438*33b1fccfSAndroid Build Coastguard Worker 		break;
439*33b1fccfSAndroid Build Coastguard Worker 	default:
440*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("unknown datalayout");
441*33b1fccfSAndroid Build Coastguard Worker 		return -EINVAL;
442*33b1fccfSAndroid Build Coastguard Worker 	}
443*33b1fccfSAndroid Build Coastguard Worker 
444*33b1fccfSAndroid Build Coastguard Worker 	while (pos < inode->i_size) {
445*33b1fccfSAndroid Build Coastguard Worker 		unsigned int alloc_rawsize;
446*33b1fccfSAndroid Build Coastguard Worker 
447*33b1fccfSAndroid Build Coastguard Worker 		map.m_la = pos;
448*33b1fccfSAndroid Build Coastguard Worker 		if (compressed)
449*33b1fccfSAndroid Build Coastguard Worker 			ret = z_erofs_map_blocks_iter(inode, &map,
450*33b1fccfSAndroid Build Coastguard Worker 					EROFS_GET_BLOCKS_FIEMAP);
451*33b1fccfSAndroid Build Coastguard Worker 		else
452*33b1fccfSAndroid Build Coastguard Worker 			ret = erofs_map_blocks(inode, &map,
453*33b1fccfSAndroid Build Coastguard Worker 					EROFS_GET_BLOCKS_FIEMAP);
454*33b1fccfSAndroid Build Coastguard Worker 		if (ret)
455*33b1fccfSAndroid Build Coastguard Worker 			goto out;
456*33b1fccfSAndroid Build Coastguard Worker 
457*33b1fccfSAndroid Build Coastguard Worker 		if (!compressed && map.m_llen != map.m_plen) {
458*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("broken chunk length m_la %" PRIu64 " m_llen %" PRIu64 " m_plen %" PRIu64,
459*33b1fccfSAndroid Build Coastguard Worker 				  map.m_la, map.m_llen, map.m_plen);
460*33b1fccfSAndroid Build Coastguard Worker 			ret = -EFSCORRUPTED;
461*33b1fccfSAndroid Build Coastguard Worker 			goto out;
462*33b1fccfSAndroid Build Coastguard Worker 		}
463*33b1fccfSAndroid Build Coastguard Worker 
464*33b1fccfSAndroid Build Coastguard Worker 		/* the last lcluster can be divided into 3 parts */
465*33b1fccfSAndroid Build Coastguard Worker 		if (map.m_la + map.m_llen > inode->i_size)
466*33b1fccfSAndroid Build Coastguard Worker 			map.m_llen = inode->i_size - map.m_la;
467*33b1fccfSAndroid Build Coastguard Worker 
468*33b1fccfSAndroid Build Coastguard Worker 		pchunk_len += map.m_plen;
469*33b1fccfSAndroid Build Coastguard Worker 		pos += map.m_llen;
470*33b1fccfSAndroid Build Coastguard Worker 
471*33b1fccfSAndroid Build Coastguard Worker 		/* should skip decomp? */
472*33b1fccfSAndroid Build Coastguard Worker 		if (map.m_la >= inode->i_size || !fsckcfg.check_decomp)
473*33b1fccfSAndroid Build Coastguard Worker 			continue;
474*33b1fccfSAndroid Build Coastguard Worker 
475*33b1fccfSAndroid Build Coastguard Worker 		if (outfd >= 0 && !(map.m_flags & EROFS_MAP_MAPPED)) {
476*33b1fccfSAndroid Build Coastguard Worker 			ret = lseek(outfd, map.m_llen, SEEK_CUR);
477*33b1fccfSAndroid Build Coastguard Worker 			if (ret < 0) {
478*33b1fccfSAndroid Build Coastguard Worker 				ret = -errno;
479*33b1fccfSAndroid Build Coastguard Worker 				goto out;
480*33b1fccfSAndroid Build Coastguard Worker 			}
481*33b1fccfSAndroid Build Coastguard Worker 			continue;
482*33b1fccfSAndroid Build Coastguard Worker 		}
483*33b1fccfSAndroid Build Coastguard Worker 
484*33b1fccfSAndroid Build Coastguard Worker 		if (map.m_plen > Z_EROFS_PCLUSTER_MAX_SIZE) {
485*33b1fccfSAndroid Build Coastguard Worker 			if (compressed) {
486*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("invalid pcluster size %" PRIu64 " @ offset %" PRIu64 " of nid %" PRIu64,
487*33b1fccfSAndroid Build Coastguard Worker 					  map.m_plen, map.m_la,
488*33b1fccfSAndroid Build Coastguard Worker 					  inode->nid | 0ULL);
489*33b1fccfSAndroid Build Coastguard Worker 				ret = -EFSCORRUPTED;
490*33b1fccfSAndroid Build Coastguard Worker 				goto out;
491*33b1fccfSAndroid Build Coastguard Worker 			}
492*33b1fccfSAndroid Build Coastguard Worker 			alloc_rawsize = Z_EROFS_PCLUSTER_MAX_SIZE;
493*33b1fccfSAndroid Build Coastguard Worker 		} else {
494*33b1fccfSAndroid Build Coastguard Worker 			alloc_rawsize = map.m_plen;
495*33b1fccfSAndroid Build Coastguard Worker 		}
496*33b1fccfSAndroid Build Coastguard Worker 
497*33b1fccfSAndroid Build Coastguard Worker 		if (alloc_rawsize > raw_size) {
498*33b1fccfSAndroid Build Coastguard Worker 			char *newraw = realloc(raw, alloc_rawsize);
499*33b1fccfSAndroid Build Coastguard Worker 
500*33b1fccfSAndroid Build Coastguard Worker 			if (!newraw) {
501*33b1fccfSAndroid Build Coastguard Worker 				ret = -ENOMEM;
502*33b1fccfSAndroid Build Coastguard Worker 				goto out;
503*33b1fccfSAndroid Build Coastguard Worker 			}
504*33b1fccfSAndroid Build Coastguard Worker 			raw = newraw;
505*33b1fccfSAndroid Build Coastguard Worker 			raw_size = alloc_rawsize;
506*33b1fccfSAndroid Build Coastguard Worker 		}
507*33b1fccfSAndroid Build Coastguard Worker 
508*33b1fccfSAndroid Build Coastguard Worker 		if (compressed) {
509*33b1fccfSAndroid Build Coastguard Worker 			if (map.m_llen > buffer_size) {
510*33b1fccfSAndroid Build Coastguard Worker 				char *newbuffer;
511*33b1fccfSAndroid Build Coastguard Worker 
512*33b1fccfSAndroid Build Coastguard Worker 				buffer_size = map.m_llen;
513*33b1fccfSAndroid Build Coastguard Worker 				newbuffer = realloc(buffer, buffer_size);
514*33b1fccfSAndroid Build Coastguard Worker 				if (!newbuffer) {
515*33b1fccfSAndroid Build Coastguard Worker 					ret = -ENOMEM;
516*33b1fccfSAndroid Build Coastguard Worker 					goto out;
517*33b1fccfSAndroid Build Coastguard Worker 				}
518*33b1fccfSAndroid Build Coastguard Worker 				buffer = newbuffer;
519*33b1fccfSAndroid Build Coastguard Worker 			}
520*33b1fccfSAndroid Build Coastguard Worker 			ret = z_erofs_read_one_data(inode, &map, raw, buffer,
521*33b1fccfSAndroid Build Coastguard Worker 						    0, map.m_llen, false);
522*33b1fccfSAndroid Build Coastguard Worker 			if (ret)
523*33b1fccfSAndroid Build Coastguard Worker 				goto out;
524*33b1fccfSAndroid Build Coastguard Worker 
525*33b1fccfSAndroid Build Coastguard Worker 			if (outfd >= 0 && write(outfd, buffer, map.m_llen) < 0)
526*33b1fccfSAndroid Build Coastguard Worker 				goto fail_eio;
527*33b1fccfSAndroid Build Coastguard Worker 		} else {
528*33b1fccfSAndroid Build Coastguard Worker 			u64 p = 0;
529*33b1fccfSAndroid Build Coastguard Worker 
530*33b1fccfSAndroid Build Coastguard Worker 			do {
531*33b1fccfSAndroid Build Coastguard Worker 				u64 count = min_t(u64, alloc_rawsize,
532*33b1fccfSAndroid Build Coastguard Worker 						  map.m_llen);
533*33b1fccfSAndroid Build Coastguard Worker 
534*33b1fccfSAndroid Build Coastguard Worker 				ret = erofs_read_one_data(inode, &map, raw, p, count);
535*33b1fccfSAndroid Build Coastguard Worker 				if (ret)
536*33b1fccfSAndroid Build Coastguard Worker 					goto out;
537*33b1fccfSAndroid Build Coastguard Worker 
538*33b1fccfSAndroid Build Coastguard Worker 				if (outfd >= 0 && write(outfd, raw, count) < 0)
539*33b1fccfSAndroid Build Coastguard Worker 					goto fail_eio;
540*33b1fccfSAndroid Build Coastguard Worker 				map.m_llen -= count;
541*33b1fccfSAndroid Build Coastguard Worker 				p += count;
542*33b1fccfSAndroid Build Coastguard Worker 			} while (map.m_llen);
543*33b1fccfSAndroid Build Coastguard Worker 		}
544*33b1fccfSAndroid Build Coastguard Worker 	}
545*33b1fccfSAndroid Build Coastguard Worker 
546*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.print_comp_ratio) {
547*33b1fccfSAndroid Build Coastguard Worker 		if (!erofs_is_packed_inode(inode))
548*33b1fccfSAndroid Build Coastguard Worker 			fsckcfg.logical_blocks += BLK_ROUND_UP(inode->sbi, inode->i_size);
549*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.physical_blocks += BLK_ROUND_UP(inode->sbi, pchunk_len);
550*33b1fccfSAndroid Build Coastguard Worker 	}
551*33b1fccfSAndroid Build Coastguard Worker out:
552*33b1fccfSAndroid Build Coastguard Worker 	if (raw)
553*33b1fccfSAndroid Build Coastguard Worker 		free(raw);
554*33b1fccfSAndroid Build Coastguard Worker 	if (buffer)
555*33b1fccfSAndroid Build Coastguard Worker 		free(buffer);
556*33b1fccfSAndroid Build Coastguard Worker 	return ret < 0 ? ret : 0;
557*33b1fccfSAndroid Build Coastguard Worker 
558*33b1fccfSAndroid Build Coastguard Worker fail_eio:
559*33b1fccfSAndroid Build Coastguard Worker 	erofs_err("I/O error occurred when verifying data chunk @ nid %llu",
560*33b1fccfSAndroid Build Coastguard Worker 		  inode->nid | 0ULL);
561*33b1fccfSAndroid Build Coastguard Worker 	ret = -EIO;
562*33b1fccfSAndroid Build Coastguard Worker 	goto out;
563*33b1fccfSAndroid Build Coastguard Worker }
564*33b1fccfSAndroid Build Coastguard Worker 
erofs_extract_dir(struct erofs_inode * inode)565*33b1fccfSAndroid Build Coastguard Worker static inline int erofs_extract_dir(struct erofs_inode *inode)
566*33b1fccfSAndroid Build Coastguard Worker {
567*33b1fccfSAndroid Build Coastguard Worker 	int ret;
568*33b1fccfSAndroid Build Coastguard Worker 
569*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("create directory %s", fsckcfg.extract_path);
570*33b1fccfSAndroid Build Coastguard Worker 
571*33b1fccfSAndroid Build Coastguard Worker 	/* verify data chunk layout */
572*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_verify_inode_data(inode, -1);
573*33b1fccfSAndroid Build Coastguard Worker 	if (ret)
574*33b1fccfSAndroid Build Coastguard Worker 		return ret;
575*33b1fccfSAndroid Build Coastguard Worker 
576*33b1fccfSAndroid Build Coastguard Worker 	/*
577*33b1fccfSAndroid Build Coastguard Worker 	 * Make directory with default user rwx permissions rather than
578*33b1fccfSAndroid Build Coastguard Worker 	 * the permissions from the filesystem, as these may not have
579*33b1fccfSAndroid Build Coastguard Worker 	 * write/execute permission.  These are fixed up later in
580*33b1fccfSAndroid Build Coastguard Worker 	 * erofsfsck_set_attributes().
581*33b1fccfSAndroid Build Coastguard Worker 	 */
582*33b1fccfSAndroid Build Coastguard Worker 	if (mkdir(fsckcfg.extract_path, 0700) < 0) {
583*33b1fccfSAndroid Build Coastguard Worker 		struct stat st;
584*33b1fccfSAndroid Build Coastguard Worker 
585*33b1fccfSAndroid Build Coastguard Worker 		if (errno != EEXIST) {
586*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to create directory: %s (%s)",
587*33b1fccfSAndroid Build Coastguard Worker 				  fsckcfg.extract_path, strerror(errno));
588*33b1fccfSAndroid Build Coastguard Worker 			return -errno;
589*33b1fccfSAndroid Build Coastguard Worker 		}
590*33b1fccfSAndroid Build Coastguard Worker 
591*33b1fccfSAndroid Build Coastguard Worker 		if (lstat(fsckcfg.extract_path, &st) ||
592*33b1fccfSAndroid Build Coastguard Worker 		    !S_ISDIR(st.st_mode)) {
593*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("path is not a directory: %s",
594*33b1fccfSAndroid Build Coastguard Worker 				  fsckcfg.extract_path);
595*33b1fccfSAndroid Build Coastguard Worker 			return -ENOTDIR;
596*33b1fccfSAndroid Build Coastguard Worker 		}
597*33b1fccfSAndroid Build Coastguard Worker 
598*33b1fccfSAndroid Build Coastguard Worker 		/*
599*33b1fccfSAndroid Build Coastguard Worker 		 * Try to change permissions of existing directory so
600*33b1fccfSAndroid Build Coastguard Worker 		 * that we can write to it
601*33b1fccfSAndroid Build Coastguard Worker 		 */
602*33b1fccfSAndroid Build Coastguard Worker 		if (chmod(fsckcfg.extract_path, 0700) < 0) {
603*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to set permissions: %s (%s)",
604*33b1fccfSAndroid Build Coastguard Worker 				  fsckcfg.extract_path, strerror(errno));
605*33b1fccfSAndroid Build Coastguard Worker 			return -errno;
606*33b1fccfSAndroid Build Coastguard Worker 		}
607*33b1fccfSAndroid Build Coastguard Worker 	}
608*33b1fccfSAndroid Build Coastguard Worker 	return 0;
609*33b1fccfSAndroid Build Coastguard Worker }
610*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_hardlink_find(erofs_nid_t nid)611*33b1fccfSAndroid Build Coastguard Worker static char *erofsfsck_hardlink_find(erofs_nid_t nid)
612*33b1fccfSAndroid Build Coastguard Worker {
613*33b1fccfSAndroid Build Coastguard Worker 	struct list_head *head =
614*33b1fccfSAndroid Build Coastguard Worker 			&erofsfsck_link_hashtable[nid % NR_HARDLINK_HASHTABLE];
615*33b1fccfSAndroid Build Coastguard Worker 	struct erofsfsck_hardlink_entry *entry;
616*33b1fccfSAndroid Build Coastguard Worker 
617*33b1fccfSAndroid Build Coastguard Worker 	list_for_each_entry(entry, head, list)
618*33b1fccfSAndroid Build Coastguard Worker 		if (entry->nid == nid)
619*33b1fccfSAndroid Build Coastguard Worker 			return entry->path;
620*33b1fccfSAndroid Build Coastguard Worker 	return NULL;
621*33b1fccfSAndroid Build Coastguard Worker }
622*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_hardlink_insert(erofs_nid_t nid,const char * path)623*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_hardlink_insert(erofs_nid_t nid, const char *path)
624*33b1fccfSAndroid Build Coastguard Worker {
625*33b1fccfSAndroid Build Coastguard Worker 	struct erofsfsck_hardlink_entry *entry;
626*33b1fccfSAndroid Build Coastguard Worker 
627*33b1fccfSAndroid Build Coastguard Worker 	entry = malloc(sizeof(*entry));
628*33b1fccfSAndroid Build Coastguard Worker 	if (!entry)
629*33b1fccfSAndroid Build Coastguard Worker 		return -ENOMEM;
630*33b1fccfSAndroid Build Coastguard Worker 
631*33b1fccfSAndroid Build Coastguard Worker 	entry->nid = nid;
632*33b1fccfSAndroid Build Coastguard Worker 	entry->path = strdup(path);
633*33b1fccfSAndroid Build Coastguard Worker 	if (!entry->path) {
634*33b1fccfSAndroid Build Coastguard Worker 		free(entry);
635*33b1fccfSAndroid Build Coastguard Worker 		return -ENOMEM;
636*33b1fccfSAndroid Build Coastguard Worker 	}
637*33b1fccfSAndroid Build Coastguard Worker 
638*33b1fccfSAndroid Build Coastguard Worker 	list_add_tail(&entry->list,
639*33b1fccfSAndroid Build Coastguard Worker 		      &erofsfsck_link_hashtable[nid % NR_HARDLINK_HASHTABLE]);
640*33b1fccfSAndroid Build Coastguard Worker 	return 0;
641*33b1fccfSAndroid Build Coastguard Worker }
642*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_hardlink_init(void)643*33b1fccfSAndroid Build Coastguard Worker static void erofsfsck_hardlink_init(void)
644*33b1fccfSAndroid Build Coastguard Worker {
645*33b1fccfSAndroid Build Coastguard Worker 	unsigned int i;
646*33b1fccfSAndroid Build Coastguard Worker 
647*33b1fccfSAndroid Build Coastguard Worker 	for (i = 0; i < NR_HARDLINK_HASHTABLE; ++i)
648*33b1fccfSAndroid Build Coastguard Worker 		init_list_head(&erofsfsck_link_hashtable[i]);
649*33b1fccfSAndroid Build Coastguard Worker }
650*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_hardlink_exit(void)651*33b1fccfSAndroid Build Coastguard Worker static void erofsfsck_hardlink_exit(void)
652*33b1fccfSAndroid Build Coastguard Worker {
653*33b1fccfSAndroid Build Coastguard Worker 	struct erofsfsck_hardlink_entry *entry, *n;
654*33b1fccfSAndroid Build Coastguard Worker 	struct list_head *head;
655*33b1fccfSAndroid Build Coastguard Worker 	unsigned int i;
656*33b1fccfSAndroid Build Coastguard Worker 
657*33b1fccfSAndroid Build Coastguard Worker 	for (i = 0; i < NR_HARDLINK_HASHTABLE; ++i) {
658*33b1fccfSAndroid Build Coastguard Worker 		head = &erofsfsck_link_hashtable[i];
659*33b1fccfSAndroid Build Coastguard Worker 
660*33b1fccfSAndroid Build Coastguard Worker 		list_for_each_entry_safe(entry, n, head, list) {
661*33b1fccfSAndroid Build Coastguard Worker 			if (entry->path)
662*33b1fccfSAndroid Build Coastguard Worker 				free(entry->path);
663*33b1fccfSAndroid Build Coastguard Worker 			free(entry);
664*33b1fccfSAndroid Build Coastguard Worker 		}
665*33b1fccfSAndroid Build Coastguard Worker 	}
666*33b1fccfSAndroid Build Coastguard Worker }
667*33b1fccfSAndroid Build Coastguard Worker 
erofs_extract_file(struct erofs_inode * inode)668*33b1fccfSAndroid Build Coastguard Worker static inline int erofs_extract_file(struct erofs_inode *inode)
669*33b1fccfSAndroid Build Coastguard Worker {
670*33b1fccfSAndroid Build Coastguard Worker 	bool tryagain = true;
671*33b1fccfSAndroid Build Coastguard Worker 	int ret, fd;
672*33b1fccfSAndroid Build Coastguard Worker 
673*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("extract file to path: %s", fsckcfg.extract_path);
674*33b1fccfSAndroid Build Coastguard Worker 
675*33b1fccfSAndroid Build Coastguard Worker again:
676*33b1fccfSAndroid Build Coastguard Worker 	fd = open(fsckcfg.extract_path,
677*33b1fccfSAndroid Build Coastguard Worker 		  O_WRONLY | O_CREAT | O_NOFOLLOW |
678*33b1fccfSAndroid Build Coastguard Worker 			(fsckcfg.overwrite ? O_TRUNC : O_EXCL), 0700);
679*33b1fccfSAndroid Build Coastguard Worker 	if (fd < 0) {
680*33b1fccfSAndroid Build Coastguard Worker 		if (fsckcfg.overwrite && tryagain) {
681*33b1fccfSAndroid Build Coastguard Worker 			if (errno == EISDIR) {
682*33b1fccfSAndroid Build Coastguard Worker 				erofs_warn("try to forcely remove directory %s",
683*33b1fccfSAndroid Build Coastguard Worker 					   fsckcfg.extract_path);
684*33b1fccfSAndroid Build Coastguard Worker 				if (rmdir(fsckcfg.extract_path) < 0) {
685*33b1fccfSAndroid Build Coastguard Worker 					erofs_err("failed to remove: %s (%s)",
686*33b1fccfSAndroid Build Coastguard Worker 						  fsckcfg.extract_path, strerror(errno));
687*33b1fccfSAndroid Build Coastguard Worker 					return -EISDIR;
688*33b1fccfSAndroid Build Coastguard Worker 				}
689*33b1fccfSAndroid Build Coastguard Worker 			} else if (errno == EACCES &&
690*33b1fccfSAndroid Build Coastguard Worker 				   chmod(fsckcfg.extract_path, 0700) < 0) {
691*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("failed to set permissions: %s (%s)",
692*33b1fccfSAndroid Build Coastguard Worker 					  fsckcfg.extract_path, strerror(errno));
693*33b1fccfSAndroid Build Coastguard Worker 				return -errno;
694*33b1fccfSAndroid Build Coastguard Worker 			}
695*33b1fccfSAndroid Build Coastguard Worker 			tryagain = false;
696*33b1fccfSAndroid Build Coastguard Worker 			goto again;
697*33b1fccfSAndroid Build Coastguard Worker 		}
698*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to open: %s (%s)", fsckcfg.extract_path,
699*33b1fccfSAndroid Build Coastguard Worker 			  strerror(errno));
700*33b1fccfSAndroid Build Coastguard Worker 		return -errno;
701*33b1fccfSAndroid Build Coastguard Worker 	}
702*33b1fccfSAndroid Build Coastguard Worker 
703*33b1fccfSAndroid Build Coastguard Worker 	/* verify data chunk layout */
704*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_verify_inode_data(inode, fd);
705*33b1fccfSAndroid Build Coastguard Worker 	close(fd);
706*33b1fccfSAndroid Build Coastguard Worker 	return ret;
707*33b1fccfSAndroid Build Coastguard Worker }
708*33b1fccfSAndroid Build Coastguard Worker 
erofs_extract_symlink(struct erofs_inode * inode)709*33b1fccfSAndroid Build Coastguard Worker static inline int erofs_extract_symlink(struct erofs_inode *inode)
710*33b1fccfSAndroid Build Coastguard Worker {
711*33b1fccfSAndroid Build Coastguard Worker 	bool tryagain = true;
712*33b1fccfSAndroid Build Coastguard Worker 	int ret;
713*33b1fccfSAndroid Build Coastguard Worker 	char *buf = NULL;
714*33b1fccfSAndroid Build Coastguard Worker 
715*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("extract symlink to path: %s", fsckcfg.extract_path);
716*33b1fccfSAndroid Build Coastguard Worker 
717*33b1fccfSAndroid Build Coastguard Worker 	/* verify data chunk layout */
718*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_verify_inode_data(inode, -1);
719*33b1fccfSAndroid Build Coastguard Worker 	if (ret)
720*33b1fccfSAndroid Build Coastguard Worker 		return ret;
721*33b1fccfSAndroid Build Coastguard Worker 
722*33b1fccfSAndroid Build Coastguard Worker 	buf = malloc(inode->i_size + 1);
723*33b1fccfSAndroid Build Coastguard Worker 	if (!buf) {
724*33b1fccfSAndroid Build Coastguard Worker 		ret = -ENOMEM;
725*33b1fccfSAndroid Build Coastguard Worker 		goto out;
726*33b1fccfSAndroid Build Coastguard Worker 	}
727*33b1fccfSAndroid Build Coastguard Worker 
728*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_pread(inode, buf, inode->i_size, 0);
729*33b1fccfSAndroid Build Coastguard Worker 	if (ret) {
730*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("I/O error occurred when reading symlink @ nid %llu: %d",
731*33b1fccfSAndroid Build Coastguard Worker 			  inode->nid | 0ULL, ret);
732*33b1fccfSAndroid Build Coastguard Worker 		goto out;
733*33b1fccfSAndroid Build Coastguard Worker 	}
734*33b1fccfSAndroid Build Coastguard Worker 
735*33b1fccfSAndroid Build Coastguard Worker 	buf[inode->i_size] = '\0';
736*33b1fccfSAndroid Build Coastguard Worker again:
737*33b1fccfSAndroid Build Coastguard Worker 	if (symlink(buf, fsckcfg.extract_path) < 0) {
738*33b1fccfSAndroid Build Coastguard Worker 		if (errno == EEXIST && fsckcfg.overwrite && tryagain) {
739*33b1fccfSAndroid Build Coastguard Worker 			erofs_warn("try to forcely remove file %s",
740*33b1fccfSAndroid Build Coastguard Worker 				   fsckcfg.extract_path);
741*33b1fccfSAndroid Build Coastguard Worker 			if (unlink(fsckcfg.extract_path) < 0) {
742*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("failed to remove: %s",
743*33b1fccfSAndroid Build Coastguard Worker 					  fsckcfg.extract_path);
744*33b1fccfSAndroid Build Coastguard Worker 				ret = -errno;
745*33b1fccfSAndroid Build Coastguard Worker 				goto out;
746*33b1fccfSAndroid Build Coastguard Worker 			}
747*33b1fccfSAndroid Build Coastguard Worker 			tryagain = false;
748*33b1fccfSAndroid Build Coastguard Worker 			goto again;
749*33b1fccfSAndroid Build Coastguard Worker 		}
750*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to create symlink: %s",
751*33b1fccfSAndroid Build Coastguard Worker 			  fsckcfg.extract_path);
752*33b1fccfSAndroid Build Coastguard Worker 		ret = -errno;
753*33b1fccfSAndroid Build Coastguard Worker 	}
754*33b1fccfSAndroid Build Coastguard Worker out:
755*33b1fccfSAndroid Build Coastguard Worker 	if (buf)
756*33b1fccfSAndroid Build Coastguard Worker 		free(buf);
757*33b1fccfSAndroid Build Coastguard Worker 	return ret;
758*33b1fccfSAndroid Build Coastguard Worker }
759*33b1fccfSAndroid Build Coastguard Worker 
erofs_extract_special(struct erofs_inode * inode)760*33b1fccfSAndroid Build Coastguard Worker static int erofs_extract_special(struct erofs_inode *inode)
761*33b1fccfSAndroid Build Coastguard Worker {
762*33b1fccfSAndroid Build Coastguard Worker 	bool tryagain = true;
763*33b1fccfSAndroid Build Coastguard Worker 	int ret;
764*33b1fccfSAndroid Build Coastguard Worker 
765*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("extract special to path: %s", fsckcfg.extract_path);
766*33b1fccfSAndroid Build Coastguard Worker 
767*33b1fccfSAndroid Build Coastguard Worker 	/* verify data chunk layout */
768*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_verify_inode_data(inode, -1);
769*33b1fccfSAndroid Build Coastguard Worker 	if (ret)
770*33b1fccfSAndroid Build Coastguard Worker 		return ret;
771*33b1fccfSAndroid Build Coastguard Worker 
772*33b1fccfSAndroid Build Coastguard Worker again:
773*33b1fccfSAndroid Build Coastguard Worker 	if (mknod(fsckcfg.extract_path, inode->i_mode, inode->u.i_rdev) < 0) {
774*33b1fccfSAndroid Build Coastguard Worker 		if (errno == EEXIST && fsckcfg.overwrite && tryagain) {
775*33b1fccfSAndroid Build Coastguard Worker 			erofs_warn("try to forcely remove file %s",
776*33b1fccfSAndroid Build Coastguard Worker 				   fsckcfg.extract_path);
777*33b1fccfSAndroid Build Coastguard Worker 			if (unlink(fsckcfg.extract_path) < 0) {
778*33b1fccfSAndroid Build Coastguard Worker 				erofs_err("failed to remove: %s",
779*33b1fccfSAndroid Build Coastguard Worker 					  fsckcfg.extract_path);
780*33b1fccfSAndroid Build Coastguard Worker 				return -errno;
781*33b1fccfSAndroid Build Coastguard Worker 			}
782*33b1fccfSAndroid Build Coastguard Worker 			tryagain = false;
783*33b1fccfSAndroid Build Coastguard Worker 			goto again;
784*33b1fccfSAndroid Build Coastguard Worker 		}
785*33b1fccfSAndroid Build Coastguard Worker 		if (errno == EEXIST || fsckcfg.superuser) {
786*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to create special file: %s",
787*33b1fccfSAndroid Build Coastguard Worker 				  fsckcfg.extract_path);
788*33b1fccfSAndroid Build Coastguard Worker 			ret = -errno;
789*33b1fccfSAndroid Build Coastguard Worker 		} else {
790*33b1fccfSAndroid Build Coastguard Worker 			erofs_warn("failed to create special file: %s, skipped",
791*33b1fccfSAndroid Build Coastguard Worker 				   fsckcfg.extract_path);
792*33b1fccfSAndroid Build Coastguard Worker 			ret = -ECANCELED;
793*33b1fccfSAndroid Build Coastguard Worker 		}
794*33b1fccfSAndroid Build Coastguard Worker 	}
795*33b1fccfSAndroid Build Coastguard Worker 	return ret;
796*33b1fccfSAndroid Build Coastguard Worker }
797*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_dirent_iter(struct erofs_dir_context * ctx)798*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_dirent_iter(struct erofs_dir_context *ctx)
799*33b1fccfSAndroid Build Coastguard Worker {
800*33b1fccfSAndroid Build Coastguard Worker 	int ret;
801*33b1fccfSAndroid Build Coastguard Worker 	size_t prev_pos, curr_pos;
802*33b1fccfSAndroid Build Coastguard Worker 
803*33b1fccfSAndroid Build Coastguard Worker 	if (ctx->dot_dotdot)
804*33b1fccfSAndroid Build Coastguard Worker 		return 0;
805*33b1fccfSAndroid Build Coastguard Worker 
806*33b1fccfSAndroid Build Coastguard Worker 	prev_pos = fsckcfg.extract_pos;
807*33b1fccfSAndroid Build Coastguard Worker 	curr_pos = prev_pos;
808*33b1fccfSAndroid Build Coastguard Worker 
809*33b1fccfSAndroid Build Coastguard Worker 	if (prev_pos + ctx->de_namelen >= PATH_MAX) {
810*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("unable to fsck since the path is too long (%u)",
811*33b1fccfSAndroid Build Coastguard Worker 			  curr_pos + ctx->de_namelen);
812*33b1fccfSAndroid Build Coastguard Worker 		return -EOPNOTSUPP;
813*33b1fccfSAndroid Build Coastguard Worker 	}
814*33b1fccfSAndroid Build Coastguard Worker 
815*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.extract_path) {
816*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.extract_path[curr_pos++] = '/';
817*33b1fccfSAndroid Build Coastguard Worker 		strncpy(fsckcfg.extract_path + curr_pos, ctx->dname,
818*33b1fccfSAndroid Build Coastguard Worker 			ctx->de_namelen);
819*33b1fccfSAndroid Build Coastguard Worker 		curr_pos += ctx->de_namelen;
820*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.extract_path[curr_pos] = '\0';
821*33b1fccfSAndroid Build Coastguard Worker 	} else {
822*33b1fccfSAndroid Build Coastguard Worker 		curr_pos += ctx->de_namelen;
823*33b1fccfSAndroid Build Coastguard Worker 	}
824*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.extract_pos = curr_pos;
825*33b1fccfSAndroid Build Coastguard Worker 	ret = erofsfsck_check_inode(ctx->dir->nid, ctx->de_nid);
826*33b1fccfSAndroid Build Coastguard Worker 
827*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.extract_path)
828*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.extract_path[prev_pos] = '\0';
829*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.extract_pos = prev_pos;
830*33b1fccfSAndroid Build Coastguard Worker 	return ret;
831*33b1fccfSAndroid Build Coastguard Worker }
832*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_extract_inode(struct erofs_inode * inode)833*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_extract_inode(struct erofs_inode *inode)
834*33b1fccfSAndroid Build Coastguard Worker {
835*33b1fccfSAndroid Build Coastguard Worker 	int ret;
836*33b1fccfSAndroid Build Coastguard Worker 	char *oldpath;
837*33b1fccfSAndroid Build Coastguard Worker 
838*33b1fccfSAndroid Build Coastguard Worker 	if (!fsckcfg.extract_path) {
839*33b1fccfSAndroid Build Coastguard Worker verify:
840*33b1fccfSAndroid Build Coastguard Worker 		/* verify data chunk layout */
841*33b1fccfSAndroid Build Coastguard Worker 		return erofs_verify_inode_data(inode, -1);
842*33b1fccfSAndroid Build Coastguard Worker 	}
843*33b1fccfSAndroid Build Coastguard Worker 
844*33b1fccfSAndroid Build Coastguard Worker 	oldpath = erofsfsck_hardlink_find(inode->nid);
845*33b1fccfSAndroid Build Coastguard Worker 	if (oldpath) {
846*33b1fccfSAndroid Build Coastguard Worker 		if (link(oldpath, fsckcfg.extract_path) == -1) {
847*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to extract hard link: %s (%s)",
848*33b1fccfSAndroid Build Coastguard Worker 				  fsckcfg.extract_path, strerror(errno));
849*33b1fccfSAndroid Build Coastguard Worker 			return -errno;
850*33b1fccfSAndroid Build Coastguard Worker 		}
851*33b1fccfSAndroid Build Coastguard Worker 		return 0;
852*33b1fccfSAndroid Build Coastguard Worker 	}
853*33b1fccfSAndroid Build Coastguard Worker 
854*33b1fccfSAndroid Build Coastguard Worker 	switch (inode->i_mode & S_IFMT) {
855*33b1fccfSAndroid Build Coastguard Worker 	case S_IFDIR:
856*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_extract_dir(inode);
857*33b1fccfSAndroid Build Coastguard Worker 		break;
858*33b1fccfSAndroid Build Coastguard Worker 	case S_IFREG:
859*33b1fccfSAndroid Build Coastguard Worker 		if (erofs_is_packed_inode(inode))
860*33b1fccfSAndroid Build Coastguard Worker 			goto verify;
861*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_extract_file(inode);
862*33b1fccfSAndroid Build Coastguard Worker 		break;
863*33b1fccfSAndroid Build Coastguard Worker 	case S_IFLNK:
864*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_extract_symlink(inode);
865*33b1fccfSAndroid Build Coastguard Worker 		break;
866*33b1fccfSAndroid Build Coastguard Worker 	case S_IFCHR:
867*33b1fccfSAndroid Build Coastguard Worker 	case S_IFBLK:
868*33b1fccfSAndroid Build Coastguard Worker 	case S_IFIFO:
869*33b1fccfSAndroid Build Coastguard Worker 	case S_IFSOCK:
870*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_extract_special(inode);
871*33b1fccfSAndroid Build Coastguard Worker 		break;
872*33b1fccfSAndroid Build Coastguard Worker 	default:
873*33b1fccfSAndroid Build Coastguard Worker 		/* TODO */
874*33b1fccfSAndroid Build Coastguard Worker 		goto verify;
875*33b1fccfSAndroid Build Coastguard Worker 	}
876*33b1fccfSAndroid Build Coastguard Worker 	if (ret && ret != -ECANCELED)
877*33b1fccfSAndroid Build Coastguard Worker 		return ret;
878*33b1fccfSAndroid Build Coastguard Worker 
879*33b1fccfSAndroid Build Coastguard Worker 	/* record nid and old path for hardlink */
880*33b1fccfSAndroid Build Coastguard Worker 	if (inode->i_nlink > 1 && !S_ISDIR(inode->i_mode))
881*33b1fccfSAndroid Build Coastguard Worker 		ret = erofsfsck_hardlink_insert(inode->nid,
882*33b1fccfSAndroid Build Coastguard Worker 						fsckcfg.extract_path);
883*33b1fccfSAndroid Build Coastguard Worker 	return ret;
884*33b1fccfSAndroid Build Coastguard Worker }
885*33b1fccfSAndroid Build Coastguard Worker 
erofsfsck_check_inode(erofs_nid_t pnid,erofs_nid_t nid)886*33b1fccfSAndroid Build Coastguard Worker static int erofsfsck_check_inode(erofs_nid_t pnid, erofs_nid_t nid)
887*33b1fccfSAndroid Build Coastguard Worker {
888*33b1fccfSAndroid Build Coastguard Worker 	int ret;
889*33b1fccfSAndroid Build Coastguard Worker 	struct erofs_inode inode;
890*33b1fccfSAndroid Build Coastguard Worker 
891*33b1fccfSAndroid Build Coastguard Worker 	erofs_dbg("check inode: nid(%llu)", nid | 0ULL);
892*33b1fccfSAndroid Build Coastguard Worker 
893*33b1fccfSAndroid Build Coastguard Worker 	inode.nid = nid;
894*33b1fccfSAndroid Build Coastguard Worker 	inode.sbi = &g_sbi;
895*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_read_inode_from_disk(&inode);
896*33b1fccfSAndroid Build Coastguard Worker 	if (ret) {
897*33b1fccfSAndroid Build Coastguard Worker 		if (ret == -EIO)
898*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("I/O error occurred when reading nid(%llu)",
899*33b1fccfSAndroid Build Coastguard Worker 				  nid | 0ULL);
900*33b1fccfSAndroid Build Coastguard Worker 		goto out;
901*33b1fccfSAndroid Build Coastguard Worker 	}
902*33b1fccfSAndroid Build Coastguard Worker 
903*33b1fccfSAndroid Build Coastguard Worker 	/* verify xattr field */
904*33b1fccfSAndroid Build Coastguard Worker 	ret = erofs_verify_xattr(&inode);
905*33b1fccfSAndroid Build Coastguard Worker 	if (ret)
906*33b1fccfSAndroid Build Coastguard Worker 		goto out;
907*33b1fccfSAndroid Build Coastguard Worker 
908*33b1fccfSAndroid Build Coastguard Worker 	ret = erofsfsck_extract_inode(&inode);
909*33b1fccfSAndroid Build Coastguard Worker 	if (ret && ret != -ECANCELED)
910*33b1fccfSAndroid Build Coastguard Worker 		goto out;
911*33b1fccfSAndroid Build Coastguard Worker 
912*33b1fccfSAndroid Build Coastguard Worker 	/* XXXX: the dir depth should be restricted in order to avoid loops */
913*33b1fccfSAndroid Build Coastguard Worker 	if (S_ISDIR(inode.i_mode)) {
914*33b1fccfSAndroid Build Coastguard Worker 		struct erofs_dir_context ctx = {
915*33b1fccfSAndroid Build Coastguard Worker 			.flags = EROFS_READDIR_VALID_PNID,
916*33b1fccfSAndroid Build Coastguard Worker 			.pnid = pnid,
917*33b1fccfSAndroid Build Coastguard Worker 			.dir = &inode,
918*33b1fccfSAndroid Build Coastguard Worker 			.cb = erofsfsck_dirent_iter,
919*33b1fccfSAndroid Build Coastguard Worker 		};
920*33b1fccfSAndroid Build Coastguard Worker 
921*33b1fccfSAndroid Build Coastguard Worker 		ret = erofs_iterate_dir(&ctx, true);
922*33b1fccfSAndroid Build Coastguard Worker 	}
923*33b1fccfSAndroid Build Coastguard Worker 
924*33b1fccfSAndroid Build Coastguard Worker 	if (!ret && !erofs_is_packed_inode(&inode))
925*33b1fccfSAndroid Build Coastguard Worker 		erofsfsck_set_attributes(&inode, fsckcfg.extract_path);
926*33b1fccfSAndroid Build Coastguard Worker 
927*33b1fccfSAndroid Build Coastguard Worker 	if (ret == -ECANCELED)
928*33b1fccfSAndroid Build Coastguard Worker 		ret = 0;
929*33b1fccfSAndroid Build Coastguard Worker out:
930*33b1fccfSAndroid Build Coastguard Worker 	if (ret && ret != -EIO)
931*33b1fccfSAndroid Build Coastguard Worker 		fsckcfg.corrupted = true;
932*33b1fccfSAndroid Build Coastguard Worker 	return ret;
933*33b1fccfSAndroid Build Coastguard Worker }
934*33b1fccfSAndroid Build Coastguard Worker 
935*33b1fccfSAndroid Build Coastguard Worker #ifdef FUZZING
erofsfsck_fuzz_one(int argc,char * argv[])936*33b1fccfSAndroid Build Coastguard Worker int erofsfsck_fuzz_one(int argc, char *argv[])
937*33b1fccfSAndroid Build Coastguard Worker #else
938*33b1fccfSAndroid Build Coastguard Worker int main(int argc, char *argv[])
939*33b1fccfSAndroid Build Coastguard Worker #endif
940*33b1fccfSAndroid Build Coastguard Worker {
941*33b1fccfSAndroid Build Coastguard Worker 	int err;
942*33b1fccfSAndroid Build Coastguard Worker 
943*33b1fccfSAndroid Build Coastguard Worker 	erofs_init_configure();
944*33b1fccfSAndroid Build Coastguard Worker 
945*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.physical_blocks = 0;
946*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.logical_blocks = 0;
947*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.extract_path = NULL;
948*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.extract_pos = 0;
949*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.umask = umask(0);
950*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.superuser = geteuid() == 0;
951*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.corrupted = false;
952*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.print_comp_ratio = false;
953*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.check_decomp = false;
954*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.force = false;
955*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.overwrite = false;
956*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.preserve_owner = fsckcfg.superuser;
957*33b1fccfSAndroid Build Coastguard Worker 	fsckcfg.preserve_perms = fsckcfg.superuser;
958*33b1fccfSAndroid Build Coastguard Worker 
959*33b1fccfSAndroid Build Coastguard Worker 	err = erofsfsck_parse_options_cfg(argc, argv);
960*33b1fccfSAndroid Build Coastguard Worker 	if (err) {
961*33b1fccfSAndroid Build Coastguard Worker 		if (err == -EINVAL)
962*33b1fccfSAndroid Build Coastguard Worker 			fprintf(stderr, "Try '%s --help' for more information.\n", argv[0]);
963*33b1fccfSAndroid Build Coastguard Worker 		goto exit;
964*33b1fccfSAndroid Build Coastguard Worker 	}
965*33b1fccfSAndroid Build Coastguard Worker 
966*33b1fccfSAndroid Build Coastguard Worker #ifdef FUZZING
967*33b1fccfSAndroid Build Coastguard Worker 	cfg.c_dbg_lvl = -1;
968*33b1fccfSAndroid Build Coastguard Worker #endif
969*33b1fccfSAndroid Build Coastguard Worker 
970*33b1fccfSAndroid Build Coastguard Worker 	err = erofs_dev_open(&g_sbi, cfg.c_img_path, O_RDONLY);
971*33b1fccfSAndroid Build Coastguard Worker 	if (err) {
972*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to open image file");
973*33b1fccfSAndroid Build Coastguard Worker 		goto exit;
974*33b1fccfSAndroid Build Coastguard Worker 	}
975*33b1fccfSAndroid Build Coastguard Worker 
976*33b1fccfSAndroid Build Coastguard Worker 	err = erofs_read_superblock(&g_sbi);
977*33b1fccfSAndroid Build Coastguard Worker 	if (err) {
978*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to read superblock");
979*33b1fccfSAndroid Build Coastguard Worker 		goto exit_dev_close;
980*33b1fccfSAndroid Build Coastguard Worker 	}
981*33b1fccfSAndroid Build Coastguard Worker 
982*33b1fccfSAndroid Build Coastguard Worker 	if (erofs_sb_has_sb_chksum(&g_sbi) && erofs_check_sb_chksum()) {
983*33b1fccfSAndroid Build Coastguard Worker 		erofs_err("failed to verify superblock checksum");
984*33b1fccfSAndroid Build Coastguard Worker 		goto exit_put_super;
985*33b1fccfSAndroid Build Coastguard Worker 	}
986*33b1fccfSAndroid Build Coastguard Worker 
987*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.extract_path)
988*33b1fccfSAndroid Build Coastguard Worker 		erofsfsck_hardlink_init();
989*33b1fccfSAndroid Build Coastguard Worker 
990*33b1fccfSAndroid Build Coastguard Worker 	if (erofs_sb_has_fragments(&g_sbi) && g_sbi.packed_nid > 0) {
991*33b1fccfSAndroid Build Coastguard Worker 		err = erofsfsck_check_inode(g_sbi.packed_nid, g_sbi.packed_nid);
992*33b1fccfSAndroid Build Coastguard Worker 		if (err) {
993*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("failed to verify packed file");
994*33b1fccfSAndroid Build Coastguard Worker 			goto exit_hardlink;
995*33b1fccfSAndroid Build Coastguard Worker 		}
996*33b1fccfSAndroid Build Coastguard Worker 	}
997*33b1fccfSAndroid Build Coastguard Worker 
998*33b1fccfSAndroid Build Coastguard Worker 	err = erofsfsck_check_inode(g_sbi.root_nid, g_sbi.root_nid);
999*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.corrupted) {
1000*33b1fccfSAndroid Build Coastguard Worker 		if (!fsckcfg.extract_path)
1001*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("Found some filesystem corruption");
1002*33b1fccfSAndroid Build Coastguard Worker 		else
1003*33b1fccfSAndroid Build Coastguard Worker 			erofs_err("Failed to extract filesystem");
1004*33b1fccfSAndroid Build Coastguard Worker 		err = -EFSCORRUPTED;
1005*33b1fccfSAndroid Build Coastguard Worker 	} else if (!err) {
1006*33b1fccfSAndroid Build Coastguard Worker 		if (!fsckcfg.extract_path)
1007*33b1fccfSAndroid Build Coastguard Worker 			erofs_info("No errors found");
1008*33b1fccfSAndroid Build Coastguard Worker 		else
1009*33b1fccfSAndroid Build Coastguard Worker 			erofs_info("Extracted filesystem successfully");
1010*33b1fccfSAndroid Build Coastguard Worker 
1011*33b1fccfSAndroid Build Coastguard Worker 		if (fsckcfg.print_comp_ratio) {
1012*33b1fccfSAndroid Build Coastguard Worker 			double comp_ratio =
1013*33b1fccfSAndroid Build Coastguard Worker 				(double)fsckcfg.physical_blocks * 100 /
1014*33b1fccfSAndroid Build Coastguard Worker 				(double)fsckcfg.logical_blocks;
1015*33b1fccfSAndroid Build Coastguard Worker 
1016*33b1fccfSAndroid Build Coastguard Worker 			erofs_info("Compression ratio: %.2f(%%)", comp_ratio);
1017*33b1fccfSAndroid Build Coastguard Worker 		}
1018*33b1fccfSAndroid Build Coastguard Worker 	}
1019*33b1fccfSAndroid Build Coastguard Worker 
1020*33b1fccfSAndroid Build Coastguard Worker exit_hardlink:
1021*33b1fccfSAndroid Build Coastguard Worker 	if (fsckcfg.extract_path)
1022*33b1fccfSAndroid Build Coastguard Worker 		erofsfsck_hardlink_exit();
1023*33b1fccfSAndroid Build Coastguard Worker exit_put_super:
1024*33b1fccfSAndroid Build Coastguard Worker 	erofs_put_super(&g_sbi);
1025*33b1fccfSAndroid Build Coastguard Worker exit_dev_close:
1026*33b1fccfSAndroid Build Coastguard Worker 	erofs_dev_close(&g_sbi);
1027*33b1fccfSAndroid Build Coastguard Worker exit:
1028*33b1fccfSAndroid Build Coastguard Worker 	erofs_blob_closeall(&g_sbi);
1029*33b1fccfSAndroid Build Coastguard Worker 	erofs_exit_configure();
1030*33b1fccfSAndroid Build Coastguard Worker 	return err ? 1 : 0;
1031*33b1fccfSAndroid Build Coastguard Worker }
1032*33b1fccfSAndroid Build Coastguard Worker 
1033*33b1fccfSAndroid Build Coastguard Worker #ifdef FUZZING
LLVMFuzzerTestOneInput(const uint8_t * Data,size_t Size)1034*33b1fccfSAndroid Build Coastguard Worker int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size)
1035*33b1fccfSAndroid Build Coastguard Worker {
1036*33b1fccfSAndroid Build Coastguard Worker 	int fd, ret;
1037*33b1fccfSAndroid Build Coastguard Worker 	char filename[] = "/tmp/erofsfsck_libfuzzer_XXXXXX";
1038*33b1fccfSAndroid Build Coastguard Worker 	char *argv[] = {
1039*33b1fccfSAndroid Build Coastguard Worker 		"fsck.erofs",
1040*33b1fccfSAndroid Build Coastguard Worker 		"--extract",
1041*33b1fccfSAndroid Build Coastguard Worker 		filename,
1042*33b1fccfSAndroid Build Coastguard Worker 	};
1043*33b1fccfSAndroid Build Coastguard Worker 
1044*33b1fccfSAndroid Build Coastguard Worker 	fd = mkstemp(filename);
1045*33b1fccfSAndroid Build Coastguard Worker 	if (fd < 0)
1046*33b1fccfSAndroid Build Coastguard Worker 		return -errno;
1047*33b1fccfSAndroid Build Coastguard Worker 	if (write(fd, Data, Size) != Size) {
1048*33b1fccfSAndroid Build Coastguard Worker 		close(fd);
1049*33b1fccfSAndroid Build Coastguard Worker 		return -EIO;
1050*33b1fccfSAndroid Build Coastguard Worker 	}
1051*33b1fccfSAndroid Build Coastguard Worker 	close(fd);
1052*33b1fccfSAndroid Build Coastguard Worker 	ret = erofsfsck_fuzz_one(ARRAY_SIZE(argv), argv);
1053*33b1fccfSAndroid Build Coastguard Worker 	unlink(filename);
1054*33b1fccfSAndroid Build Coastguard Worker 	return ret ? -1 : 0;
1055*33b1fccfSAndroid Build Coastguard Worker }
1056*33b1fccfSAndroid Build Coastguard Worker #endif
1057