forked from Minki/linux
50614bcf29
This adds a field to record the latest checkpoint number in the nilfs_segment_summary structure. This will help to recover the latest checkpoint number from logs on disk. This field is intended for crucial cases in which super blocks have lost pointer to the latest log. Even though this will change the disk format, both backward and forward compatibility is preserved by a size field prepared in the segment summary header. Signed-off-by: Ryusuke Konishi <konishi.ryusuke@lab.ntt.co.jp>
922 lines
23 KiB
C
922 lines
23 KiB
C
/*
|
|
* recovery.c - NILFS recovery logic
|
|
*
|
|
* Copyright (C) 2005-2008 Nippon Telegraph and Telephone Corporation.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Written by Ryusuke Konishi <ryusuke@osrg.net>
|
|
*/
|
|
|
|
#include <linux/buffer_head.h>
|
|
#include <linux/blkdev.h>
|
|
#include <linux/swap.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/crc32.h>
|
|
#include "nilfs.h"
|
|
#include "segment.h"
|
|
#include "sufile.h"
|
|
#include "page.h"
|
|
#include "segbuf.h"
|
|
|
|
/*
|
|
* Segment check result
|
|
*/
|
|
enum {
|
|
NILFS_SEG_VALID,
|
|
NILFS_SEG_NO_SUPER_ROOT,
|
|
NILFS_SEG_FAIL_IO,
|
|
NILFS_SEG_FAIL_MAGIC,
|
|
NILFS_SEG_FAIL_SEQ,
|
|
NILFS_SEG_FAIL_CHECKSUM_SUPER_ROOT,
|
|
NILFS_SEG_FAIL_CHECKSUM_FULL,
|
|
NILFS_SEG_FAIL_CONSISTENCY,
|
|
};
|
|
|
|
/* work structure for recovery */
|
|
struct nilfs_recovery_block {
|
|
ino_t ino; /* Inode number of the file that this block
|
|
belongs to */
|
|
sector_t blocknr; /* block number */
|
|
__u64 vblocknr; /* virtual block number */
|
|
unsigned long blkoff; /* File offset of the data block (per block) */
|
|
struct list_head list;
|
|
};
|
|
|
|
|
|
static int nilfs_warn_segment_error(int err)
|
|
{
|
|
switch (err) {
|
|
case NILFS_SEG_FAIL_IO:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: I/O error on loading last segment\n");
|
|
return -EIO;
|
|
case NILFS_SEG_FAIL_MAGIC:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: Segment magic number invalid\n");
|
|
break;
|
|
case NILFS_SEG_FAIL_SEQ:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: Sequence number mismatch\n");
|
|
break;
|
|
case NILFS_SEG_FAIL_CHECKSUM_SUPER_ROOT:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: Checksum error in super root\n");
|
|
break;
|
|
case NILFS_SEG_FAIL_CHECKSUM_FULL:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: Checksum error in segment payload\n");
|
|
break;
|
|
case NILFS_SEG_FAIL_CONSISTENCY:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: Inconsistent segment\n");
|
|
break;
|
|
case NILFS_SEG_NO_SUPER_ROOT:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: No super root in the last segment\n");
|
|
break;
|
|
}
|
|
return -EINVAL;
|
|
}
|
|
|
|
static void store_segsum_info(struct nilfs_segsum_info *ssi,
|
|
struct nilfs_segment_summary *sum,
|
|
unsigned int blocksize)
|
|
{
|
|
ssi->flags = le16_to_cpu(sum->ss_flags);
|
|
ssi->seg_seq = le64_to_cpu(sum->ss_seq);
|
|
ssi->ctime = le64_to_cpu(sum->ss_create);
|
|
ssi->next = le64_to_cpu(sum->ss_next);
|
|
ssi->nblocks = le32_to_cpu(sum->ss_nblocks);
|
|
ssi->nfinfo = le32_to_cpu(sum->ss_nfinfo);
|
|
ssi->sumbytes = le32_to_cpu(sum->ss_sumbytes);
|
|
|
|
ssi->nsumblk = DIV_ROUND_UP(ssi->sumbytes, blocksize);
|
|
ssi->nfileblk = ssi->nblocks - ssi->nsumblk - !!NILFS_SEG_HAS_SR(ssi);
|
|
|
|
/* need to verify ->ss_bytes field if read ->ss_cno */
|
|
}
|
|
|
|
/**
|
|
* calc_crc_cont - check CRC of blocks continuously
|
|
* @sbi: nilfs_sb_info
|
|
* @bhs: buffer head of start block
|
|
* @sum: place to store result
|
|
* @offset: offset bytes in the first block
|
|
* @check_bytes: number of bytes to be checked
|
|
* @start: DBN of start block
|
|
* @nblock: number of blocks to be checked
|
|
*/
|
|
static int calc_crc_cont(struct nilfs_sb_info *sbi, struct buffer_head *bhs,
|
|
u32 *sum, unsigned long offset, u64 check_bytes,
|
|
sector_t start, unsigned long nblock)
|
|
{
|
|
unsigned long blocksize = sbi->s_super->s_blocksize;
|
|
unsigned long size;
|
|
u32 crc;
|
|
|
|
BUG_ON(offset >= blocksize);
|
|
check_bytes -= offset;
|
|
size = min_t(u64, check_bytes, blocksize - offset);
|
|
crc = crc32_le(sbi->s_nilfs->ns_crc_seed,
|
|
(unsigned char *)bhs->b_data + offset, size);
|
|
if (--nblock > 0) {
|
|
do {
|
|
struct buffer_head *bh
|
|
= sb_bread(sbi->s_super, ++start);
|
|
if (!bh)
|
|
return -EIO;
|
|
check_bytes -= size;
|
|
size = min_t(u64, check_bytes, blocksize);
|
|
crc = crc32_le(crc, bh->b_data, size);
|
|
brelse(bh);
|
|
} while (--nblock > 0);
|
|
}
|
|
*sum = crc;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* nilfs_read_super_root_block - read super root block
|
|
* @sb: super_block
|
|
* @sr_block: disk block number of the super root block
|
|
* @pbh: address of a buffer_head pointer to return super root buffer
|
|
* @check: CRC check flag
|
|
*/
|
|
int nilfs_read_super_root_block(struct super_block *sb, sector_t sr_block,
|
|
struct buffer_head **pbh, int check)
|
|
{
|
|
struct buffer_head *bh_sr;
|
|
struct nilfs_super_root *sr;
|
|
u32 crc;
|
|
int ret;
|
|
|
|
*pbh = NULL;
|
|
bh_sr = sb_bread(sb, sr_block);
|
|
if (unlikely(!bh_sr)) {
|
|
ret = NILFS_SEG_FAIL_IO;
|
|
goto failed;
|
|
}
|
|
|
|
sr = (struct nilfs_super_root *)bh_sr->b_data;
|
|
if (check) {
|
|
unsigned bytes = le16_to_cpu(sr->sr_bytes);
|
|
|
|
if (bytes == 0 || bytes > sb->s_blocksize) {
|
|
ret = NILFS_SEG_FAIL_CHECKSUM_SUPER_ROOT;
|
|
goto failed_bh;
|
|
}
|
|
if (calc_crc_cont(NILFS_SB(sb), bh_sr, &crc,
|
|
sizeof(sr->sr_sum), bytes, sr_block, 1)) {
|
|
ret = NILFS_SEG_FAIL_IO;
|
|
goto failed_bh;
|
|
}
|
|
if (crc != le32_to_cpu(sr->sr_sum)) {
|
|
ret = NILFS_SEG_FAIL_CHECKSUM_SUPER_ROOT;
|
|
goto failed_bh;
|
|
}
|
|
}
|
|
*pbh = bh_sr;
|
|
return 0;
|
|
|
|
failed_bh:
|
|
brelse(bh_sr);
|
|
|
|
failed:
|
|
return nilfs_warn_segment_error(ret);
|
|
}
|
|
|
|
/**
|
|
* load_segment_summary - read segment summary of the specified partial segment
|
|
* @sbi: nilfs_sb_info
|
|
* @pseg_start: start disk block number of partial segment
|
|
* @seg_seq: sequence number requested
|
|
* @ssi: pointer to nilfs_segsum_info struct to store information
|
|
*/
|
|
static int
|
|
load_segment_summary(struct nilfs_sb_info *sbi, sector_t pseg_start,
|
|
u64 seg_seq, struct nilfs_segsum_info *ssi)
|
|
{
|
|
struct buffer_head *bh_sum;
|
|
struct nilfs_segment_summary *sum;
|
|
unsigned long nblock;
|
|
u32 crc;
|
|
int ret = NILFS_SEG_FAIL_IO;
|
|
|
|
bh_sum = sb_bread(sbi->s_super, pseg_start);
|
|
if (!bh_sum)
|
|
goto out;
|
|
|
|
sum = (struct nilfs_segment_summary *)bh_sum->b_data;
|
|
|
|
/* Check consistency of segment summary */
|
|
if (le32_to_cpu(sum->ss_magic) != NILFS_SEGSUM_MAGIC) {
|
|
ret = NILFS_SEG_FAIL_MAGIC;
|
|
goto failed;
|
|
}
|
|
store_segsum_info(ssi, sum, sbi->s_super->s_blocksize);
|
|
if (seg_seq != ssi->seg_seq) {
|
|
ret = NILFS_SEG_FAIL_SEQ;
|
|
goto failed;
|
|
}
|
|
|
|
nblock = ssi->nblocks;
|
|
if (unlikely(nblock == 0 ||
|
|
nblock > sbi->s_nilfs->ns_blocks_per_segment)) {
|
|
/* This limits the number of blocks read in the CRC check */
|
|
ret = NILFS_SEG_FAIL_CONSISTENCY;
|
|
goto failed;
|
|
}
|
|
if (calc_crc_cont(sbi, bh_sum, &crc, sizeof(sum->ss_datasum),
|
|
((u64)nblock << sbi->s_super->s_blocksize_bits),
|
|
pseg_start, nblock)) {
|
|
ret = NILFS_SEG_FAIL_IO;
|
|
goto failed;
|
|
}
|
|
if (crc == le32_to_cpu(sum->ss_datasum))
|
|
ret = 0;
|
|
else
|
|
ret = NILFS_SEG_FAIL_CHECKSUM_FULL;
|
|
failed:
|
|
brelse(bh_sum);
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static void *segsum_get(struct super_block *sb, struct buffer_head **pbh,
|
|
unsigned int *offset, unsigned int bytes)
|
|
{
|
|
void *ptr;
|
|
sector_t blocknr;
|
|
|
|
BUG_ON((*pbh)->b_size < *offset);
|
|
if (bytes > (*pbh)->b_size - *offset) {
|
|
blocknr = (*pbh)->b_blocknr;
|
|
brelse(*pbh);
|
|
*pbh = sb_bread(sb, blocknr + 1);
|
|
if (unlikely(!*pbh))
|
|
return NULL;
|
|
*offset = 0;
|
|
}
|
|
ptr = (*pbh)->b_data + *offset;
|
|
*offset += bytes;
|
|
return ptr;
|
|
}
|
|
|
|
static void segsum_skip(struct super_block *sb, struct buffer_head **pbh,
|
|
unsigned int *offset, unsigned int bytes,
|
|
unsigned long count)
|
|
{
|
|
unsigned int rest_item_in_current_block
|
|
= ((*pbh)->b_size - *offset) / bytes;
|
|
|
|
if (count <= rest_item_in_current_block) {
|
|
*offset += bytes * count;
|
|
} else {
|
|
sector_t blocknr = (*pbh)->b_blocknr;
|
|
unsigned int nitem_per_block = (*pbh)->b_size / bytes;
|
|
unsigned int bcnt;
|
|
|
|
count -= rest_item_in_current_block;
|
|
bcnt = DIV_ROUND_UP(count, nitem_per_block);
|
|
*offset = bytes * (count - (bcnt - 1) * nitem_per_block);
|
|
|
|
brelse(*pbh);
|
|
*pbh = sb_bread(sb, blocknr + bcnt);
|
|
}
|
|
}
|
|
|
|
static int
|
|
collect_blocks_from_segsum(struct nilfs_sb_info *sbi, sector_t sum_blocknr,
|
|
struct nilfs_segsum_info *ssi,
|
|
struct list_head *head)
|
|
{
|
|
struct buffer_head *bh;
|
|
unsigned int offset;
|
|
unsigned long nfinfo = ssi->nfinfo;
|
|
sector_t blocknr = sum_blocknr + ssi->nsumblk;
|
|
ino_t ino;
|
|
int err = -EIO;
|
|
|
|
if (!nfinfo)
|
|
return 0;
|
|
|
|
bh = sb_bread(sbi->s_super, sum_blocknr);
|
|
if (unlikely(!bh))
|
|
goto out;
|
|
|
|
offset = le16_to_cpu(
|
|
((struct nilfs_segment_summary *)bh->b_data)->ss_bytes);
|
|
for (;;) {
|
|
unsigned long nblocks, ndatablk, nnodeblk;
|
|
struct nilfs_finfo *finfo;
|
|
|
|
finfo = segsum_get(sbi->s_super, &bh, &offset, sizeof(*finfo));
|
|
if (unlikely(!finfo))
|
|
goto out;
|
|
|
|
ino = le64_to_cpu(finfo->fi_ino);
|
|
nblocks = le32_to_cpu(finfo->fi_nblocks);
|
|
ndatablk = le32_to_cpu(finfo->fi_ndatablk);
|
|
nnodeblk = nblocks - ndatablk;
|
|
|
|
while (ndatablk-- > 0) {
|
|
struct nilfs_recovery_block *rb;
|
|
struct nilfs_binfo_v *binfo;
|
|
|
|
binfo = segsum_get(sbi->s_super, &bh, &offset,
|
|
sizeof(*binfo));
|
|
if (unlikely(!binfo))
|
|
goto out;
|
|
|
|
rb = kmalloc(sizeof(*rb), GFP_NOFS);
|
|
if (unlikely(!rb)) {
|
|
err = -ENOMEM;
|
|
goto out;
|
|
}
|
|
rb->ino = ino;
|
|
rb->blocknr = blocknr++;
|
|
rb->vblocknr = le64_to_cpu(binfo->bi_vblocknr);
|
|
rb->blkoff = le64_to_cpu(binfo->bi_blkoff);
|
|
/* INIT_LIST_HEAD(&rb->list); */
|
|
list_add_tail(&rb->list, head);
|
|
}
|
|
if (--nfinfo == 0)
|
|
break;
|
|
blocknr += nnodeblk; /* always 0 for the data sync segments */
|
|
segsum_skip(sbi->s_super, &bh, &offset, sizeof(__le64),
|
|
nnodeblk);
|
|
if (unlikely(!bh))
|
|
goto out;
|
|
}
|
|
err = 0;
|
|
out:
|
|
brelse(bh); /* brelse(NULL) is just ignored */
|
|
return err;
|
|
}
|
|
|
|
static void dispose_recovery_list(struct list_head *head)
|
|
{
|
|
while (!list_empty(head)) {
|
|
struct nilfs_recovery_block *rb
|
|
= list_entry(head->next,
|
|
struct nilfs_recovery_block, list);
|
|
list_del(&rb->list);
|
|
kfree(rb);
|
|
}
|
|
}
|
|
|
|
struct nilfs_segment_entry {
|
|
struct list_head list;
|
|
__u64 segnum;
|
|
};
|
|
|
|
static int nilfs_segment_list_add(struct list_head *head, __u64 segnum)
|
|
{
|
|
struct nilfs_segment_entry *ent = kmalloc(sizeof(*ent), GFP_NOFS);
|
|
|
|
if (unlikely(!ent))
|
|
return -ENOMEM;
|
|
|
|
ent->segnum = segnum;
|
|
INIT_LIST_HEAD(&ent->list);
|
|
list_add_tail(&ent->list, head);
|
|
return 0;
|
|
}
|
|
|
|
void nilfs_dispose_segment_list(struct list_head *head)
|
|
{
|
|
while (!list_empty(head)) {
|
|
struct nilfs_segment_entry *ent
|
|
= list_entry(head->next,
|
|
struct nilfs_segment_entry, list);
|
|
list_del(&ent->list);
|
|
kfree(ent);
|
|
}
|
|
}
|
|
|
|
static int nilfs_prepare_segment_for_recovery(struct the_nilfs *nilfs,
|
|
struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_info *ri)
|
|
{
|
|
struct list_head *head = &ri->ri_used_segments;
|
|
struct nilfs_segment_entry *ent, *n;
|
|
struct inode *sufile = nilfs->ns_sufile;
|
|
__u64 segnum[4];
|
|
int err;
|
|
int i;
|
|
|
|
segnum[0] = nilfs->ns_segnum;
|
|
segnum[1] = nilfs->ns_nextnum;
|
|
segnum[2] = ri->ri_segnum;
|
|
segnum[3] = ri->ri_nextnum;
|
|
|
|
nilfs_attach_writer(nilfs, sbi);
|
|
/*
|
|
* Releasing the next segment of the latest super root.
|
|
* The next segment is invalidated by this recovery.
|
|
*/
|
|
err = nilfs_sufile_free(sufile, segnum[1]);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
|
|
for (i = 1; i < 4; i++) {
|
|
err = nilfs_segment_list_add(head, segnum[i]);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
}
|
|
|
|
/*
|
|
* Collecting segments written after the latest super root.
|
|
* These are marked dirty to avoid being reallocated in the next write.
|
|
*/
|
|
list_for_each_entry_safe(ent, n, head, list) {
|
|
if (ent->segnum != segnum[0]) {
|
|
err = nilfs_sufile_scrap(sufile, ent->segnum);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
}
|
|
list_del(&ent->list);
|
|
kfree(ent);
|
|
}
|
|
|
|
/* Allocate new segments for recovery */
|
|
err = nilfs_sufile_alloc(sufile, &segnum[0]);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
|
|
nilfs->ns_pseg_offset = 0;
|
|
nilfs->ns_seg_seq = ri->ri_seq + 2;
|
|
nilfs->ns_nextnum = nilfs->ns_segnum = segnum[0];
|
|
|
|
failed:
|
|
/* No need to recover sufile because it will be destroyed on error */
|
|
nilfs_detach_writer(nilfs, sbi);
|
|
return err;
|
|
}
|
|
|
|
static int nilfs_recovery_copy_block(struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_block *rb,
|
|
struct page *page)
|
|
{
|
|
struct buffer_head *bh_org;
|
|
void *kaddr;
|
|
|
|
bh_org = sb_bread(sbi->s_super, rb->blocknr);
|
|
if (unlikely(!bh_org))
|
|
return -EIO;
|
|
|
|
kaddr = kmap_atomic(page, KM_USER0);
|
|
memcpy(kaddr + bh_offset(bh_org), bh_org->b_data, bh_org->b_size);
|
|
kunmap_atomic(kaddr, KM_USER0);
|
|
brelse(bh_org);
|
|
return 0;
|
|
}
|
|
|
|
static int recover_dsync_blocks(struct nilfs_sb_info *sbi,
|
|
struct list_head *head,
|
|
unsigned long *nr_salvaged_blocks)
|
|
{
|
|
struct inode *inode;
|
|
struct nilfs_recovery_block *rb, *n;
|
|
unsigned blocksize = sbi->s_super->s_blocksize;
|
|
struct page *page;
|
|
loff_t pos;
|
|
int err = 0, err2 = 0;
|
|
|
|
list_for_each_entry_safe(rb, n, head, list) {
|
|
inode = nilfs_iget(sbi->s_super, rb->ino);
|
|
if (IS_ERR(inode)) {
|
|
err = PTR_ERR(inode);
|
|
inode = NULL;
|
|
goto failed_inode;
|
|
}
|
|
|
|
pos = rb->blkoff << inode->i_blkbits;
|
|
page = NULL;
|
|
err = block_write_begin(NULL, inode->i_mapping, pos, blocksize,
|
|
0, &page, NULL, nilfs_get_block);
|
|
if (unlikely(err))
|
|
goto failed_inode;
|
|
|
|
err = nilfs_recovery_copy_block(sbi, rb, page);
|
|
if (unlikely(err))
|
|
goto failed_page;
|
|
|
|
err = nilfs_set_file_dirty(sbi, inode, 1);
|
|
if (unlikely(err))
|
|
goto failed_page;
|
|
|
|
block_write_end(NULL, inode->i_mapping, pos, blocksize,
|
|
blocksize, page, NULL);
|
|
|
|
unlock_page(page);
|
|
page_cache_release(page);
|
|
|
|
(*nr_salvaged_blocks)++;
|
|
goto next;
|
|
|
|
failed_page:
|
|
unlock_page(page);
|
|
page_cache_release(page);
|
|
|
|
failed_inode:
|
|
printk(KERN_WARNING
|
|
"NILFS warning: error recovering data block "
|
|
"(err=%d, ino=%lu, block-offset=%llu)\n",
|
|
err, (unsigned long)rb->ino,
|
|
(unsigned long long)rb->blkoff);
|
|
if (!err2)
|
|
err2 = err;
|
|
next:
|
|
iput(inode); /* iput(NULL) is just ignored */
|
|
list_del_init(&rb->list);
|
|
kfree(rb);
|
|
}
|
|
return err2;
|
|
}
|
|
|
|
/**
|
|
* nilfs_do_roll_forward - salvage logical segments newer than the latest
|
|
* checkpoint
|
|
* @sbi: nilfs_sb_info
|
|
* @nilfs: the_nilfs
|
|
* @ri: pointer to a nilfs_recovery_info
|
|
*/
|
|
static int nilfs_do_roll_forward(struct the_nilfs *nilfs,
|
|
struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_info *ri)
|
|
{
|
|
struct nilfs_segsum_info ssi;
|
|
sector_t pseg_start;
|
|
sector_t seg_start, seg_end; /* Starting/ending DBN of full segment */
|
|
unsigned long nsalvaged_blocks = 0;
|
|
u64 seg_seq;
|
|
__u64 segnum, nextnum = 0;
|
|
int empty_seg = 0;
|
|
int err = 0, ret;
|
|
LIST_HEAD(dsync_blocks); /* list of data blocks to be recovered */
|
|
enum {
|
|
RF_INIT_ST,
|
|
RF_DSYNC_ST, /* scanning data-sync segments */
|
|
};
|
|
int state = RF_INIT_ST;
|
|
|
|
nilfs_attach_writer(nilfs, sbi);
|
|
pseg_start = ri->ri_lsegs_start;
|
|
seg_seq = ri->ri_lsegs_start_seq;
|
|
segnum = nilfs_get_segnum_of_block(nilfs, pseg_start);
|
|
nilfs_get_segment_range(nilfs, segnum, &seg_start, &seg_end);
|
|
|
|
while (segnum != ri->ri_segnum || pseg_start <= ri->ri_pseg_start) {
|
|
|
|
ret = load_segment_summary(sbi, pseg_start, seg_seq, &ssi);
|
|
if (ret) {
|
|
if (ret == NILFS_SEG_FAIL_IO) {
|
|
err = -EIO;
|
|
goto failed;
|
|
}
|
|
goto strayed;
|
|
}
|
|
if (unlikely(NILFS_SEG_HAS_SR(&ssi)))
|
|
goto confused;
|
|
|
|
/* Found a valid partial segment; do recovery actions */
|
|
nextnum = nilfs_get_segnum_of_block(nilfs, ssi.next);
|
|
empty_seg = 0;
|
|
nilfs->ns_ctime = ssi.ctime;
|
|
if (!(ssi.flags & NILFS_SS_GC))
|
|
nilfs->ns_nongc_ctime = ssi.ctime;
|
|
|
|
switch (state) {
|
|
case RF_INIT_ST:
|
|
if (!NILFS_SEG_LOGBGN(&ssi) || !NILFS_SEG_DSYNC(&ssi))
|
|
goto try_next_pseg;
|
|
state = RF_DSYNC_ST;
|
|
/* Fall through */
|
|
case RF_DSYNC_ST:
|
|
if (!NILFS_SEG_DSYNC(&ssi))
|
|
goto confused;
|
|
|
|
err = collect_blocks_from_segsum(
|
|
sbi, pseg_start, &ssi, &dsync_blocks);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
if (NILFS_SEG_LOGEND(&ssi)) {
|
|
err = recover_dsync_blocks(
|
|
sbi, &dsync_blocks, &nsalvaged_blocks);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
state = RF_INIT_ST;
|
|
}
|
|
break; /* Fall through to try_next_pseg */
|
|
}
|
|
|
|
try_next_pseg:
|
|
if (pseg_start == ri->ri_lsegs_end)
|
|
break;
|
|
pseg_start += ssi.nblocks;
|
|
if (pseg_start < seg_end)
|
|
continue;
|
|
goto feed_segment;
|
|
|
|
strayed:
|
|
if (pseg_start == ri->ri_lsegs_end)
|
|
break;
|
|
|
|
feed_segment:
|
|
/* Looking to the next full segment */
|
|
if (empty_seg++)
|
|
break;
|
|
seg_seq++;
|
|
segnum = nextnum;
|
|
nilfs_get_segment_range(nilfs, segnum, &seg_start, &seg_end);
|
|
pseg_start = seg_start;
|
|
}
|
|
|
|
if (nsalvaged_blocks) {
|
|
printk(KERN_INFO "NILFS (device %s): salvaged %lu blocks\n",
|
|
sbi->s_super->s_id, nsalvaged_blocks);
|
|
ri->ri_need_recovery = NILFS_RECOVERY_ROLLFORWARD_DONE;
|
|
}
|
|
out:
|
|
dispose_recovery_list(&dsync_blocks);
|
|
nilfs_detach_writer(sbi->s_nilfs, sbi);
|
|
return err;
|
|
|
|
confused:
|
|
err = -EINVAL;
|
|
failed:
|
|
printk(KERN_ERR
|
|
"NILFS (device %s): Error roll-forwarding "
|
|
"(err=%d, pseg block=%llu). ",
|
|
sbi->s_super->s_id, err, (unsigned long long)pseg_start);
|
|
goto out;
|
|
}
|
|
|
|
static void nilfs_finish_roll_forward(struct the_nilfs *nilfs,
|
|
struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_info *ri)
|
|
{
|
|
struct buffer_head *bh;
|
|
int err;
|
|
|
|
if (nilfs_get_segnum_of_block(nilfs, ri->ri_lsegs_start) !=
|
|
nilfs_get_segnum_of_block(nilfs, ri->ri_super_root))
|
|
return;
|
|
|
|
bh = sb_getblk(sbi->s_super, ri->ri_lsegs_start);
|
|
BUG_ON(!bh);
|
|
memset(bh->b_data, 0, bh->b_size);
|
|
set_buffer_dirty(bh);
|
|
err = sync_dirty_buffer(bh);
|
|
if (unlikely(err))
|
|
printk(KERN_WARNING
|
|
"NILFS warning: buffer sync write failed during "
|
|
"post-cleaning of recovery.\n");
|
|
brelse(bh);
|
|
}
|
|
|
|
/**
|
|
* nilfs_recover_logical_segments - salvage logical segments written after
|
|
* the latest super root
|
|
* @nilfs: the_nilfs
|
|
* @sbi: nilfs_sb_info
|
|
* @ri: pointer to a nilfs_recovery_info struct to store search results.
|
|
*
|
|
* Return Value: On success, 0 is returned. On error, one of the following
|
|
* negative error code is returned.
|
|
*
|
|
* %-EINVAL - Inconsistent filesystem state.
|
|
*
|
|
* %-EIO - I/O error
|
|
*
|
|
* %-ENOSPC - No space left on device (only in a panic state).
|
|
*
|
|
* %-ERESTARTSYS - Interrupted.
|
|
*
|
|
* %-ENOMEM - Insufficient memory available.
|
|
*/
|
|
int nilfs_recover_logical_segments(struct the_nilfs *nilfs,
|
|
struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_info *ri)
|
|
{
|
|
int err;
|
|
|
|
if (ri->ri_lsegs_start == 0 || ri->ri_lsegs_end == 0)
|
|
return 0;
|
|
|
|
err = nilfs_attach_checkpoint(sbi, ri->ri_cno);
|
|
if (unlikely(err)) {
|
|
printk(KERN_ERR
|
|
"NILFS: error loading the latest checkpoint.\n");
|
|
return err;
|
|
}
|
|
|
|
err = nilfs_do_roll_forward(nilfs, sbi, ri);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
|
|
if (ri->ri_need_recovery == NILFS_RECOVERY_ROLLFORWARD_DONE) {
|
|
err = nilfs_prepare_segment_for_recovery(nilfs, sbi, ri);
|
|
if (unlikely(err)) {
|
|
printk(KERN_ERR "NILFS: Error preparing segments for "
|
|
"recovery.\n");
|
|
goto failed;
|
|
}
|
|
|
|
err = nilfs_attach_segment_constructor(sbi);
|
|
if (unlikely(err))
|
|
goto failed;
|
|
|
|
set_nilfs_discontinued(nilfs);
|
|
err = nilfs_construct_segment(sbi->s_super);
|
|
nilfs_detach_segment_constructor(sbi);
|
|
|
|
if (unlikely(err)) {
|
|
printk(KERN_ERR "NILFS: Oops! recovery failed. "
|
|
"(err=%d)\n", err);
|
|
goto failed;
|
|
}
|
|
|
|
nilfs_finish_roll_forward(nilfs, sbi, ri);
|
|
}
|
|
|
|
failed:
|
|
nilfs_detach_checkpoint(sbi);
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* nilfs_search_super_root - search the latest valid super root
|
|
* @nilfs: the_nilfs
|
|
* @sbi: nilfs_sb_info
|
|
* @ri: pointer to a nilfs_recovery_info struct to store search results.
|
|
*
|
|
* nilfs_search_super_root() looks for the latest super-root from a partial
|
|
* segment pointed by the superblock. It sets up struct the_nilfs through
|
|
* this search. It fills nilfs_recovery_info (ri) required for recovery.
|
|
*
|
|
* Return Value: On success, 0 is returned. On error, one of the following
|
|
* negative error code is returned.
|
|
*
|
|
* %-EINVAL - No valid segment found
|
|
*
|
|
* %-EIO - I/O error
|
|
*/
|
|
int nilfs_search_super_root(struct the_nilfs *nilfs, struct nilfs_sb_info *sbi,
|
|
struct nilfs_recovery_info *ri)
|
|
{
|
|
struct nilfs_segsum_info ssi;
|
|
sector_t pseg_start, pseg_end, sr_pseg_start = 0;
|
|
sector_t seg_start, seg_end; /* range of full segment (block number) */
|
|
sector_t b, end;
|
|
u64 seg_seq;
|
|
__u64 segnum, nextnum = 0;
|
|
__u64 cno;
|
|
LIST_HEAD(segments);
|
|
int empty_seg = 0, scan_newer = 0;
|
|
int ret;
|
|
|
|
pseg_start = nilfs->ns_last_pseg;
|
|
seg_seq = nilfs->ns_last_seq;
|
|
cno = nilfs->ns_last_cno;
|
|
segnum = nilfs_get_segnum_of_block(nilfs, pseg_start);
|
|
|
|
/* Calculate range of segment */
|
|
nilfs_get_segment_range(nilfs, segnum, &seg_start, &seg_end);
|
|
|
|
/* Read ahead segment */
|
|
b = seg_start;
|
|
while (b <= seg_end)
|
|
sb_breadahead(sbi->s_super, b++);
|
|
|
|
for (;;) {
|
|
/* Load segment summary */
|
|
ret = load_segment_summary(sbi, pseg_start, seg_seq, &ssi);
|
|
if (ret) {
|
|
if (ret == NILFS_SEG_FAIL_IO)
|
|
goto failed;
|
|
goto strayed;
|
|
}
|
|
pseg_end = pseg_start + ssi.nblocks - 1;
|
|
if (unlikely(pseg_end > seg_end)) {
|
|
ret = NILFS_SEG_FAIL_CONSISTENCY;
|
|
goto strayed;
|
|
}
|
|
|
|
/* A valid partial segment */
|
|
ri->ri_pseg_start = pseg_start;
|
|
ri->ri_seq = seg_seq;
|
|
ri->ri_segnum = segnum;
|
|
nextnum = nilfs_get_segnum_of_block(nilfs, ssi.next);
|
|
ri->ri_nextnum = nextnum;
|
|
empty_seg = 0;
|
|
|
|
if (!NILFS_SEG_HAS_SR(&ssi) && !scan_newer) {
|
|
/* This will never happen because a superblock
|
|
(last_segment) always points to a pseg
|
|
having a super root. */
|
|
ret = NILFS_SEG_FAIL_CONSISTENCY;
|
|
goto failed;
|
|
}
|
|
|
|
if (pseg_start == seg_start) {
|
|
nilfs_get_segment_range(nilfs, nextnum, &b, &end);
|
|
while (b <= end)
|
|
sb_breadahead(sbi->s_super, b++);
|
|
}
|
|
if (!NILFS_SEG_HAS_SR(&ssi)) {
|
|
if (!ri->ri_lsegs_start && NILFS_SEG_LOGBGN(&ssi)) {
|
|
ri->ri_lsegs_start = pseg_start;
|
|
ri->ri_lsegs_start_seq = seg_seq;
|
|
}
|
|
if (NILFS_SEG_LOGEND(&ssi))
|
|
ri->ri_lsegs_end = pseg_start;
|
|
goto try_next_pseg;
|
|
}
|
|
|
|
/* A valid super root was found. */
|
|
ri->ri_cno = cno++;
|
|
ri->ri_super_root = pseg_end;
|
|
ri->ri_lsegs_start = ri->ri_lsegs_end = 0;
|
|
|
|
nilfs_dispose_segment_list(&segments);
|
|
nilfs->ns_pseg_offset = (sr_pseg_start = pseg_start)
|
|
+ ssi.nblocks - seg_start;
|
|
nilfs->ns_seg_seq = seg_seq;
|
|
nilfs->ns_segnum = segnum;
|
|
nilfs->ns_cno = cno; /* nilfs->ns_cno = ri->ri_cno + 1 */
|
|
nilfs->ns_ctime = ssi.ctime;
|
|
nilfs->ns_nextnum = nextnum;
|
|
|
|
if (scan_newer)
|
|
ri->ri_need_recovery = NILFS_RECOVERY_SR_UPDATED;
|
|
else {
|
|
if (nilfs->ns_mount_state & NILFS_VALID_FS)
|
|
goto super_root_found;
|
|
scan_newer = 1;
|
|
}
|
|
|
|
/* reset region for roll-forward */
|
|
pseg_start += ssi.nblocks;
|
|
if (pseg_start < seg_end)
|
|
continue;
|
|
goto feed_segment;
|
|
|
|
try_next_pseg:
|
|
/* Standing on a course, or met an inconsistent state */
|
|
pseg_start += ssi.nblocks;
|
|
if (pseg_start < seg_end)
|
|
continue;
|
|
goto feed_segment;
|
|
|
|
strayed:
|
|
/* Off the trail */
|
|
if (!scan_newer)
|
|
/*
|
|
* This can happen if a checkpoint was written without
|
|
* barriers, or as a result of an I/O failure.
|
|
*/
|
|
goto failed;
|
|
|
|
feed_segment:
|
|
/* Looking to the next full segment */
|
|
if (empty_seg++)
|
|
goto super_root_found; /* found a valid super root */
|
|
|
|
ret = nilfs_segment_list_add(&segments, segnum);
|
|
if (unlikely(ret))
|
|
goto failed;
|
|
|
|
seg_seq++;
|
|
segnum = nextnum;
|
|
nilfs_get_segment_range(nilfs, segnum, &seg_start, &seg_end);
|
|
pseg_start = seg_start;
|
|
}
|
|
|
|
super_root_found:
|
|
/* Updating pointers relating to the latest checkpoint */
|
|
list_splice_tail(&segments, &ri->ri_used_segments);
|
|
nilfs->ns_last_pseg = sr_pseg_start;
|
|
nilfs->ns_last_seq = nilfs->ns_seg_seq;
|
|
nilfs->ns_last_cno = ri->ri_cno;
|
|
return 0;
|
|
|
|
failed:
|
|
nilfs_dispose_segment_list(&segments);
|
|
return (ret < 0) ? ret : nilfs_warn_segment_error(ret);
|
|
}
|