xref: /openbmc/linux/fs/xfs/xfs_bmap_item.c (revision 680776e5)
10b61f8a4SDave Chinner // SPDX-License-Identifier: GPL-2.0+
26413a014SDarrick J. Wong /*
36413a014SDarrick J. Wong  * Copyright (C) 2016 Oracle.  All Rights Reserved.
46413a014SDarrick J. Wong  * Author: Darrick J. Wong <darrick.wong@oracle.com>
56413a014SDarrick J. Wong  */
66413a014SDarrick J. Wong #include "xfs.h"
76413a014SDarrick J. Wong #include "xfs_fs.h"
86413a014SDarrick J. Wong #include "xfs_format.h"
96413a014SDarrick J. Wong #include "xfs_log_format.h"
106413a014SDarrick J. Wong #include "xfs_trans_resv.h"
1177d61fe4SDarrick J. Wong #include "xfs_bit.h"
125467b34bSDarrick J. Wong #include "xfs_shared.h"
136413a014SDarrick J. Wong #include "xfs_mount.h"
1477d61fe4SDarrick J. Wong #include "xfs_defer.h"
1577d61fe4SDarrick J. Wong #include "xfs_inode.h"
166413a014SDarrick J. Wong #include "xfs_trans.h"
176413a014SDarrick J. Wong #include "xfs_trans_priv.h"
186413a014SDarrick J. Wong #include "xfs_bmap_item.h"
196413a014SDarrick J. Wong #include "xfs_log.h"
2077d61fe4SDarrick J. Wong #include "xfs_bmap.h"
2177d61fe4SDarrick J. Wong #include "xfs_icache.h"
22fe0be23eSDarrick J. Wong #include "xfs_bmap_btree.h"
23fe0be23eSDarrick J. Wong #include "xfs_trans_space.h"
24a5155b87SDarrick J. Wong #include "xfs_error.h"
253c6ba3cfSDarrick J. Wong #include "xfs_log_priv.h"
2686ffa471SDarrick J. Wong #include "xfs_log_recover.h"
27774a99b4SDarrick J. Wong #include "xfs_ag.h"
286413a014SDarrick J. Wong 
29182696fbSDarrick J. Wong struct kmem_cache	*xfs_bui_cache;
30182696fbSDarrick J. Wong struct kmem_cache	*xfs_bud_cache;
316413a014SDarrick J. Wong 
329329ba89SDarrick J. Wong static const struct xfs_item_ops xfs_bui_item_ops;
339329ba89SDarrick J. Wong 
BUI_ITEM(struct xfs_log_item * lip)346413a014SDarrick J. Wong static inline struct xfs_bui_log_item *BUI_ITEM(struct xfs_log_item *lip)
356413a014SDarrick J. Wong {
366413a014SDarrick J. Wong 	return container_of(lip, struct xfs_bui_log_item, bui_item);
376413a014SDarrick J. Wong }
386413a014SDarrick J. Wong 
393c6ba3cfSDarrick J. Wong STATIC void
xfs_bui_item_free(struct xfs_bui_log_item * buip)406413a014SDarrick J. Wong xfs_bui_item_free(
416413a014SDarrick J. Wong 	struct xfs_bui_log_item	*buip)
426413a014SDarrick J. Wong {
43c230a4a8SDave Chinner 	kmem_free(buip->bui_item.li_lv_shadow);
44182696fbSDarrick J. Wong 	kmem_cache_free(xfs_bui_cache, buip);
456413a014SDarrick J. Wong }
466413a014SDarrick J. Wong 
470612d116SDave Chinner /*
480612d116SDave Chinner  * Freeing the BUI requires that we remove it from the AIL if it has already
490612d116SDave Chinner  * been placed there. However, the BUI may not yet have been placed in the AIL
500612d116SDave Chinner  * when called by xfs_bui_release() from BUD processing due to the ordering of
510612d116SDave Chinner  * committed vs unpin operations in bulk insert operations. Hence the reference
520612d116SDave Chinner  * count to ensure only the last caller frees the BUI.
530612d116SDave Chinner  */
549329ba89SDarrick J. Wong STATIC void
xfs_bui_release(struct xfs_bui_log_item * buip)550612d116SDave Chinner xfs_bui_release(
560612d116SDave Chinner 	struct xfs_bui_log_item	*buip)
570612d116SDave Chinner {
580612d116SDave Chinner 	ASSERT(atomic_read(&buip->bui_refcount) > 0);
593512fc1eSDave Chinner 	if (!atomic_dec_and_test(&buip->bui_refcount))
603512fc1eSDave Chinner 		return;
613512fc1eSDave Chinner 
623512fc1eSDave Chinner 	xfs_trans_ail_delete(&buip->bui_item, 0);
630612d116SDave Chinner 	xfs_bui_item_free(buip);
640612d116SDave Chinner }
650612d116SDave Chinner 
660612d116SDave Chinner 
676413a014SDarrick J. Wong STATIC void
xfs_bui_item_size(struct xfs_log_item * lip,int * nvecs,int * nbytes)686413a014SDarrick J. Wong xfs_bui_item_size(
696413a014SDarrick J. Wong 	struct xfs_log_item	*lip,
706413a014SDarrick J. Wong 	int			*nvecs,
716413a014SDarrick J. Wong 	int			*nbytes)
726413a014SDarrick J. Wong {
736413a014SDarrick J. Wong 	struct xfs_bui_log_item	*buip = BUI_ITEM(lip);
746413a014SDarrick J. Wong 
756413a014SDarrick J. Wong 	*nvecs += 1;
766413a014SDarrick J. Wong 	*nbytes += xfs_bui_log_format_sizeof(buip->bui_format.bui_nextents);
776413a014SDarrick J. Wong }
786413a014SDarrick J. Wong 
796413a014SDarrick J. Wong /*
806413a014SDarrick J. Wong  * This is called to fill in the vector of log iovecs for the
816413a014SDarrick J. Wong  * given bui log item. We use only 1 iovec, and we point that
826413a014SDarrick J. Wong  * at the bui_log_format structure embedded in the bui item.
836413a014SDarrick J. Wong  * It is at this point that we assert that all of the extent
846413a014SDarrick J. Wong  * slots in the bui item have been filled.
856413a014SDarrick J. Wong  */
866413a014SDarrick J. Wong STATIC void
xfs_bui_item_format(struct xfs_log_item * lip,struct xfs_log_vec * lv)876413a014SDarrick J. Wong xfs_bui_item_format(
886413a014SDarrick J. Wong 	struct xfs_log_item	*lip,
896413a014SDarrick J. Wong 	struct xfs_log_vec	*lv)
906413a014SDarrick J. Wong {
916413a014SDarrick J. Wong 	struct xfs_bui_log_item	*buip = BUI_ITEM(lip);
926413a014SDarrick J. Wong 	struct xfs_log_iovec	*vecp = NULL;
936413a014SDarrick J. Wong 
946413a014SDarrick J. Wong 	ASSERT(atomic_read(&buip->bui_next_extent) ==
956413a014SDarrick J. Wong 			buip->bui_format.bui_nextents);
966413a014SDarrick J. Wong 
976413a014SDarrick J. Wong 	buip->bui_format.bui_type = XFS_LI_BUI;
986413a014SDarrick J. Wong 	buip->bui_format.bui_size = 1;
996413a014SDarrick J. Wong 
1006413a014SDarrick J. Wong 	xlog_copy_iovec(lv, &vecp, XLOG_REG_TYPE_BUI_FORMAT, &buip->bui_format,
1016413a014SDarrick J. Wong 			xfs_bui_log_format_sizeof(buip->bui_format.bui_nextents));
1026413a014SDarrick J. Wong }
1036413a014SDarrick J. Wong 
1046413a014SDarrick J. Wong /*
1056413a014SDarrick J. Wong  * The unpin operation is the last place an BUI is manipulated in the log. It is
1066413a014SDarrick J. Wong  * either inserted in the AIL or aborted in the event of a log I/O error. In
1076413a014SDarrick J. Wong  * either case, the BUI transaction has been successfully committed to make it
1086413a014SDarrick J. Wong  * this far. Therefore, we expect whoever committed the BUI to either construct
1096413a014SDarrick J. Wong  * and commit the BUD or drop the BUD's reference in the event of error. Simply
1106413a014SDarrick J. Wong  * drop the log's BUI reference now that the log is done with it.
1116413a014SDarrick J. Wong  */
1126413a014SDarrick J. Wong STATIC void
xfs_bui_item_unpin(struct xfs_log_item * lip,int remove)1136413a014SDarrick J. Wong xfs_bui_item_unpin(
1146413a014SDarrick J. Wong 	struct xfs_log_item	*lip,
1156413a014SDarrick J. Wong 	int			remove)
1166413a014SDarrick J. Wong {
1176413a014SDarrick J. Wong 	struct xfs_bui_log_item	*buip = BUI_ITEM(lip);
1186413a014SDarrick J. Wong 
1196413a014SDarrick J. Wong 	xfs_bui_release(buip);
1206413a014SDarrick J. Wong }
1216413a014SDarrick J. Wong 
1226413a014SDarrick J. Wong /*
1236413a014SDarrick J. Wong  * The BUI has been either committed or aborted if the transaction has been
1246413a014SDarrick J. Wong  * cancelled. If the transaction was cancelled, an BUD isn't going to be
1256413a014SDarrick J. Wong  * constructed and thus we free the BUI here directly.
1266413a014SDarrick J. Wong  */
1276413a014SDarrick J. Wong STATIC void
xfs_bui_item_release(struct xfs_log_item * lip)128ddf92053SChristoph Hellwig xfs_bui_item_release(
1296413a014SDarrick J. Wong 	struct xfs_log_item	*lip)
1306413a014SDarrick J. Wong {
1310612d116SDave Chinner 	xfs_bui_release(BUI_ITEM(lip));
1326413a014SDarrick J. Wong }
1336413a014SDarrick J. Wong 
1346413a014SDarrick J. Wong /*
1356413a014SDarrick J. Wong  * Allocate and initialize an bui item with the given number of extents.
1366413a014SDarrick J. Wong  */
1373c6ba3cfSDarrick J. Wong STATIC struct xfs_bui_log_item *
xfs_bui_init(struct xfs_mount * mp)1386413a014SDarrick J. Wong xfs_bui_init(
1396413a014SDarrick J. Wong 	struct xfs_mount		*mp)
1406413a014SDarrick J. Wong 
1416413a014SDarrick J. Wong {
1426413a014SDarrick J. Wong 	struct xfs_bui_log_item		*buip;
1436413a014SDarrick J. Wong 
144182696fbSDarrick J. Wong 	buip = kmem_cache_zalloc(xfs_bui_cache, GFP_KERNEL | __GFP_NOFAIL);
1456413a014SDarrick J. Wong 
1466413a014SDarrick J. Wong 	xfs_log_item_init(mp, &buip->bui_item, XFS_LI_BUI, &xfs_bui_item_ops);
1476413a014SDarrick J. Wong 	buip->bui_format.bui_nextents = XFS_BUI_MAX_FAST_EXTENTS;
1486413a014SDarrick J. Wong 	buip->bui_format.bui_id = (uintptr_t)(void *)buip;
1496413a014SDarrick J. Wong 	atomic_set(&buip->bui_next_extent, 0);
1506413a014SDarrick J. Wong 	atomic_set(&buip->bui_refcount, 2);
1516413a014SDarrick J. Wong 
1526413a014SDarrick J. Wong 	return buip;
1536413a014SDarrick J. Wong }
1546413a014SDarrick J. Wong 
BUD_ITEM(struct xfs_log_item * lip)1556413a014SDarrick J. Wong static inline struct xfs_bud_log_item *BUD_ITEM(struct xfs_log_item *lip)
1566413a014SDarrick J. Wong {
1576413a014SDarrick J. Wong 	return container_of(lip, struct xfs_bud_log_item, bud_item);
1586413a014SDarrick J. Wong }
1596413a014SDarrick J. Wong 
1606413a014SDarrick J. Wong STATIC void
xfs_bud_item_size(struct xfs_log_item * lip,int * nvecs,int * nbytes)1616413a014SDarrick J. Wong xfs_bud_item_size(
1626413a014SDarrick J. Wong 	struct xfs_log_item	*lip,
1636413a014SDarrick J. Wong 	int			*nvecs,
1646413a014SDarrick J. Wong 	int			*nbytes)
1656413a014SDarrick J. Wong {
1666413a014SDarrick J. Wong 	*nvecs += 1;
1676413a014SDarrick J. Wong 	*nbytes += sizeof(struct xfs_bud_log_format);
1686413a014SDarrick J. Wong }
1696413a014SDarrick J. Wong 
1706413a014SDarrick J. Wong /*
1716413a014SDarrick J. Wong  * This is called to fill in the vector of log iovecs for the
1726413a014SDarrick J. Wong  * given bud log item. We use only 1 iovec, and we point that
1736413a014SDarrick J. Wong  * at the bud_log_format structure embedded in the bud item.
1746413a014SDarrick J. Wong  * It is at this point that we assert that all of the extent
1756413a014SDarrick J. Wong  * slots in the bud item have been filled.
1766413a014SDarrick J. Wong  */
1776413a014SDarrick J. Wong STATIC void
xfs_bud_item_format(struct xfs_log_item * lip,struct xfs_log_vec * lv)1786413a014SDarrick J. Wong xfs_bud_item_format(
1796413a014SDarrick J. Wong 	struct xfs_log_item	*lip,
1806413a014SDarrick J. Wong 	struct xfs_log_vec	*lv)
1816413a014SDarrick J. Wong {
1826413a014SDarrick J. Wong 	struct xfs_bud_log_item	*budp = BUD_ITEM(lip);
1836413a014SDarrick J. Wong 	struct xfs_log_iovec	*vecp = NULL;
1846413a014SDarrick J. Wong 
1856413a014SDarrick J. Wong 	budp->bud_format.bud_type = XFS_LI_BUD;
1866413a014SDarrick J. Wong 	budp->bud_format.bud_size = 1;
1876413a014SDarrick J. Wong 
1886413a014SDarrick J. Wong 	xlog_copy_iovec(lv, &vecp, XLOG_REG_TYPE_BUD_FORMAT, &budp->bud_format,
1896413a014SDarrick J. Wong 			sizeof(struct xfs_bud_log_format));
1906413a014SDarrick J. Wong }
1916413a014SDarrick J. Wong 
1926413a014SDarrick J. Wong /*
1936413a014SDarrick J. Wong  * The BUD is either committed or aborted if the transaction is cancelled. If
1946413a014SDarrick J. Wong  * the transaction is cancelled, drop our reference to the BUI and free the
1956413a014SDarrick J. Wong  * BUD.
1966413a014SDarrick J. Wong  */
1976413a014SDarrick J. Wong STATIC void
xfs_bud_item_release(struct xfs_log_item * lip)198ddf92053SChristoph Hellwig xfs_bud_item_release(
1996413a014SDarrick J. Wong 	struct xfs_log_item	*lip)
2006413a014SDarrick J. Wong {
2016413a014SDarrick J. Wong 	struct xfs_bud_log_item	*budp = BUD_ITEM(lip);
2026413a014SDarrick J. Wong 
2036413a014SDarrick J. Wong 	xfs_bui_release(budp->bud_buip);
204c230a4a8SDave Chinner 	kmem_free(budp->bud_item.li_lv_shadow);
205182696fbSDarrick J. Wong 	kmem_cache_free(xfs_bud_cache, budp);
2066413a014SDarrick J. Wong }
2076413a014SDarrick J. Wong 
208c23ab603SDave Chinner static struct xfs_log_item *
xfs_bud_item_intent(struct xfs_log_item * lip)209c23ab603SDave Chinner xfs_bud_item_intent(
210c23ab603SDave Chinner 	struct xfs_log_item	*lip)
211c23ab603SDave Chinner {
212c23ab603SDave Chinner 	return &BUD_ITEM(lip)->bud_buip->bui_item;
213c23ab603SDave Chinner }
214c23ab603SDave Chinner 
2156413a014SDarrick J. Wong static const struct xfs_item_ops xfs_bud_item_ops = {
216f5b81200SDave Chinner 	.flags		= XFS_ITEM_RELEASE_WHEN_COMMITTED |
217f5b81200SDave Chinner 			  XFS_ITEM_INTENT_DONE,
2186413a014SDarrick J. Wong 	.iop_size	= xfs_bud_item_size,
2196413a014SDarrick J. Wong 	.iop_format	= xfs_bud_item_format,
220ddf92053SChristoph Hellwig 	.iop_release	= xfs_bud_item_release,
221c23ab603SDave Chinner 	.iop_intent	= xfs_bud_item_intent,
2226413a014SDarrick J. Wong };
2236413a014SDarrick J. Wong 
224caeaea98SChristoph Hellwig static struct xfs_bud_log_item *
xfs_trans_get_bud(struct xfs_trans * tp,struct xfs_bui_log_item * buip)22573f0d236SChristoph Hellwig xfs_trans_get_bud(
22673f0d236SChristoph Hellwig 	struct xfs_trans		*tp,
2276413a014SDarrick J. Wong 	struct xfs_bui_log_item		*buip)
2286413a014SDarrick J. Wong {
2296413a014SDarrick J. Wong 	struct xfs_bud_log_item		*budp;
2306413a014SDarrick J. Wong 
231182696fbSDarrick J. Wong 	budp = kmem_cache_zalloc(xfs_bud_cache, GFP_KERNEL | __GFP_NOFAIL);
23273f0d236SChristoph Hellwig 	xfs_log_item_init(tp->t_mountp, &budp->bud_item, XFS_LI_BUD,
23373f0d236SChristoph Hellwig 			  &xfs_bud_item_ops);
2346413a014SDarrick J. Wong 	budp->bud_buip = buip;
2356413a014SDarrick J. Wong 	budp->bud_format.bud_bui_id = buip->bui_format.bui_id;
2366413a014SDarrick J. Wong 
23773f0d236SChristoph Hellwig 	xfs_trans_add_item(tp, &budp->bud_item);
2386413a014SDarrick J. Wong 	return budp;
2396413a014SDarrick J. Wong }
24077d61fe4SDarrick J. Wong 
24177d61fe4SDarrick J. Wong /*
242caeaea98SChristoph Hellwig  * Finish an bmap update and log it to the BUD. Note that the
243caeaea98SChristoph Hellwig  * transaction is marked dirty regardless of whether the bmap update
244caeaea98SChristoph Hellwig  * succeeds or fails to support the BUI/BUD lifecycle rules.
245caeaea98SChristoph Hellwig  */
246caeaea98SChristoph Hellwig static int
xfs_trans_log_finish_bmap_update(struct xfs_trans * tp,struct xfs_bud_log_item * budp,struct xfs_bmap_intent * bi)247caeaea98SChristoph Hellwig xfs_trans_log_finish_bmap_update(
248caeaea98SChristoph Hellwig 	struct xfs_trans		*tp,
249caeaea98SChristoph Hellwig 	struct xfs_bud_log_item		*budp,
250ddccb81bSDarrick J. Wong 	struct xfs_bmap_intent		*bi)
251caeaea98SChristoph Hellwig {
252caeaea98SChristoph Hellwig 	int				error;
253caeaea98SChristoph Hellwig 
254ddccb81bSDarrick J. Wong 	error = xfs_bmap_finish_one(tp, bi);
255caeaea98SChristoph Hellwig 
256caeaea98SChristoph Hellwig 	/*
257caeaea98SChristoph Hellwig 	 * Mark the transaction dirty, even on error. This ensures the
258caeaea98SChristoph Hellwig 	 * transaction is aborted, which:
259caeaea98SChristoph Hellwig 	 *
260caeaea98SChristoph Hellwig 	 * 1.) releases the BUI and frees the BUD
261caeaea98SChristoph Hellwig 	 * 2.) shuts down the filesystem
262caeaea98SChristoph Hellwig 	 */
263bb7b1c9cSDave Chinner 	tp->t_flags |= XFS_TRANS_DIRTY | XFS_TRANS_HAS_INTENT_DONE;
264caeaea98SChristoph Hellwig 	set_bit(XFS_LI_DIRTY, &budp->bud_item.li_flags);
265caeaea98SChristoph Hellwig 
266caeaea98SChristoph Hellwig 	return error;
267caeaea98SChristoph Hellwig }
268caeaea98SChristoph Hellwig 
269caeaea98SChristoph Hellwig /* Sort bmap intents by inode. */
270caeaea98SChristoph Hellwig static int
xfs_bmap_update_diff_items(void * priv,const struct list_head * a,const struct list_head * b)271caeaea98SChristoph Hellwig xfs_bmap_update_diff_items(
272caeaea98SChristoph Hellwig 	void				*priv,
2734f0f586bSSami Tolvanen 	const struct list_head		*a,
2744f0f586bSSami Tolvanen 	const struct list_head		*b)
275caeaea98SChristoph Hellwig {
276caeaea98SChristoph Hellwig 	struct xfs_bmap_intent		*ba;
277caeaea98SChristoph Hellwig 	struct xfs_bmap_intent		*bb;
278caeaea98SChristoph Hellwig 
279caeaea98SChristoph Hellwig 	ba = container_of(a, struct xfs_bmap_intent, bi_list);
280caeaea98SChristoph Hellwig 	bb = container_of(b, struct xfs_bmap_intent, bi_list);
281caeaea98SChristoph Hellwig 	return ba->bi_owner->i_ino - bb->bi_owner->i_ino;
282caeaea98SChristoph Hellwig }
283caeaea98SChristoph Hellwig 
284caeaea98SChristoph Hellwig /* Set the map extent flags for this mapping. */
285caeaea98SChristoph Hellwig static void
xfs_trans_set_bmap_flags(struct xfs_map_extent * map,enum xfs_bmap_intent_type type,int whichfork,xfs_exntst_t state)286caeaea98SChristoph Hellwig xfs_trans_set_bmap_flags(
287f3ebac4cSDarrick J. Wong 	struct xfs_map_extent		*map,
288caeaea98SChristoph Hellwig 	enum xfs_bmap_intent_type	type,
289caeaea98SChristoph Hellwig 	int				whichfork,
290caeaea98SChristoph Hellwig 	xfs_exntst_t			state)
291caeaea98SChristoph Hellwig {
292f3ebac4cSDarrick J. Wong 	map->me_flags = 0;
293caeaea98SChristoph Hellwig 	switch (type) {
294caeaea98SChristoph Hellwig 	case XFS_BMAP_MAP:
295caeaea98SChristoph Hellwig 	case XFS_BMAP_UNMAP:
296f3ebac4cSDarrick J. Wong 		map->me_flags = type;
297caeaea98SChristoph Hellwig 		break;
298caeaea98SChristoph Hellwig 	default:
299caeaea98SChristoph Hellwig 		ASSERT(0);
300caeaea98SChristoph Hellwig 	}
301caeaea98SChristoph Hellwig 	if (state == XFS_EXT_UNWRITTEN)
302f3ebac4cSDarrick J. Wong 		map->me_flags |= XFS_BMAP_EXTENT_UNWRITTEN;
303caeaea98SChristoph Hellwig 	if (whichfork == XFS_ATTR_FORK)
304f3ebac4cSDarrick J. Wong 		map->me_flags |= XFS_BMAP_EXTENT_ATTR_FORK;
305caeaea98SChristoph Hellwig }
306caeaea98SChristoph Hellwig 
307caeaea98SChristoph Hellwig /* Log bmap updates in the intent item. */
308caeaea98SChristoph Hellwig STATIC void
xfs_bmap_update_log_item(struct xfs_trans * tp,struct xfs_bui_log_item * buip,struct xfs_bmap_intent * bi)309caeaea98SChristoph Hellwig xfs_bmap_update_log_item(
310caeaea98SChristoph Hellwig 	struct xfs_trans		*tp,
311c1f09188SChristoph Hellwig 	struct xfs_bui_log_item		*buip,
312f3ebac4cSDarrick J. Wong 	struct xfs_bmap_intent		*bi)
313caeaea98SChristoph Hellwig {
314caeaea98SChristoph Hellwig 	uint				next_extent;
315caeaea98SChristoph Hellwig 	struct xfs_map_extent		*map;
316caeaea98SChristoph Hellwig 
317caeaea98SChristoph Hellwig 	tp->t_flags |= XFS_TRANS_DIRTY;
318caeaea98SChristoph Hellwig 	set_bit(XFS_LI_DIRTY, &buip->bui_item.li_flags);
319caeaea98SChristoph Hellwig 
320caeaea98SChristoph Hellwig 	/*
321caeaea98SChristoph Hellwig 	 * atomic_inc_return gives us the value after the increment;
322caeaea98SChristoph Hellwig 	 * we want to use it as an array index so we need to subtract 1 from
323caeaea98SChristoph Hellwig 	 * it.
324caeaea98SChristoph Hellwig 	 */
325caeaea98SChristoph Hellwig 	next_extent = atomic_inc_return(&buip->bui_next_extent) - 1;
326caeaea98SChristoph Hellwig 	ASSERT(next_extent < buip->bui_format.bui_nextents);
327caeaea98SChristoph Hellwig 	map = &buip->bui_format.bui_extents[next_extent];
328f3ebac4cSDarrick J. Wong 	map->me_owner = bi->bi_owner->i_ino;
329f3ebac4cSDarrick J. Wong 	map->me_startblock = bi->bi_bmap.br_startblock;
330f3ebac4cSDarrick J. Wong 	map->me_startoff = bi->bi_bmap.br_startoff;
331f3ebac4cSDarrick J. Wong 	map->me_len = bi->bi_bmap.br_blockcount;
332f3ebac4cSDarrick J. Wong 	xfs_trans_set_bmap_flags(map, bi->bi_type, bi->bi_whichfork,
333f3ebac4cSDarrick J. Wong 			bi->bi_bmap.br_state);
334caeaea98SChristoph Hellwig }
335caeaea98SChristoph Hellwig 
33613a83333SChristoph Hellwig static struct xfs_log_item *
xfs_bmap_update_create_intent(struct xfs_trans * tp,struct list_head * items,unsigned int count,bool sort)337c1f09188SChristoph Hellwig xfs_bmap_update_create_intent(
338c1f09188SChristoph Hellwig 	struct xfs_trans		*tp,
339c1f09188SChristoph Hellwig 	struct list_head		*items,
340d367a868SChristoph Hellwig 	unsigned int			count,
341d367a868SChristoph Hellwig 	bool				sort)
342c1f09188SChristoph Hellwig {
343d367a868SChristoph Hellwig 	struct xfs_mount		*mp = tp->t_mountp;
344d367a868SChristoph Hellwig 	struct xfs_bui_log_item		*buip = xfs_bui_init(mp);
345f3ebac4cSDarrick J. Wong 	struct xfs_bmap_intent		*bi;
346c1f09188SChristoph Hellwig 
347c1f09188SChristoph Hellwig 	ASSERT(count == XFS_BUI_MAX_FAST_EXTENTS);
348c1f09188SChristoph Hellwig 
349c1f09188SChristoph Hellwig 	xfs_trans_add_item(tp, &buip->bui_item);
350d367a868SChristoph Hellwig 	if (sort)
351d367a868SChristoph Hellwig 		list_sort(mp, items, xfs_bmap_update_diff_items);
352f3ebac4cSDarrick J. Wong 	list_for_each_entry(bi, items, bi_list)
353f3ebac4cSDarrick J. Wong 		xfs_bmap_update_log_item(tp, buip, bi);
35413a83333SChristoph Hellwig 	return &buip->bui_item;
355c1f09188SChristoph Hellwig }
356c1f09188SChristoph Hellwig 
357caeaea98SChristoph Hellwig /* Get an BUD so we can process all the deferred rmap updates. */
358f09d167cSChristoph Hellwig static struct xfs_log_item *
xfs_bmap_update_create_done(struct xfs_trans * tp,struct xfs_log_item * intent,unsigned int count)359caeaea98SChristoph Hellwig xfs_bmap_update_create_done(
360caeaea98SChristoph Hellwig 	struct xfs_trans		*tp,
36113a83333SChristoph Hellwig 	struct xfs_log_item		*intent,
362caeaea98SChristoph Hellwig 	unsigned int			count)
363caeaea98SChristoph Hellwig {
364f09d167cSChristoph Hellwig 	return &xfs_trans_get_bud(tp, BUI_ITEM(intent))->bud_item;
365caeaea98SChristoph Hellwig }
366caeaea98SChristoph Hellwig 
367774a99b4SDarrick J. Wong /* Take a passive ref to the AG containing the space we're mapping. */
368774a99b4SDarrick J. Wong void
xfs_bmap_update_get_group(struct xfs_mount * mp,struct xfs_bmap_intent * bi)369774a99b4SDarrick J. Wong xfs_bmap_update_get_group(
370774a99b4SDarrick J. Wong 	struct xfs_mount	*mp,
371774a99b4SDarrick J. Wong 	struct xfs_bmap_intent	*bi)
372774a99b4SDarrick J. Wong {
373774a99b4SDarrick J. Wong 	xfs_agnumber_t		agno;
374774a99b4SDarrick J. Wong 
375774a99b4SDarrick J. Wong 	agno = XFS_FSB_TO_AGNO(mp, bi->bi_bmap.br_startblock);
376d5c88131SDarrick J. Wong 
377d5c88131SDarrick J. Wong 	/*
378d5c88131SDarrick J. Wong 	 * Bump the intent count on behalf of the deferred rmap and refcount
379d5c88131SDarrick J. Wong 	 * intent items that that we can queue when we finish this bmap work.
380d5c88131SDarrick J. Wong 	 * This new intent item will bump the intent count before the bmap
381d5c88131SDarrick J. Wong 	 * intent drops the intent count, ensuring that the intent count
382d5c88131SDarrick J. Wong 	 * remains nonzero across the transaction roll.
383d5c88131SDarrick J. Wong 	 */
384d5c88131SDarrick J. Wong 	bi->bi_pag = xfs_perag_intent_get(mp, agno);
385774a99b4SDarrick J. Wong }
386774a99b4SDarrick J. Wong 
387774a99b4SDarrick J. Wong /* Release a passive AG ref after finishing mapping work. */
388774a99b4SDarrick J. Wong static inline void
xfs_bmap_update_put_group(struct xfs_bmap_intent * bi)389774a99b4SDarrick J. Wong xfs_bmap_update_put_group(
390774a99b4SDarrick J. Wong 	struct xfs_bmap_intent	*bi)
391774a99b4SDarrick J. Wong {
392d5c88131SDarrick J. Wong 	xfs_perag_intent_put(bi->bi_pag);
393774a99b4SDarrick J. Wong }
394774a99b4SDarrick J. Wong 
395caeaea98SChristoph Hellwig /* Process a deferred rmap update. */
396caeaea98SChristoph Hellwig STATIC int
xfs_bmap_update_finish_item(struct xfs_trans * tp,struct xfs_log_item * done,struct list_head * item,struct xfs_btree_cur ** state)397caeaea98SChristoph Hellwig xfs_bmap_update_finish_item(
398caeaea98SChristoph Hellwig 	struct xfs_trans		*tp,
399f09d167cSChristoph Hellwig 	struct xfs_log_item		*done,
400caeaea98SChristoph Hellwig 	struct list_head		*item,
4013ec1b26cSChristoph Hellwig 	struct xfs_btree_cur		**state)
402caeaea98SChristoph Hellwig {
403ddccb81bSDarrick J. Wong 	struct xfs_bmap_intent		*bi;
404caeaea98SChristoph Hellwig 	int				error;
405caeaea98SChristoph Hellwig 
406ddccb81bSDarrick J. Wong 	bi = container_of(item, struct xfs_bmap_intent, bi_list);
407ddccb81bSDarrick J. Wong 
408ddccb81bSDarrick J. Wong 	error = xfs_trans_log_finish_bmap_update(tp, BUD_ITEM(done), bi);
409ddccb81bSDarrick J. Wong 	if (!error && bi->bi_bmap.br_blockcount > 0) {
410ddccb81bSDarrick J. Wong 		ASSERT(bi->bi_type == XFS_BMAP_UNMAP);
411caeaea98SChristoph Hellwig 		return -EAGAIN;
412caeaea98SChristoph Hellwig 	}
413774a99b4SDarrick J. Wong 
414774a99b4SDarrick J. Wong 	xfs_bmap_update_put_group(bi);
415ddccb81bSDarrick J. Wong 	kmem_cache_free(xfs_bmap_intent_cache, bi);
416caeaea98SChristoph Hellwig 	return error;
417caeaea98SChristoph Hellwig }
418caeaea98SChristoph Hellwig 
419caeaea98SChristoph Hellwig /* Abort all pending BUIs. */
420caeaea98SChristoph Hellwig STATIC void
xfs_bmap_update_abort_intent(struct xfs_log_item * intent)421caeaea98SChristoph Hellwig xfs_bmap_update_abort_intent(
42213a83333SChristoph Hellwig 	struct xfs_log_item		*intent)
423caeaea98SChristoph Hellwig {
42413a83333SChristoph Hellwig 	xfs_bui_release(BUI_ITEM(intent));
425caeaea98SChristoph Hellwig }
426caeaea98SChristoph Hellwig 
427774a99b4SDarrick J. Wong /* Cancel a deferred bmap update. */
428caeaea98SChristoph Hellwig STATIC void
xfs_bmap_update_cancel_item(struct list_head * item)429caeaea98SChristoph Hellwig xfs_bmap_update_cancel_item(
430caeaea98SChristoph Hellwig 	struct list_head		*item)
431caeaea98SChristoph Hellwig {
432f3ebac4cSDarrick J. Wong 	struct xfs_bmap_intent		*bi;
433caeaea98SChristoph Hellwig 
434f3ebac4cSDarrick J. Wong 	bi = container_of(item, struct xfs_bmap_intent, bi_list);
435774a99b4SDarrick J. Wong 
436774a99b4SDarrick J. Wong 	xfs_bmap_update_put_group(bi);
437f3ebac4cSDarrick J. Wong 	kmem_cache_free(xfs_bmap_intent_cache, bi);
438caeaea98SChristoph Hellwig }
439caeaea98SChristoph Hellwig 
440caeaea98SChristoph Hellwig const struct xfs_defer_op_type xfs_bmap_update_defer_type = {
441caeaea98SChristoph Hellwig 	.max_items	= XFS_BUI_MAX_FAST_EXTENTS,
442caeaea98SChristoph Hellwig 	.create_intent	= xfs_bmap_update_create_intent,
443caeaea98SChristoph Hellwig 	.abort_intent	= xfs_bmap_update_abort_intent,
444caeaea98SChristoph Hellwig 	.create_done	= xfs_bmap_update_create_done,
445caeaea98SChristoph Hellwig 	.finish_item	= xfs_bmap_update_finish_item,
446caeaea98SChristoph Hellwig 	.cancel_item	= xfs_bmap_update_cancel_item,
447caeaea98SChristoph Hellwig };
448caeaea98SChristoph Hellwig 
449bc525cf4SDarrick J. Wong /* Is this recovered BUI ok? */
450bc525cf4SDarrick J. Wong static inline bool
xfs_bui_validate(struct xfs_mount * mp,struct xfs_bui_log_item * buip)451bc525cf4SDarrick J. Wong xfs_bui_validate(
452bc525cf4SDarrick J. Wong 	struct xfs_mount		*mp,
453bc525cf4SDarrick J. Wong 	struct xfs_bui_log_item		*buip)
454bc525cf4SDarrick J. Wong {
455f3ebac4cSDarrick J. Wong 	struct xfs_map_extent		*map;
456bc525cf4SDarrick J. Wong 
457bc525cf4SDarrick J. Wong 	/* Only one mapping operation per BUI... */
458bc525cf4SDarrick J. Wong 	if (buip->bui_format.bui_nextents != XFS_BUI_MAX_FAST_EXTENTS)
459bc525cf4SDarrick J. Wong 		return false;
460bc525cf4SDarrick J. Wong 
461f3ebac4cSDarrick J. Wong 	map = &buip->bui_format.bui_extents[0];
462bc525cf4SDarrick J. Wong 
463f3ebac4cSDarrick J. Wong 	if (map->me_flags & ~XFS_BMAP_EXTENT_FLAGS)
464bc525cf4SDarrick J. Wong 		return false;
465bc525cf4SDarrick J. Wong 
466f3ebac4cSDarrick J. Wong 	switch (map->me_flags & XFS_BMAP_EXTENT_TYPE_MASK) {
467bc525cf4SDarrick J. Wong 	case XFS_BMAP_MAP:
468bc525cf4SDarrick J. Wong 	case XFS_BMAP_UNMAP:
469bc525cf4SDarrick J. Wong 		break;
470bc525cf4SDarrick J. Wong 	default:
471bc525cf4SDarrick J. Wong 		return false;
472bc525cf4SDarrick J. Wong 	}
473bc525cf4SDarrick J. Wong 
474f3ebac4cSDarrick J. Wong 	if (!xfs_verify_ino(mp, map->me_owner))
47567d8679bSDarrick J. Wong 		return false;
47667d8679bSDarrick J. Wong 
477f3ebac4cSDarrick J. Wong 	if (!xfs_verify_fileext(mp, map->me_startoff, map->me_len))
47867d8679bSDarrick J. Wong 		return false;
47967d8679bSDarrick J. Wong 
480f3ebac4cSDarrick J. Wong 	return xfs_verify_fsbext(mp, map->me_startblock, map->me_len);
481bc525cf4SDarrick J. Wong }
482bc525cf4SDarrick J. Wong 
483caeaea98SChristoph Hellwig /*
48477d61fe4SDarrick J. Wong  * Process a bmap update intent item that was recovered from the log.
48577d61fe4SDarrick J. Wong  * We need to update some inode's bmbt.
48677d61fe4SDarrick J. Wong  */
4879329ba89SDarrick J. Wong STATIC int
xfs_bui_item_recover(struct xfs_defer_pending * dfp,struct list_head * capture_list)48896b60f82SDarrick J. Wong xfs_bui_item_recover(
48987db24c8SDarrick J. Wong 	struct xfs_defer_pending	*dfp,
490e6fff81eSDarrick J. Wong 	struct list_head		*capture_list)
49177d61fe4SDarrick J. Wong {
492ddccb81bSDarrick J. Wong 	struct xfs_bmap_intent		fake = { };
4933c919b09SDarrick J. Wong 	struct xfs_trans_res		resv;
49487db24c8SDarrick J. Wong 	struct xfs_log_item		*lip = dfp->dfp_intent;
49596b60f82SDarrick J. Wong 	struct xfs_bui_log_item		*buip = BUI_ITEM(lip);
49696b60f82SDarrick J. Wong 	struct xfs_trans		*tp;
49796b60f82SDarrick J. Wong 	struct xfs_inode		*ip = NULL;
498d86142ddSDave Chinner 	struct xfs_mount		*mp = lip->li_log->l_mp;
499ddccb81bSDarrick J. Wong 	struct xfs_map_extent		*map;
50096b60f82SDarrick J. Wong 	struct xfs_bud_log_item		*budp;
50185ef08b5SChandan Babu R 	int				iext_delta;
50296b60f82SDarrick J. Wong 	int				error = 0;
50377d61fe4SDarrick J. Wong 
504bc525cf4SDarrick J. Wong 	if (!xfs_bui_validate(mp, buip)) {
505bc525cf4SDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp,
506bc525cf4SDarrick J. Wong 				&buip->bui_format, sizeof(buip->bui_format));
507895e196fSDarrick J. Wong 		return -EFSCORRUPTED;
508bc525cf4SDarrick J. Wong 	}
50977d61fe4SDarrick J. Wong 
510ddccb81bSDarrick J. Wong 	map = &buip->bui_format.bui_extents[0];
511ddccb81bSDarrick J. Wong 	fake.bi_whichfork = (map->me_flags & XFS_BMAP_EXTENT_ATTR_FORK) ?
512919522e8SDarrick J. Wong 			XFS_ATTR_FORK : XFS_DATA_FORK;
513ddccb81bSDarrick J. Wong 	fake.bi_type = map->me_flags & XFS_BMAP_EXTENT_TYPE_MASK;
51477d61fe4SDarrick J. Wong 
515ddccb81bSDarrick J. Wong 	error = xlog_recover_iget(mp, map->me_owner, &ip);
5169f3afb57SDarrick J. Wong 	if (error)
5179f3afb57SDarrick J. Wong 		return error;
518e6fff81eSDarrick J. Wong 
51964a3f331SDarrick J. Wong 	/* Allocate transaction and do the work. */
5203c919b09SDarrick J. Wong 	resv = xlog_recover_resv(&M_RES(mp)->tr_itruncate);
5213c919b09SDarrick J. Wong 	error = xfs_trans_alloc(mp, &resv,
52264a3f331SDarrick J. Wong 			XFS_EXTENTADD_SPACE_RES(mp, XFS_DATA_FORK), 0, 0, &tp);
52364a3f331SDarrick J. Wong 	if (error)
52464a3f331SDarrick J. Wong 		goto err_rele;
52564a3f331SDarrick J. Wong 
52664a3f331SDarrick J. Wong 	budp = xfs_trans_get_bud(tp, buip);
527680776e5SDarrick J. Wong 	xlog_recover_transfer_intent(tp, dfp);
528680776e5SDarrick J. Wong 
52964a3f331SDarrick J. Wong 	xfs_ilock(ip, XFS_ILOCK_EXCL);
5309f3afb57SDarrick J. Wong 	xfs_trans_ijoin(tp, ip, 0);
5319f3afb57SDarrick J. Wong 
532ddccb81bSDarrick J. Wong 	if (fake.bi_type == XFS_BMAP_MAP)
53385ef08b5SChandan Babu R 		iext_delta = XFS_IEXT_ADD_NOSPLIT_CNT;
53485ef08b5SChandan Babu R 	else
53585ef08b5SChandan Babu R 		iext_delta = XFS_IEXT_PUNCH_HOLE_CNT;
53685ef08b5SChandan Babu R 
537ddccb81bSDarrick J. Wong 	error = xfs_iext_count_may_overflow(ip, fake.bi_whichfork, iext_delta);
5384f86bb4bSChandan Babu R 	if (error == -EFBIG)
5394f86bb4bSChandan Babu R 		error = xfs_iext_count_upgrade(tp, ip, iext_delta);
540727e1acdSChandan Babu R 	if (error)
541727e1acdSChandan Babu R 		goto err_cancel;
542727e1acdSChandan Babu R 
543ddccb81bSDarrick J. Wong 	fake.bi_owner = ip;
544ddccb81bSDarrick J. Wong 	fake.bi_bmap.br_startblock = map->me_startblock;
545ddccb81bSDarrick J. Wong 	fake.bi_bmap.br_startoff = map->me_startoff;
546ddccb81bSDarrick J. Wong 	fake.bi_bmap.br_blockcount = map->me_len;
547ddccb81bSDarrick J. Wong 	fake.bi_bmap.br_state = (map->me_flags & XFS_BMAP_EXTENT_UNWRITTEN) ?
548ddccb81bSDarrick J. Wong 			XFS_EXT_UNWRITTEN : XFS_EXT_NORM;
549ddccb81bSDarrick J. Wong 
550774a99b4SDarrick J. Wong 	xfs_bmap_update_get_group(mp, &fake);
551ddccb81bSDarrick J. Wong 	error = xfs_trans_log_finish_bmap_update(tp, budp, &fake);
55243059d54SDarrick J. Wong 	if (error == -EFSCORRUPTED)
553ddccb81bSDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp, map,
554ddccb81bSDarrick J. Wong 				sizeof(*map));
555774a99b4SDarrick J. Wong 	xfs_bmap_update_put_group(&fake);
5569f3afb57SDarrick J. Wong 	if (error)
55764a3f331SDarrick J. Wong 		goto err_cancel;
5589f3afb57SDarrick J. Wong 
559ddccb81bSDarrick J. Wong 	if (fake.bi_bmap.br_blockcount > 0) {
560ddccb81bSDarrick J. Wong 		ASSERT(fake.bi_type == XFS_BMAP_UNMAP);
561ddccb81bSDarrick J. Wong 		xfs_bmap_unmap_extent(tp, ip, &fake.bi_bmap);
562e1a4e37cSDarrick J. Wong 	}
563e1a4e37cSDarrick J. Wong 
564ff4ab5e0SDarrick J. Wong 	/*
565ff4ab5e0SDarrick J. Wong 	 * Commit transaction, which frees the transaction and saves the inode
566ff4ab5e0SDarrick J. Wong 	 * for later replay activities.
567ff4ab5e0SDarrick J. Wong 	 */
568512edfacSDarrick J. Wong 	error = xfs_defer_ops_capture_and_commit(tp, capture_list);
56964a3f331SDarrick J. Wong 	if (error)
57064a3f331SDarrick J. Wong 		goto err_unlock;
5719f3afb57SDarrick J. Wong 
5729f3afb57SDarrick J. Wong 	xfs_iunlock(ip, XFS_ILOCK_EXCL);
57344a8736bSDarrick J. Wong 	xfs_irele(ip);
57464a3f331SDarrick J. Wong 	return 0;
57564a3f331SDarrick J. Wong 
57664a3f331SDarrick J. Wong err_cancel:
57764a3f331SDarrick J. Wong 	xfs_trans_cancel(tp);
57864a3f331SDarrick J. Wong err_unlock:
57964a3f331SDarrick J. Wong 	xfs_iunlock(ip, XFS_ILOCK_EXCL);
58064a3f331SDarrick J. Wong err_rele:
58164a3f331SDarrick J. Wong 	xfs_irele(ip);
58277d61fe4SDarrick J. Wong 	return error;
58377d61fe4SDarrick J. Wong }
58486ffa471SDarrick J. Wong 
585154c733aSDarrick J. Wong STATIC bool
xfs_bui_item_match(struct xfs_log_item * lip,uint64_t intent_id)586154c733aSDarrick J. Wong xfs_bui_item_match(
587154c733aSDarrick J. Wong 	struct xfs_log_item	*lip,
588154c733aSDarrick J. Wong 	uint64_t		intent_id)
589154c733aSDarrick J. Wong {
590154c733aSDarrick J. Wong 	return BUI_ITEM(lip)->bui_format.bui_id == intent_id;
591154c733aSDarrick J. Wong }
592154c733aSDarrick J. Wong 
5934e919af7SDarrick J. Wong /* Relog an intent item to push the log tail forward. */
5944e919af7SDarrick J. Wong static struct xfs_log_item *
xfs_bui_item_relog(struct xfs_log_item * intent,struct xfs_trans * tp)5954e919af7SDarrick J. Wong xfs_bui_item_relog(
5964e919af7SDarrick J. Wong 	struct xfs_log_item		*intent,
5974e919af7SDarrick J. Wong 	struct xfs_trans		*tp)
5984e919af7SDarrick J. Wong {
5994e919af7SDarrick J. Wong 	struct xfs_bud_log_item		*budp;
6004e919af7SDarrick J. Wong 	struct xfs_bui_log_item		*buip;
601f3ebac4cSDarrick J. Wong 	struct xfs_map_extent		*map;
6024e919af7SDarrick J. Wong 	unsigned int			count;
6034e919af7SDarrick J. Wong 
6044e919af7SDarrick J. Wong 	count = BUI_ITEM(intent)->bui_format.bui_nextents;
605f3ebac4cSDarrick J. Wong 	map = BUI_ITEM(intent)->bui_format.bui_extents;
6064e919af7SDarrick J. Wong 
6074e919af7SDarrick J. Wong 	tp->t_flags |= XFS_TRANS_DIRTY;
6084e919af7SDarrick J. Wong 	budp = xfs_trans_get_bud(tp, BUI_ITEM(intent));
6094e919af7SDarrick J. Wong 	set_bit(XFS_LI_DIRTY, &budp->bud_item.li_flags);
6104e919af7SDarrick J. Wong 
6114e919af7SDarrick J. Wong 	buip = xfs_bui_init(tp->t_mountp);
612f3ebac4cSDarrick J. Wong 	memcpy(buip->bui_format.bui_extents, map, count * sizeof(*map));
6134e919af7SDarrick J. Wong 	atomic_set(&buip->bui_next_extent, count);
6144e919af7SDarrick J. Wong 	xfs_trans_add_item(tp, &buip->bui_item);
6154e919af7SDarrick J. Wong 	set_bit(XFS_LI_DIRTY, &buip->bui_item.li_flags);
6164e919af7SDarrick J. Wong 	return &buip->bui_item;
6174e919af7SDarrick J. Wong }
6184e919af7SDarrick J. Wong 
6199329ba89SDarrick J. Wong static const struct xfs_item_ops xfs_bui_item_ops = {
620f5b81200SDave Chinner 	.flags		= XFS_ITEM_INTENT,
6219329ba89SDarrick J. Wong 	.iop_size	= xfs_bui_item_size,
6229329ba89SDarrick J. Wong 	.iop_format	= xfs_bui_item_format,
6239329ba89SDarrick J. Wong 	.iop_unpin	= xfs_bui_item_unpin,
6249329ba89SDarrick J. Wong 	.iop_release	= xfs_bui_item_release,
6259329ba89SDarrick J. Wong 	.iop_recover	= xfs_bui_item_recover,
626154c733aSDarrick J. Wong 	.iop_match	= xfs_bui_item_match,
6274e919af7SDarrick J. Wong 	.iop_relog	= xfs_bui_item_relog,
6289329ba89SDarrick J. Wong };
6299329ba89SDarrick J. Wong 
630a38ebce1SDarrick J. Wong static inline void
xfs_bui_copy_format(struct xfs_bui_log_format * dst,const struct xfs_bui_log_format * src)6313c6ba3cfSDarrick J. Wong xfs_bui_copy_format(
632a38ebce1SDarrick J. Wong 	struct xfs_bui_log_format	*dst,
633a38ebce1SDarrick J. Wong 	const struct xfs_bui_log_format	*src)
6343c6ba3cfSDarrick J. Wong {
635a38ebce1SDarrick J. Wong 	unsigned int			i;
6363c6ba3cfSDarrick J. Wong 
637a38ebce1SDarrick J. Wong 	memcpy(dst, src, offsetof(struct xfs_bui_log_format, bui_extents));
6383c6ba3cfSDarrick J. Wong 
639a38ebce1SDarrick J. Wong 	for (i = 0; i < src->bui_nextents; i++)
640a38ebce1SDarrick J. Wong 		memcpy(&dst->bui_extents[i], &src->bui_extents[i],
641a38ebce1SDarrick J. Wong 				sizeof(struct xfs_map_extent));
6423c6ba3cfSDarrick J. Wong }
6433c6ba3cfSDarrick J. Wong 
6443c6ba3cfSDarrick J. Wong /*
6453c6ba3cfSDarrick J. Wong  * This routine is called to create an in-core extent bmap update
6463c6ba3cfSDarrick J. Wong  * item from the bui format structure which was logged on disk.
6473c6ba3cfSDarrick J. Wong  * It allocates an in-core bui, copies the extents from the format
6483c6ba3cfSDarrick J. Wong  * structure into it, and adds the bui to the AIL with the given
6493c6ba3cfSDarrick J. Wong  * LSN.
6503c6ba3cfSDarrick J. Wong  */
6513c6ba3cfSDarrick J. Wong STATIC int
xlog_recover_bui_commit_pass2(struct xlog * log,struct list_head * buffer_list,struct xlog_recover_item * item,xfs_lsn_t lsn)6523c6ba3cfSDarrick J. Wong xlog_recover_bui_commit_pass2(
6533c6ba3cfSDarrick J. Wong 	struct xlog			*log,
6543c6ba3cfSDarrick J. Wong 	struct list_head		*buffer_list,
6553c6ba3cfSDarrick J. Wong 	struct xlog_recover_item	*item,
6563c6ba3cfSDarrick J. Wong 	xfs_lsn_t			lsn)
6573c6ba3cfSDarrick J. Wong {
6583c6ba3cfSDarrick J. Wong 	struct xfs_mount		*mp = log->l_mp;
6593c6ba3cfSDarrick J. Wong 	struct xfs_bui_log_item		*buip;
6603c6ba3cfSDarrick J. Wong 	struct xfs_bui_log_format	*bui_formatp;
661a38ebce1SDarrick J. Wong 	size_t				len;
6623c6ba3cfSDarrick J. Wong 
6633c6ba3cfSDarrick J. Wong 	bui_formatp = item->ri_buf[0].i_addr;
6643c6ba3cfSDarrick J. Wong 
665a38ebce1SDarrick J. Wong 	if (item->ri_buf[0].i_len < xfs_bui_log_format_sizeof(0)) {
666950f0d50SDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp,
667950f0d50SDarrick J. Wong 				item->ri_buf[0].i_addr, item->ri_buf[0].i_len);
668a38ebce1SDarrick J. Wong 		return -EFSCORRUPTED;
669a38ebce1SDarrick J. Wong 	}
670a38ebce1SDarrick J. Wong 
6713c6ba3cfSDarrick J. Wong 	if (bui_formatp->bui_nextents != XFS_BUI_MAX_FAST_EXTENTS) {
672950f0d50SDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp,
673950f0d50SDarrick J. Wong 				item->ri_buf[0].i_addr, item->ri_buf[0].i_len);
6743c6ba3cfSDarrick J. Wong 		return -EFSCORRUPTED;
6753c6ba3cfSDarrick J. Wong 	}
676a38ebce1SDarrick J. Wong 
677a38ebce1SDarrick J. Wong 	len = xfs_bui_log_format_sizeof(bui_formatp->bui_nextents);
678a38ebce1SDarrick J. Wong 	if (item->ri_buf[0].i_len != len) {
679950f0d50SDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, mp,
680950f0d50SDarrick J. Wong 				item->ri_buf[0].i_addr, item->ri_buf[0].i_len);
681a38ebce1SDarrick J. Wong 		return -EFSCORRUPTED;
6823c6ba3cfSDarrick J. Wong 	}
683a38ebce1SDarrick J. Wong 
684a38ebce1SDarrick J. Wong 	buip = xfs_bui_init(mp);
685a38ebce1SDarrick J. Wong 	xfs_bui_copy_format(&buip->bui_format, bui_formatp);
6863c6ba3cfSDarrick J. Wong 	atomic_set(&buip->bui_next_extent, bui_formatp->bui_nextents);
687cd3c2cf3SDarrick J. Wong 
688cd3c2cf3SDarrick J. Wong 	xlog_recover_intent_item(log, &buip->bui_item, lsn,
689cd3c2cf3SDarrick J. Wong 			XFS_DEFER_OPS_TYPE_BMAP);
6903c6ba3cfSDarrick J. Wong 	return 0;
6913c6ba3cfSDarrick J. Wong }
6923c6ba3cfSDarrick J. Wong 
69386ffa471SDarrick J. Wong const struct xlog_recover_item_ops xlog_bui_item_ops = {
69486ffa471SDarrick J. Wong 	.item_type		= XFS_LI_BUI,
6953c6ba3cfSDarrick J. Wong 	.commit_pass2		= xlog_recover_bui_commit_pass2,
69686ffa471SDarrick J. Wong };
69786ffa471SDarrick J. Wong 
6983c6ba3cfSDarrick J. Wong /*
6993c6ba3cfSDarrick J. Wong  * This routine is called when an BUD format structure is found in a committed
7003c6ba3cfSDarrick J. Wong  * transaction in the log. Its purpose is to cancel the corresponding BUI if it
7013c6ba3cfSDarrick J. Wong  * was still in the log. To do this it searches the AIL for the BUI with an id
7023c6ba3cfSDarrick J. Wong  * equal to that in the BUD format structure. If we find it we drop the BUD
7033c6ba3cfSDarrick J. Wong  * reference, which removes the BUI from the AIL and frees it.
7043c6ba3cfSDarrick J. Wong  */
7053c6ba3cfSDarrick J. Wong STATIC int
xlog_recover_bud_commit_pass2(struct xlog * log,struct list_head * buffer_list,struct xlog_recover_item * item,xfs_lsn_t lsn)7063c6ba3cfSDarrick J. Wong xlog_recover_bud_commit_pass2(
7073c6ba3cfSDarrick J. Wong 	struct xlog			*log,
7083c6ba3cfSDarrick J. Wong 	struct list_head		*buffer_list,
7093c6ba3cfSDarrick J. Wong 	struct xlog_recover_item	*item,
7103c6ba3cfSDarrick J. Wong 	xfs_lsn_t			lsn)
7113c6ba3cfSDarrick J. Wong {
7123c6ba3cfSDarrick J. Wong 	struct xfs_bud_log_format	*bud_formatp;
7133c6ba3cfSDarrick J. Wong 
7143c6ba3cfSDarrick J. Wong 	bud_formatp = item->ri_buf[0].i_addr;
7153c6ba3cfSDarrick J. Wong 	if (item->ri_buf[0].i_len != sizeof(struct xfs_bud_log_format)) {
716950f0d50SDarrick J. Wong 		XFS_CORRUPTION_ERROR(__func__, XFS_ERRLEVEL_LOW, log->l_mp,
717950f0d50SDarrick J. Wong 				item->ri_buf[0].i_addr, item->ri_buf[0].i_len);
7183c6ba3cfSDarrick J. Wong 		return -EFSCORRUPTED;
7193c6ba3cfSDarrick J. Wong 	}
7203c6ba3cfSDarrick J. Wong 
721154c733aSDarrick J. Wong 	xlog_recover_release_intent(log, XFS_LI_BUI, bud_formatp->bud_bui_id);
7223c6ba3cfSDarrick J. Wong 	return 0;
7233c6ba3cfSDarrick J. Wong }
7243c6ba3cfSDarrick J. Wong 
72586ffa471SDarrick J. Wong const struct xlog_recover_item_ops xlog_bud_item_ops = {
72686ffa471SDarrick J. Wong 	.item_type		= XFS_LI_BUD,
7273c6ba3cfSDarrick J. Wong 	.commit_pass2		= xlog_recover_bud_commit_pass2,
72886ffa471SDarrick J. Wong };
729