Message ID | 0914bad6138d2cfafc9cfe762bd06c1883ceb9d2.1744657692.git.josef@toxicpanda.com (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | btrfs: adjust subpage bit start based on sectorsize | expand |
On Mon, Apr 14, 2025 at 03:08:16PM -0400, Josef Bacik wrote: > When running machines with 64k page size and a 16k nodesize we started > seeing tree log corruption in production. This turned out to be because > we were not writing out dirty blocks sometimes, so this in fact affects > all metadata writes. > > When writing out a subpage EB we scan the subpage bitmap for a dirty > range. If the range isn't dirty we do > > bit_start++; > > to move onto the next bit. The problem is the bitmap is based on the > number of sectors that an EB has. So in this case, we have a 64k > pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 > bits for every node. With a 64k page size we end up with 4 nodes per > page. > > To make this easier this is how everything looks > > [0 16k 32k 48k ] logical address > [0 4 8 12 ] radix tree offset > [ 64k page ] folio > [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers > [ | | | | | | | | | | | | | | | | ] bitmap > > Now we use all of our addressing based on fs_info->sectorsize_bits, so > as you can see the above our 16k eb->start turns into radix entry 4. > > When we find a dirty range for our eb, we correctly do bit_start += > sectors_per_node, because if we start at bit 0, the next bit for the > next eb is 4, to correspond to eb->start 16k. > > However if our range is clean, we will do bit_start++, which will now > put us offset from our radix tree entries. > > In our case, assume that the first time we check the bitmap the block is > not dirty, we increment bit_start so now it == 1, and then we loop > around and check again. This time it is dirty, and we go to find that > start using the following equation > > start = folio_start + bit_start * fs_info->sectorsize; > > so in the case above, eb->start 0 is now dirty, and we calculate start > as > > 0 + 1 * fs_info->sectorsize = 4096 > 4096 >> 12 = 1 > > Now we're looking up the radix tree for 1, and we won't find an eb. > What's worse is now we're using bit_start == 1, so we do bit_start += > sectors_per_node, which is now 5. If that eb is dirty we will run into > the same thing, we will look at an offset that is not populated in the > radix tree, and now we're skipping the writeout of dirty extent buffers. > > The best fix for this is to not use sectorsize_bits to address nodes, > but that's a larger change. Since this is a fs corruption problem fix > it simply by always using sectors_per_node to increment the start bit. > > Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a subpage metadata page") > Signed-off-by: Josef Bacik <josef@toxicpanda.com> Reviewed-by: Boris Burkov <boris@bur.io> > --- > fs/btrfs/extent_io.c | 2 +- > 1 file changed, 1 insertion(+), 1 deletion(-) > > diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c > index 5f08615b334f..6cfd286b8bbc 100644 > --- a/fs/btrfs/extent_io.c > +++ b/fs/btrfs/extent_io.c > @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio *folio, struct writeback_control *wbc) > subpage->bitmaps)) { > spin_unlock_irqrestore(&subpage->lock, flags); > spin_unlock(&folio->mapping->i_private_lock); > - bit_start++; > + bit_start += sectors_per_node; > continue; > } > > -- > 2.48.1 >
在 2025/4/15 04:38, Josef Bacik 写道: > When running machines with 64k page size and a 16k nodesize we started > seeing tree log corruption in production. This turned out to be because > we were not writing out dirty blocks sometimes, so this in fact affects > all metadata writes. > > When writing out a subpage EB we scan the subpage bitmap for a dirty > range. If the range isn't dirty we do > > bit_start++; > > to move onto the next bit. The problem is the bitmap is based on the > number of sectors that an EB has. So in this case, we have a 64k > pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 > bits for every node. With a 64k page size we end up with 4 nodes per > page. > > To make this easier this is how everything looks > > [0 16k 32k 48k ] logical address > [0 4 8 12 ] radix tree offset > [ 64k page ] folio > [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers > [ | | | | | | | | | | | | | | | | ] bitmap > > Now we use all of our addressing based on fs_info->sectorsize_bits, so > as you can see the above our 16k eb->start turns into radix entry 4. > > When we find a dirty range for our eb, we correctly do bit_start += > sectors_per_node, because if we start at bit 0, the next bit for the > next eb is 4, to correspond to eb->start 16k. > > However if our range is clean, we will do bit_start++, which will now > put us offset from our radix tree entries. > > In our case, assume that the first time we check the bitmap the block is > not dirty, we increment bit_start so now it == 1, and then we loop > around and check again. This time it is dirty, and we go to find that > start using the following equation > > start = folio_start + bit_start * fs_info->sectorsize; > > so in the case above, eb->start 0 is now dirty, and we calculate start > as > > 0 + 1 * fs_info->sectorsize = 4096 > 4096 >> 12 = 1 > > Now we're looking up the radix tree for 1, and we won't find an eb. > What's worse is now we're using bit_start == 1, so we do bit_start += > sectors_per_node, which is now 5. If that eb is dirty we will run into > the same thing, we will look at an offset that is not populated in the > radix tree, and now we're skipping the writeout of dirty extent buffers. > > The best fix for this is to not use sectorsize_bits to address nodes, > but that's a larger change. Since this is a fs corruption problem fix > it simply by always using sectors_per_node to increment the start bit. > > Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a subpage metadata page") > Signed-off-by: Josef Bacik <josef@toxicpanda.com> > --- > fs/btrfs/extent_io.c | 2 +- > 1 file changed, 1 insertion(+), 1 deletion(-) > > diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c > index 5f08615b334f..6cfd286b8bbc 100644 > --- a/fs/btrfs/extent_io.c > +++ b/fs/btrfs/extent_io.c > @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio *folio, struct writeback_control *wbc) > subpage->bitmaps)) { > spin_unlock_irqrestore(&subpage->lock, flags); > spin_unlock(&folio->mapping->i_private_lock); > - bit_start++; > + bit_start += sectors_per_node; The problem is, we can not ensure all extent buffers are nodesize aligned. If we have an eb whose bytenr is only block aligned but not node size aligned, this will lead to the same problem. We need an extra check to reject tree blocks which are not node size aligned, which is another big change and not suitable for a quick fix. Can we do a gang radix tree lookup for the involved ranges that can cover the block, then increase bit_start to the end of the found eb instead? Thanks, Qu > continue; > } >
在 2025/4/15 07:38, Qu Wenruo 写道: > > > 在 2025/4/15 04:38, Josef Bacik 写道: >> When running machines with 64k page size and a 16k nodesize we started >> seeing tree log corruption in production. This turned out to be because >> we were not writing out dirty blocks sometimes, so this in fact affects >> all metadata writes. >> >> When writing out a subpage EB we scan the subpage bitmap for a dirty >> range. If the range isn't dirty we do >> >> bit_start++; >> >> to move onto the next bit. The problem is the bitmap is based on the >> number of sectors that an EB has. So in this case, we have a 64k >> pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 >> bits for every node. With a 64k page size we end up with 4 nodes per >> page. >> >> To make this easier this is how everything looks >> >> [0 16k 32k 48k ] logical address >> [0 4 8 12 ] radix tree offset >> [ 64k page ] folio >> [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers >> [ | | | | | | | | | | | | | | | | ] bitmap >> >> Now we use all of our addressing based on fs_info->sectorsize_bits, so >> as you can see the above our 16k eb->start turns into radix entry 4. >> >> When we find a dirty range for our eb, we correctly do bit_start += >> sectors_per_node, because if we start at bit 0, the next bit for the >> next eb is 4, to correspond to eb->start 16k. >> >> However if our range is clean, we will do bit_start++, which will now >> put us offset from our radix tree entries. >> >> In our case, assume that the first time we check the bitmap the block is >> not dirty, we increment bit_start so now it == 1, and then we loop >> around and check again. This time it is dirty, and we go to find that >> start using the following equation >> >> start = folio_start + bit_start * fs_info->sectorsize; >> >> so in the case above, eb->start 0 is now dirty, and we calculate start >> as >> >> 0 + 1 * fs_info->sectorsize = 4096 >> 4096 >> 12 = 1 >> >> Now we're looking up the radix tree for 1, and we won't find an eb. >> What's worse is now we're using bit_start == 1, so we do bit_start += >> sectors_per_node, which is now 5. If that eb is dirty we will run into >> the same thing, we will look at an offset that is not populated in the >> radix tree, and now we're skipping the writeout of dirty extent buffers. >> >> The best fix for this is to not use sectorsize_bits to address nodes, >> but that's a larger change. Since this is a fs corruption problem fix >> it simply by always using sectors_per_node to increment the start bit. >> >> Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a >> subpage metadata page") >> Signed-off-by: Josef Bacik <josef@toxicpanda.com> >> --- >> fs/btrfs/extent_io.c | 2 +- >> 1 file changed, 1 insertion(+), 1 deletion(-) >> >> diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c >> index 5f08615b334f..6cfd286b8bbc 100644 >> --- a/fs/btrfs/extent_io.c >> +++ b/fs/btrfs/extent_io.c >> @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio >> *folio, struct writeback_control *wbc) >> subpage->bitmaps)) { >> spin_unlock_irqrestore(&subpage->lock, flags); >> spin_unlock(&folio->mapping->i_private_lock); >> - bit_start++; >> + bit_start += sectors_per_node; > > The problem is, we can not ensure all extent buffers are nodesize aligned. > > If we have an eb whose bytenr is only block aligned but not node size > aligned, this will lead to the same problem. > > We need an extra check to reject tree blocks which are not node size > aligned, which is another big change and not suitable for a quick fix. > > > Can we do a gang radix tree lookup for the involved ranges that can > cover the block, then increase bit_start to the end of the found eb > instead? In fact, I think it's better to fix both this and the missing eb write bugs together in one go. With something like this: static int find_subpage_dirty_subpage(struct folio *folio) { struct extent_buffer *gang[BTRFS_MAX_EB_SIZE/MIN_BLOCKSIZE]; struct extent_buffer *ret = NULL; rcu_read_lock() ret = radix_tree_gang_lookup(); for (int i = 0; i < ret; i++) { if (eb && atomic_inc_not_zero(&eb->refs)) { if (!test_bit(EXTENT_BUFFER_DIRTY) { atomic_dec(&eb->refs); continue; } ret = eb; break; } } rcu_read_unlock() return ret; } And make submit_eb_subpage() no longer relies on subpage dirty bitmap, but above helper to grab any dirty ebs. By this, we fix both bugs by: - No more bitmap search So no increment mismatch, and can still handle unaligned one (as long as they don't cross page boundary). - No more missing writeback As the gang lookup is always for the whole folio, and we always test eb dirty flags, we should always catch dirty ebs in the folio. Thanks, Qu > > Thanks, > Qu > >> continue; >> } >
On Tue, Apr 15, 2025 at 08:07:08AM +0930, Qu Wenruo wrote: > > > 在 2025/4/15 07:38, Qu Wenruo 写道: > > > > > > 在 2025/4/15 04:38, Josef Bacik 写道: > > > When running machines with 64k page size and a 16k nodesize we started > > > seeing tree log corruption in production. This turned out to be because > > > we were not writing out dirty blocks sometimes, so this in fact affects > > > all metadata writes. > > > > > > When writing out a subpage EB we scan the subpage bitmap for a dirty > > > range. If the range isn't dirty we do > > > > > > bit_start++; > > > > > > to move onto the next bit. The problem is the bitmap is based on the > > > number of sectors that an EB has. So in this case, we have a 64k > > > pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 > > > bits for every node. With a 64k page size we end up with 4 nodes per > > > page. > > > > > > To make this easier this is how everything looks > > > > > > [0 16k 32k 48k ] logical address > > > [0 4 8 12 ] radix tree offset > > > [ 64k page ] folio > > > [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers > > > [ | | | | | | | | | | | | | | | | ] bitmap > > > > > > Now we use all of our addressing based on fs_info->sectorsize_bits, so > > > as you can see the above our 16k eb->start turns into radix entry 4. > > > > > > When we find a dirty range for our eb, we correctly do bit_start += > > > sectors_per_node, because if we start at bit 0, the next bit for the > > > next eb is 4, to correspond to eb->start 16k. > > > > > > However if our range is clean, we will do bit_start++, which will now > > > put us offset from our radix tree entries. > > > > > > In our case, assume that the first time we check the bitmap the block is > > > not dirty, we increment bit_start so now it == 1, and then we loop > > > around and check again. This time it is dirty, and we go to find that > > > start using the following equation > > > > > > start = folio_start + bit_start * fs_info->sectorsize; > > > > > > so in the case above, eb->start 0 is now dirty, and we calculate start > > > as > > > > > > 0 + 1 * fs_info->sectorsize = 4096 > > > 4096 >> 12 = 1 > > > > > > Now we're looking up the radix tree for 1, and we won't find an eb. > > > What's worse is now we're using bit_start == 1, so we do bit_start += > > > sectors_per_node, which is now 5. If that eb is dirty we will run into > > > the same thing, we will look at an offset that is not populated in the > > > radix tree, and now we're skipping the writeout of dirty extent buffers. > > > > > > The best fix for this is to not use sectorsize_bits to address nodes, > > > but that's a larger change. Since this is a fs corruption problem fix > > > it simply by always using sectors_per_node to increment the start bit. > > > > > > Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a > > > subpage metadata page") > > > Signed-off-by: Josef Bacik <josef@toxicpanda.com> > > > --- > > > fs/btrfs/extent_io.c | 2 +- > > > 1 file changed, 1 insertion(+), 1 deletion(-) > > > > > > diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c > > > index 5f08615b334f..6cfd286b8bbc 100644 > > > --- a/fs/btrfs/extent_io.c > > > +++ b/fs/btrfs/extent_io.c > > > @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio > > > *folio, struct writeback_control *wbc) > > > subpage->bitmaps)) { > > > spin_unlock_irqrestore(&subpage->lock, flags); > > > spin_unlock(&folio->mapping->i_private_lock); > > > - bit_start++; > > > + bit_start += sectors_per_node; > > > > The problem is, we can not ensure all extent buffers are nodesize aligned. > > In theory because the allocator can do whatever it wants, or in practice because of mixed block groups? It seems to me that in practice without mixed block groups they ought to always be aligned. > > If we have an eb whose bytenr is only block aligned but not node size > > aligned, this will lead to the same problem. > > But then the existing code for the non error path is broken, right? How was this intended to work? Is there any correct way to loop over the ebs in a folio when nodesize < pagesize? Your proposed gang lookup? I guess to put my question a different way, was it intentional to mix the increment size in the two codepaths in this function? > > We need an extra check to reject tree blocks which are not node size > > aligned, which is another big change and not suitable for a quick fix. > > > > > > Can we do a gang radix tree lookup for the involved ranges that can > > cover the block, then increase bit_start to the end of the found eb > > instead? > > In fact, I think it's better to fix both this and the missing eb write > bugs together in one go. > > With something like this: > > static int find_subpage_dirty_subpage(struct folio *folio) > { > struct extent_buffer *gang[BTRFS_MAX_EB_SIZE/MIN_BLOCKSIZE]; > struct extent_buffer *ret = NULL; > > rcu_read_lock() > ret = radix_tree_gang_lookup(); > for (int i = 0; i < ret; i++) { > if (eb && atomic_inc_not_zero(&eb->refs)) { > if (!test_bit(EXTENT_BUFFER_DIRTY) { > atomic_dec(&eb->refs); > continue; > } > ret = eb; > break; > } > } > rcu_read_unlock() > return ret; > } > > And make submit_eb_subpage() no longer relies on subpage dirty bitmap, > but above helper to grab any dirty ebs. > > By this, we fix both bugs by: > > - No more bitmap search > So no increment mismatch, and can still handle unaligned one (as long > as they don't cross page boundary). > > - No more missing writeback > As the gang lookup is always for the whole folio, and we always test > eb dirty flags, we should always catch dirty ebs in the folio. I don't see why this is the case. The race Josef fixed is quite narrow but is fundamentally based on the TOWRITE mark getting cleared mid subpage iteration. If all we do is change subpage bitmap to this gang lookup, but still clear the TOWRITE tag whenever the folio has the first eb call meta_folio_set_writeback(), then it is possible for other threads to come in and dirty a different eb, write it back, tag it TOWRITE, then lose the tag before doing the tagged lookup for TOWRITE folios. Thanks, Boris > > Thanks, > Qu > > > > > Thanks, > > Qu > > > > > continue; > > > } > > >
On Tue, Apr 15, 2025 at 07:38:29AM +0930, Qu Wenruo wrote: > > > 在 2025/4/15 04:38, Josef Bacik 写道: > > When running machines with 64k page size and a 16k nodesize we started > > seeing tree log corruption in production. This turned out to be because > > we were not writing out dirty blocks sometimes, so this in fact affects > > all metadata writes. > > > > When writing out a subpage EB we scan the subpage bitmap for a dirty > > range. If the range isn't dirty we do > > > > bit_start++; > > > > to move onto the next bit. The problem is the bitmap is based on the > > number of sectors that an EB has. So in this case, we have a 64k > > pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 > > bits for every node. With a 64k page size we end up with 4 nodes per > > page. > > > > To make this easier this is how everything looks > > > > [0 16k 32k 48k ] logical address > > [0 4 8 12 ] radix tree offset > > [ 64k page ] folio > > [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers > > [ | | | | | | | | | | | | | | | | ] bitmap > > > > Now we use all of our addressing based on fs_info->sectorsize_bits, so > > as you can see the above our 16k eb->start turns into radix entry 4. > > > > When we find a dirty range for our eb, we correctly do bit_start += > > sectors_per_node, because if we start at bit 0, the next bit for the > > next eb is 4, to correspond to eb->start 16k. > > > > However if our range is clean, we will do bit_start++, which will now > > put us offset from our radix tree entries. > > > > In our case, assume that the first time we check the bitmap the block is > > not dirty, we increment bit_start so now it == 1, and then we loop > > around and check again. This time it is dirty, and we go to find that > > start using the following equation > > > > start = folio_start + bit_start * fs_info->sectorsize; > > > > so in the case above, eb->start 0 is now dirty, and we calculate start > > as > > > > 0 + 1 * fs_info->sectorsize = 4096 > > 4096 >> 12 = 1 > > > > Now we're looking up the radix tree for 1, and we won't find an eb. > > What's worse is now we're using bit_start == 1, so we do bit_start += > > sectors_per_node, which is now 5. If that eb is dirty we will run into > > the same thing, we will look at an offset that is not populated in the > > radix tree, and now we're skipping the writeout of dirty extent buffers. > > > > The best fix for this is to not use sectorsize_bits to address nodes, > > but that's a larger change. Since this is a fs corruption problem fix > > it simply by always using sectors_per_node to increment the start bit. > > > > Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a subpage metadata page") > > Signed-off-by: Josef Bacik <josef@toxicpanda.com> > > --- > > fs/btrfs/extent_io.c | 2 +- > > 1 file changed, 1 insertion(+), 1 deletion(-) > > > > diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c > > index 5f08615b334f..6cfd286b8bbc 100644 > > --- a/fs/btrfs/extent_io.c > > +++ b/fs/btrfs/extent_io.c > > @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio *folio, struct writeback_control *wbc) > > subpage->bitmaps)) { > > spin_unlock_irqrestore(&subpage->lock, flags); > > spin_unlock(&folio->mapping->i_private_lock); > > - bit_start++; > > + bit_start += sectors_per_node; > > The problem is, we can not ensure all extent buffers are nodesize aligned. > > If we have an eb whose bytenr is only block aligned but not node size > aligned, this will lead to the same problem. > > We need an extra check to reject tree blocks which are not node size > aligned, which is another big change and not suitable for a quick fix. We already have this. > > > Can we do a gang radix tree lookup for the involved ranges that can cover > the block, then increase bit_start to the end of the found eb instead? That's a followup patch that I'm testing now. I've started with the simplest fix so that they can be pulled back to all the affected kernels, and then I'm following up with much more invasive changes to make these problems go away in general. Thanks, Josef
On Tue, Apr 15, 2025 at 08:07:08AM +0930, Qu Wenruo wrote: > > > 在 2025/4/15 07:38, Qu Wenruo 写道: > > > > > > 在 2025/4/15 04:38, Josef Bacik 写道: > > > When running machines with 64k page size and a 16k nodesize we started > > > seeing tree log corruption in production. This turned out to be because > > > we were not writing out dirty blocks sometimes, so this in fact affects > > > all metadata writes. > > > > > > When writing out a subpage EB we scan the subpage bitmap for a dirty > > > range. If the range isn't dirty we do > > > > > > bit_start++; > > > > > > to move onto the next bit. The problem is the bitmap is based on the > > > number of sectors that an EB has. So in this case, we have a 64k > > > pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 > > > bits for every node. With a 64k page size we end up with 4 nodes per > > > page. > > > > > > To make this easier this is how everything looks > > > > > > [0 16k 32k 48k ] logical address > > > [0 4 8 12 ] radix tree offset > > > [ 64k page ] folio > > > [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers > > > [ | | | | | | | | | | | | | | | | ] bitmap > > > > > > Now we use all of our addressing based on fs_info->sectorsize_bits, so > > > as you can see the above our 16k eb->start turns into radix entry 4. > > > > > > When we find a dirty range for our eb, we correctly do bit_start += > > > sectors_per_node, because if we start at bit 0, the next bit for the > > > next eb is 4, to correspond to eb->start 16k. > > > > > > However if our range is clean, we will do bit_start++, which will now > > > put us offset from our radix tree entries. > > > > > > In our case, assume that the first time we check the bitmap the block is > > > not dirty, we increment bit_start so now it == 1, and then we loop > > > around and check again. This time it is dirty, and we go to find that > > > start using the following equation > > > > > > start = folio_start + bit_start * fs_info->sectorsize; > > > > > > so in the case above, eb->start 0 is now dirty, and we calculate start > > > as > > > > > > 0 + 1 * fs_info->sectorsize = 4096 > > > 4096 >> 12 = 1 > > > > > > Now we're looking up the radix tree for 1, and we won't find an eb. > > > What's worse is now we're using bit_start == 1, so we do bit_start += > > > sectors_per_node, which is now 5. If that eb is dirty we will run into > > > the same thing, we will look at an offset that is not populated in the > > > radix tree, and now we're skipping the writeout of dirty extent buffers. > > > > > > The best fix for this is to not use sectorsize_bits to address nodes, > > > but that's a larger change. Since this is a fs corruption problem fix > > > it simply by always using sectors_per_node to increment the start bit. > > > > > > Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a > > > subpage metadata page") > > > Signed-off-by: Josef Bacik <josef@toxicpanda.com> > > > --- > > > fs/btrfs/extent_io.c | 2 +- > > > 1 file changed, 1 insertion(+), 1 deletion(-) > > > > > > diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c > > > index 5f08615b334f..6cfd286b8bbc 100644 > > > --- a/fs/btrfs/extent_io.c > > > +++ b/fs/btrfs/extent_io.c > > > @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio > > > *folio, struct writeback_control *wbc) > > > subpage->bitmaps)) { > > > spin_unlock_irqrestore(&subpage->lock, flags); > > > spin_unlock(&folio->mapping->i_private_lock); > > > - bit_start++; > > > + bit_start += sectors_per_node; > > > > The problem is, we can not ensure all extent buffers are nodesize aligned. > > > > If we have an eb whose bytenr is only block aligned but not node size > > aligned, this will lead to the same problem. > > > > We need an extra check to reject tree blocks which are not node size > > aligned, which is another big change and not suitable for a quick fix. > > > > > > Can we do a gang radix tree lookup for the involved ranges that can > > cover the block, then increase bit_start to the end of the found eb > > instead? > > In fact, I think it's better to fix both this and the missing eb write > bugs together in one go. > > With something like this: > > static int find_subpage_dirty_subpage(struct folio *folio) > { > struct extent_buffer *gang[BTRFS_MAX_EB_SIZE/MIN_BLOCKSIZE]; > struct extent_buffer *ret = NULL; > > rcu_read_lock() > ret = radix_tree_gang_lookup(); > for (int i = 0; i < ret; i++) { > if (eb && atomic_inc_not_zero(&eb->refs)) { > if (!test_bit(EXTENT_BUFFER_DIRTY) { > atomic_dec(&eb->refs); > continue; > } > ret = eb; > break; > } > } > rcu_read_unlock() > return ret; > } Again, I'm following up with a better solution for all of this. The current patch needs to be pulled back to a ton of kernels, so this is targeted at fixing the problem, and then we can make it look better with a series that has a longer bake time. Thanks, Josef
diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c index 5f08615b334f..6cfd286b8bbc 100644 --- a/fs/btrfs/extent_io.c +++ b/fs/btrfs/extent_io.c @@ -2034,7 +2034,7 @@ static int submit_eb_subpage(struct folio *folio, struct writeback_control *wbc) subpage->bitmaps)) { spin_unlock_irqrestore(&subpage->lock, flags); spin_unlock(&folio->mapping->i_private_lock); - bit_start++; + bit_start += sectors_per_node; continue; }
When running machines with 64k page size and a 16k nodesize we started seeing tree log corruption in production. This turned out to be because we were not writing out dirty blocks sometimes, so this in fact affects all metadata writes. When writing out a subpage EB we scan the subpage bitmap for a dirty range. If the range isn't dirty we do bit_start++; to move onto the next bit. The problem is the bitmap is based on the number of sectors that an EB has. So in this case, we have a 64k pagesize, 16k nodesize, but a 4k sectorsize. This means our bitmap is 4 bits for every node. With a 64k page size we end up with 4 nodes per page. To make this easier this is how everything looks [0 16k 32k 48k ] logical address [0 4 8 12 ] radix tree offset [ 64k page ] folio [ 16k eb ][ 16k eb ][ 16k eb ][ 16k eb ] extent buffers [ | | | | | | | | | | | | | | | | ] bitmap Now we use all of our addressing based on fs_info->sectorsize_bits, so as you can see the above our 16k eb->start turns into radix entry 4. When we find a dirty range for our eb, we correctly do bit_start += sectors_per_node, because if we start at bit 0, the next bit for the next eb is 4, to correspond to eb->start 16k. However if our range is clean, we will do bit_start++, which will now put us offset from our radix tree entries. In our case, assume that the first time we check the bitmap the block is not dirty, we increment bit_start so now it == 1, and then we loop around and check again. This time it is dirty, and we go to find that start using the following equation start = folio_start + bit_start * fs_info->sectorsize; so in the case above, eb->start 0 is now dirty, and we calculate start as 0 + 1 * fs_info->sectorsize = 4096 4096 >> 12 = 1 Now we're looking up the radix tree for 1, and we won't find an eb. What's worse is now we're using bit_start == 1, so we do bit_start += sectors_per_node, which is now 5. If that eb is dirty we will run into the same thing, we will look at an offset that is not populated in the radix tree, and now we're skipping the writeout of dirty extent buffers. The best fix for this is to not use sectorsize_bits to address nodes, but that's a larger change. Since this is a fs corruption problem fix it simply by always using sectors_per_node to increment the start bit. Fixes: c4aec299fa8f ("btrfs: introduce submit_eb_subpage() to submit a subpage metadata page") Signed-off-by: Josef Bacik <josef@toxicpanda.com> --- fs/btrfs/extent_io.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-)