bcachefs: Fix moving compressed data

bio_uncompress_inplace() used to potentially need to extend the bio to
be big enough for the uncompressed data, which has become problematic
with multipage bvecs - but, the move extent path actually already
allocated the bios to be big enough for the uncompressed data.

The promote path needed to be fixed, though.

Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
This commit is contained in:
Kent Overstreet 2019-07-09 11:16:33 -04:00 committed by Kent Overstreet
parent 0fd7263e2e
commit a6f4d5bb24
2 changed files with 34 additions and 39 deletions

View File

@ -241,19 +241,10 @@ int bch2_bio_uncompress_inplace(struct bch_fs *c, struct bio *bio,
} }
/* /*
* might have to free existing pages and retry allocation from mempool - * XXX: don't have a good way to assert that the bio was allocated with
* do this _after_ decompressing: * enough space, we depend on bch2_move_extent doing the right thing
*/ */
if (bio->bi_iter.bi_size < crc->live_size << 9) { bio->bi_iter.bi_size = crc->live_size << 9;
if (bch2_bio_alloc_pages(bio, (crc->live_size << 9) -
bio->bi_iter.bi_size,
GFP_NOFS)) {
bch2_bio_free_pages_pool(c, bio);
bio->bi_iter.bi_size = 0;
bio->bi_vcnt = 0;
bch2_bio_alloc_pages_pool(c, bio, crc->live_size << 9);
}
}
memcpy_to_bio(bio, bio->bi_iter, data.b + (crc->offset << 9)); memcpy_to_bio(bio, bio->bi_iter, data.b + (crc->offset << 9));

View File

@ -1039,22 +1039,18 @@ static struct promote_op *__promote_alloc(struct bch_fs *c,
struct bpos pos, struct bpos pos,
struct extent_ptr_decoded *pick, struct extent_ptr_decoded *pick,
struct bch_io_opts opts, struct bch_io_opts opts,
unsigned rbio_sectors, unsigned sectors,
struct bch_read_bio **rbio) struct bch_read_bio **rbio)
{ {
struct promote_op *op = NULL; struct promote_op *op = NULL;
struct bio *bio; struct bio *bio;
unsigned rbio_pages = DIV_ROUND_UP(rbio_sectors, PAGE_SECTORS); unsigned pages = DIV_ROUND_UP(sectors, PAGE_SECTORS);
/* data might have to be decompressed in the write path: */
unsigned wbio_pages = DIV_ROUND_UP(pick->crc.uncompressed_size,
PAGE_SECTORS);
int ret; int ret;
if (!percpu_ref_tryget(&c->writes)) if (!percpu_ref_tryget(&c->writes))
return NULL; return NULL;
op = kzalloc(sizeof(*op) + sizeof(struct bio_vec) * wbio_pages, op = kzalloc(sizeof(*op) + sizeof(struct bio_vec) * pages, GFP_NOIO);
GFP_NOIO);
if (!op) if (!op)
goto err; goto err;
@ -1062,34 +1058,32 @@ static struct promote_op *__promote_alloc(struct bch_fs *c,
op->pos = pos; op->pos = pos;
/* /*
* promotes require bouncing, but if the extent isn't * We don't use the mempool here because extents that aren't
* checksummed/compressed it might be too big for the mempool: * checksummed or compressed can be too big for the mempool:
*/ */
if (rbio_sectors > c->sb.encoded_extent_max) { *rbio = kzalloc(sizeof(struct bch_read_bio) +
*rbio = kzalloc(sizeof(struct bch_read_bio) + sizeof(struct bio_vec) * pages,
sizeof(struct bio_vec) * rbio_pages, GFP_NOIO);
GFP_NOIO); if (!*rbio)
if (!*rbio) goto err;
goto err;
rbio_init(&(*rbio)->bio, opts); rbio_init(&(*rbio)->bio, opts);
bio_init(&(*rbio)->bio, NULL, (*rbio)->bio.bi_inline_vecs, rbio_pages, 0); bio_init(&(*rbio)->bio, NULL, (*rbio)->bio.bi_inline_vecs, pages, 0);
if (bch2_bio_alloc_pages(&(*rbio)->bio, rbio_sectors << 9, if (bch2_bio_alloc_pages(&(*rbio)->bio, sectors << 9,
GFP_NOIO)) GFP_NOIO))
goto err; goto err;
(*rbio)->bounce = true; (*rbio)->bounce = true;
(*rbio)->split = true; (*rbio)->split = true;
(*rbio)->kmalloc = true; (*rbio)->kmalloc = true;
}
if (rhashtable_lookup_insert_fast(&c->promote_table, &op->hash, if (rhashtable_lookup_insert_fast(&c->promote_table, &op->hash,
bch_promote_params)) bch_promote_params))
goto err; goto err;
bio = &op->write.op.wbio.bio; bio = &op->write.op.wbio.bio;
bio_init(bio, NULL, bio->bi_inline_vecs, wbio_pages, 0); bio_init(bio, NULL, bio->bi_inline_vecs, pages, 0);
ret = bch2_migrate_write_init(c, &op->write, ret = bch2_migrate_write_init(c, &op->write,
writepoint_hashed((unsigned long) current), writepoint_hashed((unsigned long) current),
@ -1123,8 +1117,9 @@ static inline struct promote_op *promote_alloc(struct bch_fs *c,
bool *read_full) bool *read_full)
{ {
bool promote_full = *read_full || READ_ONCE(c->promote_whole_extents); bool promote_full = *read_full || READ_ONCE(c->promote_whole_extents);
/* data might have to be decompressed in the write path: */
unsigned sectors = promote_full unsigned sectors = promote_full
? pick->crc.compressed_size ? max(pick->crc.compressed_size, pick->crc.live_size)
: bvec_iter_sectors(iter); : bvec_iter_sectors(iter);
struct bpos pos = promote_full struct bpos pos = promote_full
? bkey_start_pos(k.k) ? bkey_start_pos(k.k)
@ -1659,7 +1654,16 @@ int __bch2_read_extent(struct bch_fs *c, struct bch_read_bio *orig,
} }
if (rbio) { if (rbio) {
/* promote already allocated bounce rbio */ /*
* promote already allocated bounce rbio:
* promote needs to allocate a bio big enough for uncompressing
* data in the write path, but we're not going to use it all
* here:
*/
BUG_ON(rbio->bio.bi_iter.bi_size <
pick.crc.compressed_size << 9);
rbio->bio.bi_iter.bi_size =
pick.crc.compressed_size << 9;
} else if (bounce) { } else if (bounce) {
unsigned sectors = pick.crc.compressed_size; unsigned sectors = pick.crc.compressed_size;