2018-05-13 23:10:08 -07:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0 */
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Red Hat, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __LIBXFS_AG_H
|
|
|
|
#define __LIBXFS_AG_H 1
|
|
|
|
|
2024-11-03 20:18:38 -08:00
|
|
|
#include "xfs_group.h"
|
|
|
|
|
2018-05-13 23:10:08 -07:00
|
|
|
struct xfs_mount;
|
|
|
|
struct xfs_trans;
|
2021-06-02 10:48:24 +10:00
|
|
|
struct xfs_perag;
|
|
|
|
|
|
|
|
/*
|
2021-06-02 10:48:24 +10:00
|
|
|
* Per-ag infrastructure
|
2021-06-02 10:48:24 +10:00
|
|
|
*/
|
2021-06-02 10:48:24 +10:00
|
|
|
|
|
|
|
/* per-AG block reservation data structures*/
|
|
|
|
struct xfs_ag_resv {
|
|
|
|
/* number of blocks originally reserved here */
|
|
|
|
xfs_extlen_t ar_orig_reserved;
|
|
|
|
/* number of blocks reserved here */
|
|
|
|
xfs_extlen_t ar_reserved;
|
|
|
|
/* number of blocks originally asked for */
|
|
|
|
xfs_extlen_t ar_asked;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Per-ag incore structure, copies of information in agf and agi, to improve the
|
|
|
|
* performance of allocation group selection.
|
|
|
|
*/
|
2021-06-02 10:48:51 +10:00
|
|
|
struct xfs_perag {
|
2024-11-03 20:18:38 -08:00
|
|
|
struct xfs_group pag_group;
|
2023-02-13 09:14:52 +11:00
|
|
|
unsigned long pag_opstate;
|
2024-02-22 12:39:46 -08:00
|
|
|
uint8_t pagf_bno_level; /* # of levels in bno btree */
|
|
|
|
uint8_t pagf_cnt_level; /* # of levels in cnt btree */
|
|
|
|
uint8_t pagf_rmap_level;/* # of levels in rmap btree */
|
2021-06-02 10:48:24 +10:00
|
|
|
uint32_t pagf_flcount; /* count of blocks in freelist */
|
|
|
|
xfs_extlen_t pagf_freeblks; /* total free blocks */
|
|
|
|
xfs_extlen_t pagf_longest; /* longest free space */
|
|
|
|
uint32_t pagf_btreeblks; /* # of blocks held in AGF btrees */
|
|
|
|
xfs_agino_t pagi_freecount; /* number of free inodes */
|
|
|
|
xfs_agino_t pagi_count; /* number of allocated inodes */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Inode allocation search lookup optimisation.
|
|
|
|
* If the pagino matches, the search for new inodes
|
|
|
|
* doesn't need to search the near ones again straight away
|
|
|
|
*/
|
|
|
|
xfs_agino_t pagl_pagino;
|
|
|
|
xfs_agino_t pagl_leftrec;
|
|
|
|
xfs_agino_t pagl_rightrec;
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
2022-07-07 19:13:02 +10:00
|
|
|
/* Precalculated geometry info */
|
2022-07-07 19:13:10 +10:00
|
|
|
xfs_agino_t agino_min;
|
|
|
|
xfs_agino_t agino_max;
|
2022-07-07 19:13:02 +10:00
|
|
|
|
2021-11-09 18:18:50 -08:00
|
|
|
#ifdef __KERNEL__
|
2021-06-02 10:48:24 +10:00
|
|
|
/* -- kernel only structures below this line -- */
|
|
|
|
|
2023-12-15 10:03:32 -08:00
|
|
|
#ifdef CONFIG_XFS_ONLINE_REPAIR
|
|
|
|
/*
|
|
|
|
* Alternate btree heights so that online repair won't trip the write
|
|
|
|
* verifiers while rebuilding the AG btrees.
|
|
|
|
*/
|
2024-02-22 12:39:46 -08:00
|
|
|
uint8_t pagf_repair_bno_level;
|
|
|
|
uint8_t pagf_repair_cnt_level;
|
2023-12-15 10:03:33 -08:00
|
|
|
uint8_t pagf_repair_refcount_level;
|
2024-02-22 12:43:38 -08:00
|
|
|
uint8_t pagf_repair_rmap_level;
|
2023-12-15 10:03:32 -08:00
|
|
|
#endif
|
|
|
|
|
2021-06-02 10:48:24 +10:00
|
|
|
atomic_t pagf_fstrms; /* # of filestreams active in this AG */
|
|
|
|
|
|
|
|
spinlock_t pag_ici_lock; /* incore inode cache lock */
|
|
|
|
struct radix_tree_root pag_ici_root; /* incore inode cache root */
|
|
|
|
int pag_ici_reclaimable; /* reclaimable inodes */
|
|
|
|
unsigned long pag_ici_reclaim_cursor; /* reclaim restart point */
|
|
|
|
|
2024-02-22 12:42:58 -08:00
|
|
|
struct xfs_buf_cache pag_bcache;
|
2021-06-02 10:48:24 +10:00
|
|
|
|
|
|
|
/* background prealloc block trimming */
|
|
|
|
struct delayed_work pag_blockgc_work;
|
2021-11-09 18:18:50 -08:00
|
|
|
#endif /* __KERNEL__ */
|
2021-06-02 10:48:51 +10:00
|
|
|
};
|
2021-06-02 10:48:24 +10:00
|
|
|
|
2024-11-03 20:18:38 -08:00
|
|
|
static inline struct xfs_perag *to_perag(struct xfs_group *xg)
|
|
|
|
{
|
|
|
|
return container_of(xg, struct xfs_perag, pag_group);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct xfs_group *pag_group(struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
return &pag->pag_group;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct xfs_mount *pag_mount(const struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
return pag->pag_group.xg_mount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xfs_agnumber_t pag_agno(const struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
return pag->pag_group.xg_gno;
|
|
|
|
}
|
|
|
|
|
2023-02-13 09:14:52 +11:00
|
|
|
/*
|
|
|
|
* Per-AG operational state. These are atomic flag bits.
|
|
|
|
*/
|
|
|
|
#define XFS_AGSTATE_AGF_INIT 0
|
|
|
|
#define XFS_AGSTATE_AGI_INIT 1
|
|
|
|
#define XFS_AGSTATE_PREFERS_METADATA 2
|
|
|
|
#define XFS_AGSTATE_ALLOWS_INODES 3
|
|
|
|
#define XFS_AGSTATE_AGFL_NEEDS_RESET 4
|
|
|
|
|
|
|
|
#define __XFS_AG_OPSTATE(name, NAME) \
|
|
|
|
static inline bool xfs_perag_ ## name (struct xfs_perag *pag) \
|
|
|
|
{ \
|
|
|
|
return test_bit(XFS_AGSTATE_ ## NAME, &pag->pag_opstate); \
|
|
|
|
}
|
|
|
|
|
|
|
|
__XFS_AG_OPSTATE(initialised_agf, AGF_INIT)
|
|
|
|
__XFS_AG_OPSTATE(initialised_agi, AGI_INIT)
|
|
|
|
__XFS_AG_OPSTATE(prefers_metadata, PREFERS_METADATA)
|
|
|
|
__XFS_AG_OPSTATE(allows_inodes, ALLOWS_INODES)
|
|
|
|
__XFS_AG_OPSTATE(agfl_needs_reset, AGFL_NEEDS_RESET)
|
|
|
|
|
2024-11-03 20:18:36 -08:00
|
|
|
int xfs_initialize_perag(struct xfs_mount *mp, xfs_agnumber_t orig_agcount,
|
|
|
|
xfs_agnumber_t new_agcount, xfs_rfsblock_t dcount,
|
2024-10-14 08:04:50 +02:00
|
|
|
xfs_agnumber_t *maxagi);
|
2024-10-14 08:04:51 +02:00
|
|
|
void xfs_free_perag_range(struct xfs_mount *mp, xfs_agnumber_t first_agno,
|
|
|
|
xfs_agnumber_t end_agno);
|
xfs: clean up incore inode walk functions
This ambitious series aims to cleans up redundant inode walk code in
xfs_icache.c, hide implementation details of the quotaoff dquot release
code, and eliminates indirect function calls from incore inode walks.
The first thing it does is to move all the code that quotaoff calls to
release dquots from all incore inodes into xfs_icache.c. Next, it
separates the goal of an inode walk from the actual radix tree tags that
may or may not be involved and drops the kludgy XFS_ICI_NO_TAG thing.
Finally, we split the speculative preallocation (blockgc) and quotaoff
dquot release code paths into separate functions so that we can keep the
implementations cohesive.
Christoph suggested last cycle that we 'simply' change quotaoff not to
allow deactivating quota entirely, but as these cleanups are to enable
one major change in behavior (deferred inode inactivation) I do not want
to add a second behavior change (quotaoff) as a dependency.
To be blunt: Additional cleanups are not in scope for this series.
Next, I made two observations about incore inode radix tree walks --
since there's a 1:1 mapping between the walk goal and the per-inode
processing function passed in, we can use the goal to make a direct call
to the processing function. Furthermore, the only caller to supply a
nonzero iter_flags argument is quotaoff, and there's only one INEW flag.
From that observation, I concluded that it's quite possible to remove
two parameters from the xfs_inode_walk* function signatures -- the
iter_flags, and the execute function pointer. The middle of the series
moves the INEW functionality into the one piece (quotaoff) that wants
it, and removes the indirect calls.
The final observation is that the inode reclaim walk loop is now almost
the same as xfs_inode_walk, so it's silly to maintain two copies. Merge
the reclaim loop code into xfs_inode_walk.
Lastly, refactor the per-ag radix tagging functions since there's
duplicated code that can be consolidated.
This series is a prerequisite for the next two patchsets, since deferred
inode inactivation will add another inode radix tree tag and iterator
function to xfs_inode_walk.
v2: walk the vfs inode list when running quotaoff instead of the radix
tree, then rework the (now completely internal) inode walk function
to take the tag as the main parameter.
v3: merge the reclaim loop into xfs_inode_walk, then consolidate the
radix tree tagging functions
v4: rebase to 5.13-rc4
v5: combine with the quotaoff patchset, reorder functions to minimize
forward declarations, split inode walk goals from radix tree tags
to reduce conceptual confusion
v6: start moving the inode cache code towards the xfs_icwalk prefix
-----BEGIN PGP SIGNATURE-----
iQIzBAABCgAdFiEEUzaAxoMeQq6m2jMV+H93GTRKtOsFAmC5Yv0ACgkQ+H93GTRK
tOv7Fg//Z7cKph0zSg6qsukMEMZxscuNcEBydCW1bu9gSx1NpszDpiGqAiO5ZB3X
wP2XkCqjuatbNGGvkNLHS/M4sbLX3ELogvYmMRvUhDoaSFxT/KKgxvsyNffiCSS7
xRB/rvWRp9MGRpBWPF0ZUxFU6VBzhCrYdMsNhvW95AEup8S/j+NplwoIif0gzaZZ
Q6Fl4Ca9VEBvJQPV+/zkLih19iFItmARJhPHUs4BO1nZv+CzZBFQHg7Ijw7nW92j
eSY68W4LH/IQ5cqm+HrD/+Z6ns0P7J2viewzVymkNEGnuX4a0xrQrzQ8ydRsAxTi
9EDrpIe3MbSI5YjJfmRe8G3LX5p7vBpqc8TeyZdRDMGWkFjT33HPlQNb6WxKLQbA
mjKdfr8AYZR/UQKW/7oZFrJnOoMpYRAQ4Sn/9BAYZQYm7tiLzuZsrEZ7JBwiUA56
XHmlsDDeLzJeKvjmUu8M3H4oh4Nwf5/I2vJwHjueTfhl83uJP04igIXC4rnq56bM
AAAjH9uV11Fo3q0ywAnRtN2HYj8PEJlCMK5CNskILrGeMITsBPGht0SbaA6hDI2h
GYmltKInHzuPhHC9NfyPVrVr3BrmPR5cBsVFESiz5A4E9rbuKmmna6Yk8MFlMyl8
FRIA3zVatJ2qQXtsAcdI8AZzMd7ciYhkAgCqFKxv8qK/qxITHh4=
=Rxdn
-----END PGP SIGNATURE-----
Merge tag 'inode-walk-cleanups-5.14_2021-06-03' of https://git.kernel.org/pub/scm/linux/kernel/git/djwong/xfs-linux into xfs-5.14-merge2
xfs: clean up incore inode walk functions
This ambitious series aims to cleans up redundant inode walk code in
xfs_icache.c, hide implementation details of the quotaoff dquot release
code, and eliminates indirect function calls from incore inode walks.
The first thing it does is to move all the code that quotaoff calls to
release dquots from all incore inodes into xfs_icache.c. Next, it
separates the goal of an inode walk from the actual radix tree tags that
may or may not be involved and drops the kludgy XFS_ICI_NO_TAG thing.
Finally, we split the speculative preallocation (blockgc) and quotaoff
dquot release code paths into separate functions so that we can keep the
implementations cohesive.
Christoph suggested last cycle that we 'simply' change quotaoff not to
allow deactivating quota entirely, but as these cleanups are to enable
one major change in behavior (deferred inode inactivation) I do not want
to add a second behavior change (quotaoff) as a dependency.
To be blunt: Additional cleanups are not in scope for this series.
Next, I made two observations about incore inode radix tree walks --
since there's a 1:1 mapping between the walk goal and the per-inode
processing function passed in, we can use the goal to make a direct call
to the processing function. Furthermore, the only caller to supply a
nonzero iter_flags argument is quotaoff, and there's only one INEW flag.
From that observation, I concluded that it's quite possible to remove
two parameters from the xfs_inode_walk* function signatures -- the
iter_flags, and the execute function pointer. The middle of the series
moves the INEW functionality into the one piece (quotaoff) that wants
it, and removes the indirect calls.
The final observation is that the inode reclaim walk loop is now almost
the same as xfs_inode_walk, so it's silly to maintain two copies. Merge
the reclaim loop code into xfs_inode_walk.
Lastly, refactor the per-ag radix tagging functions since there's
duplicated code that can be consolidated.
This series is a prerequisite for the next two patchsets, since deferred
inode inactivation will add another inode radix tree tag and iterator
function to xfs_inode_walk.
v2: walk the vfs inode list when running quotaoff instead of the radix
tree, then rework the (now completely internal) inode walk function
to take the tag as the main parameter.
v3: merge the reclaim loop into xfs_inode_walk, then consolidate the
radix tree tagging functions
v4: rebase to 5.13-rc4
v5: combine with the quotaoff patchset, reorder functions to minimize
forward declarations, split inode walk goals from radix tree tags
to reduce conceptual confusion
v6: start moving the inode cache code towards the xfs_icwalk prefix
* tag 'inode-walk-cleanups-5.14_2021-06-03' of https://git.kernel.org/pub/scm/linux/kernel/git/djwong/xfs-linux:
xfs: refactor per-AG inode tagging functions
xfs: merge xfs_reclaim_inodes_ag into xfs_inode_walk_ag
xfs: pass struct xfs_eofblocks to the inode scan callback
xfs: fix radix tree tag signs
xfs: make the icwalk processing functions clean up the grab state
xfs: clean up inode state flag tests in xfs_blockgc_igrab
xfs: remove indirect calls from xfs_inode_walk{,_ag}
xfs: remove iter_flags parameter from xfs_inode_walk_*
xfs: move xfs_inew_wait call into xfs_dqrele_inode
xfs: separate the dqrele_all inode grab logic from xfs_inode_walk_ag_grab
xfs: pass the goal of the incore inode walk to xfs_inode_walk()
xfs: rename xfs_inode_walk functions to xfs_icwalk
xfs: move the inode walk functions further down
xfs: detach inode dquots at the end of inactivation
xfs: move the quotaoff dqrele inode walk into xfs_icache.c
[djwong: added variable names to function declarations while fixing
merge conflicts]
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2021-06-08 09:26:44 -07:00
|
|
|
int xfs_initialize_perag_data(struct xfs_mount *mp, xfs_agnumber_t agno);
|
2024-10-14 08:04:55 +02:00
|
|
|
int xfs_update_last_ag_size(struct xfs_mount *mp, xfs_agnumber_t prev_agcount);
|
2021-06-02 10:48:24 +10:00
|
|
|
|
2023-02-13 09:14:42 +11:00
|
|
|
/* Passive AG references */
|
2024-11-03 20:18:38 -08:00
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_get(
|
|
|
|
struct xfs_mount *mp,
|
|
|
|
xfs_agnumber_t agno)
|
|
|
|
{
|
|
|
|
return to_perag(xfs_group_get(mp, agno, XG_TYPE_AG));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_hold(
|
|
|
|
struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
return to_perag(xfs_group_hold(pag_group(pag)));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
xfs_perag_put(
|
|
|
|
struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
xfs_group_put(pag_group(pag));
|
|
|
|
}
|
2018-05-13 23:10:08 -07:00
|
|
|
|
2023-02-13 09:14:42 +11:00
|
|
|
/* Active AG references */
|
2024-11-03 20:18:38 -08:00
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_grab(
|
|
|
|
struct xfs_mount *mp,
|
|
|
|
xfs_agnumber_t agno)
|
|
|
|
{
|
|
|
|
return to_perag(xfs_group_grab(mp, agno, XG_TYPE_AG));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
xfs_perag_rele(
|
|
|
|
struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
xfs_group_rele(pag_group(pag));
|
|
|
|
}
|
2023-02-13 09:14:42 +11:00
|
|
|
|
2024-11-03 20:18:39 -08:00
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_next_range(
|
|
|
|
struct xfs_mount *mp,
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agnumber_t start_agno,
|
|
|
|
xfs_agnumber_t end_agno)
|
|
|
|
{
|
|
|
|
return to_perag(xfs_group_next_range(mp, pag ? pag_group(pag) : NULL,
|
|
|
|
start_agno, end_agno, XG_TYPE_AG));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_next_from(
|
|
|
|
struct xfs_mount *mp,
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agnumber_t start_agno)
|
|
|
|
{
|
|
|
|
return xfs_perag_next_range(mp, pag, start_agno, mp->m_sb.sb_agcount - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_next(
|
|
|
|
struct xfs_mount *mp,
|
|
|
|
struct xfs_perag *pag)
|
|
|
|
{
|
|
|
|
return xfs_perag_next_from(mp, pag, 0);
|
|
|
|
}
|
|
|
|
|
2022-07-07 19:13:02 +10:00
|
|
|
/*
|
|
|
|
* Per-ag geometry infomation and validation
|
|
|
|
*/
|
|
|
|
xfs_agblock_t xfs_ag_block_count(struct xfs_mount *mp, xfs_agnumber_t agno);
|
2022-07-07 19:13:10 +10:00
|
|
|
void xfs_agino_range(struct xfs_mount *mp, xfs_agnumber_t agno,
|
|
|
|
xfs_agino_t *first, xfs_agino_t *last);
|
2022-07-07 19:13:02 +10:00
|
|
|
|
|
|
|
static inline bool
|
|
|
|
xfs_verify_agbno(struct xfs_perag *pag, xfs_agblock_t agbno)
|
|
|
|
{
|
2024-11-03 20:19:35 -08:00
|
|
|
return xfs_verify_gbno(pag_group(pag), agbno);
|
2022-07-07 19:13:02 +10:00
|
|
|
}
|
|
|
|
|
2022-10-28 15:48:58 -07:00
|
|
|
static inline bool
|
|
|
|
xfs_verify_agbext(
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agblock_t agbno,
|
|
|
|
xfs_agblock_t len)
|
|
|
|
{
|
2024-11-03 20:19:35 -08:00
|
|
|
return xfs_verify_gbext(pag_group(pag), agbno, len);
|
2022-10-28 15:48:58 -07:00
|
|
|
}
|
|
|
|
|
2022-07-07 19:13:10 +10:00
|
|
|
/*
|
|
|
|
* Verify that an AG inode number pointer neither points outside the AG
|
|
|
|
* nor points at static metadata.
|
|
|
|
*/
|
|
|
|
static inline bool
|
|
|
|
xfs_verify_agino(struct xfs_perag *pag, xfs_agino_t agino)
|
|
|
|
{
|
|
|
|
if (agino < pag->agino_min)
|
|
|
|
return false;
|
|
|
|
if (agino > pag->agino_max)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify that an AG inode number pointer neither points outside the AG
|
|
|
|
* nor points at static metadata, or is NULLAGINO.
|
|
|
|
*/
|
|
|
|
static inline bool
|
|
|
|
xfs_verify_agino_or_null(struct xfs_perag *pag, xfs_agino_t agino)
|
|
|
|
{
|
|
|
|
if (agino == NULLAGINO)
|
|
|
|
return true;
|
|
|
|
return xfs_verify_agino(pag, agino);
|
|
|
|
}
|
|
|
|
|
2022-07-07 19:13:21 +10:00
|
|
|
static inline bool
|
|
|
|
xfs_ag_contains_log(struct xfs_mount *mp, xfs_agnumber_t agno)
|
|
|
|
{
|
|
|
|
return mp->m_sb.sb_logstart > 0 &&
|
|
|
|
agno == XFS_FSB_TO_AGNO(mp, mp->m_sb.sb_logstart);
|
|
|
|
}
|
|
|
|
|
2023-02-13 09:14:53 +11:00
|
|
|
static inline struct xfs_perag *
|
|
|
|
xfs_perag_next_wrap(
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agnumber_t *agno,
|
|
|
|
xfs_agnumber_t stop_agno,
|
2023-02-13 09:14:54 +11:00
|
|
|
xfs_agnumber_t restart_agno,
|
2023-02-13 09:14:53 +11:00
|
|
|
xfs_agnumber_t wrap_agno)
|
|
|
|
{
|
2024-11-03 20:18:38 -08:00
|
|
|
struct xfs_mount *mp = pag_mount(pag);
|
2023-02-13 09:14:53 +11:00
|
|
|
|
2024-11-03 20:18:38 -08:00
|
|
|
*agno = pag_agno(pag) + 1;
|
2023-02-13 09:14:53 +11:00
|
|
|
xfs_perag_rele(pag);
|
|
|
|
while (*agno != stop_agno) {
|
2023-02-13 09:14:54 +11:00
|
|
|
if (*agno >= wrap_agno) {
|
|
|
|
if (restart_agno >= stop_agno)
|
|
|
|
break;
|
|
|
|
*agno = restart_agno;
|
|
|
|
}
|
2023-02-13 09:14:53 +11:00
|
|
|
|
|
|
|
pag = xfs_perag_grab(mp, *agno);
|
|
|
|
if (pag)
|
|
|
|
return pag;
|
|
|
|
(*agno)++;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2023-02-13 09:14:54 +11:00
|
|
|
* Iterate all AGs from start_agno through wrap_agno, then restart_agno through
|
2023-02-13 09:14:53 +11:00
|
|
|
* (start_agno - 1).
|
|
|
|
*/
|
2023-02-13 09:14:54 +11:00
|
|
|
#define for_each_perag_wrap_range(mp, start_agno, restart_agno, wrap_agno, agno, pag) \
|
2023-02-13 09:14:53 +11:00
|
|
|
for ((agno) = (start_agno), (pag) = xfs_perag_grab((mp), (agno)); \
|
|
|
|
(pag) != NULL; \
|
|
|
|
(pag) = xfs_perag_next_wrap((pag), &(agno), (start_agno), \
|
2023-02-13 09:14:54 +11:00
|
|
|
(restart_agno), (wrap_agno)))
|
|
|
|
/*
|
|
|
|
* Iterate all AGs from start_agno through wrap_agno, then 0 through
|
|
|
|
* (start_agno - 1).
|
|
|
|
*/
|
|
|
|
#define for_each_perag_wrap_at(mp, start_agno, wrap_agno, agno, pag) \
|
|
|
|
for_each_perag_wrap_range((mp), (start_agno), 0, (wrap_agno), (agno), (pag))
|
2023-02-13 09:14:53 +11:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Iterate all AGs from start_agno through to the end of the filesystem, then 0
|
|
|
|
* through (start_agno - 1).
|
|
|
|
*/
|
|
|
|
#define for_each_perag_wrap(mp, start_agno, agno, pag) \
|
|
|
|
for_each_perag_wrap_at((mp), (start_agno), (mp)->m_sb.sb_agcount, \
|
|
|
|
(agno), (pag))
|
|
|
|
|
|
|
|
|
2018-05-13 23:10:08 -07:00
|
|
|
struct aghdr_init_data {
|
|
|
|
/* per ag data */
|
|
|
|
xfs_agblock_t agno; /* ag to init */
|
|
|
|
xfs_extlen_t agsize; /* new AG size */
|
|
|
|
struct list_head buffer_list; /* buffer writeback list */
|
|
|
|
xfs_rfsblock_t nfree; /* cumulative new free space */
|
|
|
|
|
|
|
|
/* per header data */
|
|
|
|
xfs_daddr_t daddr; /* header location */
|
|
|
|
size_t numblks; /* size of header */
|
2024-02-22 12:35:16 -08:00
|
|
|
const struct xfs_btree_ops *bc_ops; /* btree ops */
|
2018-05-13 23:10:08 -07:00
|
|
|
};
|
|
|
|
|
2018-05-13 23:10:08 -07:00
|
|
|
int xfs_ag_init_headers(struct xfs_mount *mp, struct aghdr_init_data *id);
|
2022-07-07 19:07:09 +10:00
|
|
|
int xfs_ag_shrink_space(struct xfs_perag *pag, struct xfs_trans **tpp,
|
|
|
|
xfs_extlen_t delta);
|
|
|
|
int xfs_ag_extend_space(struct xfs_perag *pag, struct xfs_trans *tp,
|
|
|
|
xfs_extlen_t len);
|
|
|
|
int xfs_ag_get_geometry(struct xfs_perag *pag, struct xfs_ag_geometry *ageo);
|
2018-05-13 23:10:08 -07:00
|
|
|
|
2024-11-03 20:18:28 -08:00
|
|
|
static inline xfs_fsblock_t
|
|
|
|
xfs_agbno_to_fsb(
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agblock_t agbno)
|
|
|
|
{
|
2024-11-03 20:18:38 -08:00
|
|
|
return XFS_AGB_TO_FSB(pag_mount(pag), pag_agno(pag), agbno);
|
2024-11-03 20:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline xfs_daddr_t
|
|
|
|
xfs_agbno_to_daddr(
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agblock_t agbno)
|
|
|
|
{
|
2024-11-03 20:18:38 -08:00
|
|
|
return XFS_AGB_TO_DADDR(pag_mount(pag), pag_agno(pag), agbno);
|
2024-11-03 20:18:28 -08:00
|
|
|
}
|
|
|
|
|
2024-11-03 20:18:28 -08:00
|
|
|
static inline xfs_ino_t
|
|
|
|
xfs_agino_to_ino(
|
|
|
|
struct xfs_perag *pag,
|
|
|
|
xfs_agino_t agino)
|
|
|
|
{
|
2024-11-03 20:18:38 -08:00
|
|
|
return XFS_AGINO_TO_INO(pag_mount(pag), pag_agno(pag), agino);
|
2024-11-03 20:18:28 -08:00
|
|
|
}
|
|
|
|
|
2018-05-13 23:10:08 -07:00
|
|
|
#endif /* __LIBXFS_AG_H */
|