|
@@ -516,8 +516,11 @@ xfs_inode_ag_walk(
|
|
|
struct xfs_mount *mp,
|
|
|
struct xfs_perag *pag,
|
|
|
int (*execute)(struct xfs_inode *ip,
|
|
|
- struct xfs_perag *pag, int flags),
|
|
|
- int flags)
|
|
|
+ struct xfs_perag *pag, int flags,
|
|
|
+ void *args),
|
|
|
+ int flags,
|
|
|
+ void *args,
|
|
|
+ int tag)
|
|
|
{
|
|
|
uint32_t first_index;
|
|
|
int last_error = 0;
|
|
@@ -536,9 +539,17 @@ restart:
|
|
|
int i;
|
|
|
|
|
|
rcu_read_lock();
|
|
|
- nr_found = radix_tree_gang_lookup(&pag->pag_ici_root,
|
|
|
+
|
|
|
+ if (tag == -1)
|
|
|
+ nr_found = radix_tree_gang_lookup(&pag->pag_ici_root,
|
|
|
(void **)batch, first_index,
|
|
|
XFS_LOOKUP_BATCH);
|
|
|
+ else
|
|
|
+ nr_found = radix_tree_gang_lookup_tag(
|
|
|
+ &pag->pag_ici_root,
|
|
|
+ (void **) batch, first_index,
|
|
|
+ XFS_LOOKUP_BATCH, tag);
|
|
|
+
|
|
|
if (!nr_found) {
|
|
|
rcu_read_unlock();
|
|
|
break;
|
|
@@ -579,7 +590,7 @@ restart:
|
|
|
for (i = 0; i < nr_found; i++) {
|
|
|
if (!batch[i])
|
|
|
continue;
|
|
|
- error = execute(batch[i], pag, flags);
|
|
|
+ error = execute(batch[i], pag, flags, args);
|
|
|
IRELE(batch[i]);
|
|
|
if (error == EAGAIN) {
|
|
|
skipped++;
|
|
@@ -608,8 +619,10 @@ int
|
|
|
xfs_inode_ag_iterator(
|
|
|
struct xfs_mount *mp,
|
|
|
int (*execute)(struct xfs_inode *ip,
|
|
|
- struct xfs_perag *pag, int flags),
|
|
|
- int flags)
|
|
|
+ struct xfs_perag *pag, int flags,
|
|
|
+ void *args),
|
|
|
+ int flags,
|
|
|
+ void *args)
|
|
|
{
|
|
|
struct xfs_perag *pag;
|
|
|
int error = 0;
|
|
@@ -619,7 +632,36 @@ xfs_inode_ag_iterator(
|
|
|
ag = 0;
|
|
|
while ((pag = xfs_perag_get(mp, ag))) {
|
|
|
ag = pag->pag_agno + 1;
|
|
|
- error = xfs_inode_ag_walk(mp, pag, execute, flags);
|
|
|
+ error = xfs_inode_ag_walk(mp, pag, execute, flags, args, -1);
|
|
|
+ xfs_perag_put(pag);
|
|
|
+ if (error) {
|
|
|
+ last_error = error;
|
|
|
+ if (error == EFSCORRUPTED)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return XFS_ERROR(last_error);
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+xfs_inode_ag_iterator_tag(
|
|
|
+ struct xfs_mount *mp,
|
|
|
+ int (*execute)(struct xfs_inode *ip,
|
|
|
+ struct xfs_perag *pag, int flags,
|
|
|
+ void *args),
|
|
|
+ int flags,
|
|
|
+ void *args,
|
|
|
+ int tag)
|
|
|
+{
|
|
|
+ struct xfs_perag *pag;
|
|
|
+ int error = 0;
|
|
|
+ int last_error = 0;
|
|
|
+ xfs_agnumber_t ag;
|
|
|
+
|
|
|
+ ag = 0;
|
|
|
+ while ((pag = xfs_perag_get_tag(mp, ag, tag))) {
|
|
|
+ ag = pag->pag_agno + 1;
|
|
|
+ error = xfs_inode_ag_walk(mp, pag, execute, flags, args, tag);
|
|
|
xfs_perag_put(pag);
|
|
|
if (error) {
|
|
|
last_error = error;
|