xfs_itable.c 17.0 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2 3
 * Copyright (c) 2000-2002,2005 Silicon Graphics, Inc.
 * All Rights Reserved.
L
Linus Torvalds 已提交
4
 *
5 6
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
L
Linus Torvalds 已提交
7 8
 * published by the Free Software Foundation.
 *
9 10 11 12
 * This program is distributed in the hope that it would be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
L
Linus Torvalds 已提交
13
 *
14 15 16
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write the Free Software Foundation,
 * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
L
Linus Torvalds 已提交
17 18
 */
#include "xfs.h"
19
#include "xfs_fs.h"
20
#include "xfs_shared.h"
21
#include "xfs_format.h"
22 23
#include "xfs_log_format.h"
#include "xfs_trans_resv.h"
L
Linus Torvalds 已提交
24 25
#include "xfs_mount.h"
#include "xfs_inode.h"
26
#include "xfs_btree.h"
L
Linus Torvalds 已提交
27
#include "xfs_ialloc.h"
28
#include "xfs_ialloc_btree.h"
L
Linus Torvalds 已提交
29 30
#include "xfs_itable.h"
#include "xfs_error.h"
C
Christoph Hellwig 已提交
31
#include "xfs_trace.h"
D
Dave Chinner 已提交
32
#include "xfs_icache.h"
L
Linus Torvalds 已提交
33

34
STATIC int
35 36 37 38 39
xfs_internal_inum(
	xfs_mount_t	*mp,
	xfs_ino_t	ino)
{
	return (ino == mp->m_sb.sb_rbmino || ino == mp->m_sb.sb_rsumino ||
40
		(xfs_sb_version_hasquota(&mp->m_sb) &&
41
		 xfs_is_quota_inode(&mp->m_sb, ino)));
42 43
}

44 45 46 47 48 49 50 51 52 53 54 55 56
/*
 * Return stat information for one inode.
 * Return 0 if ok, else errno.
 */
int
xfs_bulkstat_one_int(
	struct xfs_mount	*mp,		/* mount point for filesystem */
	xfs_ino_t		ino,		/* inode to get data for */
	void __user		*buffer,	/* buffer to place output in */
	int			ubsize,		/* size of buffer */
	bulkstat_one_fmt_pf	formatter,	/* formatter, copy to user */
	int			*ubused,	/* bytes used by me */
	int			*stat)		/* BULKSTAT_RV_... */
L
Linus Torvalds 已提交
57
{
58 59 60 61 62 63 64 65
	struct xfs_icdinode	*dic;		/* dinode core info pointer */
	struct xfs_inode	*ip;		/* incore inode pointer */
	struct xfs_bstat	*buf;		/* return buffer */
	int			error = 0;	/* error value */

	*stat = BULKSTAT_RV_NOTHING;

	if (!buffer || xfs_internal_inum(mp, ino))
D
Dave Chinner 已提交
66
		return -EINVAL;
67 68 69

	buf = kmem_alloc(sizeof(*buf), KM_SLEEP | KM_MAYFAIL);
	if (!buf)
D
Dave Chinner 已提交
70
		return -ENOMEM;
L
Linus Torvalds 已提交
71

72
	error = xfs_iget(mp, NULL, ino,
73 74
			 (XFS_IGET_DONTCACHE | XFS_IGET_UNTRUSTED),
			 XFS_ILOCK_SHARED, &ip);
75
	if (error)
76
		goto out_free;
L
Linus Torvalds 已提交
77 78

	ASSERT(ip != NULL);
79
	ASSERT(ip->i_imap.im_blkno != 0);
L
Linus Torvalds 已提交
80 81 82 83 84 85 86

	dic = &ip->i_d;

	/* xfs_iget returns the following without needing
	 * further change.
	 */
	buf->bs_nlink = dic->di_nlink;
87 88
	buf->bs_projid_lo = dic->di_projid_lo;
	buf->bs_projid_hi = dic->di_projid_hi;
L
Linus Torvalds 已提交
89 90 91 92 93
	buf->bs_ino = ino;
	buf->bs_mode = dic->di_mode;
	buf->bs_uid = dic->di_uid;
	buf->bs_gid = dic->di_gid;
	buf->bs_size = dic->di_size;
C
Christoph Hellwig 已提交
94 95 96 97 98 99
	buf->bs_atime.tv_sec = dic->di_atime.t_sec;
	buf->bs_atime.tv_nsec = dic->di_atime.t_nsec;
	buf->bs_mtime.tv_sec = dic->di_mtime.t_sec;
	buf->bs_mtime.tv_nsec = dic->di_mtime.t_nsec;
	buf->bs_ctime.tv_sec = dic->di_ctime.t_sec;
	buf->bs_ctime.tv_nsec = dic->di_ctime.t_nsec;
L
Linus Torvalds 已提交
100 101 102 103 104 105 106 107
	buf->bs_xflags = xfs_ip2xflags(ip);
	buf->bs_extsize = dic->di_extsize << mp->m_sb.sb_blocklog;
	buf->bs_extents = dic->di_nextents;
	buf->bs_gen = dic->di_gen;
	memset(buf->bs_pad, 0, sizeof(buf->bs_pad));
	buf->bs_dmevmask = dic->di_dmevmask;
	buf->bs_dmstate = dic->di_dmstate;
	buf->bs_aextents = dic->di_anextents;
108
	buf->bs_forkoff = XFS_IFORK_BOFF(ip);
L
Linus Torvalds 已提交
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128

	switch (dic->di_format) {
	case XFS_DINODE_FMT_DEV:
		buf->bs_rdev = ip->i_df.if_u2.if_rdev;
		buf->bs_blksize = BLKDEV_IOSIZE;
		buf->bs_blocks = 0;
		break;
	case XFS_DINODE_FMT_LOCAL:
	case XFS_DINODE_FMT_UUID:
		buf->bs_rdev = 0;
		buf->bs_blksize = mp->m_sb.sb_blocksize;
		buf->bs_blocks = 0;
		break;
	case XFS_DINODE_FMT_EXTENTS:
	case XFS_DINODE_FMT_BTREE:
		buf->bs_rdev = 0;
		buf->bs_blksize = mp->m_sb.sb_blocksize;
		buf->bs_blocks = dic->di_nblocks + ip->i_delayed_blks;
		break;
	}
C
Christoph Hellwig 已提交
129 130
	xfs_iunlock(ip, XFS_ILOCK_SHARED);
	IRELE(ip);
L
Linus Torvalds 已提交
131

132 133 134
	error = formatter(buffer, ubsize, ubused, buf);
	if (!error)
		*stat = BULKSTAT_RV_DIDONE;
L
Linus Torvalds 已提交
135

136 137 138
 out_free:
	kmem_free(buf);
	return error;
L
Linus Torvalds 已提交
139 140
}

141
/* Return 0 on success or positive error */
142 143 144
STATIC int
xfs_bulkstat_one_fmt(
	void			__user *ubuffer,
145 146
	int			ubsize,
	int			*ubused,
147 148
	const xfs_bstat_t	*buffer)
{
149
	if (ubsize < sizeof(*buffer))
D
Dave Chinner 已提交
150
		return -ENOMEM;
151
	if (copy_to_user(ubuffer, buffer, sizeof(*buffer)))
D
Dave Chinner 已提交
152
		return -EFAULT;
153 154 155
	if (ubused)
		*ubused = sizeof(*buffer);
	return 0;
156 157
}

158 159 160 161 162 163 164 165 166 167
int
xfs_bulkstat_one(
	xfs_mount_t	*mp,		/* mount point for filesystem */
	xfs_ino_t	ino,		/* inode number to get data for */
	void		__user *buffer,	/* buffer to place output in */
	int		ubsize,		/* size of buffer */
	int		*ubused,	/* bytes used by me */
	int		*stat)		/* BULKSTAT_RV_... */
{
	return xfs_bulkstat_one_int(mp, ino, buffer, ubsize,
168
				    xfs_bulkstat_one_fmt, ubused, stat);
169 170
}

J
Jie Liu 已提交
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
/*
 * Loop over all clusters in a chunk for a given incore inode allocation btree
 * record.  Do a readahead if there are any allocated inodes in that cluster.
 */
STATIC void
xfs_bulkstat_ichunk_ra(
	struct xfs_mount		*mp,
	xfs_agnumber_t			agno,
	struct xfs_inobt_rec_incore	*irec)
{
	xfs_agblock_t			agbno;
	struct blk_plug			plug;
	int				blks_per_cluster;
	int				inodes_per_cluster;
	int				i;	/* inode chunk index */

	agbno = XFS_AGINO_TO_AGBNO(mp, irec->ir_startino);
	blks_per_cluster = xfs_icluster_size_fsb(mp);
	inodes_per_cluster = blks_per_cluster << mp->m_sb.sb_inopblog;

	blk_start_plug(&plug);
	for (i = 0; i < XFS_INODES_PER_CHUNK;
	     i += inodes_per_cluster, agbno += blks_per_cluster) {
		if (xfs_inobt_maskn(i, inodes_per_cluster) & ~irec->ir_free) {
			xfs_btree_reada_bufs(mp, agno, agbno, blks_per_cluster,
					     &xfs_inode_buf_ops);
		}
	}
	blk_finish_plug(&plug);
}

202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231
/*
 * Lookup the inode chunk that the given inode lives in and then get the record
 * if we found the chunk.  If the inode was not the last in the chunk and there
 * are some left allocated, update the data for the pointed-to record as well as
 * return the count of grabbed inodes.
 */
STATIC int
xfs_bulkstat_grab_ichunk(
	struct xfs_btree_cur		*cur,	/* btree cursor */
	xfs_agino_t			agino,	/* starting inode of chunk */
	int				*icount,/* return # of inodes grabbed */
	struct xfs_inobt_rec_incore	*irec)	/* btree record */
{
	int				idx;	/* index into inode chunk */
	int				stat;
	int				error = 0;

	/* Lookup the inode chunk that this inode lives in */
	error = xfs_inobt_lookup(cur, agino, XFS_LOOKUP_LE, &stat);
	if (error)
		return error;
	if (!stat) {
		*icount = 0;
		return error;
	}

	/* Get the record, should always work */
	error = xfs_inobt_get_rec(cur, irec, &stat);
	if (error)
		return error;
232
	XFS_WANT_CORRUPTED_RETURN(cur->bc_mp, stat == 1);
233 234

	/* Check if the record contains the inode in request */
235 236 237 238
	if (irec->ir_startino + XFS_INODES_PER_CHUNK <= agino) {
		*icount = 0;
		return 0;
	}
239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260

	idx = agino - irec->ir_startino + 1;
	if (idx < XFS_INODES_PER_CHUNK &&
	    (xfs_inobt_maskn(idx, XFS_INODES_PER_CHUNK - idx) & ~irec->ir_free)) {
		int	i;

		/* We got a right chunk with some left inodes allocated at it.
		 * Grab the chunk record.  Mark all the uninteresting inodes
		 * free -- because they're before our start point.
		 */
		for (i = 0; i < idx; i++) {
			if (XFS_INOBT_MASK(i) & ~irec->ir_free)
				irec->ir_freecount++;
		}

		irec->ir_free |= xfs_inobt_maskn(0, idx);
		*icount = XFS_INODES_PER_CHUNK - irec->ir_freecount;
	}

	return 0;
}

261 262
#define XFS_BULKSTAT_UBLEFT(ubleft)	((ubleft) >= statstruct_size)

263 264 265 266 267 268
struct xfs_bulkstat_agichunk {
	char		__user **ac_ubuffer;/* pointer into user's buffer */
	int		ac_ubleft;	/* bytes left in user's buffer */
	int		ac_ubelem;	/* spaces used in user's buffer */
};

J
Jie Liu 已提交
269 270 271 272
/*
 * Process inodes in chunk with a pointer to a formatter function
 * that will iget the inode and fill in the appropriate structure.
 */
273
static int
J
Jie Liu 已提交
274 275 276 277 278 279
xfs_bulkstat_ag_ichunk(
	struct xfs_mount		*mp,
	xfs_agnumber_t			agno,
	struct xfs_inobt_rec_incore	*irbp,
	bulkstat_one_pf			formatter,
	size_t				statstruct_size,
280
	struct xfs_bulkstat_agichunk	*acp,
281
	xfs_agino_t			*last_agino)
J
Jie Liu 已提交
282 283
{
	char				__user **ubufp = acp->ac_ubuffer;
284
	int				chunkidx;
J
Jie Liu 已提交
285
	int				error = 0;
286
	xfs_agino_t			agino = irbp->ir_startino;
J
Jie Liu 已提交
287

288 289 290
	for (chunkidx = 0; chunkidx < XFS_INODES_PER_CHUNK;
	     chunkidx++, agino++) {
		int		fmterror;
J
Jie Liu 已提交
291
		int		ubused;
292 293 294 295

		/* inode won't fit in buffer, we are done */
		if (acp->ac_ubleft < statstruct_size)
			break;
J
Jie Liu 已提交
296 297

		/* Skip if this inode is free */
298
		if (XFS_INOBT_MASK(chunkidx) & irbp->ir_free)
J
Jie Liu 已提交
299 300 301 302
			continue;

		/* Get the inode and fill in a single buffer */
		ubused = statstruct_size;
303 304 305
		error = formatter(mp, XFS_AGINO_TO_INO(mp, agno, agino),
				  *ubufp, acp->ac_ubleft, &ubused, &fmterror);

306 307 308
		if (fmterror == BULKSTAT_RV_GIVEUP ||
		    (error && error != -ENOENT && error != -EINVAL)) {
			acp->ac_ubleft = 0;
J
Jie Liu 已提交
309 310 311
			ASSERT(error);
			break;
		}
312 313 314 315 316 317 318 319 320 321 322

		/* be careful not to leak error if at end of chunk */
		if (fmterror == BULKSTAT_RV_NOTHING || error) {
			error = 0;
			continue;
		}

		*ubufp += ubused;
		acp->ac_ubleft -= ubused;
		acp->ac_ubelem++;
	}
J
Jie Liu 已提交
323

324 325 326 327 328 329 330 331 332 333
	/*
	 * Post-update *last_agino. At this point, agino will always point one
	 * inode past the last inode we processed successfully. Hence we
	 * substract that inode when setting the *last_agino cursor so that we
	 * return the correct cookie to userspace. On the next bulkstat call,
	 * the inode under the lastino cookie will be skipped as we have already
	 * processed it here.
	 */
	*last_agino = agino - 1;

J
Jie Liu 已提交
334 335 336
	return error;
}

L
Linus Torvalds 已提交
337 338 339 340 341 342 343 344 345 346 347
/*
 * Return stat information in bulk (by-inode) for the filesystem.
 */
int					/* error status */
xfs_bulkstat(
	xfs_mount_t		*mp,	/* mount point for filesystem */
	xfs_ino_t		*lastinop, /* last inode returned */
	int			*ubcountp, /* size of buffer/count returned */
	bulkstat_one_pf		formatter, /* func that'd fill a single buf */
	size_t			statstruct_size, /* sizeof struct filling */
	char			__user *ubuffer, /* buffer with inode stats */
348
	int			*done)	/* 1 if there are more stats to get */
L
Linus Torvalds 已提交
349 350 351 352 353
{
	xfs_buf_t		*agbp;	/* agi header buffer */
	xfs_agino_t		agino;	/* inode # in allocation group */
	xfs_agnumber_t		agno;	/* allocation group number */
	xfs_btree_cur_t		*cur;	/* btree cursor for ialloc btree */
354
	size_t			irbsize; /* size of irec buffer in bytes */
355
	xfs_inobt_rec_incore_t	*irbuf;	/* start of irec buffer */
L
Linus Torvalds 已提交
356 357
	int			nirbuf;	/* size of irbuf */
	int			ubcount; /* size of user's buffer */
358
	struct xfs_bulkstat_agichunk ac;
359
	int			error = 0;
L
Linus Torvalds 已提交
360 361 362 363

	/*
	 * Get the last inode value, see if there's nothing to do.
	 */
364 365
	agno = XFS_INO_TO_AGNO(mp, *lastinop);
	agino = XFS_INO_TO_AGINO(mp, *lastinop);
L
Linus Torvalds 已提交
366
	if (agno >= mp->m_sb.sb_agcount ||
367
	    *lastinop != XFS_AGINO_TO_INO(mp, agno, agino)) {
L
Linus Torvalds 已提交
368 369 370 371
		*done = 1;
		*ubcountp = 0;
		return 0;
	}
372

L
Linus Torvalds 已提交
373
	ubcount = *ubcountp; /* statstruct's */
374 375 376 377 378
	ac.ac_ubuffer = &ubuffer;
	ac.ac_ubleft = ubcount * statstruct_size; /* bytes */;
	ac.ac_ubelem = 0;

	*ubcountp = 0;
L
Linus Torvalds 已提交
379
	*done = 0;
380

381 382
	irbuf = kmem_zalloc_greedy(&irbsize, PAGE_SIZE, PAGE_SIZE * 4);
	if (!irbuf)
D
Dave Chinner 已提交
383
		return -ENOMEM;
384

385 386
	nirbuf = irbsize / sizeof(*irbuf);

L
Linus Torvalds 已提交
387 388 389 390
	/*
	 * Loop over the allocation groups, starting from the last
	 * inode returned; 0 means start of the allocation group.
	 */
391 392 393 394 395 396 397
	while (agno < mp->m_sb.sb_agcount) {
		struct xfs_inobt_rec_incore	*irbp = irbuf;
		struct xfs_inobt_rec_incore	*irbufend = irbuf + nirbuf;
		bool				end_of_ag = false;
		int				icount = 0;
		int				stat;

L
Linus Torvalds 已提交
398
		error = xfs_ialloc_read_agi(mp, NULL, agno, &agbp);
399 400
		if (error)
			break;
L
Linus Torvalds 已提交
401 402 403
		/*
		 * Allocate and initialize a btree cursor for ialloc btree.
		 */
404 405
		cur = xfs_inobt_init_cursor(mp, NULL, agbp, agno,
					    XFS_BTNUM_INO);
L
Linus Torvalds 已提交
406 407
		if (agino > 0) {
			/*
408 409
			 * In the middle of an allocation group, we need to get
			 * the remainder of the chunk we're in.
L
Linus Torvalds 已提交
410
			 */
411 412 413 414
			struct xfs_inobt_rec_incore	r;

			error = xfs_bulkstat_grab_ichunk(cur, agino, &icount, &r);
			if (error)
415
				goto del_cursor;
416
			if (icount) {
417 418 419
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
420 421
				irbp++;
			}
422
			/* Increment to the next record */
423
			error = xfs_btree_increment(cur, 0, &stat);
L
Linus Torvalds 已提交
424
		} else {
425
			/* Start of ag.  Lookup the first inode chunk */
426
			error = xfs_inobt_lookup(cur, 0, XFS_LOOKUP_GE, &stat);
L
Linus Torvalds 已提交
427
		}
428
		if (error || stat == 0) {
429
			end_of_ag = true;
430
			goto del_cursor;
431
		}
432

L
Linus Torvalds 已提交
433 434 435 436 437
		/*
		 * Loop through inode btree records in this ag,
		 * until we run out of inodes or space in the buffer.
		 */
		while (irbp < irbufend && icount < ubcount) {
438
			struct xfs_inobt_rec_incore	r;
439

440 441
			error = xfs_inobt_get_rec(cur, &r, &stat);
			if (error || stat == 0) {
442
				end_of_ag = true;
443
				goto del_cursor;
L
Linus Torvalds 已提交
444
			}
445

L
Linus Torvalds 已提交
446 447
			/*
			 * If this chunk has any allocated inodes, save it.
448
			 * Also start read-ahead now for this chunk.
L
Linus Torvalds 已提交
449
			 */
450
			if (r.ir_freecount < XFS_INODES_PER_CHUNK) {
J
Jie Liu 已提交
451
				xfs_bulkstat_ichunk_ra(mp, agno, &r);
452 453 454
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
455
				irbp++;
456
				icount += XFS_INODES_PER_CHUNK - r.ir_freecount;
L
Linus Torvalds 已提交
457
			}
458 459
			error = xfs_btree_increment(cur, 0, &stat);
			if (error || stat == 0) {
460
				end_of_ag = true;
461 462
				goto del_cursor;
			}
463
			cond_resched();
L
Linus Torvalds 已提交
464
		}
465

L
Linus Torvalds 已提交
466
		/*
467 468 469
		 * Drop the btree buffers and the agi buffer as we can't hold any
		 * of the locks these represent when calling iget. If there is a
		 * pending error, then we are done.
L
Linus Torvalds 已提交
470
		 */
471
del_cursor:
L
Linus Torvalds 已提交
472 473
		xfs_btree_del_cursor(cur, XFS_BTREE_NOERROR);
		xfs_buf_relse(agbp);
474 475
		if (error)
			break;
L
Linus Torvalds 已提交
476
		/*
477 478 479
		 * Now format all the good inodes into the user's buffer. The
		 * call to xfs_bulkstat_ag_ichunk() sets up the agino pointer
		 * for the next loop iteration.
L
Linus Torvalds 已提交
480 481 482
		 */
		irbufend = irbp;
		for (irbp = irbuf;
483
		     irbp < irbufend && ac.ac_ubleft >= statstruct_size;
484
		     irbp++) {
J
Jie Liu 已提交
485
			error = xfs_bulkstat_ag_ichunk(mp, agno, irbp,
486
					formatter, statstruct_size, &ac,
487
					&agino);
J
Jie Liu 已提交
488
			if (error)
489
				break;
J
Jie Liu 已提交
490

491
			cond_resched();
L
Linus Torvalds 已提交
492
		}
493

494 495 496 497 498
		/*
		 * If we've run out of space or had a formatting error, we
		 * are now done
		 */
		if (ac.ac_ubleft < statstruct_size || error)
L
Linus Torvalds 已提交
499
			break;
500 501 502 503

		if (end_of_ag) {
			agno++;
			agino = 0;
504
		}
L
Linus Torvalds 已提交
505 506 507 508
	}
	/*
	 * Done, we're either out of filesystem or space to put the data.
	 */
509
	kmem_free(irbuf);
510
	*ubcountp = ac.ac_ubelem;
511

512
	/*
513 514 515 516 517
	 * We found some inodes, so clear the error status and return them.
	 * The lastino pointer will point directly at the inode that triggered
	 * any error that occurred, so on the next call the error will be
	 * triggered again and propagated to userspace as there will be no
	 * formatted inodes in the buffer.
518
	 */
519
	if (ac.ac_ubelem)
520 521
		error = 0;

522 523 524 525 526 527
	/*
	 * If we ran out of filesystem, lastino will point off the end of
	 * the filesystem so the next call will return immediately.
	 */
	*lastinop = XFS_AGINO_TO_INO(mp, agno, agino);
	if (agno >= mp->m_sb.sb_agcount)
L
Linus Torvalds 已提交
528 529
		*done = 1;

530
	return error;
L
Linus Torvalds 已提交
531 532
}

533 534 535
int
xfs_inumbers_fmt(
	void			__user *ubuffer, /* buffer to write to */
J
Jie Liu 已提交
536
	const struct xfs_inogrp	*buffer,	/* buffer to read from */
537 538 539 540 541 542 543 544 545
	long			count,		/* # of elements to read */
	long			*written)	/* # of bytes written */
{
	if (copy_to_user(ubuffer, buffer, count * sizeof(*buffer)))
		return -EFAULT;
	*written = count * sizeof(*buffer);
	return 0;
}

L
Linus Torvalds 已提交
546 547 548 549 550
/*
 * Return inode number table for the filesystem.
 */
int					/* error status */
xfs_inumbers(
J
Jie Liu 已提交
551 552 553 554 555
	struct xfs_mount	*mp,/* mount point for filesystem */
	xfs_ino_t		*lastino,/* last inode returned */
	int			*count,/* size of buffer/count returned */
	void			__user *ubuffer,/* buffer with inode descriptions */
	inumbers_fmt_pf		formatter)
L
Linus Torvalds 已提交
556
{
J
Jie Liu 已提交
557 558 559
	xfs_agnumber_t		agno = XFS_INO_TO_AGNO(mp, *lastino);
	xfs_agino_t		agino = XFS_INO_TO_AGINO(mp, *lastino);
	struct xfs_btree_cur	*cur = NULL;
560
	struct xfs_buf		*agbp = NULL;
J
Jie Liu 已提交
561 562 563 564 565 566
	struct xfs_inogrp	*buffer;
	int			bcount;
	int			left = *count;
	int			bufidx = 0;
	int			error = 0;

L
Linus Torvalds 已提交
567
	*count = 0;
J
Jie Liu 已提交
568 569 570 571
	if (agno >= mp->m_sb.sb_agcount ||
	    *lastino != XFS_AGINO_TO_INO(mp, agno, agino))
		return error;

572
	bcount = MIN(left, (int)(PAGE_SIZE / sizeof(*buffer)));
L
Linus Torvalds 已提交
573
	buffer = kmem_alloc(bcount * sizeof(*buffer), KM_SLEEP);
574
	do {
J
Jie Liu 已提交
575 576 577
		struct xfs_inobt_rec_incore	r;
		int				stat;

578
		if (!agbp) {
L
Linus Torvalds 已提交
579
			error = xfs_ialloc_read_agi(mp, NULL, agno, &agbp);
580 581 582
			if (error)
				break;

583 584
			cur = xfs_inobt_init_cursor(mp, NULL, agbp, agno,
						    XFS_BTNUM_INO);
585
			error = xfs_inobt_lookup(cur, agino, XFS_LOOKUP_GE,
J
Jie Liu 已提交
586
						 &stat);
587 588 589 590
			if (error)
				break;
			if (!stat)
				goto next_ag;
L
Linus Torvalds 已提交
591
		}
592

J
Jie Liu 已提交
593
		error = xfs_inobt_get_rec(cur, &r, &stat);
594 595 596 597 598
		if (error)
			break;
		if (!stat)
			goto next_ag;

599 600 601 602 603 604
		agino = r.ir_startino + XFS_INODES_PER_CHUNK - 1;
		buffer[bufidx].xi_startino =
			XFS_AGINO_TO_INO(mp, agno, r.ir_startino);
		buffer[bufidx].xi_alloccount =
			XFS_INODES_PER_CHUNK - r.ir_freecount;
		buffer[bufidx].xi_allocmask = ~r.ir_free;
605 606 607
		if (++bufidx == bcount) {
			long	written;

J
Jie Liu 已提交
608 609
			error = formatter(ubuffer, buffer, bufidx, &written);
			if (error)
L
Linus Torvalds 已提交
610
				break;
611
			ubuffer += written;
L
Linus Torvalds 已提交
612 613 614
			*count += bufidx;
			bufidx = 0;
		}
615 616 617 618 619 620 621 622 623 624 625 626 627 628 629
		if (!--left)
			break;

		error = xfs_btree_increment(cur, 0, &stat);
		if (error)
			break;
		if (stat)
			continue;

next_ag:
		xfs_btree_del_cursor(cur, XFS_BTREE_ERROR);
		cur = NULL;
		xfs_buf_relse(agbp);
		agbp = NULL;
		agino = 0;
630 631
		agno++;
	} while (agno < mp->m_sb.sb_agcount);
632

L
Linus Torvalds 已提交
633 634
	if (!error) {
		if (bufidx) {
635 636
			long	written;

J
Jie Liu 已提交
637 638
			error = formatter(ubuffer, buffer, bufidx, &written);
			if (!error)
L
Linus Torvalds 已提交
639 640 641 642
				*count += bufidx;
		}
		*lastino = XFS_AGINO_TO_INO(mp, agno, agino);
	}
643

644
	kmem_free(buffer);
L
Linus Torvalds 已提交
645 646 647 648 649
	if (cur)
		xfs_btree_del_cursor(cur, (error ? XFS_BTREE_ERROR :
					   XFS_BTREE_NOERROR));
	if (agbp)
		xfs_buf_relse(agbp);
650

L
Linus Torvalds 已提交
651 652
	return error;
}