xfs_itable.c 17.1 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"
24
#include "xfs_inum.h"
L
Linus Torvalds 已提交
25
#include "xfs_sb.h"
26
#include "xfs_ag.h"
L
Linus Torvalds 已提交
27 28
#include "xfs_mount.h"
#include "xfs_inode.h"
29
#include "xfs_btree.h"
L
Linus Torvalds 已提交
30
#include "xfs_ialloc.h"
31
#include "xfs_ialloc_btree.h"
L
Linus Torvalds 已提交
32 33
#include "xfs_itable.h"
#include "xfs_error.h"
C
Christoph Hellwig 已提交
34
#include "xfs_trace.h"
D
Dave Chinner 已提交
35
#include "xfs_icache.h"
36
#include "xfs_dinode.h"
L
Linus Torvalds 已提交
37

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

48 49 50 51 52 53 54 55 56 57 58 59 60
/*
 * 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 已提交
61
{
62 63 64 65 66 67 68 69
	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 已提交
70
		return -EINVAL;
71 72 73

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

76
	error = xfs_iget(mp, NULL, ino,
77 78
			 (XFS_IGET_DONTCACHE | XFS_IGET_UNTRUSTED),
			 XFS_ILOCK_SHARED, &ip);
79
	if (error)
80
		goto out_free;
L
Linus Torvalds 已提交
81 82

	ASSERT(ip != NULL);
83
	ASSERT(ip->i_imap.im_blkno != 0);
L
Linus Torvalds 已提交
84 85 86 87 88 89 90

	dic = &ip->i_d;

	/* xfs_iget returns the following without needing
	 * further change.
	 */
	buf->bs_nlink = dic->di_nlink;
91 92
	buf->bs_projid_lo = dic->di_projid_lo;
	buf->bs_projid_hi = dic->di_projid_hi;
L
Linus Torvalds 已提交
93 94 95 96 97
	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 已提交
98 99 100 101 102 103
	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 已提交
104 105 106 107 108 109 110 111
	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;
112
	buf->bs_forkoff = XFS_IFORK_BOFF(ip);
L
Linus Torvalds 已提交
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132

	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 已提交
133 134
	xfs_iunlock(ip, XFS_ILOCK_SHARED);
	IRELE(ip);
L
Linus Torvalds 已提交
135

136 137 138
	error = formatter(buffer, ubsize, ubused, buf);
	if (!error)
		*stat = BULKSTAT_RV_DIDONE;
L
Linus Torvalds 已提交
139

140 141 142
 out_free:
	kmem_free(buf);
	return error;
L
Linus Torvalds 已提交
143 144
}

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

162 163 164 165 166 167 168 169 170 171
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,
172
				    xfs_bulkstat_one_fmt, ubused, stat);
173 174
}

J
Jie Liu 已提交
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 202 203 204 205
/*
 * 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);
}

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 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
/*
 * 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;
	XFS_WANT_CORRUPTED_RETURN(stat == 1);

	/* Check if the record contains the inode in request */
	if (irec->ir_startino + XFS_INODES_PER_CHUNK <= agino)
		return -EINVAL;

	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;
}

263 264
#define XFS_BULKSTAT_UBLEFT(ubleft)	((ubleft) >= statstruct_size)

265 266 267 268 269 270
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 已提交
271 272 273 274
/*
 * Process inodes in chunk with a pointer to a formatter function
 * that will iget the inode and fill in the appropriate structure.
 */
275
static int
J
Jie Liu 已提交
276 277 278 279 280 281
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,
282 283
	struct xfs_bulkstat_agichunk	*acp,
	xfs_ino_t			*lastino)
J
Jie Liu 已提交
284 285
{
	char				__user **ubufp = acp->ac_ubuffer;
286
	int				chunkidx;
J
Jie Liu 已提交
287 288 289
	int				error = 0;
	xfs_agino_t			agino;

290 291 292 293
	agino = irbp->ir_startino;
	for (chunkidx = 0; chunkidx < XFS_INODES_PER_CHUNK;
	     chunkidx++, agino++) {
		int		fmterror;
J
Jie Liu 已提交
294 295 296 297 298
		int		ubused;
		xfs_ino_t	ino = XFS_AGINO_TO_INO(mp, agno, agino);

		/* Skip if this inode is free */
		if (XFS_INOBT_MASK(chunkidx) & irbp->ir_free) {
299
			*lastino = ino;
J
Jie Liu 已提交
300 301 302 303 304
			continue;
		}

		/* Get the inode and fill in a single buffer */
		ubused = statstruct_size;
305 306 307 308 309
		error = formatter(mp, ino, *ubufp, acp->ac_ubleft,
				  &ubused, &fmterror);
		if (fmterror == BULKSTAT_RV_GIVEUP ||
		    (error && error != -ENOENT && error != -EINVAL)) {
			acp->ac_ubleft = 0;
J
Jie Liu 已提交
310 311 312
			ASSERT(error);
			break;
		}
313 314 315 316 317 318 319 320 321 322 323

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

		*ubufp += ubused;
		acp->ac_ubleft -= ubused;
		acp->ac_ubelem++;
324
		*lastino = ino;
J
Jie Liu 已提交
325

326 327 328
		if (acp->ac_ubleft < statstruct_size)
			break;
	}
J
Jie Liu 已提交
329 330 331 332

	return error;
}

L
Linus Torvalds 已提交
333 334 335 336 337 338 339 340 341 342 343
/*
 * 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 */
344
	int			*done)	/* 1 if there are more stats to get */
L
Linus Torvalds 已提交
345 346 347 348 349 350 351 352 353
{
	xfs_buf_t		*agbp;	/* agi header buffer */
	xfs_agi_t		*agi;	/* agi header data */
	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 */
	int			end_of_ag; /* set if we've seen the ag end */
	int			error;	/* error code */
	int			icount;	/* count of inodes good in irbuf */
354
	size_t			irbsize; /* size of irec buffer in bytes */
L
Linus Torvalds 已提交
355
	xfs_ino_t		ino;	/* inode number (filesystem) */
356 357 358
	xfs_inobt_rec_incore_t	*irbp;	/* current irec buffer pointer */
	xfs_inobt_rec_incore_t	*irbuf;	/* start of irec buffer */
	xfs_inobt_rec_incore_t	*irbufend; /* end of good irec buffer entries */
359
	xfs_ino_t		lastino; /* last inode number returned */
L
Linus Torvalds 已提交
360 361 362
	int			nirbuf;	/* size of irbuf */
	int			rval;	/* return value error code */
	int			ubcount; /* size of user's buffer */
363
	int			stat;
364
	struct xfs_bulkstat_agichunk ac;
L
Linus Torvalds 已提交
365 366 367 368 369

	/*
	 * Get the last inode value, see if there's nothing to do.
	 */
	ino = (xfs_ino_t)*lastinop;
370
	lastino = ino;
L
Linus Torvalds 已提交
371 372 373 374 375 376 377 378
	agno = XFS_INO_TO_AGNO(mp, ino);
	agino = XFS_INO_TO_AGINO(mp, ino);
	if (agno >= mp->m_sb.sb_agcount ||
	    ino != XFS_AGINO_TO_INO(mp, agno, agino)) {
		*done = 1;
		*ubcountp = 0;
		return 0;
	}
379

L
Linus Torvalds 已提交
380
	ubcount = *ubcountp; /* statstruct's */
381 382 383 384 385
	ac.ac_ubuffer = &ubuffer;
	ac.ac_ubleft = ubcount * statstruct_size; /* bytes */;
	ac.ac_ubelem = 0;

	*ubcountp = 0;
L
Linus Torvalds 已提交
386
	*done = 0;
387

388 389
	irbuf = kmem_zalloc_greedy(&irbsize, PAGE_SIZE, PAGE_SIZE * 4);
	if (!irbuf)
D
Dave Chinner 已提交
390
		return -ENOMEM;
391

392 393
	nirbuf = irbsize / sizeof(*irbuf);

L
Linus Torvalds 已提交
394 395 396 397 398
	/*
	 * Loop over the allocation groups, starting from the last
	 * inode returned; 0 means start of the allocation group.
	 */
	rval = 0;
399
	while (XFS_BULKSTAT_UBLEFT(ac.ac_ubleft) && agno < mp->m_sb.sb_agcount) {
400
		cond_resched();
L
Linus Torvalds 已提交
401
		error = xfs_ialloc_read_agi(mp, NULL, agno, &agbp);
402 403
		if (error)
			break;
L
Linus Torvalds 已提交
404 405 406 407
		agi = XFS_BUF_TO_AGI(agbp);
		/*
		 * Allocate and initialize a btree cursor for ialloc btree.
		 */
408 409
		cur = xfs_inobt_init_cursor(mp, NULL, agbp, agno,
					    XFS_BTNUM_INO);
L
Linus Torvalds 已提交
410 411 412
		irbp = irbuf;
		irbufend = irbuf + nirbuf;
		end_of_ag = 0;
413
		icount = 0;
L
Linus Torvalds 已提交
414 415
		if (agino > 0) {
			/*
416 417
			 * In the middle of an allocation group, we need to get
			 * the remainder of the chunk we're in.
L
Linus Torvalds 已提交
418
			 */
419 420 421 422
			struct xfs_inobt_rec_incore	r;

			error = xfs_bulkstat_grab_ichunk(cur, agino, &icount, &r);
			if (error)
423
				goto del_cursor;
424
			if (icount) {
425 426 427
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
428
				irbp++;
429
				agino = r.ir_startino + XFS_INODES_PER_CHUNK;
L
Linus Torvalds 已提交
430
			}
431
			/* Increment to the next record */
432
			error = xfs_btree_increment(cur, 0, &stat);
L
Linus Torvalds 已提交
433
		} else {
434
			/* Start of ag.  Lookup the first inode chunk */
435
			error = xfs_inobt_lookup(cur, 0, XFS_LOOKUP_GE, &stat);
L
Linus Torvalds 已提交
436
		}
437 438
		if (error || stat == 0) {
			end_of_ag = 1;
439
			goto del_cursor;
440
		}
441

L
Linus Torvalds 已提交
442 443 444 445 446
		/*
		 * Loop through inode btree records in this ag,
		 * until we run out of inodes or space in the buffer.
		 */
		while (irbp < irbufend && icount < ubcount) {
447
			struct xfs_inobt_rec_incore	r;
448

449 450
			error = xfs_inobt_get_rec(cur, &r, &stat);
			if (error || stat == 0) {
L
Linus Torvalds 已提交
451
				end_of_ag = 1;
452
				goto del_cursor;
L
Linus Torvalds 已提交
453
			}
454

L
Linus Torvalds 已提交
455 456
			/*
			 * If this chunk has any allocated inodes, save it.
457
			 * Also start read-ahead now for this chunk.
L
Linus Torvalds 已提交
458
			 */
459
			if (r.ir_freecount < XFS_INODES_PER_CHUNK) {
J
Jie Liu 已提交
460
				xfs_bulkstat_ichunk_ra(mp, agno, &r);
461 462 463
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
464
				irbp++;
465
				icount += XFS_INODES_PER_CHUNK - r.ir_freecount;
L
Linus Torvalds 已提交
466 467 468 469
			}
			/*
			 * Set agino to after this chunk and bump the cursor.
			 */
470
			agino = r.ir_startino + XFS_INODES_PER_CHUNK;
471 472
			error = xfs_btree_increment(cur, 0, &stat);
			if (error || stat == 0) {
473 474 475
				end_of_ag = 1;
				goto del_cursor;
			}
476
			cond_resched();
L
Linus Torvalds 已提交
477
		}
478

L
Linus Torvalds 已提交
479
		/*
480 481 482
		 * 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 已提交
483
		 */
484
del_cursor:
L
Linus Torvalds 已提交
485 486
		xfs_btree_del_cursor(cur, XFS_BTREE_NOERROR);
		xfs_buf_relse(agbp);
487 488
		if (error)
			break;
L
Linus Torvalds 已提交
489 490 491 492 493
		/*
		 * Now format all the good inodes into the user's buffer.
		 */
		irbufend = irbp;
		for (irbp = irbuf;
494 495
		     irbp < irbufend && XFS_BULKSTAT_UBLEFT(ac.ac_ubleft);
		     irbp++) {
J
Jie Liu 已提交
496
			error = xfs_bulkstat_ag_ichunk(mp, agno, irbp,
497 498
					formatter, statstruct_size, &ac,
					&lastino);
J
Jie Liu 已提交
499 500 501
			if (error)
				rval = error;

502
			cond_resched();
L
Linus Torvalds 已提交
503
		}
504

L
Linus Torvalds 已提交
505 506 507
		/*
		 * Set up for the next loop iteration.
		 */
508
		if (XFS_BULKSTAT_UBLEFT(ac.ac_ubleft)) {
L
Linus Torvalds 已提交
509 510 511
			if (end_of_ag) {
				agno++;
				agino = 0;
512 513
			} else
				agino = XFS_INO_TO_AGINO(mp, lastino);
L
Linus Torvalds 已提交
514 515 516 517 518 519
		} else
			break;
	}
	/*
	 * Done, we're either out of filesystem or space to put the data.
	 */
520
	kmem_free(irbuf);
521
	*ubcountp = ac.ac_ubelem;
522 523 524
	/*
	 * Found some inodes, return them now and return the error next time.
	 */
525
	if (ac.ac_ubelem)
526
		rval = 0;
L
Linus Torvalds 已提交
527 528 529 530 531 532 533 534 535 536 537 538 539 540
	if (agno >= mp->m_sb.sb_agcount) {
		/*
		 * If we ran out of filesystem, mark lastino as off
		 * the end of the filesystem, so the next call
		 * will return immediately.
		 */
		*lastinop = (xfs_ino_t)XFS_AGINO_TO_INO(mp, agno, 0);
		*done = 1;
	} else
		*lastinop = (xfs_ino_t)lastino;

	return rval;
}

541 542 543
int
xfs_inumbers_fmt(
	void			__user *ubuffer, /* buffer to write to */
J
Jie Liu 已提交
544
	const struct xfs_inogrp	*buffer,	/* buffer to read from */
545 546 547 548 549 550 551 552 553
	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 已提交
554 555 556 557 558
/*
 * Return inode number table for the filesystem.
 */
int					/* error status */
xfs_inumbers(
J
Jie Liu 已提交
559 560 561 562 563
	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 已提交
564
{
J
Jie Liu 已提交
565 566 567
	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;
568
	struct xfs_buf		*agbp = NULL;
J
Jie Liu 已提交
569 570 571 572 573 574
	struct xfs_inogrp	*buffer;
	int			bcount;
	int			left = *count;
	int			bufidx = 0;
	int			error = 0;

L
Linus Torvalds 已提交
575
	*count = 0;
J
Jie Liu 已提交
576 577 578 579
	if (agno >= mp->m_sb.sb_agcount ||
	    *lastino != XFS_AGINO_TO_INO(mp, agno, agino))
		return error;

580
	bcount = MIN(left, (int)(PAGE_SIZE / sizeof(*buffer)));
L
Linus Torvalds 已提交
581
	buffer = kmem_alloc(bcount * sizeof(*buffer), KM_SLEEP);
582
	do {
J
Jie Liu 已提交
583 584 585
		struct xfs_inobt_rec_incore	r;
		int				stat;

586
		if (!agbp) {
L
Linus Torvalds 已提交
587
			error = xfs_ialloc_read_agi(mp, NULL, agno, &agbp);
588 589 590
			if (error)
				break;

591 592
			cur = xfs_inobt_init_cursor(mp, NULL, agbp, agno,
						    XFS_BTNUM_INO);
593
			error = xfs_inobt_lookup(cur, agino, XFS_LOOKUP_GE,
J
Jie Liu 已提交
594
						 &stat);
595 596 597 598
			if (error)
				break;
			if (!stat)
				goto next_ag;
L
Linus Torvalds 已提交
599
		}
600

J
Jie Liu 已提交
601
		error = xfs_inobt_get_rec(cur, &r, &stat);
602 603 604 605 606
		if (error)
			break;
		if (!stat)
			goto next_ag;

607 608 609 610 611 612
		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;
613 614 615
		if (++bufidx == bcount) {
			long	written;

J
Jie Liu 已提交
616 617
			error = formatter(ubuffer, buffer, bufidx, &written);
			if (error)
L
Linus Torvalds 已提交
618
				break;
619
			ubuffer += written;
L
Linus Torvalds 已提交
620 621 622
			*count += bufidx;
			bufidx = 0;
		}
623 624 625 626 627 628 629 630 631 632 633 634 635 636 637
		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;
638 639
		agno++;
	} while (agno < mp->m_sb.sb_agcount);
640

L
Linus Torvalds 已提交
641 642
	if (!error) {
		if (bufidx) {
643 644
			long	written;

J
Jie Liu 已提交
645 646
			error = formatter(ubuffer, buffer, bufidx, &written);
			if (!error)
L
Linus Torvalds 已提交
647 648 649 650
				*count += bufidx;
		}
		*lastino = XFS_AGINO_TO_INO(mp, agno, agino);
	}
651

652
	kmem_free(buffer);
L
Linus Torvalds 已提交
653 654 655 656 657
	if (cur)
		xfs_btree_del_cursor(cur, (error ? XFS_BTREE_ERROR :
					   XFS_BTREE_NOERROR));
	if (agbp)
		xfs_buf_relse(agbp);
658

L
Linus Torvalds 已提交
659 660
	return error;
}