Commit 9edbcc72 authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Fix bch2_evict_subvolume_inodes()

This fixes a bug in bch2_evict_subvolume_inodes(): d_mark_dontcache()
doesn't handle the case where i_count is already 0, we need to grab and
put the inode in order for it to be dropped.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent e1e7ecaf
...@@ -971,6 +971,10 @@ struct bch_fs { ...@@ -971,6 +971,10 @@ struct bch_fs {
reflink_gc_table reflink_gc_table; reflink_gc_table reflink_gc_table;
size_t reflink_gc_nr; size_t reflink_gc_nr;
/* fs.c */
struct list_head vfs_inodes_list;
struct mutex vfs_inodes_lock;
/* VFS IO PATH - fs-io.c */ /* VFS IO PATH - fs-io.c */
struct bio_set writepage_bioset; struct bio_set writepage_bioset;
struct bio_set dio_write_bioset; struct bio_set dio_write_bioset;
......
...@@ -19,11 +19,11 @@ struct { \ ...@@ -19,11 +19,11 @@ struct { \
typedef DARRAY(void) darray_void; typedef DARRAY(void) darray_void;
static inline int __darray_make_room(darray_void *d, size_t t_size, size_t more) static inline int __darray_make_room(darray_void *d, size_t t_size, size_t more, gfp_t gfp)
{ {
if (d->nr + more > d->size) { if (d->nr + more > d->size) {
size_t new_size = roundup_pow_of_two(d->nr + more); size_t new_size = roundup_pow_of_two(d->nr + more);
void *data = krealloc_array(d->data, new_size, t_size, GFP_KERNEL); void *data = krealloc_array(d->data, new_size, t_size, gfp);
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
...@@ -35,20 +35,25 @@ static inline int __darray_make_room(darray_void *d, size_t t_size, size_t more) ...@@ -35,20 +35,25 @@ static inline int __darray_make_room(darray_void *d, size_t t_size, size_t more)
return 0; return 0;
} }
#define darray_make_room_gfp(_d, _more, _gfp) \
__darray_make_room((darray_void *) (_d), sizeof((_d)->data[0]), (_more), _gfp)
#define darray_make_room(_d, _more) \ #define darray_make_room(_d, _more) \
__darray_make_room((darray_void *) (_d), sizeof((_d)->data[0]), (_more)) darray_make_room_gfp(_d, _more, GFP_KERNEL)
#define darray_top(_d) ((_d).data[(_d).nr]) #define darray_top(_d) ((_d).data[(_d).nr])
#define darray_push(_d, _item) \ #define darray_push_gfp(_d, _item, _gfp) \
({ \ ({ \
int _ret = darray_make_room((_d), 1); \ int _ret = darray_make_room_gfp((_d), 1, _gfp); \
\ \
if (!_ret) \ if (!_ret) \
(_d)->data[(_d)->nr++] = (_item); \ (_d)->data[(_d)->nr++] = (_item); \
_ret; \ _ret; \
}) })
#define darray_push(_d, _item) darray_push_gfp(_d, _item, GFP_KERNEL)
#define darray_insert_item(_d, _pos, _item) \ #define darray_insert_item(_d, _pos, _item) \
({ \ ({ \
size_t pos = (_pos); \ size_t pos = (_pos); \
......
...@@ -201,6 +201,10 @@ struct inode *bch2_vfs_inode_get(struct bch_fs *c, subvol_inum inum) ...@@ -201,6 +201,10 @@ struct inode *bch2_vfs_inode_get(struct bch_fs *c, subvol_inum inum)
return ERR_PTR(ret); return ERR_PTR(ret);
} }
mutex_lock(&c->vfs_inodes_lock);
list_add(&inode->ei_vfs_inode_list, &c->vfs_inodes_list);
mutex_unlock(&c->vfs_inodes_lock);
unlock_new_inode(&inode->v); unlock_new_inode(&inode->v);
return &inode->v; return &inode->v;
...@@ -314,6 +318,9 @@ __bch2_create(struct mnt_idmap *idmap, ...@@ -314,6 +318,9 @@ __bch2_create(struct mnt_idmap *idmap,
inode = old; inode = old;
} else { } else {
mutex_lock(&c->vfs_inodes_lock);
list_add(&inode->ei_vfs_inode_list, &c->vfs_inodes_list);
mutex_unlock(&c->vfs_inodes_lock);
/* /*
* we really don't want insert_inode_locked2() to be setting * we really don't want insert_inode_locked2() to be setting
* I_NEW... * I_NEW...
...@@ -1370,6 +1377,7 @@ static struct inode *bch2_alloc_inode(struct super_block *sb) ...@@ -1370,6 +1377,7 @@ static struct inode *bch2_alloc_inode(struct super_block *sb)
inode_init_once(&inode->v); inode_init_once(&inode->v);
mutex_init(&inode->ei_update_lock); mutex_init(&inode->ei_update_lock);
two_state_lock_init(&inode->ei_pagecache_lock); two_state_lock_init(&inode->ei_pagecache_lock);
INIT_LIST_HEAD(&inode->ei_vfs_inode_list);
mutex_init(&inode->ei_quota_lock); mutex_init(&inode->ei_quota_lock);
return &inode->v; return &inode->v;
...@@ -1434,53 +1442,78 @@ static void bch2_evict_inode(struct inode *vinode) ...@@ -1434,53 +1442,78 @@ static void bch2_evict_inode(struct inode *vinode)
KEY_TYPE_QUOTA_WARN); KEY_TYPE_QUOTA_WARN);
bch2_inode_rm(c, inode_inum(inode)); bch2_inode_rm(c, inode_inum(inode));
} }
mutex_lock(&c->vfs_inodes_lock);
list_del_init(&inode->ei_vfs_inode_list);
mutex_unlock(&c->vfs_inodes_lock);
} }
void bch2_evict_subvolume_inodes(struct bch_fs *c, void bch2_evict_subvolume_inodes(struct bch_fs *c, snapshot_id_list *s)
snapshot_id_list *s)
{ {
struct super_block *sb = c->vfs_sb; struct bch_inode_info *inode, **i;
struct inode *inode; DARRAY(struct bch_inode_info *) grabbed;
bool clean_pass = false, this_pass_clean;
spin_lock(&sb->s_inode_list_lock); /*
list_for_each_entry(inode, &sb->s_inodes, i_sb_list) { * Initially, we scan for inodes without I_DONTCACHE, then mark them to
if (!snapshot_list_has_id(s, to_bch_ei(inode)->ei_subvol) || * be pruned with d_mark_dontcache().
(inode->i_state & I_FREEING)) *
continue; * Once we've had a clean pass where we didn't find any inodes without
* I_DONTCACHE, we wait for them to be freed:
*/
d_mark_dontcache(inode); darray_init(&grabbed);
d_prune_aliases(inode); darray_make_room(&grabbed, 1024);
}
spin_unlock(&sb->s_inode_list_lock);
again: again:
cond_resched(); cond_resched();
spin_lock(&sb->s_inode_list_lock); this_pass_clean = true;
list_for_each_entry(inode, &sb->s_inodes, i_sb_list) {
if (!snapshot_list_has_id(s, to_bch_ei(inode)->ei_subvol) || mutex_lock(&c->vfs_inodes_lock);
(inode->i_state & I_FREEING)) list_for_each_entry(inode, &c->vfs_inodes_list, ei_vfs_inode_list) {
if (!snapshot_list_has_id(s, inode->ei_subvol))
continue; continue;
if (!(inode->i_state & I_DONTCACHE)) { if (!(inode->v.i_state & I_DONTCACHE) &&
d_mark_dontcache(inode); !(inode->v.i_state & I_FREEING)) {
d_prune_aliases(inode); this_pass_clean = false;
}
d_mark_dontcache(&inode->v);
d_prune_aliases(&inode->v);
/*
* If i_count was zero, we have to take and release a
* ref in order for I_DONTCACHE to be noticed and the
* inode to be dropped;
*/
if (!atomic_read(&inode->v.i_count) &&
igrab(&inode->v) &&
darray_push_gfp(&grabbed, inode, GFP_ATOMIC|__GFP_NOWARN))
break;
} else if (clean_pass && this_pass_clean) {
wait_queue_head_t *wq = bit_waitqueue(&inode->v.i_state, __I_NEW);
DEFINE_WAIT_BIT(wait, &inode->v.i_state, __I_NEW);
spin_lock(&inode->i_lock);
if (snapshot_list_has_id(s, to_bch_ei(inode)->ei_subvol) &&
!(inode->i_state & I_FREEING)) {
wait_queue_head_t *wq = bit_waitqueue(&inode->i_state, __I_NEW);
DEFINE_WAIT_BIT(wait, &inode->i_state, __I_NEW);
prepare_to_wait(wq, &wait.wq_entry, TASK_UNINTERRUPTIBLE); prepare_to_wait(wq, &wait.wq_entry, TASK_UNINTERRUPTIBLE);
spin_unlock(&inode->i_lock); mutex_unlock(&c->vfs_inodes_lock);
spin_unlock(&sb->s_inode_list_lock);
schedule(); schedule();
finish_wait(wq, &wait.wq_entry); finish_wait(wq, &wait.wq_entry);
goto again; goto again;
} }
}
mutex_unlock(&c->vfs_inodes_lock);
darray_for_each(grabbed, i)
iput(&(*i)->v);
grabbed.nr = 0;
spin_unlock(&inode->i_lock); if (!clean_pass || !this_pass_clean) {
clean_pass = this_pass_clean;
goto again;
} }
spin_unlock(&sb->s_inode_list_lock);
darray_exit(&grabbed);
} }
static int bch2_statfs(struct dentry *dentry, struct kstatfs *buf) static int bch2_statfs(struct dentry *dentry, struct kstatfs *buf)
......
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
struct bch_inode_info { struct bch_inode_info {
struct inode v; struct inode v;
struct list_head ei_vfs_inode_list;
unsigned long ei_flags; unsigned long ei_flags;
struct mutex ei_update_lock; struct mutex ei_update_lock;
......
...@@ -803,9 +803,6 @@ int bch2_inode_rm(struct bch_fs *c, subvol_inum inum) ...@@ -803,9 +803,6 @@ int bch2_inode_rm(struct bch_fs *c, subvol_inum inum)
bch2_inode_unpack(k, &inode_u); bch2_inode_unpack(k, &inode_u);
/* Subvolume root? */
BUG_ON(inode_u.bi_subvol);
bkey_inode_generation_init(&delete.k_i); bkey_inode_generation_init(&delete.k_i);
delete.k.p = iter.pos; delete.k.p = iter.pos;
delete.v.bi_generation = cpu_to_le32(inode_u.bi_generation + 1); delete.v.bi_generation = cpu_to_le32(inode_u.bi_generation + 1);
......
...@@ -709,6 +709,9 @@ static struct bch_fs *bch2_fs_alloc(struct bch_sb *sb, struct bch_opts opts) ...@@ -709,6 +709,9 @@ static struct bch_fs *bch2_fs_alloc(struct bch_sb *sb, struct bch_opts opts)
sema_init(&c->io_in_flight, 128); sema_init(&c->io_in_flight, 128);
INIT_LIST_HEAD(&c->vfs_inodes_list);
mutex_init(&c->vfs_inodes_lock);
c->copy_gc_enabled = 1; c->copy_gc_enabled = 1;
c->rebalance.enabled = 1; c->rebalance.enabled = 1;
c->promote_whole_extents = true; c->promote_whole_extents = true;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment