提交 0dcd18e4 编写于 作者: A Artem Bityutskiy

UBIFS: check ubifs_scan error codes better

The 'ubifs_scan()' function returns -EUCLEAN if something is corrupted
and recovery is needed, otherwise it returns other error codes. However,
in few places UBIFS does not check the error codes and runs recovery.
This patch changes this behavior and makes UBIFS start recovery only
on -EUCLEAN errors.
Signed-off-by: NArtem Bityutskiy <Artem.Bityutskiy@nokia.com>
Reviewed-by: NAdrian Hunter <Adrian.Hunter@nokia.com>
上级 348709ba
...@@ -29,7 +29,8 @@ ...@@ -29,7 +29,8 @@
* @c: UBIFS file-system description object * @c: UBIFS file-system description object
* *
* This function scans the master node LEBs and search for the latest master * This function scans the master node LEBs and search for the latest master
* node. Returns zero in case of success and a negative error code in case of * node. Returns zero in case of success, %-EUCLEAN if there master area is
* corrupted and requires recovery, and a negative error code in case of
* failure. * failure.
*/ */
static int scan_for_master(struct ubifs_info *c) static int scan_for_master(struct ubifs_info *c)
...@@ -48,7 +49,7 @@ static int scan_for_master(struct ubifs_info *c) ...@@ -48,7 +49,7 @@ static int scan_for_master(struct ubifs_info *c)
snod = list_entry(sleb->nodes.prev, struct ubifs_scan_node, snod = list_entry(sleb->nodes.prev, struct ubifs_scan_node,
list); list);
if (snod->type != UBIFS_MST_NODE) if (snod->type != UBIFS_MST_NODE)
goto out; goto out_dump;
memcpy(c->mst_node, snod->node, snod->len); memcpy(c->mst_node, snod->node, snod->len);
offs = snod->offs; offs = snod->offs;
} }
...@@ -65,7 +66,7 @@ static int scan_for_master(struct ubifs_info *c) ...@@ -65,7 +66,7 @@ static int scan_for_master(struct ubifs_info *c)
goto out; goto out;
snod = list_entry(sleb->nodes.prev, struct ubifs_scan_node, list); snod = list_entry(sleb->nodes.prev, struct ubifs_scan_node, list);
if (snod->type != UBIFS_MST_NODE) if (snod->type != UBIFS_MST_NODE)
goto out; goto out_dump;
if (snod->offs != offs) if (snod->offs != offs)
goto out; goto out;
if (memcmp((void *)c->mst_node + UBIFS_CH_SZ, if (memcmp((void *)c->mst_node + UBIFS_CH_SZ,
...@@ -78,6 +79,12 @@ static int scan_for_master(struct ubifs_info *c) ...@@ -78,6 +79,12 @@ static int scan_for_master(struct ubifs_info *c)
out: out:
ubifs_scan_destroy(sleb); ubifs_scan_destroy(sleb);
return -EUCLEAN;
out_dump:
ubifs_err("unexpected node type %d master LEB %d:%d",
snod->type, lnum, snod->offs);
ubifs_scan_destroy(sleb);
return -EINVAL; return -EINVAL;
} }
...@@ -256,6 +263,7 @@ int ubifs_read_master(struct ubifs_info *c) ...@@ -256,6 +263,7 @@ int ubifs_read_master(struct ubifs_info *c)
err = scan_for_master(c); err = scan_for_master(c);
if (err) { if (err) {
if (err == -EUCLEAN)
err = ubifs_recover_master_node(c); err = ubifs_recover_master_node(c);
if (err) if (err)
/* /*
......
...@@ -672,6 +672,7 @@ static int kill_orphans(struct ubifs_info *c) ...@@ -672,6 +672,7 @@ static int kill_orphans(struct ubifs_info *c)
dbg_rcvry("LEB %d", lnum); dbg_rcvry("LEB %d", lnum);
sleb = ubifs_scan(c, lnum, 0, c->sbuf, 1); sleb = ubifs_scan(c, lnum, 0, c->sbuf, 1);
if (IS_ERR(sleb)) { if (IS_ERR(sleb)) {
if (PTR_ERR(sleb) == -EUCLEAN)
sleb = ubifs_recover_leb(c, lnum, 0, c->sbuf, 0); sleb = ubifs_recover_leb(c, lnum, 0, c->sbuf, 0);
if (IS_ERR(sleb)) { if (IS_ERR(sleb)) {
err = PTR_ERR(sleb); err = PTR_ERR(sleb);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册