xref: /aosp_15_r20/external/e2fsprogs/debugfs/icheck.c (revision 6a54128f25917bfc36a8a6e9d722c04a0b4641b6)
1*6a54128fSAndroid Build Coastguard Worker /*
2*6a54128fSAndroid Build Coastguard Worker  * icheck.c --- given a list of blocks, generate a list of inodes
3*6a54128fSAndroid Build Coastguard Worker  *
4*6a54128fSAndroid Build Coastguard Worker  * Copyright (C) 1994 Theodore Ts'o.  This file may be redistributed
5*6a54128fSAndroid Build Coastguard Worker  * under the terms of the GNU Public License.
6*6a54128fSAndroid Build Coastguard Worker  */
7*6a54128fSAndroid Build Coastguard Worker 
8*6a54128fSAndroid Build Coastguard Worker #include "config.h"
9*6a54128fSAndroid Build Coastguard Worker #include <stdio.h>
10*6a54128fSAndroid Build Coastguard Worker #include <unistd.h>
11*6a54128fSAndroid Build Coastguard Worker #include <stdlib.h>
12*6a54128fSAndroid Build Coastguard Worker #include <ctype.h>
13*6a54128fSAndroid Build Coastguard Worker #include <string.h>
14*6a54128fSAndroid Build Coastguard Worker #include <time.h>
15*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_ERRNO_H
16*6a54128fSAndroid Build Coastguard Worker #include <errno.h>
17*6a54128fSAndroid Build Coastguard Worker #endif
18*6a54128fSAndroid Build Coastguard Worker #include <sys/types.h>
19*6a54128fSAndroid Build Coastguard Worker 
20*6a54128fSAndroid Build Coastguard Worker #include "debugfs.h"
21*6a54128fSAndroid Build Coastguard Worker 
22*6a54128fSAndroid Build Coastguard Worker struct block_info {
23*6a54128fSAndroid Build Coastguard Worker 	blk64_t		blk;
24*6a54128fSAndroid Build Coastguard Worker 	ext2_ino_t	ino;
25*6a54128fSAndroid Build Coastguard Worker };
26*6a54128fSAndroid Build Coastguard Worker 
27*6a54128fSAndroid Build Coastguard Worker struct block_walk_struct {
28*6a54128fSAndroid Build Coastguard Worker 	struct block_info	*barray;
29*6a54128fSAndroid Build Coastguard Worker 	e2_blkcnt_t		blocks_left;
30*6a54128fSAndroid Build Coastguard Worker 	e2_blkcnt_t		num_blocks;
31*6a54128fSAndroid Build Coastguard Worker 	ext2_ino_t		inode;
32*6a54128fSAndroid Build Coastguard Worker };
33*6a54128fSAndroid Build Coastguard Worker 
icheck_proc(ext2_filsys fs EXT2FS_ATTR ((unused)),blk64_t * block_nr,e2_blkcnt_t blockcnt EXT2FS_ATTR ((unused)),blk64_t ref_block EXT2FS_ATTR ((unused)),int ref_offset EXT2FS_ATTR ((unused)),void * private)34*6a54128fSAndroid Build Coastguard Worker static int icheck_proc(ext2_filsys fs EXT2FS_ATTR((unused)),
35*6a54128fSAndroid Build Coastguard Worker 		       blk64_t	*block_nr,
36*6a54128fSAndroid Build Coastguard Worker 		       e2_blkcnt_t blockcnt EXT2FS_ATTR((unused)),
37*6a54128fSAndroid Build Coastguard Worker 		       blk64_t ref_block EXT2FS_ATTR((unused)),
38*6a54128fSAndroid Build Coastguard Worker 		       int ref_offset EXT2FS_ATTR((unused)),
39*6a54128fSAndroid Build Coastguard Worker 		       void *private)
40*6a54128fSAndroid Build Coastguard Worker {
41*6a54128fSAndroid Build Coastguard Worker 	struct block_walk_struct *bw = (struct block_walk_struct *) private;
42*6a54128fSAndroid Build Coastguard Worker 	e2_blkcnt_t	i;
43*6a54128fSAndroid Build Coastguard Worker 
44*6a54128fSAndroid Build Coastguard Worker 	for (i=0; i < bw->num_blocks; i++) {
45*6a54128fSAndroid Build Coastguard Worker 		if (!bw->barray[i].ino && bw->barray[i].blk == *block_nr) {
46*6a54128fSAndroid Build Coastguard Worker 			bw->barray[i].ino = bw->inode;
47*6a54128fSAndroid Build Coastguard Worker 			bw->blocks_left--;
48*6a54128fSAndroid Build Coastguard Worker 		}
49*6a54128fSAndroid Build Coastguard Worker 	}
50*6a54128fSAndroid Build Coastguard Worker 	if (!bw->blocks_left)
51*6a54128fSAndroid Build Coastguard Worker 		return BLOCK_ABORT;
52*6a54128fSAndroid Build Coastguard Worker 
53*6a54128fSAndroid Build Coastguard Worker 	return 0;
54*6a54128fSAndroid Build Coastguard Worker }
55*6a54128fSAndroid Build Coastguard Worker 
do_icheck(int argc,char ** argv,int sci_idx EXT2FS_ATTR ((unused)),void * infop EXT2FS_ATTR ((unused)))56*6a54128fSAndroid Build Coastguard Worker void do_icheck(int argc, char **argv, int sci_idx EXT2FS_ATTR((unused)),
57*6a54128fSAndroid Build Coastguard Worker 	       void *infop EXT2FS_ATTR((unused)))
58*6a54128fSAndroid Build Coastguard Worker {
59*6a54128fSAndroid Build Coastguard Worker 	struct block_walk_struct bw;
60*6a54128fSAndroid Build Coastguard Worker 	struct block_info	*binfo;
61*6a54128fSAndroid Build Coastguard Worker 	int			i;
62*6a54128fSAndroid Build Coastguard Worker 	ext2_inode_scan		scan = 0;
63*6a54128fSAndroid Build Coastguard Worker 	ext2_ino_t		ino;
64*6a54128fSAndroid Build Coastguard Worker 	struct ext2_inode	inode;
65*6a54128fSAndroid Build Coastguard Worker 	errcode_t		retval;
66*6a54128fSAndroid Build Coastguard Worker 	char			*block_buf;
67*6a54128fSAndroid Build Coastguard Worker 
68*6a54128fSAndroid Build Coastguard Worker 	if (argc < 2) {
69*6a54128fSAndroid Build Coastguard Worker 		com_err(argv[0], 0, "Usage: icheck <block number> ...");
70*6a54128fSAndroid Build Coastguard Worker 		return;
71*6a54128fSAndroid Build Coastguard Worker 	}
72*6a54128fSAndroid Build Coastguard Worker 	if (check_fs_open(argv[0]))
73*6a54128fSAndroid Build Coastguard Worker 		return;
74*6a54128fSAndroid Build Coastguard Worker 
75*6a54128fSAndroid Build Coastguard Worker 	bw.barray = malloc(sizeof(struct block_info) * argc);
76*6a54128fSAndroid Build Coastguard Worker 	if (!bw.barray) {
77*6a54128fSAndroid Build Coastguard Worker 		com_err("icheck", ENOMEM,
78*6a54128fSAndroid Build Coastguard Worker 			"while allocating inode info array");
79*6a54128fSAndroid Build Coastguard Worker 		return;
80*6a54128fSAndroid Build Coastguard Worker 	}
81*6a54128fSAndroid Build Coastguard Worker 	memset(bw.barray, 0, sizeof(struct block_info) * argc);
82*6a54128fSAndroid Build Coastguard Worker 
83*6a54128fSAndroid Build Coastguard Worker 	block_buf = malloc(current_fs->blocksize * 3);
84*6a54128fSAndroid Build Coastguard Worker 	if (!block_buf) {
85*6a54128fSAndroid Build Coastguard Worker 		com_err("icheck", ENOMEM, "while allocating block buffer");
86*6a54128fSAndroid Build Coastguard Worker 		goto error_out;
87*6a54128fSAndroid Build Coastguard Worker 	}
88*6a54128fSAndroid Build Coastguard Worker 
89*6a54128fSAndroid Build Coastguard Worker 	for (i=1; i < argc; i++) {
90*6a54128fSAndroid Build Coastguard Worker 		if (strtoblk(argv[0], argv[i], NULL, &bw.barray[i-1].blk))
91*6a54128fSAndroid Build Coastguard Worker 			goto error_out;
92*6a54128fSAndroid Build Coastguard Worker 	}
93*6a54128fSAndroid Build Coastguard Worker 
94*6a54128fSAndroid Build Coastguard Worker 	bw.num_blocks = bw.blocks_left = argc-1;
95*6a54128fSAndroid Build Coastguard Worker 
96*6a54128fSAndroid Build Coastguard Worker 	retval = ext2fs_open_inode_scan(current_fs, 0, &scan);
97*6a54128fSAndroid Build Coastguard Worker 	if (retval) {
98*6a54128fSAndroid Build Coastguard Worker 		com_err("icheck", retval, "while opening inode scan");
99*6a54128fSAndroid Build Coastguard Worker 		goto error_out;
100*6a54128fSAndroid Build Coastguard Worker 	}
101*6a54128fSAndroid Build Coastguard Worker 
102*6a54128fSAndroid Build Coastguard Worker 	do {
103*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_get_next_inode(scan, &ino, &inode);
104*6a54128fSAndroid Build Coastguard Worker 	} while (retval == EXT2_ET_BAD_BLOCK_IN_INODE_TABLE);
105*6a54128fSAndroid Build Coastguard Worker 	if (retval) {
106*6a54128fSAndroid Build Coastguard Worker 		com_err("icheck", retval, "while starting inode scan");
107*6a54128fSAndroid Build Coastguard Worker 		goto error_out;
108*6a54128fSAndroid Build Coastguard Worker 	}
109*6a54128fSAndroid Build Coastguard Worker 
110*6a54128fSAndroid Build Coastguard Worker 	while (ino) {
111*6a54128fSAndroid Build Coastguard Worker 		blk64_t blk;
112*6a54128fSAndroid Build Coastguard Worker 
113*6a54128fSAndroid Build Coastguard Worker 		if (!inode.i_links_count)
114*6a54128fSAndroid Build Coastguard Worker 			goto next;
115*6a54128fSAndroid Build Coastguard Worker 
116*6a54128fSAndroid Build Coastguard Worker 		bw.inode = ino;
117*6a54128fSAndroid Build Coastguard Worker 
118*6a54128fSAndroid Build Coastguard Worker 		blk = ext2fs_file_acl_block(current_fs, &inode);
119*6a54128fSAndroid Build Coastguard Worker 		if (blk) {
120*6a54128fSAndroid Build Coastguard Worker 			icheck_proc(current_fs, &blk, 0,
121*6a54128fSAndroid Build Coastguard Worker 				    0, 0, &bw);
122*6a54128fSAndroid Build Coastguard Worker 			if (bw.blocks_left == 0)
123*6a54128fSAndroid Build Coastguard Worker 				break;
124*6a54128fSAndroid Build Coastguard Worker 			ext2fs_file_acl_block_set(current_fs, &inode, blk);
125*6a54128fSAndroid Build Coastguard Worker 		}
126*6a54128fSAndroid Build Coastguard Worker 
127*6a54128fSAndroid Build Coastguard Worker 		if (!ext2fs_inode_has_valid_blocks2(current_fs, &inode))
128*6a54128fSAndroid Build Coastguard Worker 			goto next;
129*6a54128fSAndroid Build Coastguard Worker 		/*
130*6a54128fSAndroid Build Coastguard Worker 		 * To handle filesystems touched by 0.3c extfs; can be
131*6a54128fSAndroid Build Coastguard Worker 		 * removed later.
132*6a54128fSAndroid Build Coastguard Worker 		 */
133*6a54128fSAndroid Build Coastguard Worker 		if (inode.i_dtime)
134*6a54128fSAndroid Build Coastguard Worker 			goto next;
135*6a54128fSAndroid Build Coastguard Worker 
136*6a54128fSAndroid Build Coastguard Worker 		retval = ext2fs_block_iterate3(current_fs, ino,
137*6a54128fSAndroid Build Coastguard Worker 					       BLOCK_FLAG_READ_ONLY, block_buf,
138*6a54128fSAndroid Build Coastguard Worker 					       icheck_proc, &bw);
139*6a54128fSAndroid Build Coastguard Worker 		if (retval) {
140*6a54128fSAndroid Build Coastguard Worker 			com_err("icheck", retval,
141*6a54128fSAndroid Build Coastguard Worker 				"while calling ext2fs_block_iterate");
142*6a54128fSAndroid Build Coastguard Worker 			goto next;
143*6a54128fSAndroid Build Coastguard Worker 		}
144*6a54128fSAndroid Build Coastguard Worker 
145*6a54128fSAndroid Build Coastguard Worker 		if (bw.blocks_left == 0)
146*6a54128fSAndroid Build Coastguard Worker 			break;
147*6a54128fSAndroid Build Coastguard Worker 
148*6a54128fSAndroid Build Coastguard Worker 	next:
149*6a54128fSAndroid Build Coastguard Worker 		do {
150*6a54128fSAndroid Build Coastguard Worker 			retval = ext2fs_get_next_inode(scan, &ino, &inode);
151*6a54128fSAndroid Build Coastguard Worker 		} while (retval == EXT2_ET_BAD_BLOCK_IN_INODE_TABLE);
152*6a54128fSAndroid Build Coastguard Worker 		if (retval) {
153*6a54128fSAndroid Build Coastguard Worker 			com_err("icheck", retval,
154*6a54128fSAndroid Build Coastguard Worker 				"while doing inode scan");
155*6a54128fSAndroid Build Coastguard Worker 			goto error_out;
156*6a54128fSAndroid Build Coastguard Worker 		}
157*6a54128fSAndroid Build Coastguard Worker 	}
158*6a54128fSAndroid Build Coastguard Worker 
159*6a54128fSAndroid Build Coastguard Worker 	printf("Block\tInode number\n");
160*6a54128fSAndroid Build Coastguard Worker 	for (i=0, binfo = bw.barray; i < bw.num_blocks; i++, binfo++) {
161*6a54128fSAndroid Build Coastguard Worker 		if (binfo->ino == 0) {
162*6a54128fSAndroid Build Coastguard Worker 			printf("%llu\t<block not found>\n",
163*6a54128fSAndroid Build Coastguard Worker 			       (unsigned long long) binfo->blk);
164*6a54128fSAndroid Build Coastguard Worker 			continue;
165*6a54128fSAndroid Build Coastguard Worker 		}
166*6a54128fSAndroid Build Coastguard Worker 		printf("%llu\t%u\n", (unsigned long long) binfo->blk,
167*6a54128fSAndroid Build Coastguard Worker 		       binfo->ino);
168*6a54128fSAndroid Build Coastguard Worker 	}
169*6a54128fSAndroid Build Coastguard Worker 
170*6a54128fSAndroid Build Coastguard Worker error_out:
171*6a54128fSAndroid Build Coastguard Worker 	free(bw.barray);
172*6a54128fSAndroid Build Coastguard Worker 	free(block_buf);
173*6a54128fSAndroid Build Coastguard Worker 	if (scan)
174*6a54128fSAndroid Build Coastguard Worker 		ext2fs_close_inode_scan(scan);
175*6a54128fSAndroid Build Coastguard Worker 	return;
176*6a54128fSAndroid Build Coastguard Worker }
177