aboutsummaryrefslogtreecommitdiffstats
path: root/fs/f2fs/gc.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/f2fs/gc.c')
-rw-r--r--fs/f2fs/gc.c60
1 files changed, 21 insertions, 39 deletions
diff --git a/fs/f2fs/gc.c b/fs/f2fs/gc.c
index b0ec721e984a..b4dd90cf1f18 100644
--- a/fs/f2fs/gc.c
+++ b/fs/f2fs/gc.c
@@ -78,7 +78,7 @@ static int gc_thread_func(void *data)
78 78
79 sbi->bg_gc++; 79 sbi->bg_gc++;
80 80
81 if (f2fs_gc(sbi, 1) == GC_NONE) 81 if (f2fs_gc(sbi) == GC_NONE)
82 wait_ms = GC_THREAD_NOGC_SLEEP_TIME; 82 wait_ms = GC_THREAD_NOGC_SLEEP_TIME;
83 else if (wait_ms == GC_THREAD_NOGC_SLEEP_TIME) 83 else if (wait_ms == GC_THREAD_NOGC_SLEEP_TIME)
84 wait_ms = GC_THREAD_MAX_SLEEP_TIME; 84 wait_ms = GC_THREAD_MAX_SLEEP_TIME;
@@ -651,62 +651,44 @@ static int do_garbage_collect(struct f2fs_sb_info *sbi, unsigned int segno,
651 return ret; 651 return ret;
652} 652}
653 653
654int f2fs_gc(struct f2fs_sb_info *sbi, int nGC) 654int f2fs_gc(struct f2fs_sb_info *sbi)
655{ 655{
656 unsigned int segno;
657 int old_free_secs, cur_free_secs;
658 int gc_status, nfree;
659 struct list_head ilist; 656 struct list_head ilist;
657 unsigned int segno, i;
660 int gc_type = BG_GC; 658 int gc_type = BG_GC;
659 int gc_status = GC_NONE;
661 660
662 INIT_LIST_HEAD(&ilist); 661 INIT_LIST_HEAD(&ilist);
663gc_more: 662gc_more:
664 nfree = 0; 663 if (!(sbi->sb->s_flags & MS_ACTIVE))
665 gc_status = GC_NONE; 664 goto stop;
666 665
667 if (has_not_enough_free_secs(sbi)) 666 if (has_not_enough_free_secs(sbi))
668 old_free_secs = reserved_sections(sbi); 667 gc_type = FG_GC;
669 else
670 old_free_secs = free_sections(sbi);
671
672 while (sbi->sb->s_flags & MS_ACTIVE) {
673 int i;
674 if (has_not_enough_free_secs(sbi))
675 gc_type = FG_GC;
676 668
677 cur_free_secs = free_sections(sbi) + nfree; 669 if (!__get_victim(sbi, &segno, gc_type, NO_CHECK_TYPE))
670 goto stop;
678 671
679 /* We got free space successfully. */ 672 for (i = 0; i < sbi->segs_per_sec; i++) {
680 if (nGC < cur_free_secs - old_free_secs) 673 /*
681 break; 674 * do_garbage_collect will give us three gc_status:
682 675 * GC_ERROR, GC_DONE, and GC_BLOCKED.
683 if (!__get_victim(sbi, &segno, gc_type, NO_CHECK_TYPE)) 676 * If GC is finished uncleanly, we have to return
677 * the victim to dirty segment list.
678 */
679 gc_status = do_garbage_collect(sbi, segno + i, &ilist, gc_type);
680 if (gc_status != GC_DONE)
684 break; 681 break;
685
686 for (i = 0; i < sbi->segs_per_sec; i++) {
687 /*
688 * do_garbage_collect will give us three gc_status:
689 * GC_ERROR, GC_DONE, and GC_BLOCKED.
690 * If GC is finished uncleanly, we have to return
691 * the victim to dirty segment list.
692 */
693 gc_status = do_garbage_collect(sbi, segno + i,
694 &ilist, gc_type);
695 if (gc_status != GC_DONE)
696 goto stop;
697 nfree++;
698 }
699 } 682 }
700stop: 683 if (has_not_enough_free_secs(sbi)) {
701 if (has_not_enough_free_secs(sbi) || gc_status == GC_BLOCKED) {
702 write_checkpoint(sbi, (gc_status == GC_BLOCKED), false); 684 write_checkpoint(sbi, (gc_status == GC_BLOCKED), false);
703 if (nfree) 685 if (has_not_enough_free_secs(sbi))
704 goto gc_more; 686 goto gc_more;
705 } 687 }
688stop:
706 mutex_unlock(&sbi->gc_mutex); 689 mutex_unlock(&sbi->gc_mutex);
707 690
708 put_gc_inode(&ilist); 691 put_gc_inode(&ilist);
709 BUG_ON(!list_empty(&ilist));
710 return gc_status; 692 return gc_status;
711} 693}
712 694