#include "bcachefs.h"
#include "alloc_background.h"
#include "alloc_foreground.h"
#include "backpointers.h"
#include "bkey_buf.h"
#include "bset.h"
#include "btree_gc.h"
#include "btree_update.h"
#include "btree_write_buffer.h"
#include "buckets.h"
#include "checksum.h"
#include "disk_accounting.h"
#include "disk_groups.h"
#include "ec.h"
#include "error.h"
#include "io_read.h"
#include "io_write.h"
#include "keylist.h"
#include "recovery.h"
#include "replicas.h"
#include "super-io.h"
#include "util.h"
#include <linux/sort.h>
#ifdef __KERNEL__
#include <linux/raid/pq.h>
#include <linux/raid/xor.h>
static void raid5_recov(unsigned disks, unsigned failed_idx,
size_t size, void **data)
{ … }
static void raid_gen(int nd, int np, size_t size, void **v)
{ … }
static void raid_rec(int nr, int *ir, int nd, int np, size_t size, void **v)
{ … }
#else
#include <raid/raid.h>
#endif
struct ec_bio { … };
int bch2_stripe_validate(struct bch_fs *c, struct bkey_s_c k,
enum bch_validate_flags flags)
{ … }
void bch2_stripe_to_text(struct printbuf *out, struct bch_fs *c,
struct bkey_s_c k)
{ … }
static int __mark_stripe_bucket(struct btree_trans *trans,
struct bch_dev *ca,
struct bkey_s_c_stripe s,
unsigned ptr_idx, bool deleting,
struct bpos bucket,
struct bch_alloc_v4 *a,
enum btree_iter_update_trigger_flags flags)
{ … }
static int mark_stripe_bucket(struct btree_trans *trans,
struct bkey_s_c_stripe s,
unsigned ptr_idx, bool deleting,
enum btree_iter_update_trigger_flags flags)
{ … }
static int mark_stripe_buckets(struct btree_trans *trans,
struct bkey_s_c old, struct bkey_s_c new,
enum btree_iter_update_trigger_flags flags)
{ … }
static inline void stripe_to_mem(struct stripe *m, const struct bch_stripe *s)
{ … }
int bch2_trigger_stripe(struct btree_trans *trans,
enum btree_id btree, unsigned level,
struct bkey_s_c old, struct bkey_s _new,
enum btree_iter_update_trigger_flags flags)
{ … }
static const struct bch_extent_ptr *bkey_matches_stripe(struct bch_stripe *s,
struct bkey_s_c k, unsigned *block)
{ … }
static bool extent_has_stripe_ptr(struct bkey_s_c k, u64 idx)
{ … }
static void ec_stripe_buf_exit(struct ec_stripe_buf *buf)
{ … }
static int ec_stripe_buf_init(struct ec_stripe_buf *buf,
unsigned offset, unsigned size)
{ … }
static struct bch_csum ec_block_checksum(struct ec_stripe_buf *buf,
unsigned block, unsigned offset)
{ … }
static void ec_generate_checksums(struct ec_stripe_buf *buf)
{ … }
static void ec_validate_checksums(struct bch_fs *c, struct ec_stripe_buf *buf)
{ … }
static void ec_generate_ec(struct ec_stripe_buf *buf)
{ … }
static unsigned ec_nr_failed(struct ec_stripe_buf *buf)
{ … }
static int ec_do_recov(struct bch_fs *c, struct ec_stripe_buf *buf)
{ … }
static void ec_block_endio(struct bio *bio)
{ … }
static void ec_block_io(struct bch_fs *c, struct ec_stripe_buf *buf,
blk_opf_t opf, unsigned idx, struct closure *cl)
{ … }
static int get_stripe_key_trans(struct btree_trans *trans, u64 idx,
struct ec_stripe_buf *stripe)
{ … }
int bch2_ec_read_extent(struct btree_trans *trans, struct bch_read_bio *rbio,
struct bkey_s_c orig_k)
{ … }
static int __ec_stripe_mem_alloc(struct bch_fs *c, size_t idx, gfp_t gfp)
{ … }
static int ec_stripe_mem_alloc(struct btree_trans *trans,
struct btree_iter *iter)
{ … }
static bool __bch2_stripe_is_open(struct bch_fs *c, u64 idx)
{ … }
static bool bch2_stripe_is_open(struct bch_fs *c, u64 idx)
{ … }
static bool bch2_try_open_stripe(struct bch_fs *c,
struct ec_stripe_new *s,
u64 idx)
{ … }
static void bch2_stripe_close(struct bch_fs *c, struct ec_stripe_new *s)
{ … }
static u64 stripe_idx_to_delete(struct bch_fs *c)
{ … }
static inline void ec_stripes_heap_set_backpointer(ec_stripes_heap *h,
size_t i)
{ … }
static inline bool ec_stripes_heap_cmp(const void *l, const void *r, void __always_unused *args)
{ … }
static inline void ec_stripes_heap_swap(void *l, void *r, void *h)
{ … }
static void heap_verify_backpointer(struct bch_fs *c, size_t idx)
{ … }
void bch2_stripes_heap_del(struct bch_fs *c,
struct stripe *m, size_t idx)
{ … }
void bch2_stripes_heap_insert(struct bch_fs *c,
struct stripe *m, size_t idx)
{ … }
void bch2_stripes_heap_update(struct bch_fs *c,
struct stripe *m, size_t idx)
{ … }
static int ec_stripe_delete(struct btree_trans *trans, u64 idx)
{ … }
static void ec_stripe_delete_work(struct work_struct *work)
{ … }
void bch2_do_stripe_deletes(struct bch_fs *c)
{ … }
static int ec_stripe_key_update(struct btree_trans *trans,
struct bkey_i_stripe *old,
struct bkey_i_stripe *new)
{ … }
static int ec_stripe_update_extent(struct btree_trans *trans,
struct bch_dev *ca,
struct bpos bucket, u8 gen,
struct ec_stripe_buf *s,
struct bpos *bp_pos)
{ … }
static int ec_stripe_update_bucket(struct btree_trans *trans, struct ec_stripe_buf *s,
unsigned block)
{ … }
static int ec_stripe_update_extents(struct bch_fs *c, struct ec_stripe_buf *s)
{ … }
static void zero_out_rest_of_ec_bucket(struct bch_fs *c,
struct ec_stripe_new *s,
unsigned block,
struct open_bucket *ob)
{ … }
void bch2_ec_stripe_new_free(struct bch_fs *c, struct ec_stripe_new *s)
{ … }
static void ec_stripe_create(struct ec_stripe_new *s)
{ … }
static struct ec_stripe_new *get_pending_stripe(struct bch_fs *c)
{ … }
static void ec_stripe_create_work(struct work_struct *work)
{ … }
void bch2_ec_do_stripe_creates(struct bch_fs *c)
{ … }
static void ec_stripe_new_set_pending(struct bch_fs *c, struct ec_stripe_head *h)
{ … }
static void ec_stripe_new_cancel(struct bch_fs *c, struct ec_stripe_head *h, int err)
{ … }
void bch2_ec_bucket_cancel(struct bch_fs *c, struct open_bucket *ob)
{ … }
void *bch2_writepoint_ec_buf(struct bch_fs *c, struct write_point *wp)
{ … }
static int unsigned_cmp(const void *_l, const void *_r)
{ … }
static unsigned pick_blocksize(struct bch_fs *c,
struct bch_devs_mask *devs)
{ … }
static bool may_create_new_stripe(struct bch_fs *c)
{ … }
static void ec_stripe_key_init(struct bch_fs *c,
struct bkey_i *k,
unsigned nr_data,
unsigned nr_parity,
unsigned stripe_size,
unsigned disk_label)
{ … }
static int ec_new_stripe_alloc(struct bch_fs *c, struct ec_stripe_head *h)
{ … }
static void ec_stripe_head_devs_update(struct bch_fs *c, struct ec_stripe_head *h)
{ … }
static struct ec_stripe_head *
ec_new_stripe_head_alloc(struct bch_fs *c, unsigned disk_label,
unsigned algo, unsigned redundancy,
enum bch_watermark watermark)
{ … }
void bch2_ec_stripe_head_put(struct bch_fs *c, struct ec_stripe_head *h)
{ … }
static struct ec_stripe_head *
__bch2_ec_stripe_head_get(struct btree_trans *trans,
unsigned disk_label,
unsigned algo,
unsigned redundancy,
enum bch_watermark watermark)
{ … }
static int new_stripe_alloc_buckets(struct btree_trans *trans, struct ec_stripe_head *h,
enum bch_watermark watermark, struct closure *cl)
{ … }
static s64 get_existing_stripe(struct bch_fs *c,
struct ec_stripe_head *head)
{ … }
static int __bch2_ec_stripe_head_reuse(struct btree_trans *trans, struct ec_stripe_head *h)
{ … }
static int __bch2_ec_stripe_head_reserve(struct btree_trans *trans, struct ec_stripe_head *h)
{ … }
struct ec_stripe_head *bch2_ec_stripe_head_get(struct btree_trans *trans,
unsigned target,
unsigned algo,
unsigned redundancy,
enum bch_watermark watermark,
struct closure *cl)
{ … }
static int bch2_invalidate_stripe_to_dev(struct btree_trans *trans, struct bkey_s_c k_a)
{ … }
int bch2_dev_remove_stripes(struct bch_fs *c, unsigned dev_idx)
{ … }
static void __bch2_ec_stop(struct bch_fs *c, struct bch_dev *ca)
{ … }
void bch2_ec_stop_dev(struct bch_fs *c, struct bch_dev *ca)
{ … }
void bch2_fs_ec_stop(struct bch_fs *c)
{ … }
static bool bch2_fs_ec_flush_done(struct bch_fs *c)
{ … }
void bch2_fs_ec_flush(struct bch_fs *c)
{ … }
int bch2_stripes_read(struct bch_fs *c)
{ … }
void bch2_stripes_heap_to_text(struct printbuf *out, struct bch_fs *c)
{ … }
static void bch2_new_stripe_to_text(struct printbuf *out, struct bch_fs *c,
struct ec_stripe_new *s)
{ … }
void bch2_new_stripes_to_text(struct printbuf *out, struct bch_fs *c)
{ … }
void bch2_fs_ec_exit(struct bch_fs *c)
{ … }
void bch2_fs_ec_init_early(struct bch_fs *c)
{ … }
int bch2_fs_ec_init(struct bch_fs *c)
{ … }