ocfs2: block read meta ecc.
Add block check calls to the read_block validate functions. This is the almost all of the read-side checking of metaecc. xattr buckets are not checked yet. Writes are also unchecked, and so a read-write mount will quickly fail. Signed-off-by: Joel Becker <joel.becker@oracle.com> Signed-off-by: Mark Fasheh <mfasheh@suse.com>
This commit is contained in:
parent
684ef27837
commit
d6b32bbb3e
@ -37,6 +37,7 @@
|
|||||||
|
|
||||||
#include "alloc.h"
|
#include "alloc.h"
|
||||||
#include "aops.h"
|
#include "aops.h"
|
||||||
|
#include "blockcheck.h"
|
||||||
#include "dlmglue.h"
|
#include "dlmglue.h"
|
||||||
#include "extent_map.h"
|
#include "extent_map.h"
|
||||||
#include "inode.h"
|
#include "inode.h"
|
||||||
@ -682,12 +683,28 @@ struct ocfs2_merge_ctxt {
|
|||||||
static int ocfs2_validate_extent_block(struct super_block *sb,
|
static int ocfs2_validate_extent_block(struct super_block *sb,
|
||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
|
int rc;
|
||||||
struct ocfs2_extent_block *eb =
|
struct ocfs2_extent_block *eb =
|
||||||
(struct ocfs2_extent_block *)bh->b_data;
|
(struct ocfs2_extent_block *)bh->b_data;
|
||||||
|
|
||||||
mlog(0, "Validating extent block %llu\n",
|
mlog(0, "Validating extent block %llu\n",
|
||||||
(unsigned long long)bh->b_blocknr);
|
(unsigned long long)bh->b_blocknr);
|
||||||
|
|
||||||
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
rc = ocfs2_validate_meta_ecc(sb, bh->b_data, &eb->h_check);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Errors after here are fatal.
|
||||||
|
*/
|
||||||
|
|
||||||
if (!OCFS2_IS_VALID_EXTENT_BLOCK(eb)) {
|
if (!OCFS2_IS_VALID_EXTENT_BLOCK(eb)) {
|
||||||
ocfs2_error(sb,
|
ocfs2_error(sb,
|
||||||
"Extent block #%llu has bad signature %.*s",
|
"Extent block #%llu has bad signature %.*s",
|
||||||
|
@ -24,6 +24,8 @@
|
|||||||
#include <linux/bitops.h>
|
#include <linux/bitops.h>
|
||||||
#include <asm/byteorder.h>
|
#include <asm/byteorder.h>
|
||||||
|
|
||||||
|
#include <cluster/masklog.h>
|
||||||
|
|
||||||
#include "ocfs2.h"
|
#include "ocfs2.h"
|
||||||
|
|
||||||
#include "blockcheck.h"
|
#include "blockcheck.h"
|
||||||
@ -292,6 +294,10 @@ int ocfs2_block_check_validate(void *data, size_t blocksize,
|
|||||||
if (crc == check.bc_crc32e)
|
if (crc == check.bc_crc32e)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
mlog(ML_ERROR,
|
||||||
|
"CRC32 failed: stored: %u, computed %u. Applying ECC.\n",
|
||||||
|
(unsigned int)check.bc_crc32e, (unsigned int)crc);
|
||||||
|
|
||||||
/* Ok, try ECC fixups */
|
/* Ok, try ECC fixups */
|
||||||
ecc = ocfs2_hamming_encode_block(data, blocksize);
|
ecc = ocfs2_hamming_encode_block(data, blocksize);
|
||||||
ocfs2_hamming_fix_block(data, blocksize, ecc ^ check.bc_ecc);
|
ocfs2_hamming_fix_block(data, blocksize, ecc ^ check.bc_ecc);
|
||||||
@ -301,6 +307,9 @@ int ocfs2_block_check_validate(void *data, size_t blocksize,
|
|||||||
if (crc == check.bc_crc32e)
|
if (crc == check.bc_crc32e)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
mlog(ML_ERROR, "Fixed CRC32 failed: stored: %u, computed %u\n",
|
||||||
|
(unsigned int)check.bc_crc32e, (unsigned int)crc);
|
||||||
|
|
||||||
rc = -EIO;
|
rc = -EIO;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
|
@ -38,6 +38,7 @@
|
|||||||
#include "ocfs2.h"
|
#include "ocfs2.h"
|
||||||
|
|
||||||
#include "alloc.h"
|
#include "alloc.h"
|
||||||
|
#include "blockcheck.h"
|
||||||
#include "dlmglue.h"
|
#include "dlmglue.h"
|
||||||
#include "extent_map.h"
|
#include "extent_map.h"
|
||||||
#include "file.h"
|
#include "file.h"
|
||||||
@ -1262,7 +1263,7 @@ void ocfs2_refresh_inode(struct inode *inode,
|
|||||||
int ocfs2_validate_inode_block(struct super_block *sb,
|
int ocfs2_validate_inode_block(struct super_block *sb,
|
||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
int rc = -EINVAL;
|
int rc;
|
||||||
struct ocfs2_dinode *di = (struct ocfs2_dinode *)bh->b_data;
|
struct ocfs2_dinode *di = (struct ocfs2_dinode *)bh->b_data;
|
||||||
|
|
||||||
mlog(0, "Validating dinode %llu\n",
|
mlog(0, "Validating dinode %llu\n",
|
||||||
@ -1270,6 +1271,21 @@ int ocfs2_validate_inode_block(struct super_block *sb,
|
|||||||
|
|
||||||
BUG_ON(!buffer_uptodate(bh));
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
rc = ocfs2_validate_meta_ecc(sb, bh->b_data, &di->i_check);
|
||||||
|
if (rc)
|
||||||
|
goto bail;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Errors after here are fatal.
|
||||||
|
*/
|
||||||
|
|
||||||
|
rc = -EINVAL;
|
||||||
|
|
||||||
if (!OCFS2_IS_VALID_DINODE(di)) {
|
if (!OCFS2_IS_VALID_DINODE(di)) {
|
||||||
ocfs2_error(sb, "Invalid dinode #%llu: signature = %.*s\n",
|
ocfs2_error(sb, "Invalid dinode #%llu: signature = %.*s\n",
|
||||||
(unsigned long long)bh->b_blocknr, 7,
|
(unsigned long long)bh->b_blocknr, 7,
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
#include "ocfs2_fs.h"
|
#include "ocfs2_fs.h"
|
||||||
#include "ocfs2.h"
|
#include "ocfs2.h"
|
||||||
#include "alloc.h"
|
#include "alloc.h"
|
||||||
|
#include "blockcheck.h"
|
||||||
#include "inode.h"
|
#include "inode.h"
|
||||||
#include "journal.h"
|
#include "journal.h"
|
||||||
#include "file.h"
|
#include "file.h"
|
||||||
@ -90,12 +91,20 @@ struct qtree_fmt_operations ocfs2_global_ops = {
|
|||||||
static int ocfs2_validate_quota_block(struct super_block *sb,
|
static int ocfs2_validate_quota_block(struct super_block *sb,
|
||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
struct ocfs2_disk_dqtrailer *dqt = ocfs2_dq_trailer(sb, bh->b_data);
|
struct ocfs2_disk_dqtrailer *dqt =
|
||||||
|
ocfs2_block_dqtrailer(sb->s_blocksize, bh->b_data);
|
||||||
|
|
||||||
mlog(0, "Validating quota block %llu\n",
|
mlog(0, "Validating quota block %llu\n",
|
||||||
(unsigned long long)bh->b_blocknr);
|
(unsigned long long)bh->b_blocknr);
|
||||||
|
|
||||||
return 0;
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
return ocfs2_validate_meta_ecc(sb, bh->b_data, &dqt->dq_check);
|
||||||
}
|
}
|
||||||
|
|
||||||
int ocfs2_read_quota_block(struct inode *inode, u64 v_block,
|
int ocfs2_read_quota_block(struct inode *inode, u64 v_block,
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
#include "ocfs2.h"
|
#include "ocfs2.h"
|
||||||
|
|
||||||
#include "alloc.h"
|
#include "alloc.h"
|
||||||
|
#include "blockcheck.h"
|
||||||
#include "dlmglue.h"
|
#include "dlmglue.h"
|
||||||
#include "inode.h"
|
#include "inode.h"
|
||||||
#include "journal.h"
|
#include "journal.h"
|
||||||
@ -250,8 +251,18 @@ int ocfs2_check_group_descriptor(struct super_block *sb,
|
|||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
|
struct ocfs2_group_desc *gd = (struct ocfs2_group_desc *)bh->b_data;
|
||||||
|
|
||||||
rc = ocfs2_validate_gd_self(sb, bh, 1);
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
rc = ocfs2_validate_meta_ecc(sb, bh->b_data, &gd->bg_check);
|
||||||
|
if (!rc)
|
||||||
|
rc = ocfs2_validate_gd_self(sb, bh, 1);
|
||||||
if (!rc)
|
if (!rc)
|
||||||
rc = ocfs2_validate_gd_parent(sb, di, bh, 1);
|
rc = ocfs2_validate_gd_parent(sb, di, bh, 1);
|
||||||
|
|
||||||
@ -261,9 +272,27 @@ int ocfs2_check_group_descriptor(struct super_block *sb,
|
|||||||
static int ocfs2_validate_group_descriptor(struct super_block *sb,
|
static int ocfs2_validate_group_descriptor(struct super_block *sb,
|
||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
|
int rc;
|
||||||
|
struct ocfs2_group_desc *gd = (struct ocfs2_group_desc *)bh->b_data;
|
||||||
|
|
||||||
mlog(0, "Validating group descriptor %llu\n",
|
mlog(0, "Validating group descriptor %llu\n",
|
||||||
(unsigned long long)bh->b_blocknr);
|
(unsigned long long)bh->b_blocknr);
|
||||||
|
|
||||||
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
rc = ocfs2_validate_meta_ecc(sb, bh->b_data, &gd->bg_check);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Errors after here are fatal.
|
||||||
|
*/
|
||||||
|
|
||||||
return ocfs2_validate_gd_self(sb, bh, 0);
|
return ocfs2_validate_gd_self(sb, bh, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,6 +42,7 @@
|
|||||||
|
|
||||||
#include "ocfs2.h"
|
#include "ocfs2.h"
|
||||||
#include "alloc.h"
|
#include "alloc.h"
|
||||||
|
#include "blockcheck.h"
|
||||||
#include "dlmglue.h"
|
#include "dlmglue.h"
|
||||||
#include "file.h"
|
#include "file.h"
|
||||||
#include "symlink.h"
|
#include "symlink.h"
|
||||||
@ -322,12 +323,28 @@ static void ocfs2_xattr_bucket_copy_data(struct ocfs2_xattr_bucket *dest,
|
|||||||
static int ocfs2_validate_xattr_block(struct super_block *sb,
|
static int ocfs2_validate_xattr_block(struct super_block *sb,
|
||||||
struct buffer_head *bh)
|
struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
|
int rc;
|
||||||
struct ocfs2_xattr_block *xb =
|
struct ocfs2_xattr_block *xb =
|
||||||
(struct ocfs2_xattr_block *)bh->b_data;
|
(struct ocfs2_xattr_block *)bh->b_data;
|
||||||
|
|
||||||
mlog(0, "Validating xattr block %llu\n",
|
mlog(0, "Validating xattr block %llu\n",
|
||||||
(unsigned long long)bh->b_blocknr);
|
(unsigned long long)bh->b_blocknr);
|
||||||
|
|
||||||
|
BUG_ON(!buffer_uptodate(bh));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the ecc fails, we return the error but otherwise
|
||||||
|
* leave the filesystem running. We know any error is
|
||||||
|
* local to this block.
|
||||||
|
*/
|
||||||
|
rc = ocfs2_validate_meta_ecc(sb, bh->b_data, &xb->xb_check);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Errors after here are fatal
|
||||||
|
*/
|
||||||
|
|
||||||
if (!OCFS2_IS_VALID_XATTR_BLOCK(xb)) {
|
if (!OCFS2_IS_VALID_XATTR_BLOCK(xb)) {
|
||||||
ocfs2_error(sb,
|
ocfs2_error(sb,
|
||||||
"Extended attribute block #%llu has bad "
|
"Extended attribute block #%llu has bad "
|
||||||
|
Loading…
Reference in New Issue
Block a user