aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorStephen C. Tweedie <sct@redhat.com>2005-01-04 05:27:49 -0800
committerLinus Torvalds <torvalds@ppc970.osdl.org>2005-01-04 05:27:49 -0800
commita319278834bd1e9e7b14910ffe3ddf7485817941 (patch)
tree2c2ef65a6da3096dcf52760f312ddfcdf62fab2b /fs
parentc579b4e22af013e1c6745023a6bb37756d38fc52 (diff)
downloadhistory-a319278834bd1e9e7b14910ffe3ddf7485817941.tar.gz
[PATCH] ext3: handle attempted double-delete of metadata.
This patch improves ext3's ability to deal with corruption on-disk. If we try to delete a metadata block twice, we confuse ext3's internal revoke error-checking, resulting in a BUG(). But this can occur in practice due to a corrupt indirect block, so we should attempt to fail gracefully. Downgrade the assert failure to a JH_EXPECT_BH failure, and return EIO when it occurs. This is easily reproduced with a sample ext3 fs image containing an inode which references the same indirect block more than once. Deleting that inode will BUG() an unfixed kernel with: Assertion failure in journal_revoke() at fs/jbd/revoke.c:379: "!buffer_revoked(bh)" With the fix, ext3 recovers gracefully. Signed-off-by: Stephen Tweedie <sct@redhat.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'fs')
-rw-r--r--fs/jbd/revoke.c7
1 files changed, 6 insertions, 1 deletions
diff --git a/fs/jbd/revoke.c b/fs/jbd/revoke.c
index 0bb60dba76cc7c..d327a598f86128 100644
--- a/fs/jbd/revoke.c
+++ b/fs/jbd/revoke.c
@@ -376,7 +376,12 @@ int journal_revoke(handle_t *handle, unsigned long blocknr,
first having the revoke cancelled: it's illegal to free a
block twice without allocating it in between! */
if (bh) {
- J_ASSERT_BH(bh, !buffer_revoked(bh));
+ if (!J_EXPECT_BH(bh, !buffer_revoked(bh),
+ "inconsistent data on disk")) {
+ if (!bh_in)
+ brelse(bh);
+ return -EIO;
+ }
set_buffer_revoked(bh);
set_buffer_revokevalid(bh);
if (bh_in) {