On Wed, Apr 28, 2021 at 12:21:52PM +0530, Chandan Babu R wrote: > Executing xfs/538 after disabling injection of bmap_alloc_minlen_extent error > can cause several tasks to trigger hung task timeout. Most of the tasks are > blocked on getting a lock on an AG's AGF buffer. However, The task which has > the lock on the AG's AGF buffer has the following call trace, > > PID: 1341 TASK: ffff8881073f3700 CPU: 1 COMMAND: "fsstress" > __schedule+0x22f at ffffffff81f75e8f > schedule+0x46 at ffffffff81f76366 > xfs_extent_busy_flush+0x69 at ffffffff81477d99 > xfs_alloc_ag_vextent_size+0x16a at ffffffff8141711a > xfs_alloc_ag_vextent+0x19b at ffffffff81417edb > xfs_alloc_fix_freelist+0x22f at ffffffff8141896f > xfs_free_extent_fix_freelist+0x6a at ffffffff8141939a > __xfs_free_extent+0x99 at ffffffff81419499 > xfs_trans_free_extent+0x3e at ffffffff814a6fee > xfs_extent_free_finish_item+0x24 at ffffffff814a70d4 > xfs_defer_finish_noroll+0x1f7 at ffffffff81441407 > xfs_defer_finish+0x11 at ffffffff814417e1 > xfs_itruncate_extents_flags+0x13d at ffffffff8148b7dd > xfs_inactive_truncate+0xb9 at ffffffff8148bb89 > xfs_inactive+0x227 at ffffffff8148c4f7 > xfs_fs_destroy_inode+0xb8 at ffffffff81496898 > destroy_inode+0x3b at ffffffff8127d2ab > do_unlinkat+0x1d1 at ffffffff81270df1 > do_syscall_64+0x40 at ffffffff81f6b5f0 > entry_SYSCALL_64_after_hwframe+0x44 at ffffffff8200007c > > The following sequence of events lead to the above listed call trace, > > 1. The task frees atleast two extents belonging to the file being truncated. > 2. The corresponding xfs_extent_free_items are stored in the list pointed to > by xfs_defer_pending->dfp_work. > 3. When executing the next step of the rolling transaction, The first of the > above mentioned extents is freed. The corresponding busy extent entry is > added to the current transaction's tp->t_busy list as well as to the perag > rb tree at xfs_perag->pagb_tree. > 4. When trying to free the second extent, XFS determines that the AGFL needs > to be populated and hence tries to allocate free blocks. > 5. The only free extent whose size is >= xfs_alloc_arg->maxlen > happens to be the first extent that was freed by the current transaction. > 6. Hence xfs_alloc_ag_vextent_size() flushes the CIL in the hope of clearing > the busy status of the extent and waits for the busy generation number to > change. > 7. However, flushing the CIL is futile since the busy extent is still in the > current transaction's tp->t_busy list. > > Here the task ends up waiting indefinitely. > > This commit fixes the bug by preventing a CIL flush if all free extents are > busy and all of them are in the transaction's tp->t_busy list. Hmmm. I don't doubt that this fixes the symptom you are seeing, but the way it is being fixed doesn't seem right to me at all. We're rtying to populate the AGFL here, and the fact is that a multi-block allocation is simply an optimisation to minimise the number of extents we need to allocate to fill the AGFL. The extent that gets allocated gets broken up into single blocks to be inserted into the AGFL, so we don't actually need a continuguous extent to be allocated here. Hence, if the extent we find is busy when allocating for the AGFL, we should just skip it and choose another extent. args->minlen is set to zero for the allocation, so we can actually return any extent that has a length <= args->maxlen. We know this is an AGFL allocation because args->resv == XFS_AG_RESV_AGFL, so if we find a busy extent that would require a log force to be able to use before we can place it in the AGFL, we should just skip it entirely and select another extent to allocate from. Adding another two boolean conditionals to the already complex extent selection for this specific case makes the code much harder to follow and reason about. I'd much prefer that we just do something like: if (busy && args->resv == XFS_AG_RESV_AGFL) { /* * Extent might have just been freed in this * transaction so we can't use it. Move to the next * best extent candidate and try that instead. */ <increment/decrement and continue the search loop> } IOWs, we should not be issuing a log force to flush busy extents if we can't use the largest candidate free extent for the AGFL - we should just keep searching until we find one we can use.... Cheers, Dave. -- Dave Chinner david@xxxxxxxxxxxxx