2017-03-16 22:18:50 -08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
#include "bcachefs.h"
|
2020-12-17 15:08:58 -05:00
|
|
|
#include "bkey_buf.h"
|
2018-10-06 00:46:55 -04:00
|
|
|
#include "alloc_background.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "btree_gc.h"
|
|
|
|
#include "btree_update.h"
|
|
|
|
#include "btree_update_interior.h"
|
|
|
|
#include "btree_io.h"
|
2019-01-24 20:25:40 -05:00
|
|
|
#include "buckets.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "dirent.h"
|
2018-11-01 15:13:19 -04:00
|
|
|
#include "ec.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "error.h"
|
2019-10-02 18:35:36 -04:00
|
|
|
#include "fs-common.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "fsck.h"
|
|
|
|
#include "journal_io.h"
|
2019-04-11 22:39:39 -04:00
|
|
|
#include "journal_reclaim.h"
|
2019-04-04 21:53:12 -04:00
|
|
|
#include "journal_seq_blacklist.h"
|
2022-02-17 03:11:39 -05:00
|
|
|
#include "lru.h"
|
2021-03-22 18:39:16 -04:00
|
|
|
#include "move.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "quota.h"
|
|
|
|
#include "recovery.h"
|
2019-01-24 19:09:49 -05:00
|
|
|
#include "replicas.h"
|
2021-03-16 00:42:25 -04:00
|
|
|
#include "subvolume.h"
|
2017-03-16 22:18:50 -08:00
|
|
|
#include "super-io.h"
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
#include <linux/sort.h>
|
2017-03-16 22:18:50 -08:00
|
|
|
#include <linux/stat.h>
|
|
|
|
|
|
|
|
#define QSTR(n) { { { .len = strlen(n) } }, .name = n }
|
|
|
|
|
2020-10-24 21:20:16 -04:00
|
|
|
/* for -o reconstruct_alloc: */
|
|
|
|
static void drop_alloc_keys(struct journal_keys *keys)
|
|
|
|
{
|
|
|
|
size_t src, dst;
|
|
|
|
|
|
|
|
for (src = 0, dst = 0; src < keys->nr; src++)
|
2021-02-20 19:27:37 -05:00
|
|
|
if (keys->d[src].btree_id != BTREE_ID_alloc)
|
2020-10-24 21:20:16 -04:00
|
|
|
keys->d[dst++] = keys->d[src];
|
|
|
|
|
|
|
|
keys->nr = dst;
|
|
|
|
}
|
|
|
|
|
2021-07-30 14:33:06 -04:00
|
|
|
/*
|
|
|
|
* Btree node pointers have a field to stack a pointer to the in memory btree
|
|
|
|
* node; we need to zero out this field when reading in btree nodes, or when
|
|
|
|
* reading in keys from the journal:
|
|
|
|
*/
|
|
|
|
static void zero_out_btree_mem_ptr(struct journal_keys *keys)
|
|
|
|
{
|
|
|
|
struct journal_key *i;
|
|
|
|
|
|
|
|
for (i = keys->d; i < keys->d + keys->nr; i++)
|
|
|
|
if (i->k->k.type == KEY_TYPE_btree_ptr_v2)
|
|
|
|
bkey_i_to_btree_ptr_v2(i->k)->v.mem_ptr = 0;
|
|
|
|
}
|
|
|
|
|
2019-07-12 17:08:32 -04:00
|
|
|
/* iterate over keys read from the journal: */
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
static int __journal_key_cmp(enum btree_id l_btree_id,
|
|
|
|
unsigned l_level,
|
|
|
|
struct bpos l_pos,
|
2021-12-25 20:07:00 -05:00
|
|
|
const struct journal_key *r)
|
2021-01-26 20:15:46 -05:00
|
|
|
{
|
|
|
|
return (cmp_int(l_btree_id, r->btree_id) ?:
|
|
|
|
cmp_int(l_level, r->level) ?:
|
2021-03-04 16:20:16 -05:00
|
|
|
bpos_cmp(l_pos, r->k->k.p));
|
2021-01-26 20:15:46 -05:00
|
|
|
}
|
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
static int journal_key_cmp(const struct journal_key *l, const struct journal_key *r)
|
2021-01-26 20:15:46 -05:00
|
|
|
{
|
2021-12-25 20:07:00 -05:00
|
|
|
return __journal_key_cmp(l->btree_id, l->level, l->k->k.p, r);
|
2021-01-26 20:15:46 -05:00
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
static inline size_t idx_to_pos(struct journal_keys *keys, size_t idx)
|
|
|
|
{
|
|
|
|
size_t gap_size = keys->size - keys->nr;
|
|
|
|
|
|
|
|
if (idx >= keys->gap)
|
|
|
|
idx += gap_size;
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct journal_key *idx_to_key(struct journal_keys *keys, size_t idx)
|
|
|
|
{
|
|
|
|
return keys->d + idx_to_pos(keys, idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t bch2_journal_key_search(struct journal_keys *keys,
|
2021-12-25 20:07:00 -05:00
|
|
|
enum btree_id id, unsigned level,
|
|
|
|
struct bpos pos)
|
2019-07-12 17:08:32 -04:00
|
|
|
{
|
2022-04-04 01:09:26 -04:00
|
|
|
size_t l = 0, r = keys->nr, m;
|
2019-07-12 17:08:32 -04:00
|
|
|
|
2020-03-15 23:29:43 -04:00
|
|
|
while (l < r) {
|
|
|
|
m = l + ((r - l) >> 1);
|
2022-04-04 01:09:26 -04:00
|
|
|
if (__journal_key_cmp(id, level, pos, idx_to_key(keys, m)) > 0)
|
2020-03-15 23:29:43 -04:00
|
|
|
l = m + 1;
|
|
|
|
else
|
|
|
|
r = m;
|
2019-07-12 17:08:32 -04:00
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
BUG_ON(l < keys->nr &&
|
|
|
|
__journal_key_cmp(id, level, pos, idx_to_key(keys, l)) > 0);
|
2020-03-15 23:29:43 -04:00
|
|
|
|
|
|
|
BUG_ON(l &&
|
2022-04-04 01:09:26 -04:00
|
|
|
__journal_key_cmp(id, level, pos, idx_to_key(keys, l - 1)) <= 0);
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
return idx_to_pos(keys, l);
|
2021-01-26 20:15:46 -05:00
|
|
|
}
|
|
|
|
|
2022-02-24 08:08:53 -05:00
|
|
|
struct bkey_i *bch2_journal_keys_peek(struct bch_fs *c, enum btree_id btree_id,
|
|
|
|
unsigned level, struct bpos pos)
|
|
|
|
{
|
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
2022-04-04 01:09:26 -04:00
|
|
|
size_t idx = bch2_journal_key_search(keys, btree_id, level, pos);
|
2022-02-24 08:08:53 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
while (idx < keys->size &&
|
|
|
|
keys->d[idx].overwritten) {
|
|
|
|
idx++;
|
|
|
|
if (idx == keys->gap)
|
|
|
|
idx += keys->size - keys->nr;
|
|
|
|
}
|
2022-02-24 08:08:53 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
if (idx < keys->size &&
|
|
|
|
keys->d[idx].btree_id == btree_id &&
|
|
|
|
keys->d[idx].level == level)
|
|
|
|
return keys->d[idx].k;
|
2022-02-24 08:08:53 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
static void journal_iters_fix(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
|
|
|
/* The key we just inserted is immediately before the gap: */
|
|
|
|
struct journal_key *n = &keys->d[keys->gap - 1];
|
|
|
|
size_t gap_end = keys->gap + (keys->size - keys->nr);
|
|
|
|
struct btree_and_journal_iter *iter;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If an iterator points one after the key we just inserted,
|
|
|
|
* and the key we just inserted compares >= the iterator's position,
|
|
|
|
* decrement the iterator so it points at the key we just inserted:
|
|
|
|
*/
|
|
|
|
list_for_each_entry(iter, &c->journal_iters, journal.list)
|
|
|
|
if (iter->journal.idx == gap_end &&
|
|
|
|
iter->last &&
|
|
|
|
iter->b->c.btree_id == n->btree_id &&
|
|
|
|
iter->b->c.level == n->level &&
|
|
|
|
bpos_cmp(n->k->k.p, iter->unpacked.p) >= 0)
|
|
|
|
iter->journal.idx = keys->gap - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void journal_iters_move_gap(struct bch_fs *c, size_t old_gap, size_t new_gap)
|
2021-01-26 20:15:46 -05:00
|
|
|
{
|
2022-04-04 01:09:26 -04:00
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
|
|
|
struct journal_iter *iter;
|
|
|
|
size_t gap_size = keys->size - keys->nr;
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
list_for_each_entry(iter, &c->journal_iters, list) {
|
|
|
|
if (iter->idx > old_gap)
|
|
|
|
iter->idx -= gap_size;
|
|
|
|
if (iter->idx >= new_gap)
|
|
|
|
iter->idx += gap_size;
|
|
|
|
}
|
2021-01-26 20:15:46 -05:00
|
|
|
}
|
|
|
|
|
2021-12-23 21:35:28 -05:00
|
|
|
int bch2_journal_key_insert_take(struct bch_fs *c, enum btree_id id,
|
|
|
|
unsigned level, struct bkey_i *k)
|
2021-01-26 20:15:46 -05:00
|
|
|
{
|
|
|
|
struct journal_key n = {
|
|
|
|
.btree_id = id,
|
|
|
|
.level = level,
|
|
|
|
.k = k,
|
2022-01-01 20:45:30 -05:00
|
|
|
.allocated = true,
|
|
|
|
/*
|
|
|
|
* Ensure these keys are done last by journal replay, to unblock
|
|
|
|
* journal reclaim:
|
|
|
|
*/
|
|
|
|
.journal_seq = U32_MAX,
|
2021-01-26 20:15:46 -05:00
|
|
|
};
|
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
2021-12-25 20:07:00 -05:00
|
|
|
size_t idx = bch2_journal_key_search(keys, id, level, k->k.p);
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-01-01 20:45:30 -05:00
|
|
|
BUG_ON(test_bit(BCH_FS_RW, &c->flags));
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
if (idx < keys->size &&
|
2021-01-26 20:15:46 -05:00
|
|
|
journal_key_cmp(&n, &keys->d[idx]) == 0) {
|
|
|
|
if (keys->d[idx].allocated)
|
|
|
|
kfree(keys->d[idx].k);
|
|
|
|
keys->d[idx] = n;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
if (idx > keys->gap)
|
|
|
|
idx -= keys->size - keys->nr;
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
if (keys->nr == keys->size) {
|
|
|
|
struct journal_keys new_keys = {
|
|
|
|
.nr = keys->nr,
|
2022-03-21 00:15:53 -04:00
|
|
|
.size = max(keys->size, 8UL) * 2,
|
2021-01-26 20:15:46 -05:00
|
|
|
.journal_seq_base = keys->journal_seq_base,
|
|
|
|
};
|
|
|
|
|
|
|
|
new_keys.d = kvmalloc(sizeof(new_keys.d[0]) * new_keys.size, GFP_KERNEL);
|
2021-02-23 15:16:41 -05:00
|
|
|
if (!new_keys.d) {
|
|
|
|
bch_err(c, "%s: error allocating new key array (size %zu)",
|
|
|
|
__func__, new_keys.size);
|
2021-01-26 20:15:46 -05:00
|
|
|
return -ENOMEM;
|
2021-02-23 15:16:41 -05:00
|
|
|
}
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
/* Since @keys was full, there was no gap: */
|
2021-01-26 20:15:46 -05:00
|
|
|
memcpy(new_keys.d, keys->d, sizeof(keys->d[0]) * keys->nr);
|
|
|
|
kvfree(keys->d);
|
|
|
|
*keys = new_keys;
|
2022-04-04 01:09:26 -04:00
|
|
|
|
|
|
|
/* And now the gap is at the end: */
|
|
|
|
keys->gap = keys->nr;
|
2021-01-26 20:15:46 -05:00
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
journal_iters_move_gap(c, keys->gap, idx);
|
|
|
|
|
|
|
|
move_gap(keys->d, keys->nr, keys->size, keys->gap, idx);
|
|
|
|
keys->gap = idx;
|
|
|
|
|
|
|
|
keys->nr++;
|
|
|
|
keys->d[keys->gap++] = n;
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
journal_iters_fix(c);
|
2021-01-26 20:15:46 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
/*
|
|
|
|
* Can only be used from the recovery thread while we're still RO - can't be
|
|
|
|
* used once we've got RW, as journal_keys is at that point used by multiple
|
|
|
|
* threads:
|
|
|
|
*/
|
2021-12-23 21:35:28 -05:00
|
|
|
int bch2_journal_key_insert(struct bch_fs *c, enum btree_id id,
|
|
|
|
unsigned level, struct bkey_i *k)
|
2021-01-26 20:15:46 -05:00
|
|
|
{
|
2021-12-23 21:35:28 -05:00
|
|
|
struct bkey_i *n;
|
2021-01-26 20:15:46 -05:00
|
|
|
int ret;
|
|
|
|
|
2021-12-23 21:35:28 -05:00
|
|
|
n = kmalloc(bkey_bytes(&k->k), GFP_KERNEL);
|
|
|
|
if (!n)
|
2021-01-26 20:15:46 -05:00
|
|
|
return -ENOMEM;
|
2020-03-15 23:29:43 -04:00
|
|
|
|
2021-12-23 21:35:28 -05:00
|
|
|
bkey_copy(n, k);
|
|
|
|
ret = bch2_journal_key_insert_take(c, id, level, n);
|
2021-01-26 20:15:46 -05:00
|
|
|
if (ret)
|
2021-12-23 21:35:28 -05:00
|
|
|
kfree(n);
|
2021-01-26 20:15:46 -05:00
|
|
|
return ret;
|
2020-03-15 23:29:43 -04:00
|
|
|
}
|
|
|
|
|
2021-12-23 21:35:28 -05:00
|
|
|
int bch2_journal_key_delete(struct bch_fs *c, enum btree_id id,
|
|
|
|
unsigned level, struct bpos pos)
|
|
|
|
{
|
|
|
|
struct bkey_i whiteout;
|
|
|
|
|
|
|
|
bkey_init(&whiteout.k);
|
|
|
|
whiteout.k.p = pos;
|
|
|
|
|
|
|
|
return bch2_journal_key_insert(c, id, level, &whiteout);
|
|
|
|
}
|
|
|
|
|
2021-12-31 17:54:13 -05:00
|
|
|
void bch2_journal_key_overwritten(struct bch_fs *c, enum btree_id btree,
|
|
|
|
unsigned level, struct bpos pos)
|
|
|
|
{
|
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
2021-12-25 20:07:00 -05:00
|
|
|
size_t idx = bch2_journal_key_search(keys, btree, level, pos);
|
2021-12-31 17:54:13 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
if (idx < keys->size &&
|
2021-12-31 17:54:13 -05:00
|
|
|
keys->d[idx].btree_id == btree &&
|
|
|
|
keys->d[idx].level == level &&
|
|
|
|
!bpos_cmp(keys->d[idx].k->k.p, pos))
|
|
|
|
keys->d[idx].overwritten = true;
|
|
|
|
}
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
static void bch2_journal_iter_advance(struct journal_iter *iter)
|
|
|
|
{
|
|
|
|
if (iter->idx < iter->keys->size) {
|
|
|
|
iter->idx++;
|
|
|
|
if (iter->idx == iter->keys->gap)
|
|
|
|
iter->idx += iter->keys->size - iter->keys->nr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct bkey_i *bch2_journal_iter_peek(struct journal_iter *iter)
|
2020-03-15 23:29:43 -04:00
|
|
|
{
|
2021-12-25 20:07:00 -05:00
|
|
|
struct journal_key *k = iter->keys->d + iter->idx;
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
while (k < iter->keys->d + iter->keys->size &&
|
2021-12-25 20:07:00 -05:00
|
|
|
k->btree_id == iter->btree_id &&
|
|
|
|
k->level == iter->level) {
|
|
|
|
if (!k->overwritten)
|
|
|
|
return k->k;
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
bch2_journal_iter_advance(iter);
|
2021-12-25 20:07:00 -05:00
|
|
|
k = iter->keys->d + iter->idx;
|
|
|
|
}
|
2020-03-15 23:29:43 -04:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
static void bch2_journal_iter_exit(struct journal_iter *iter)
|
|
|
|
{
|
|
|
|
list_del(&iter->list);
|
2019-07-12 17:08:32 -04:00
|
|
|
}
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
static void bch2_journal_iter_init(struct bch_fs *c,
|
|
|
|
struct journal_iter *iter,
|
2020-03-15 23:29:43 -04:00
|
|
|
enum btree_id id, unsigned level,
|
|
|
|
struct bpos pos)
|
2019-07-12 17:08:32 -04:00
|
|
|
{
|
2020-03-15 23:29:43 -04:00
|
|
|
iter->btree_id = id;
|
|
|
|
iter->level = level;
|
2021-01-26 20:15:46 -05:00
|
|
|
iter->keys = &c->journal_keys;
|
2021-12-25 20:07:00 -05:00
|
|
|
iter->idx = bch2_journal_key_search(&c->journal_keys, id, level, pos);
|
2020-03-15 23:29:43 -04:00
|
|
|
}
|
2019-07-12 17:08:32 -04:00
|
|
|
|
2020-03-15 23:29:43 -04:00
|
|
|
static struct bkey_s_c bch2_journal_iter_peek_btree(struct btree_and_journal_iter *iter)
|
|
|
|
{
|
2021-01-26 20:15:46 -05:00
|
|
|
return bch2_btree_node_iter_peek_unpack(&iter->node_iter,
|
|
|
|
iter->b, &iter->unpacked);
|
2020-03-15 23:29:43 -04:00
|
|
|
}
|
2019-12-27 20:51:35 -05:00
|
|
|
|
2020-03-15 23:29:43 -04:00
|
|
|
static void bch2_journal_iter_advance_btree(struct btree_and_journal_iter *iter)
|
|
|
|
{
|
2021-01-26 20:15:46 -05:00
|
|
|
bch2_btree_node_iter_advance(&iter->node_iter, iter->b);
|
2019-07-12 17:08:32 -04:00
|
|
|
}
|
|
|
|
|
2019-12-27 20:51:35 -05:00
|
|
|
void bch2_btree_and_journal_iter_advance(struct btree_and_journal_iter *iter)
|
|
|
|
{
|
|
|
|
switch (iter->last) {
|
|
|
|
case none:
|
|
|
|
break;
|
|
|
|
case btree:
|
2020-03-15 23:29:43 -04:00
|
|
|
bch2_journal_iter_advance_btree(iter);
|
2019-12-27 20:51:35 -05:00
|
|
|
break;
|
|
|
|
case journal:
|
2020-03-15 23:29:43 -04:00
|
|
|
bch2_journal_iter_advance(&iter->journal);
|
2019-12-27 20:51:35 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
iter->last = none;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct bkey_s_c bch2_btree_and_journal_iter_peek(struct btree_and_journal_iter *iter)
|
|
|
|
{
|
|
|
|
struct bkey_s_c ret;
|
|
|
|
|
|
|
|
while (1) {
|
2020-03-15 23:29:43 -04:00
|
|
|
struct bkey_s_c btree_k =
|
|
|
|
bch2_journal_iter_peek_btree(iter);
|
|
|
|
struct bkey_s_c journal_k =
|
|
|
|
bkey_i_to_s_c(bch2_journal_iter_peek(&iter->journal));
|
2019-12-27 20:51:35 -05:00
|
|
|
|
|
|
|
if (btree_k.k && journal_k.k) {
|
2021-03-04 16:20:16 -05:00
|
|
|
int cmp = bpos_cmp(btree_k.k->p, journal_k.k->p);
|
2019-12-27 20:51:35 -05:00
|
|
|
|
|
|
|
if (!cmp)
|
2020-03-15 23:29:43 -04:00
|
|
|
bch2_journal_iter_advance_btree(iter);
|
2019-12-27 20:51:35 -05:00
|
|
|
|
|
|
|
iter->last = cmp < 0 ? btree : journal;
|
|
|
|
} else if (btree_k.k) {
|
|
|
|
iter->last = btree;
|
|
|
|
} else if (journal_k.k) {
|
|
|
|
iter->last = journal;
|
|
|
|
} else {
|
|
|
|
iter->last = none;
|
|
|
|
return bkey_s_c_null;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = iter->last == journal ? journal_k : btree_k;
|
2020-03-15 23:29:43 -04:00
|
|
|
|
|
|
|
if (iter->b &&
|
2021-03-04 16:20:16 -05:00
|
|
|
bpos_cmp(ret.k->p, iter->b->data->max_key) > 0) {
|
2021-01-26 20:15:46 -05:00
|
|
|
iter->journal.idx = iter->journal.keys->nr;
|
2020-03-15 23:29:43 -04:00
|
|
|
iter->last = none;
|
|
|
|
return bkey_s_c_null;
|
|
|
|
}
|
|
|
|
|
2019-12-27 20:51:35 -05:00
|
|
|
if (!bkey_deleted(ret.k))
|
|
|
|
break;
|
|
|
|
|
|
|
|
bch2_btree_and_journal_iter_advance(iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct bkey_s_c bch2_btree_and_journal_iter_next(struct btree_and_journal_iter *iter)
|
|
|
|
{
|
|
|
|
bch2_btree_and_journal_iter_advance(iter);
|
|
|
|
|
|
|
|
return bch2_btree_and_journal_iter_peek(iter);
|
|
|
|
}
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
void bch2_btree_and_journal_iter_exit(struct btree_and_journal_iter *iter)
|
2019-12-27 20:51:35 -05:00
|
|
|
{
|
2021-01-26 20:15:46 -05:00
|
|
|
bch2_journal_iter_exit(&iter->journal);
|
2020-03-15 23:29:43 -04:00
|
|
|
}
|
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
void __bch2_btree_and_journal_iter_init_node_iter(struct btree_and_journal_iter *iter,
|
|
|
|
struct bch_fs *c,
|
|
|
|
struct btree *b,
|
|
|
|
struct btree_node_iter node_iter,
|
|
|
|
struct bpos pos)
|
2020-03-15 23:29:43 -04:00
|
|
|
{
|
|
|
|
memset(iter, 0, sizeof(*iter));
|
|
|
|
|
|
|
|
iter->b = b;
|
2021-12-25 20:07:00 -05:00
|
|
|
iter->node_iter = node_iter;
|
|
|
|
bch2_journal_iter_init(c, &iter->journal, b->c.btree_id, b->c.level, pos);
|
|
|
|
INIT_LIST_HEAD(&iter->journal.list);
|
2019-12-27 20:51:35 -05:00
|
|
|
}
|
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
/*
|
|
|
|
* this version is used by btree_gc before filesystem has gone RW and
|
|
|
|
* multithreaded, so uses the journal_iters list:
|
|
|
|
*/
|
|
|
|
void bch2_btree_and_journal_iter_init_node_iter(struct btree_and_journal_iter *iter,
|
|
|
|
struct bch_fs *c,
|
|
|
|
struct btree *b)
|
2020-05-24 14:06:10 -04:00
|
|
|
{
|
2021-12-25 20:07:00 -05:00
|
|
|
struct btree_node_iter node_iter;
|
2020-05-24 14:06:10 -04:00
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
bch2_btree_node_iter_init_from_start(&node_iter, b);
|
|
|
|
__bch2_btree_and_journal_iter_init_node_iter(iter, c, b, node_iter, b->data->min_key);
|
|
|
|
list_add(&iter->journal.list, &c->journal_iters);
|
2020-05-24 14:06:10 -04:00
|
|
|
}
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
/* sort and dedup all keys in the journal: */
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2020-03-25 16:12:33 -04:00
|
|
|
void bch2_journal_entries_free(struct list_head *list)
|
2017-03-16 22:18:50 -08:00
|
|
|
{
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
while (!list_empty(list)) {
|
|
|
|
struct journal_replay *i =
|
|
|
|
list_first_entry(list, struct journal_replay, list);
|
|
|
|
list_del(&i->list);
|
|
|
|
kvpfree(i, offsetof(struct journal_replay, j) +
|
|
|
|
vstruct_bytes(&i->j));
|
2017-03-16 22:18:50 -08:00
|
|
|
}
|
2019-04-11 22:39:39 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-12-30 14:37:25 -05:00
|
|
|
/*
|
|
|
|
* When keys compare equal, oldest compares first:
|
|
|
|
*/
|
2019-04-11 22:39:39 -04:00
|
|
|
static int journal_sort_key_cmp(const void *_l, const void *_r)
|
|
|
|
{
|
|
|
|
const struct journal_key *l = _l;
|
|
|
|
const struct journal_key *r = _r;
|
|
|
|
|
2021-12-25 20:07:00 -05:00
|
|
|
return journal_key_cmp(l, r) ?:
|
2019-04-11 22:39:39 -04:00
|
|
|
cmp_int(l->journal_seq, r->journal_seq) ?:
|
|
|
|
cmp_int(l->journal_offset, r->journal_offset);
|
|
|
|
}
|
|
|
|
|
2020-03-25 16:12:33 -04:00
|
|
|
void bch2_journal_keys_free(struct journal_keys *keys)
|
2019-04-11 22:39:39 -04:00
|
|
|
{
|
2021-01-26 20:15:46 -05:00
|
|
|
struct journal_key *i;
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
move_gap(keys->d, keys->nr, keys->size, keys->gap, keys->nr);
|
|
|
|
keys->gap = keys->nr;
|
|
|
|
|
2021-01-26 20:15:46 -05:00
|
|
|
for (i = keys->d; i < keys->d + keys->nr; i++)
|
|
|
|
if (i->allocated)
|
|
|
|
kfree(i->k);
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
kvfree(keys->d);
|
|
|
|
keys->d = NULL;
|
2022-04-04 01:09:26 -04:00
|
|
|
keys->nr = keys->gap = keys->size = 0;
|
2019-04-11 22:39:39 -04:00
|
|
|
}
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
static int journal_keys_sort(struct bch_fs *c)
|
2019-04-11 22:39:39 -04:00
|
|
|
{
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
struct journal_replay *i;
|
2019-04-11 22:39:39 -04:00
|
|
|
struct jset_entry *entry;
|
|
|
|
struct bkey_i *k, *_n;
|
2022-03-21 00:15:53 -04:00
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
2019-12-30 14:37:25 -05:00
|
|
|
struct journal_key *src, *dst;
|
2019-04-11 22:39:39 -04:00
|
|
|
size_t nr_keys = 0;
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
list_for_each_entry(i, &c->journal_entries, list) {
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
if (i->ignore)
|
2020-06-13 18:43:14 -04:00
|
|
|
continue;
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
if (!keys->journal_seq_base)
|
|
|
|
keys->journal_seq_base = le64_to_cpu(i->j.seq);
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
|
|
|
|
for_each_jset_key(k, _n, entry, &i->j)
|
2019-04-11 22:39:39 -04:00
|
|
|
nr_keys++;
|
2020-06-13 18:43:14 -04:00
|
|
|
}
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
if (!nr_keys)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
keys->size = roundup_pow_of_two(nr_keys);
|
2021-01-26 20:15:46 -05:00
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
keys->d = kvmalloc(sizeof(keys->d[0]) * keys->size, GFP_KERNEL);
|
|
|
|
if (!keys->d)
|
|
|
|
return -ENOMEM;
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
list_for_each_entry(i, &c->journal_entries, list) {
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
if (i->ignore)
|
2020-06-13 18:43:14 -04:00
|
|
|
continue;
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
BUG_ON(le64_to_cpu(i->j.seq) - keys->journal_seq_base > U32_MAX);
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
|
|
|
|
for_each_jset_key(k, _n, entry, &i->j)
|
2022-03-21 00:15:53 -04:00
|
|
|
keys->d[keys->nr++] = (struct journal_key) {
|
2019-04-11 22:39:39 -04:00
|
|
|
.btree_id = entry->btree_id,
|
2020-03-15 23:29:43 -04:00
|
|
|
.level = entry->level,
|
2019-04-11 22:39:39 -04:00
|
|
|
.k = k,
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
.journal_seq = le64_to_cpu(i->j.seq) -
|
2022-03-21 00:15:53 -04:00
|
|
|
keys->journal_seq_base,
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
.journal_offset = k->_data - i->j._data,
|
2019-04-11 22:39:39 -04:00
|
|
|
};
|
2020-06-13 18:43:14 -04:00
|
|
|
}
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
sort(keys->d, keys->nr, sizeof(keys->d[0]), journal_sort_key_cmp, NULL);
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
src = dst = keys->d;
|
|
|
|
while (src < keys->d + keys->nr) {
|
|
|
|
while (src + 1 < keys->d + keys->nr &&
|
2020-03-15 23:29:43 -04:00
|
|
|
src[0].btree_id == src[1].btree_id &&
|
|
|
|
src[0].level == src[1].level &&
|
2021-03-04 16:20:16 -05:00
|
|
|
!bpos_cmp(src[0].k->k.p, src[1].k->k.p))
|
2019-12-30 14:37:25 -05:00
|
|
|
src++;
|
2019-04-11 22:39:39 -04:00
|
|
|
|
2019-12-30 14:37:25 -05:00
|
|
|
*dst++ = *src++;
|
2019-04-11 22:39:39 -04:00
|
|
|
}
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
keys->nr = dst - keys->d;
|
|
|
|
keys->gap = keys->nr;
|
|
|
|
return 0;
|
2019-04-11 22:39:39 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* journal replay: */
|
|
|
|
|
|
|
|
static void replay_now_at(struct journal *j, u64 seq)
|
|
|
|
{
|
|
|
|
BUG_ON(seq < j->replay_journal_seq);
|
|
|
|
BUG_ON(seq > j->replay_journal_seq_end);
|
|
|
|
|
|
|
|
while (j->replay_journal_seq < seq)
|
|
|
|
bch2_journal_pin_put(j, j->replay_journal_seq++);
|
|
|
|
}
|
|
|
|
|
2021-12-27 23:10:06 -05:00
|
|
|
static int bch2_journal_replay_key(struct btree_trans *trans,
|
|
|
|
struct journal_key *k)
|
2019-12-31 16:17:42 -05:00
|
|
|
{
|
2021-08-30 15:18:31 -04:00
|
|
|
struct btree_iter iter;
|
2021-10-26 17:35:58 -04:00
|
|
|
unsigned iter_flags =
|
|
|
|
BTREE_ITER_INTENT|
|
|
|
|
BTREE_ITER_NOT_EXTENTS;
|
2020-03-09 16:15:54 -04:00
|
|
|
int ret;
|
2019-12-31 16:17:42 -05:00
|
|
|
|
2021-10-26 17:35:58 -04:00
|
|
|
if (!k->level && k->btree_id == BTREE_ID_alloc)
|
2021-12-27 23:10:06 -05:00
|
|
|
iter_flags |= BTREE_ITER_CACHED;
|
2021-10-26 17:35:58 -04:00
|
|
|
|
|
|
|
bch2_trans_node_iter_init(trans, &iter, k->btree_id, k->k->k.p,
|
|
|
|
BTREE_MAX_DEPTH, k->level,
|
|
|
|
iter_flags);
|
2021-12-31 17:54:13 -05:00
|
|
|
ret = bch2_btree_iter_traverse(&iter);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/* Must be checked with btree locked: */
|
|
|
|
if (k->overwritten)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = bch2_trans_update(trans, &iter, k->k, BTREE_TRIGGER_NORUN);
|
|
|
|
out:
|
2021-08-30 15:18:31 -04:00
|
|
|
bch2_trans_iter_exit(trans, &iter);
|
2020-03-09 16:15:54 -04:00
|
|
|
return ret;
|
2019-12-31 16:17:42 -05:00
|
|
|
}
|
|
|
|
|
2019-10-05 12:54:53 -04:00
|
|
|
static int journal_sort_seq_cmp(const void *_l, const void *_r)
|
|
|
|
{
|
2021-12-29 15:55:25 -05:00
|
|
|
const struct journal_key *l = *((const struct journal_key **)_l);
|
|
|
|
const struct journal_key *r = *((const struct journal_key **)_r);
|
2019-10-05 12:54:53 -04:00
|
|
|
|
2021-12-27 23:10:06 -05:00
|
|
|
return cmp_int(l->journal_seq, r->journal_seq);
|
2019-10-05 12:54:53 -04:00
|
|
|
}
|
|
|
|
|
2021-12-29 15:55:25 -05:00
|
|
|
static int bch2_journal_replay(struct bch_fs *c)
|
2019-04-11 22:39:39 -04:00
|
|
|
{
|
2021-12-29 15:55:25 -05:00
|
|
|
struct journal_keys *keys = &c->journal_keys;
|
|
|
|
struct journal_key **keys_sorted, *k;
|
2019-04-11 22:39:39 -04:00
|
|
|
struct journal *j = &c->journal;
|
2021-12-29 15:55:25 -05:00
|
|
|
size_t i;
|
|
|
|
int ret;
|
|
|
|
|
2022-04-04 01:09:26 -04:00
|
|
|
move_gap(keys->d, keys->nr, keys->size, keys->gap, keys->nr);
|
|
|
|
keys->gap = keys->nr;
|
|
|
|
|
2022-01-03 04:17:02 -05:00
|
|
|
keys_sorted = kvmalloc_array(sizeof(*keys_sorted), keys->nr, GFP_KERNEL);
|
2021-12-29 15:55:25 -05:00
|
|
|
if (!keys_sorted)
|
|
|
|
return -ENOMEM;
|
2019-03-29 19:13:54 -04:00
|
|
|
|
2021-12-29 15:55:25 -05:00
|
|
|
for (i = 0; i < keys->nr; i++)
|
|
|
|
keys_sorted[i] = &keys->d[i];
|
2019-03-29 19:13:54 -04:00
|
|
|
|
2021-12-29 15:55:25 -05:00
|
|
|
sort(keys_sorted, keys->nr,
|
|
|
|
sizeof(keys_sorted[0]),
|
|
|
|
journal_sort_seq_cmp, NULL);
|
|
|
|
|
2022-01-04 19:05:08 -05:00
|
|
|
if (keys->nr)
|
2021-12-29 15:55:25 -05:00
|
|
|
replay_now_at(j, keys->journal_seq_base);
|
2020-03-15 22:32:03 -04:00
|
|
|
|
2021-12-29 15:55:25 -05:00
|
|
|
for (i = 0; i < keys->nr; i++) {
|
|
|
|
k = keys_sorted[i];
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
cond_resched();
|
2019-10-05 12:54:53 -04:00
|
|
|
|
2021-12-27 23:10:06 -05:00
|
|
|
if (!k->allocated)
|
|
|
|
replay_now_at(j, keys->journal_seq_base + k->journal_seq);
|
2019-10-05 12:54:53 -04:00
|
|
|
|
2021-12-27 23:10:06 -05:00
|
|
|
ret = bch2_trans_do(c, NULL, NULL,
|
|
|
|
BTREE_INSERT_LAZY_RW|
|
|
|
|
BTREE_INSERT_NOFAIL|
|
2022-03-14 21:48:42 -04:00
|
|
|
(!k->allocated
|
|
|
|
? BTREE_INSERT_JOURNAL_REPLAY|JOURNAL_WATERMARK_reserved
|
|
|
|
: 0),
|
2021-12-27 23:10:06 -05:00
|
|
|
bch2_journal_replay_key(&trans, k));
|
|
|
|
if (ret) {
|
|
|
|
bch_err(c, "journal replay: error %d while replaying key at btree %s level %u",
|
|
|
|
ret, bch2_btree_ids[k->btree_id], k->level);
|
2019-10-05 12:54:53 -04:00
|
|
|
goto err;
|
2021-12-27 23:10:06 -05:00
|
|
|
}
|
2019-03-29 19:13:54 -04:00
|
|
|
}
|
2019-04-11 22:39:39 -04:00
|
|
|
|
|
|
|
replay_now_at(j, j->replay_journal_seq_end);
|
|
|
|
j->replay_journal_seq = 0;
|
|
|
|
|
|
|
|
bch2_journal_set_replay_done(j);
|
|
|
|
bch2_journal_flush_all_pins(j);
|
2021-12-27 23:10:06 -05:00
|
|
|
ret = bch2_journal_error(j);
|
2022-03-10 14:25:16 -05:00
|
|
|
|
|
|
|
if (keys->nr && !ret)
|
|
|
|
bch2_journal_log_msg(&c->journal, "journal replay finished");
|
2019-10-05 12:54:53 -04:00
|
|
|
err:
|
2022-01-03 04:17:02 -05:00
|
|
|
kvfree(keys_sorted);
|
2019-10-05 12:54:53 -04:00
|
|
|
return ret;
|
2019-03-29 19:13:54 -04:00
|
|
|
}
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
/* journal replay early: */
|
2019-03-29 19:13:54 -04:00
|
|
|
|
2019-01-24 19:09:49 -05:00
|
|
|
static int journal_replay_entry_early(struct bch_fs *c,
|
|
|
|
struct jset_entry *entry)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
switch (entry->type) {
|
|
|
|
case BCH_JSET_ENTRY_btree_root: {
|
2019-06-24 18:11:35 -04:00
|
|
|
struct btree_root *r;
|
|
|
|
|
|
|
|
if (entry->btree_id >= BTREE_ID_NR) {
|
|
|
|
bch_err(c, "filesystem has unknown btree type %u",
|
|
|
|
entry->btree_id);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
r = &c->btree_roots[entry->btree_id];
|
2019-01-24 19:09:49 -05:00
|
|
|
|
|
|
|
if (entry->u64s) {
|
|
|
|
r->level = entry->level;
|
|
|
|
bkey_copy(&r->key, &entry->start[0]);
|
|
|
|
r->error = 0;
|
|
|
|
} else {
|
|
|
|
r->error = -EIO;
|
|
|
|
}
|
|
|
|
r->alive = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case BCH_JSET_ENTRY_usage: {
|
|
|
|
struct jset_entry_usage *u =
|
|
|
|
container_of(entry, struct jset_entry_usage, entry);
|
|
|
|
|
2019-02-09 19:20:57 -05:00
|
|
|
switch (entry->btree_id) {
|
2021-12-31 17:06:29 -05:00
|
|
|
case BCH_FS_USAGE_reserved:
|
2019-02-09 19:20:57 -05:00
|
|
|
if (entry->level < BCH_REPLICAS_MAX)
|
2019-02-10 19:34:47 -05:00
|
|
|
c->usage_base->persistent_reserved[entry->level] =
|
|
|
|
le64_to_cpu(u->v);
|
2019-01-24 19:09:49 -05:00
|
|
|
break;
|
2021-12-31 17:06:29 -05:00
|
|
|
case BCH_FS_USAGE_inodes:
|
2019-02-10 19:34:47 -05:00
|
|
|
c->usage_base->nr_inodes = le64_to_cpu(u->v);
|
2019-01-24 19:09:49 -05:00
|
|
|
break;
|
2021-12-31 17:06:29 -05:00
|
|
|
case BCH_FS_USAGE_key_version:
|
2019-01-24 19:09:49 -05:00
|
|
|
atomic64_set(&c->key_version,
|
2019-02-09 19:20:57 -05:00
|
|
|
le64_to_cpu(u->v));
|
2019-01-24 19:09:49 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-02-09 19:20:57 -05:00
|
|
|
case BCH_JSET_ENTRY_data_usage: {
|
|
|
|
struct jset_entry_data_usage *u =
|
|
|
|
container_of(entry, struct jset_entry_data_usage, entry);
|
2021-01-21 21:52:06 -05:00
|
|
|
|
2019-02-09 19:20:57 -05:00
|
|
|
ret = bch2_replicas_set_usage(c, &u->r,
|
|
|
|
le64_to_cpu(u->v));
|
|
|
|
break;
|
|
|
|
}
|
2021-01-21 21:52:06 -05:00
|
|
|
case BCH_JSET_ENTRY_dev_usage: {
|
|
|
|
struct jset_entry_dev_usage *u =
|
|
|
|
container_of(entry, struct jset_entry_dev_usage, entry);
|
2021-05-23 17:04:13 -04:00
|
|
|
struct bch_dev *ca = bch_dev_bkey_exists(c, le32_to_cpu(u->dev));
|
2021-12-31 17:06:29 -05:00
|
|
|
unsigned i, nr_types = jset_entry_dev_usage_nr_types(u);
|
2021-01-21 21:52:06 -05:00
|
|
|
|
|
|
|
ca->usage_base->buckets_ec = le64_to_cpu(u->buckets_ec);
|
|
|
|
|
2021-06-10 23:51:09 -04:00
|
|
|
for (i = 0; i < min_t(unsigned, nr_types, BCH_DATA_NR); i++) {
|
2021-01-21 21:52:06 -05:00
|
|
|
ca->usage_base->d[i].buckets = le64_to_cpu(u->d[i].buckets);
|
|
|
|
ca->usage_base->d[i].sectors = le64_to_cpu(u->d[i].sectors);
|
|
|
|
ca->usage_base->d[i].fragmented = le64_to_cpu(u->d[i].fragmented);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-04-04 21:53:12 -04:00
|
|
|
case BCH_JSET_ENTRY_blacklist: {
|
|
|
|
struct jset_entry_blacklist *bl_entry =
|
|
|
|
container_of(entry, struct jset_entry_blacklist, entry);
|
|
|
|
|
|
|
|
ret = bch2_journal_seq_blacklist_add(c,
|
|
|
|
le64_to_cpu(bl_entry->seq),
|
|
|
|
le64_to_cpu(bl_entry->seq) + 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case BCH_JSET_ENTRY_blacklist_v2: {
|
|
|
|
struct jset_entry_blacklist_v2 *bl_entry =
|
|
|
|
container_of(entry, struct jset_entry_blacklist_v2, entry);
|
|
|
|
|
|
|
|
ret = bch2_journal_seq_blacklist_add(c,
|
|
|
|
le64_to_cpu(bl_entry->start),
|
|
|
|
le64_to_cpu(bl_entry->end) + 1);
|
|
|
|
break;
|
|
|
|
}
|
2021-01-21 15:28:59 -05:00
|
|
|
case BCH_JSET_ENTRY_clock: {
|
|
|
|
struct jset_entry_clock *clock =
|
|
|
|
container_of(entry, struct jset_entry_clock, entry);
|
|
|
|
|
2021-05-23 17:04:13 -04:00
|
|
|
atomic64_set(&c->io_clock[clock->rw].now, le64_to_cpu(clock->time));
|
2021-01-21 15:28:59 -05:00
|
|
|
}
|
2019-01-24 19:09:49 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-04-04 21:53:12 -04:00
|
|
|
static int journal_replay_early(struct bch_fs *c,
|
|
|
|
struct bch_sb_field_clean *clean,
|
|
|
|
struct list_head *journal)
|
2017-03-16 22:18:50 -08:00
|
|
|
{
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
struct journal_replay *i;
|
2019-03-29 19:13:54 -04:00
|
|
|
struct jset_entry *entry;
|
|
|
|
int ret;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
if (clean) {
|
|
|
|
for (entry = clean->start;
|
|
|
|
entry != vstruct_end(&clean->field);
|
|
|
|
entry = vstruct_next(entry)) {
|
|
|
|
ret = journal_replay_entry_early(c, entry);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
} else {
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
list_for_each_entry(i, journal, list) {
|
|
|
|
if (i->ignore)
|
|
|
|
continue;
|
2019-03-29 19:13:54 -04:00
|
|
|
|
|
|
|
vstruct_for_each(&i->j, entry) {
|
|
|
|
ret = journal_replay_entry_early(c, entry);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
}
|
2019-03-21 22:19:57 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
bch2_fs_usage_initialize(c);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
/* sb clean section: */
|
|
|
|
|
|
|
|
static struct bkey_i *btree_root_find(struct bch_fs *c,
|
|
|
|
struct bch_sb_field_clean *clean,
|
|
|
|
struct jset *j,
|
|
|
|
enum btree_id id, unsigned *level)
|
|
|
|
{
|
|
|
|
struct bkey_i *k;
|
|
|
|
struct jset_entry *entry, *start, *end;
|
|
|
|
|
|
|
|
if (clean) {
|
|
|
|
start = clean->start;
|
|
|
|
end = vstruct_end(&clean->field);
|
|
|
|
} else {
|
|
|
|
start = j->start;
|
|
|
|
end = vstruct_last(j);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (entry = start; entry < end; entry = vstruct_next(entry))
|
|
|
|
if (entry->type == BCH_JSET_ENTRY_btree_root &&
|
|
|
|
entry->btree_id == id)
|
|
|
|
goto found;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
found:
|
|
|
|
if (!entry->u64s)
|
|
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
|
|
|
|
k = entry->start;
|
|
|
|
*level = entry->level;
|
|
|
|
return k;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int verify_superblock_clean(struct bch_fs *c,
|
|
|
|
struct bch_sb_field_clean **cleanp,
|
|
|
|
struct jset *j)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
struct bch_sb_field_clean *clean = *cleanp;
|
2022-02-25 13:18:19 -05:00
|
|
|
struct printbuf buf1 = PRINTBUF;
|
|
|
|
struct printbuf buf2 = PRINTBUF;
|
2019-04-11 22:39:39 -04:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (mustfix_fsck_err_on(j->seq != clean->journal_seq, c,
|
|
|
|
"superblock journal seq (%llu) doesn't match journal (%llu) after clean shutdown",
|
|
|
|
le64_to_cpu(clean->journal_seq),
|
|
|
|
le64_to_cpu(j->seq))) {
|
|
|
|
kfree(clean);
|
|
|
|
*cleanp = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < BTREE_ID_NR; i++) {
|
|
|
|
struct bkey_i *k1, *k2;
|
|
|
|
unsigned l1 = 0, l2 = 0;
|
|
|
|
|
|
|
|
k1 = btree_root_find(c, clean, NULL, i, &l1);
|
|
|
|
k2 = btree_root_find(c, NULL, j, i, &l2);
|
|
|
|
|
|
|
|
if (!k1 && !k2)
|
|
|
|
continue;
|
|
|
|
|
2022-02-25 13:18:19 -05:00
|
|
|
printbuf_reset(&buf1);
|
|
|
|
printbuf_reset(&buf2);
|
|
|
|
|
|
|
|
if (k1)
|
|
|
|
bch2_bkey_val_to_text(&buf1, c, bkey_i_to_s_c(k1));
|
|
|
|
else
|
|
|
|
pr_buf(&buf1, "(none)");
|
|
|
|
|
|
|
|
if (k2)
|
|
|
|
bch2_bkey_val_to_text(&buf2, c, bkey_i_to_s_c(k2));
|
|
|
|
else
|
|
|
|
pr_buf(&buf2, "(none)");
|
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
mustfix_fsck_err_on(!k1 || !k2 ||
|
|
|
|
IS_ERR(k1) ||
|
|
|
|
IS_ERR(k2) ||
|
|
|
|
k1->k.u64s != k2->k.u64s ||
|
|
|
|
memcmp(k1, k2, bkey_bytes(k1)) ||
|
|
|
|
l1 != l2, c,
|
2020-05-25 14:57:06 -04:00
|
|
|
"superblock btree root %u doesn't match journal after clean shutdown\n"
|
|
|
|
"sb: l=%u %s\n"
|
|
|
|
"journal: l=%u %s\n", i,
|
2022-02-25 13:18:19 -05:00
|
|
|
l1, buf1.buf,
|
|
|
|
l2, buf2.buf);
|
2019-04-11 22:39:39 -04:00
|
|
|
}
|
|
|
|
fsck_err:
|
2022-02-25 13:18:19 -05:00
|
|
|
printbuf_exit(&buf2);
|
|
|
|
printbuf_exit(&buf1);
|
2019-04-11 22:39:39 -04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct bch_sb_field_clean *read_superblock_clean(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
struct bch_sb_field_clean *clean, *sb_clean;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&c->sb_lock);
|
|
|
|
sb_clean = bch2_sb_get_clean(c->disk_sb.sb);
|
|
|
|
|
|
|
|
if (fsck_err_on(!sb_clean, c,
|
|
|
|
"superblock marked clean but clean section not present")) {
|
|
|
|
SET_BCH_SB_CLEAN(c->disk_sb.sb, false);
|
|
|
|
c->sb.clean = false;
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
clean = kmemdup(sb_clean, vstruct_bytes(&sb_clean->field),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!clean) {
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
}
|
|
|
|
|
2022-02-20 05:00:45 -05:00
|
|
|
ret = bch2_sb_clean_validate_late(c, clean, READ);
|
2021-03-04 19:06:26 -05:00
|
|
|
if (ret) {
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
}
|
2019-04-11 22:39:39 -04:00
|
|
|
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
|
|
|
|
return clean;
|
|
|
|
fsck_err:
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
}
|
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
static int read_btree_roots(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
int ret = 0;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
|
|
|
for (i = 0; i < BTREE_ID_NR; i++) {
|
2019-03-29 19:13:54 -04:00
|
|
|
struct btree_root *r = &c->btree_roots[i];
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
if (!r->alive)
|
|
|
|
continue;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2021-02-20 19:27:37 -05:00
|
|
|
if (i == BTREE_ID_alloc &&
|
2019-08-28 13:20:31 -04:00
|
|
|
c->opts.reconstruct_alloc) {
|
2021-04-04 21:57:35 -04:00
|
|
|
c->sb.compat &= ~(1ULL << BCH_COMPAT_alloc_info);
|
2017-03-16 22:18:50 -08:00
|
|
|
continue;
|
2019-03-29 19:13:54 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
if (r->error) {
|
2021-02-20 19:27:37 -05:00
|
|
|
__fsck_err(c, i == BTREE_ID_alloc
|
2019-03-29 19:13:54 -04:00
|
|
|
? FSCK_CAN_IGNORE : 0,
|
|
|
|
"invalid btree root %s",
|
|
|
|
bch2_btree_ids[i]);
|
2021-02-20 19:27:37 -05:00
|
|
|
if (i == BTREE_ID_alloc)
|
2021-04-04 21:57:35 -04:00
|
|
|
c->sb.compat &= ~(1ULL << BCH_COMPAT_alloc_info);
|
2019-03-29 19:13:54 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = bch2_btree_root_read(c, i, &r->key, r->level);
|
|
|
|
if (ret) {
|
2021-02-20 19:27:37 -05:00
|
|
|
__fsck_err(c, i == BTREE_ID_alloc
|
2019-03-29 19:13:54 -04:00
|
|
|
? FSCK_CAN_IGNORE : 0,
|
|
|
|
"error reading btree root %s",
|
|
|
|
bch2_btree_ids[i]);
|
2021-02-20 19:27:37 -05:00
|
|
|
if (i == BTREE_ID_alloc)
|
2021-04-04 21:57:35 -04:00
|
|
|
c->sb.compat &= ~(1ULL << BCH_COMPAT_alloc_info);
|
2019-03-29 19:13:54 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
}
|
2019-03-29 19:13:54 -04:00
|
|
|
|
|
|
|
for (i = 0; i < BTREE_ID_NR; i++)
|
|
|
|
if (!c->btree_roots[i].b)
|
|
|
|
bch2_btree_root_alloc(c, i);
|
2017-03-16 22:18:50 -08:00
|
|
|
fsck_err:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-16 00:42:25 -04:00
|
|
|
static int bch2_fs_initialize_subvolumes(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
struct bkey_i_snapshot root_snapshot;
|
|
|
|
struct bkey_i_subvolume root_volume;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
bkey_snapshot_init(&root_snapshot.k_i);
|
|
|
|
root_snapshot.k.p.offset = U32_MAX;
|
|
|
|
root_snapshot.v.flags = 0;
|
|
|
|
root_snapshot.v.parent = 0;
|
|
|
|
root_snapshot.v.subvol = BCACHEFS_ROOT_SUBVOL;
|
|
|
|
root_snapshot.v.pad = 0;
|
|
|
|
SET_BCH_SNAPSHOT_SUBVOL(&root_snapshot.v, true);
|
|
|
|
|
|
|
|
ret = bch2_btree_insert(c, BTREE_ID_snapshots,
|
|
|
|
&root_snapshot.k_i,
|
|
|
|
NULL, NULL, 0);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
|
|
bkey_subvolume_init(&root_volume.k_i);
|
|
|
|
root_volume.k.p.offset = BCACHEFS_ROOT_SUBVOL;
|
|
|
|
root_volume.v.flags = 0;
|
|
|
|
root_volume.v.snapshot = cpu_to_le32(U32_MAX);
|
|
|
|
root_volume.v.inode = cpu_to_le64(BCACHEFS_ROOT_INO);
|
|
|
|
|
|
|
|
ret = bch2_btree_insert(c, BTREE_ID_subvolumes,
|
|
|
|
&root_volume.k_i,
|
|
|
|
NULL, NULL, 0);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bch2_fs_upgrade_for_subvolumes(struct btree_trans *trans)
|
|
|
|
{
|
|
|
|
struct btree_iter iter;
|
|
|
|
struct bkey_s_c k;
|
|
|
|
struct bch_inode_unpacked inode;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
bch2_trans_iter_init(trans, &iter, BTREE_ID_inodes,
|
2021-09-27 13:25:18 -04:00
|
|
|
SPOS(0, BCACHEFS_ROOT_INO, U32_MAX), 0);
|
2021-03-16 00:42:25 -04:00
|
|
|
k = bch2_btree_iter_peek_slot(&iter);
|
|
|
|
ret = bkey_err(k);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
2021-10-29 21:14:23 -04:00
|
|
|
if (!bkey_is_inode(k.k)) {
|
2022-01-04 19:05:08 -05:00
|
|
|
bch_err(trans->c, "root inode not found");
|
2021-03-16 00:42:25 -04:00
|
|
|
ret = -ENOENT;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:14:23 -04:00
|
|
|
ret = bch2_inode_unpack(k, &inode);
|
2021-03-16 00:42:25 -04:00
|
|
|
BUG_ON(ret);
|
|
|
|
|
|
|
|
inode.bi_subvol = BCACHEFS_ROOT_SUBVOL;
|
|
|
|
|
2021-09-27 13:25:18 -04:00
|
|
|
ret = bch2_inode_write(trans, &iter, &inode);
|
2021-03-16 00:42:25 -04:00
|
|
|
err:
|
|
|
|
bch2_trans_iter_exit(trans, &iter);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-03-16 22:18:50 -08:00
|
|
|
int bch2_fs_recovery(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
const char *err = "cannot allocate memory";
|
2019-04-04 21:53:12 -04:00
|
|
|
struct bch_sb_field_clean *clean = NULL;
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
struct jset *last_journal_entry = NULL;
|
|
|
|
u64 blacklist_seq, journal_seq;
|
2021-01-08 21:20:58 -05:00
|
|
|
bool write_sb = false;
|
2021-04-24 18:02:59 -04:00
|
|
|
int ret = 0;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-04-04 21:53:12 -04:00
|
|
|
if (c->sb.clean)
|
|
|
|
clean = read_superblock_clean(c);
|
|
|
|
ret = PTR_ERR_OR_ZERO(clean);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
if (c->sb.clean)
|
2017-03-16 22:18:50 -08:00
|
|
|
bch_info(c, "recovering from clean shutdown, journal seq %llu",
|
|
|
|
le64_to_cpu(clean->journal_seq));
|
2021-11-15 15:03:06 -05:00
|
|
|
else
|
|
|
|
bch_info(c, "recovering from unclean shutdown");
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2021-02-20 00:00:23 -05:00
|
|
|
if (!(c->sb.features & (1ULL << BCH_FEATURE_new_extent_overwrite))) {
|
|
|
|
bch_err(c, "feature new_extent_overwrite not set, filesystem no longer supported");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2021-03-21 16:20:40 -04:00
|
|
|
if (!c->sb.clean &&
|
|
|
|
!(c->sb.features & (1ULL << BCH_FEATURE_extents_above_btree_updates))) {
|
|
|
|
bch_err(c, "filesystem needs recovery from older version; run fsck from older bcachefs-tools to fix");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
bcachefs: Start using bpos.snapshot field
This patch starts treating the bpos.snapshot field like part of the key
in the btree code:
* bpos_successor() and bpos_predecessor() now include the snapshot field
* Keys in btrees that will be using snapshots (extents, inodes, dirents
and xattrs) now always have their snapshot field set to U32_MAX
The btree iterator code gets a new flag, BTREE_ITER_ALL_SNAPSHOTS, that
determines whether we're iterating over keys in all snapshots or not -
internally, this controlls whether bkey_(successor|predecessor)
increment/decrement the snapshot field, or only the higher bits of the
key.
We add a new member to struct btree_iter, iter->snapshot: when
BTREE_ITER_ALL_SNAPSHOTS is not set, iter->pos.snapshot should always
equal iter->snapshot, which will be 0 for btrees that don't use
snapshots, and alsways U32_MAX for btrees that will use snapshots
(until we enable snapshot creation).
This patch also introduces a new metadata version number, and compat
code for reading from/writing to older versions - this isn't a forced
upgrade (yet).
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2021-03-24 18:02:16 -04:00
|
|
|
if (!(c->sb.compat & (1ULL << BCH_COMPAT_bformat_overflow_done))) {
|
|
|
|
bch_err(c, "filesystem may have incompatible bkey formats; run fsck from the compat branch to fix");
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:31:17 -05:00
|
|
|
if (!(c->sb.features & (1ULL << BCH_FEATURE_alloc_v2))) {
|
|
|
|
bch_info(c, "alloc_v2 feature bit not set, fsck required");
|
|
|
|
c->opts.fsck = true;
|
|
|
|
c->opts.fix_errors = FSCK_OPT_YES;
|
|
|
|
}
|
|
|
|
|
2020-07-20 15:51:05 -04:00
|
|
|
if (!c->replicas.entries ||
|
|
|
|
c->opts.rebuild_replicas) {
|
2019-04-04 21:53:12 -04:00
|
|
|
bch_info(c, "building replicas info");
|
|
|
|
set_bit(BCH_FS_REBUILD_REPLICAS, &c->flags);
|
|
|
|
}
|
|
|
|
|
2021-10-27 17:53:20 -04:00
|
|
|
if (!c->opts.nochanges) {
|
2022-04-01 01:29:59 -04:00
|
|
|
if (c->sb.version < bcachefs_metadata_version_new_data_types) {
|
|
|
|
bch_info(c, "version prior to new_data_types, upgrade and fsck required");
|
2021-10-27 17:53:20 -04:00
|
|
|
c->opts.version_upgrade = true;
|
|
|
|
c->opts.fsck = true;
|
|
|
|
c->opts.fix_errors = FSCK_OPT_YES;
|
|
|
|
}
|
2021-07-10 13:44:42 -04:00
|
|
|
}
|
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
ret = bch2_blacklist_table_initialize(c);
|
|
|
|
if (ret) {
|
|
|
|
bch_err(c, "error initializing blacklist table");
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2020-03-28 18:26:01 -04:00
|
|
|
if (!c->sb.clean || c->opts.fsck || c->opts.keep_journal) {
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
struct journal_replay *i;
|
2019-04-04 21:53:12 -04:00
|
|
|
|
2022-01-04 00:06:49 -05:00
|
|
|
bch_verbose(c, "starting journal read");
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
ret = bch2_journal_read(c, &c->journal_entries,
|
|
|
|
&blacklist_seq, &journal_seq);
|
2017-03-16 22:18:50 -08:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
list_for_each_entry_reverse(i, &c->journal_entries, list)
|
|
|
|
if (!i->ignore) {
|
|
|
|
last_journal_entry = &i->j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mustfix_fsck_err_on(c->sb.clean &&
|
|
|
|
last_journal_entry &&
|
|
|
|
!journal_entry_empty(last_journal_entry), c,
|
2019-03-11 14:59:58 -04:00
|
|
|
"filesystem marked clean but journal not empty")) {
|
2021-04-04 21:57:35 -04:00
|
|
|
c->sb.compat &= ~(1ULL << BCH_COMPAT_alloc_info);
|
2019-03-11 14:59:58 -04:00
|
|
|
SET_BCH_SB_CLEAN(c->disk_sb.sb, false);
|
|
|
|
c->sb.clean = false;
|
|
|
|
}
|
2019-04-04 21:53:12 -04:00
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
if (!last_journal_entry) {
|
|
|
|
fsck_err_on(!c->sb.clean, c, "no journal entries found");
|
|
|
|
goto use_clean;
|
2019-04-04 21:53:12 -04:00
|
|
|
}
|
|
|
|
|
2022-03-21 00:15:53 -04:00
|
|
|
ret = journal_keys_sort(c);
|
|
|
|
if (ret)
|
2019-04-11 22:39:39 -04:00
|
|
|
goto err;
|
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
if (c->sb.clean && last_journal_entry) {
|
|
|
|
ret = verify_superblock_clean(c, &clean,
|
|
|
|
last_journal_entry);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
use_clean:
|
|
|
|
if (!clean) {
|
|
|
|
bch_err(c, "no superblock clean section found");
|
|
|
|
ret = BCH_FSCK_REPAIR_IMPOSSIBLE;
|
2019-03-29 19:13:54 -04:00
|
|
|
goto err;
|
2019-04-04 21:53:12 -04:00
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
}
|
|
|
|
blacklist_seq = journal_seq = le64_to_cpu(clean->journal_seq) + 1;
|
2019-04-04 21:53:12 -04:00
|
|
|
}
|
|
|
|
|
2022-02-19 05:15:53 -05:00
|
|
|
if (c->opts.read_journal_only)
|
|
|
|
goto out;
|
|
|
|
|
2020-10-24 21:20:16 -04:00
|
|
|
if (c->opts.reconstruct_alloc) {
|
2021-04-04 21:57:35 -04:00
|
|
|
c->sb.compat &= ~(1ULL << BCH_COMPAT_alloc_info);
|
2020-10-24 21:20:16 -04:00
|
|
|
drop_alloc_keys(&c->journal_keys);
|
|
|
|
}
|
|
|
|
|
2021-07-30 14:33:06 -04:00
|
|
|
zero_out_btree_mem_ptr(&c->journal_keys);
|
|
|
|
|
2020-03-25 16:12:33 -04:00
|
|
|
ret = journal_replay_early(c, clean, &c->journal_entries);
|
2019-04-04 21:53:12 -04:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
2022-01-04 19:41:23 -05:00
|
|
|
/*
|
|
|
|
* After an unclean shutdown, skip then next few journal sequence
|
|
|
|
* numbers as they may have been referenced by btree writes that
|
|
|
|
* happened before their corresponding journal writes - those btree
|
|
|
|
* writes need to be ignored, by skipping and blacklisting the next few
|
|
|
|
* journal sequence numbers:
|
|
|
|
*/
|
|
|
|
if (!c->sb.clean)
|
|
|
|
journal_seq += 8;
|
|
|
|
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
if (blacklist_seq != journal_seq) {
|
2019-04-04 21:53:12 -04:00
|
|
|
ret = bch2_journal_seq_blacklist_add(c,
|
bcachefs: Don't require flush/fua on every journal write
This patch adds a flag to journal entries which, if set, indicates that
they weren't done as flush/fua writes.
- non flush/fua journal writes don't update last_seq (i.e. they don't
free up space in the journal), thus the journal free space
calculations now check whether nonflush journal writes are currently
allowed (i.e. are we low on free space, or would doing a flush write
free up a lot of space in the journal)
- write_delay_ms, the user configurable option for when open journal
entries are automatically written, is now interpreted as the max
delay between flush journal writes (default 1 second).
- bch2_journal_flush_seq_async is changed to ensure a flush write >=
the requested sequence number has happened
- journal read/replay must now ignore, and blacklist, any journal
entries newer than the most recent flush entry in the journal. Also,
the way the read_entire_journal option is handled has been improved;
struct journal_replay now has an entry, 'ignore', for entries that
were read but should not be used.
- assorted refactoring and improvements related to journal read in
journal_io.c and recovery.c
Previously, we'd have to issue a flush/fua write every time we
accumulated a full journal entry - typically the bucket size. Now we
need to issue them much less frequently: when an fsync is requested, or
it's been more than write_delay_ms since the last flush, or when we need
to free up space in the journal. This is a significant performance
improvement on many write heavy workloads.
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2020-11-14 09:59:58 -05:00
|
|
|
blacklist_seq, journal_seq);
|
2019-04-04 21:53:12 -04:00
|
|
|
if (ret) {
|
|
|
|
bch_err(c, "error creating new journal seq blacklist entry");
|
2019-03-29 19:13:54 -04:00
|
|
|
goto err;
|
2019-04-04 21:53:12 -04:00
|
|
|
}
|
2019-06-18 19:37:39 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-04-11 22:39:39 -04:00
|
|
|
ret = bch2_fs_journal_start(&c->journal, journal_seq,
|
2020-03-25 16:12:33 -04:00
|
|
|
&c->journal_entries);
|
2019-03-29 19:13:54 -04:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
ret = read_btree_roots(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-11 14:59:58 -04:00
|
|
|
bch_verbose(c, "starting alloc read");
|
2017-03-16 22:18:50 -08:00
|
|
|
err = "error reading allocation information";
|
2021-12-25 20:39:19 -05:00
|
|
|
|
|
|
|
down_read(&c->gc_lock);
|
2022-02-10 19:26:55 -05:00
|
|
|
ret = bch2_alloc_read(c);
|
2021-12-25 20:39:19 -05:00
|
|
|
up_read(&c->gc_lock);
|
|
|
|
|
2017-03-16 22:18:50 -08:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
2019-03-11 14:59:58 -04:00
|
|
|
bch_verbose(c, "alloc read done");
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-02-09 16:50:53 -05:00
|
|
|
bch_verbose(c, "starting stripes_read");
|
2019-03-11 14:59:58 -04:00
|
|
|
err = "error reading stripes";
|
2021-04-29 15:37:47 -04:00
|
|
|
ret = bch2_stripes_read(c);
|
2018-11-23 02:50:33 -05:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
2019-02-09 16:50:53 -05:00
|
|
|
bch_verbose(c, "stripes_read done");
|
2018-11-25 20:53:51 -05:00
|
|
|
|
2022-01-01 19:46:12 -05:00
|
|
|
/*
|
|
|
|
* If we're not running fsck, this ensures bch2_fsck_err() calls are
|
|
|
|
* instead interpreted as bch2_inconsistent_err() calls:
|
|
|
|
*/
|
|
|
|
if (!c->opts.fsck)
|
|
|
|
set_bit(BCH_FS_FSCK_DONE, &c->flags);
|
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
if (c->opts.fsck ||
|
2021-04-04 21:57:35 -04:00
|
|
|
!(c->sb.compat & (1ULL << BCH_COMPAT_alloc_info)) ||
|
|
|
|
!(c->sb.compat & (1ULL << BCH_COMPAT_alloc_metadata)) ||
|
2019-03-29 19:13:54 -04:00
|
|
|
test_bit(BCH_FS_REBUILD_REPLICAS, &c->flags)) {
|
2021-04-16 16:54:11 -04:00
|
|
|
bool metadata_only = c->opts.norecovery;
|
|
|
|
|
2022-02-13 20:47:05 -05:00
|
|
|
bch_info(c, "checking allocations");
|
2022-02-17 03:11:39 -05:00
|
|
|
err = "error checking allocations";
|
2021-04-16 16:54:11 -04:00
|
|
|
ret = bch2_gc(c, true, metadata_only);
|
2020-10-16 21:36:26 -04:00
|
|
|
if (ret)
|
2021-01-08 21:20:58 -05:00
|
|
|
goto err;
|
2022-02-13 20:47:05 -05:00
|
|
|
bch_verbose(c, "done checking allocations");
|
2019-02-06 11:56:51 -05:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2022-02-17 03:11:39 -05:00
|
|
|
if (c->opts.fsck) {
|
|
|
|
bch_info(c, "checking need_discard and freespace btrees");
|
|
|
|
err = "error checking need_discard and freespace btrees";
|
2022-04-05 13:44:18 -04:00
|
|
|
ret = bch2_check_alloc_info(c);
|
2022-02-17 03:11:39 -05:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = bch2_check_lrus(c, true);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "done checking need_discard and freespace btrees");
|
|
|
|
}
|
|
|
|
|
2021-01-14 16:19:23 -05:00
|
|
|
bch2_stripes_heap_start(c);
|
|
|
|
|
2018-11-03 22:00:50 -04:00
|
|
|
clear_bit(BCH_FS_REBUILD_REPLICAS, &c->flags);
|
2019-02-06 11:56:51 -05:00
|
|
|
set_bit(BCH_FS_INITIAL_GC_DONE, &c->flags);
|
2022-02-19 02:48:27 -05:00
|
|
|
set_bit(BCH_FS_MAY_GO_RW, &c->flags);
|
2018-11-03 22:00:50 -04:00
|
|
|
|
2019-01-24 19:09:49 -05:00
|
|
|
/*
|
|
|
|
* Skip past versions that might have possibly been used (as nonces),
|
|
|
|
* but hadn't had their pointers written:
|
|
|
|
*/
|
|
|
|
if (c->sb.encryption_type && !c->sb.clean)
|
|
|
|
atomic64_add(1 << 16, &c->key_version);
|
|
|
|
|
2019-04-17 18:21:19 -04:00
|
|
|
if (c->opts.norecovery)
|
2019-03-29 19:13:54 -04:00
|
|
|
goto out;
|
|
|
|
|
2022-01-04 19:05:08 -05:00
|
|
|
bch_verbose(c, "starting journal replay, %zu keys", c->journal_keys.nr);
|
2017-03-16 22:18:50 -08:00
|
|
|
err = "journal replay failed";
|
2021-12-29 15:55:25 -05:00
|
|
|
ret = bch2_journal_replay(c);
|
2017-03-16 22:18:50 -08:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
2021-12-27 23:10:06 -05:00
|
|
|
if (c->opts.verbose || !c->sb.clean)
|
|
|
|
bch_info(c, "journal replay done");
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2021-12-11 17:13:09 -05:00
|
|
|
err = "error initializing freespace";
|
|
|
|
ret = bch2_fs_freespace_init(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
2022-02-17 03:11:39 -05:00
|
|
|
if (c->opts.fsck) {
|
|
|
|
bch_info(c, "checking alloc to lru refs");
|
|
|
|
err = "error checking alloc to lru refs";
|
|
|
|
ret = bch2_check_alloc_to_lru_refs(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = bch2_check_lrus(c, true);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "done checking alloc to lru refs");
|
|
|
|
}
|
|
|
|
|
2021-09-27 13:25:18 -04:00
|
|
|
if (c->sb.version < bcachefs_metadata_version_snapshot_2) {
|
|
|
|
bch2_fs_lazy_rw(c);
|
|
|
|
|
2021-03-16 00:42:25 -04:00
|
|
|
err = "error creating root snapshot node";
|
|
|
|
ret = bch2_fs_initialize_subvolumes(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
bch_verbose(c, "reading snapshots table");
|
|
|
|
err = "error reading snapshots table";
|
|
|
|
ret = bch2_fs_snapshots_start(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "reading snapshots done");
|
|
|
|
|
2021-09-27 13:25:18 -04:00
|
|
|
if (c->sb.version < bcachefs_metadata_version_snapshot_2) {
|
2021-03-16 00:42:25 -04:00
|
|
|
/* set bi_subvol on root inode */
|
|
|
|
err = "error upgrade root inode for subvolumes";
|
|
|
|
ret = bch2_trans_do(c, NULL, NULL, BTREE_INSERT_LAZY_RW,
|
|
|
|
bch2_fs_upgrade_for_subvolumes(&trans));
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2019-04-17 18:21:19 -04:00
|
|
|
if (c->opts.fsck) {
|
|
|
|
bch_info(c, "starting fsck");
|
|
|
|
err = "error in fsck";
|
|
|
|
ret = bch2_fsck_full(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "fsck done");
|
2021-04-06 21:19:25 -04:00
|
|
|
} else if (!c->sb.clean) {
|
|
|
|
bch_verbose(c, "checking for deleted inodes");
|
|
|
|
err = "error in recovery";
|
|
|
|
ret = bch2_fsck_walk_inodes_only(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "check inodes done");
|
2019-04-17 18:21:19 -04:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2019-03-29 19:13:54 -04:00
|
|
|
if (enabled_qtypes(c)) {
|
2019-04-17 18:21:19 -04:00
|
|
|
bch_verbose(c, "reading quotas");
|
2019-03-29 19:13:54 -04:00
|
|
|
ret = bch2_fs_quota_read(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "quotas done");
|
|
|
|
}
|
|
|
|
|
2018-11-01 15:10:01 -04:00
|
|
|
mutex_lock(&c->sb_lock);
|
|
|
|
if (c->opts.version_upgrade) {
|
2021-05-23 17:04:13 -04:00
|
|
|
c->disk_sb.sb->version = cpu_to_le16(bcachefs_metadata_version_current);
|
|
|
|
c->disk_sb.sb->features[0] |= cpu_to_le64(BCH_SB_FEATURES_ALL);
|
2019-03-11 14:59:58 -04:00
|
|
|
write_sb = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!test_bit(BCH_FS_ERROR, &c->flags)) {
|
2021-05-23 17:04:13 -04:00
|
|
|
c->disk_sb.sb->compat[0] |= cpu_to_le64(1ULL << BCH_COMPAT_alloc_info);
|
2019-03-11 14:59:58 -04:00
|
|
|
write_sb = true;
|
2018-07-14 21:06:51 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 09:34:55 -04:00
|
|
|
if (c->opts.fsck &&
|
2021-04-13 10:26:59 -04:00
|
|
|
!test_bit(BCH_FS_ERROR, &c->flags) &&
|
2021-04-24 16:32:35 -04:00
|
|
|
!test_bit(BCH_FS_ERRORS_NOT_FIXED, &c->flags)) {
|
2019-03-28 09:34:55 -04:00
|
|
|
SET_BCH_SB_HAS_ERRORS(c->disk_sb.sb, 0);
|
2021-04-24 16:32:35 -04:00
|
|
|
SET_BCH_SB_HAS_TOPOLOGY_ERRORS(c->disk_sb.sb, 0);
|
2019-03-11 14:59:58 -04:00
|
|
|
write_sb = true;
|
2019-03-28 09:34:55 -04:00
|
|
|
}
|
2019-03-11 14:59:58 -04:00
|
|
|
|
|
|
|
if (write_sb)
|
|
|
|
bch2_write_super(c);
|
2018-11-01 15:10:01 -04:00
|
|
|
mutex_unlock(&c->sb_lock);
|
2019-04-04 21:53:12 -04:00
|
|
|
|
2021-12-27 20:05:07 -05:00
|
|
|
if (!(c->sb.compat & (1ULL << BCH_COMPAT_extents_above_btree_updates_done)) ||
|
|
|
|
!(c->sb.compat & (1ULL << BCH_COMPAT_bformat_overflow_done)) ||
|
|
|
|
le16_to_cpu(c->sb.version_min) < bcachefs_metadata_version_btree_ptr_sectors_written) {
|
|
|
|
struct bch_move_stats stats;
|
|
|
|
|
|
|
|
bch_move_stats_init(&stats, "recovery");
|
|
|
|
|
|
|
|
bch_info(c, "scanning for old btree nodes");
|
|
|
|
ret = bch2_fs_read_write(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = bch2_scan_old_btree_nodes(c, &stats);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_info(c, "scanning for old btree nodes done");
|
|
|
|
}
|
|
|
|
|
2022-01-04 19:41:23 -05:00
|
|
|
if (c->journal_seq_blacklist_table &&
|
|
|
|
c->journal_seq_blacklist_table->nr > 128)
|
|
|
|
queue_work(system_long_wq, &c->journal_seq_blacklist_gc_work);
|
|
|
|
|
2019-04-17 18:21:19 -04:00
|
|
|
ret = 0;
|
2021-04-24 18:02:59 -04:00
|
|
|
out:
|
2019-09-07 12:42:27 -04:00
|
|
|
set_bit(BCH_FS_FSCK_DONE, &c->flags);
|
2019-04-17 18:21:19 -04:00
|
|
|
bch2_flush_fsck_errs(c);
|
2019-09-07 12:42:27 -04:00
|
|
|
|
2020-03-25 16:12:33 -04:00
|
|
|
if (!c->opts.keep_journal) {
|
|
|
|
bch2_journal_keys_free(&c->journal_keys);
|
|
|
|
bch2_journal_entries_free(&c->journal_entries);
|
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
kfree(clean);
|
2019-04-17 18:21:19 -04:00
|
|
|
if (ret)
|
|
|
|
bch_err(c, "Error in recovery: %s (%i)", err, ret);
|
|
|
|
else
|
|
|
|
bch_verbose(c, "ret %i", ret);
|
2017-03-16 22:18:50 -08:00
|
|
|
return ret;
|
2021-04-24 18:02:59 -04:00
|
|
|
err:
|
|
|
|
fsck_err:
|
|
|
|
bch2_fs_emergency_read_only(c);
|
|
|
|
goto out;
|
2017-03-16 22:18:50 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
int bch2_fs_initialize(struct bch_fs *c)
|
|
|
|
{
|
|
|
|
struct bch_inode_unpacked root_inode, lostfound_inode;
|
|
|
|
struct bkey_inode_buf packed_inode;
|
|
|
|
struct qstr lostfound = QSTR("lost+found");
|
|
|
|
const char *err = "cannot allocate memory";
|
|
|
|
struct bch_dev *ca;
|
|
|
|
LIST_HEAD(journal);
|
|
|
|
unsigned i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
bch_notice(c, "initializing new filesystem");
|
|
|
|
|
2019-01-24 20:25:40 -05:00
|
|
|
mutex_lock(&c->sb_lock);
|
2021-05-23 17:04:13 -04:00
|
|
|
c->disk_sb.sb->compat[0] |= cpu_to_le64(1ULL << BCH_COMPAT_extents_above_btree_updates_done);
|
|
|
|
c->disk_sb.sb->compat[0] |= cpu_to_le64(1ULL << BCH_COMPAT_bformat_overflow_done);
|
2020-06-03 16:20:22 -04:00
|
|
|
|
2021-03-21 16:20:40 -04:00
|
|
|
if (c->opts.version_upgrade) {
|
2021-05-23 17:04:13 -04:00
|
|
|
c->disk_sb.sb->version = cpu_to_le16(bcachefs_metadata_version_current);
|
|
|
|
c->disk_sb.sb->features[0] |= cpu_to_le64(BCH_SB_FEATURES_ALL);
|
2021-03-21 16:20:40 -04:00
|
|
|
bch2_write_super(c);
|
|
|
|
}
|
2020-06-03 16:20:22 -04:00
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
|
2019-02-10 19:16:55 -05:00
|
|
|
set_bit(BCH_FS_INITIAL_GC_DONE, &c->flags);
|
2022-02-19 02:48:27 -05:00
|
|
|
set_bit(BCH_FS_MAY_GO_RW, &c->flags);
|
2022-01-01 19:46:12 -05:00
|
|
|
set_bit(BCH_FS_FSCK_DONE, &c->flags);
|
2017-03-16 22:18:50 -08:00
|
|
|
|
2018-11-24 17:09:44 -05:00
|
|
|
for (i = 0; i < BTREE_ID_NR; i++)
|
|
|
|
bch2_btree_root_alloc(c, i);
|
|
|
|
|
2022-04-01 01:29:59 -04:00
|
|
|
for_each_online_member(ca, c, i)
|
|
|
|
bch2_dev_usage_init(ca);
|
|
|
|
|
2017-03-16 22:18:50 -08:00
|
|
|
err = "unable to allocate journal buckets";
|
2019-02-28 22:33:06 -05:00
|
|
|
for_each_online_member(ca, c, i) {
|
|
|
|
ret = bch2_dev_journal_alloc(ca);
|
|
|
|
if (ret) {
|
2017-03-16 22:18:50 -08:00
|
|
|
percpu_ref_put(&ca->io_ref);
|
|
|
|
goto err;
|
|
|
|
}
|
2019-02-28 22:33:06 -05:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* journal_res_get() will crash if called before this has
|
|
|
|
* set up the journal.pin FIFO and journal.cur pointer:
|
|
|
|
*/
|
2019-04-04 21:53:12 -04:00
|
|
|
bch2_fs_journal_start(&c->journal, 1, &journal);
|
2017-03-16 22:18:50 -08:00
|
|
|
bch2_journal_set_replay_done(&c->journal);
|
|
|
|
|
2020-10-16 21:36:26 -04:00
|
|
|
err = "error going read-write";
|
|
|
|
ret = bch2_fs_read_write_early(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write out the superblock and journal buckets, now that we can do
|
|
|
|
* btree updates
|
|
|
|
*/
|
2022-01-09 20:48:31 -05:00
|
|
|
bch_verbose(c, "marking superblocks");
|
2021-04-14 20:25:33 -04:00
|
|
|
err = "error marking superblock and journal";
|
|
|
|
for_each_member_device(ca, c, i) {
|
|
|
|
ret = bch2_trans_mark_dev_sb(c, ca);
|
2021-05-07 20:43:43 -04:00
|
|
|
if (ret) {
|
|
|
|
percpu_ref_put(&ca->ref);
|
2021-04-14 20:25:33 -04:00
|
|
|
goto err;
|
2021-05-07 20:43:43 -04:00
|
|
|
}
|
2021-12-24 04:22:20 -05:00
|
|
|
|
|
|
|
ca->new_fs_bucket_idx = 0;
|
2021-04-14 20:25:33 -04:00
|
|
|
}
|
2020-10-16 21:36:26 -04:00
|
|
|
|
2022-01-09 20:48:31 -05:00
|
|
|
bch_verbose(c, "initializing freespace");
|
2021-12-11 17:13:09 -05:00
|
|
|
err = "error initializing freespace";
|
|
|
|
ret = bch2_fs_freespace_init(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
2021-03-16 00:42:25 -04:00
|
|
|
err = "error creating root snapshot node";
|
|
|
|
ret = bch2_fs_initialize_subvolumes(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
bch_verbose(c, "reading snapshots table");
|
|
|
|
err = "error reading snapshots table";
|
|
|
|
ret = bch2_fs_snapshots_start(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
bch_verbose(c, "reading snapshots done");
|
|
|
|
|
2017-03-16 22:18:50 -08:00
|
|
|
bch2_inode_init(c, &root_inode, 0, 0,
|
|
|
|
S_IFDIR|S_IRWXU|S_IRUGO|S_IXUGO, 0, NULL);
|
2021-03-16 00:42:25 -04:00
|
|
|
root_inode.bi_inum = BCACHEFS_ROOT_INO;
|
|
|
|
root_inode.bi_subvol = BCACHEFS_ROOT_SUBVOL;
|
2020-11-05 23:39:33 -05:00
|
|
|
bch2_inode_pack(c, &packed_inode, &root_inode);
|
bcachefs: Start using bpos.snapshot field
This patch starts treating the bpos.snapshot field like part of the key
in the btree code:
* bpos_successor() and bpos_predecessor() now include the snapshot field
* Keys in btrees that will be using snapshots (extents, inodes, dirents
and xattrs) now always have their snapshot field set to U32_MAX
The btree iterator code gets a new flag, BTREE_ITER_ALL_SNAPSHOTS, that
determines whether we're iterating over keys in all snapshots or not -
internally, this controlls whether bkey_(successor|predecessor)
increment/decrement the snapshot field, or only the higher bits of the
key.
We add a new member to struct btree_iter, iter->snapshot: when
BTREE_ITER_ALL_SNAPSHOTS is not set, iter->pos.snapshot should always
equal iter->snapshot, which will be 0 for btrees that don't use
snapshots, and alsways U32_MAX for btrees that will use snapshots
(until we enable snapshot creation).
This patch also introduces a new metadata version number, and compat
code for reading from/writing to older versions - this isn't a forced
upgrade (yet).
Signed-off-by: Kent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
2021-03-24 18:02:16 -04:00
|
|
|
packed_inode.inode.k.p.snapshot = U32_MAX;
|
2017-03-16 22:18:50 -08:00
|
|
|
|
|
|
|
err = "error creating root directory";
|
2021-02-20 19:27:37 -05:00
|
|
|
ret = bch2_btree_insert(c, BTREE_ID_inodes,
|
2017-03-16 22:18:50 -08:00
|
|
|
&packed_inode.inode.k_i,
|
2020-10-16 21:36:26 -04:00
|
|
|
NULL, NULL, 0);
|
2017-03-16 22:18:50 -08:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
2019-10-02 18:35:36 -04:00
|
|
|
bch2_inode_init_early(c, &lostfound_inode);
|
2017-03-16 22:18:50 -08:00
|
|
|
|
|
|
|
err = "error creating lost+found";
|
2019-12-22 23:39:28 -05:00
|
|
|
ret = bch2_trans_do(c, NULL, NULL, 0,
|
2021-03-16 00:28:17 -04:00
|
|
|
bch2_create_trans(&trans,
|
|
|
|
BCACHEFS_ROOT_SUBVOL_INUM,
|
2019-10-02 18:35:36 -04:00
|
|
|
&root_inode, &lostfound_inode,
|
|
|
|
&lostfound,
|
2019-11-09 19:15:40 -08:00
|
|
|
0, 0, S_IFDIR|0700, 0,
|
2021-03-16 23:28:43 -04:00
|
|
|
NULL, NULL, (subvol_inum) { 0 }, 0));
|
2021-02-23 15:16:41 -05:00
|
|
|
if (ret) {
|
|
|
|
bch_err(c, "error creating lost+found");
|
2017-03-16 22:18:50 -08:00
|
|
|
goto err;
|
2021-02-23 15:16:41 -05:00
|
|
|
}
|
2017-03-16 22:18:50 -08:00
|
|
|
|
|
|
|
if (enabled_qtypes(c)) {
|
|
|
|
ret = bch2_fs_quota_read(c);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = "error writing first journal entry";
|
2021-11-15 17:30:11 -05:00
|
|
|
ret = bch2_journal_flush(&c->journal);
|
2017-03-16 22:18:50 -08:00
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
mutex_lock(&c->sb_lock);
|
|
|
|
SET_BCH_SB_INITIALIZED(c->disk_sb.sb, true);
|
|
|
|
SET_BCH_SB_CLEAN(c->disk_sb.sb, false);
|
|
|
|
|
|
|
|
bch2_write_super(c);
|
|
|
|
mutex_unlock(&c->sb_lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
err:
|
2018-11-04 22:09:51 -05:00
|
|
|
pr_err("Error initializing new filesystem: %s (%i)", err, ret);
|
2017-03-16 22:18:50 -08:00
|
|
|
return ret;
|
|
|
|
}
|