xfs_alloc_btree.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 2,070 行 · 第 1/5 页

C
2,070
字号
		xfs_alloc_rec_t	*lrp;	/* record pointer for left block */		lrp = XFS_ALLOC_REC_ADDR(left, nrec, cur);		rrp = XFS_ALLOC_REC_ADDR(right, 1, cur);		*lrp = *rrp;		xfs_alloc_log_recs(cur, lbp, nrec, nrec);		xfs_btree_check_rec(cur->bc_btnum, lrp - 1, lrp);	}	/*	 * Bump and log left's numrecs, decrement and log right's numrecs.	 */	INT_MOD(left->bb_numrecs, ARCH_CONVERT, +1);	xfs_alloc_log_block(cur->bc_tp, lbp, XFS_BB_NUMRECS);	INT_MOD(right->bb_numrecs, ARCH_CONVERT, -1);	xfs_alloc_log_block(cur->bc_tp, rbp, XFS_BB_NUMRECS);	/*	 * Slide the contents of right down one entry.	 */	if (level > 0) {#ifdef DEBUG		for (i = 0; i < INT_GET(right->bb_numrecs, ARCH_CONVERT); i++) {			if ((error = xfs_btree_check_sptr(cur, INT_GET(rpp[i + 1], ARCH_CONVERT),					level)))				return error;		}#endif		memmove(rkp, rkp + 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rkp));		memmove(rpp, rpp + 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rpp));		xfs_alloc_log_keys(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT));		xfs_alloc_log_ptrs(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT));	} else {		memmove(rrp, rrp + 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rrp));		xfs_alloc_log_recs(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT));		key.ar_startblock = rrp->ar_startblock; /* INT_: direct copy */		key.ar_blockcount = rrp->ar_blockcount; /* INT_: direct copy */		rkp = &key;	}	/*	 * Update the parent key values of right.	 */	if ((error = xfs_alloc_updkey(cur, rkp, level + 1)))		return error;	/*	 * Slide the cursor value left one.	 */	cur->bc_ptrs[level]--;	*stat = 1;	return 0;}/* * Allocate a new root block, fill it in. */STATIC int				/* error */xfs_alloc_newroot(	xfs_btree_cur_t		*cur,	/* btree cursor */	int			*stat)	/* success/failure */{	int			error;	/* error return value */	xfs_agblock_t		lbno;	/* left block number */	xfs_buf_t		*lbp;	/* left btree buffer */	xfs_alloc_block_t	*left;	/* left btree block */	xfs_mount_t		*mp;	/* mount structure */	xfs_agblock_t		nbno;	/* new block number */	xfs_buf_t		*nbp;	/* new (root) buffer */	xfs_alloc_block_t	*new;	/* new (root) btree block */	int			nptr;	/* new value for key index, 1 or 2 */	xfs_agblock_t		rbno;	/* right block number */	xfs_buf_t		*rbp;	/* right btree buffer */	xfs_alloc_block_t	*right;	/* right btree block */	mp = cur->bc_mp;	ASSERT(cur->bc_nlevels < XFS_AG_MAXLEVELS(mp));	/*	 * Get a buffer from the freelist blocks, for the new root.	 */	if ((error = xfs_alloc_get_freelist(cur->bc_tp, cur->bc_private.a.agbp,			&nbno)))		return error;	/*	 * None available, we fail.	 */	if (nbno == NULLAGBLOCK) {		*stat = 0;		return 0;	}	xfs_trans_agbtree_delta(cur->bc_tp, 1);	nbp = xfs_btree_get_bufs(mp, cur->bc_tp, cur->bc_private.a.agno, nbno,		0);	new = XFS_BUF_TO_ALLOC_BLOCK(nbp);	/*	 * Set the root data in the a.g. freespace structure.	 */	{		xfs_agf_t	*agf;	/* a.g. freespace header */		xfs_agnumber_t	seqno;		agf = XFS_BUF_TO_AGF(cur->bc_private.a.agbp);		INT_SET(agf->agf_roots[cur->bc_btnum], ARCH_CONVERT, nbno);		INT_MOD(agf->agf_levels[cur->bc_btnum], ARCH_CONVERT, 1);		seqno = INT_GET(agf->agf_seqno, ARCH_CONVERT);		mp->m_perag[seqno].pagf_levels[cur->bc_btnum]++;		xfs_alloc_log_agf(cur->bc_tp, cur->bc_private.a.agbp,			XFS_AGF_ROOTS | XFS_AGF_LEVELS);	}	/*	 * At the previous root level there are now two blocks: the old	 * root, and the new block generated when it was split.	 * We don't know which one the cursor is pointing at, so we	 * set up variables "left" and "right" for each case.	 */	lbp = cur->bc_bufs[cur->bc_nlevels - 1];	left = XFS_BUF_TO_ALLOC_BLOCK(lbp);#ifdef DEBUG	if ((error = xfs_btree_check_sblock(cur, left, cur->bc_nlevels - 1, lbp)))		return error;#endif	if (INT_GET(left->bb_rightsib, ARCH_CONVERT) != NULLAGBLOCK) {		/*		 * Our block is left, pick up the right block.		 */		lbno = XFS_DADDR_TO_AGBNO(mp, XFS_BUF_ADDR(lbp));		rbno = INT_GET(left->bb_rightsib, ARCH_CONVERT);		if ((error = xfs_btree_read_bufs(mp, cur->bc_tp,				cur->bc_private.a.agno, rbno, 0, &rbp,				XFS_ALLOC_BTREE_REF)))			return error;		right = XFS_BUF_TO_ALLOC_BLOCK(rbp);		if ((error = xfs_btree_check_sblock(cur, right,				cur->bc_nlevels - 1, rbp)))			return error;		nptr = 1;	} else {		/*		 * Our block is right, pick up the left block.		 */		rbp = lbp;		right = left;		rbno = XFS_DADDR_TO_AGBNO(mp, XFS_BUF_ADDR(rbp));		lbno = INT_GET(right->bb_leftsib, ARCH_CONVERT);		if ((error = xfs_btree_read_bufs(mp, cur->bc_tp,				cur->bc_private.a.agno, lbno, 0, &lbp,				XFS_ALLOC_BTREE_REF)))			return error;		left = XFS_BUF_TO_ALLOC_BLOCK(lbp);		if ((error = xfs_btree_check_sblock(cur, left,				cur->bc_nlevels - 1, lbp)))			return error;		nptr = 2;	}	/*	 * Fill in the new block's btree header and log it.	 */	INT_SET(new->bb_magic, ARCH_CONVERT, xfs_magics[cur->bc_btnum]);	INT_SET(new->bb_level, ARCH_CONVERT, (__uint16_t)cur->bc_nlevels);	INT_SET(new->bb_numrecs, ARCH_CONVERT, 2);	INT_SET(new->bb_leftsib, ARCH_CONVERT, NULLAGBLOCK);	INT_SET(new->bb_rightsib, ARCH_CONVERT, NULLAGBLOCK);	xfs_alloc_log_block(cur->bc_tp, nbp, XFS_BB_ALL_BITS);	ASSERT(lbno != NULLAGBLOCK && rbno != NULLAGBLOCK);	/*	 * Fill in the key data in the new root.	 */	{		xfs_alloc_key_t		*kp;	/* btree key pointer */		kp = XFS_ALLOC_KEY_ADDR(new, 1, cur);		if (INT_GET(left->bb_level, ARCH_CONVERT) > 0) {			kp[0] = *XFS_ALLOC_KEY_ADDR(left, 1, cur); /* INT_: structure copy */			kp[1] = *XFS_ALLOC_KEY_ADDR(right, 1, cur);/* INT_: structure copy */		} else {			xfs_alloc_rec_t	*rp;	/* btree record pointer */			rp = XFS_ALLOC_REC_ADDR(left, 1, cur);			kp[0].ar_startblock = rp->ar_startblock; /* INT_: direct copy */			kp[0].ar_blockcount = rp->ar_blockcount; /* INT_: direct copy */			rp = XFS_ALLOC_REC_ADDR(right, 1, cur);			kp[1].ar_startblock = rp->ar_startblock; /* INT_: direct copy */			kp[1].ar_blockcount = rp->ar_blockcount; /* INT_: direct copy */		}	}	xfs_alloc_log_keys(cur, nbp, 1, 2);	/*	 * Fill in the pointer data in the new root.	 */	{		xfs_alloc_ptr_t		*pp;	/* btree address pointer */		pp = XFS_ALLOC_PTR_ADDR(new, 1, cur);		INT_SET(pp[0], ARCH_CONVERT, lbno);		INT_SET(pp[1], ARCH_CONVERT, rbno);	}	xfs_alloc_log_ptrs(cur, nbp, 1, 2);	/*	 * Fix up the cursor.	 */	xfs_btree_setbuf(cur, cur->bc_nlevels, nbp);	cur->bc_ptrs[cur->bc_nlevels] = nptr;	cur->bc_nlevels++;	*stat = 1;	return 0;}/* * Move 1 record right from cur/level if possible. * Update cur to reflect the new path. */STATIC int				/* error */xfs_alloc_rshift(	xfs_btree_cur_t		*cur,	/* btree cursor */	int			level,	/* level to shift record on */	int			*stat)	/* success/failure */{	int			error;	/* error return value */	int			i;	/* loop index */	xfs_alloc_key_t		key;	/* key value for leaf level upward */	xfs_buf_t		*lbp;	/* buffer for left (current) block */	xfs_alloc_block_t	*left;	/* left (current) btree block */	xfs_buf_t		*rbp;	/* buffer for right neighbor block */	xfs_alloc_block_t	*right;	/* right neighbor btree block */	xfs_alloc_key_t		*rkp;	/* key pointer for right block */	xfs_btree_cur_t		*tcur;	/* temporary cursor */	/*	 * Set up variables for this block as "left".	 */	lbp = cur->bc_bufs[level];	left = XFS_BUF_TO_ALLOC_BLOCK(lbp);#ifdef DEBUG	if ((error = xfs_btree_check_sblock(cur, left, level, lbp)))		return error;#endif	/*	 * If we've got no right sibling then we can't shift an entry right.	 */	if (INT_GET(left->bb_rightsib, ARCH_CONVERT) == NULLAGBLOCK) {		*stat = 0;		return 0;	}	/*	 * If the cursor entry is the one that would be moved, don't	 * do it... it's too complicated.	 */	if (cur->bc_ptrs[level] >= INT_GET(left->bb_numrecs, ARCH_CONVERT)) {		*stat = 0;		return 0;	}	/*	 * Set up the right neighbor as "right".	 */	if ((error = xfs_btree_read_bufs(cur->bc_mp, cur->bc_tp,			cur->bc_private.a.agno, INT_GET(left->bb_rightsib, ARCH_CONVERT), 0, &rbp,			XFS_ALLOC_BTREE_REF)))		return error;	right = XFS_BUF_TO_ALLOC_BLOCK(rbp);	if ((error = xfs_btree_check_sblock(cur, right, level, rbp)))		return error;	/*	 * If it's full, it can't take another entry.	 */	if (INT_GET(right->bb_numrecs, ARCH_CONVERT) == XFS_ALLOC_BLOCK_MAXRECS(level, cur)) {		*stat = 0;		return 0;	}	/*	 * Make a hole at the start of the right neighbor block, then	 * copy the last left block entry to the hole.	 */	if (level > 0) {		xfs_alloc_key_t	*lkp;	/* key pointer for left block */		xfs_alloc_ptr_t	*lpp;	/* address pointer for left block */		xfs_alloc_ptr_t	*rpp;	/* address pointer for right block */		lkp = XFS_ALLOC_KEY_ADDR(left, INT_GET(left->bb_numrecs, ARCH_CONVERT), cur);		lpp = XFS_ALLOC_PTR_ADDR(left, INT_GET(left->bb_numrecs, ARCH_CONVERT), cur);		rkp = XFS_ALLOC_KEY_ADDR(right, 1, cur);		rpp = XFS_ALLOC_PTR_ADDR(right, 1, cur);#ifdef DEBUG		for (i = INT_GET(right->bb_numrecs, ARCH_CONVERT) - 1; i >= 0; i--) {			if ((error = xfs_btree_check_sptr(cur, INT_GET(rpp[i], ARCH_CONVERT), level)))				return error;		}#endif		memmove(rkp + 1, rkp, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rkp));		memmove(rpp + 1, rpp, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rpp));#ifdef DEBUG		if ((error = xfs_btree_check_sptr(cur, INT_GET(*lpp, ARCH_CONVERT), level)))			return error;#endif		*rkp = *lkp; /* INT_: copy */		*rpp = *lpp; /* INT_: copy */		xfs_alloc_log_keys(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) + 1);		xfs_alloc_log_ptrs(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) + 1);		xfs_btree_check_key(cur->bc_btnum, rkp, rkp + 1);	} else {		xfs_alloc_rec_t	*lrp;	/* record pointer for left block */		xfs_alloc_rec_t	*rrp;	/* record pointer for right block */		lrp = XFS_ALLOC_REC_ADDR(left, INT_GET(left->bb_numrecs, ARCH_CONVERT), cur);		rrp = XFS_ALLOC_REC_ADDR(right, 1, cur);		memmove(rrp + 1, rrp, INT_GET(right->bb_numrecs, ARCH_CONVERT) * sizeof(*rrp));		*rrp = *lrp;		xfs_alloc_log_recs(cur, rbp, 1, INT_GET(right->bb_numrecs, ARCH_CONVERT) + 1);		key.ar_startblock = rrp->ar_startblock; /* INT_: direct copy */		key.ar_blockcount = rrp->ar_blockcount; /* INT_: direct copy */		rkp = &key;		xfs_btree_check_rec(cur->bc_btnum, rrp, rrp + 1);	}	/*	 * Decrement and log left's numrecs, bump and log right's numrecs.	 */	INT_MOD(left->bb_numrecs, ARCH_CONVERT, -1);	xfs_alloc_log_block(cur->bc_tp, lbp, XFS_BB_NUMRECS);	INT_MOD(right->bb_numrecs, ARCH_CONVERT, +1);	xfs_alloc_log_block(cur->bc_tp, rbp, XFS_BB_NUMRECS);	/*	 * Using a temporary cursor, update the parent key values of the	 * block on the right.	 */	if ((error = xfs_btree_dup_cursor(cur, &tcur)))		return error;	i = xfs_btree_lastrec(tcur, level);	XFS_WANT_CORRUPTED_GOTO(i == 1, error0);	if ((error = xfs_alloc_increment(tcur, level, &i)) ||	    (error = xfs_alloc_updkey(tcur, rkp, level + 1)))		goto error0;	xfs_btree_del_cursor(tcur, XFS_BTREE_NOERROR);	*stat = 1;	return 0;error0:	xfs_btree_del_cursor(tcur, XFS_BTREE_ERROR);	return error;}/* * Split cur/level block in half. * Return new block number and its first record (to be inserted into parent). */STATIC int				/* error */xfs_alloc_split(	xfs_btree_cur_t		*cur,	/* btree cursor */	int			level,	/* level to split */	xfs_agblock_t		*bnop,	/* output: block number allocated */	xfs_alloc_key_t		*keyp,	/* output: first key of new block */	xfs_btree_cur_t		**curp,	/* output: new cursor */	int			*stat)	/* success/failure */{	int			error;	/* error return value */	int			i;	/* loop index/record number */	xfs_agblock_t		lbno;	/* left (current) block number */	xfs_buf_t		*lbp;	/* buffer for left block */	xfs_alloc_block_t	*left;	/* left (current) btree block */	xfs_agblock_t		rbno;	/* right (new) block number */	xfs_buf_t		*rbp;	/* buffer for right block */	xfs_alloc_block_t	*right;	/* right (new) btree block */	/*	 * Allocate the new block from the freelist.	 * If we can't do it, we're toast.  Give up.	 */	if ((error = xfs_alloc_get_freelist(cur->bc_tp, cur->bc_private.a.agbp,			&rbno)))		return error;	if (rbno == NULLAGBLOCK) {		*stat = 0;		return 0;	}	xfs_trans_agbtree_delta(cur->bc_tp, 1);	rbp = xfs_btree_get_bufs(cur->bc_mp, cur->bc_tp, cur->bc_private.a.agno,		rbno, 0);	/*	 * Set up the new block as "right".	 */	right = XFS_BUF_TO_ALLOC_BLOCK(rbp);	/*	 * "Left" is the current (according to the cursor) block.	 */	lbp = cur->bc_bufs[level];	left = XFS_BUF_TO_ALLOC_BLOCK(lbp);#ifdef DEBUG	if ((error = xfs_btree_check_sblock(cur, left, level, lbp)))		return error;#endif	/*	 * Fill in the btree header for the new block.	 */	INT_SET(right->bb_magic, ARCH_CONVERT, xfs_magics[cur->bc_btnum]);	right->bb_level = left->bb_level; /* INT_: direct copy */	INT_SET(right->bb_numrecs, ARCH_CONVERT, (__uint16_t)(INT_GET(left->bb_numrecs, ARCH_CONVERT) / 2));	/*	 * Make sure that if there's an odd number of entries now, that	 * each new block will have the same number of entries.	 */	if ((INT_GET(left->bb_numrecs, ARCH_CONVERT) & 1) &&	    cur->bc_ptrs[level] <= INT_GET(right->bb_numrecs, ARCH_CONVERT) + 1)		INT_MOD(right->bb_numrecs, ARCH_CONVERT, +1);	i = INT_GET(left->bb_numrecs, ARCH_CONVERT) - INT_GET(right->bb_numrecs, ARCH_CONVERT) + 1;	/*	 * For non-leaf blocks, copy keys and addresses over to the new block.	 */	if (level > 0) {		xfs_alloc_key_t	*lkp;	/* left btree key pointer */		xfs_alloc_ptr_t	*lpp;	/* left btree address pointer */		xfs_alloc_key_t	*rkp;	/* right btree key pointer */		xfs_alloc_ptr_t	*rpp;	/* right btree address pointer */		lkp = XFS_ALLOC_KEY_ADDR(left, i, cur);		lpp = XFS_ALLOC_PTR_ADDR(left, i, cur);		rkp = XFS_ALLOC_KEY_ADDR(right, 1, cur);		rpp = XFS_ALLOC_PTR_ADDR(right, 1, cur);#ifdef DEBUG		for (i = 0; i < INT_GET(right->bb_numrecs, ARCH_CONVERT); i++) {			if ((error = xfs_btree_check_sptr(cur, INT_GET(lpp[i], ARCH_CONVERT), level)))

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?