forked from mirrors/linux
		
	xfs: make for_each_perag... a first class citizen
for_each_perag_tag() is defined in xfs_icache.c for local use. Promote this to xfs_ag.h and define equivalent iteration functions so that we can use them to iterate AGs instead to replace open coded perag walks and perag lookups. We also convert as many of the straight forward open coded AG walks to use these iterators as possible. Anything that is not a direct conversion to an iterator is ignored and will be updated in future commits. Signed-off-by: Dave Chinner <dchinner@redhat.com> Reviewed-by: Brian Foster <bfoster@redhat.com> Reviewed-by: Darrick J. Wong <djwong@kernel.org>
This commit is contained in:
		
							parent
							
								
									07b6403a68
								
							
						
					
					
						commit
						f250eedcf7
					
				
					 6 changed files with 39 additions and 52 deletions
				
			
		| 
						 | 
					@ -114,6 +114,23 @@ struct xfs_perag *xfs_perag_get_tag(struct xfs_mount *, xfs_agnumber_t,
 | 
				
			||||||
				   int tag);
 | 
									   int tag);
 | 
				
			||||||
void	xfs_perag_put(struct xfs_perag *pag);
 | 
					void	xfs_perag_put(struct xfs_perag *pag);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Perag iteration APIs
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#define for_each_perag(mp, next_agno, pag) \
 | 
				
			||||||
 | 
						for ((next_agno) = 0, (pag) = xfs_perag_get((mp), 0); \
 | 
				
			||||||
 | 
							(pag) != NULL; \
 | 
				
			||||||
 | 
							(next_agno) = (pag)->pag_agno + 1, \
 | 
				
			||||||
 | 
							xfs_perag_put(pag), \
 | 
				
			||||||
 | 
							(pag) = xfs_perag_get((mp), (next_agno)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define for_each_perag_tag(mp, next_agno, pag, tag) \
 | 
				
			||||||
 | 
						for ((next_agno) = 0, (pag) = xfs_perag_get_tag((mp), 0, (tag)); \
 | 
				
			||||||
 | 
							(pag) != NULL; \
 | 
				
			||||||
 | 
							(next_agno) = (pag)->pag_agno + 1, \
 | 
				
			||||||
 | 
							xfs_perag_put(pag), \
 | 
				
			||||||
 | 
							(pag) = xfs_perag_get_tag((mp), (next_agno), (tag)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct aghdr_init_data {
 | 
					struct aghdr_init_data {
 | 
				
			||||||
	/* per ag data */
 | 
						/* per ag data */
 | 
				
			||||||
	xfs_agblock_t		agno;		/* ag to init */
 | 
						xfs_agblock_t		agno;		/* ag to init */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -71,11 +71,11 @@ xchk_fscount_warmup(
 | 
				
			||||||
	xfs_agnumber_t		agno;
 | 
						xfs_agnumber_t		agno;
 | 
				
			||||||
	int			error = 0;
 | 
						int			error = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		pag = xfs_perag_get(mp, agno);
 | 
							if (xchk_should_terminate(sc, &error))
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
		if (pag->pagi_init && pag->pagf_init)
 | 
							if (pag->pagi_init && pag->pagf_init)
 | 
				
			||||||
			goto next_loop_perag;
 | 
								continue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/* Lock both AG headers. */
 | 
							/* Lock both AG headers. */
 | 
				
			||||||
		error = xfs_ialloc_read_agi(mp, sc->tp, agno, &agi_bp);
 | 
							error = xfs_ialloc_read_agi(mp, sc->tp, agno, &agi_bp);
 | 
				
			||||||
| 
						 | 
					@ -89,21 +89,15 @@ xchk_fscount_warmup(
 | 
				
			||||||
		 * These are supposed to be initialized by the header read
 | 
							 * These are supposed to be initialized by the header read
 | 
				
			||||||
		 * function.
 | 
							 * function.
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		error = -EFSCORRUPTED;
 | 
							if (!pag->pagi_init || !pag->pagf_init) {
 | 
				
			||||||
		if (!pag->pagi_init || !pag->pagf_init)
 | 
								error = -EFSCORRUPTED;
 | 
				
			||||||
			break;
 | 
								break;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		xfs_buf_relse(agf_bp);
 | 
							xfs_buf_relse(agf_bp);
 | 
				
			||||||
		agf_bp = NULL;
 | 
							agf_bp = NULL;
 | 
				
			||||||
		xfs_buf_relse(agi_bp);
 | 
							xfs_buf_relse(agi_bp);
 | 
				
			||||||
		agi_bp = NULL;
 | 
							agi_bp = NULL;
 | 
				
			||||||
next_loop_perag:
 | 
					 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
		pag = NULL;
 | 
					 | 
				
			||||||
		error = 0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (xchk_should_terminate(sc, &error))
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (agf_bp)
 | 
						if (agf_bp)
 | 
				
			||||||
| 
						 | 
					@ -196,13 +190,14 @@ xchk_fscount_aggregate_agcounts(
 | 
				
			||||||
	fsc->ifree = 0;
 | 
						fsc->ifree = 0;
 | 
				
			||||||
	fsc->fdblocks = 0;
 | 
						fsc->fdblocks = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		pag = xfs_perag_get(mp, agno);
 | 
							if (xchk_should_terminate(sc, &error))
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/* This somehow got unset since the warmup? */
 | 
							/* This somehow got unset since the warmup? */
 | 
				
			||||||
		if (!pag->pagi_init || !pag->pagf_init) {
 | 
							if (!pag->pagi_init || !pag->pagf_init) {
 | 
				
			||||||
			xfs_perag_put(pag);
 | 
								error = -EFSCORRUPTED;
 | 
				
			||||||
			return -EFSCORRUPTED;
 | 
								break;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/* Count all the inodes */
 | 
							/* Count all the inodes */
 | 
				
			||||||
| 
						 | 
					@ -216,10 +211,8 @@ xchk_fscount_aggregate_agcounts(
 | 
				
			||||||
			fsc->fdblocks += pag->pagf_btreeblks;
 | 
								fsc->fdblocks += pag->pagf_btreeblks;
 | 
				
			||||||
		} else {
 | 
							} else {
 | 
				
			||||||
			error = xchk_fscount_btreeblks(sc, fsc, agno);
 | 
								error = xchk_fscount_btreeblks(sc, fsc, agno);
 | 
				
			||||||
			if (error) {
 | 
								if (error)
 | 
				
			||||||
				xfs_perag_put(pag);
 | 
					 | 
				
			||||||
				break;
 | 
									break;
 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/*
 | 
							/*
 | 
				
			||||||
| 
						 | 
					@ -229,12 +222,9 @@ xchk_fscount_aggregate_agcounts(
 | 
				
			||||||
		fsc->fdblocks -= pag->pag_meta_resv.ar_reserved;
 | 
							fsc->fdblocks -= pag->pag_meta_resv.ar_reserved;
 | 
				
			||||||
		fsc->fdblocks -= pag->pag_rmapbt_resv.ar_orig_reserved;
 | 
							fsc->fdblocks -= pag->pag_rmapbt_resv.ar_orig_reserved;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (xchk_should_terminate(sc, &error))
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if (pag)
 | 
				
			||||||
 | 
							xfs_perag_put(pag);
 | 
				
			||||||
	if (error)
 | 
						if (error)
 | 
				
			||||||
		return error;
 | 
							return error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -605,12 +605,11 @@ void
 | 
				
			||||||
xfs_extent_busy_wait_all(
 | 
					xfs_extent_busy_wait_all(
 | 
				
			||||||
	struct xfs_mount	*mp)
 | 
						struct xfs_mount	*mp)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
						struct xfs_perag	*pag;
 | 
				
			||||||
	DEFINE_WAIT		(wait);
 | 
						DEFINE_WAIT		(wait);
 | 
				
			||||||
	xfs_agnumber_t		agno;
 | 
						xfs_agnumber_t		agno;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		struct xfs_perag *pag = xfs_perag_get(mp, agno);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		do {
 | 
							do {
 | 
				
			||||||
			prepare_to_wait(&pag->pagb_wait, &wait, TASK_KILLABLE);
 | 
								prepare_to_wait(&pag->pagb_wait, &wait, TASK_KILLABLE);
 | 
				
			||||||
			if  (RB_EMPTY_ROOT(&pag->pagb_tree))
 | 
								if  (RB_EMPTY_ROOT(&pag->pagb_tree))
 | 
				
			||||||
| 
						 | 
					@ -618,8 +617,6 @@ xfs_extent_busy_wait_all(
 | 
				
			||||||
			schedule();
 | 
								schedule();
 | 
				
			||||||
		} while (1);
 | 
							} while (1);
 | 
				
			||||||
		finish_wait(&pag->pagb_wait, &wait);
 | 
							finish_wait(&pag->pagb_wait, &wait);
 | 
				
			||||||
 | 
					 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -576,10 +576,8 @@ xfs_fs_reserve_ag_blocks(
 | 
				
			||||||
	int			err2;
 | 
						int			err2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	mp->m_finobt_nores = false;
 | 
						mp->m_finobt_nores = false;
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		pag = xfs_perag_get(mp, agno);
 | 
					 | 
				
			||||||
		err2 = xfs_ag_resv_init(pag, NULL);
 | 
							err2 = xfs_ag_resv_init(pag, NULL);
 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
		if (err2 && !error)
 | 
							if (err2 && !error)
 | 
				
			||||||
			error = err2;
 | 
								error = err2;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -605,10 +603,8 @@ xfs_fs_unreserve_ag_blocks(
 | 
				
			||||||
	int			error = 0;
 | 
						int			error = 0;
 | 
				
			||||||
	int			err2;
 | 
						int			err2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		pag = xfs_perag_get(mp, agno);
 | 
					 | 
				
			||||||
		err2 = xfs_ag_resv_free(pag);
 | 
							err2 = xfs_ag_resv_free(pag);
 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
		if (err2 && !error)
 | 
							if (err2 && !error)
 | 
				
			||||||
			error = err2;
 | 
								error = err2;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -34,14 +34,12 @@ xfs_health_unmount(
 | 
				
			||||||
		return;
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Measure AG corruption levels. */
 | 
						/* Measure AG corruption levels. */
 | 
				
			||||||
	for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
 | 
						for_each_perag(mp, agno, pag) {
 | 
				
			||||||
		pag = xfs_perag_get(mp, agno);
 | 
					 | 
				
			||||||
		xfs_ag_measure_sickness(pag, &sick, &checked);
 | 
							xfs_ag_measure_sickness(pag, &sick, &checked);
 | 
				
			||||||
		if (sick) {
 | 
							if (sick) {
 | 
				
			||||||
			trace_xfs_ag_unfixed_corruption(mp, agno, sick);
 | 
								trace_xfs_ag_unfixed_corruption(mp, agno, sick);
 | 
				
			||||||
			warn = true;
 | 
								warn = true;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Measure realtime volume corruption levels. */
 | 
						/* Measure realtime volume corruption levels. */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1061,15 +1061,13 @@ xfs_reclaim_inodes_ag(
 | 
				
			||||||
	int			*nr_to_scan)
 | 
						int			*nr_to_scan)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct xfs_perag	*pag;
 | 
						struct xfs_perag	*pag;
 | 
				
			||||||
	xfs_agnumber_t		ag = 0;
 | 
						xfs_agnumber_t		agno;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	while ((pag = xfs_perag_get_tag(mp, ag, XFS_ICI_RECLAIM_TAG))) {
 | 
						for_each_perag_tag(mp, agno, pag, XFS_ICI_RECLAIM_TAG) {
 | 
				
			||||||
		unsigned long	first_index = 0;
 | 
							unsigned long	first_index = 0;
 | 
				
			||||||
		int		done = 0;
 | 
							int		done = 0;
 | 
				
			||||||
		int		nr_found = 0;
 | 
							int		nr_found = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		ag = pag->pag_agno + 1;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		first_index = READ_ONCE(pag->pag_ici_reclaim_cursor);
 | 
							first_index = READ_ONCE(pag->pag_ici_reclaim_cursor);
 | 
				
			||||||
		do {
 | 
							do {
 | 
				
			||||||
			struct xfs_inode *batch[XFS_LOOKUP_BATCH];
 | 
								struct xfs_inode *batch[XFS_LOOKUP_BATCH];
 | 
				
			||||||
| 
						 | 
					@ -1134,7 +1132,6 @@ xfs_reclaim_inodes_ag(
 | 
				
			||||||
		if (done)
 | 
							if (done)
 | 
				
			||||||
			first_index = 0;
 | 
								first_index = 0;
 | 
				
			||||||
		WRITE_ONCE(pag->pag_ici_reclaim_cursor, first_index);
 | 
							WRITE_ONCE(pag->pag_ici_reclaim_cursor, first_index);
 | 
				
			||||||
		xfs_perag_put(pag);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1554,14 +1551,6 @@ xfs_inode_clear_cowblocks_tag(
 | 
				
			||||||
	return xfs_blockgc_clear_iflag(ip, XFS_ICOWBLOCKS);
 | 
						return xfs_blockgc_clear_iflag(ip, XFS_ICOWBLOCKS);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define for_each_perag_tag(mp, next_agno, pag, tag) \
 | 
					 | 
				
			||||||
	for ((next_agno) = 0, (pag) = xfs_perag_get_tag((mp), 0, (tag)); \
 | 
					 | 
				
			||||||
		(pag) != NULL; \
 | 
					 | 
				
			||||||
		(next_agno) = (pag)->pag_agno + 1, \
 | 
					 | 
				
			||||||
		xfs_perag_put(pag), \
 | 
					 | 
				
			||||||
		(pag) = xfs_perag_get_tag((mp), (next_agno), (tag)))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/* Disable post-EOF and CoW block auto-reclamation. */
 | 
					/* Disable post-EOF and CoW block auto-reclamation. */
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
xfs_blockgc_stop(
 | 
					xfs_blockgc_stop(
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue