예제 #1
0
/*
 * This routine gets the lost_and_found inode, making it a directory
 * if necessary
 */
ext2_ino_t e2fsck_get_lost_and_found(e2fsck_t ctx, int fix)
{
    ext2_filsys fs = ctx->fs;
    ext2_ino_t			ino;
    blk64_t			blk;
    errcode_t		retval;
    struct ext2_inode	inode;
    char *			block;
    static const char	name[] = "lost+found";
    struct 	problem_context	pctx;

    if (ctx->lost_and_found)
        return ctx->lost_and_found;

    clear_problem_context(&pctx);

    retval = ext2fs_lookup(fs, EXT2_ROOT_INO, name,
                           sizeof(name)-1, 0, &ino);
    if (retval && !fix)
        return 0;
    if (!retval) {
        if (ext2fs_test_inode_bitmap2(ctx->inode_dir_map, ino)) {
            ctx->lost_and_found = ino;
            return ino;
        }

        /* Lost+found isn't a directory! */
        if (!fix)
            return 0;
        pctx.ino = ino;
        if (!fix_problem(ctx, PR_3_LPF_NOTDIR, &pctx))
            return 0;

        /* OK, unlink the old /lost+found file. */
        pctx.errcode = ext2fs_unlink(fs, EXT2_ROOT_INO, name, ino, 0);
        if (pctx.errcode) {
            pctx.str = "ext2fs_unlink";
            fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
            return 0;
        }
        (void) e2fsck_dir_info_set_parent(ctx, ino, 0);
        e2fsck_adjust_inode_count(ctx, ino, -1);
    } else if (retval != EXT2_ET_FILE_NOT_FOUND) {
        pctx.errcode = retval;
        fix_problem(ctx, PR_3_ERR_FIND_LPF, &pctx);
    }
    if (!fix_problem(ctx, PR_3_NO_LF_DIR, 0))
        return 0;

    /*
     * Read the inode and block bitmaps in; we'll be messing with
     * them.
     */
    e2fsck_read_bitmaps(ctx);

    /*
     * First, find a free block
     */
    retval = ext2fs_new_block2(fs, 0, ctx->block_found_map, &blk);
    if (retval) {
        pctx.errcode = retval;
        fix_problem(ctx, PR_3_ERR_LPF_NEW_BLOCK, &pctx);
        return 0;
    }
    ext2fs_mark_block_bitmap2(ctx->block_found_map, blk);
    ext2fs_block_alloc_stats2(fs, blk, +1);

    /*
     * Next find a free inode.
     */
    retval = ext2fs_new_inode(fs, EXT2_ROOT_INO, 040700,
                              ctx->inode_used_map, &ino);
    if (retval) {
        pctx.errcode = retval;
        fix_problem(ctx, PR_3_ERR_LPF_NEW_INODE, &pctx);
        return 0;
    }
    ext2fs_mark_inode_bitmap2(ctx->inode_used_map, ino);
    ext2fs_mark_inode_bitmap2(ctx->inode_dir_map, ino);
    ext2fs_inode_alloc_stats2(fs, ino, +1, 1);

    /*
     * Now let's create the actual data block for the inode
     */
    retval = ext2fs_new_dir_block(fs, ino, EXT2_ROOT_INO, &block);
    if (retval) {
        pctx.errcode = retval;
        fix_problem(ctx, PR_3_ERR_LPF_NEW_DIR_BLOCK, &pctx);
        return 0;
    }

    retval = ext2fs_write_dir_block(fs, blk, block);
    ext2fs_free_mem(&block);
    if (retval) {
        pctx.errcode = retval;
        fix_problem(ctx, PR_3_ERR_LPF_WRITE_BLOCK, &pctx);
        return 0;
    }

    /*
     * Set up the inode structure
     */
    memset(&inode, 0, sizeof(inode));
    inode.i_mode = 040700;
    inode.i_size = fs->blocksize;
    inode.i_atime = inode.i_ctime = inode.i_mtime = ctx->now;
    inode.i_links_count = 2;
    ext2fs_iblk_set(fs, &inode, 1);
    inode.i_block[0] = blk;

    /*
     * Next, write out the inode.
     */
    pctx.errcode = ext2fs_write_new_inode(fs, ino, &inode);
    if (pctx.errcode) {
        pctx.str = "ext2fs_write_inode";
        fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
        return 0;
    }
    /*
     * Finally, create the directory link
     */
    pctx.errcode = ext2fs_link(fs, EXT2_ROOT_INO, name, ino, EXT2_FT_DIR);
    if (pctx.errcode) {
        pctx.str = "ext2fs_link";
        fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
        return 0;
    }

    /*
     * Miscellaneous bookkeeping that needs to be kept straight.
     */
    e2fsck_add_dir_info(ctx, ino, EXT2_ROOT_INO);
    e2fsck_adjust_inode_count(ctx, EXT2_ROOT_INO, 1);
    ext2fs_icount_store(ctx->inode_count, ino, 2);
    ext2fs_icount_store(ctx->inode_link_info, ino, 2);
    ctx->lost_and_found = ino;
    quota_data_add(ctx->qctx, &inode, ino, fs->blocksize);
    quota_data_inodes(ctx->qctx, &inode, ino, +1);
#if 0
    printf("/lost+found created; inode #%lu\n", ino);
#endif
    return ino;
}
예제 #2
0
/*
 * This routine gets the lost_and_found inode, making it a directory
 * if necessary
 */
ext2_ino_t e2fsck_get_lost_and_found(e2fsck_t ctx, int fix)
{
	ext2_filsys fs = ctx->fs;
	ext2_ino_t			ino;
	blk64_t			blk;
	errcode_t		retval;
	struct ext2_inode	inode;
	char *			block;
	static const char	name[] = "lost+found";
	struct 	problem_context	pctx;
	int			will_rehash, flags;

	if (ctx->lost_and_found)
		return ctx->lost_and_found;

	clear_problem_context(&pctx);

	will_rehash = e2fsck_dir_will_be_rehashed(ctx, EXT2_ROOT_INO);
	if (will_rehash) {
		flags = ctx->fs->flags;
		ctx->fs->flags |= EXT2_FLAG_IGNORE_CSUM_ERRORS;
	}
	retval = ext2fs_lookup(fs, EXT2_ROOT_INO, name,
			       sizeof(name)-1, 0, &ino);
	if (will_rehash)
		ctx->fs->flags = (flags & EXT2_FLAG_IGNORE_CSUM_ERRORS) |
			(ctx->fs->flags & ~EXT2_FLAG_IGNORE_CSUM_ERRORS);
	if (retval && !fix)
		return 0;
	if (!retval) {
		/* Lost+found shouldn't have inline data */
		retval = ext2fs_read_inode(fs, ino, &inode);
		if (fix && retval)
			return 0;

		if (fix && (inode.i_flags & EXT4_INLINE_DATA_FL)) {
			if (!fix_problem(ctx, PR_3_LPF_INLINE_DATA, &pctx))
				return 0;
			goto unlink;
		}

		if (fix && (inode.i_flags & EXT4_ENCRYPT_FL)) {
			if (!fix_problem(ctx, PR_3_LPF_ENCRYPTED, &pctx))
				return 0;
			goto unlink;
		}

		if (ext2fs_check_directory(fs, ino) == 0) {
			ctx->lost_and_found = ino;
			return ino;
		}

		/* Lost+found isn't a directory! */
		if (!fix)
			return 0;
		pctx.ino = ino;
		if (!fix_problem(ctx, PR_3_LPF_NOTDIR, &pctx))
			return 0;

unlink:
		/* OK, unlink the old /lost+found file. */
		pctx.errcode = ext2fs_unlink(fs, EXT2_ROOT_INO, name, ino, 0);
		if (pctx.errcode) {
			pctx.str = "ext2fs_unlink";
			fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
			return 0;
		}
		(void) e2fsck_dir_info_set_parent(ctx, ino, 0);
		e2fsck_adjust_inode_count(ctx, ino, -1);
		/*
		 * If the old lost+found was a directory, we've just
		 * disconnected it from the directory tree, which
		 * means we need to restart the directory tree scan.
		 * The simplest way to do this is restart the whole
		 * e2fsck operation.
		 */
		if (LINUX_S_ISDIR(inode.i_mode))
			ctx->flags |= E2F_FLAG_RESTART;
	} else if (retval != EXT2_ET_FILE_NOT_FOUND) {
		pctx.errcode = retval;
		fix_problem(ctx, PR_3_ERR_FIND_LPF, &pctx);
	}
	if (!fix_problem(ctx, PR_3_NO_LF_DIR, 0))
		return 0;

	/*
	 * Read the inode and block bitmaps in; we'll be messing with
	 * them.
	 */
	e2fsck_read_bitmaps(ctx);

	/*
	 * First, find a free block
	 */
	if (ctx->lnf_repair_block) {
		blk = ctx->lnf_repair_block;
		ctx->lnf_repair_block = 0;
		goto skip_new_block;
	}
	retval = ext2fs_new_block2(fs, 0, ctx->block_found_map, &blk);
	if (retval == EXT2_ET_BLOCK_ALLOC_FAIL &&
	    fix_problem(ctx, PR_3_LPF_NO_SPACE, &pctx)) {
		fix_problem(ctx, PR_3_NO_SPACE_TO_RECOVER, &pctx);
		ctx->lost_and_found = EXT2_ROOT_INO;
		return 0;
	}
	if (retval) {
		pctx.errcode = retval;
		fix_problem(ctx, PR_3_ERR_LPF_NEW_BLOCK, &pctx);
		return 0;
	}
	ext2fs_mark_block_bitmap2(ctx->block_found_map, blk);
skip_new_block:
	ext2fs_block_alloc_stats2(fs, blk, +1);

	/*
	 * Next find a free inode.
	 */
	retval = ext2fs_new_inode(fs, EXT2_ROOT_INO, 040700,
				  ctx->inode_used_map, &ino);
	if (retval == EXT2_ET_INODE_ALLOC_FAIL &&
	    fix_problem(ctx, PR_3_LPF_NO_SPACE, &pctx)) {
		fix_problem(ctx, PR_3_NO_SPACE_TO_RECOVER, &pctx);
		ctx->lost_and_found = EXT2_ROOT_INO;
		return 0;
	}
	if (retval) {
		pctx.errcode = retval;
		fix_problem(ctx, PR_3_ERR_LPF_NEW_INODE, &pctx);
		return 0;
	}
	ext2fs_mark_inode_bitmap2(ctx->inode_used_map, ino);
	ext2fs_mark_inode_bitmap2(ctx->inode_dir_map, ino);
	ext2fs_inode_alloc_stats2(fs, ino, +1, 1);

	/*
	 * Set up the inode structure
	 */
	memset(&inode, 0, sizeof(inode));
	inode.i_mode = 040700;
	inode.i_size = fs->blocksize;
	inode.i_atime = inode.i_ctime = inode.i_mtime = ctx->now;
	inode.i_links_count = 2;
	ext2fs_iblk_set(fs, &inode, 1);
	inode.i_block[0] = blk;

	/*
	 * Next, write out the inode.
	 */
	pctx.errcode = ext2fs_write_new_inode(fs, ino, &inode);
	if (pctx.errcode) {
		pctx.str = "ext2fs_write_inode";
		fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
		return 0;
	}

	/*
	 * Now let's create the actual data block for the inode.
	 * Due to metadata_csum, the directory block MUST be written
	 * after the inode is written to disk!
	 */
	retval = ext2fs_new_dir_block(fs, ino, EXT2_ROOT_INO, &block);
	if (retval) {
		pctx.errcode = retval;
		fix_problem(ctx, PR_3_ERR_LPF_NEW_DIR_BLOCK, &pctx);
		return 0;
	}

	retval = ext2fs_write_dir_block4(fs, blk, block, 0, ino);
	ext2fs_free_mem(&block);
	if (retval) {
		pctx.errcode = retval;
		fix_problem(ctx, PR_3_ERR_LPF_WRITE_BLOCK, &pctx);
		return 0;
	}

	/*
	 * Finally, create the directory link
	 */
	pctx.errcode = ext2fs_link(fs, EXT2_ROOT_INO, name, ino, EXT2_FT_DIR);
	if (pctx.errcode == EXT2_ET_DIR_NO_SPACE) {
		pctx.errcode = ext2fs_expand_dir(fs, EXT2_ROOT_INO);
		if (pctx.errcode)
			goto link_error;
		pctx.errcode = ext2fs_link(fs, EXT2_ROOT_INO, name, ino,
					   EXT2_FT_DIR);
	}
	if (pctx.errcode) {
link_error:
		pctx.str = "ext2fs_link";
		fix_problem(ctx, PR_3_CREATE_LPF_ERROR, &pctx);
		return 0;
	}

	/*
	 * Miscellaneous bookkeeping that needs to be kept straight.
	 */
	e2fsck_add_dir_info(ctx, ino, EXT2_ROOT_INO);
	e2fsck_adjust_inode_count(ctx, EXT2_ROOT_INO, 1);
	ext2fs_icount_store(ctx->inode_count, ino, 2);
	ext2fs_icount_store(ctx->inode_link_info, ino, 2);
	ctx->lost_and_found = ino;
	quota_data_add(ctx->qctx, &inode, ino, fs->blocksize);
	quota_data_inodes(ctx->qctx, &inode, ino, +1);
#if 0
	printf("/lost+found created; inode #%lu\n", ino);
#endif
	return ino;
}