Commit 74b9aa63 authored by Dave Chinner's avatar Dave Chinner

xfs: introduce xfs_alloc_vextent_prepare()

Now that we have wrapper functions for each type of allocation we
can ask for, we can start unravelling xfs_alloc_ag_vextent(). That
is essentially just a prepare stage, the allocation multiplexer
and a post-allocation accounting step is the allocation proceeded.

The current xfs_alloc_vextent*() wrappers all have a prepare stage,
the allocation operation and a post-allocation accounting step.

We can consolidate this by moving the AG alloc prep code into the
wrapper functions, the accounting code in the wrapper accounting
functions, and cut out the multiplexer layer entirely.

This patch consolidates the AG preparation stage.
Signed-off-by: default avatarDave Chinner <dchinner@redhat.com>
Reviewed-by: default avatarDarrick J. Wong <djwong@kernel.org>
parent 5f36b2ce
...@@ -1144,31 +1144,8 @@ static int ...@@ -1144,31 +1144,8 @@ static int
xfs_alloc_ag_vextent( xfs_alloc_ag_vextent(
struct xfs_alloc_arg *args) struct xfs_alloc_arg *args)
{ {
struct xfs_mount *mp = args->mp;
int error = 0; int error = 0;
ASSERT(args->minlen > 0);
ASSERT(args->maxlen > 0);
ASSERT(args->minlen <= args->maxlen);
ASSERT(args->mod < args->prod);
ASSERT(args->alignment > 0);
ASSERT(args->resv != XFS_AG_RESV_AGFL);
error = xfs_alloc_fix_freelist(args, 0);
if (error) {
trace_xfs_alloc_vextent_nofix(args);
return error;
}
if (!args->agbp) {
/* cannot allocate in this AG at all */
trace_xfs_alloc_vextent_noagbp(args);
args->agbno = NULLAGBLOCK;
return 0;
}
args->agbno = XFS_FSB_TO_AGBNO(mp, args->fsbno);
args->wasfromfl = 0;
/* /*
* Branch to correct routine based on the type. * Branch to correct routine based on the type.
*/ */
...@@ -3201,11 +3178,18 @@ xfs_alloc_vextent_check_args( ...@@ -3201,11 +3178,18 @@ xfs_alloc_vextent_check_args(
args->maxlen = agsize; args->maxlen = agsize;
if (args->alignment == 0) if (args->alignment == 0)
args->alignment = 1; args->alignment = 1;
ASSERT(args->minlen > 0);
ASSERT(args->maxlen > 0);
ASSERT(args->alignment > 0);
ASSERT(args->resv != XFS_AG_RESV_AGFL);
ASSERT(XFS_FSB_TO_AGNO(mp, target) < mp->m_sb.sb_agcount); ASSERT(XFS_FSB_TO_AGNO(mp, target) < mp->m_sb.sb_agcount);
ASSERT(XFS_FSB_TO_AGBNO(mp, target) < agsize); ASSERT(XFS_FSB_TO_AGBNO(mp, target) < agsize);
ASSERT(args->minlen <= args->maxlen); ASSERT(args->minlen <= args->maxlen);
ASSERT(args->minlen <= agsize); ASSERT(args->minlen <= agsize);
ASSERT(args->mod < args->prod); ASSERT(args->mod < args->prod);
if (XFS_FSB_TO_AGNO(mp, target) >= mp->m_sb.sb_agcount || if (XFS_FSB_TO_AGNO(mp, target) >= mp->m_sb.sb_agcount ||
XFS_FSB_TO_AGBNO(mp, target) >= agsize || XFS_FSB_TO_AGBNO(mp, target) >= agsize ||
args->minlen > args->maxlen || args->minlen > agsize || args->minlen > args->maxlen || args->minlen > agsize ||
...@@ -3217,6 +3201,41 @@ xfs_alloc_vextent_check_args( ...@@ -3217,6 +3201,41 @@ xfs_alloc_vextent_check_args(
return 0; return 0;
} }
/*
* Prepare an AG for allocation. If the AG is not prepared to accept the
* allocation, return failure.
*
* XXX(dgc): The complexity of "need_pag" will go away as all caller paths are
* modified to hold their own perag references.
*/
static int
xfs_alloc_vextent_prepare_ag(
struct xfs_alloc_arg *args)
{
bool need_pag = !args->pag;
int error;
if (need_pag)
args->pag = xfs_perag_get(args->mp, args->agno);
error = xfs_alloc_fix_freelist(args, 0);
if (error) {
trace_xfs_alloc_vextent_nofix(args);
if (need_pag)
xfs_perag_put(args->pag);
args->agbno = NULLAGBLOCK;
return error;
}
if (!args->agbp) {
/* cannot allocate in this AG at all */
trace_xfs_alloc_vextent_noagbp(args);
args->agbno = NULLAGBLOCK;
return 0;
}
args->wasfromfl = 0;
return 0;
}
/* /*
* Post-process allocation results to set the allocated block number correctly * Post-process allocation results to set the allocated block number correctly
* for the caller. * for the caller.
...@@ -3268,7 +3287,8 @@ xfs_alloc_vextent_set_fsbno( ...@@ -3268,7 +3287,8 @@ xfs_alloc_vextent_set_fsbno(
} }
/* /*
* Allocate within a single AG only. * Allocate within a single AG only. Caller is expected to hold a
* perag reference in args->pag.
*/ */
int int
xfs_alloc_vextent_this_ag( xfs_alloc_vextent_this_ag(
...@@ -3301,7 +3321,10 @@ xfs_alloc_vextent_this_ag( ...@@ -3301,7 +3321,10 @@ xfs_alloc_vextent_this_ag(
args->fsbno = target; args->fsbno = target;
args->type = XFS_ALLOCTYPE_THIS_AG; args->type = XFS_ALLOCTYPE_THIS_AG;
error = xfs_alloc_vextent_prepare_ag(args);
if (!error && args->agbp)
error = xfs_alloc_ag_vextent(args); error = xfs_alloc_ag_vextent(args);
xfs_alloc_vextent_set_fsbno(args, minimum_agno); xfs_alloc_vextent_set_fsbno(args, minimum_agno);
return error; return error;
} }
...@@ -3339,13 +3362,19 @@ xfs_alloc_vextent_iterate_ags( ...@@ -3339,13 +3362,19 @@ xfs_alloc_vextent_iterate_ags(
args->agno = start_agno; args->agno = start_agno;
for (;;) { for (;;) {
args->pag = xfs_perag_get(mp, args->agno); args->pag = xfs_perag_get(mp, args->agno);
args->agbno = XFS_FSB_TO_AGBNO(mp, args->fsbno);
error = xfs_alloc_vextent_prepare_ag(args);
if (error)
break;
if (args->agbp) {
/*
* Allocation is supposed to succeed now, so break out
* of the loop regardless of whether we succeed or not.
*/
error = xfs_alloc_ag_vextent(args); error = xfs_alloc_ag_vextent(args);
if (error) {
args->agbno = NULLAGBLOCK;
break; break;
} }
if (args->agbp)
break;
trace_xfs_alloc_vextent_loopfailed(args); trace_xfs_alloc_vextent_loopfailed(args);
...@@ -3378,11 +3407,9 @@ xfs_alloc_vextent_iterate_ags( ...@@ -3378,11 +3407,9 @@ xfs_alloc_vextent_iterate_ags(
} }
flags = 0; flags = 0;
if (args->otype == XFS_ALLOCTYPE_NEAR_BNO) { if (args->otype == XFS_ALLOCTYPE_NEAR_BNO)
args->agbno = XFS_FSB_TO_AGBNO(mp, args->fsbno);
args->type = XFS_ALLOCTYPE_NEAR_BNO; args->type = XFS_ALLOCTYPE_NEAR_BNO;
} }
}
xfs_perag_put(args->pag); xfs_perag_put(args->pag);
args->pag = NULL; args->pag = NULL;
} }
...@@ -3485,7 +3512,8 @@ xfs_alloc_vextent_first_ag( ...@@ -3485,7 +3512,8 @@ xfs_alloc_vextent_first_ag(
} }
/* /*
* Allocate within a single AG only. * Allocate at the exact block target or fail. Caller is expected to hold a
* perag reference in args->pag.
*/ */
int int
xfs_alloc_vextent_exact_bno( xfs_alloc_vextent_exact_bno(
...@@ -3515,9 +3543,10 @@ xfs_alloc_vextent_exact_bno( ...@@ -3515,9 +3543,10 @@ xfs_alloc_vextent_exact_bno(
args->agbno = XFS_FSB_TO_AGBNO(mp, target); args->agbno = XFS_FSB_TO_AGBNO(mp, target);
args->fsbno = target; args->fsbno = target;
args->type = XFS_ALLOCTYPE_THIS_BNO; args->type = XFS_ALLOCTYPE_THIS_BNO;
error = xfs_alloc_vextent_prepare_ag(args);
if (!error && args->agbp)
error = xfs_alloc_ag_vextent(args); error = xfs_alloc_ag_vextent(args);
if (error)
return error;
xfs_alloc_vextent_set_fsbno(args, minimum_agno); xfs_alloc_vextent_set_fsbno(args, minimum_agno);
return 0; return 0;
...@@ -3526,6 +3555,8 @@ xfs_alloc_vextent_exact_bno( ...@@ -3526,6 +3555,8 @@ xfs_alloc_vextent_exact_bno(
/* /*
* Allocate an extent as close to the target as possible. If there are not * Allocate an extent as close to the target as possible. If there are not
* viable candidates in the AG, then fail the allocation. * viable candidates in the AG, then fail the allocation.
*
* Caller may or may not have a per-ag reference in args->pag.
*/ */
int int
xfs_alloc_vextent_near_bno( xfs_alloc_vextent_near_bno(
...@@ -3550,21 +3581,22 @@ xfs_alloc_vextent_near_bno( ...@@ -3550,21 +3581,22 @@ xfs_alloc_vextent_near_bno(
args->agno = XFS_FSB_TO_AGNO(mp, target); args->agno = XFS_FSB_TO_AGNO(mp, target);
if (minimum_agno > args->agno) { if (minimum_agno > args->agno) {
trace_xfs_alloc_vextent_skip_deadlock(args); trace_xfs_alloc_vextent_skip_deadlock(args);
args->fsbno = NULLFSBLOCK;
return 0; return 0;
} }
args->agbno = XFS_FSB_TO_AGBNO(mp, target); args->agbno = XFS_FSB_TO_AGBNO(mp, target);
args->type = XFS_ALLOCTYPE_NEAR_BNO; args->type = XFS_ALLOCTYPE_NEAR_BNO;
if (need_pag)
args->pag = xfs_perag_get(args->mp, args->agno); error = xfs_alloc_vextent_prepare_ag(args);
if (!error && args->agbp)
error = xfs_alloc_ag_vextent(args); error = xfs_alloc_ag_vextent(args);
xfs_alloc_vextent_set_fsbno(args, minimum_agno);
if (need_pag) if (need_pag)
xfs_perag_put(args->pag); xfs_perag_put(args->pag);
if (error)
return error;
xfs_alloc_vextent_set_fsbno(args, minimum_agno); return error;
return 0;
} }
/* Ensure that the freelist is at full capacity. */ /* Ensure that the freelist is at full capacity. */
......
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