diff options
author | Kent Overstreet <kent.overstreet@gmail.com> | 2018-11-23 02:06:18 -0500 |
---|---|---|
committer | Kent Overstreet <kent.overstreet@linux.dev> | 2023-10-22 17:08:12 -0400 |
commit | de5bb710f93fb87aef8303336a49d09323286822 (patch) | |
tree | 3f6fc1fc3fd7e1ff8531773225ebb21489331ca0 /fs | |
parent | 4e65431c855e959700cc9456f305fcfd94ee6241 (diff) | |
download | lwn-de5bb710f93fb87aef8303336a49d09323286822.tar.gz lwn-de5bb710f93fb87aef8303336a49d09323286822.zip |
bcachefs: shim for userspace raid library
Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/bcachefs/ec.c | 127 |
1 files changed, 73 insertions, 54 deletions
diff --git a/fs/bcachefs/ec.c b/fs/bcachefs/ec.c index f6314aa6a0f1..727324f15f43 100644 --- a/fs/bcachefs/ec.c +++ b/fs/bcachefs/ec.c @@ -16,9 +16,79 @@ #include "super-io.h" #include "util.h" +#include <linux/sort.h> + +#ifdef __KERNEL__ + #include <linux/raid/pq.h> #include <linux/raid/xor.h> -#include <linux/sort.h> + +static void raid5_recov(unsigned disks, unsigned failed_idx, + size_t size, void **data) +{ + unsigned i = 2, nr; + + BUG_ON(failed_idx >= disks); + + swap(data[0], data[failed_idx]); + memcpy(data[0], data[1], size); + + while (i < disks) { + nr = min_t(unsigned, disks - i, MAX_XOR_BLOCKS); + xor_blocks(nr, size, data[0], data + i); + i += nr; + } + + swap(data[0], data[failed_idx]); +} + +static void raid_gen(int nd, int np, size_t size, void **v) +{ + if (np >= 1) + raid5_recov(nd + np, nd, size, v); + if (np >= 2) + raid6_call.gen_syndrome(nd + np, size, v); + BUG_ON(np > 2); +} + +static void raid_rec(int nr, int *ir, int nd, int np, size_t size, void **v) +{ + switch (nr) { + case 0: + break; + case 1: + if (ir[0] < nd + 1) + raid5_recov(nd + 1, ir[0], size, v); + else + raid6_call.gen_syndrome(nd + np, size, v); + break; + case 2: + if (ir[1] < nd) { + /* data+data failure. */ + raid6_2data_recov(nd + np, size, ir[0], ir[1], v); + } else if (ir[0] < nd) { + /* data + p/q failure */ + + if (ir[1] == nd) /* data + p failure */ + raid6_datap_recov(nd + np, size, ir[0], v); + else { /* data + q failure */ + raid5_recov(nd + 1, ir[0], size, v); + raid6_call.gen_syndrome(nd + np, size, v); + } + } else { + raid_gen(nd, np, size, v); + } + break; + default: + BUG(); + } +} + +#else + +#include <raid/raid.h> + +#endif struct ec_bio { struct bch_dev *ca; @@ -251,41 +321,13 @@ static void ec_validate_checksums(struct bch_fs *c, struct ec_stripe_buf *buf) /* Erasure coding: */ -static void raid5_recov(unsigned disks, unsigned bytes, - unsigned failed, void **data) -{ - unsigned i = 2, nr; - - BUG_ON(failed >= disks); - - swap(data[0], data[failed]); - memcpy(data[0], data[1], bytes); - - while (i < disks) { - nr = min_t(unsigned, disks - i, MAX_XOR_BLOCKS); - xor_blocks(nr, bytes, data[0], data + i); - i += nr; - } - - swap(data[0], data[failed]); -} - static void ec_generate_ec(struct ec_stripe_buf *buf) { struct bch_stripe *v = &buf->key.v; unsigned nr_data = v->nr_blocks - v->nr_redundant; unsigned bytes = le16_to_cpu(v->sectors) << 9; - switch (v->nr_redundant) { - case 2: - raid6_call.gen_syndrome(v->nr_blocks, bytes, buf->data); - fallthrough; - case 1: - raid5_recov(v->nr_blocks, bytes, nr_data, buf->data); - break; - default: - BUG(); - } + raid_gen(nr_data, v->nr_redundant, bytes, buf->data); } static unsigned __ec_nr_failed(struct ec_stripe_buf *buf, unsigned nr) @@ -315,30 +357,7 @@ static int ec_do_recov(struct bch_fs *c, struct ec_stripe_buf *buf) if (!test_bit(i, buf->valid)) failed[nr_failed++] = i; - switch (nr_failed) { - case 0: - break; - case 1: - if (test_bit(nr_data, buf->valid)) - raid5_recov(nr_data + 1, bytes, failed[0], buf->data); - else - raid6_datap_recov(v->nr_blocks, bytes, failed[0], buf->data); - break; - case 2: - /* data+data failure. */ - raid6_2data_recov(v->nr_blocks, bytes, failed[0], failed[1], buf->data); - break; - - default: - BUG(); - } - - for (i = nr_data; i < v->nr_blocks; i++) - if (!test_bit(i, buf->valid)) { - ec_generate_ec(buf); - break; - } - + raid_rec(nr_failed, failed, nr_data, v->nr_redundant, bytes, buf->data); return 0; } |