Re: generic/475 deadlock?

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Fri, Mar 22, 2019 at 10:50:45AM +1100, Dave Chinner wrote:
> On Thu, Mar 21, 2019 at 05:53:07PM -0400, Brian Foster wrote:
> > On Fri, Mar 22, 2019 at 08:10:51AM +1100, Dave Chinner wrote:
> > > On Thu, Mar 21, 2019 at 08:11:52AM -0400, Brian Foster wrote:
> > > > On Thu, Mar 21, 2019 at 08:49:38AM +1100, Dave Chinner wrote:
> > > > The current xfsaild() log force logic is still tied to PINNED objects,
> > > > but it has additional logic to detect skipped buffers at delwri queue
> > > > submit time (i.e., due to being pinned) that should cover the pinned
> > > > inode cluster buffer case.
> > > 
> > > And it's not skipped buffers, either. Skipped buffers are only for
> > > determining timeouts - it's the ail->ail_log_flush counter that
> > > determines whether a log flush should be done or not, and that's
> > > only incremented on ITEM_PINNED objects. This is why I was
> > > suggesting returning that if the backing buffer is pinned, but I
> > > forgot all about how we already handle this case...
> > > 
> > > > But given the above, couldn't we just remove the log force from
> > > > xfs_iflush() and let the existing xfsaild_push() logic handle it?
> > > 
> > > We can, but not for the reasons you are suggesting.
> > > 
> > 
> > The reason I suggested is because we currently "detect skipped buffers
> > at delwri queue submit time (i.e., due to being pinned)," which causes
> > xfsaild to force the log.  I'm not sure how exactly you read that, but it
> > refers exactly to the code snippet you posted below where we bump the
> > log flush counter due to skipped buffers on submit (because they were
> > pinned).
> 
> In the context of the AIL pushing, "skipped" has specific meaning -
> it refers to items that could not be flushed for whatever reason.
> And the actions that are taken based on the number of "skipped"
> objects is taken after delwri queue submit time.
> 
> You used all the terminology that refers to /skipped/ item
> handling, not /pinned/ item handling - if you re-read what I wrote
> above with this difference in mind, then it will make a lot more
> sense to you.
> 

Perhaps I could have phrased it better, but I specifically referred to
skipped buffers (not items), in the context of delwri submission and
qualified it with those skipped due to being pinned. ;) Even in my
earlier reply right after Darrick corrected my mischaracterization of
the deadlock, I said the following about the log force in xfs_iflush():

"I was wondering why we'd need that given that xfsaild() has its own log
force logic in cases where the delwri submit didn't fully complete
(i.e., for pinned buffers)."

But no matter, I can see where the disconnect is now at least. I just
think you misread me in this case.

> FWIW, this skipped/pinned terminology difference goes right down
> into the delwri list handling I quoted - I'll add the next few lines
> so it's clear what I'm talking about:
> 
> 
> > > 	list_for_each_entry_safe(bp, n, buffer_list, b_list) {
> > > 		if (!wait_list) {
> > > 			if (xfs_buf_ispinned(bp)) {
> > > 				pinned++;
> > > 				continue;
> > > 			}
> 			if (!xfs_buf_trylock(bp))
> 				continue;
> 		}
> 
> This code branch also avoid blocking on locked buffers. In AIL
> terms these locked buffers have been "skipped", but the code only
> counts and indicates pinned buffers to the caller. And the delwri
> submission code in the AIL pushing acts on this specific
> distinction - it triggers a log force if pinned buffers were
> encountered, not if the flush skipped locked buffers.
> 

Yep..

> > > ....
> > > 	return pinned;
> > > 
> > 
> > "detect skipped buffers at delwri queue submit time (i.e., due to being
> > pinned)"
> 
> If the AIL code was detecting skipped buffers, it would do:
> 
> 	xfs_buf_delwri_submit_nowait(&ailp->ail_buf_list);
> 	if (!list_empty(&ailp->ail_buf_list)) {
> 		/* we skipped buffers */
> 		.....
> 	}
> 
> But it's not doing this.
> 

In general, if you think I'm mischaracterizing code/behavior, it's
helpful to point out how you interpret the specified reasoning so we can
correct mistaken assumptions/understandings, clarify descriptions or use
better terminology or whatever. That helps to get onto the same page
more efficiently, particularly for cases where it so happens we were
talking about the same exact thing in the first place and just happen to
think about it a bit differently. ;)

For example, to just say "We can, but not for the reasons you are
suggesting." doesn't tell me much about how you interpreted my
reasoning. I thus assume you interpreted it exactly as I meant it. That
ambiguity leads to confusion when you then go on and explain the same
behavior I was (trying to) refer to. ;P If you said something like "we
can but not because of <reason>," it would have probably been
immediately apparent to me that the reason I intended to describe is not
exactly how you read it.

> i.e. "Pinned" vs "skipped" may be a subtle terminology difference,
> but it is an important difference and they are not interchangable...
> 
> > > remove them and issue IO on them, and tell the caller that we left
> > > pinned/locked buffers on the list that still need to be submitted.
> > > The AIL takes this as an indication it needs to flush the log to
> > > unpin the buffers it can't flush, and because they are still on it's
> > > delwri list, it will attempt to submit them again next time around
> > > after the log has been forced.
> > > 
> > 
> > Right, so this is exactly the reason I suggested.
> 
> OK, we're on the same page. :)
> 

Indeed.

> Do want to submit a patch that fixes the inode and dquot flushing?
> I haven't looked any further at the busy extent stuff, that's likely
> to be a more difficult problem to solve...
> 

I'll add it to the todo list and make a pass through the log items...

I missed where the busy extent stuff came into play here.. are you
pointing out that log forces from the busy extent code can occur under
buffer locks and thus we're susceptible to similar problems there? If
so, aren't we protected by the transaction in that context?

IOW, wouldn't a log force from busy extent context always occur with
locked buffers joined to a transaction? If so, then doesn't the active
transaction hold a bli reference and prevent such items from being
"freed" in log completion context (i.e. xfs_buf_item_unpin()) if they
happened to be pinned? Perhaps I'm missing something...

Brian

> Cheers,
> 
> Dave.
> -- 
> Dave Chinner
> david@xxxxxxxxxxxxx



[Index of Archives]     [XFS Filesystem Development (older mail)]     [Linux Filesystem Development]     [Linux Audio Users]     [Yosemite Trails]     [Linux Kernel]     [Linux RAID]     [Linux SCSI]


  Powered by Linux