Commit 14d7d61f authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Fix btree_gc when multiple passes required

We weren't resetting filesystem & device usage when restarting gc, which
was spotted when free bucket counters overflowed - whoops.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 1ae40fd8
...@@ -1282,8 +1282,7 @@ static int bch2_gc_done(struct bch_fs *c, ...@@ -1282,8 +1282,7 @@ static int bch2_gc_done(struct bch_fs *c,
return ret; return ret;
} }
static int bch2_gc_start(struct bch_fs *c, static int bch2_gc_start(struct bch_fs *c)
bool metadata_only)
{ {
struct bch_dev *ca = NULL; struct bch_dev *ca = NULL;
unsigned i; unsigned i;
...@@ -1298,7 +1297,6 @@ static int bch2_gc_start(struct bch_fs *c, ...@@ -1298,7 +1297,6 @@ static int bch2_gc_start(struct bch_fs *c,
} }
for_each_member_device(ca, c, i) { for_each_member_device(ca, c, i) {
BUG_ON(ca->buckets_gc);
BUG_ON(ca->usage_gc); BUG_ON(ca->usage_gc);
ca->usage_gc = alloc_percpu(struct bch_dev_usage); ca->usage_gc = alloc_percpu(struct bch_dev_usage);
...@@ -1315,6 +1313,22 @@ static int bch2_gc_start(struct bch_fs *c, ...@@ -1315,6 +1313,22 @@ static int bch2_gc_start(struct bch_fs *c,
return 0; return 0;
} }
static int bch2_gc_reset(struct bch_fs *c)
{
struct bch_dev *ca;
unsigned i;
for_each_member_device(ca, c, i) {
free_percpu(ca->usage_gc);
ca->usage_gc = NULL;
}
free_percpu(c->usage_gc);
c->usage_gc = NULL;
return bch2_gc_start(c);
}
/* returns true if not equal */ /* returns true if not equal */
static inline bool bch2_alloc_v4_cmp(struct bch_alloc_v4 l, static inline bool bch2_alloc_v4_cmp(struct bch_alloc_v4 l,
struct bch_alloc_v4 r) struct bch_alloc_v4 r)
...@@ -1761,7 +1775,7 @@ int bch2_gc(struct bch_fs *c, bool initial, bool metadata_only) ...@@ -1761,7 +1775,7 @@ int bch2_gc(struct bch_fs *c, bool initial, bool metadata_only)
bch2_btree_interior_updates_flush(c); bch2_btree_interior_updates_flush(c);
ret = bch2_gc_start(c, metadata_only) ?: ret = bch2_gc_start(c) ?:
bch2_gc_alloc_start(c, metadata_only) ?: bch2_gc_alloc_start(c, metadata_only) ?:
bch2_gc_reflink_start(c, metadata_only); bch2_gc_reflink_start(c, metadata_only);
if (ret) if (ret)
...@@ -1822,6 +1836,9 @@ int bch2_gc(struct bch_fs *c, bool initial, bool metadata_only) ...@@ -1822,6 +1836,9 @@ int bch2_gc(struct bch_fs *c, bool initial, bool metadata_only)
bch2_gc_stripes_reset(c, metadata_only); bch2_gc_stripes_reset(c, metadata_only);
bch2_gc_alloc_reset(c, metadata_only); bch2_gc_alloc_reset(c, metadata_only);
bch2_gc_reflink_reset(c, metadata_only); bch2_gc_reflink_reset(c, metadata_only);
ret = bch2_gc_reset(c);
if (ret)
goto out;
/* flush fsck errors, reset counters */ /* flush fsck errors, reset counters */
bch2_flush_fsck_errs(c); bch2_flush_fsck_errs(c);
......
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