diff options
-rw-r--r-- | fs/nilfs2/mdt.c | 67 | ||||
-rw-r--r-- | fs/nilfs2/mdt.h | 3 | ||||
-rw-r--r-- | fs/nilfs2/page.c | 2 | ||||
-rw-r--r-- | fs/nilfs2/page.h | 2 | ||||
-rw-r--r-- | fs/nilfs2/segment.c | 1 |
5 files changed, 75 insertions, 0 deletions
diff --git a/fs/nilfs2/mdt.c b/fs/nilfs2/mdt.c index 0066468609da..532f85acf273 100644 --- a/fs/nilfs2/mdt.c +++ b/fs/nilfs2/mdt.c | |||
@@ -622,6 +622,72 @@ int nilfs_mdt_save_to_shadow_map(struct inode *inode) | |||
622 | return ret; | 622 | return ret; |
623 | } | 623 | } |
624 | 624 | ||
625 | int nilfs_mdt_freeze_buffer(struct inode *inode, struct buffer_head *bh) | ||
626 | { | ||
627 | struct nilfs_shadow_map *shadow = NILFS_MDT(inode)->mi_shadow; | ||
628 | struct buffer_head *bh_frozen; | ||
629 | struct page *page; | ||
630 | int blkbits = inode->i_blkbits; | ||
631 | int ret = -ENOMEM; | ||
632 | |||
633 | page = grab_cache_page(&shadow->frozen_data, bh->b_page->index); | ||
634 | if (!page) | ||
635 | return ret; | ||
636 | |||
637 | if (!page_has_buffers(page)) | ||
638 | create_empty_buffers(page, 1 << blkbits, 0); | ||
639 | |||
640 | bh_frozen = nilfs_page_get_nth_block(page, bh_offset(bh) >> blkbits); | ||
641 | if (bh_frozen) { | ||
642 | if (!buffer_uptodate(bh_frozen)) | ||
643 | nilfs_copy_buffer(bh_frozen, bh); | ||
644 | if (list_empty(&bh_frozen->b_assoc_buffers)) { | ||
645 | list_add_tail(&bh_frozen->b_assoc_buffers, | ||
646 | &shadow->frozen_buffers); | ||
647 | set_buffer_nilfs_redirected(bh); | ||
648 | } else { | ||
649 | brelse(bh_frozen); /* already frozen */ | ||
650 | } | ||
651 | ret = 0; | ||
652 | } | ||
653 | unlock_page(page); | ||
654 | page_cache_release(page); | ||
655 | return ret; | ||
656 | } | ||
657 | |||
658 | struct buffer_head * | ||
659 | nilfs_mdt_get_frozen_buffer(struct inode *inode, struct buffer_head *bh) | ||
660 | { | ||
661 | struct nilfs_shadow_map *shadow = NILFS_MDT(inode)->mi_shadow; | ||
662 | struct buffer_head *bh_frozen = NULL; | ||
663 | struct page *page; | ||
664 | int n; | ||
665 | |||
666 | page = find_lock_page(&shadow->frozen_data, bh->b_page->index); | ||
667 | if (page) { | ||
668 | if (page_has_buffers(page)) { | ||
669 | n = bh_offset(bh) >> inode->i_blkbits; | ||
670 | bh_frozen = nilfs_page_get_nth_block(page, n); | ||
671 | } | ||
672 | unlock_page(page); | ||
673 | page_cache_release(page); | ||
674 | } | ||
675 | return bh_frozen; | ||
676 | } | ||
677 | |||
678 | static void nilfs_release_frozen_buffers(struct nilfs_shadow_map *shadow) | ||
679 | { | ||
680 | struct list_head *head = &shadow->frozen_buffers; | ||
681 | struct buffer_head *bh; | ||
682 | |||
683 | while (!list_empty(head)) { | ||
684 | bh = list_first_entry(head, struct buffer_head, | ||
685 | b_assoc_buffers); | ||
686 | list_del_init(&bh->b_assoc_buffers); | ||
687 | brelse(bh); /* drop ref-count to make it releasable */ | ||
688 | } | ||
689 | } | ||
690 | |||
625 | /** | 691 | /** |
626 | * nilfs_mdt_restore_from_shadow_map - restore dirty pages and bmap state | 692 | * nilfs_mdt_restore_from_shadow_map - restore dirty pages and bmap state |
627 | * @inode: inode of the metadata file | 693 | * @inode: inode of the metadata file |
@@ -658,6 +724,7 @@ void nilfs_mdt_clear_shadow_map(struct inode *inode) | |||
658 | struct nilfs_shadow_map *shadow = mi->mi_shadow; | 724 | struct nilfs_shadow_map *shadow = mi->mi_shadow; |
659 | 725 | ||
660 | down_write(&mi->mi_sem); | 726 | down_write(&mi->mi_sem); |
727 | nilfs_release_frozen_buffers(shadow); | ||
661 | truncate_inode_pages(&shadow->frozen_data, 0); | 728 | truncate_inode_pages(&shadow->frozen_data, 0); |
662 | truncate_inode_pages(&shadow->frozen_btnodes, 0); | 729 | truncate_inode_pages(&shadow->frozen_btnodes, 0); |
663 | up_write(&mi->mi_sem); | 730 | up_write(&mi->mi_sem); |
diff --git a/fs/nilfs2/mdt.h b/fs/nilfs2/mdt.h index e7f0d158c527..e60bbfe899f1 100644 --- a/fs/nilfs2/mdt.h +++ b/fs/nilfs2/mdt.h | |||
@@ -100,6 +100,9 @@ int nilfs_mdt_setup_shadow_map(struct inode *inode, | |||
100 | int nilfs_mdt_save_to_shadow_map(struct inode *inode); | 100 | int nilfs_mdt_save_to_shadow_map(struct inode *inode); |
101 | void nilfs_mdt_restore_from_shadow_map(struct inode *inode); | 101 | void nilfs_mdt_restore_from_shadow_map(struct inode *inode); |
102 | void nilfs_mdt_clear_shadow_map(struct inode *inode); | 102 | void nilfs_mdt_clear_shadow_map(struct inode *inode); |
103 | int nilfs_mdt_freeze_buffer(struct inode *inode, struct buffer_head *bh); | ||
104 | struct buffer_head *nilfs_mdt_get_frozen_buffer(struct inode *inode, | ||
105 | struct buffer_head *bh); | ||
103 | 106 | ||
104 | #define nilfs_mdt_mark_buffer_dirty(bh) nilfs_mark_buffer_dirty(bh) | 107 | #define nilfs_mdt_mark_buffer_dirty(bh) nilfs_mark_buffer_dirty(bh) |
105 | 108 | ||
diff --git a/fs/nilfs2/page.c b/fs/nilfs2/page.c index 6384ac14c0c8..7083344ac881 100644 --- a/fs/nilfs2/page.c +++ b/fs/nilfs2/page.c | |||
@@ -131,6 +131,7 @@ void nilfs_forget_buffer(struct buffer_head *bh) | |||
131 | lock_buffer(bh); | 131 | lock_buffer(bh); |
132 | clear_buffer_nilfs_volatile(bh); | 132 | clear_buffer_nilfs_volatile(bh); |
133 | clear_buffer_nilfs_checked(bh); | 133 | clear_buffer_nilfs_checked(bh); |
134 | clear_buffer_nilfs_redirected(bh); | ||
134 | clear_buffer_dirty(bh); | 135 | clear_buffer_dirty(bh); |
135 | if (nilfs_page_buffers_clean(page)) | 136 | if (nilfs_page_buffers_clean(page)) |
136 | __nilfs_clear_page_dirty(page); | 137 | __nilfs_clear_page_dirty(page); |
@@ -483,6 +484,7 @@ void nilfs_clear_dirty_pages(struct address_space *mapping) | |||
483 | clear_buffer_dirty(bh); | 484 | clear_buffer_dirty(bh); |
484 | clear_buffer_nilfs_volatile(bh); | 485 | clear_buffer_nilfs_volatile(bh); |
485 | clear_buffer_nilfs_checked(bh); | 486 | clear_buffer_nilfs_checked(bh); |
487 | clear_buffer_nilfs_redirected(bh); | ||
486 | clear_buffer_uptodate(bh); | 488 | clear_buffer_uptodate(bh); |
487 | clear_buffer_mapped(bh); | 489 | clear_buffer_mapped(bh); |
488 | unlock_buffer(bh); | 490 | unlock_buffer(bh); |
diff --git a/fs/nilfs2/page.h b/fs/nilfs2/page.h index 6ec4f498fc2b..fb9e8a8a2038 100644 --- a/fs/nilfs2/page.h +++ b/fs/nilfs2/page.h | |||
@@ -35,12 +35,14 @@ enum { | |||
35 | BH_NILFS_Node, | 35 | BH_NILFS_Node, |
36 | BH_NILFS_Volatile, | 36 | BH_NILFS_Volatile, |
37 | BH_NILFS_Checked, | 37 | BH_NILFS_Checked, |
38 | BH_NILFS_Redirected, | ||
38 | }; | 39 | }; |
39 | 40 | ||
40 | BUFFER_FNS(NILFS_Allocated, nilfs_allocated) /* nilfs private buffers */ | 41 | BUFFER_FNS(NILFS_Allocated, nilfs_allocated) /* nilfs private buffers */ |
41 | BUFFER_FNS(NILFS_Node, nilfs_node) /* nilfs node buffers */ | 42 | BUFFER_FNS(NILFS_Node, nilfs_node) /* nilfs node buffers */ |
42 | BUFFER_FNS(NILFS_Volatile, nilfs_volatile) | 43 | BUFFER_FNS(NILFS_Volatile, nilfs_volatile) |
43 | BUFFER_FNS(NILFS_Checked, nilfs_checked) /* buffer is verified */ | 44 | BUFFER_FNS(NILFS_Checked, nilfs_checked) /* buffer is verified */ |
45 | BUFFER_FNS(NILFS_Redirected, nilfs_redirected) /* redirected to a copy */ | ||
44 | 46 | ||
45 | 47 | ||
46 | void nilfs_mark_buffer_dirty(struct buffer_head *bh); | 48 | void nilfs_mark_buffer_dirty(struct buffer_head *bh); |
diff --git a/fs/nilfs2/segment.c b/fs/nilfs2/segment.c index b75306d642c2..91dc0668ec83 100644 --- a/fs/nilfs2/segment.c +++ b/fs/nilfs2/segment.c | |||
@@ -1908,6 +1908,7 @@ static void nilfs_segctor_complete_write(struct nilfs_sc_info *sci) | |||
1908 | set_buffer_uptodate(bh); | 1908 | set_buffer_uptodate(bh); |
1909 | clear_buffer_dirty(bh); | 1909 | clear_buffer_dirty(bh); |
1910 | clear_buffer_nilfs_volatile(bh); | 1910 | clear_buffer_nilfs_volatile(bh); |
1911 | clear_buffer_nilfs_redirected(bh); | ||
1911 | if (bh == segbuf->sb_super_root) { | 1912 | if (bh == segbuf->sb_super_root) { |
1912 | if (bh->b_page != bd_page) { | 1913 | if (bh->b_page != bd_page) { |
1913 | end_page_writeback(bd_page); | 1914 | end_page_writeback(bd_page); |