Commit 6b9857b2 authored by Brian Foster's avatar Brian Foster Committed by Kent Overstreet

bcachefs: use u64 for folio end pos to avoid overflows

Some of the folio_end_*() helpers are prone to overflow of signed
64-bit types because the mapping is only limited by the max value of
loff_t and the associated helpers return the start offset of the
next folio. Therefore, a folio_end_pos() of the max allowable folio in a
mapping returns a value that overflows loff_t.

This makes it hard to rely on such values when doing folio
processing across a range of a file, as bcachefs attempts to do with
the recent folio changes. For example, generic/564 causes problems
in the buffered write path when testing writes at max boundary
conditions.

The current understanding is that the pagecache historically limited
the mapping to one less page to avoid this problem and this was
dropped with some of the folio conversions, but may be reinstated to
properly address the problem. In the meantime, update the internal
folio_end_*() helpers in bcachefs to return a u64, and all of the
associated code to use or cast to u64 to avoid overflow problems.
This allows generic/564 to pass and can be reverted back to using
loff_t if at any point the pagecache subsystem can guarantee these
boundary conditions will not overflow.
Signed-off-by: default avatarBrian Foster <bfoster@redhat.com>
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 335f7d4f
...@@ -78,7 +78,13 @@ static inline struct folio_vec bio_iter_iovec_folio(struct bio *bio, ...@@ -78,7 +78,13 @@ static inline struct folio_vec bio_iter_iovec_folio(struct bio *bio,
#define bio_for_each_folio(bvl, bio, iter) \ #define bio_for_each_folio(bvl, bio, iter) \
__bio_for_each_folio(bvl, bio, iter, (bio)->bi_iter) __bio_for_each_folio(bvl, bio, iter, (bio)->bi_iter)
static inline loff_t folio_end_pos(struct folio *folio) /*
* Use u64 for the end pos and sector helpers because if the folio covers the
* max supported range of the mapping, the start offset of the next folio
* overflows loff_t. This breaks much of the range based processing in the
* buffered write path.
*/
static inline u64 folio_end_pos(struct folio *folio)
{ {
return folio_pos(folio) + folio_size(folio); return folio_pos(folio) + folio_size(folio);
} }
...@@ -93,7 +99,7 @@ static inline loff_t folio_sector(struct folio *folio) ...@@ -93,7 +99,7 @@ static inline loff_t folio_sector(struct folio *folio)
return folio_pos(folio) >> 9; return folio_pos(folio) >> 9;
} }
static inline loff_t folio_end_sector(struct folio *folio) static inline u64 folio_end_sector(struct folio *folio)
{ {
return folio_end_pos(folio) >> 9; return folio_end_pos(folio) >> 9;
} }
...@@ -101,12 +107,12 @@ static inline loff_t folio_end_sector(struct folio *folio) ...@@ -101,12 +107,12 @@ static inline loff_t folio_end_sector(struct folio *folio)
typedef DARRAY(struct folio *) folios; typedef DARRAY(struct folio *) folios;
static int filemap_get_contig_folios_d(struct address_space *mapping, static int filemap_get_contig_folios_d(struct address_space *mapping,
loff_t start, loff_t end, loff_t start, u64 end,
int fgp_flags, gfp_t gfp, int fgp_flags, gfp_t gfp,
folios *folios) folios *folios)
{ {
struct folio *f; struct folio *f;
loff_t pos = start; u64 pos = start;
int ret = 0; int ret = 0;
while (pos < end) { while (pos < end) {
...@@ -1859,7 +1865,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode, ...@@ -1859,7 +1865,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
folios folios; folios folios;
struct folio **fi, *f; struct folio **fi, *f;
unsigned copied = 0, f_offset; unsigned copied = 0, f_offset;
loff_t end = pos + len, f_pos; u64 end = pos + len, f_pos;
loff_t last_folio_pos = inode->v.i_size; loff_t last_folio_pos = inode->v.i_size;
int ret = 0; int ret = 0;
...@@ -1901,7 +1907,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode, ...@@ -1901,7 +1907,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
f_offset = pos - folio_pos(darray_first(folios)); f_offset = pos - folio_pos(darray_first(folios));
darray_for_each(folios, fi) { darray_for_each(folios, fi) {
struct folio *f = *fi; struct folio *f = *fi;
unsigned f_len = min(end, folio_end_pos(f)) - f_pos; u64 f_len = min(end, folio_end_pos(f)) - f_pos;
if (!bch2_folio_create(f, __GFP_NOFAIL)->uptodate) { if (!bch2_folio_create(f, __GFP_NOFAIL)->uptodate) {
ret = bch2_folio_set(c, inode_inum(inode), fi, ret = bch2_folio_set(c, inode_inum(inode), fi,
...@@ -1940,7 +1946,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode, ...@@ -1940,7 +1946,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
f_offset = pos - folio_pos(darray_first(folios)); f_offset = pos - folio_pos(darray_first(folios));
darray_for_each(folios, fi) { darray_for_each(folios, fi) {
struct folio *f = *fi; struct folio *f = *fi;
unsigned f_len = min(end, folio_end_pos(f)) - f_pos; u64 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); unsigned f_copied = copy_page_from_iter_atomic(&f->page, f_offset, f_len, iter);
if (!f_copied) { if (!f_copied) {
...@@ -1982,7 +1988,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode, ...@@ -1982,7 +1988,7 @@ static int __bch2_buffered_write(struct bch_inode_info *inode,
f_offset = pos - folio_pos(darray_first(folios)); f_offset = pos - folio_pos(darray_first(folios));
darray_for_each(folios, fi) { darray_for_each(folios, fi) {
struct folio *f = *fi; struct folio *f = *fi;
unsigned f_len = min(end, folio_end_pos(f)) - f_pos; u64 f_len = min(end, folio_end_pos(f)) - f_pos;
if (!folio_test_uptodate(f)) if (!folio_test_uptodate(f))
folio_mark_uptodate(f); folio_mark_uptodate(f);
...@@ -2818,7 +2824,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode, ...@@ -2818,7 +2824,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode,
struct folio *folio; struct folio *folio;
s64 i_sectors_delta = 0; s64 i_sectors_delta = 0;
int ret = 0; int ret = 0;
loff_t end_pos; u64 end_pos;
folio = filemap_lock_folio(mapping, index); folio = filemap_lock_folio(mapping, index);
if (!folio) { if (!folio) {
...@@ -2844,7 +2850,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode, ...@@ -2844,7 +2850,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode,
BUG_ON(end <= folio_pos(folio)); BUG_ON(end <= folio_pos(folio));
start_offset = max(start, folio_pos(folio)) - folio_pos(folio); start_offset = max(start, folio_pos(folio)) - folio_pos(folio);
end_offset = min(end, folio_end_pos(folio)) - folio_pos(folio); end_offset = min_t(u64, end, folio_end_pos(folio)) - folio_pos(folio);
/* Folio boundary? Nothing to do */ /* Folio boundary? Nothing to do */
if (start_offset == 0 && if (start_offset == 0 &&
...@@ -2895,7 +2901,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode, ...@@ -2895,7 +2901,7 @@ static int __bch2_truncate_folio(struct bch_inode_info *inode,
WARN_ON_ONCE(folio_pos(folio) >= inode->v.i_size); WARN_ON_ONCE(folio_pos(folio) >= inode->v.i_size);
end_pos = folio_end_pos(folio); end_pos = folio_end_pos(folio);
if (inode->v.i_size > folio_pos(folio)) if (inode->v.i_size > folio_pos(folio))
end_pos = min(inode->v.i_size, end_pos); end_pos = min_t(u64, inode->v.i_size, end_pos);
ret = s->s[(end_pos - folio_pos(folio) - 1) >> 9].state >= SECTOR_dirty; ret = s->s[(end_pos - folio_pos(folio) - 1) >> 9].state >= SECTOR_dirty;
folio_zero_segment(folio, start_offset, end_offset); folio_zero_segment(folio, start_offset, end_offset);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment