diff options
Diffstat (limited to 'libbcachefs/btree_gc.h')
-rw-r--r-- | libbcachefs/btree_gc.h | 87 |
1 files changed, 31 insertions, 56 deletions
diff --git a/libbcachefs/btree_gc.h b/libbcachefs/btree_gc.h index 214a3fe3..8a47e8bd 100644 --- a/libbcachefs/btree_gc.h +++ b/libbcachefs/btree_gc.h @@ -1,19 +1,13 @@ +/* SPDX-License-Identifier: GPL-2.0 */ #ifndef _BCACHEFS_BTREE_GC_H #define _BCACHEFS_BTREE_GC_H +#include "bkey.h" +#include "btree_gc_types.h" #include "btree_types.h" -enum bkey_type; - -void bch2_coalesce(struct bch_fs *); -void bch2_gc(struct bch_fs *); -void bch2_gc_thread_stop(struct bch_fs *); -int bch2_gc_thread_start(struct bch_fs *); -int bch2_initial_gc(struct bch_fs *, struct list_head *); -u8 bch2_btree_key_recalc_oldest_gen(struct bch_fs *, struct bkey_s_c); -int bch2_btree_mark_key_initial(struct bch_fs *, enum bkey_type, - struct bkey_s_c); -void bch2_mark_dev_superblock(struct bch_fs *, struct bch_dev *, unsigned); +int bch2_check_topology(struct bch_fs *); +int bch2_check_allocations(struct bch_fs *); /* * For concurrent mark and sweep (with other index updates), we define a total @@ -39,74 +33,55 @@ void bch2_mark_dev_superblock(struct bch_fs *, struct bch_dev *, unsigned); /* Position of (the start of) a gc phase: */ static inline struct gc_pos gc_phase(enum gc_phase phase) { - return (struct gc_pos) { - .phase = phase, - .pos = POS_MIN, - .level = 0, - }; + return (struct gc_pos) { .phase = phase, }; } -static inline int gc_pos_cmp(struct gc_pos l, struct gc_pos r) -{ - if (l.phase != r.phase) - return l.phase < r.phase ? -1 : 1; - if (bkey_cmp(l.pos, r.pos)) - return bkey_cmp(l.pos, r.pos); - if (l.level != r.level) - return l.level < r.level ? -1 : 1; - return 0; -} - -static inline struct gc_pos gc_pos_btree(enum btree_id id, - struct bpos pos, unsigned level) +static inline struct gc_pos gc_pos_btree(enum btree_id btree, unsigned level, + struct bpos pos) { return (struct gc_pos) { - .phase = GC_PHASE_BTREE_EXTENTS + id, - .pos = pos, + .phase = GC_PHASE_btree, + .btree = btree, .level = level, + .pos = pos, }; } -/* - * GC position of the pointers within a btree node: note, _not_ for &b->key - * itself, that lives in the parent node: - */ -static inline struct gc_pos gc_pos_btree_node(struct btree *b) +static inline int gc_btree_order(enum btree_id btree) { - return gc_pos_btree(b->btree_id, b->key.k.p, b->level); + if (btree == BTREE_ID_alloc) + return -2; + if (btree == BTREE_ID_stripes) + return -1; + return btree; } -/* - * GC position of the pointer to a btree root: we don't use - * gc_pos_pointer_to_btree_node() here to avoid a potential race with - * btree_split() increasing the tree depth - the new root will have level > the - * old root and thus have a greater gc position than the old root, but that - * would be incorrect since once gc has marked the root it's not coming back. - */ -static inline struct gc_pos gc_pos_btree_root(enum btree_id id) -{ - return gc_pos_btree(id, POS_MAX, BTREE_MAX_DEPTH); -} - -static inline struct gc_pos gc_pos_alloc(struct bch_fs *c, struct open_bucket *ob) +static inline int gc_pos_cmp(struct gc_pos l, struct gc_pos r) { - return (struct gc_pos) { - .phase = GC_PHASE_ALLOC, - .pos = POS(ob ? ob - c->open_buckets : 0, 0), - }; + return cmp_int(l.phase, r.phase) ?: + cmp_int(gc_btree_order(l.btree), + gc_btree_order(r.btree)) ?: + cmp_int(l.level, r.level) ?: + bpos_cmp(l.pos, r.pos); } -static inline bool gc_will_visit(struct bch_fs *c, struct gc_pos pos) +static inline bool gc_visited(struct bch_fs *c, struct gc_pos pos) { unsigned seq; bool ret; do { seq = read_seqcount_begin(&c->gc_pos_lock); - ret = gc_pos_cmp(c->gc_pos, pos) < 0; + ret = gc_pos_cmp(pos, c->gc_pos) <= 0; } while (read_seqcount_retry(&c->gc_pos_lock, seq)); return ret; } +void bch2_gc_pos_to_text(struct printbuf *, struct gc_pos *); + +int bch2_gc_gens(struct bch_fs *); +void bch2_gc_gens_async(struct bch_fs *); +void bch2_fs_gc_init(struct bch_fs *); + #endif /* _BCACHEFS_BTREE_GC_H */ |