f2fs: use bitmap in discard_entry
This patch changes to use bitmap instead of extent in struct discard_entry to indicate discard range in one segment, for fragmented space, this implementation can save memory footprint. Signed-off-by: Chao Yu <yuchao0@huawei.com> Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
This commit is contained in:
@ -182,11 +182,11 @@ struct inode_entry {
|
|||||||
struct inode *inode; /* vfs inode pointer */
|
struct inode *inode; /* vfs inode pointer */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* for the list of blockaddresses to be discarded */
|
/* for the bitmap indicate blocks to be discarded */
|
||||||
struct discard_entry {
|
struct discard_entry {
|
||||||
struct list_head list; /* list head */
|
struct list_head list; /* list head */
|
||||||
block_t blkaddr; /* block address to be discarded */
|
block_t start_blkaddr; /* start blockaddr of current segment */
|
||||||
int len; /* # of consecutive blocks of the discard */
|
unsigned char discard_map[SIT_VBLOCK_MAP_SIZE]; /* segment discard bitmap */
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
|
@ -962,32 +962,6 @@ static int f2fs_issue_discard(struct f2fs_sb_info *sbi,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __add_discard_entry(struct f2fs_sb_info *sbi,
|
|
||||||
struct cp_control *cpc, struct seg_entry *se,
|
|
||||||
unsigned int start, unsigned int end)
|
|
||||||
{
|
|
||||||
struct list_head *head = &SM_I(sbi)->dcc_info->discard_entry_list;
|
|
||||||
struct discard_entry *new, *last;
|
|
||||||
|
|
||||||
if (!list_empty(head)) {
|
|
||||||
last = list_last_entry(head, struct discard_entry, list);
|
|
||||||
if (START_BLOCK(sbi, cpc->trim_start) + start ==
|
|
||||||
last->blkaddr + last->len &&
|
|
||||||
last->len < MAX_DISCARD_BLOCKS(sbi)) {
|
|
||||||
last->len += end - start;
|
|
||||||
goto done;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
new = f2fs_kmem_cache_alloc(discard_entry_slab, GFP_NOFS);
|
|
||||||
INIT_LIST_HEAD(&new->list);
|
|
||||||
new->blkaddr = START_BLOCK(sbi, cpc->trim_start) + start;
|
|
||||||
new->len = end - start;
|
|
||||||
list_add_tail(&new->list, head);
|
|
||||||
done:
|
|
||||||
SM_I(sbi)->dcc_info->nr_discards += end - start;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool add_discard_addrs(struct f2fs_sb_info *sbi, struct cp_control *cpc,
|
static bool add_discard_addrs(struct f2fs_sb_info *sbi, struct cp_control *cpc,
|
||||||
bool check_only)
|
bool check_only)
|
||||||
{
|
{
|
||||||
@ -1000,6 +974,8 @@ static bool add_discard_addrs(struct f2fs_sb_info *sbi, struct cp_control *cpc,
|
|||||||
unsigned long *dmap = SIT_I(sbi)->tmp_map;
|
unsigned long *dmap = SIT_I(sbi)->tmp_map;
|
||||||
unsigned int start = 0, end = -1;
|
unsigned int start = 0, end = -1;
|
||||||
bool force = (cpc->reason == CP_DISCARD);
|
bool force = (cpc->reason == CP_DISCARD);
|
||||||
|
struct discard_entry *de = NULL;
|
||||||
|
struct list_head *head = &SM_I(sbi)->dcc_info->discard_entry_list;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (se->valid_blocks == max_blocks || !f2fs_discard_en(sbi))
|
if (se->valid_blocks == max_blocks || !f2fs_discard_en(sbi))
|
||||||
@ -1031,7 +1007,17 @@ static bool add_discard_addrs(struct f2fs_sb_info *sbi, struct cp_control *cpc,
|
|||||||
if (check_only)
|
if (check_only)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
__add_discard_entry(sbi, cpc, se, start, end);
|
if (!de) {
|
||||||
|
de = f2fs_kmem_cache_alloc(discard_entry_slab,
|
||||||
|
GFP_F2FS_ZERO);
|
||||||
|
de->start_blkaddr = START_BLOCK(sbi, cpc->trim_start);
|
||||||
|
list_add_tail(&de->list, head);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = start; i < end; i++)
|
||||||
|
__set_bit_le(i, (void *)de->discard_map);
|
||||||
|
|
||||||
|
SM_I(sbi)->dcc_info->nr_discards += end - start;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1117,13 +1103,35 @@ next:
|
|||||||
|
|
||||||
/* send small discards */
|
/* send small discards */
|
||||||
list_for_each_entry_safe(entry, this, head, list) {
|
list_for_each_entry_safe(entry, this, head, list) {
|
||||||
if (force && entry->len < cpc->trim_minlen)
|
unsigned int cur_pos = 0, next_pos, len, total_len = 0;
|
||||||
goto skip;
|
bool is_valid = test_bit_le(0, entry->discard_map);
|
||||||
f2fs_issue_discard(sbi, entry->blkaddr, entry->len);
|
|
||||||
cpc->trimmed += entry->len;
|
find_next:
|
||||||
|
if (is_valid) {
|
||||||
|
next_pos = find_next_zero_bit_le(entry->discard_map,
|
||||||
|
sbi->blocks_per_seg, cur_pos);
|
||||||
|
len = next_pos - cur_pos;
|
||||||
|
|
||||||
|
if (force && len < cpc->trim_minlen)
|
||||||
|
goto skip;
|
||||||
|
|
||||||
|
f2fs_issue_discard(sbi, entry->start_blkaddr + cur_pos,
|
||||||
|
len);
|
||||||
|
cpc->trimmed += len;
|
||||||
|
total_len += len;
|
||||||
|
} else {
|
||||||
|
next_pos = find_next_bit_le(entry->discard_map,
|
||||||
|
sbi->blocks_per_seg, cur_pos);
|
||||||
|
}
|
||||||
skip:
|
skip:
|
||||||
|
cur_pos = next_pos;
|
||||||
|
is_valid = !is_valid;
|
||||||
|
|
||||||
|
if (cur_pos < sbi->blocks_per_seg)
|
||||||
|
goto find_next;
|
||||||
|
|
||||||
list_del(&entry->list);
|
list_del(&entry->list);
|
||||||
SM_I(sbi)->dcc_info->nr_discards -= entry->len;
|
SM_I(sbi)->dcc_info->nr_discards -= total_len;
|
||||||
kmem_cache_free(discard_entry_slab, entry);
|
kmem_cache_free(discard_entry_slab, entry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user