8fcdf81418
This improves copygc pipelining across multiple buckets: we now track each in flight bucket we're evacuating, with separate moving_contexts. This means that whereas previously we had to wait for outstanding moves to complete to ensure we didn't try to evacuate the same bucket twice, we can now just check buckets we want to evacuate against the pending list. This also mean we can run the verify_bucket_evacuated() check without killing pipelining - meaning it can now always be enabled, not just on debug builds. This is going to be important for the upcoming erasure coding work, where moving IOs that are being erasure coded will now skip the initial replication step; instead the IOs will wait on the stripe to complete. Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
90 lines
2.4 KiB
C
90 lines
2.4 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef _BCACHEFS_MOVE_H
|
|
#define _BCACHEFS_MOVE_H
|
|
|
|
#include "btree_iter.h"
|
|
#include "buckets.h"
|
|
#include "data_update.h"
|
|
#include "move_types.h"
|
|
|
|
struct bch_read_bio;
|
|
|
|
struct moving_context {
|
|
struct bch_fs *c;
|
|
struct bch_ratelimit *rate;
|
|
struct bch_move_stats *stats;
|
|
struct write_point_specifier wp;
|
|
bool wait_on_copygc;
|
|
bool write_error;
|
|
|
|
/* For waiting on outstanding reads and writes: */
|
|
struct closure cl;
|
|
struct list_head reads;
|
|
|
|
/* in flight sectors: */
|
|
atomic_t read_sectors;
|
|
atomic_t write_sectors;
|
|
atomic_t read_ios;
|
|
atomic_t write_ios;
|
|
|
|
wait_queue_head_t wait;
|
|
};
|
|
|
|
void bch2_verify_bucket_evacuated(struct btree_trans *, struct bpos, int);
|
|
|
|
#define move_ctxt_wait_event(_ctxt, _trans, _cond) \
|
|
do { \
|
|
bool cond_finished = false; \
|
|
bch2_moving_ctxt_do_pending_writes(_ctxt, _trans); \
|
|
\
|
|
if (_cond) \
|
|
break; \
|
|
__wait_event((_ctxt)->wait, \
|
|
bch2_moving_ctxt_next_pending_write(_ctxt) || \
|
|
(cond_finished = (_cond))); \
|
|
if (cond_finished) \
|
|
break; \
|
|
} while (1)
|
|
|
|
typedef bool (*move_pred_fn)(struct bch_fs *, void *, struct bkey_s_c,
|
|
struct bch_io_opts *, struct data_update_opts *);
|
|
|
|
void bch2_moving_ctxt_exit(struct moving_context *);
|
|
void bch2_moving_ctxt_init(struct moving_context *, struct bch_fs *,
|
|
struct bch_ratelimit *, struct bch_move_stats *,
|
|
struct write_point_specifier, bool);
|
|
struct moving_io *bch2_moving_ctxt_next_pending_write(struct moving_context *);
|
|
void bch2_moving_ctxt_do_pending_writes(struct moving_context *,
|
|
struct btree_trans *);
|
|
|
|
int bch2_scan_old_btree_nodes(struct bch_fs *, struct bch_move_stats *);
|
|
|
|
int bch2_move_data(struct bch_fs *,
|
|
enum btree_id, struct bpos,
|
|
enum btree_id, struct bpos,
|
|
struct bch_ratelimit *,
|
|
struct bch_move_stats *,
|
|
struct write_point_specifier,
|
|
bool,
|
|
move_pred_fn, void *);
|
|
|
|
int __bch2_evacuate_bucket(struct btree_trans *,
|
|
struct moving_context *,
|
|
struct move_bucket_in_flight *,
|
|
struct bpos, int,
|
|
struct data_update_opts);
|
|
int bch2_evacuate_bucket(struct bch_fs *, struct bpos, int,
|
|
struct data_update_opts,
|
|
struct bch_ratelimit *,
|
|
struct bch_move_stats *,
|
|
struct write_point_specifier,
|
|
bool);
|
|
int bch2_data_job(struct bch_fs *,
|
|
struct bch_move_stats *,
|
|
struct bch_ioctl_data);
|
|
|
|
void bch2_move_stats_init(struct bch_move_stats *stats, char *name);
|
|
|
|
|
|
#endif /* _BCACHEFS_MOVE_H */
|