bcachefs: Run check_key_has_snapshot in snapshot_delete_keys()
authorKent Overstreet <kent.overstreet@linux.dev>
Sun, 26 May 2024 16:38:30 +0000 (12:38 -0400)
committerKent Overstreet <kent.overstreet@linux.dev>
Tue, 28 May 2024 15:29:26 +0000 (11:29 -0400)
delete_dead_snapshots now runs before the main fsck.c passes which check
for keys for invalid snapshots; thus, it needs those checks as well.

Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
fs/bcachefs/fsck.c
fs/bcachefs/snapshot.c
fs/bcachefs/snapshot.h

index 4cd28db9bad8deded3a314952905e6bd97989286..fd277bd58ed34afe3ad5a86f93210ea621756b62 100644 (file)
@@ -766,25 +766,6 @@ static int get_visible_inodes(struct btree_trans *trans,
        return ret;
 }
 
-static int check_key_has_snapshot(struct btree_trans *trans,
-                                 struct btree_iter *iter,
-                                 struct bkey_s_c k)
-{
-       struct bch_fs *c = trans->c;
-       struct printbuf buf = PRINTBUF;
-       int ret = 0;
-
-       if (mustfix_fsck_err_on(!bch2_snapshot_equiv(c, k.k->p.snapshot), c,
-                               bkey_in_missing_snapshot,
-                               "key in missing snapshot: %s",
-                               (bch2_bkey_val_to_text(&buf, c, k), buf.buf)))
-               ret = bch2_btree_delete_at(trans, iter,
-                                           BTREE_UPDATE_internal_snapshot_node) ?: 1;
-fsck_err:
-       printbuf_exit(&buf);
-       return ret;
-}
-
 static int hash_redo_key(struct btree_trans *trans,
                         const struct bch_hash_desc desc,
                         struct bch_hash_info *hash_info,
@@ -979,7 +960,7 @@ static int check_inode(struct btree_trans *trans,
        bool do_update = false;
        int ret;
 
-       ret = check_key_has_snapshot(trans, iter, k);
+       ret = bch2_check_key_has_snapshot(trans, iter, k);
        if (ret < 0)
                goto err;
        if (ret)
@@ -1483,7 +1464,7 @@ static int check_extent(struct btree_trans *trans, struct btree_iter *iter,
        struct printbuf buf = PRINTBUF;
        int ret = 0;
 
-       ret = check_key_has_snapshot(trans, iter, k);
+       ret = bch2_check_key_has_snapshot(trans, iter, k);
        if (ret) {
                ret = ret < 0 ? ret : 0;
                goto out;
@@ -2006,7 +1987,7 @@ static int check_dirent(struct btree_trans *trans, struct btree_iter *iter,
        struct printbuf buf = PRINTBUF;
        int ret = 0;
 
-       ret = check_key_has_snapshot(trans, iter, k);
+       ret = bch2_check_key_has_snapshot(trans, iter, k);
        if (ret) {
                ret = ret < 0 ? ret : 0;
                goto out;
@@ -2161,7 +2142,7 @@ static int check_xattr(struct btree_trans *trans, struct btree_iter *iter,
        struct inode_walker_entry *i;
        int ret;
 
-       ret = check_key_has_snapshot(trans, iter, k);
+       ret = bch2_check_key_has_snapshot(trans, iter, k);
        if (ret < 0)
                return ret;
        if (ret)
index 466fa3e6a4b656aae48fc1ce357b09aa1c08e7a6..51918acfd72681cc2249a0b123f1e020ecf6e3d5 100644 (file)
@@ -1042,6 +1042,25 @@ err:
        return ret;
 }
 
+int bch2_check_key_has_snapshot(struct btree_trans *trans,
+                               struct btree_iter *iter,
+                               struct bkey_s_c k)
+{
+       struct bch_fs *c = trans->c;
+       struct printbuf buf = PRINTBUF;
+       int ret = 0;
+
+       if (fsck_err_on(!bch2_snapshot_equiv(c, k.k->p.snapshot), c,
+                       bkey_in_missing_snapshot,
+                       "key in missing snapshot %s, delete?",
+                       (bch2_bkey_val_to_text(&buf, c, k), buf.buf)))
+               ret = bch2_btree_delete_at(trans, iter,
+                                           BTREE_UPDATE_internal_snapshot_node) ?: 1;
+fsck_err:
+       printbuf_exit(&buf);
+       return ret;
+}
+
 /*
  * Mark a snapshot as deleted, for future cleanup:
  */
@@ -1358,6 +1377,10 @@ static int delete_dead_snapshots_process_key(struct btree_trans *trans,
                               snapshot_id_list *equiv_seen,
                               struct bpos *last_pos)
 {
+       int ret = bch2_check_key_has_snapshot(trans, iter, k);
+       if (ret)
+               return ret < 0 ? ret : 0;
+
        struct bch_fs *c = trans->c;
        u32 equiv = bch2_snapshot_equiv(c, k.k->p.snapshot);
        if (!equiv) /* key for invalid snapshot node, but we chose not to delete */
@@ -1377,7 +1400,7 @@ static int delete_dead_snapshots_process_key(struct btree_trans *trans,
 
        *last_pos = k.k->p;
 
-       int ret = snapshot_list_add_nodup(c, equiv_seen, equiv);
+       ret = snapshot_list_add_nodup(c, equiv_seen, equiv);
        if (ret)
                return ret;
 
index ab13d8f4b41e1e934c1b5f5c21975358df128019..31b0ee03e96288e8dd525fab7f65e9eb56020990 100644 (file)
@@ -242,6 +242,7 @@ int bch2_snapshot_node_create(struct btree_trans *, u32,
 int bch2_check_snapshot_trees(struct bch_fs *);
 int bch2_check_snapshots(struct bch_fs *);
 int bch2_reconstruct_snapshots(struct bch_fs *);
+int bch2_check_key_has_snapshot(struct btree_trans *, struct btree_iter *, struct bkey_s_c);
 
 int bch2_snapshot_node_set_deleted(struct btree_trans *, u32);
 void bch2_delete_dead_snapshots_work(struct work_struct *);