Commit 52c6e6f9 authored by Artem Bityutskiy's avatar Artem Bityutskiy

UBIFS: seek journal heads to the latest bud in replay

This is the second fix of the following symptom:

UBIFS error (pid 34456): could not find an empty LEB

which sometimes happens after power cuts when we mount the file-system - UBIFS
refuses it with the above error message which comes from the
'ubifs_rcvry_gc_commit()' function. I can reproduce this using the integck test
with the UBIFS power cut emulation enabled.

Analysis of the problem.

Currently UBIFS replay seeks the journal heads to the last _replayed_ bud.
But the buds are replayed out-of-order, so the replay basically seeks journal
heads to the "random" bud belonging to this head, and not to the _last_ one.

The result of this is that the GC head may be seeked to a full LEB with no free
space, or very little free space. And 'ubifs_rcvry_gc_commit()' tries to find a
fully or mostly dirty LEB to match the current GC head (because we need to
garbage-collect that dirty LEB at one go, because we do not have @c->gc_lnum).
So 'ubifs_find_dirty_leb()' fails and we fall back to finding an empty LEB and
also fail. As a result - recovery fails and mounting fails.

This patch teaches the replay to initialize the GC heads exactly to the latest
buds, i.e. the buds which have the largest sequence number in corresponding
log reference nodes.
Signed-off-by: default avatarArtem Bityutskiy <Artem.Bityutskiy@nokia.com>
Cc: stable@kernel.org
parent b50b9f40
...@@ -59,6 +59,7 @@ enum { ...@@ -59,6 +59,7 @@ enum {
* @new_size: truncation new size * @new_size: truncation new size
* @free: amount of free space in a bud * @free: amount of free space in a bud
* @dirty: amount of dirty space in a bud from padding and deletion nodes * @dirty: amount of dirty space in a bud from padding and deletion nodes
* @jhead: journal head number of the bud
* *
* UBIFS journal replay must compare node sequence numbers, which means it must * UBIFS journal replay must compare node sequence numbers, which means it must
* build a tree of node information to insert into the TNC. * build a tree of node information to insert into the TNC.
...@@ -80,6 +81,7 @@ struct replay_entry { ...@@ -80,6 +81,7 @@ struct replay_entry {
struct { struct {
int free; int free;
int dirty; int dirty;
int jhead;
}; };
}; };
}; };
...@@ -159,6 +161,11 @@ static int set_bud_lprops(struct ubifs_info *c, struct replay_entry *r) ...@@ -159,6 +161,11 @@ static int set_bud_lprops(struct ubifs_info *c, struct replay_entry *r)
err = PTR_ERR(lp); err = PTR_ERR(lp);
goto out; goto out;
} }
/* Make sure the journal head points to the latest bud */
err = ubifs_wbuf_seek_nolock(&c->jheads[r->jhead].wbuf, r->lnum,
c->leb_size - r->free, UBI_SHORTTERM);
out: out:
ubifs_release_lprops(c); ubifs_release_lprops(c);
return err; return err;
...@@ -627,10 +634,6 @@ static int replay_bud(struct ubifs_info *c, int lnum, int offs, int jhead, ...@@ -627,10 +634,6 @@ static int replay_bud(struct ubifs_info *c, int lnum, int offs, int jhead,
ubifs_assert(sleb->endpt - offs >= used); ubifs_assert(sleb->endpt - offs >= used);
ubifs_assert(sleb->endpt % c->min_io_size == 0); ubifs_assert(sleb->endpt % c->min_io_size == 0);
if (sleb->endpt + c->min_io_size <= c->leb_size && !c->ro_mount)
err = ubifs_wbuf_seek_nolock(&c->jheads[jhead].wbuf, lnum,
sleb->endpt, UBI_SHORTTERM);
*dirty = sleb->endpt - offs - used; *dirty = sleb->endpt - offs - used;
*free = c->leb_size - sleb->endpt; *free = c->leb_size - sleb->endpt;
...@@ -653,12 +656,14 @@ static int replay_bud(struct ubifs_info *c, int lnum, int offs, int jhead, ...@@ -653,12 +656,14 @@ static int replay_bud(struct ubifs_info *c, int lnum, int offs, int jhead,
* @sqnum: sequence number * @sqnum: sequence number
* @free: amount of free space in bud * @free: amount of free space in bud
* @dirty: amount of dirty space from padding and deletion nodes * @dirty: amount of dirty space from padding and deletion nodes
* @jhead: journal head number for the bud
* *
* This function inserts a reference node to the replay tree and returns zero * This function inserts a reference node to the replay tree and returns zero
* in case of success or a negative error code in case of failure. * in case of success or a negative error code in case of failure.
*/ */
static int insert_ref_node(struct ubifs_info *c, int lnum, int offs, static int insert_ref_node(struct ubifs_info *c, int lnum, int offs,
unsigned long long sqnum, int free, int dirty) unsigned long long sqnum, int free, int dirty,
int jhead)
{ {
struct rb_node **p = &c->replay_tree.rb_node, *parent = NULL; struct rb_node **p = &c->replay_tree.rb_node, *parent = NULL;
struct replay_entry *r; struct replay_entry *r;
...@@ -688,6 +693,7 @@ static int insert_ref_node(struct ubifs_info *c, int lnum, int offs, ...@@ -688,6 +693,7 @@ static int insert_ref_node(struct ubifs_info *c, int lnum, int offs,
r->flags = REPLAY_REF; r->flags = REPLAY_REF;
r->free = free; r->free = free;
r->dirty = dirty; r->dirty = dirty;
r->jhead = jhead;
rb_link_node(&r->rb, parent, p); rb_link_node(&r->rb, parent, p);
rb_insert_color(&r->rb, &c->replay_tree); rb_insert_color(&r->rb, &c->replay_tree);
...@@ -712,7 +718,7 @@ static int replay_buds(struct ubifs_info *c) ...@@ -712,7 +718,7 @@ static int replay_buds(struct ubifs_info *c)
if (err) if (err)
return err; return err;
err = insert_ref_node(c, b->bud->lnum, b->bud->start, b->sqnum, err = insert_ref_node(c, b->bud->lnum, b->bud->start, b->sqnum,
free, dirty); free, dirty, b->bud->jhead);
if (err) if (err)
return err; return err;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment