Commit 61aa005a authored by Dave Chinner's avatar Dave Chinner Committed by Dave Chinner

xfs: prepare for moving perag definitions and support to libxfs

The perag structures really need to be defined with the rest of the
AG support infrastructure. The struct xfs_perag and init/teardown
has been placed in xfs_mount.[ch] because there are differences in
the structure between kernel and userspace. Mainly that userspace
doesn't have a lot of the internal stuff that the kernel has for
caches and discard and other such structures.

However, it makes more sense to move this to libxfs than to keep
this separation because we are now moving to use struct perags
everywhere in the code instead of passing raw agnumber_t values
about. Hence we shoudl really move the support infrastructure to
libxfs/xfs_ag.[ch].

To do this without breaking userspace, first we need to rearrange
the structures and code so that all the kernel specific code is
located together. This makes it simple for userspace to ifdef out
the all the parts it does not need, minimising the code differences
between kernel and userspace. The next commit will do the move...
Signed-off-by: default avatarDave Chinner <dchinner@redhat.com>
Reviewed-by: default avatarBrian Foster <bfoster@redhat.com>
Reviewed-by: default avatarDarrick J. Wong <djwong@kernel.org>
parent 9bbafc71
...@@ -148,9 +148,11 @@ xfs_free_perag( ...@@ -148,9 +148,11 @@ xfs_free_perag(
spin_unlock(&mp->m_perag_lock); spin_unlock(&mp->m_perag_lock);
ASSERT(pag); ASSERT(pag);
ASSERT(atomic_read(&pag->pag_ref) == 0); ASSERT(atomic_read(&pag->pag_ref) == 0);
cancel_delayed_work_sync(&pag->pag_blockgc_work); cancel_delayed_work_sync(&pag->pag_blockgc_work);
xfs_iunlink_destroy(pag); xfs_iunlink_destroy(pag);
xfs_buf_hash_destroy(pag); xfs_buf_hash_destroy(pag);
call_rcu(&pag->rcu_head, __xfs_free_perag); call_rcu(&pag->rcu_head, __xfs_free_perag);
} }
} }
...@@ -175,14 +177,14 @@ xfs_sb_validate_fsb_count( ...@@ -175,14 +177,14 @@ xfs_sb_validate_fsb_count(
int int
xfs_initialize_perag( xfs_initialize_perag(
xfs_mount_t *mp, struct xfs_mount *mp,
xfs_agnumber_t agcount, xfs_agnumber_t agcount,
xfs_agnumber_t *maxagi) xfs_agnumber_t *maxagi)
{ {
struct xfs_perag *pag;
xfs_agnumber_t index; xfs_agnumber_t index;
xfs_agnumber_t first_initialised = NULLAGNUMBER; xfs_agnumber_t first_initialised = NULLAGNUMBER;
xfs_perag_t *pag; int error;
int error = -ENOMEM;
/* /*
* Walk the current per-ag tree so we don't try to initialise AGs * Walk the current per-ag tree so we don't try to initialise AGs
...@@ -203,21 +205,10 @@ xfs_initialize_perag( ...@@ -203,21 +205,10 @@ xfs_initialize_perag(
} }
pag->pag_agno = index; pag->pag_agno = index;
pag->pag_mount = mp; pag->pag_mount = mp;
spin_lock_init(&pag->pag_ici_lock);
INIT_DELAYED_WORK(&pag->pag_blockgc_work, xfs_blockgc_worker);
INIT_RADIX_TREE(&pag->pag_ici_root, GFP_ATOMIC);
error = xfs_buf_hash_init(pag);
if (error)
goto out_free_pag;
init_waitqueue_head(&pag->pagb_wait);
spin_lock_init(&pag->pagb_lock);
pag->pagb_count = 0;
pag->pagb_tree = RB_ROOT;
error = radix_tree_preload(GFP_NOFS); error = radix_tree_preload(GFP_NOFS);
if (error) if (error)
goto out_hash_destroy; goto out_free_pag;
spin_lock(&mp->m_perag_lock); spin_lock(&mp->m_perag_lock);
if (radix_tree_insert(&mp->m_perag_tree, index, pag)) { if (radix_tree_insert(&mp->m_perag_tree, index, pag)) {
...@@ -225,17 +216,32 @@ xfs_initialize_perag( ...@@ -225,17 +216,32 @@ xfs_initialize_perag(
spin_unlock(&mp->m_perag_lock); spin_unlock(&mp->m_perag_lock);
radix_tree_preload_end(); radix_tree_preload_end();
error = -EEXIST; error = -EEXIST;
goto out_hash_destroy; goto out_free_pag;
} }
spin_unlock(&mp->m_perag_lock); spin_unlock(&mp->m_perag_lock);
radix_tree_preload_end(); radix_tree_preload_end();
/* first new pag is fully initialized */
if (first_initialised == NULLAGNUMBER) /* Place kernel structure only init below this point. */
first_initialised = index; spin_lock_init(&pag->pag_ici_lock);
spin_lock_init(&pag->pagb_lock);
spin_lock_init(&pag->pag_state_lock);
INIT_DELAYED_WORK(&pag->pag_blockgc_work, xfs_blockgc_worker);
INIT_RADIX_TREE(&pag->pag_ici_root, GFP_ATOMIC);
init_waitqueue_head(&pag->pagb_wait);
pag->pagb_count = 0;
pag->pagb_tree = RB_ROOT;
error = xfs_buf_hash_init(pag);
if (error)
goto out_remove_pag;
error = xfs_iunlink_init(pag); error = xfs_iunlink_init(pag);
if (error) if (error)
goto out_hash_destroy; goto out_hash_destroy;
spin_lock_init(&pag->pag_state_lock);
/* first new pag is fully initialized */
if (first_initialised == NULLAGNUMBER)
first_initialised = index;
} }
index = xfs_set_inode_alloc(mp, agcount); index = xfs_set_inode_alloc(mp, agcount);
...@@ -248,6 +254,8 @@ xfs_initialize_perag( ...@@ -248,6 +254,8 @@ xfs_initialize_perag(
out_hash_destroy: out_hash_destroy:
xfs_buf_hash_destroy(pag); xfs_buf_hash_destroy(pag);
out_remove_pag:
radix_tree_delete(&mp->m_perag_tree, index);
out_free_pag: out_free_pag:
kmem_free(pag); kmem_free(pag);
out_unwind_new_pags: out_unwind_new_pags:
......
...@@ -338,6 +338,16 @@ typedef struct xfs_perag { ...@@ -338,6 +338,16 @@ typedef struct xfs_perag {
xfs_agino_t pagl_leftrec; xfs_agino_t pagl_leftrec;
xfs_agino_t pagl_rightrec; xfs_agino_t pagl_rightrec;
int pagb_count; /* pagb slots in use */
uint8_t pagf_refcount_level; /* recount btree height */
/* Blocks reserved for all kinds of metadata. */
struct xfs_ag_resv pag_meta_resv;
/* Blocks reserved for the reverse mapping btree. */
struct xfs_ag_resv pag_rmapbt_resv;
/* -- kernel only structures below this line -- */
/* /*
* Bitsets of per-ag metadata that have been checked and/or are sick. * Bitsets of per-ag metadata that have been checked and/or are sick.
* Callers should hold pag_state_lock before accessing this field. * Callers should hold pag_state_lock before accessing this field.
...@@ -364,19 +374,10 @@ typedef struct xfs_perag { ...@@ -364,19 +374,10 @@ typedef struct xfs_perag {
/* for rcu-safe freeing */ /* for rcu-safe freeing */
struct rcu_head rcu_head; struct rcu_head rcu_head;
int pagb_count; /* pagb slots in use */
/* Blocks reserved for all kinds of metadata. */
struct xfs_ag_resv pag_meta_resv;
/* Blocks reserved for the reverse mapping btree. */
struct xfs_ag_resv pag_rmapbt_resv;
/* background prealloc block trimming */ /* background prealloc block trimming */
struct delayed_work pag_blockgc_work; struct delayed_work pag_blockgc_work;
/* reference count */
uint8_t pagf_refcount_level;
/* /*
* Unlinked inode information. This incore information reflects * Unlinked inode information. This incore information reflects
* data stored in the AGI, so callers must hold the AGI buffer lock * data stored in the AGI, so callers must hold the AGI buffer lock
......
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