1 /*
2 * badblocks.c --- replace/append bad blocks to the bad block inode
3 *
4 * Copyright (C) 1993, 1994 Theodore Ts'o. This file may be
5 * redistributed under the terms of the GNU Public License.
6 */
7
8 #include <time.h>
9 #ifdef HAVE_ERRNO_H
10 #include <errno.h>
11 #endif
12
13 #include <et/com_err.h>
14 #include "e2fsck.h"
15
16 static int check_bb_inode_blocks(ext2_filsys fs, blk_t *block_nr, int blockcnt,
17 void *priv_data);
18
19
invalid_block(ext2_filsys fs EXT2FS_ATTR ((unused)),blk_t blk)20 static void invalid_block(ext2_filsys fs EXT2FS_ATTR((unused)), blk_t blk)
21 {
22 printf(_("Bad block %u out of range; ignored.\n"), blk);
23 return;
24 }
25
read_bad_blocks_file(e2fsck_t ctx,const char * bad_blocks_file,int replace_bad_blocks)26 void read_bad_blocks_file(e2fsck_t ctx, const char *bad_blocks_file,
27 int replace_bad_blocks)
28 {
29 ext2_filsys fs = ctx->fs;
30 errcode_t retval;
31 badblocks_list bb_list = 0;
32 FILE *f;
33 char buf[1024];
34
35 e2fsck_read_bitmaps(ctx);
36
37 /*
38 * Make sure the bad block inode is sane. If there are any
39 * illegal blocks, clear them.
40 */
41 retval = ext2fs_block_iterate(fs, EXT2_BAD_INO, 0, 0,
42 check_bb_inode_blocks, 0);
43 if (retval) {
44 com_err("ext2fs_block_iterate", retval, "%s",
45 _("while sanity checking the bad blocks inode"));
46 goto fatal;
47 }
48
49 /*
50 * If we're appending to the bad blocks inode, read in the
51 * current bad blocks.
52 */
53 if (!replace_bad_blocks) {
54 retval = ext2fs_read_bb_inode(fs, &bb_list);
55 if (retval) {
56 com_err("ext2fs_read_bb_inode", retval, "%s",
57 _("while reading the bad blocks inode"));
58 goto fatal;
59 }
60 }
61
62 /*
63 * Now read in the bad blocks from the file; if
64 * bad_blocks_file is null, then try to run the badblocks
65 * command.
66 */
67 if (bad_blocks_file) {
68 f = fopen(bad_blocks_file, "r");
69 if (!f) {
70 com_err("read_bad_blocks_file", errno,
71 _("while trying to open %s"), bad_blocks_file);
72 goto fatal;
73 }
74 } else {
75 sprintf(buf, "badblocks -b %d -X %s%s%s %llu", fs->blocksize,
76 (ctx->options & E2F_OPT_PREEN) ? "" : "-s ",
77 (ctx->options & E2F_OPT_WRITECHECK) ? "-n " : "",
78 fs->device_name, ext2fs_blocks_count(fs->super)-1);
79 f = popen(buf, "r");
80 if (!f) {
81 com_err("read_bad_blocks_file", errno,
82 _("while trying popen '%s'"), buf);
83 goto fatal;
84 }
85 }
86 retval = ext2fs_read_bb_FILE(fs, f, &bb_list, invalid_block);
87 if (bad_blocks_file)
88 fclose(f);
89 else
90 pclose(f);
91 if (retval) {
92 com_err("ext2fs_read_bb_FILE", retval, "%s",
93 _("while reading in list of bad blocks from file"));
94 goto fatal;
95 }
96
97 /*
98 * Finally, update the bad blocks from the bad_block_map
99 */
100 printf("%s: Updating bad block inode.\n", ctx->device_name);
101 retval = ext2fs_update_bb_inode(fs, bb_list);
102 if (retval) {
103 com_err("ext2fs_update_bb_inode", retval, "%s",
104 _("while updating bad block inode"));
105 goto fatal;
106 }
107
108 ext2fs_badblocks_list_free(bb_list);
109 return;
110
111 fatal:
112 ctx->flags |= E2F_FLAG_ABORT;
113 return;
114
115 }
116
check_bb_inode_blocks(ext2_filsys fs,blk_t * block_nr,int blockcnt EXT2FS_ATTR ((unused)),void * priv_data EXT2FS_ATTR ((unused)))117 static int check_bb_inode_blocks(ext2_filsys fs,
118 blk_t *block_nr,
119 int blockcnt EXT2FS_ATTR((unused)),
120 void *priv_data EXT2FS_ATTR((unused)))
121 {
122 if (!*block_nr)
123 return 0;
124
125 /*
126 * If the block number is outrageous, clear it and ignore it.
127 */
128 if (*block_nr >= ext2fs_blocks_count(fs->super) ||
129 *block_nr < fs->super->s_first_data_block) {
130 printf(_("Warning: illegal block %u found in bad block inode. "
131 "Cleared.\n"), *block_nr);
132 *block_nr = 0;
133 return BLOCK_CHANGED;
134 }
135
136 return 0;
137 }
138
139