mirror of
https://git.kernel.org/pub/scm/fs/ext2/e2fsprogs.git
synced 2025-01-01 14:13:26 +08:00
f3db3566b5
Checkin of e2fsprogs 0.5b
163 lines
3.5 KiB
C
163 lines
3.5 KiB
C
/*
|
|
* badblocks.c --- replace/append bad blocks to the bad block inode
|
|
*
|
|
* Copyright (C) 1993, 1994 Theodore Ts'o. This file may be
|
|
* redistributed under the terms of the GNU Public License.
|
|
*/
|
|
|
|
#include <time.h>
|
|
|
|
#include <et/com_err.h>
|
|
#include "e2fsck.h"
|
|
|
|
static int check_bb_inode_blocks(ext2_filsys fs, blk_t *block_nr, int blockcnt,
|
|
void *private);
|
|
|
|
|
|
static void invalid_block(ext2_filsys fs, blk_t blk)
|
|
{
|
|
printf("Bad block %lu out of range; ignored.\n", blk);
|
|
return;
|
|
}
|
|
|
|
void read_bad_blocks_file(ext2_filsys fs, const char *bad_blocks_file,
|
|
int replace_bad_blocks)
|
|
{
|
|
errcode_t retval;
|
|
badblocks_list bb_list = 0;
|
|
FILE *f;
|
|
|
|
read_bitmaps(fs);
|
|
|
|
/*
|
|
* Make sure the bad block inode is sane. If there are any
|
|
* illegal blocks, clear them.
|
|
*/
|
|
retval = ext2fs_block_iterate(fs, EXT2_BAD_INO, 0, 0,
|
|
check_bb_inode_blocks, 0);
|
|
if (retval) {
|
|
com_err("ext2fs_block_iterate", retval,
|
|
"while sanity checking the bad blocks inode");
|
|
fatal_error(0);
|
|
}
|
|
|
|
|
|
/*
|
|
* If we're appending to the bad blocks inode, read in the
|
|
* current bad blocks.
|
|
*/
|
|
if (!replace_bad_blocks) {
|
|
retval = ext2fs_read_bb_inode(fs, &bb_list);
|
|
if (retval) {
|
|
com_err("ext2fs_read_bb_inode", retval,
|
|
"while reading the bad blocks inode");
|
|
fatal_error(0);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Now read in the bad blocks from the file.
|
|
*/
|
|
f = fopen(bad_blocks_file, "r");
|
|
if (!f) {
|
|
com_err("read_bad_blocks_file", errno,
|
|
"while trying to open %s", bad_blocks_file);
|
|
fatal_error(0);
|
|
}
|
|
retval = ext2fs_read_bb_FILE(fs, f, &bb_list, invalid_block);
|
|
fclose (f);
|
|
if (retval) {
|
|
com_err("ext2fs_read_bb_FILE", retval,
|
|
"while reading in list of bad blocks from file");
|
|
fatal_error(0);
|
|
}
|
|
|
|
/*
|
|
* Finally, update the bad blocks from the bad_block_map
|
|
*/
|
|
retval = ext2fs_update_bb_inode(fs, bb_list);
|
|
if (retval) {
|
|
com_err("ext2fs_update_bb_inode", retval,
|
|
"while updating bad block inode");
|
|
fatal_error(0);
|
|
}
|
|
|
|
badblocks_list_free(bb_list);
|
|
return;
|
|
}
|
|
|
|
static int check_bb_inode_blocks(ext2_filsys fs, blk_t *block_nr, int blockcnt,
|
|
void *private)
|
|
{
|
|
if (!*block_nr)
|
|
return 0;
|
|
|
|
/*
|
|
* If the block number is outrageous, clear it and ignore it.
|
|
*/
|
|
if (*block_nr >= fs->super->s_blocks_count ||
|
|
*block_nr < fs->super->s_first_data_block) {
|
|
printf("Warning illegal block %lu found in bad block inode. Cleared.\n", *block_nr);
|
|
*block_nr = 0;
|
|
return BLOCK_CHANGED;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void test_disk(ext2_filsys fs)
|
|
{
|
|
errcode_t retval;
|
|
badblocks_list bb_list = 0;
|
|
FILE *f;
|
|
char buf[1024];
|
|
|
|
read_bitmaps(fs);
|
|
|
|
/*
|
|
* Always read in the current list of bad blocks.
|
|
*/
|
|
retval = ext2fs_read_bb_inode(fs, &bb_list);
|
|
if (retval) {
|
|
com_err("ext2fs_read_bb_inode", retval,
|
|
"while reading the bad blocks inode");
|
|
fatal_error(0);
|
|
}
|
|
|
|
/*
|
|
* Now run the bad blocks program
|
|
*/
|
|
sprintf(buf, "badblocks %s%s %ld", preen ? "" : "-s ",
|
|
fs->device_name,
|
|
fs->super->s_blocks_count);
|
|
if (verbose)
|
|
printf("Running command: %s\n", buf);
|
|
f = popen(buf, "r");
|
|
if (!f) {
|
|
com_err("popen", errno,
|
|
"while trying to run %s", buf);
|
|
fatal_error(0);
|
|
}
|
|
retval = ext2fs_read_bb_FILE(fs, f, &bb_list, invalid_block);
|
|
fclose (f);
|
|
if (retval) {
|
|
com_err("ext2fs_read_bb_FILE", retval,
|
|
"while processing list of bad blocks from program");
|
|
fatal_error(0);
|
|
}
|
|
|
|
/*
|
|
* Finally, update the bad blocks from the bad_block_map
|
|
*/
|
|
retval = ext2fs_update_bb_inode(fs, bb_list);
|
|
if (retval) {
|
|
com_err("ext2fs_update_bb_inode", retval,
|
|
"while updating bad block inode");
|
|
fatal_error(0);
|
|
}
|
|
|
|
badblocks_list_free(bb_list);
|
|
return;
|
|
}
|
|
|