xfs_attr_leaf.c 81.7 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 * Copyright (c) 2000-2005 Silicon Graphics, Inc.
D
Dave Chinner 已提交
3
 * Copyright (c) 2013 Red Hat, Inc.
4
 * All Rights Reserved.
L
Linus Torvalds 已提交
5
 *
6 7
 * 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 已提交
8 9
 * published by the Free Software Foundation.
 *
10 11 12 13
 * 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 已提交
14
 *
15 16 17
 * 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 已提交
18 19
 */
#include "xfs.h"
20
#include "xfs_fs.h"
21
#include "xfs_shared.h"
22
#include "xfs_format.h"
23 24
#include "xfs_log_format.h"
#include "xfs_trans_resv.h"
25
#include "xfs_bit.h"
L
Linus Torvalds 已提交
26 27
#include "xfs_sb.h"
#include "xfs_mount.h"
28
#include "xfs_da_format.h"
29
#include "xfs_da_btree.h"
L
Linus Torvalds 已提交
30
#include "xfs_inode.h"
31
#include "xfs_trans.h"
32
#include "xfs_inode_item.h"
33
#include "xfs_bmap_btree.h"
L
Linus Torvalds 已提交
34
#include "xfs_bmap.h"
35 36
#include "xfs_attr_sf.h"
#include "xfs_attr_remote.h"
L
Linus Torvalds 已提交
37 38 39
#include "xfs_attr.h"
#include "xfs_attr_leaf.h"
#include "xfs_error.h"
C
Christoph Hellwig 已提交
40
#include "xfs_trace.h"
D
Dave Chinner 已提交
41 42
#include "xfs_buf_item.h"
#include "xfs_cksum.h"
D
Dave Chinner 已提交
43
#include "xfs_dir2.h"
44
#include "xfs_log.h"
D
Dave Chinner 已提交
45

L
Linus Torvalds 已提交
46 47 48 49 50 51 52 53 54 55 56 57 58 59

/*
 * xfs_attr_leaf.c
 *
 * Routines to implement leaf blocks of attributes as Btrees of hashed names.
 */

/*========================================================================
 * Function prototypes for the kernel.
 *========================================================================*/

/*
 * Routines used for growing the Btree.
 */
D
Dave Chinner 已提交
60 61 62 63 64 65 66 67 68
STATIC int xfs_attr3_leaf_create(struct xfs_da_args *args,
				 xfs_dablk_t which_block, struct xfs_buf **bpp);
STATIC int xfs_attr3_leaf_add_work(struct xfs_buf *leaf_buffer,
				   struct xfs_attr3_icleaf_hdr *ichdr,
				   struct xfs_da_args *args, int freemap_index);
STATIC void xfs_attr3_leaf_compact(struct xfs_da_args *args,
				   struct xfs_attr3_icleaf_hdr *ichdr,
				   struct xfs_buf *leaf_buffer);
STATIC void xfs_attr3_leaf_rebalance(xfs_da_state_t *state,
L
Linus Torvalds 已提交
69 70
						   xfs_da_state_blk_t *blk1,
						   xfs_da_state_blk_t *blk2);
D
Dave Chinner 已提交
71 72 73 74 75 76 77
STATIC int xfs_attr3_leaf_figure_balance(xfs_da_state_t *state,
			xfs_da_state_blk_t *leaf_blk_1,
			struct xfs_attr3_icleaf_hdr *ichdr1,
			xfs_da_state_blk_t *leaf_blk_2,
			struct xfs_attr3_icleaf_hdr *ichdr2,
			int *number_entries_in_blk1,
			int *number_usedbytes_in_blk1);
L
Linus Torvalds 已提交
78 79 80 81

/*
 * Utility routines.
 */
82 83
STATIC void xfs_attr3_leaf_moveents(struct xfs_da_args *args,
			struct xfs_attr_leafblock *src_leaf,
D
Dave Chinner 已提交
84 85 86
			struct xfs_attr3_icleaf_hdr *src_ichdr, int src_start,
			struct xfs_attr_leafblock *dst_leaf,
			struct xfs_attr3_icleaf_hdr *dst_ichdr, int dst_start,
87
			int move_count);
88
STATIC int xfs_attr_leaf_entsize(xfs_attr_leafblock_t *leaf, int index);
89

90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
/*
 * attr3 block 'firstused' conversion helpers.
 *
 * firstused refers to the offset of the first used byte of the nameval region
 * of an attr leaf block. The region starts at the tail of the block and expands
 * backwards towards the middle. As such, firstused is initialized to the block
 * size for an empty leaf block and is reduced from there.
 *
 * The attr3 block size is pegged to the fsb size and the maximum fsb is 64k.
 * The in-core firstused field is 32-bit and thus supports the maximum fsb size.
 * The on-disk field is only 16-bit, however, and overflows at 64k. Since this
 * only occurs at exactly 64k, we use zero as a magic on-disk value to represent
 * the attr block size. The following helpers manage the conversion between the
 * in-core and on-disk formats.
 */

static void
xfs_attr3_leaf_firstused_from_disk(
	struct xfs_da_geometry		*geo,
	struct xfs_attr3_icleaf_hdr	*to,
	struct xfs_attr_leafblock	*from)
{
	struct xfs_attr3_leaf_hdr	*hdr3;

	if (from->hdr.info.magic == cpu_to_be16(XFS_ATTR3_LEAF_MAGIC)) {
		hdr3 = (struct xfs_attr3_leaf_hdr *) from;
		to->firstused = be16_to_cpu(hdr3->firstused);
	} else {
		to->firstused = be16_to_cpu(from->hdr.firstused);
	}

	/*
	 * Convert from the magic fsb size value to actual blocksize. This
	 * should only occur for empty blocks when the block size overflows
	 * 16-bits.
	 */
	if (to->firstused == XFS_ATTR3_LEAF_NULLOFF) {
		ASSERT(!to->count && !to->usedbytes);
		ASSERT(geo->blksize > USHRT_MAX);
		to->firstused = geo->blksize;
	}
}

static void
xfs_attr3_leaf_firstused_to_disk(
	struct xfs_da_geometry		*geo,
	struct xfs_attr_leafblock	*to,
	struct xfs_attr3_icleaf_hdr	*from)
{
	struct xfs_attr3_leaf_hdr	*hdr3;
	uint32_t			firstused;

	/* magic value should only be seen on disk */
	ASSERT(from->firstused != XFS_ATTR3_LEAF_NULLOFF);

	/*
	 * Scale down the 32-bit in-core firstused value to the 16-bit on-disk
	 * value. This only overflows at the max supported value of 64k. Use the
	 * magic on-disk value to represent block size in this case.
	 */
	firstused = from->firstused;
	if (firstused > USHRT_MAX) {
		ASSERT(from->firstused == geo->blksize);
		firstused = XFS_ATTR3_LEAF_NULLOFF;
	}

	if (from->magic == XFS_ATTR3_LEAF_MAGIC) {
		hdr3 = (struct xfs_attr3_leaf_hdr *) to;
		hdr3->firstused = cpu_to_be16(firstused);
	} else {
		to->hdr.firstused = cpu_to_be16(firstused);
	}
}

D
Dave Chinner 已提交
164 165
void
xfs_attr3_leaf_hdr_from_disk(
166
	struct xfs_da_geometry		*geo,
D
Dave Chinner 已提交
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182
	struct xfs_attr3_icleaf_hdr	*to,
	struct xfs_attr_leafblock	*from)
{
	int	i;

	ASSERT(from->hdr.info.magic == cpu_to_be16(XFS_ATTR_LEAF_MAGIC) ||
	       from->hdr.info.magic == cpu_to_be16(XFS_ATTR3_LEAF_MAGIC));

	if (from->hdr.info.magic == cpu_to_be16(XFS_ATTR3_LEAF_MAGIC)) {
		struct xfs_attr3_leaf_hdr *hdr3 = (struct xfs_attr3_leaf_hdr *)from;

		to->forw = be32_to_cpu(hdr3->info.hdr.forw);
		to->back = be32_to_cpu(hdr3->info.hdr.back);
		to->magic = be16_to_cpu(hdr3->info.hdr.magic);
		to->count = be16_to_cpu(hdr3->count);
		to->usedbytes = be16_to_cpu(hdr3->usedbytes);
183
		xfs_attr3_leaf_firstused_from_disk(geo, to, from);
D
Dave Chinner 已提交
184 185 186 187 188 189 190 191 192 193 194 195 196
		to->holes = hdr3->holes;

		for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
			to->freemap[i].base = be16_to_cpu(hdr3->freemap[i].base);
			to->freemap[i].size = be16_to_cpu(hdr3->freemap[i].size);
		}
		return;
	}
	to->forw = be32_to_cpu(from->hdr.info.forw);
	to->back = be32_to_cpu(from->hdr.info.back);
	to->magic = be16_to_cpu(from->hdr.info.magic);
	to->count = be16_to_cpu(from->hdr.count);
	to->usedbytes = be16_to_cpu(from->hdr.usedbytes);
197
	xfs_attr3_leaf_firstused_from_disk(geo, to, from);
D
Dave Chinner 已提交
198 199 200 201 202 203 204 205 206 207
	to->holes = from->hdr.holes;

	for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
		to->freemap[i].base = be16_to_cpu(from->hdr.freemap[i].base);
		to->freemap[i].size = be16_to_cpu(from->hdr.freemap[i].size);
	}
}

void
xfs_attr3_leaf_hdr_to_disk(
208
	struct xfs_da_geometry		*geo,
D
Dave Chinner 已提交
209 210 211
	struct xfs_attr_leafblock	*to,
	struct xfs_attr3_icleaf_hdr	*from)
{
212
	int				i;
D
Dave Chinner 已提交
213 214 215 216 217 218 219 220 221 222 223 224

	ASSERT(from->magic == XFS_ATTR_LEAF_MAGIC ||
	       from->magic == XFS_ATTR3_LEAF_MAGIC);

	if (from->magic == XFS_ATTR3_LEAF_MAGIC) {
		struct xfs_attr3_leaf_hdr *hdr3 = (struct xfs_attr3_leaf_hdr *)to;

		hdr3->info.hdr.forw = cpu_to_be32(from->forw);
		hdr3->info.hdr.back = cpu_to_be32(from->back);
		hdr3->info.hdr.magic = cpu_to_be16(from->magic);
		hdr3->count = cpu_to_be16(from->count);
		hdr3->usedbytes = cpu_to_be16(from->usedbytes);
225
		xfs_attr3_leaf_firstused_to_disk(geo, to, from);
D
Dave Chinner 已提交
226 227 228 229 230 231 232 233 234 235 236 237 238 239
		hdr3->holes = from->holes;
		hdr3->pad1 = 0;

		for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
			hdr3->freemap[i].base = cpu_to_be16(from->freemap[i].base);
			hdr3->freemap[i].size = cpu_to_be16(from->freemap[i].size);
		}
		return;
	}
	to->hdr.info.forw = cpu_to_be32(from->forw);
	to->hdr.info.back = cpu_to_be32(from->back);
	to->hdr.info.magic = cpu_to_be16(from->magic);
	to->hdr.count = cpu_to_be16(from->count);
	to->hdr.usedbytes = cpu_to_be16(from->usedbytes);
240
	xfs_attr3_leaf_firstused_to_disk(geo, to, from);
D
Dave Chinner 已提交
241 242 243 244 245 246 247 248 249
	to->hdr.holes = from->holes;
	to->hdr.pad1 = 0;

	for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
		to->hdr.freemap[i].base = cpu_to_be16(from->freemap[i].base);
		to->hdr.freemap[i].size = cpu_to_be16(from->freemap[i].size);
	}
}

250
static xfs_failaddr_t
D
Dave Chinner 已提交
251
xfs_attr3_leaf_verify(
252
	struct xfs_buf			*bp)
253
{
254 255 256 257 258
	struct xfs_attr3_icleaf_hdr	ichdr;
	struct xfs_mount		*mp = bp->b_target->bt_mount;
	struct xfs_attr_leafblock	*leaf = bp->b_addr;
	struct xfs_perag		*pag = bp->b_pag;
	struct xfs_attr_leaf_entry	*entries;
259

260
	xfs_attr3_leaf_hdr_from_disk(mp->m_attr_geo, &ichdr, leaf);
D
Dave Chinner 已提交
261 262 263 264 265

	if (xfs_sb_version_hascrc(&mp->m_sb)) {
		struct xfs_da3_node_hdr *hdr3 = bp->b_addr;

		if (ichdr.magic != XFS_ATTR3_LEAF_MAGIC)
266
			return __this_address;
D
Dave Chinner 已提交
267

268
		if (!uuid_equal(&hdr3->info.uuid, &mp->m_sb.sb_meta_uuid))
269
			return __this_address;
D
Dave Chinner 已提交
270
		if (be64_to_cpu(hdr3->info.blkno) != bp->b_bn)
271
			return __this_address;
272
		if (!xfs_log_check_lsn(mp, be64_to_cpu(hdr3->info.lsn)))
273
			return __this_address;
D
Dave Chinner 已提交
274 275
	} else {
		if (ichdr.magic != XFS_ATTR_LEAF_MAGIC)
276
			return __this_address;
277
	}
278 279 280 281 282 283
	/*
	 * In recovery there is a transient state where count == 0 is valid
	 * because we may have transitioned an empty shortform attr to a leaf
	 * if the attr didn't fit in shortform.
	 */
	if (pag && pag->pagf_init && ichdr.count == 0)
284
		return __this_address;
D
Dave Chinner 已提交
285

286 287 288 289 290 291 292 293 294 295 296 297 298 299 300
	/*
	 * firstused is the block offset of the first name info structure.
	 * Make sure it doesn't go off the block or crash into the header.
	 */
	if (ichdr.firstused > mp->m_attr_geo->blksize)
		return __this_address;
	if (ichdr.firstused < xfs_attr3_leaf_hdr_size(leaf))
		return __this_address;

	/* Make sure the entries array doesn't crash into the name info. */
	entries = xfs_attr3_leaf_entryp(bp->b_addr);
	if ((char *)&entries[ichdr.count] >
	    (char *)bp->b_addr + ichdr.firstused)
		return __this_address;

D
Dave Chinner 已提交
301 302 303
	/* XXX: need to range check rest of attr header values */
	/* XXX: hash order check? */

304
	return NULL;
305 306 307
}

static void
D
Dave Chinner 已提交
308
xfs_attr3_leaf_write_verify(
309 310
	struct xfs_buf	*bp)
{
D
Dave Chinner 已提交
311
	struct xfs_mount	*mp = bp->b_target->bt_mount;
C
Carlos Maiolino 已提交
312
	struct xfs_buf_log_item	*bip = bp->b_log_item;
D
Dave Chinner 已提交
313
	struct xfs_attr3_leaf_hdr *hdr3 = bp->b_addr;
314
	xfs_failaddr_t		fa;
D
Dave Chinner 已提交
315

316 317 318
	fa = xfs_attr3_leaf_verify(bp);
	if (fa) {
		xfs_verifier_error(bp, -EFSCORRUPTED, fa);
D
Dave Chinner 已提交
319 320 321 322 323 324 325 326 327
		return;
	}

	if (!xfs_sb_version_hascrc(&mp->m_sb))
		return;

	if (bip)
		hdr3->info.lsn = cpu_to_be64(bip->bli_item.li_lsn);

328
	xfs_buf_update_cksum(bp, XFS_ATTR3_LEAF_CRC_OFF);
329
}
330

D
Dave Chinner 已提交
331 332 333 334 335 336
/*
 * leaf/node format detection on trees is sketchy, so a node read can be done on
 * leaf level blocks when detection identifies the tree as a node format tree
 * incorrectly. In this case, we need to swap the verifier to match the correct
 * format of the block being read.
 */
337
static void
D
Dave Chinner 已提交
338 339
xfs_attr3_leaf_read_verify(
	struct xfs_buf		*bp)
340
{
D
Dave Chinner 已提交
341
	struct xfs_mount	*mp = bp->b_target->bt_mount;
342
	xfs_failaddr_t		fa;
D
Dave Chinner 已提交
343

344 345
	if (xfs_sb_version_hascrc(&mp->m_sb) &&
	     !xfs_buf_verify_cksum(bp, XFS_ATTR3_LEAF_CRC_OFF))
346 347 348 349 350 351
		xfs_verifier_error(bp, -EFSBADCRC, __this_address);
	else {
		fa = xfs_attr3_leaf_verify(bp);
		if (fa)
			xfs_verifier_error(bp, -EFSCORRUPTED, fa);
	}
352 353
}

D
Dave Chinner 已提交
354
const struct xfs_buf_ops xfs_attr3_leaf_buf_ops = {
355
	.name = "xfs_attr3_leaf",
D
Dave Chinner 已提交
356 357
	.verify_read = xfs_attr3_leaf_read_verify,
	.verify_write = xfs_attr3_leaf_write_verify,
358
	.verify_struct = xfs_attr3_leaf_verify,
359
};
360

361
int
D
Dave Chinner 已提交
362
xfs_attr3_leaf_read(
363 364 365 366 367 368
	struct xfs_trans	*tp,
	struct xfs_inode	*dp,
	xfs_dablk_t		bno,
	xfs_daddr_t		mappedbno,
	struct xfs_buf		**bpp)
{
369 370 371
	int			err;

	err = xfs_da_read_buf(tp, dp, bno, mappedbno, bpp,
D
Dave Chinner 已提交
372
				XFS_ATTR_FORK, &xfs_attr3_leaf_buf_ops);
373
	if (!err && tp && *bpp)
374
		xfs_trans_buf_set_type(tp, *bpp, XFS_BLFT_ATTR_LEAF_BUF);
375
	return err;
376 377
}

378 379 380 381 382 383 384 385
/*========================================================================
 * Namespace helper routines
 *========================================================================*/

/*
 * If namespace bits don't match return 0.
 * If all match then return 1.
 */
386
STATIC int
387 388 389 390 391
xfs_attr_namesp_match(int arg_flags, int ondisk_flags)
{
	return XFS_ATTR_NSP_ONDISK(ondisk_flags) == XFS_ATTR_NSP_ARGS_TO_ONDISK(arg_flags);
}

L
Linus Torvalds 已提交
392 393

/*========================================================================
394
 * External routines when attribute fork size < XFS_LITINO(mp).
L
Linus Torvalds 已提交
395 396 397
 *========================================================================*/

/*
398 399
 * Query whether the requested number of additional bytes of extended
 * attribute space will be able to fit inline.
400
 *
401 402 403 404 405
 * Returns zero if not, else the di_forkoff fork offset to be used in the
 * literal area for attribute data once the new bytes have been added.
 *
 * di_forkoff must be 8 byte aligned, hence is stored as a >>3 value;
 * special case for dev/uuid inodes, they have fixed size data forks.
L
Linus Torvalds 已提交
406 407
 */
int
408 409 410 411 412
xfs_attr_shortform_bytesfit(xfs_inode_t *dp, int bytes)
{
	int offset;
	int minforkoff;	/* lower limit on valid forkoff locations */
	int maxforkoff;	/* upper limit on valid forkoff locations */
413
	int dsize;
414 415
	xfs_mount_t *mp = dp->i_mount;

416 417
	/* rounded down */
	offset = (XFS_LITINO(mp, dp->i_d.di_version) - bytes) >> 3;
418

419
	if (dp->i_d.di_format == XFS_DINODE_FMT_DEV) {
420 421 422 423
		minforkoff = roundup(sizeof(xfs_dev_t), 8) >> 3;
		return (offset >= minforkoff) ? minforkoff : 0;
	}

424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442
	/*
	 * If the requested numbers of bytes is smaller or equal to the
	 * current attribute fork size we can always proceed.
	 *
	 * Note that if_bytes in the data fork might actually be larger than
	 * the current data fork size is due to delalloc extents. In that
	 * case either the extent count will go down when they are converted
	 * to real extents, or the delalloc conversion will take care of the
	 * literal area rebalancing.
	 */
	if (bytes <= XFS_IFORK_ASIZE(dp))
		return dp->i_d.di_forkoff;

	/*
	 * For attr2 we can try to move the forkoff if there is space in the
	 * literal area, but for the old format we are done if there is no
	 * space in the fixed attribute fork.
	 */
	if (!(mp->m_flags & XFS_MOUNT_ATTR2))
443 444
		return 0;

445
	dsize = dp->i_df.if_bytes;
446

447 448
	switch (dp->i_d.di_format) {
	case XFS_DINODE_FMT_EXTENTS:
449
		/*
450
		 * If there is no attr fork and the data fork is extents, 
451 452 453
		 * determine if creating the default attr fork will result
		 * in the extents form migrating to btree. If so, the
		 * minimum offset only needs to be the space required for
454
		 * the btree root.
455
		 */
C
Christoph Hellwig 已提交
456 457
		if (!dp->i_d.di_forkoff && dp->i_df.if_bytes >
		    xfs_default_attroffset(dp))
458 459 460 461
			dsize = XFS_BMDR_SPACE_CALC(MINDBTPTRS);
		break;
	case XFS_DINODE_FMT_BTREE:
		/*
462 463 464
		 * If we have a data btree then keep forkoff if we have one,
		 * otherwise we are adding a new attr, so then we set
		 * minforkoff to where the btree root can finish so we have
465 466 467
		 * plenty of room for attrs
		 */
		if (dp->i_d.di_forkoff) {
468
			if (offset < dp->i_d.di_forkoff)
469
				return 0;
470 471
			return dp->i_d.di_forkoff;
		}
472
		dsize = XFS_BMAP_BROOT_SPACE(mp, dp->i_df.if_broot);
473 474
		break;
	}
475 476 477

	/*
	 * A data fork btree root must have space for at least
478 479 480
	 * MINDBTPTRS key/ptr pairs if the data fork is small or empty.
	 */
	minforkoff = MAX(dsize, XFS_BMDR_SPACE_CALC(MINDBTPTRS));
481 482 483
	minforkoff = roundup(minforkoff, 8) >> 3;

	/* attr fork btree root can have at least this many key/ptr pairs */
484 485
	maxforkoff = XFS_LITINO(mp, dp->i_d.di_version) -
			XFS_BMDR_SPACE_CALC(MINABTPTRS);
486 487 488 489
	maxforkoff = maxforkoff >> 3;	/* rounded down */

	if (offset >= maxforkoff)
		return maxforkoff;
490 491
	if (offset >= minforkoff)
		return offset;
492 493 494 495 496 497 498 499 500
	return 0;
}

/*
 * Switch on the ATTR2 superblock bit (implies also FEATURES2)
 */
STATIC void
xfs_sbversion_add_attr2(xfs_mount_t *mp, xfs_trans_t *tp)
{
501
	if ((mp->m_flags & XFS_MOUNT_ATTR2) &&
502
	    !(xfs_sb_version_hasattr2(&mp->m_sb))) {
E
Eric Sandeen 已提交
503
		spin_lock(&mp->m_sb_lock);
504 505
		if (!xfs_sb_version_hasattr2(&mp->m_sb)) {
			xfs_sb_version_addattr2(&mp->m_sb);
E
Eric Sandeen 已提交
506
			spin_unlock(&mp->m_sb_lock);
507
			xfs_log_sb(tp);
508
		} else
E
Eric Sandeen 已提交
509
			spin_unlock(&mp->m_sb_lock);
510 511 512 513 514 515 516
	}
}

/*
 * Create the initial contents of a shortform attribute list.
 */
void
L
Linus Torvalds 已提交
517 518 519 520 521 522
xfs_attr_shortform_create(xfs_da_args_t *args)
{
	xfs_attr_sf_hdr_t *hdr;
	xfs_inode_t *dp;
	xfs_ifork_t *ifp;

523 524
	trace_xfs_attr_sf_create(args);

L
Linus Torvalds 已提交
525 526 527 528 529 530 531 532 533 534 535 536 537 538 539
	dp = args->dp;
	ASSERT(dp != NULL);
	ifp = dp->i_afp;
	ASSERT(ifp != NULL);
	ASSERT(ifp->if_bytes == 0);
	if (dp->i_d.di_aformat == XFS_DINODE_FMT_EXTENTS) {
		ifp->if_flags &= ~XFS_IFEXTENTS;	/* just in case */
		dp->i_d.di_aformat = XFS_DINODE_FMT_LOCAL;
		ifp->if_flags |= XFS_IFINLINE;
	} else {
		ASSERT(ifp->if_flags & XFS_IFINLINE);
	}
	xfs_idata_realloc(dp, sizeof(*hdr), XFS_ATTR_FORK);
	hdr = (xfs_attr_sf_hdr_t *)ifp->if_u1.if_data;
	hdr->count = 0;
540
	hdr->totsize = cpu_to_be16(sizeof(*hdr));
L
Linus Torvalds 已提交
541 542 543 544 545 546 547
	xfs_trans_log_inode(args->trans, dp, XFS_ILOG_CORE | XFS_ILOG_ADATA);
}

/*
 * Add a name/value pair to the shortform attribute list.
 * Overflow from the inode has already been checked for.
 */
548 549
void
xfs_attr_shortform_add(xfs_da_args_t *args, int forkoff)
L
Linus Torvalds 已提交
550 551 552 553
{
	xfs_attr_shortform_t *sf;
	xfs_attr_sf_entry_t *sfe;
	int i, offset, size;
554
	xfs_mount_t *mp;
L
Linus Torvalds 已提交
555 556 557
	xfs_inode_t *dp;
	xfs_ifork_t *ifp;

558 559
	trace_xfs_attr_sf_add(args);

L
Linus Torvalds 已提交
560
	dp = args->dp;
561 562 563
	mp = dp->i_mount;
	dp->i_d.di_forkoff = forkoff;

L
Linus Torvalds 已提交
564 565 566 567
	ifp = dp->i_afp;
	ASSERT(ifp->if_flags & XFS_IFINLINE);
	sf = (xfs_attr_shortform_t *)ifp->if_u1.if_data;
	sfe = &sf->list[0];
568
	for (i = 0; i < sf->hdr.count; sfe = XFS_ATTR_SF_NEXTENTRY(sfe), i++) {
569
#ifdef DEBUG
L
Linus Torvalds 已提交
570 571 572 573
		if (sfe->namelen != args->namelen)
			continue;
		if (memcmp(args->name, sfe->nameval, args->namelen) != 0)
			continue;
574
		if (!xfs_attr_namesp_match(args->flags, sfe->flags))
L
Linus Torvalds 已提交
575
			continue;
576 577
		ASSERT(0);
#endif
L
Linus Torvalds 已提交
578 579 580 581 582 583 584 585 586
	}

	offset = (char *)sfe - (char *)sf;
	size = XFS_ATTR_SF_ENTSIZE_BYNAME(args->namelen, args->valuelen);
	xfs_idata_realloc(dp, size, XFS_ATTR_FORK);
	sf = (xfs_attr_shortform_t *)ifp->if_u1.if_data;
	sfe = (xfs_attr_sf_entry_t *)((char *)sf + offset);

	sfe->namelen = args->namelen;
587
	sfe->valuelen = args->valuelen;
588
	sfe->flags = XFS_ATTR_NSP_ARGS_TO_ONDISK(args->flags);
L
Linus Torvalds 已提交
589 590
	memcpy(sfe->nameval, args->name, args->namelen);
	memcpy(&sfe->nameval[args->namelen], args->value, args->valuelen);
591
	sf->hdr.count++;
592
	be16_add_cpu(&sf->hdr.totsize, size);
L
Linus Torvalds 已提交
593 594
	xfs_trans_log_inode(args->trans, dp, XFS_ILOG_CORE | XFS_ILOG_ADATA);

595
	xfs_sbversion_add_attr2(mp, args->trans);
L
Linus Torvalds 已提交
596 597
}

598 599 600 601
/*
 * After the last attribute is removed revert to original inode format,
 * making all literal area available to the data fork once more.
 */
602 603
void
xfs_attr_fork_remove(
604 605 606 607 608 609 610 611 612 613 614 615 616
	struct xfs_inode	*ip,
	struct xfs_trans	*tp)
{
	xfs_idestroy_fork(ip, XFS_ATTR_FORK);
	ip->i_d.di_forkoff = 0;
	ip->i_d.di_aformat = XFS_DINODE_FMT_EXTENTS;

	ASSERT(ip->i_d.di_anextents == 0);
	ASSERT(ip->i_afp == NULL);

	xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
}

L
Linus Torvalds 已提交
617
/*
618
 * Remove an attribute from the shortform attribute list structure.
L
Linus Torvalds 已提交
619 620 621 622 623 624 625
 */
int
xfs_attr_shortform_remove(xfs_da_args_t *args)
{
	xfs_attr_shortform_t *sf;
	xfs_attr_sf_entry_t *sfe;
	int base, size=0, end, totsize, i;
626
	xfs_mount_t *mp;
L
Linus Torvalds 已提交
627 628
	xfs_inode_t *dp;

629 630
	trace_xfs_attr_sf_remove(args);

L
Linus Torvalds 已提交
631
	dp = args->dp;
632
	mp = dp->i_mount;
L
Linus Torvalds 已提交
633 634 635
	base = sizeof(xfs_attr_sf_hdr_t);
	sf = (xfs_attr_shortform_t *)dp->i_afp->if_u1.if_data;
	sfe = &sf->list[0];
636
	end = sf->hdr.count;
637
	for (i = 0; i < end; sfe = XFS_ATTR_SF_NEXTENTRY(sfe),
L
Linus Torvalds 已提交
638 639 640 641 642 643
					base += size, i++) {
		size = XFS_ATTR_SF_ENTSIZE(sfe);
		if (sfe->namelen != args->namelen)
			continue;
		if (memcmp(sfe->nameval, args->name, args->namelen) != 0)
			continue;
644
		if (!xfs_attr_namesp_match(args->flags, sfe->flags))
L
Linus Torvalds 已提交
645 646 647
			continue;
		break;
	}
648
	if (i == end)
D
Dave Chinner 已提交
649
		return -ENOATTR;
L
Linus Torvalds 已提交
650

651 652 653
	/*
	 * Fix up the attribute fork data, covering the hole
	 */
L
Linus Torvalds 已提交
654
	end = base + size;
655
	totsize = be16_to_cpu(sf->hdr.totsize);
656 657
	if (end != totsize)
		memmove(&((char *)sf)[base], &((char *)sf)[end], totsize - end);
658
	sf->hdr.count--;
659
	be16_add_cpu(&sf->hdr.totsize, -size);
660 661 662 663 664

	/*
	 * Fix up the start offset of the attribute fork
	 */
	totsize -= size;
665
	if (totsize == sizeof(xfs_attr_sf_hdr_t) &&
666 667 668
	    (mp->m_flags & XFS_MOUNT_ATTR2) &&
	    (dp->i_d.di_format != XFS_DINODE_FMT_BTREE) &&
	    !(args->op_flags & XFS_DA_OP_ADDNAME)) {
669
		xfs_attr_fork_remove(dp, args->trans);
670 671 672 673
	} else {
		xfs_idata_realloc(dp, -size, XFS_ATTR_FORK);
		dp->i_d.di_forkoff = xfs_attr_shortform_bytesfit(dp, totsize);
		ASSERT(dp->i_d.di_forkoff);
674 675 676 677
		ASSERT(totsize > sizeof(xfs_attr_sf_hdr_t) ||
				(args->op_flags & XFS_DA_OP_ADDNAME) ||
				!(mp->m_flags & XFS_MOUNT_ATTR2) ||
				dp->i_d.di_format == XFS_DINODE_FMT_BTREE);
678 679 680 681 682
		xfs_trans_log_inode(args->trans, dp,
					XFS_ILOG_CORE | XFS_ILOG_ADATA);
	}

	xfs_sbversion_add_attr2(mp, args->trans);
L
Linus Torvalds 已提交
683

E
Eric Sandeen 已提交
684
	return 0;
L
Linus Torvalds 已提交
685 686 687 688 689 690 691 692 693 694 695 696 697 698
}

/*
 * Look up a name in a shortform attribute list structure.
 */
/*ARGSUSED*/
int
xfs_attr_shortform_lookup(xfs_da_args_t *args)
{
	xfs_attr_shortform_t *sf;
	xfs_attr_sf_entry_t *sfe;
	int i;
	xfs_ifork_t *ifp;

699 700
	trace_xfs_attr_sf_lookup(args);

L
Linus Torvalds 已提交
701 702 703 704
	ifp = args->dp->i_afp;
	ASSERT(ifp->if_flags & XFS_IFINLINE);
	sf = (xfs_attr_shortform_t *)ifp->if_u1.if_data;
	sfe = &sf->list[0];
705
	for (i = 0; i < sf->hdr.count;
L
Linus Torvalds 已提交
706 707 708 709 710
				sfe = XFS_ATTR_SF_NEXTENTRY(sfe), i++) {
		if (sfe->namelen != args->namelen)
			continue;
		if (memcmp(args->name, sfe->nameval, args->namelen) != 0)
			continue;
711
		if (!xfs_attr_namesp_match(args->flags, sfe->flags))
L
Linus Torvalds 已提交
712
			continue;
D
Dave Chinner 已提交
713
		return -EEXIST;
L
Linus Torvalds 已提交
714
	}
D
Dave Chinner 已提交
715
	return -ENOATTR;
L
Linus Torvalds 已提交
716 717 718 719 720 721 722 723 724 725 726 727 728
}

/*
 * Look up a name in a shortform attribute list structure.
 */
/*ARGSUSED*/
int
xfs_attr_shortform_getvalue(xfs_da_args_t *args)
{
	xfs_attr_shortform_t *sf;
	xfs_attr_sf_entry_t *sfe;
	int i;

729
	ASSERT(args->dp->i_afp->if_flags == XFS_IFINLINE);
L
Linus Torvalds 已提交
730 731
	sf = (xfs_attr_shortform_t *)args->dp->i_afp->if_u1.if_data;
	sfe = &sf->list[0];
732
	for (i = 0; i < sf->hdr.count;
L
Linus Torvalds 已提交
733 734 735 736 737
				sfe = XFS_ATTR_SF_NEXTENTRY(sfe), i++) {
		if (sfe->namelen != args->namelen)
			continue;
		if (memcmp(args->name, sfe->nameval, args->namelen) != 0)
			continue;
738
		if (!xfs_attr_namesp_match(args->flags, sfe->flags))
L
Linus Torvalds 已提交
739 740
			continue;
		if (args->flags & ATTR_KERNOVAL) {
741
			args->valuelen = sfe->valuelen;
D
Dave Chinner 已提交
742
			return -EEXIST;
L
Linus Torvalds 已提交
743
		}
744 745
		if (args->valuelen < sfe->valuelen) {
			args->valuelen = sfe->valuelen;
D
Dave Chinner 已提交
746
			return -ERANGE;
L
Linus Torvalds 已提交
747
		}
748
		args->valuelen = sfe->valuelen;
L
Linus Torvalds 已提交
749 750
		memcpy(args->value, &sfe->nameval[args->namelen],
						    args->valuelen);
D
Dave Chinner 已提交
751
		return -EEXIST;
L
Linus Torvalds 已提交
752
	}
D
Dave Chinner 已提交
753
	return -ENOATTR;
L
Linus Torvalds 已提交
754 755 756
}

/*
757 758
 * Convert from using the shortform to the leaf.  On success, return the
 * buffer so that we can keep it locked until we're totally done with it.
L
Linus Torvalds 已提交
759 760
 */
int
761 762 763
xfs_attr_shortform_to_leaf(
	struct xfs_da_args	*args,
	struct xfs_buf		**leaf_bp)
L
Linus Torvalds 已提交
764 765 766 767 768 769 770 771
{
	xfs_inode_t *dp;
	xfs_attr_shortform_t *sf;
	xfs_attr_sf_entry_t *sfe;
	xfs_da_args_t nargs;
	char *tmpbuffer;
	int error, i, size;
	xfs_dablk_t blkno;
772
	struct xfs_buf *bp;
L
Linus Torvalds 已提交
773 774
	xfs_ifork_t *ifp;

775 776
	trace_xfs_attr_sf_to_leaf(args);

L
Linus Torvalds 已提交
777 778 779
	dp = args->dp;
	ifp = dp->i_afp;
	sf = (xfs_attr_shortform_t *)ifp->if_u1.if_data;
780
	size = be16_to_cpu(sf->hdr.totsize);
L
Linus Torvalds 已提交
781 782 783 784 785 786
	tmpbuffer = kmem_alloc(size, KM_SLEEP);
	ASSERT(tmpbuffer != NULL);
	memcpy(tmpbuffer, ifp->if_u1.if_data, size);
	sf = (xfs_attr_shortform_t *)tmpbuffer;

	xfs_idata_realloc(dp, -size, XFS_ATTR_FORK);
787 788
	xfs_bmap_local_to_extents_empty(dp, XFS_ATTR_FORK);

L
Linus Torvalds 已提交
789 790 791 792 793 794 795
	bp = NULL;
	error = xfs_da_grow_inode(args, &blkno);
	if (error) {
		/*
		 * If we hit an IO error middle of the transaction inside
		 * grow_inode(), we may have inconsistent data. Bail out.
		 */
D
Dave Chinner 已提交
796
		if (error == -EIO)
L
Linus Torvalds 已提交
797 798 799 800 801 802 803
			goto out;
		xfs_idata_realloc(dp, size, XFS_ATTR_FORK);	/* try to put */
		memcpy(ifp->if_u1.if_data, tmpbuffer, size);	/* it back */
		goto out;
	}

	ASSERT(blkno == 0);
D
Dave Chinner 已提交
804
	error = xfs_attr3_leaf_create(args, blkno, &bp);
L
Linus Torvalds 已提交
805 806 807 808 809 810 811 812 813 814 815 816
	if (error) {
		error = xfs_da_shrink_inode(args, 0, bp);
		bp = NULL;
		if (error)
			goto out;
		xfs_idata_realloc(dp, size, XFS_ATTR_FORK);	/* try to put */
		memcpy(ifp->if_u1.if_data, tmpbuffer, size);	/* it back */
		goto out;
	}

	memset((char *)&nargs, 0, sizeof(nargs));
	nargs.dp = dp;
817
	nargs.geo = args->geo;
L
Linus Torvalds 已提交
818
	nargs.firstblock = args->firstblock;
819
	nargs.dfops = args->dfops;
L
Linus Torvalds 已提交
820 821 822
	nargs.total = args->total;
	nargs.whichfork = XFS_ATTR_FORK;
	nargs.trans = args->trans;
823
	nargs.op_flags = XFS_DA_OP_OKNOENT;
L
Linus Torvalds 已提交
824 825

	sfe = &sf->list[0];
826
	for (i = 0; i < sf->hdr.count; i++) {
827
		nargs.name = sfe->nameval;
L
Linus Torvalds 已提交
828
		nargs.namelen = sfe->namelen;
829
		nargs.value = &sfe->nameval[nargs.namelen];
830
		nargs.valuelen = sfe->valuelen;
831
		nargs.hashval = xfs_da_hashname(sfe->nameval,
L
Linus Torvalds 已提交
832
						sfe->namelen);
833
		nargs.flags = XFS_ATTR_NSP_ONDISK_TO_ARGS(sfe->flags);
D
Dave Chinner 已提交
834
		error = xfs_attr3_leaf_lookup_int(bp, &nargs); /* set a->index */
D
Dave Chinner 已提交
835
		ASSERT(error == -ENOATTR);
D
Dave Chinner 已提交
836
		error = xfs_attr3_leaf_add(bp, &nargs);
D
Dave Chinner 已提交
837
		ASSERT(error != -ENOSPC);
L
Linus Torvalds 已提交
838 839 840 841 842
		if (error)
			goto out;
		sfe = XFS_ATTR_SF_NEXTENTRY(sfe);
	}
	error = 0;
843
	*leaf_bp = bp;
L
Linus Torvalds 已提交
844
out:
845
	kmem_free(tmpbuffer);
E
Eric Sandeen 已提交
846
	return error;
L
Linus Torvalds 已提交
847 848 849 850 851 852 853
}

/*
 * Check a leaf attribute block to see if all the entries would fit into
 * a shortform attribute list.
 */
int
854
xfs_attr_shortform_allfit(
855 856
	struct xfs_buf		*bp,
	struct xfs_inode	*dp)
L
Linus Torvalds 已提交
857
{
858 859
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr_leaf_entry *entry;
L
Linus Torvalds 已提交
860
	xfs_attr_leaf_name_local_t *name_loc;
861 862 863
	struct xfs_attr3_icleaf_hdr leafhdr;
	int			bytes;
	int			i;
864
	struct xfs_mount	*mp = bp->b_target->bt_mount;
L
Linus Torvalds 已提交
865

866
	leaf = bp->b_addr;
867
	xfs_attr3_leaf_hdr_from_disk(mp->m_attr_geo, &leafhdr, leaf);
868
	entry = xfs_attr3_leaf_entryp(leaf);
L
Linus Torvalds 已提交
869 870

	bytes = sizeof(struct xfs_attr_sf_hdr);
871
	for (i = 0; i < leafhdr.count; entry++, i++) {
L
Linus Torvalds 已提交
872 873 874
		if (entry->flags & XFS_ATTR_INCOMPLETE)
			continue;		/* don't copy partial entries */
		if (!(entry->flags & XFS_ATTR_LOCAL))
E
Eric Sandeen 已提交
875
			return 0;
D
Dave Chinner 已提交
876
		name_loc = xfs_attr3_leaf_name_local(leaf, i);
L
Linus Torvalds 已提交
877
		if (name_loc->namelen >= XFS_ATTR_SF_ENTSIZE_MAX)
E
Eric Sandeen 已提交
878
			return 0;
879
		if (be16_to_cpu(name_loc->valuelen) >= XFS_ATTR_SF_ENTSIZE_MAX)
E
Eric Sandeen 已提交
880
			return 0;
881
		bytes += sizeof(struct xfs_attr_sf_entry) - 1
L
Linus Torvalds 已提交
882
				+ name_loc->namelen
883
				+ be16_to_cpu(name_loc->valuelen);
L
Linus Torvalds 已提交
884
	}
885
	if ((dp->i_mount->m_flags & XFS_MOUNT_ATTR2) &&
886
	    (dp->i_d.di_format != XFS_DINODE_FMT_BTREE) &&
887
	    (bytes == sizeof(struct xfs_attr_sf_hdr)))
888 889
		return -1;
	return xfs_attr_shortform_bytesfit(dp, bytes);
L
Linus Torvalds 已提交
890 891
}

892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965
/* Verify the consistency of an inline attribute fork. */
xfs_failaddr_t
xfs_attr_shortform_verify(
	struct xfs_inode		*ip)
{
	struct xfs_attr_shortform	*sfp;
	struct xfs_attr_sf_entry	*sfep;
	struct xfs_attr_sf_entry	*next_sfep;
	char				*endp;
	struct xfs_ifork		*ifp;
	int				i;
	int				size;

	ASSERT(ip->i_d.di_aformat == XFS_DINODE_FMT_LOCAL);
	ifp = XFS_IFORK_PTR(ip, XFS_ATTR_FORK);
	sfp = (struct xfs_attr_shortform *)ifp->if_u1.if_data;
	size = ifp->if_bytes;

	/*
	 * Give up if the attribute is way too short.
	 */
	if (size < sizeof(struct xfs_attr_sf_hdr))
		return __this_address;

	endp = (char *)sfp + size;

	/* Check all reported entries */
	sfep = &sfp->list[0];
	for (i = 0; i < sfp->hdr.count; i++) {
		/*
		 * struct xfs_attr_sf_entry has a variable length.
		 * Check the fixed-offset parts of the structure are
		 * within the data buffer.
		 */
		if (((char *)sfep + sizeof(*sfep)) >= endp)
			return __this_address;

		/* Don't allow names with known bad length. */
		if (sfep->namelen == 0)
			return __this_address;

		/*
		 * Check that the variable-length part of the structure is
		 * within the data buffer.  The next entry starts after the
		 * name component, so nextentry is an acceptable test.
		 */
		next_sfep = XFS_ATTR_SF_NEXTENTRY(sfep);
		if ((char *)next_sfep > endp)
			return __this_address;

		/*
		 * Check for unknown flags.  Short form doesn't support
		 * the incomplete or local bits, so we can use the namespace
		 * mask here.
		 */
		if (sfep->flags & ~XFS_ATTR_NSP_ONDISK_MASK)
			return __this_address;

		/*
		 * Check for invalid namespace combinations.  We only allow
		 * one namespace flag per xattr, so we can just count the
		 * bits (i.e. hweight) here.
		 */
		if (hweight8(sfep->flags & XFS_ATTR_NSP_ONDISK_MASK) > 1)
			return __this_address;

		sfep = next_sfep;
	}
	if ((void *)sfep != (void *)endp)
		return __this_address;

	return NULL;
}

L
Linus Torvalds 已提交
966 967 968 969
/*
 * Convert a leaf attribute list to shortform attribute list
 */
int
D
Dave Chinner 已提交
970 971 972 973
xfs_attr3_leaf_to_shortform(
	struct xfs_buf		*bp,
	struct xfs_da_args	*args,
	int			forkoff)
L
Linus Torvalds 已提交
974
{
D
Dave Chinner 已提交
975 976 977 978 979 980 981 982 983
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_name_local *name_loc;
	struct xfs_da_args	nargs;
	struct xfs_inode	*dp = args->dp;
	char			*tmpbuffer;
	int			error;
	int			i;
L
Linus Torvalds 已提交
984

985 986
	trace_xfs_attr_leaf_to_sf(args);

987
	tmpbuffer = kmem_alloc(args->geo->blksize, KM_SLEEP);
D
Dave Chinner 已提交
988
	if (!tmpbuffer)
D
Dave Chinner 已提交
989
		return -ENOMEM;
L
Linus Torvalds 已提交
990

991
	memcpy(tmpbuffer, bp->b_addr, args->geo->blksize);
D
Dave Chinner 已提交
992

L
Linus Torvalds 已提交
993
	leaf = (xfs_attr_leafblock_t *)tmpbuffer;
994
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
995 996 997
	entry = xfs_attr3_leaf_entryp(leaf);

	/* XXX (dgc): buffer is about to be marked stale - why zero it? */
998
	memset(bp->b_addr, 0, args->geo->blksize);
L
Linus Torvalds 已提交
999 1000 1001 1002 1003 1004 1005

	/*
	 * Clean out the prior contents of the attribute list.
	 */
	error = xfs_da_shrink_inode(args, 0, bp);
	if (error)
		goto out;
1006 1007

	if (forkoff == -1) {
1008
		ASSERT(dp->i_mount->m_flags & XFS_MOUNT_ATTR2);
1009
		ASSERT(dp->i_d.di_format != XFS_DINODE_FMT_BTREE);
1010
		xfs_attr_fork_remove(dp, args->trans);
L
Linus Torvalds 已提交
1011
		goto out;
1012 1013 1014
	}

	xfs_attr_shortform_create(args);
L
Linus Torvalds 已提交
1015 1016 1017 1018 1019

	/*
	 * Copy the attributes
	 */
	memset((char *)&nargs, 0, sizeof(nargs));
1020
	nargs.geo = args->geo;
L
Linus Torvalds 已提交
1021 1022
	nargs.dp = dp;
	nargs.firstblock = args->firstblock;
1023
	nargs.dfops = args->dfops;
L
Linus Torvalds 已提交
1024 1025 1026
	nargs.total = args->total;
	nargs.whichfork = XFS_ATTR_FORK;
	nargs.trans = args->trans;
1027
	nargs.op_flags = XFS_DA_OP_OKNOENT;
D
Dave Chinner 已提交
1028 1029

	for (i = 0; i < ichdr.count; entry++, i++) {
L
Linus Torvalds 已提交
1030 1031 1032 1033 1034
		if (entry->flags & XFS_ATTR_INCOMPLETE)
			continue;	/* don't copy partial entries */
		if (!entry->nameidx)
			continue;
		ASSERT(entry->flags & XFS_ATTR_LOCAL);
D
Dave Chinner 已提交
1035
		name_loc = xfs_attr3_leaf_name_local(leaf, i);
1036
		nargs.name = name_loc->nameval;
L
Linus Torvalds 已提交
1037
		nargs.namelen = name_loc->namelen;
1038
		nargs.value = &name_loc->nameval[nargs.namelen];
1039
		nargs.valuelen = be16_to_cpu(name_loc->valuelen);
1040
		nargs.hashval = be32_to_cpu(entry->hashval);
1041
		nargs.flags = XFS_ATTR_NSP_ONDISK_TO_ARGS(entry->flags);
1042
		xfs_attr_shortform_add(&nargs, forkoff);
L
Linus Torvalds 已提交
1043 1044 1045 1046
	}
	error = 0;

out:
1047
	kmem_free(tmpbuffer);
D
Dave Chinner 已提交
1048
	return error;
L
Linus Torvalds 已提交
1049 1050 1051 1052 1053 1054
}

/*
 * Convert from using a single leaf to a root node and a leaf.
 */
int
D
Dave Chinner 已提交
1055 1056
xfs_attr3_leaf_to_node(
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
1057
{
D
Dave Chinner 已提交
1058 1059 1060
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr icleafhdr;
	struct xfs_attr_leaf_entry *entries;
1061
	struct xfs_da_node_entry *btree;
D
Dave Chinner 已提交
1062 1063 1064 1065 1066 1067 1068 1069
	struct xfs_da3_icnode_hdr icnodehdr;
	struct xfs_da_intnode	*node;
	struct xfs_inode	*dp = args->dp;
	struct xfs_mount	*mp = dp->i_mount;
	struct xfs_buf		*bp1 = NULL;
	struct xfs_buf		*bp2 = NULL;
	xfs_dablk_t		blkno;
	int			error;
L
Linus Torvalds 已提交
1070

1071 1072
	trace_xfs_attr_leaf_to_node(args);

L
Linus Torvalds 已提交
1073 1074 1075
	error = xfs_da_grow_inode(args, &blkno);
	if (error)
		goto out;
D
Dave Chinner 已提交
1076
	error = xfs_attr3_leaf_read(args->trans, dp, 0, -1, &bp1);
L
Linus Torvalds 已提交
1077 1078
	if (error)
		goto out;
1079

D
Dave Chinner 已提交
1080
	error = xfs_da_get_buf(args->trans, dp, blkno, -1, &bp2, XFS_ATTR_FORK);
L
Linus Torvalds 已提交
1081 1082
	if (error)
		goto out;
D
Dave Chinner 已提交
1083 1084

	/* copy leaf to new buffer, update identifiers */
1085
	xfs_trans_buf_set_type(args->trans, bp2, XFS_BLFT_ATTR_LEAF_BUF);
1086
	bp2->b_ops = bp1->b_ops;
1087
	memcpy(bp2->b_addr, bp1->b_addr, args->geo->blksize);
D
Dave Chinner 已提交
1088 1089 1090 1091
	if (xfs_sb_version_hascrc(&mp->m_sb)) {
		struct xfs_da3_blkinfo *hdr3 = bp2->b_addr;
		hdr3->blkno = cpu_to_be64(bp2->b_bn);
	}
1092
	xfs_trans_log_buf(args->trans, bp2, 0, args->geo->blksize - 1);
L
Linus Torvalds 已提交
1093 1094 1095 1096

	/*
	 * Set up the new root node.
	 */
1097
	error = xfs_da3_node_create(args, 0, 1, &bp1, XFS_ATTR_FORK);
L
Linus Torvalds 已提交
1098 1099
	if (error)
		goto out;
1100
	node = bp1->b_addr;
1101
	dp->d_ops->node_hdr_from_disk(&icnodehdr, node);
D
Dave Chinner 已提交
1102
	btree = dp->d_ops->node_tree_p(node);
D
Dave Chinner 已提交
1103

1104
	leaf = bp2->b_addr;
1105
	xfs_attr3_leaf_hdr_from_disk(args->geo, &icleafhdr, leaf);
D
Dave Chinner 已提交
1106 1107
	entries = xfs_attr3_leaf_entryp(leaf);

L
Linus Torvalds 已提交
1108
	/* both on-disk, don't endian-flip twice */
D
Dave Chinner 已提交
1109
	btree[0].hashval = entries[icleafhdr.count - 1].hashval;
1110
	btree[0].before = cpu_to_be32(blkno);
D
Dave Chinner 已提交
1111
	icnodehdr.count = 1;
1112
	dp->d_ops->node_hdr_to_disk(node, &icnodehdr);
1113
	xfs_trans_log_buf(args->trans, bp1, 0, args->geo->blksize - 1);
L
Linus Torvalds 已提交
1114 1115
	error = 0;
out:
D
Dave Chinner 已提交
1116
	return error;
L
Linus Torvalds 已提交
1117 1118 1119 1120 1121 1122 1123 1124 1125 1126
}

/*========================================================================
 * Routines used for growing the Btree.
 *========================================================================*/

/*
 * Create the initial contents of a leaf attribute list
 * or a leaf in a node attribute list.
 */
1127
STATIC int
D
Dave Chinner 已提交
1128 1129 1130 1131
xfs_attr3_leaf_create(
	struct xfs_da_args	*args,
	xfs_dablk_t		blkno,
	struct xfs_buf		**bpp)
L
Linus Torvalds 已提交
1132
{
D
Dave Chinner 已提交
1133 1134 1135 1136 1137 1138
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_inode	*dp = args->dp;
	struct xfs_mount	*mp = dp->i_mount;
	struct xfs_buf		*bp;
	int			error;
L
Linus Torvalds 已提交
1139

1140 1141
	trace_xfs_attr_leaf_create(args);

L
Linus Torvalds 已提交
1142 1143 1144
	error = xfs_da_get_buf(args->trans, args->dp, blkno, -1, &bp,
					    XFS_ATTR_FORK);
	if (error)
D
Dave Chinner 已提交
1145 1146
		return error;
	bp->b_ops = &xfs_attr3_leaf_buf_ops;
1147
	xfs_trans_buf_set_type(args->trans, bp, XFS_BLFT_ATTR_LEAF_BUF);
1148
	leaf = bp->b_addr;
1149
	memset(leaf, 0, args->geo->blksize);
D
Dave Chinner 已提交
1150 1151

	memset(&ichdr, 0, sizeof(ichdr));
1152
	ichdr.firstused = args->geo->blksize;
D
Dave Chinner 已提交
1153 1154 1155 1156 1157 1158 1159 1160

	if (xfs_sb_version_hascrc(&mp->m_sb)) {
		struct xfs_da3_blkinfo *hdr3 = bp->b_addr;

		ichdr.magic = XFS_ATTR3_LEAF_MAGIC;

		hdr3->blkno = cpu_to_be64(bp->b_bn);
		hdr3->owner = cpu_to_be64(dp->i_ino);
1161
		uuid_copy(&hdr3->uuid, &mp->m_sb.sb_meta_uuid);
L
Linus Torvalds 已提交
1162

D
Dave Chinner 已提交
1163 1164 1165 1166 1167 1168
		ichdr.freemap[0].base = sizeof(struct xfs_attr3_leaf_hdr);
	} else {
		ichdr.magic = XFS_ATTR_LEAF_MAGIC;
		ichdr.freemap[0].base = sizeof(struct xfs_attr_leaf_hdr);
	}
	ichdr.freemap[0].size = ichdr.firstused - ichdr.freemap[0].base;
L
Linus Torvalds 已提交
1169

1170
	xfs_attr3_leaf_hdr_to_disk(args->geo, leaf, &ichdr);
1171
	xfs_trans_log_buf(args->trans, bp, 0, args->geo->blksize - 1);
L
Linus Torvalds 已提交
1172 1173

	*bpp = bp;
D
Dave Chinner 已提交
1174
	return 0;
L
Linus Torvalds 已提交
1175 1176 1177 1178 1179 1180
}

/*
 * Split the leaf node, rebalance, then add the new entry.
 */
int
D
Dave Chinner 已提交
1181 1182 1183 1184
xfs_attr3_leaf_split(
	struct xfs_da_state	*state,
	struct xfs_da_state_blk	*oldblk,
	struct xfs_da_state_blk	*newblk)
L
Linus Torvalds 已提交
1185 1186 1187 1188
{
	xfs_dablk_t blkno;
	int error;

1189 1190
	trace_xfs_attr_leaf_split(state->args);

L
Linus Torvalds 已提交
1191 1192 1193 1194 1195 1196
	/*
	 * Allocate space for a new leaf node.
	 */
	ASSERT(oldblk->magic == XFS_ATTR_LEAF_MAGIC);
	error = xfs_da_grow_inode(state->args, &blkno);
	if (error)
E
Eric Sandeen 已提交
1197
		return error;
D
Dave Chinner 已提交
1198
	error = xfs_attr3_leaf_create(state->args, blkno, &newblk->bp);
L
Linus Torvalds 已提交
1199
	if (error)
E
Eric Sandeen 已提交
1200
		return error;
L
Linus Torvalds 已提交
1201 1202 1203 1204 1205 1206 1207
	newblk->blkno = blkno;
	newblk->magic = XFS_ATTR_LEAF_MAGIC;

	/*
	 * Rebalance the entries across the two leaves.
	 * NOTE: rebalance() currently depends on the 2nd block being empty.
	 */
D
Dave Chinner 已提交
1208
	xfs_attr3_leaf_rebalance(state, oldblk, newblk);
1209
	error = xfs_da3_blk_link(state, oldblk, newblk);
L
Linus Torvalds 已提交
1210
	if (error)
E
Eric Sandeen 已提交
1211
		return error;
L
Linus Torvalds 已提交
1212 1213 1214 1215 1216 1217 1218 1219

	/*
	 * Save info on "old" attribute for "atomic rename" ops, leaf_add()
	 * modifies the index/blkno/rmtblk/rmtblkcnt fields to show the
	 * "new" attrs info.  Will need the "old" info to remove it later.
	 *
	 * Insert the "new" entry in the correct block.
	 */
1220 1221
	if (state->inleaf) {
		trace_xfs_attr_leaf_add_old(state->args);
D
Dave Chinner 已提交
1222
		error = xfs_attr3_leaf_add(oldblk->bp, state->args);
1223 1224
	} else {
		trace_xfs_attr_leaf_add_new(state->args);
D
Dave Chinner 已提交
1225
		error = xfs_attr3_leaf_add(newblk->bp, state->args);
1226
	}
L
Linus Torvalds 已提交
1227 1228 1229 1230 1231 1232

	/*
	 * Update last hashval in each block since we added the name.
	 */
	oldblk->hashval = xfs_attr_leaf_lasthash(oldblk->bp, NULL);
	newblk->hashval = xfs_attr_leaf_lasthash(newblk->bp, NULL);
E
Eric Sandeen 已提交
1233
	return error;
L
Linus Torvalds 已提交
1234 1235 1236 1237 1238 1239
}

/*
 * Add a name to the leaf attribute list structure.
 */
int
D
Dave Chinner 已提交
1240
xfs_attr3_leaf_add(
1241 1242
	struct xfs_buf		*bp,
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
1243
{
D
Dave Chinner 已提交
1244 1245 1246 1247 1248 1249 1250
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	int			tablesize;
	int			entsize;
	int			sum;
	int			tmp;
	int			i;
L
Linus Torvalds 已提交
1251

1252 1253
	trace_xfs_attr_leaf_add(args);

1254
	leaf = bp->b_addr;
1255
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
1256
	ASSERT(args->index >= 0 && args->index <= ichdr.count);
1257
	entsize = xfs_attr_leaf_newentsize(args, NULL);
L
Linus Torvalds 已提交
1258 1259 1260 1261 1262

	/*
	 * Search through freemap for first-fit on new name length.
	 * (may need to figure in size of entry struct too)
	 */
D
Dave Chinner 已提交
1263 1264 1265 1266 1267
	tablesize = (ichdr.count + 1) * sizeof(xfs_attr_leaf_entry_t)
					+ xfs_attr3_leaf_hdr_size(leaf);
	for (sum = 0, i = XFS_ATTR_LEAF_MAPSIZE - 1; i >= 0; i--) {
		if (tablesize > ichdr.firstused) {
			sum += ichdr.freemap[i].size;
L
Linus Torvalds 已提交
1268 1269
			continue;
		}
D
Dave Chinner 已提交
1270
		if (!ichdr.freemap[i].size)
L
Linus Torvalds 已提交
1271 1272
			continue;	/* no space in this map */
		tmp = entsize;
D
Dave Chinner 已提交
1273
		if (ichdr.freemap[i].base < ichdr.firstused)
L
Linus Torvalds 已提交
1274
			tmp += sizeof(xfs_attr_leaf_entry_t);
D
Dave Chinner 已提交
1275 1276 1277
		if (ichdr.freemap[i].size >= tmp) {
			tmp = xfs_attr3_leaf_add_work(bp, &ichdr, args, i);
			goto out_log_hdr;
L
Linus Torvalds 已提交
1278
		}
D
Dave Chinner 已提交
1279
		sum += ichdr.freemap[i].size;
L
Linus Torvalds 已提交
1280 1281 1282 1283 1284 1285 1286
	}

	/*
	 * If there are no holes in the address space of the block,
	 * and we don't have enough freespace, then compaction will do us
	 * no good and we should just give up.
	 */
D
Dave Chinner 已提交
1287
	if (!ichdr.holes && sum < entsize)
D
Dave Chinner 已提交
1288
		return -ENOSPC;
L
Linus Torvalds 已提交
1289 1290 1291 1292 1293

	/*
	 * Compact the entries to coalesce free space.
	 * This may change the hdr->count via dropping INCOMPLETE entries.
	 */
D
Dave Chinner 已提交
1294
	xfs_attr3_leaf_compact(args, &ichdr, bp);
L
Linus Torvalds 已提交
1295 1296 1297 1298 1299

	/*
	 * After compaction, the block is guaranteed to have only one
	 * free region, in freemap[0].  If it is not big enough, give up.
	 */
D
Dave Chinner 已提交
1300
	if (ichdr.freemap[0].size < (entsize + sizeof(xfs_attr_leaf_entry_t))) {
D
Dave Chinner 已提交
1301
		tmp = -ENOSPC;
D
Dave Chinner 已提交
1302 1303 1304 1305
		goto out_log_hdr;
	}

	tmp = xfs_attr3_leaf_add_work(bp, &ichdr, args, 0);
L
Linus Torvalds 已提交
1306

D
Dave Chinner 已提交
1307
out_log_hdr:
1308
	xfs_attr3_leaf_hdr_to_disk(args->geo, leaf, &ichdr);
D
Dave Chinner 已提交
1309 1310 1311 1312
	xfs_trans_log_buf(args->trans, bp,
		XFS_DA_LOGRANGE(leaf, &leaf->hdr,
				xfs_attr3_leaf_hdr_size(leaf)));
	return tmp;
L
Linus Torvalds 已提交
1313 1314 1315 1316 1317 1318
}

/*
 * Add a name to a leaf attribute list structure.
 */
STATIC int
D
Dave Chinner 已提交
1319 1320 1321 1322 1323
xfs_attr3_leaf_add_work(
	struct xfs_buf		*bp,
	struct xfs_attr3_icleaf_hdr *ichdr,
	struct xfs_da_args	*args,
	int			mapindex)
L
Linus Torvalds 已提交
1324
{
D
Dave Chinner 已提交
1325 1326 1327 1328 1329 1330 1331
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_name_local *name_loc;
	struct xfs_attr_leaf_name_remote *name_rmt;
	struct xfs_mount	*mp;
	int			tmp;
	int			i;
L
Linus Torvalds 已提交
1332

1333 1334
	trace_xfs_attr_leaf_add_work(args);

1335
	leaf = bp->b_addr;
D
Dave Chinner 已提交
1336 1337
	ASSERT(mapindex >= 0 && mapindex < XFS_ATTR_LEAF_MAPSIZE);
	ASSERT(args->index >= 0 && args->index <= ichdr->count);
L
Linus Torvalds 已提交
1338 1339 1340 1341

	/*
	 * Force open some space in the entry array and fill it in.
	 */
D
Dave Chinner 已提交
1342 1343 1344
	entry = &xfs_attr3_leaf_entryp(leaf)[args->index];
	if (args->index < ichdr->count) {
		tmp  = ichdr->count - args->index;
L
Linus Torvalds 已提交
1345
		tmp *= sizeof(xfs_attr_leaf_entry_t);
D
Dave Chinner 已提交
1346
		memmove(entry + 1, entry, tmp);
1347
		xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
1348 1349
		    XFS_DA_LOGRANGE(leaf, entry, tmp + sizeof(*entry)));
	}
D
Dave Chinner 已提交
1350
	ichdr->count++;
L
Linus Torvalds 已提交
1351 1352 1353 1354 1355

	/*
	 * Allocate space for the new string (at the end of the run).
	 */
	mp = args->trans->t_mountp;
1356
	ASSERT(ichdr->freemap[mapindex].base < args->geo->blksize);
D
Dave Chinner 已提交
1357 1358
	ASSERT((ichdr->freemap[mapindex].base & 0x3) == 0);
	ASSERT(ichdr->freemap[mapindex].size >=
1359
		xfs_attr_leaf_newentsize(args, NULL));
1360
	ASSERT(ichdr->freemap[mapindex].size < args->geo->blksize);
D
Dave Chinner 已提交
1361 1362
	ASSERT((ichdr->freemap[mapindex].size & 0x3) == 0);

1363
	ichdr->freemap[mapindex].size -= xfs_attr_leaf_newentsize(args, &tmp);
D
Dave Chinner 已提交
1364 1365 1366

	entry->nameidx = cpu_to_be16(ichdr->freemap[mapindex].base +
				     ichdr->freemap[mapindex].size);
1367
	entry->hashval = cpu_to_be32(args->hashval);
L
Linus Torvalds 已提交
1368
	entry->flags = tmp ? XFS_ATTR_LOCAL : 0;
1369
	entry->flags |= XFS_ATTR_NSP_ARGS_TO_ONDISK(args->flags);
1370
	if (args->op_flags & XFS_DA_OP_RENAME) {
L
Linus Torvalds 已提交
1371 1372 1373 1374 1375 1376
		entry->flags |= XFS_ATTR_INCOMPLETE;
		if ((args->blkno2 == args->blkno) &&
		    (args->index2 <= args->index)) {
			args->index2++;
		}
	}
1377
	xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
1378
			  XFS_DA_LOGRANGE(leaf, entry, sizeof(*entry)));
1379 1380
	ASSERT((args->index == 0) ||
	       (be32_to_cpu(entry->hashval) >= be32_to_cpu((entry-1)->hashval)));
D
Dave Chinner 已提交
1381
	ASSERT((args->index == ichdr->count - 1) ||
1382
	       (be32_to_cpu(entry->hashval) <= be32_to_cpu((entry+1)->hashval)));
L
Linus Torvalds 已提交
1383 1384 1385 1386 1387 1388 1389 1390 1391

	/*
	 * For "remote" attribute values, simply note that we need to
	 * allocate space for the "remote" value.  We can't actually
	 * allocate the extents in this transaction, and we can't decide
	 * which blocks they should be as we might allocate more blocks
	 * as part of this transaction (a split operation for example).
	 */
	if (entry->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
1392
		name_loc = xfs_attr3_leaf_name_local(leaf, args->index);
L
Linus Torvalds 已提交
1393
		name_loc->namelen = args->namelen;
1394
		name_loc->valuelen = cpu_to_be16(args->valuelen);
L
Linus Torvalds 已提交
1395 1396
		memcpy((char *)name_loc->nameval, args->name, args->namelen);
		memcpy((char *)&name_loc->nameval[args->namelen], args->value,
1397
				   be16_to_cpu(name_loc->valuelen));
L
Linus Torvalds 已提交
1398
	} else {
D
Dave Chinner 已提交
1399
		name_rmt = xfs_attr3_leaf_name_remote(leaf, args->index);
L
Linus Torvalds 已提交
1400 1401 1402 1403 1404 1405 1406
		name_rmt->namelen = args->namelen;
		memcpy((char *)name_rmt->name, args->name, args->namelen);
		entry->flags |= XFS_ATTR_INCOMPLETE;
		/* just in case */
		name_rmt->valuelen = 0;
		name_rmt->valueblk = 0;
		args->rmtblkno = 1;
D
Dave Chinner 已提交
1407
		args->rmtblkcnt = xfs_attr3_rmt_blocks(mp, args->valuelen);
1408
		args->rmtvaluelen = args->valuelen;
L
Linus Torvalds 已提交
1409
	}
1410
	xfs_trans_log_buf(args->trans, bp,
D
Dave Chinner 已提交
1411
	     XFS_DA_LOGRANGE(leaf, xfs_attr3_leaf_name(leaf, args->index),
L
Linus Torvalds 已提交
1412 1413 1414 1415 1416
				   xfs_attr_leaf_entsize(leaf, args->index)));

	/*
	 * Update the control info for this leaf node
	 */
D
Dave Chinner 已提交
1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428
	if (be16_to_cpu(entry->nameidx) < ichdr->firstused)
		ichdr->firstused = be16_to_cpu(entry->nameidx);

	ASSERT(ichdr->firstused >= ichdr->count * sizeof(xfs_attr_leaf_entry_t)
					+ xfs_attr3_leaf_hdr_size(leaf));
	tmp = (ichdr->count - 1) * sizeof(xfs_attr_leaf_entry_t)
					+ xfs_attr3_leaf_hdr_size(leaf);

	for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
		if (ichdr->freemap[i].base == tmp) {
			ichdr->freemap[i].base += sizeof(xfs_attr_leaf_entry_t);
			ichdr->freemap[i].size -= sizeof(xfs_attr_leaf_entry_t);
L
Linus Torvalds 已提交
1429 1430
		}
	}
D
Dave Chinner 已提交
1431 1432
	ichdr->usedbytes += xfs_attr_leaf_entsize(leaf, args->index);
	return 0;
L
Linus Torvalds 已提交
1433 1434 1435 1436 1437 1438
}

/*
 * Garbage collect a leaf attribute list block by copying it to a new buffer.
 */
STATIC void
D
Dave Chinner 已提交
1439
xfs_attr3_leaf_compact(
1440
	struct xfs_da_args	*args,
1441
	struct xfs_attr3_icleaf_hdr *ichdr_dst,
1442
	struct xfs_buf		*bp)
L
Linus Torvalds 已提交
1443
{
1444 1445 1446
	struct xfs_attr_leafblock *leaf_src;
	struct xfs_attr_leafblock *leaf_dst;
	struct xfs_attr3_icleaf_hdr ichdr_src;
1447 1448 1449 1450
	struct xfs_trans	*trans = args->trans;
	char			*tmpbuffer;

	trace_xfs_attr_leaf_compact(args);
L
Linus Torvalds 已提交
1451

1452 1453 1454
	tmpbuffer = kmem_alloc(args->geo->blksize, KM_SLEEP);
	memcpy(tmpbuffer, bp->b_addr, args->geo->blksize);
	memset(bp->b_addr, 0, args->geo->blksize);
1455 1456
	leaf_src = (xfs_attr_leafblock_t *)tmpbuffer;
	leaf_dst = bp->b_addr;
L
Linus Torvalds 已提交
1457 1458

	/*
1459 1460 1461
	 * Copy the on-disk header back into the destination buffer to ensure
	 * all the information in the header that is not part of the incore
	 * header structure is preserved.
L
Linus Torvalds 已提交
1462
	 */
1463 1464 1465 1466
	memcpy(bp->b_addr, tmpbuffer, xfs_attr3_leaf_hdr_size(leaf_src));

	/* Initialise the incore headers */
	ichdr_src = *ichdr_dst;	/* struct copy */
1467
	ichdr_dst->firstused = args->geo->blksize;
1468 1469 1470 1471 1472 1473 1474 1475
	ichdr_dst->usedbytes = 0;
	ichdr_dst->count = 0;
	ichdr_dst->holes = 0;
	ichdr_dst->freemap[0].base = xfs_attr3_leaf_hdr_size(leaf_src);
	ichdr_dst->freemap[0].size = ichdr_dst->firstused -
						ichdr_dst->freemap[0].base;

	/* write the header back to initialise the underlying buffer */
1476
	xfs_attr3_leaf_hdr_to_disk(args->geo, leaf_dst, ichdr_dst);
L
Linus Torvalds 已提交
1477 1478 1479 1480 1481

	/*
	 * Copy all entry's in the same (sorted) order,
	 * but allocate name/value pairs packed and in sequence.
	 */
1482 1483
	xfs_attr3_leaf_moveents(args, leaf_src, &ichdr_src, 0,
				leaf_dst, ichdr_dst, 0, ichdr_src.count);
D
Dave Chinner 已提交
1484 1485 1486 1487
	/*
	 * this logs the entire buffer, but the caller must write the header
	 * back to the buffer when it is finished modifying it.
	 */
1488
	xfs_trans_log_buf(trans, bp, 0, args->geo->blksize - 1);
L
Linus Torvalds 已提交
1489

1490
	kmem_free(tmpbuffer);
L
Linus Torvalds 已提交
1491 1492
}

D
Dave Chinner 已提交
1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525
/*
 * Compare two leaf blocks "order".
 * Return 0 unless leaf2 should go before leaf1.
 */
static int
xfs_attr3_leaf_order(
	struct xfs_buf	*leaf1_bp,
	struct xfs_attr3_icleaf_hdr *leaf1hdr,
	struct xfs_buf	*leaf2_bp,
	struct xfs_attr3_icleaf_hdr *leaf2hdr)
{
	struct xfs_attr_leaf_entry *entries1;
	struct xfs_attr_leaf_entry *entries2;

	entries1 = xfs_attr3_leaf_entryp(leaf1_bp->b_addr);
	entries2 = xfs_attr3_leaf_entryp(leaf2_bp->b_addr);
	if (leaf1hdr->count > 0 && leaf2hdr->count > 0 &&
	    ((be32_to_cpu(entries2[0].hashval) <
	      be32_to_cpu(entries1[0].hashval)) ||
	     (be32_to_cpu(entries2[leaf2hdr->count - 1].hashval) <
	      be32_to_cpu(entries1[leaf1hdr->count - 1].hashval)))) {
		return 1;
	}
	return 0;
}

int
xfs_attr_leaf_order(
	struct xfs_buf	*leaf1_bp,
	struct xfs_buf	*leaf2_bp)
{
	struct xfs_attr3_icleaf_hdr ichdr1;
	struct xfs_attr3_icleaf_hdr ichdr2;
1526
	struct xfs_mount *mp = leaf1_bp->b_target->bt_mount;
D
Dave Chinner 已提交
1527

1528 1529
	xfs_attr3_leaf_hdr_from_disk(mp->m_attr_geo, &ichdr1, leaf1_bp->b_addr);
	xfs_attr3_leaf_hdr_from_disk(mp->m_attr_geo, &ichdr2, leaf2_bp->b_addr);
D
Dave Chinner 已提交
1530 1531 1532
	return xfs_attr3_leaf_order(leaf1_bp, &ichdr1, leaf2_bp, &ichdr2);
}

L
Linus Torvalds 已提交
1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545
/*
 * Redistribute the attribute list entries between two leaf nodes,
 * taking into account the size of the new entry.
 *
 * NOTE: if new block is empty, then it will get the upper half of the
 * old block.  At present, all (one) callers pass in an empty second block.
 *
 * This code adjusts the args->index/blkno and args->index2/blkno2 fields
 * to match what it is doing in splitting the attribute leaf block.  Those
 * values are used in "atomic rename" operations on attributes.  Note that
 * the "new" and "old" values can end up in different blocks.
 */
STATIC void
D
Dave Chinner 已提交
1546 1547 1548 1549
xfs_attr3_leaf_rebalance(
	struct xfs_da_state	*state,
	struct xfs_da_state_blk	*blk1,
	struct xfs_da_state_blk	*blk2)
L
Linus Torvalds 已提交
1550
{
D
Dave Chinner 已提交
1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562
	struct xfs_da_args	*args;
	struct xfs_attr_leafblock *leaf1;
	struct xfs_attr_leafblock *leaf2;
	struct xfs_attr3_icleaf_hdr ichdr1;
	struct xfs_attr3_icleaf_hdr ichdr2;
	struct xfs_attr_leaf_entry *entries1;
	struct xfs_attr_leaf_entry *entries2;
	int			count;
	int			totallen;
	int			max;
	int			space;
	int			swap;
L
Linus Torvalds 已提交
1563 1564 1565 1566 1567 1568

	/*
	 * Set up environment.
	 */
	ASSERT(blk1->magic == XFS_ATTR_LEAF_MAGIC);
	ASSERT(blk2->magic == XFS_ATTR_LEAF_MAGIC);
1569 1570
	leaf1 = blk1->bp->b_addr;
	leaf2 = blk2->bp->b_addr;
1571 1572
	xfs_attr3_leaf_hdr_from_disk(state->args->geo, &ichdr1, leaf1);
	xfs_attr3_leaf_hdr_from_disk(state->args->geo, &ichdr2, leaf2);
D
Dave Chinner 已提交
1573
	ASSERT(ichdr2.count == 0);
L
Linus Torvalds 已提交
1574 1575
	args = state->args;

1576 1577
	trace_xfs_attr_leaf_rebalance(args);

L
Linus Torvalds 已提交
1578 1579 1580 1581 1582 1583 1584
	/*
	 * Check ordering of blocks, reverse if it makes things simpler.
	 *
	 * NOTE: Given that all (current) callers pass in an empty
	 * second block, this code should never set "swap".
	 */
	swap = 0;
D
Dave Chinner 已提交
1585 1586 1587 1588
	if (xfs_attr3_leaf_order(blk1->bp, &ichdr1, blk2->bp, &ichdr2)) {
		struct xfs_da_state_blk	*tmp_blk;
		struct xfs_attr3_icleaf_hdr tmp_ichdr;

L
Linus Torvalds 已提交
1589 1590 1591
		tmp_blk = blk1;
		blk1 = blk2;
		blk2 = tmp_blk;
D
Dave Chinner 已提交
1592 1593 1594 1595 1596 1597

		/* struct copies to swap them rather than reconverting */
		tmp_ichdr = ichdr1;
		ichdr1 = ichdr2;
		ichdr2 = tmp_ichdr;

1598 1599
		leaf1 = blk1->bp->b_addr;
		leaf2 = blk2->bp->b_addr;
L
Linus Torvalds 已提交
1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610
		swap = 1;
	}

	/*
	 * Examine entries until we reduce the absolute difference in
	 * byte usage between the two blocks to a minimum.  Then get
	 * the direction to copy and the number of elements to move.
	 *
	 * "inleaf" is true if the new entry should be inserted into blk1.
	 * If "swap" is also true, then reverse the sense of "inleaf".
	 */
D
Dave Chinner 已提交
1611 1612 1613
	state->inleaf = xfs_attr3_leaf_figure_balance(state, blk1, &ichdr1,
						      blk2, &ichdr2,
						      &count, &totallen);
L
Linus Torvalds 已提交
1614 1615 1616 1617 1618 1619
	if (swap)
		state->inleaf = !state->inleaf;

	/*
	 * Move any entries required from leaf to leaf:
	 */
D
Dave Chinner 已提交
1620
	if (count < ichdr1.count) {
L
Linus Torvalds 已提交
1621 1622 1623 1624
		/*
		 * Figure the total bytes to be added to the destination leaf.
		 */
		/* number entries being moved */
D
Dave Chinner 已提交
1625 1626
		count = ichdr1.count - count;
		space  = ichdr1.usedbytes - totallen;
L
Linus Torvalds 已提交
1627 1628 1629 1630 1631
		space += count * sizeof(xfs_attr_leaf_entry_t);

		/*
		 * leaf2 is the destination, compact it if it looks tight.
		 */
D
Dave Chinner 已提交
1632 1633
		max  = ichdr2.firstused - xfs_attr3_leaf_hdr_size(leaf1);
		max -= ichdr2.count * sizeof(xfs_attr_leaf_entry_t);
1634
		if (space > max)
D
Dave Chinner 已提交
1635
			xfs_attr3_leaf_compact(args, &ichdr2, blk2->bp);
L
Linus Torvalds 已提交
1636 1637 1638 1639

		/*
		 * Move high entries from leaf1 to low end of leaf2.
		 */
1640 1641
		xfs_attr3_leaf_moveents(args, leaf1, &ichdr1,
				ichdr1.count - count, leaf2, &ichdr2, 0, count);
L
Linus Torvalds 已提交
1642

D
Dave Chinner 已提交
1643
	} else if (count > ichdr1.count) {
L
Linus Torvalds 已提交
1644 1645 1646 1647
		/*
		 * I assert that since all callers pass in an empty
		 * second buffer, this code should never execute.
		 */
1648
		ASSERT(0);
L
Linus Torvalds 已提交
1649 1650 1651 1652 1653

		/*
		 * Figure the total bytes to be added to the destination leaf.
		 */
		/* number entries being moved */
D
Dave Chinner 已提交
1654 1655
		count -= ichdr1.count;
		space  = totallen - ichdr1.usedbytes;
L
Linus Torvalds 已提交
1656 1657 1658 1659 1660
		space += count * sizeof(xfs_attr_leaf_entry_t);

		/*
		 * leaf1 is the destination, compact it if it looks tight.
		 */
D
Dave Chinner 已提交
1661 1662
		max  = ichdr1.firstused - xfs_attr3_leaf_hdr_size(leaf1);
		max -= ichdr1.count * sizeof(xfs_attr_leaf_entry_t);
1663
		if (space > max)
D
Dave Chinner 已提交
1664
			xfs_attr3_leaf_compact(args, &ichdr1, blk1->bp);
L
Linus Torvalds 已提交
1665 1666 1667 1668

		/*
		 * Move low entries from leaf2 to high end of leaf1.
		 */
1669 1670
		xfs_attr3_leaf_moveents(args, leaf2, &ichdr2, 0, leaf1, &ichdr1,
					ichdr1.count, count);
L
Linus Torvalds 已提交
1671 1672
	}

1673 1674
	xfs_attr3_leaf_hdr_to_disk(state->args->geo, leaf1, &ichdr1);
	xfs_attr3_leaf_hdr_to_disk(state->args->geo, leaf2, &ichdr2);
1675 1676
	xfs_trans_log_buf(args->trans, blk1->bp, 0, args->geo->blksize - 1);
	xfs_trans_log_buf(args->trans, blk2->bp, 0, args->geo->blksize - 1);
D
Dave Chinner 已提交
1677

L
Linus Torvalds 已提交
1678 1679 1680
	/*
	 * Copy out last hashval in each block for B-tree code.
	 */
D
Dave Chinner 已提交
1681 1682 1683 1684
	entries1 = xfs_attr3_leaf_entryp(leaf1);
	entries2 = xfs_attr3_leaf_entryp(leaf2);
	blk1->hashval = be32_to_cpu(entries1[ichdr1.count - 1].hashval);
	blk2->hashval = be32_to_cpu(entries2[ichdr2.count - 1].hashval);
L
Linus Torvalds 已提交
1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697

	/*
	 * Adjust the expected index for insertion.
	 * NOTE: this code depends on the (current) situation that the
	 * second block was originally empty.
	 *
	 * If the insertion point moved to the 2nd block, we must adjust
	 * the index.  We must also track the entry just following the
	 * new entry for use in an "atomic rename" operation, that entry
	 * is always the "old" entry and the "new" entry is what we are
	 * inserting.  The index/blkno fields refer to the "old" entry,
	 * while the index2/blkno2 fields refer to the "new" entry.
	 */
D
Dave Chinner 已提交
1698
	if (blk1->index > ichdr1.count) {
L
Linus Torvalds 已提交
1699
		ASSERT(state->inleaf == 0);
D
Dave Chinner 已提交
1700
		blk2->index = blk1->index - ichdr1.count;
L
Linus Torvalds 已提交
1701 1702
		args->index = args->index2 = blk2->index;
		args->blkno = args->blkno2 = blk2->blkno;
D
Dave Chinner 已提交
1703
	} else if (blk1->index == ichdr1.count) {
L
Linus Torvalds 已提交
1704 1705 1706 1707 1708 1709
		if (state->inleaf) {
			args->index = blk1->index;
			args->blkno = blk1->blkno;
			args->index2 = 0;
			args->blkno2 = blk2->blkno;
		} else {
1710 1711 1712 1713 1714
			/*
			 * On a double leaf split, the original attr location
			 * is already stored in blkno2/index2, so don't
			 * overwrite it overwise we corrupt the tree.
			 */
D
Dave Chinner 已提交
1715
			blk2->index = blk1->index - ichdr1.count;
1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726
			args->index = blk2->index;
			args->blkno = blk2->blkno;
			if (!state->extravalid) {
				/*
				 * set the new attr location to match the old
				 * one and let the higher level split code
				 * decide where in the leaf to place it.
				 */
				args->index2 = blk2->index;
				args->blkno2 = blk2->blkno;
			}
L
Linus Torvalds 已提交
1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742
		}
	} else {
		ASSERT(state->inleaf == 1);
		args->index = args->index2 = blk1->index;
		args->blkno = args->blkno2 = blk1->blkno;
	}
}

/*
 * Examine entries until we reduce the absolute difference in
 * byte usage between the two blocks to a minimum.
 * GROT: Is this really necessary?  With other than a 512 byte blocksize,
 * GROT: there will always be enough room in either block for a new entry.
 * GROT: Do a double-split for this case?
 */
STATIC int
D
Dave Chinner 已提交
1743 1744 1745 1746 1747 1748 1749 1750
xfs_attr3_leaf_figure_balance(
	struct xfs_da_state		*state,
	struct xfs_da_state_blk		*blk1,
	struct xfs_attr3_icleaf_hdr	*ichdr1,
	struct xfs_da_state_blk		*blk2,
	struct xfs_attr3_icleaf_hdr	*ichdr2,
	int				*countarg,
	int				*usedbytesarg)
L
Linus Torvalds 已提交
1751
{
D
Dave Chinner 已提交
1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762
	struct xfs_attr_leafblock	*leaf1 = blk1->bp->b_addr;
	struct xfs_attr_leafblock	*leaf2 = blk2->bp->b_addr;
	struct xfs_attr_leaf_entry	*entry;
	int				count;
	int				max;
	int				index;
	int				totallen = 0;
	int				half;
	int				lastdelta;
	int				foundit = 0;
	int				tmp;
L
Linus Torvalds 已提交
1763 1764 1765 1766 1767

	/*
	 * Examine entries until we reduce the absolute difference in
	 * byte usage between the two blocks to a minimum.
	 */
D
Dave Chinner 已提交
1768 1769 1770
	max = ichdr1->count + ichdr2->count;
	half = (max + 1) * sizeof(*entry);
	half += ichdr1->usedbytes + ichdr2->usedbytes +
1771
			xfs_attr_leaf_newentsize(state->args, NULL);
L
Linus Torvalds 已提交
1772
	half /= 2;
1773
	lastdelta = state->args->geo->blksize;
D
Dave Chinner 已提交
1774
	entry = xfs_attr3_leaf_entryp(leaf1);
L
Linus Torvalds 已提交
1775 1776 1777 1778 1779 1780 1781 1782
	for (count = index = 0; count < max; entry++, index++, count++) {

#define XFS_ATTR_ABS(A)	(((A) < 0) ? -(A) : (A))
		/*
		 * The new entry is in the first block, account for it.
		 */
		if (count == blk1->index) {
			tmp = totallen + sizeof(*entry) +
1783
				xfs_attr_leaf_newentsize(state->args, NULL);
L
Linus Torvalds 已提交
1784 1785 1786 1787 1788 1789 1790 1791 1792 1793
			if (XFS_ATTR_ABS(half - tmp) > lastdelta)
				break;
			lastdelta = XFS_ATTR_ABS(half - tmp);
			totallen = tmp;
			foundit = 1;
		}

		/*
		 * Wrap around into the second block if necessary.
		 */
D
Dave Chinner 已提交
1794
		if (count == ichdr1->count) {
L
Linus Torvalds 已提交
1795
			leaf1 = leaf2;
D
Dave Chinner 已提交
1796
			entry = xfs_attr3_leaf_entryp(leaf1);
L
Linus Torvalds 已提交
1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818
			index = 0;
		}

		/*
		 * Figure out if next leaf entry would be too much.
		 */
		tmp = totallen + sizeof(*entry) + xfs_attr_leaf_entsize(leaf1,
									index);
		if (XFS_ATTR_ABS(half - tmp) > lastdelta)
			break;
		lastdelta = XFS_ATTR_ABS(half - tmp);
		totallen = tmp;
#undef XFS_ATTR_ABS
	}

	/*
	 * Calculate the number of usedbytes that will end up in lower block.
	 * If new entry not in lower block, fix up the count.
	 */
	totallen -= count * sizeof(*entry);
	if (foundit) {
		totallen -= sizeof(*entry) +
1819
				xfs_attr_leaf_newentsize(state->args, NULL);
L
Linus Torvalds 已提交
1820 1821 1822 1823
	}

	*countarg = count;
	*usedbytesarg = totallen;
D
Dave Chinner 已提交
1824
	return foundit;
L
Linus Torvalds 已提交
1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842
}

/*========================================================================
 * Routines used for shrinking the Btree.
 *========================================================================*/

/*
 * Check a leaf block and its neighbors to see if the block should be
 * collapsed into one or the other neighbor.  Always keep the block
 * with the smaller block number.
 * If the current block is over 50% full, don't try to join it, return 0.
 * If the block is empty, fill in the state structure and return 2.
 * If it can be collapsed, fill in the state structure and return 1.
 * If nothing can be done, return 0.
 *
 * GROT: allow for INCOMPLETE entries in calculation.
 */
int
D
Dave Chinner 已提交
1843 1844 1845
xfs_attr3_leaf_toosmall(
	struct xfs_da_state	*state,
	int			*action)
L
Linus Torvalds 已提交
1846
{
D
Dave Chinner 已提交
1847 1848 1849 1850 1851 1852 1853 1854 1855 1856
	struct xfs_attr_leafblock *leaf;
	struct xfs_da_state_blk	*blk;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_buf		*bp;
	xfs_dablk_t		blkno;
	int			bytes;
	int			forward;
	int			error;
	int			retval;
	int			i;
L
Linus Torvalds 已提交
1857

1858 1859
	trace_xfs_attr_leaf_toosmall(state->args);

L
Linus Torvalds 已提交
1860 1861 1862 1863 1864 1865
	/*
	 * Check for the degenerate case of the block being over 50% full.
	 * If so, it's not worth even looking to see if we might be able
	 * to coalesce with a sibling.
	 */
	blk = &state->path.blk[ state->path.active-1 ];
D
Dave Chinner 已提交
1866
	leaf = blk->bp->b_addr;
1867
	xfs_attr3_leaf_hdr_from_disk(state->args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
1868 1869 1870
	bytes = xfs_attr3_leaf_hdr_size(leaf) +
		ichdr.count * sizeof(xfs_attr_leaf_entry_t) +
		ichdr.usedbytes;
1871
	if (bytes > (state->args->geo->blksize >> 1)) {
L
Linus Torvalds 已提交
1872
		*action = 0;	/* blk over 50%, don't try to join */
E
Eric Sandeen 已提交
1873
		return 0;
L
Linus Torvalds 已提交
1874 1875 1876 1877 1878
	}

	/*
	 * Check for the degenerate case of the block being empty.
	 * If the block is empty, we'll simply delete it, no need to
1879
	 * coalesce it with a sibling block.  We choose (arbitrarily)
L
Linus Torvalds 已提交
1880 1881
	 * to merge with the forward block unless it is NULL.
	 */
D
Dave Chinner 已提交
1882
	if (ichdr.count == 0) {
L
Linus Torvalds 已提交
1883 1884 1885 1886
		/*
		 * Make altpath point to the block we want to keep and
		 * path point to the block we want to drop (this one).
		 */
D
Dave Chinner 已提交
1887
		forward = (ichdr.forw != 0);
L
Linus Torvalds 已提交
1888
		memcpy(&state->altpath, &state->path, sizeof(state->path));
1889
		error = xfs_da3_path_shift(state, &state->altpath, forward,
L
Linus Torvalds 已提交
1890 1891
						 0, &retval);
		if (error)
E
Eric Sandeen 已提交
1892
			return error;
L
Linus Torvalds 已提交
1893 1894 1895 1896 1897
		if (retval) {
			*action = 0;
		} else {
			*action = 2;
		}
D
Dave Chinner 已提交
1898
		return 0;
L
Linus Torvalds 已提交
1899 1900 1901 1902 1903 1904 1905 1906 1907 1908
	}

	/*
	 * Examine each sibling block to see if we can coalesce with
	 * at least 25% free space to spare.  We need to figure out
	 * whether to merge with the forward or the backward block.
	 * We prefer coalescing with the lower numbered sibling so as
	 * to shrink an attribute list over time.
	 */
	/* start with smaller blk num */
D
Dave Chinner 已提交
1909
	forward = ichdr.forw < ichdr.back;
L
Linus Torvalds 已提交
1910
	for (i = 0; i < 2; forward = !forward, i++) {
D
Dave Chinner 已提交
1911
		struct xfs_attr3_icleaf_hdr ichdr2;
L
Linus Torvalds 已提交
1912
		if (forward)
D
Dave Chinner 已提交
1913
			blkno = ichdr.forw;
L
Linus Torvalds 已提交
1914
		else
D
Dave Chinner 已提交
1915
			blkno = ichdr.back;
L
Linus Torvalds 已提交
1916 1917
		if (blkno == 0)
			continue;
D
Dave Chinner 已提交
1918
		error = xfs_attr3_leaf_read(state->args->trans, state->args->dp,
1919
					blkno, -1, &bp);
L
Linus Torvalds 已提交
1920
		if (error)
E
Eric Sandeen 已提交
1921
			return error;
L
Linus Torvalds 已提交
1922

1923
		xfs_attr3_leaf_hdr_from_disk(state->args->geo, &ichdr2, bp->b_addr);
D
Dave Chinner 已提交
1924

1925 1926
		bytes = state->args->geo->blksize -
			(state->args->geo->blksize >> 2) -
D
Dave Chinner 已提交
1927 1928 1929 1930 1931
			ichdr.usedbytes - ichdr2.usedbytes -
			((ichdr.count + ichdr2.count) *
					sizeof(xfs_attr_leaf_entry_t)) -
			xfs_attr3_leaf_hdr_size(leaf);

1932
		xfs_trans_brelse(state->args->trans, bp);
L
Linus Torvalds 已提交
1933 1934 1935 1936 1937
		if (bytes >= 0)
			break;	/* fits with at least 25% to spare */
	}
	if (i >= 2) {
		*action = 0;
E
Eric Sandeen 已提交
1938
		return 0;
L
Linus Torvalds 已提交
1939 1940 1941 1942 1943 1944 1945 1946
	}

	/*
	 * Make altpath point to the block we want to keep (the lower
	 * numbered block) and path point to the block we want to drop.
	 */
	memcpy(&state->altpath, &state->path, sizeof(state->path));
	if (blkno < blk->blkno) {
1947
		error = xfs_da3_path_shift(state, &state->altpath, forward,
L
Linus Torvalds 已提交
1948 1949
						 0, &retval);
	} else {
1950
		error = xfs_da3_path_shift(state, &state->path, forward,
L
Linus Torvalds 已提交
1951 1952 1953
						 0, &retval);
	}
	if (error)
E
Eric Sandeen 已提交
1954
		return error;
L
Linus Torvalds 已提交
1955 1956 1957 1958 1959
	if (retval) {
		*action = 0;
	} else {
		*action = 1;
	}
E
Eric Sandeen 已提交
1960
	return 0;
L
Linus Torvalds 已提交
1961 1962 1963 1964 1965 1966 1967 1968 1969
}

/*
 * Remove a name from the leaf attribute list structure.
 *
 * Return 1 if leaf is less than 37% full, 0 if >= 37% full.
 * If two leaves are 37% full, when combined they will leave 25% free.
 */
int
D
Dave Chinner 已提交
1970 1971 1972
xfs_attr3_leaf_remove(
	struct xfs_buf		*bp,
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
1973
{
D
Dave Chinner 已提交
1974 1975 1976 1977 1978 1979 1980 1981 1982 1983
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_attr_leaf_entry *entry;
	int			before;
	int			after;
	int			smallest;
	int			entsize;
	int			tablesize;
	int			tmp;
	int			i;
L
Linus Torvalds 已提交
1984

1985 1986
	trace_xfs_attr_leaf_remove(args);

1987
	leaf = bp->b_addr;
1988
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
1989

1990
	ASSERT(ichdr.count > 0 && ichdr.count < args->geo->blksize / 8);
D
Dave Chinner 已提交
1991 1992 1993 1994 1995 1996 1997
	ASSERT(args->index >= 0 && args->index < ichdr.count);
	ASSERT(ichdr.firstused >= ichdr.count * sizeof(*entry) +
					xfs_attr3_leaf_hdr_size(leaf));

	entry = &xfs_attr3_leaf_entryp(leaf)[args->index];

	ASSERT(be16_to_cpu(entry->nameidx) >= ichdr.firstused);
1998
	ASSERT(be16_to_cpu(entry->nameidx) < args->geo->blksize);
L
Linus Torvalds 已提交
1999 2000 2001 2002 2003 2004 2005

	/*
	 * Scan through free region table:
	 *    check for adjacency of free'd entry with an existing one,
	 *    find smallest free region in case we need to replace it,
	 *    adjust any map that borders the entry table,
	 */
D
Dave Chinner 已提交
2006 2007 2008
	tablesize = ichdr.count * sizeof(xfs_attr_leaf_entry_t)
					+ xfs_attr3_leaf_hdr_size(leaf);
	tmp = ichdr.freemap[0].size;
L
Linus Torvalds 已提交
2009 2010 2011
	before = after = -1;
	smallest = XFS_ATTR_LEAF_MAPSIZE - 1;
	entsize = xfs_attr_leaf_entsize(leaf, args->index);
D
Dave Chinner 已提交
2012
	for (i = 0; i < XFS_ATTR_LEAF_MAPSIZE; i++) {
2013 2014
		ASSERT(ichdr.freemap[i].base < args->geo->blksize);
		ASSERT(ichdr.freemap[i].size < args->geo->blksize);
D
Dave Chinner 已提交
2015 2016 2017
		if (ichdr.freemap[i].base == tablesize) {
			ichdr.freemap[i].base -= sizeof(xfs_attr_leaf_entry_t);
			ichdr.freemap[i].size += sizeof(xfs_attr_leaf_entry_t);
L
Linus Torvalds 已提交
2018 2019
		}

D
Dave Chinner 已提交
2020 2021
		if (ichdr.freemap[i].base + ichdr.freemap[i].size ==
				be16_to_cpu(entry->nameidx)) {
L
Linus Torvalds 已提交
2022
			before = i;
D
Dave Chinner 已提交
2023 2024
		} else if (ichdr.freemap[i].base ==
				(be16_to_cpu(entry->nameidx) + entsize)) {
L
Linus Torvalds 已提交
2025
			after = i;
D
Dave Chinner 已提交
2026 2027
		} else if (ichdr.freemap[i].size < tmp) {
			tmp = ichdr.freemap[i].size;
L
Linus Torvalds 已提交
2028 2029 2030 2031 2032 2033 2034 2035 2036 2037
			smallest = i;
		}
	}

	/*
	 * Coalesce adjacent freemap regions,
	 * or replace the smallest region.
	 */
	if ((before >= 0) || (after >= 0)) {
		if ((before >= 0) && (after >= 0)) {
D
Dave Chinner 已提交
2038 2039 2040 2041
			ichdr.freemap[before].size += entsize;
			ichdr.freemap[before].size += ichdr.freemap[after].size;
			ichdr.freemap[after].base = 0;
			ichdr.freemap[after].size = 0;
L
Linus Torvalds 已提交
2042
		} else if (before >= 0) {
D
Dave Chinner 已提交
2043
			ichdr.freemap[before].size += entsize;
L
Linus Torvalds 已提交
2044
		} else {
D
Dave Chinner 已提交
2045 2046
			ichdr.freemap[after].base = be16_to_cpu(entry->nameidx);
			ichdr.freemap[after].size += entsize;
L
Linus Torvalds 已提交
2047 2048 2049 2050 2051
		}
	} else {
		/*
		 * Replace smallest region (if it is smaller than free'd entry)
		 */
D
Dave Chinner 已提交
2052 2053 2054
		if (ichdr.freemap[smallest].size < entsize) {
			ichdr.freemap[smallest].base = be16_to_cpu(entry->nameidx);
			ichdr.freemap[smallest].size = entsize;
L
Linus Torvalds 已提交
2055 2056 2057 2058 2059 2060
		}
	}

	/*
	 * Did we remove the first entry?
	 */
D
Dave Chinner 已提交
2061
	if (be16_to_cpu(entry->nameidx) == ichdr.firstused)
L
Linus Torvalds 已提交
2062 2063 2064 2065 2066 2067 2068
		smallest = 1;
	else
		smallest = 0;

	/*
	 * Compress the remaining entries and zero out the removed stuff.
	 */
D
Dave Chinner 已提交
2069 2070
	memset(xfs_attr3_leaf_name(leaf, args->index), 0, entsize);
	ichdr.usedbytes -= entsize;
2071
	xfs_trans_log_buf(args->trans, bp,
D
Dave Chinner 已提交
2072
	     XFS_DA_LOGRANGE(leaf, xfs_attr3_leaf_name(leaf, args->index),
L
Linus Torvalds 已提交
2073 2074
				   entsize));

D
Dave Chinner 已提交
2075 2076 2077
	tmp = (ichdr.count - args->index) * sizeof(xfs_attr_leaf_entry_t);
	memmove(entry, entry + 1, tmp);
	ichdr.count--;
2078
	xfs_trans_log_buf(args->trans, bp,
D
Dave Chinner 已提交
2079 2080 2081 2082
	    XFS_DA_LOGRANGE(leaf, entry, tmp + sizeof(xfs_attr_leaf_entry_t)));

	entry = &xfs_attr3_leaf_entryp(leaf)[ichdr.count];
	memset(entry, 0, sizeof(xfs_attr_leaf_entry_t));
L
Linus Torvalds 已提交
2083 2084 2085 2086 2087 2088 2089 2090

	/*
	 * If we removed the first entry, re-find the first used byte
	 * in the name area.  Note that if the entry was the "firstused",
	 * then we don't have a "hole" in our block resulting from
	 * removing the name.
	 */
	if (smallest) {
2091
		tmp = args->geo->blksize;
D
Dave Chinner 已提交
2092 2093 2094
		entry = xfs_attr3_leaf_entryp(leaf);
		for (i = ichdr.count - 1; i >= 0; entry++, i--) {
			ASSERT(be16_to_cpu(entry->nameidx) >= ichdr.firstused);
2095
			ASSERT(be16_to_cpu(entry->nameidx) < args->geo->blksize);
2096 2097 2098

			if (be16_to_cpu(entry->nameidx) < tmp)
				tmp = be16_to_cpu(entry->nameidx);
L
Linus Torvalds 已提交
2099
		}
D
Dave Chinner 已提交
2100
		ichdr.firstused = tmp;
2101
		ASSERT(ichdr.firstused != 0);
L
Linus Torvalds 已提交
2102
	} else {
D
Dave Chinner 已提交
2103
		ichdr.holes = 1;	/* mark as needing compaction */
L
Linus Torvalds 已提交
2104
	}
2105
	xfs_attr3_leaf_hdr_to_disk(args->geo, leaf, &ichdr);
2106
	xfs_trans_log_buf(args->trans, bp,
D
Dave Chinner 已提交
2107 2108
			  XFS_DA_LOGRANGE(leaf, &leaf->hdr,
					  xfs_attr3_leaf_hdr_size(leaf)));
L
Linus Torvalds 已提交
2109 2110 2111 2112 2113

	/*
	 * Check if leaf is less than 50% full, caller may want to
	 * "join" the leaf with a sibling if so.
	 */
D
Dave Chinner 已提交
2114 2115 2116
	tmp = ichdr.usedbytes + xfs_attr3_leaf_hdr_size(leaf) +
	      ichdr.count * sizeof(xfs_attr_leaf_entry_t);

2117
	return tmp < args->geo->magicpct; /* leaf is < 37% full */
L
Linus Torvalds 已提交
2118 2119 2120 2121 2122 2123
}

/*
 * Move all the attribute list entries from drop_leaf into save_leaf.
 */
void
D
Dave Chinner 已提交
2124 2125 2126 2127
xfs_attr3_leaf_unbalance(
	struct xfs_da_state	*state,
	struct xfs_da_state_blk	*drop_blk,
	struct xfs_da_state_blk	*save_blk)
L
Linus Torvalds 已提交
2128
{
D
Dave Chinner 已提交
2129 2130 2131 2132 2133
	struct xfs_attr_leafblock *drop_leaf = drop_blk->bp->b_addr;
	struct xfs_attr_leafblock *save_leaf = save_blk->bp->b_addr;
	struct xfs_attr3_icleaf_hdr drophdr;
	struct xfs_attr3_icleaf_hdr savehdr;
	struct xfs_attr_leaf_entry *entry;
L
Linus Torvalds 已提交
2134

2135 2136
	trace_xfs_attr_leaf_unbalance(state->args);

2137 2138
	drop_leaf = drop_blk->bp->b_addr;
	save_leaf = save_blk->bp->b_addr;
2139 2140
	xfs_attr3_leaf_hdr_from_disk(state->args->geo, &drophdr, drop_leaf);
	xfs_attr3_leaf_hdr_from_disk(state->args->geo, &savehdr, save_leaf);
D
Dave Chinner 已提交
2141
	entry = xfs_attr3_leaf_entryp(drop_leaf);
L
Linus Torvalds 已提交
2142 2143 2144 2145

	/*
	 * Save last hashval from dying block for later Btree fixup.
	 */
D
Dave Chinner 已提交
2146
	drop_blk->hashval = be32_to_cpu(entry[drophdr.count - 1].hashval);
L
Linus Torvalds 已提交
2147 2148 2149 2150 2151 2152

	/*
	 * Check if we need a temp buffer, or can we do it in place.
	 * Note that we don't check "leaf" for holes because we will
	 * always be dropping it, toosmall() decided that for us already.
	 */
D
Dave Chinner 已提交
2153
	if (savehdr.holes == 0) {
L
Linus Torvalds 已提交
2154 2155 2156 2157
		/*
		 * dest leaf has no holes, so we add there.  May need
		 * to make some room in the entry array.
		 */
D
Dave Chinner 已提交
2158 2159
		if (xfs_attr3_leaf_order(save_blk->bp, &savehdr,
					 drop_blk->bp, &drophdr)) {
2160 2161
			xfs_attr3_leaf_moveents(state->args,
						drop_leaf, &drophdr, 0,
D
Dave Chinner 已提交
2162
						save_leaf, &savehdr, 0,
2163
						drophdr.count);
L
Linus Torvalds 已提交
2164
		} else {
2165 2166
			xfs_attr3_leaf_moveents(state->args,
						drop_leaf, &drophdr, 0,
D
Dave Chinner 已提交
2167
						save_leaf, &savehdr,
2168
						savehdr.count, drophdr.count);
L
Linus Torvalds 已提交
2169 2170 2171 2172 2173 2174
		}
	} else {
		/*
		 * Destination has holes, so we make a temporary copy
		 * of the leaf and add them both to that.
		 */
D
Dave Chinner 已提交
2175 2176 2177
		struct xfs_attr_leafblock *tmp_leaf;
		struct xfs_attr3_icleaf_hdr tmphdr;

2178
		tmp_leaf = kmem_zalloc(state->args->geo->blksize, KM_SLEEP);
2179 2180 2181 2182 2183 2184 2185

		/*
		 * Copy the header into the temp leaf so that all the stuff
		 * not in the incore header is present and gets copied back in
		 * once we've moved all the entries.
		 */
		memcpy(tmp_leaf, save_leaf, xfs_attr3_leaf_hdr_size(save_leaf));
D
Dave Chinner 已提交
2186

2187
		memset(&tmphdr, 0, sizeof(tmphdr));
D
Dave Chinner 已提交
2188 2189 2190
		tmphdr.magic = savehdr.magic;
		tmphdr.forw = savehdr.forw;
		tmphdr.back = savehdr.back;
2191
		tmphdr.firstused = state->args->geo->blksize;
2192 2193

		/* write the header to the temp buffer to initialise it */
2194
		xfs_attr3_leaf_hdr_to_disk(state->args->geo, tmp_leaf, &tmphdr);
2195

D
Dave Chinner 已提交
2196 2197
		if (xfs_attr3_leaf_order(save_blk->bp, &savehdr,
					 drop_blk->bp, &drophdr)) {
2198 2199
			xfs_attr3_leaf_moveents(state->args,
						drop_leaf, &drophdr, 0,
D
Dave Chinner 已提交
2200
						tmp_leaf, &tmphdr, 0,
2201 2202 2203
						drophdr.count);
			xfs_attr3_leaf_moveents(state->args,
						save_leaf, &savehdr, 0,
D
Dave Chinner 已提交
2204
						tmp_leaf, &tmphdr, tmphdr.count,
2205
						savehdr.count);
L
Linus Torvalds 已提交
2206
		} else {
2207 2208
			xfs_attr3_leaf_moveents(state->args,
						save_leaf, &savehdr, 0,
D
Dave Chinner 已提交
2209
						tmp_leaf, &tmphdr, 0,
2210 2211 2212
						savehdr.count);
			xfs_attr3_leaf_moveents(state->args,
						drop_leaf, &drophdr, 0,
D
Dave Chinner 已提交
2213
						tmp_leaf, &tmphdr, tmphdr.count,
2214
						drophdr.count);
L
Linus Torvalds 已提交
2215
		}
2216
		memcpy(save_leaf, tmp_leaf, state->args->geo->blksize);
D
Dave Chinner 已提交
2217 2218
		savehdr = tmphdr; /* struct copy */
		kmem_free(tmp_leaf);
L
Linus Torvalds 已提交
2219 2220
	}

2221
	xfs_attr3_leaf_hdr_to_disk(state->args->geo, save_leaf, &savehdr);
2222
	xfs_trans_log_buf(state->args->trans, save_blk->bp, 0,
2223
					   state->args->geo->blksize - 1);
L
Linus Torvalds 已提交
2224 2225 2226 2227

	/*
	 * Copy out last hashval in each block for B-tree code.
	 */
D
Dave Chinner 已提交
2228 2229
	entry = xfs_attr3_leaf_entryp(save_leaf);
	save_blk->hashval = be32_to_cpu(entry[savehdr.count - 1].hashval);
L
Linus Torvalds 已提交
2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249
}

/*========================================================================
 * Routines used for finding things in the Btree.
 *========================================================================*/

/*
 * Look up a name in a leaf attribute list structure.
 * This is the internal routine, it uses the caller's buffer.
 *
 * Note that duplicate keys are allowed, but only check within the
 * current leaf node.  The Btree code must check in adjacent leaf nodes.
 *
 * Return in args->index the index into the entry[] array of either
 * the found entry, or where the entry should have been (insert before
 * that entry).
 *
 * Don't change the args->value unless we find the attribute.
 */
int
D
Dave Chinner 已提交
2250 2251 2252
xfs_attr3_leaf_lookup_int(
	struct xfs_buf		*bp,
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
2253
{
D
Dave Chinner 已提交
2254 2255 2256 2257 2258 2259 2260 2261 2262
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_entry *entries;
	struct xfs_attr_leaf_name_local *name_loc;
	struct xfs_attr_leaf_name_remote *name_rmt;
	xfs_dahash_t		hashval;
	int			probe;
	int			span;
L
Linus Torvalds 已提交
2263

2264 2265
	trace_xfs_attr_leaf_lookup(args);

2266
	leaf = bp->b_addr;
2267
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
2268
	entries = xfs_attr3_leaf_entryp(leaf);
2269 2270
	if (ichdr.count >= args->geo->blksize / 8)
		return -EFSCORRUPTED;
L
Linus Torvalds 已提交
2271 2272 2273 2274 2275

	/*
	 * Binary search.  (note: small blocks will skip this loop)
	 */
	hashval = args->hashval;
D
Dave Chinner 已提交
2276 2277
	probe = span = ichdr.count / 2;
	for (entry = &entries[probe]; span > 4; entry = &entries[probe]) {
L
Linus Torvalds 已提交
2278
		span /= 2;
2279
		if (be32_to_cpu(entry->hashval) < hashval)
L
Linus Torvalds 已提交
2280
			probe += span;
2281
		else if (be32_to_cpu(entry->hashval) > hashval)
L
Linus Torvalds 已提交
2282 2283 2284 2285
			probe -= span;
		else
			break;
	}
2286 2287 2288 2289
	if (!(probe >= 0 && (!ichdr.count || probe < ichdr.count)))
		return -EFSCORRUPTED;
	if (!(span <= 4 || be32_to_cpu(entry->hashval) == hashval))
		return -EFSCORRUPTED;
L
Linus Torvalds 已提交
2290 2291 2292 2293 2294

	/*
	 * Since we may have duplicate hashval's, find the first matching
	 * hashval in the leaf.
	 */
D
Dave Chinner 已提交
2295
	while (probe > 0 && be32_to_cpu(entry->hashval) >= hashval) {
L
Linus Torvalds 已提交
2296 2297 2298
		entry--;
		probe--;
	}
D
Dave Chinner 已提交
2299 2300
	while (probe < ichdr.count &&
	       be32_to_cpu(entry->hashval) < hashval) {
L
Linus Torvalds 已提交
2301 2302 2303
		entry++;
		probe++;
	}
D
Dave Chinner 已提交
2304
	if (probe == ichdr.count || be32_to_cpu(entry->hashval) != hashval) {
L
Linus Torvalds 已提交
2305
		args->index = probe;
D
Dave Chinner 已提交
2306
		return -ENOATTR;
L
Linus Torvalds 已提交
2307 2308 2309 2310 2311
	}

	/*
	 * Duplicate keys may be present, so search all of them for a match.
	 */
D
Dave Chinner 已提交
2312
	for (; probe < ichdr.count && (be32_to_cpu(entry->hashval) == hashval);
L
Linus Torvalds 已提交
2313 2314 2315 2316 2317 2318 2319 2320 2321 2322 2323 2324 2325
			entry++, probe++) {
/*
 * GROT: Add code to remove incomplete entries.
 */
		/*
		 * If we are looking for INCOMPLETE entries, show only those.
		 * If we are looking for complete entries, show only those.
		 */
		if ((args->flags & XFS_ATTR_INCOMPLETE) !=
		    (entry->flags & XFS_ATTR_INCOMPLETE)) {
			continue;
		}
		if (entry->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
2326
			name_loc = xfs_attr3_leaf_name_local(leaf, probe);
L
Linus Torvalds 已提交
2327 2328
			if (name_loc->namelen != args->namelen)
				continue;
D
Dave Chinner 已提交
2329 2330
			if (memcmp(args->name, name_loc->nameval,
							args->namelen) != 0)
L
Linus Torvalds 已提交
2331
				continue;
2332
			if (!xfs_attr_namesp_match(args->flags, entry->flags))
L
Linus Torvalds 已提交
2333 2334
				continue;
			args->index = probe;
D
Dave Chinner 已提交
2335
			return -EEXIST;
L
Linus Torvalds 已提交
2336
		} else {
D
Dave Chinner 已提交
2337
			name_rmt = xfs_attr3_leaf_name_remote(leaf, probe);
L
Linus Torvalds 已提交
2338 2339
			if (name_rmt->namelen != args->namelen)
				continue;
D
Dave Chinner 已提交
2340 2341
			if (memcmp(args->name, name_rmt->name,
							args->namelen) != 0)
L
Linus Torvalds 已提交
2342
				continue;
2343
			if (!xfs_attr_namesp_match(args->flags, entry->flags))
L
Linus Torvalds 已提交
2344 2345
				continue;
			args->index = probe;
2346
			args->rmtvaluelen = be32_to_cpu(name_rmt->valuelen);
2347
			args->rmtblkno = be32_to_cpu(name_rmt->valueblk);
D
Dave Chinner 已提交
2348 2349
			args->rmtblkcnt = xfs_attr3_rmt_blocks(
							args->dp->i_mount,
2350
							args->rmtvaluelen);
D
Dave Chinner 已提交
2351
			return -EEXIST;
L
Linus Torvalds 已提交
2352 2353 2354
		}
	}
	args->index = probe;
D
Dave Chinner 已提交
2355
	return -ENOATTR;
L
Linus Torvalds 已提交
2356 2357 2358 2359 2360 2361 2362
}

/*
 * Get the value associated with an attribute name from a leaf attribute
 * list structure.
 */
int
D
Dave Chinner 已提交
2363 2364 2365
xfs_attr3_leaf_getvalue(
	struct xfs_buf		*bp,
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
2366
{
D
Dave Chinner 已提交
2367 2368 2369 2370 2371 2372
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_name_local *name_loc;
	struct xfs_attr_leaf_name_remote *name_rmt;
	int			valuelen;
L
Linus Torvalds 已提交
2373

2374
	leaf = bp->b_addr;
2375
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
2376
	ASSERT(ichdr.count < args->geo->blksize / 8);
D
Dave Chinner 已提交
2377
	ASSERT(args->index < ichdr.count);
L
Linus Torvalds 已提交
2378

D
Dave Chinner 已提交
2379
	entry = &xfs_attr3_leaf_entryp(leaf)[args->index];
L
Linus Torvalds 已提交
2380
	if (entry->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
2381
		name_loc = xfs_attr3_leaf_name_local(leaf, args->index);
L
Linus Torvalds 已提交
2382 2383
		ASSERT(name_loc->namelen == args->namelen);
		ASSERT(memcmp(args->name, name_loc->nameval, args->namelen) == 0);
2384
		valuelen = be16_to_cpu(name_loc->valuelen);
L
Linus Torvalds 已提交
2385 2386
		if (args->flags & ATTR_KERNOVAL) {
			args->valuelen = valuelen;
D
Dave Chinner 已提交
2387
			return 0;
L
Linus Torvalds 已提交
2388 2389 2390
		}
		if (args->valuelen < valuelen) {
			args->valuelen = valuelen;
D
Dave Chinner 已提交
2391
			return -ERANGE;
L
Linus Torvalds 已提交
2392 2393 2394 2395
		}
		args->valuelen = valuelen;
		memcpy(args->value, &name_loc->nameval[args->namelen], valuelen);
	} else {
D
Dave Chinner 已提交
2396
		name_rmt = xfs_attr3_leaf_name_remote(leaf, args->index);
L
Linus Torvalds 已提交
2397 2398
		ASSERT(name_rmt->namelen == args->namelen);
		ASSERT(memcmp(args->name, name_rmt->name, args->namelen) == 0);
2399
		args->rmtvaluelen = be32_to_cpu(name_rmt->valuelen);
2400
		args->rmtblkno = be32_to_cpu(name_rmt->valueblk);
D
Dave Chinner 已提交
2401
		args->rmtblkcnt = xfs_attr3_rmt_blocks(args->dp->i_mount,
2402
						       args->rmtvaluelen);
L
Linus Torvalds 已提交
2403
		if (args->flags & ATTR_KERNOVAL) {
2404
			args->valuelen = args->rmtvaluelen;
D
Dave Chinner 已提交
2405
			return 0;
L
Linus Torvalds 已提交
2406
		}
2407 2408
		if (args->valuelen < args->rmtvaluelen) {
			args->valuelen = args->rmtvaluelen;
D
Dave Chinner 已提交
2409
			return -ERANGE;
L
Linus Torvalds 已提交
2410
		}
2411
		args->valuelen = args->rmtvaluelen;
L
Linus Torvalds 已提交
2412
	}
D
Dave Chinner 已提交
2413
	return 0;
L
Linus Torvalds 已提交
2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425
}

/*========================================================================
 * Utility routines.
 *========================================================================*/

/*
 * Move the indicated entries from one leaf to another.
 * NOTE: this routine modifies both source and destination leaves.
 */
/*ARGSUSED*/
STATIC void
D
Dave Chinner 已提交
2426
xfs_attr3_leaf_moveents(
2427
	struct xfs_da_args		*args,
D
Dave Chinner 已提交
2428 2429 2430 2431 2432 2433
	struct xfs_attr_leafblock	*leaf_s,
	struct xfs_attr3_icleaf_hdr	*ichdr_s,
	int				start_s,
	struct xfs_attr_leafblock	*leaf_d,
	struct xfs_attr3_icleaf_hdr	*ichdr_d,
	int				start_d,
2434
	int				count)
L
Linus Torvalds 已提交
2435
{
D
Dave Chinner 已提交
2436 2437 2438 2439 2440
	struct xfs_attr_leaf_entry	*entry_s;
	struct xfs_attr_leaf_entry	*entry_d;
	int				desti;
	int				tmp;
	int				i;
L
Linus Torvalds 已提交
2441 2442 2443 2444 2445 2446 2447 2448 2449 2450

	/*
	 * Check for nothing to do.
	 */
	if (count == 0)
		return;

	/*
	 * Set up environment.
	 */
D
Dave Chinner 已提交
2451 2452 2453
	ASSERT(ichdr_s->magic == XFS_ATTR_LEAF_MAGIC ||
	       ichdr_s->magic == XFS_ATTR3_LEAF_MAGIC);
	ASSERT(ichdr_s->magic == ichdr_d->magic);
2454
	ASSERT(ichdr_s->count > 0 && ichdr_s->count < args->geo->blksize / 8);
D
Dave Chinner 已提交
2455 2456
	ASSERT(ichdr_s->firstused >= (ichdr_s->count * sizeof(*entry_s))
					+ xfs_attr3_leaf_hdr_size(leaf_s));
2457
	ASSERT(ichdr_d->count < args->geo->blksize / 8);
D
Dave Chinner 已提交
2458 2459 2460 2461 2462 2463 2464
	ASSERT(ichdr_d->firstused >= (ichdr_d->count * sizeof(*entry_d))
					+ xfs_attr3_leaf_hdr_size(leaf_d));

	ASSERT(start_s < ichdr_s->count);
	ASSERT(start_d <= ichdr_d->count);
	ASSERT(count <= ichdr_s->count);

L
Linus Torvalds 已提交
2465 2466 2467 2468

	/*
	 * Move the entries in the destination leaf up to make a hole?
	 */
D
Dave Chinner 已提交
2469 2470
	if (start_d < ichdr_d->count) {
		tmp  = ichdr_d->count - start_d;
L
Linus Torvalds 已提交
2471
		tmp *= sizeof(xfs_attr_leaf_entry_t);
D
Dave Chinner 已提交
2472 2473 2474
		entry_s = &xfs_attr3_leaf_entryp(leaf_d)[start_d];
		entry_d = &xfs_attr3_leaf_entryp(leaf_d)[start_d + count];
		memmove(entry_d, entry_s, tmp);
L
Linus Torvalds 已提交
2475 2476 2477 2478 2479 2480
	}

	/*
	 * Copy all entry's in the same (sorted) order,
	 * but allocate attribute info packed and in sequence.
	 */
D
Dave Chinner 已提交
2481 2482
	entry_s = &xfs_attr3_leaf_entryp(leaf_s)[start_s];
	entry_d = &xfs_attr3_leaf_entryp(leaf_d)[start_d];
L
Linus Torvalds 已提交
2483 2484
	desti = start_d;
	for (i = 0; i < count; entry_s++, entry_d++, desti++, i++) {
D
Dave Chinner 已提交
2485
		ASSERT(be16_to_cpu(entry_s->nameidx) >= ichdr_s->firstused);
L
Linus Torvalds 已提交
2486 2487 2488 2489 2490 2491 2492 2493
		tmp = xfs_attr_leaf_entsize(leaf_s, start_s + i);
#ifdef GROT
		/*
		 * Code to drop INCOMPLETE entries.  Difficult to use as we
		 * may also need to change the insertion index.  Code turned
		 * off for 6.2, should be revisited later.
		 */
		if (entry_s->flags & XFS_ATTR_INCOMPLETE) { /* skip partials? */
D
Dave Chinner 已提交
2494 2495 2496
			memset(xfs_attr3_leaf_name(leaf_s, start_s + i), 0, tmp);
			ichdr_s->usedbytes -= tmp;
			ichdr_s->count -= 1;
L
Linus Torvalds 已提交
2497 2498 2499 2500 2501 2502
			entry_d--;	/* to compensate for ++ in loop hdr */
			desti--;
			if ((start_s + i) < offset)
				result++;	/* insertion index adjustment */
		} else {
#endif /* GROT */
D
Dave Chinner 已提交
2503
			ichdr_d->firstused -= tmp;
L
Linus Torvalds 已提交
2504 2505
			/* both on-disk, don't endian flip twice */
			entry_d->hashval = entry_s->hashval;
D
Dave Chinner 已提交
2506
			entry_d->nameidx = cpu_to_be16(ichdr_d->firstused);
L
Linus Torvalds 已提交
2507
			entry_d->flags = entry_s->flags;
2508
			ASSERT(be16_to_cpu(entry_d->nameidx) + tmp
2509
							<= args->geo->blksize);
D
Dave Chinner 已提交
2510 2511
			memmove(xfs_attr3_leaf_name(leaf_d, desti),
				xfs_attr3_leaf_name(leaf_s, start_s + i), tmp);
2512
			ASSERT(be16_to_cpu(entry_s->nameidx) + tmp
2513
							<= args->geo->blksize);
D
Dave Chinner 已提交
2514 2515 2516 2517 2518 2519 2520 2521
			memset(xfs_attr3_leaf_name(leaf_s, start_s + i), 0, tmp);
			ichdr_s->usedbytes -= tmp;
			ichdr_d->usedbytes += tmp;
			ichdr_s->count -= 1;
			ichdr_d->count += 1;
			tmp = ichdr_d->count * sizeof(xfs_attr_leaf_entry_t)
					+ xfs_attr3_leaf_hdr_size(leaf_d);
			ASSERT(ichdr_d->firstused >= tmp);
L
Linus Torvalds 已提交
2522 2523 2524 2525 2526 2527 2528 2529
#ifdef GROT
		}
#endif /* GROT */
	}

	/*
	 * Zero out the entries we just copied.
	 */
D
Dave Chinner 已提交
2530
	if (start_s == ichdr_s->count) {
L
Linus Torvalds 已提交
2531
		tmp = count * sizeof(xfs_attr_leaf_entry_t);
D
Dave Chinner 已提交
2532
		entry_s = &xfs_attr3_leaf_entryp(leaf_s)[start_s];
L
Linus Torvalds 已提交
2533
		ASSERT(((char *)entry_s + tmp) <=
2534
		       ((char *)leaf_s + args->geo->blksize));
D
Dave Chinner 已提交
2535
		memset(entry_s, 0, tmp);
L
Linus Torvalds 已提交
2536 2537 2538 2539 2540
	} else {
		/*
		 * Move the remaining entries down to fill the hole,
		 * then zero the entries at the top.
		 */
D
Dave Chinner 已提交
2541 2542 2543 2544
		tmp  = (ichdr_s->count - count) * sizeof(xfs_attr_leaf_entry_t);
		entry_s = &xfs_attr3_leaf_entryp(leaf_s)[start_s + count];
		entry_d = &xfs_attr3_leaf_entryp(leaf_s)[start_s];
		memmove(entry_d, entry_s, tmp);
L
Linus Torvalds 已提交
2545 2546

		tmp = count * sizeof(xfs_attr_leaf_entry_t);
D
Dave Chinner 已提交
2547
		entry_s = &xfs_attr3_leaf_entryp(leaf_s)[ichdr_s->count];
L
Linus Torvalds 已提交
2548
		ASSERT(((char *)entry_s + tmp) <=
2549
		       ((char *)leaf_s + args->geo->blksize));
D
Dave Chinner 已提交
2550
		memset(entry_s, 0, tmp);
L
Linus Torvalds 已提交
2551 2552 2553 2554 2555
	}

	/*
	 * Fill in the freemap information
	 */
D
Dave Chinner 已提交
2556 2557 2558 2559 2560 2561 2562 2563
	ichdr_d->freemap[0].base = xfs_attr3_leaf_hdr_size(leaf_d);
	ichdr_d->freemap[0].base += ichdr_d->count * sizeof(xfs_attr_leaf_entry_t);
	ichdr_d->freemap[0].size = ichdr_d->firstused - ichdr_d->freemap[0].base;
	ichdr_d->freemap[1].base = 0;
	ichdr_d->freemap[2].base = 0;
	ichdr_d->freemap[1].size = 0;
	ichdr_d->freemap[2].size = 0;
	ichdr_s->holes = 1;	/* leaf may not be compact */
L
Linus Torvalds 已提交
2564 2565 2566 2567 2568 2569
}

/*
 * Pick up the last hashvalue from a leaf block.
 */
xfs_dahash_t
2570 2571 2572
xfs_attr_leaf_lasthash(
	struct xfs_buf	*bp,
	int		*count)
L
Linus Torvalds 已提交
2573
{
D
Dave Chinner 已提交
2574 2575
	struct xfs_attr3_icleaf_hdr ichdr;
	struct xfs_attr_leaf_entry *entries;
2576
	struct xfs_mount *mp = bp->b_target->bt_mount;
L
Linus Torvalds 已提交
2577

2578
	xfs_attr3_leaf_hdr_from_disk(mp->m_attr_geo, &ichdr, bp->b_addr);
D
Dave Chinner 已提交
2579
	entries = xfs_attr3_leaf_entryp(bp->b_addr);
L
Linus Torvalds 已提交
2580
	if (count)
D
Dave Chinner 已提交
2581 2582 2583 2584
		*count = ichdr.count;
	if (!ichdr.count)
		return 0;
	return be32_to_cpu(entries[ichdr.count - 1].hashval);
L
Linus Torvalds 已提交
2585 2586 2587 2588 2589 2590
}

/*
 * Calculate the number of bytes used to store the indicated attribute
 * (whether local or remote only calculate bytes in this block).
 */
2591
STATIC int
L
Linus Torvalds 已提交
2592 2593
xfs_attr_leaf_entsize(xfs_attr_leafblock_t *leaf, int index)
{
D
Dave Chinner 已提交
2594
	struct xfs_attr_leaf_entry *entries;
L
Linus Torvalds 已提交
2595 2596 2597 2598
	xfs_attr_leaf_name_local_t *name_loc;
	xfs_attr_leaf_name_remote_t *name_rmt;
	int size;

D
Dave Chinner 已提交
2599 2600 2601
	entries = xfs_attr3_leaf_entryp(leaf);
	if (entries[index].flags & XFS_ATTR_LOCAL) {
		name_loc = xfs_attr3_leaf_name_local(leaf, index);
2602
		size = xfs_attr_leaf_entsize_local(name_loc->namelen,
2603
						   be16_to_cpu(name_loc->valuelen));
L
Linus Torvalds 已提交
2604
	} else {
D
Dave Chinner 已提交
2605
		name_rmt = xfs_attr3_leaf_name_remote(leaf, index);
2606
		size = xfs_attr_leaf_entsize_remote(name_rmt->namelen);
L
Linus Torvalds 已提交
2607
	}
D
Dave Chinner 已提交
2608
	return size;
L
Linus Torvalds 已提交
2609 2610 2611 2612 2613 2614 2615 2616 2617
}

/*
 * Calculate the number of bytes that would be required to store the new
 * attribute (whether local or remote only calculate bytes in this block).
 * This routine decides as a side effect whether the attribute will be
 * a "local" or a "remote" attribute.
 */
int
2618 2619 2620
xfs_attr_leaf_newentsize(
	struct xfs_da_args	*args,
	int			*local)
L
Linus Torvalds 已提交
2621
{
2622
	int			size;
L
Linus Torvalds 已提交
2623

2624 2625 2626
	size = xfs_attr_leaf_entsize_local(args->namelen, args->valuelen);
	if (size < xfs_attr_leaf_entsize_local_max(args->geo->blksize)) {
		if (local)
L
Linus Torvalds 已提交
2627
			*local = 1;
2628
		return size;
L
Linus Torvalds 已提交
2629
	}
2630 2631 2632
	if (local)
		*local = 0;
	return xfs_attr_leaf_entsize_remote(args->namelen);
L
Linus Torvalds 已提交
2633 2634 2635 2636 2637 2638 2639 2640 2641 2642 2643
}


/*========================================================================
 * Manage the INCOMPLETE flag in a leaf entry
 *========================================================================*/

/*
 * Clear the INCOMPLETE flag on an entry in a leaf block.
 */
int
D
Dave Chinner 已提交
2644 2645
xfs_attr3_leaf_clearflag(
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
2646
{
D
Dave Chinner 已提交
2647 2648 2649 2650 2651
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_name_remote *name_rmt;
	struct xfs_buf		*bp;
	int			error;
L
Linus Torvalds 已提交
2652
#ifdef DEBUG
D
Dave Chinner 已提交
2653
	struct xfs_attr3_icleaf_hdr ichdr;
L
Linus Torvalds 已提交
2654 2655 2656 2657 2658
	xfs_attr_leaf_name_local_t *name_loc;
	int namelen;
	char *name;
#endif /* DEBUG */

2659
	trace_xfs_attr_leaf_clearflag(args);
L
Linus Torvalds 已提交
2660 2661 2662
	/*
	 * Set up the operation.
	 */
D
Dave Chinner 已提交
2663
	error = xfs_attr3_leaf_read(args->trans, args->dp, args->blkno, -1, &bp);
2664
	if (error)
E
Eric Sandeen 已提交
2665
		return error;
L
Linus Torvalds 已提交
2666

2667
	leaf = bp->b_addr;
D
Dave Chinner 已提交
2668
	entry = &xfs_attr3_leaf_entryp(leaf)[args->index];
L
Linus Torvalds 已提交
2669 2670 2671
	ASSERT(entry->flags & XFS_ATTR_INCOMPLETE);

#ifdef DEBUG
2672
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
2673 2674 2675
	ASSERT(args->index < ichdr.count);
	ASSERT(args->index >= 0);

L
Linus Torvalds 已提交
2676
	if (entry->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
2677
		name_loc = xfs_attr3_leaf_name_local(leaf, args->index);
L
Linus Torvalds 已提交
2678 2679 2680
		namelen = name_loc->namelen;
		name = (char *)name_loc->nameval;
	} else {
D
Dave Chinner 已提交
2681
		name_rmt = xfs_attr3_leaf_name_remote(leaf, args->index);
L
Linus Torvalds 已提交
2682 2683 2684
		namelen = name_rmt->namelen;
		name = (char *)name_rmt->name;
	}
2685
	ASSERT(be32_to_cpu(entry->hashval) == args->hashval);
L
Linus Torvalds 已提交
2686 2687 2688 2689 2690
	ASSERT(namelen == args->namelen);
	ASSERT(memcmp(name, args->name, namelen) == 0);
#endif /* DEBUG */

	entry->flags &= ~XFS_ATTR_INCOMPLETE;
2691
	xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
2692 2693 2694 2695
			 XFS_DA_LOGRANGE(leaf, entry, sizeof(*entry)));

	if (args->rmtblkno) {
		ASSERT((entry->flags & XFS_ATTR_LOCAL) == 0);
D
Dave Chinner 已提交
2696
		name_rmt = xfs_attr3_leaf_name_remote(leaf, args->index);
2697
		name_rmt->valueblk = cpu_to_be32(args->rmtblkno);
2698
		name_rmt->valuelen = cpu_to_be32(args->rmtvaluelen);
2699
		xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
2700 2701 2702 2703 2704 2705
			 XFS_DA_LOGRANGE(leaf, name_rmt, sizeof(*name_rmt)));
	}

	/*
	 * Commit the flag value change and start the next trans in series.
	 */
C
Christoph Hellwig 已提交
2706
	return xfs_trans_roll_inode(&args->trans, args->dp);
L
Linus Torvalds 已提交
2707 2708 2709 2710 2711 2712
}

/*
 * Set the INCOMPLETE flag on an entry in a leaf block.
 */
int
D
Dave Chinner 已提交
2713 2714
xfs_attr3_leaf_setflag(
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
2715
{
D
Dave Chinner 已提交
2716 2717 2718 2719
	struct xfs_attr_leafblock *leaf;
	struct xfs_attr_leaf_entry *entry;
	struct xfs_attr_leaf_name_remote *name_rmt;
	struct xfs_buf		*bp;
L
Linus Torvalds 已提交
2720
	int error;
D
Dave Chinner 已提交
2721 2722 2723
#ifdef DEBUG
	struct xfs_attr3_icleaf_hdr ichdr;
#endif
L
Linus Torvalds 已提交
2724

2725 2726
	trace_xfs_attr_leaf_setflag(args);

L
Linus Torvalds 已提交
2727 2728 2729
	/*
	 * Set up the operation.
	 */
D
Dave Chinner 已提交
2730
	error = xfs_attr3_leaf_read(args->trans, args->dp, args->blkno, -1, &bp);
2731
	if (error)
E
Eric Sandeen 已提交
2732
		return error;
L
Linus Torvalds 已提交
2733

2734
	leaf = bp->b_addr;
D
Dave Chinner 已提交
2735
#ifdef DEBUG
2736
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr, leaf);
D
Dave Chinner 已提交
2737
	ASSERT(args->index < ichdr.count);
L
Linus Torvalds 已提交
2738
	ASSERT(args->index >= 0);
D
Dave Chinner 已提交
2739 2740
#endif
	entry = &xfs_attr3_leaf_entryp(leaf)[args->index];
L
Linus Torvalds 已提交
2741 2742 2743

	ASSERT((entry->flags & XFS_ATTR_INCOMPLETE) == 0);
	entry->flags |= XFS_ATTR_INCOMPLETE;
2744
	xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
2745 2746
			XFS_DA_LOGRANGE(leaf, entry, sizeof(*entry)));
	if ((entry->flags & XFS_ATTR_LOCAL) == 0) {
D
Dave Chinner 已提交
2747
		name_rmt = xfs_attr3_leaf_name_remote(leaf, args->index);
L
Linus Torvalds 已提交
2748 2749
		name_rmt->valueblk = 0;
		name_rmt->valuelen = 0;
2750
		xfs_trans_log_buf(args->trans, bp,
L
Linus Torvalds 已提交
2751 2752 2753 2754 2755 2756
			 XFS_DA_LOGRANGE(leaf, name_rmt, sizeof(*name_rmt)));
	}

	/*
	 * Commit the flag value change and start the next trans in series.
	 */
C
Christoph Hellwig 已提交
2757
	return xfs_trans_roll_inode(&args->trans, args->dp);
L
Linus Torvalds 已提交
2758 2759 2760 2761 2762 2763 2764 2765 2766 2767
}

/*
 * In a single transaction, clear the INCOMPLETE flag on the leaf entry
 * given by args->blkno/index and set the INCOMPLETE flag on the leaf
 * entry given by args->blkno2/index2.
 *
 * Note that they could be in different blocks, or in the same block.
 */
int
D
Dave Chinner 已提交
2768 2769
xfs_attr3_leaf_flipflags(
	struct xfs_da_args	*args)
L
Linus Torvalds 已提交
2770
{
D
Dave Chinner 已提交
2771 2772 2773 2774 2775 2776 2777
	struct xfs_attr_leafblock *leaf1;
	struct xfs_attr_leafblock *leaf2;
	struct xfs_attr_leaf_entry *entry1;
	struct xfs_attr_leaf_entry *entry2;
	struct xfs_attr_leaf_name_remote *name_rmt;
	struct xfs_buf		*bp1;
	struct xfs_buf		*bp2;
L
Linus Torvalds 已提交
2778 2779
	int error;
#ifdef DEBUG
D
Dave Chinner 已提交
2780 2781
	struct xfs_attr3_icleaf_hdr ichdr1;
	struct xfs_attr3_icleaf_hdr ichdr2;
L
Linus Torvalds 已提交
2782 2783 2784 2785 2786
	xfs_attr_leaf_name_local_t *name_loc;
	int namelen1, namelen2;
	char *name1, *name2;
#endif /* DEBUG */

2787 2788
	trace_xfs_attr_leaf_flipflags(args);

L
Linus Torvalds 已提交
2789 2790 2791
	/*
	 * Read the block containing the "old" attr
	 */
D
Dave Chinner 已提交
2792
	error = xfs_attr3_leaf_read(args->trans, args->dp, args->blkno, -1, &bp1);
2793 2794
	if (error)
		return error;
L
Linus Torvalds 已提交
2795 2796 2797 2798 2799

	/*
	 * Read the block containing the "new" attr, if it is different
	 */
	if (args->blkno2 != args->blkno) {
D
Dave Chinner 已提交
2800
		error = xfs_attr3_leaf_read(args->trans, args->dp, args->blkno2,
2801 2802 2803
					   -1, &bp2);
		if (error)
			return error;
L
Linus Torvalds 已提交
2804 2805 2806 2807
	} else {
		bp2 = bp1;
	}

2808
	leaf1 = bp1->b_addr;
D
Dave Chinner 已提交
2809
	entry1 = &xfs_attr3_leaf_entryp(leaf1)[args->index];
L
Linus Torvalds 已提交
2810

2811
	leaf2 = bp2->b_addr;
D
Dave Chinner 已提交
2812
	entry2 = &xfs_attr3_leaf_entryp(leaf2)[args->index2];
L
Linus Torvalds 已提交
2813 2814

#ifdef DEBUG
2815
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr1, leaf1);
D
Dave Chinner 已提交
2816 2817 2818
	ASSERT(args->index < ichdr1.count);
	ASSERT(args->index >= 0);

2819
	xfs_attr3_leaf_hdr_from_disk(args->geo, &ichdr2, leaf2);
D
Dave Chinner 已提交
2820 2821 2822
	ASSERT(args->index2 < ichdr2.count);
	ASSERT(args->index2 >= 0);

L
Linus Torvalds 已提交
2823
	if (entry1->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
2824
		name_loc = xfs_attr3_leaf_name_local(leaf1, args->index);
L
Linus Torvalds 已提交
2825 2826 2827
		namelen1 = name_loc->namelen;
		name1 = (char *)name_loc->nameval;
	} else {
D
Dave Chinner 已提交
2828
		name_rmt = xfs_attr3_leaf_name_remote(leaf1, args->index);
L
Linus Torvalds 已提交
2829 2830 2831 2832
		namelen1 = name_rmt->namelen;
		name1 = (char *)name_rmt->name;
	}
	if (entry2->flags & XFS_ATTR_LOCAL) {
D
Dave Chinner 已提交
2833
		name_loc = xfs_attr3_leaf_name_local(leaf2, args->index2);
L
Linus Torvalds 已提交
2834 2835 2836
		namelen2 = name_loc->namelen;
		name2 = (char *)name_loc->nameval;
	} else {
D
Dave Chinner 已提交
2837
		name_rmt = xfs_attr3_leaf_name_remote(leaf2, args->index2);
L
Linus Torvalds 已提交
2838 2839 2840
		namelen2 = name_rmt->namelen;
		name2 = (char *)name_rmt->name;
	}
2841
	ASSERT(be32_to_cpu(entry1->hashval) == be32_to_cpu(entry2->hashval));
L
Linus Torvalds 已提交
2842 2843 2844 2845 2846 2847 2848 2849
	ASSERT(namelen1 == namelen2);
	ASSERT(memcmp(name1, name2, namelen1) == 0);
#endif /* DEBUG */

	ASSERT(entry1->flags & XFS_ATTR_INCOMPLETE);
	ASSERT((entry2->flags & XFS_ATTR_INCOMPLETE) == 0);

	entry1->flags &= ~XFS_ATTR_INCOMPLETE;
2850
	xfs_trans_log_buf(args->trans, bp1,
L
Linus Torvalds 已提交
2851 2852 2853
			  XFS_DA_LOGRANGE(leaf1, entry1, sizeof(*entry1)));
	if (args->rmtblkno) {
		ASSERT((entry1->flags & XFS_ATTR_LOCAL) == 0);
D
Dave Chinner 已提交
2854
		name_rmt = xfs_attr3_leaf_name_remote(leaf1, args->index);
2855
		name_rmt->valueblk = cpu_to_be32(args->rmtblkno);
2856
		name_rmt->valuelen = cpu_to_be32(args->rmtvaluelen);
2857
		xfs_trans_log_buf(args->trans, bp1,
L
Linus Torvalds 已提交
2858 2859 2860 2861
			 XFS_DA_LOGRANGE(leaf1, name_rmt, sizeof(*name_rmt)));
	}

	entry2->flags |= XFS_ATTR_INCOMPLETE;
2862
	xfs_trans_log_buf(args->trans, bp2,
L
Linus Torvalds 已提交
2863 2864
			  XFS_DA_LOGRANGE(leaf2, entry2, sizeof(*entry2)));
	if ((entry2->flags & XFS_ATTR_LOCAL) == 0) {
D
Dave Chinner 已提交
2865
		name_rmt = xfs_attr3_leaf_name_remote(leaf2, args->index2);
L
Linus Torvalds 已提交
2866 2867
		name_rmt->valueblk = 0;
		name_rmt->valuelen = 0;
2868
		xfs_trans_log_buf(args->trans, bp2,
L
Linus Torvalds 已提交
2869 2870 2871 2872 2873 2874
			 XFS_DA_LOGRANGE(leaf2, name_rmt, sizeof(*name_rmt)));
	}

	/*
	 * Commit the flag value change and start the next trans in series.
	 */
C
Christoph Hellwig 已提交
2875
	error = xfs_trans_roll_inode(&args->trans, args->dp);
L
Linus Torvalds 已提交
2876

D
Dave Chinner 已提交
2877
	return error;
L
Linus Torvalds 已提交
2878
}