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