Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 57864ae5 authored by Jaegeuk Kim's avatar Jaegeuk Kim
Browse files

f2fs: limit # of inmemory pages



If some abnormal users try lots of atomic write operations, f2fs is able to
produce pinned pages in the main memory which affects system performance.
This patch limits that as 20% over total memory size, and if f2fs reaches
to the limit, it will drop all the inmemory pages.

Signed-off-by: default avatarJaegeuk Kim <jaegeuk@kernel.org>
parent ab383be5
Loading
Loading
Loading
Loading
+8 −0
Original line number Original line Diff line number Diff line
@@ -1944,6 +1944,12 @@ static int f2fs_write_begin(struct file *file, struct address_space *mapping,


	trace_f2fs_write_begin(inode, pos, len, flags);
	trace_f2fs_write_begin(inode, pos, len, flags);


	if (f2fs_is_atomic_file(inode) &&
			!available_free_memory(sbi, INMEM_PAGES)) {
		err = -ENOMEM;
		goto fail;
	}

	/*
	/*
	 * We should check this at this moment to avoid deadlock on inode page
	 * We should check this at this moment to avoid deadlock on inode page
	 * and #0 page. The locking rule for inline_data conversion should be:
	 * and #0 page. The locking rule for inline_data conversion should be:
@@ -2021,6 +2027,8 @@ static int f2fs_write_begin(struct file *file, struct address_space *mapping,
fail:
fail:
	f2fs_put_page(page, 1);
	f2fs_put_page(page, 1);
	f2fs_write_failed(mapping, pos + len);
	f2fs_write_failed(mapping, pos + len);
	if (f2fs_is_atomic_file(inode))
		drop_inmem_pages_all(sbi);
	return err;
	return err;
}
}


+3 −0
Original line number Original line Diff line number Diff line
@@ -601,6 +601,7 @@ struct f2fs_inode_info {
#endif
#endif
	struct list_head dirty_list;	/* dirty list for dirs and files */
	struct list_head dirty_list;	/* dirty list for dirs and files */
	struct list_head gdirty_list;	/* linked in global dirty list */
	struct list_head gdirty_list;	/* linked in global dirty list */
	struct list_head inmem_ilist;	/* list for inmem inodes */
	struct list_head inmem_pages;	/* inmemory pages managed by f2fs */
	struct list_head inmem_pages;	/* inmemory pages managed by f2fs */
	struct task_struct *inmem_task;	/* store inmemory task */
	struct task_struct *inmem_task;	/* store inmemory task */
	struct mutex inmem_lock;	/* lock for inmemory pages */
	struct mutex inmem_lock;	/* lock for inmemory pages */
@@ -965,6 +966,7 @@ enum inode_type {
	DIR_INODE,			/* for dirty dir inode */
	DIR_INODE,			/* for dirty dir inode */
	FILE_INODE,			/* for dirty regular/symlink inode */
	FILE_INODE,			/* for dirty regular/symlink inode */
	DIRTY_META,			/* for all dirtied inode metadata */
	DIRTY_META,			/* for all dirtied inode metadata */
	ATOMIC_FILE,			/* for all atomic files */
	NR_INODE_TYPE,
	NR_INODE_TYPE,
};
};


@@ -2544,6 +2546,7 @@ void destroy_node_manager_caches(void);
 */
 */
bool need_SSR(struct f2fs_sb_info *sbi);
bool need_SSR(struct f2fs_sb_info *sbi);
void register_inmem_page(struct inode *inode, struct page *page);
void register_inmem_page(struct inode *inode, struct page *page);
void drop_inmem_pages_all(struct f2fs_sb_info *sbi);
void drop_inmem_pages(struct inode *inode);
void drop_inmem_pages(struct inode *inode);
void drop_inmem_page(struct inode *inode, struct page *page);
void drop_inmem_page(struct inode *inode, struct page *page);
int commit_inmem_pages(struct inode *inode);
int commit_inmem_pages(struct inode *inode);
+4 −0
Original line number Original line Diff line number Diff line
@@ -74,6 +74,10 @@ bool available_free_memory(struct f2fs_sb_info *sbi, int type)
				atomic_read(&sbi->total_ext_node) *
				atomic_read(&sbi->total_ext_node) *
				sizeof(struct extent_node)) >> PAGE_SHIFT;
				sizeof(struct extent_node)) >> PAGE_SHIFT;
		res = mem_size < ((avail_ram * nm_i->ram_thresh / 100) >> 1);
		res = mem_size < ((avail_ram * nm_i->ram_thresh / 100) >> 1);
	} else if (type == INMEM_PAGES) {
		/* it allows 20% / total_ram for inmemory pages */
		mem_size = get_pages(sbi, F2FS_INMEM_PAGES);
		res = mem_size < (val.totalram / 5);
	} else {
	} else {
		if (!sbi->sb->s_bdi->wb.dirty_exceeded)
		if (!sbi->sb->s_bdi->wb.dirty_exceeded)
			return true;
			return true;
+1 −0
Original line number Original line Diff line number Diff line
@@ -140,6 +140,7 @@ enum mem_type {
	DIRTY_DENTS,	/* indicates dirty dentry pages */
	DIRTY_DENTS,	/* indicates dirty dentry pages */
	INO_ENTRIES,	/* indicates inode entries */
	INO_ENTRIES,	/* indicates inode entries */
	EXTENT_CACHE,	/* indicates extent cache */
	EXTENT_CACHE,	/* indicates extent cache */
	INMEM_PAGES,	/* indicates inmemory pages */
	BASE_CHECK,	/* check kernel status */
	BASE_CHECK,	/* check kernel status */
};
};


+38 −0
Original line number Original line Diff line number Diff line
@@ -186,6 +186,7 @@ bool need_SSR(struct f2fs_sb_info *sbi)


void register_inmem_page(struct inode *inode, struct page *page)
void register_inmem_page(struct inode *inode, struct page *page)
{
{
	struct f2fs_sb_info *sbi = F2FS_I_SB(inode);
	struct f2fs_inode_info *fi = F2FS_I(inode);
	struct f2fs_inode_info *fi = F2FS_I(inode);
	struct inmem_pages *new;
	struct inmem_pages *new;


@@ -204,6 +205,10 @@ void register_inmem_page(struct inode *inode, struct page *page)
	mutex_lock(&fi->inmem_lock);
	mutex_lock(&fi->inmem_lock);
	get_page(page);
	get_page(page);
	list_add_tail(&new->list, &fi->inmem_pages);
	list_add_tail(&new->list, &fi->inmem_pages);
	spin_lock(&sbi->inode_lock[ATOMIC_FILE]);
	if (list_empty(&fi->inmem_ilist))
		list_add_tail(&fi->inmem_ilist, &sbi->inode_list[ATOMIC_FILE]);
	spin_unlock(&sbi->inode_lock[ATOMIC_FILE]);
	inc_page_count(F2FS_I_SB(inode), F2FS_INMEM_PAGES);
	inc_page_count(F2FS_I_SB(inode), F2FS_INMEM_PAGES);
	mutex_unlock(&fi->inmem_lock);
	mutex_unlock(&fi->inmem_lock);


@@ -262,12 +267,41 @@ static int __revoke_inmem_pages(struct inode *inode,
	return err;
	return err;
}
}


void drop_inmem_pages_all(struct f2fs_sb_info *sbi)
{
	struct list_head *head = &sbi->inode_list[ATOMIC_FILE];
	struct inode *inode;
	struct f2fs_inode_info *fi;
next:
	spin_lock(&sbi->inode_lock[ATOMIC_FILE]);
	if (list_empty(head)) {
		spin_unlock(&sbi->inode_lock[ATOMIC_FILE]);
		return;
	}
	fi = list_first_entry(head, struct f2fs_inode_info, inmem_ilist);
	inode = igrab(&fi->vfs_inode);
	spin_unlock(&sbi->inode_lock[ATOMIC_FILE]);

	if (inode) {
		drop_inmem_pages(inode);
		iput(inode);
	}
	congestion_wait(BLK_RW_ASYNC, HZ/50);
	cond_resched();
	goto next;
}

void drop_inmem_pages(struct inode *inode)
void drop_inmem_pages(struct inode *inode)
{
{
	struct f2fs_sb_info *sbi = F2FS_I_SB(inode);
	struct f2fs_inode_info *fi = F2FS_I(inode);
	struct f2fs_inode_info *fi = F2FS_I(inode);


	mutex_lock(&fi->inmem_lock);
	mutex_lock(&fi->inmem_lock);
	__revoke_inmem_pages(inode, &fi->inmem_pages, true, false);
	__revoke_inmem_pages(inode, &fi->inmem_pages, true, false);
	spin_lock(&sbi->inode_lock[ATOMIC_FILE]);
	if (!list_empty(&fi->inmem_ilist))
		list_del_init(&fi->inmem_ilist);
	spin_unlock(&sbi->inode_lock[ATOMIC_FILE]);
	mutex_unlock(&fi->inmem_lock);
	mutex_unlock(&fi->inmem_lock);


	clear_inode_flag(inode, FI_ATOMIC_FILE);
	clear_inode_flag(inode, FI_ATOMIC_FILE);
@@ -399,6 +433,10 @@ int commit_inmem_pages(struct inode *inode)
		/* drop all uncommitted pages */
		/* drop all uncommitted pages */
		__revoke_inmem_pages(inode, &fi->inmem_pages, true, false);
		__revoke_inmem_pages(inode, &fi->inmem_pages, true, false);
	}
	}
	spin_lock(&sbi->inode_lock[ATOMIC_FILE]);
	if (!list_empty(&fi->inmem_ilist))
		list_del_init(&fi->inmem_ilist);
	spin_unlock(&sbi->inode_lock[ATOMIC_FILE]);
	mutex_unlock(&fi->inmem_lock);
	mutex_unlock(&fi->inmem_lock);


	clear_inode_flag(inode, FI_ATOMIC_COMMIT);
	clear_inode_flag(inode, FI_ATOMIC_COMMIT);
Loading