xfs_itable.c 17.4 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"
L
Linus Torvalds 已提交
36

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

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

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

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

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

	dic = &ip->i_d;

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

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

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

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

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

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

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

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

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

J
Jie Liu 已提交
264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336
/*
 * Process inodes in chunk with a pointer to a formatter function
 * that will iget the inode and fill in the appropriate structure.
 */
int
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,
	struct xfs_bulkstat_agichunk	*acp)
{
	xfs_ino_t			lastino = acp->ac_lastino;
	char				__user **ubufp = acp->ac_ubuffer;
	int				ubleft = acp->ac_ubleft;
	int				ubelem = acp->ac_ubelem;
	int				chunkidx, clustidx;
	int				error = 0;
	xfs_agino_t			agino;

	for (agino = irbp->ir_startino, chunkidx = clustidx = 0;
	     XFS_BULKSTAT_UBLEFT(ubleft) &&
	     irbp->ir_freecount < XFS_INODES_PER_CHUNK;
	     chunkidx++, clustidx++, agino++) {
		int		fmterror;	/* bulkstat formatter result */
		int		ubused;
		xfs_ino_t	ino = XFS_AGINO_TO_INO(mp, agno, agino);

		ASSERT(chunkidx < XFS_INODES_PER_CHUNK);

		/* Skip if this inode is free */
		if (XFS_INOBT_MASK(chunkidx) & irbp->ir_free) {
			lastino = ino;
			continue;
		}

		/*
		 * Count used inodes as free so we can tell when the
		 * chunk is used up.
		 */
		irbp->ir_freecount++;

		/* Get the inode and fill in a single buffer */
		ubused = statstruct_size;
		error = formatter(mp, ino, *ubufp, ubleft, &ubused, &fmterror);
		if (fmterror == BULKSTAT_RV_NOTHING) {
			if (error && error != -ENOENT && error != -EINVAL) {
				ubleft = 0;
				break;
			}
			lastino = ino;
			continue;
		}
		if (fmterror == BULKSTAT_RV_GIVEUP) {
			ubleft = 0;
			ASSERT(error);
			break;
		}
		if (*ubufp)
			*ubufp += ubused;
		ubleft -= ubused;
		ubelem++;
		lastino = ino;
	}

	acp->ac_lastino = lastino;
	acp->ac_ubleft = ubleft;
	acp->ac_ubelem = ubelem;

	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 354 355 356 357 358 359
{
	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                     fmterror;/* bulkstat formatter result */
	int			i;	/* loop index */
	int			icount;	/* count of inodes good in irbuf */
360
	size_t			irbsize; /* size of irec buffer in bytes */
L
Linus Torvalds 已提交
361
	xfs_ino_t		ino;	/* inode number (filesystem) */
362 363 364
	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 */
365
	xfs_ino_t		lastino; /* last inode number returned */
L
Linus Torvalds 已提交
366 367 368 369 370 371 372 373 374 375 376 377
	int			nirbuf;	/* size of irbuf */
	int			rval;	/* return value error code */
	int			tmp;	/* result value from btree calls */
	int			ubcount; /* size of user's buffer */
	int			ubleft;	/* bytes left in user's buffer */
	char			__user *ubufp;	/* pointer into user's buffer */
	int			ubelem;	/* spaces used in user's buffer */

	/*
	 * Get the last inode value, see if there's nothing to do.
	 */
	ino = (xfs_ino_t)*lastinop;
378
	lastino = ino;
L
Linus Torvalds 已提交
379 380 381 382 383 384 385 386
	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;
	}
387

L
Linus Torvalds 已提交
388 389 390 391 392 393
	ubcount = *ubcountp; /* statstruct's */
	ubleft = ubcount * statstruct_size; /* bytes */
	*ubcountp = ubelem = 0;
	*done = 0;
	fmterror = 0;
	ubufp = ubuffer;
394 395
	irbuf = kmem_zalloc_greedy(&irbsize, PAGE_SIZE, PAGE_SIZE * 4);
	if (!irbuf)
D
Dave Chinner 已提交
396
		return -ENOMEM;
397

398 399
	nirbuf = irbsize / sizeof(*irbuf);

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

			error = xfs_bulkstat_grab_ichunk(cur, agino, &icount, &r);
			if (error)
				break;
			if (icount) {
431 432 433
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
434
				irbp++;
435
				agino = r.ir_startino + XFS_INODES_PER_CHUNK;
L
Linus Torvalds 已提交
436
			}
437 438
			/* Increment to the next record */
			error = xfs_btree_increment(cur, 0, &tmp);
L
Linus Torvalds 已提交
439
		} else {
440
			/* Start of ag.  Lookup the first inode chunk */
441
			error = xfs_inobt_lookup(cur, 0, XFS_LOOKUP_GE, &tmp);
L
Linus Torvalds 已提交
442
		}
443 444 445
		if (error)
			break;

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

			error = xfs_inobt_get_rec(cur, &r, &i);
			if (error || i == 0) {
L
Linus Torvalds 已提交
455 456 457
				end_of_ag = 1;
				break;
			}
458

L
Linus Torvalds 已提交
459 460
			/*
			 * If this chunk has any allocated inodes, save it.
461
			 * Also start read-ahead now for this chunk.
L
Linus Torvalds 已提交
462
			 */
463
			if (r.ir_freecount < XFS_INODES_PER_CHUNK) {
J
Jie Liu 已提交
464
				xfs_bulkstat_ichunk_ra(mp, agno, &r);
465 466 467
				irbp->ir_startino = r.ir_startino;
				irbp->ir_freecount = r.ir_freecount;
				irbp->ir_free = r.ir_free;
L
Linus Torvalds 已提交
468
				irbp++;
469
				icount += XFS_INODES_PER_CHUNK - r.ir_freecount;
L
Linus Torvalds 已提交
470 471 472 473
			}
			/*
			 * Set agino to after this chunk and bump the cursor.
			 */
474
			agino = r.ir_startino + XFS_INODES_PER_CHUNK;
475
			error = xfs_btree_increment(cur, 0, &tmp);
476
			cond_resched();
L
Linus Torvalds 已提交
477 478 479 480 481 482 483 484 485 486 487 488 489
		}
		/*
		 * Drop the btree buffers and the agi buffer.
		 * We can't hold any of the locks these represent
		 * when calling iget.
		 */
		xfs_btree_del_cursor(cur, XFS_BTREE_NOERROR);
		xfs_buf_relse(agbp);
		/*
		 * Now format all the good inodes into the user's buffer.
		 */
		irbufend = irbp;
		for (irbp = irbuf;
490
		     irbp < irbufend && XFS_BULKSTAT_UBLEFT(ubleft); irbp++) {
J
Jie Liu 已提交
491 492 493 494 495 496 497 498 499 500 501 502 503 504
			struct xfs_bulkstat_agichunk ac;

			ac.ac_lastino = lastino;
			ac.ac_ubuffer = &ubuffer;
			ac.ac_ubleft = ubleft;
			ac.ac_ubelem = ubelem;
			error = xfs_bulkstat_ag_ichunk(mp, agno, irbp,
					formatter, statstruct_size, &ac);
			if (error)
				rval = error;

			lastino = ac.ac_lastino;
			ubleft = ac.ac_ubleft;
			ubelem = ac.ac_ubelem;
505 506

			cond_resched();
L
Linus Torvalds 已提交
507 508 509 510
		}
		/*
		 * Set up for the next loop iteration.
		 */
511
		if (XFS_BULKSTAT_UBLEFT(ubleft)) {
L
Linus Torvalds 已提交
512 513 514
			if (end_of_ag) {
				agno++;
				agino = 0;
515 516
			} else
				agino = XFS_INO_TO_AGINO(mp, lastino);
L
Linus Torvalds 已提交
517 518 519 520 521 522
		} else
			break;
	}
	/*
	 * Done, we're either out of filesystem or space to put the data.
	 */
523
	kmem_free(irbuf);
L
Linus Torvalds 已提交
524
	*ubcountp = ubelem;
525 526 527 528 529
	/*
	 * Found some inodes, return them now and return the error next time.
	 */
	if (ubelem)
		rval = 0;
L
Linus Torvalds 已提交
530 531 532 533 534 535 536 537 538 539 540 541 542 543
	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;
}

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

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

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

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

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

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

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

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

L
Linus Torvalds 已提交
644 645
	if (!error) {
		if (bufidx) {
646 647
			long	written;

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

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

L
Linus Torvalds 已提交
662 663
	return error;
}