summaryrefslogtreecommitdiff
path: root/fs/bcachefs/fs-io.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/bcachefs/fs-io.c')
-rw-r--r--fs/bcachefs/fs-io.c189
1 files changed, 109 insertions, 80 deletions
diff --git a/fs/bcachefs/fs-io.c b/fs/bcachefs/fs-io.c
index ad0b0c136e65..14abb7aad18f 100644
--- a/fs/bcachefs/fs-io.c
+++ b/fs/bcachefs/fs-io.c
@@ -1719,7 +1719,17 @@ int bch2_write_end(struct file *file, struct address_space *mapping,
return copied;
}
-#define WRITE_BATCH_PAGES 32
+typedef DARRAY(struct folio *) folios;
+
+static noinline void folios_trunc(folios *folios, struct folio **fi)
+{
+ while (folios->data + folios->nr > fi) {
+ struct folio *f = darray_pop(folios);
+
+ folio_unlock(f);
+ folio_put(f);
+ }
+}
static int __bch2_buffered_write(struct bch_inode_info *inode,
struct address_space *mapping,
@@ -1727,64 +1737,73 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
loff_t pos, unsigned len)
{
struct bch_fs *c = inode->v.i_sb->s_fs_info;
- struct folio *folios[WRITE_BATCH_PAGES];
struct bch2_folio_reservation res;
- unsigned long index = pos >> PAGE_SHIFT;
- unsigned offset = pos & (PAGE_SIZE - 1);
- unsigned nr_folios = DIV_ROUND_UP(offset + len, PAGE_SIZE);
- unsigned i, reserved = 0, set_dirty = 0;
- unsigned copied = 0, nr_folios_copied = 0;
+ folios folios;
+ struct folio **fi, *f;
+ unsigned copied = 0, f_offset;
+ loff_t end = pos + len, f_pos;
int ret = 0;
BUG_ON(!len);
- BUG_ON(nr_folios > ARRAY_SIZE(folios));
bch2_folio_reservation_init(c, inode, &res);
+ darray_init(&folios);
- for (i = 0; i < nr_folios; i++) {
- folios[i] = __filemap_get_folio(mapping, index + i,
- FGP_LOCK|FGP_WRITE|FGP_CREAT|FGP_STABLE,
- mapping_gfp_mask(mapping));
- if (!folios[i]) {
- nr_folios = i;
- if (!i) {
- ret = -ENOMEM;
- goto out;
- }
- len = min_t(unsigned, len,
- nr_folios * PAGE_SIZE - offset);
+ f_pos = pos;
+ while (f_pos < end) {
+ unsigned fgp_flags = FGP_LOCK|FGP_WRITE|FGP_STABLE;
+
+ if ((u64) f_pos < (u64) pos + (1U << 20))
+ fgp_flags |= FGP_CREAT;
+
+ if (darray_make_room_gfp(&folios, 1,
+ mapping_gfp_mask(mapping) & GFP_KERNEL))
break;
- }
+
+ f = __filemap_get_folio(mapping, f_pos >> PAGE_SHIFT,
+ fgp_flags, mapping_gfp_mask(mapping));
+ if (!f)
+ break;
+
+ BUG_ON(folios.nr && folio_pos(f) != f_pos);
+
+ f_pos = folio_end_pos(f);
+ darray_push(&folios, f);
+ }
+
+ end = min(end, f_pos);
+ if (end == pos) {
+ ret = -ENOMEM;
+ goto out;
}
- if (offset && !folio_test_uptodate(folios[0])) {
- ret = bch2_read_single_folio(folios[0], mapping);
+ f = darray_first(folios);
+ if (pos != folio_pos(f) && !folio_test_uptodate(f)) {
+ ret = bch2_read_single_folio(f, mapping);
if (ret)
goto out;
}
- if ((pos + len) & (PAGE_SIZE - 1) &&
- !folio_test_uptodate(folios[nr_folios - 1])) {
- if ((index + nr_folios - 1) << PAGE_SHIFT >= inode->v.i_size) {
- folio_zero_range(folios[nr_folios - 1], 0,
- folio_size(folios[nr_folios - 1]));
+ f = darray_last(folios);
+ if (end != folio_end_pos(f) && !folio_test_uptodate(f)) {
+ if (end >= inode->v.i_size) {
+ folio_zero_range(f, 0, folio_size(f));
} else {
- ret = bch2_read_single_folio(folios[nr_folios - 1], mapping);
+ ret = bch2_read_single_folio(f, mapping);
if (ret)
goto out;
}
}
- while (reserved < len) {
- unsigned i = (offset + reserved) >> PAGE_SHIFT;
- struct folio *folio = folios[i];
- unsigned folio_offset = (offset + reserved) & (PAGE_SIZE - 1);
- unsigned folio_len = min_t(unsigned, len - reserved,
- PAGE_SIZE - folio_offset);
+ f_pos = pos;
+ f_offset = pos - folio_pos(darray_first(folios));
+ darray_for_each(folios, fi) {
+ struct folio *f = *fi;
+ unsigned f_len = min(end, folio_end_pos(f)) - f_pos;
- if (!bch2_folio_create(folio, __GFP_NOFAIL)->uptodate) {
- ret = bch2_folio_set(c, inode_inum(inode),
- folios + i, nr_folios - i);
+ if (!bch2_folio_create(f, __GFP_NOFAIL)->uptodate) {
+ ret = bch2_folio_set(c, inode_inum(inode), fi,
+ folios.data + folios.nr - fi);
if (ret)
goto out;
}
@@ -1797,78 +1816,89 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
* we aren't completely out of disk space - we don't do that
* yet:
*/
- ret = bch2_folio_reservation_get(c, inode, folio, &res,
- folio_offset, folio_len);
+ ret = bch2_folio_reservation_get(c, inode, f, &res, f_offset, f_len);
if (unlikely(ret)) {
- if (!reserved)
+ folios_trunc(&folios, fi);
+ if (!folios.nr)
goto out;
+
+ end = min(end, folio_end_pos(darray_last(folios)));
break;
}
- reserved += folio_len;
+ f_pos = folio_end_pos(f);
+ f_offset = 0;
}
if (mapping_writably_mapped(mapping))
- for (i = 0; i < nr_folios; i++)
- flush_dcache_folio(folios[i]);
-
- while (copied < reserved) {
- struct folio *folio = folios[(offset + copied) >> PAGE_SHIFT];
- unsigned folio_offset = (offset + copied) & (PAGE_SIZE - 1);
- unsigned folio_len = min_t(unsigned, reserved - copied,
- PAGE_SIZE - folio_offset);
- unsigned folio_copied = copy_page_from_iter_atomic(&folio->page,
- folio_offset, folio_len, iter);
-
- if (!folio_copied)
+ darray_for_each(folios, fi)
+ flush_dcache_folio(*fi);
+
+ f_pos = pos;
+ f_offset = pos - folio_pos(darray_first(folios));
+ darray_for_each(folios, fi) {
+ struct folio *f = *fi;
+ unsigned f_len = min(end, folio_end_pos(f)) - f_pos;
+ unsigned f_copied = copy_page_from_iter_atomic(&f->page, f_offset, f_len, iter);
+
+ if (!f_copied) {
+ folios_trunc(&folios, fi);
break;
+ }
- if (!folio_test_uptodate(folio) &&
- folio_copied != PAGE_SIZE &&
- pos + copied + folio_copied < inode->v.i_size) {
- folio_zero_range(folio, 0, folio_size(folio));
+ if (!folio_test_uptodate(f) &&
+ f_copied != folio_size(f) &&
+ pos + copied + f_copied < inode->v.i_size) {
+ folio_zero_range(f, 0, folio_size(f));
+ folios_trunc(&folios, fi);
break;
}
- flush_dcache_folio(folio);
- copied += folio_copied;
+ flush_dcache_folio(f);
+ copied += f_copied;
- if (folio_copied != folio_len)
+ if (f_copied != f_len) {
+ folios_trunc(&folios, fi + 1);
break;
+ }
+
+ f_pos = folio_end_pos(f);
+ f_offset = 0;
}
if (!copied)
goto out;
+ end = pos + copied;
+
spin_lock(&inode->v.i_lock);
- if (pos + copied > inode->v.i_size)
- i_size_write(&inode->v, pos + copied);
+ if (end > inode->v.i_size)
+ i_size_write(&inode->v, end);
spin_unlock(&inode->v.i_lock);
- while (set_dirty < copied) {
- struct folio *folio = folios[(offset + set_dirty) >> PAGE_SHIFT];
- unsigned folio_offset = (offset + set_dirty) & (PAGE_SIZE - 1);
- unsigned folio_len = min_t(unsigned, copied - set_dirty,
- PAGE_SIZE - folio_offset);
+ f_pos = pos;
+ f_offset = pos - folio_pos(darray_first(folios));
+ darray_for_each(folios, fi) {
+ struct folio *f = *fi;
+ unsigned f_len = min(end, folio_end_pos(f)) - f_pos;
- if (!folio_test_uptodate(folio))
- folio_mark_uptodate(folio);
+ if (!folio_test_uptodate(f))
+ folio_mark_uptodate(f);
- bch2_set_folio_dirty(c, inode, folio, &res, folio_offset, folio_len);
- folio_unlock(folio);
- folio_put(folio);
+ bch2_set_folio_dirty(c, inode, f, &res, f_offset, f_len);
- set_dirty += folio_len;
+ f_pos = folio_end_pos(f);
+ f_offset = 0;
}
- nr_folios_copied = DIV_ROUND_UP(offset + copied, PAGE_SIZE);
inode->ei_last_dirtied = (unsigned long) current;
out:
- for (i = nr_folios_copied; i < nr_folios; i++) {
- folio_unlock(folios[i]);
- folio_put(folios[i]);
+ darray_for_each(folios, fi) {
+ folio_unlock(*fi);
+ folio_put(*fi);
}
+ darray_exit(&folios);
bch2_folio_reservation_put(c, inode, &res);
return copied ?: ret;
@@ -1887,8 +1917,7 @@ static ssize_t bch2_buffered_write(struct kiocb *iocb, struct iov_iter *iter)
do {
unsigned offset = pos & (PAGE_SIZE - 1);
- unsigned bytes = min_t(unsigned long, iov_iter_count(iter),
- PAGE_SIZE * WRITE_BATCH_PAGES - offset);
+ unsigned bytes = iov_iter_count(iter);
again:
/*
* Bring in the user page that we will copy from _first_.