summaryrefslogtreecommitdiff
path: root/fs/bcachefs/extent_update.c
diff options
context:
space:
mode:
authorKent Overstreet <kent.overstreet@gmail.com>2019-11-21 00:22:49 +0300
committerKent Overstreet <kent.overstreet@linux.dev>2023-10-23 00:08:32 +0300
commit183797e31d43ce2fbfc596ff3f4d034f1ba144d0 (patch)
tree7228c46c33b7b79fe90af6b46b91ceaadf1998d3 /fs/bcachefs/extent_update.c
parentc201e2d97615e7dc6f2f99dcdb8bf8d64657b761 (diff)
downloadlinux-183797e31d43ce2fbfc596ff3f4d034f1ba144d0.tar.xz
bcachefs: Always emit new extents on partial overwrite
This is prep work for snapshots: the algorithm in bch2_extent_sort_fix_overlapping() will break when we have multiple overlapping extents in unrelated snapshots - but, we'll be able to make extents work like regular keys and use bch2_key_sort_fix_overlapping() for extent btree nodes if we make a couple changes - the main one being to always emit new extents when we partially overwrite an existing (written) extent. Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com> Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
Diffstat (limited to 'fs/bcachefs/extent_update.c')
-rw-r--r--fs/bcachefs/extent_update.c293
1 files changed, 125 insertions, 168 deletions
diff --git a/fs/bcachefs/extent_update.c b/fs/bcachefs/extent_update.c
index 742b4d78cb3a..e021e1623a91 100644
--- a/fs/bcachefs/extent_update.c
+++ b/fs/bcachefs/extent_update.c
@@ -171,49 +171,51 @@ bch2_extent_can_insert(struct btree_trans *trans,
{
struct btree_iter_level *l = &insert->iter->l[0];
struct btree_node_iter node_iter = l->iter;
- enum bch_extent_overlap overlap;
struct bkey_packed *_k;
struct bkey unpacked;
- struct bkey_s_c k;
int sectors;
- /*
- * We avoid creating whiteouts whenever possible when deleting, but
- * those optimizations mean we may potentially insert two whiteouts
- * instead of one (when we overlap with the front of one extent and the
- * back of another):
- */
- if (bkey_whiteout(&insert->k->k))
- *u64s += BKEY_U64s;
-
- _k = bch2_btree_node_iter_peek_filter(&node_iter, l->b,
- KEY_TYPE_discard);
- if (!_k)
- return BTREE_INSERT_OK;
-
- k = bkey_disassemble(l->b, _k, &unpacked);
-
- overlap = bch2_extent_overlap(&insert->k->k, k.k);
-
- /* account for having to split existing extent: */
- if (overlap == BCH_EXTENT_OVERLAP_MIDDLE)
- *u64s += _k->u64s;
-
- if (overlap == BCH_EXTENT_OVERLAP_MIDDLE &&
- (sectors = bch2_bkey_sectors_compressed(k))) {
- int flags = trans->flags & BTREE_INSERT_NOFAIL
- ? BCH_DISK_RESERVATION_NOFAIL : 0;
-
- switch (bch2_disk_reservation_add(trans->c,
- trans->disk_res,
- sectors, flags)) {
- case 0:
+ while ((_k = bch2_btree_node_iter_peek_filter(&node_iter, l->b,
+ KEY_TYPE_discard))) {
+ struct bkey_s_c k = bkey_disassemble(l->b, _k, &unpacked);
+ enum bch_extent_overlap overlap =
+ bch2_extent_overlap(&insert->k->k, k.k);
+
+ if (bkey_cmp(bkey_start_pos(k.k), insert->k->k.p) >= 0)
break;
- case -ENOSPC:
- return BTREE_INSERT_ENOSPC;
- default:
- BUG();
+
+ overlap = bch2_extent_overlap(&insert->k->k, k.k);
+
+ if (bkey_written(l->b, _k) &&
+ overlap != BCH_EXTENT_OVERLAP_ALL)
+ *u64s += _k->u64s;
+
+ /* account for having to split existing extent: */
+ if (overlap == BCH_EXTENT_OVERLAP_MIDDLE)
+ *u64s += _k->u64s;
+
+ if (overlap == BCH_EXTENT_OVERLAP_MIDDLE &&
+ (sectors = bch2_bkey_sectors_compressed(k))) {
+ int flags = trans->flags & BTREE_INSERT_NOFAIL
+ ? BCH_DISK_RESERVATION_NOFAIL : 0;
+
+ switch (bch2_disk_reservation_add(trans->c,
+ trans->disk_res,
+ sectors, flags)) {
+ case 0:
+ break;
+ case -ENOSPC:
+ return BTREE_INSERT_ENOSPC;
+ default:
+ BUG();
+ }
}
+
+ if (overlap == BCH_EXTENT_OVERLAP_FRONT ||
+ overlap == BCH_EXTENT_OVERLAP_MIDDLE)
+ break;
+
+ bch2_btree_node_iter_advance(&node_iter, l->b);
}
return BTREE_INSERT_OK;
@@ -285,101 +287,106 @@ static void extent_bset_insert(struct bch_fs *c, struct btree_iter *iter,
}
static void
-extent_squash(struct bch_fs *c, struct btree_iter *iter,
- struct bkey_i *insert,
- struct bkey_packed *_k, struct bkey_s k,
- enum bch_extent_overlap overlap)
+extent_drop(struct bch_fs *c, struct btree_iter *iter,
+ struct bkey_packed *_k, struct bkey_s k)
{
struct btree_iter_level *l = &iter->l[0];
- int u64s_delta;
- switch (overlap) {
- case BCH_EXTENT_OVERLAP_FRONT:
- /* insert overlaps with start of k: */
- u64s_delta = bch2_cut_front_s(insert->k.p, k);
- btree_keys_account_val_delta(l->b, _k, u64s_delta);
+ if (!bkey_whiteout(k.k))
+ btree_account_key_drop(l->b, _k);
- EBUG_ON(bkey_deleted(k.k));
- extent_save(l->b, _k, k.k);
- bch2_btree_iter_fix_key_modified(iter, l->b, _k);
- break;
+ k.k->size = 0;
+ k.k->type = KEY_TYPE_deleted;
+ k.k->needs_whiteout = false;
- case BCH_EXTENT_OVERLAP_BACK:
- /* insert overlaps with end of k: */
- u64s_delta = bch2_cut_back_s(bkey_start_pos(&insert->k), k);
- btree_keys_account_val_delta(l->b, _k, u64s_delta);
+ if (_k >= btree_bset_last(l->b)->start) {
+ unsigned u64s = _k->u64s;
- EBUG_ON(bkey_deleted(k.k));
+ bch2_bset_delete(l->b, _k, _k->u64s);
+ bch2_btree_node_iter_fix(iter, l->b, &l->iter, _k, u64s, 0);
+ } else {
extent_save(l->b, _k, k.k);
+ bch2_btree_iter_fix_key_modified(iter, l->b, _k);
+ }
+}
- /*
- * As the auxiliary tree is indexed by the end of the
- * key and we've just changed the end, update the
- * auxiliary tree.
- */
- bch2_bset_fix_invalidated_key(l->b, _k);
- bch2_btree_node_iter_fix(iter, l->b, &l->iter,
- _k, _k->u64s, _k->u64s);
- break;
-
- case BCH_EXTENT_OVERLAP_ALL: {
- /* The insert key completely covers k, invalidate k */
- if (!bkey_whiteout(k.k))
- btree_account_key_drop(l->b, _k);
+static void
+extent_squash(struct bch_fs *c, struct btree_iter *iter,
+ struct bkey_i *insert,
+ struct bkey_packed *_k, struct bkey_s k,
+ enum bch_extent_overlap overlap)
+{
+ struct btree_iter_level *l = &iter->l[0];
+ struct bkey_on_stack tmp, split;
- k.k->size = 0;
- k.k->type = KEY_TYPE_deleted;
+ bkey_on_stack_init(&tmp);
+ bkey_on_stack_init(&split);
- if (_k >= btree_bset_last(l->b)->start) {
- unsigned u64s = _k->u64s;
+ switch (overlap) {
+ case BCH_EXTENT_OVERLAP_FRONT:
+ if (bkey_written(l->b, _k)) {
+ bkey_on_stack_reassemble(&tmp, c, k.s_c);
+ bch2_cut_front(insert->k.p, tmp.k);
- bch2_bset_delete(l->b, _k, _k->u64s);
- bch2_btree_node_iter_fix(iter, l->b, &l->iter,
- _k, u64s, 0);
+ extent_drop(c, iter, _k, k);
+ extent_bset_insert(c, iter, tmp.k);
} else {
+ btree_keys_account_val_delta(l->b, _k,
+ bch2_cut_front_s(insert->k.p, k));
+
extent_save(l->b, _k, k.k);
+ /*
+ * No need to call bset_fix_invalidated_key, start of
+ * extent changed but extents are indexed by where they
+ * end
+ */
bch2_btree_iter_fix_key_modified(iter, l->b, _k);
}
-
break;
- }
- case BCH_EXTENT_OVERLAP_MIDDLE: {
- struct bkey_on_stack split;
-
- bkey_on_stack_init(&split);
- bkey_on_stack_reassemble(&split, c, k.s_c);
+ case BCH_EXTENT_OVERLAP_BACK:
+ if (bkey_written(l->b, _k)) {
+ bkey_on_stack_reassemble(&tmp, c, k.s_c);
+ bch2_cut_back(bkey_start_pos(&insert->k), tmp.k);
- /*
- * The insert key falls 'in the middle' of k
- * The insert key splits k in 3:
- * - start only in k, preserve
- * - middle common section, invalidate in k
- * - end only in k, preserve
- *
- * We update the old key to preserve the start,
- * insert will be the new common section,
- * we manually insert the end that we are preserving.
- *
- * modify k _before_ doing the insert (which will move
- * what k points to)
- */
- split.k->k.needs_whiteout |= bkey_written(l->b, _k);
+ extent_drop(c, iter, _k, k);
+ extent_bset_insert(c, iter, tmp.k);
+ } else {
+ btree_keys_account_val_delta(l->b, _k,
+ bch2_cut_back_s(bkey_start_pos(&insert->k), k));
+ extent_save(l->b, _k, k.k);
+ bch2_bset_fix_invalidated_key(l->b, _k);
+ bch2_btree_node_iter_fix(iter, l->b, &l->iter,
+ _k, _k->u64s, _k->u64s);
+ }
+ break;
+ case BCH_EXTENT_OVERLAP_ALL:
+ extent_drop(c, iter, _k, k);
+ break;
+ case BCH_EXTENT_OVERLAP_MIDDLE:
+ bkey_on_stack_reassemble(&split, c, k.s_c);
bch2_cut_back(bkey_start_pos(&insert->k), split.k);
- BUG_ON(bkey_deleted(&split.k->k));
- u64s_delta = bch2_cut_front_s(insert->k.p, k);
- btree_keys_account_val_delta(l->b, _k, u64s_delta);
+ if (bkey_written(l->b, _k)) {
+ bkey_on_stack_reassemble(&tmp, c, k.s_c);
+ bch2_cut_front(insert->k.p, tmp.k);
- BUG_ON(bkey_deleted(k.k));
- extent_save(l->b, _k, k.k);
- bch2_btree_iter_fix_key_modified(iter, l->b, _k);
+ extent_drop(c, iter, _k, k);
+ extent_bset_insert(c, iter, tmp.k);
+ } else {
+ btree_keys_account_val_delta(l->b, _k,
+ bch2_cut_front_s(insert->k.p, k));
+
+ extent_save(l->b, _k, k.k);
+ bch2_btree_iter_fix_key_modified(iter, l->b, _k);
+ }
extent_bset_insert(c, iter, split.k);
- bkey_on_stack_exit(&split, c);
break;
}
- }
+
+ bkey_on_stack_exit(&split, c);
+ bkey_on_stack_exit(&tmp, c);
}
/**
@@ -429,10 +436,7 @@ void bch2_insert_fixup_extent(struct btree_trans *trans,
struct bkey_i *insert = insert_entry->k;
struct btree_iter_level *l = &iter->l[0];
struct btree_node_iter node_iter = l->iter;
- bool deleting = bkey_whiteout(&insert->k);
- bool update_journal = !deleting;
- bool update_btree = !deleting;
- struct bkey_i whiteout = *insert;
+ bool do_update = !bkey_whiteout(&insert->k);
struct bkey_packed *_k;
struct bkey unpacked;
@@ -443,7 +447,6 @@ void bch2_insert_fixup_extent(struct btree_trans *trans,
while ((_k = bch2_btree_node_iter_peek_filter(&l->iter, l->b,
KEY_TYPE_discard))) {
struct bkey_s k = __bkey_disassemble(l->b, _k, &unpacked);
- struct bpos cur_end = bpos_min(insert->k.p, k.k->p);
enum bch_extent_overlap overlap =
bch2_extent_overlap(&insert->k, k.k);
@@ -451,52 +454,18 @@ void bch2_insert_fixup_extent(struct btree_trans *trans,
break;
if (!bkey_whiteout(k.k))
- update_journal = true;
+ do_update = true;
+
+ if (!do_update) {
+ struct bpos cur_end = bpos_min(insert->k.p, k.k->p);
- if (!update_journal) {
bch2_cut_front(cur_end, insert);
- bch2_cut_front(cur_end, &whiteout);
bch2_btree_iter_set_pos_same_leaf(iter, cur_end);
- goto next;
- }
-
- /*
- * When deleting, if possible just do it by switching the type
- * of the key we're deleting, instead of creating and inserting
- * a new whiteout:
- */
- if (deleting &&
- !update_btree &&
- !bkey_cmp(insert->k.p, k.k->p) &&
- !bkey_cmp(bkey_start_pos(&insert->k), bkey_start_pos(k.k))) {
- if (!bkey_whiteout(k.k)) {
- btree_account_key_drop(l->b, _k);
- _k->type = KEY_TYPE_discard;
- reserve_whiteout(l->b, _k);
- bch2_btree_iter_fix_key_modified(iter,
- l->b, _k);
- }
- break;
- }
-
- if (k.k->needs_whiteout || bkey_written(l->b, _k)) {
- insert->k.needs_whiteout = true;
- update_btree = true;
- }
-
- if (update_btree &&
- overlap == BCH_EXTENT_OVERLAP_ALL &&
- bkey_whiteout(k.k) &&
- k.k->needs_whiteout) {
- unreserve_whiteout(l->b, _k);
- _k->needs_whiteout = false;
+ } else {
+ insert->k.needs_whiteout |= k.k->needs_whiteout;
+ extent_squash(c, iter, insert, _k, k, overlap);
}
- extent_squash(c, iter, insert, _k, k, overlap);
-
- if (!update_btree)
- bch2_cut_front(cur_end, insert);
-next:
node_iter = l->iter;
if (overlap == BCH_EXTENT_OVERLAP_FRONT ||
@@ -507,24 +476,12 @@ next:
l->iter = node_iter;
bch2_btree_iter_set_pos_same_leaf(iter, insert->k.p);
- if (update_btree) {
- if (deleting)
+ if (do_update) {
+ if (insert->k.type == KEY_TYPE_deleted)
insert->k.type = KEY_TYPE_discard;
- EBUG_ON(bkey_deleted(&insert->k) || !insert->k.size);
-
extent_bset_insert(c, iter, insert);
- }
-
- if (update_journal) {
- struct bkey_i *k = !deleting ? insert : &whiteout;
-
- if (deleting)
- k->k.type = KEY_TYPE_discard;
-
- EBUG_ON(bkey_deleted(&k->k) || !k->k.size);
-
- bch2_btree_journal_key(trans, iter, k);
+ bch2_btree_journal_key(trans, iter, insert);
}
bch2_cut_front(insert->k.p, insert);