Blame view

fs/xfs/xfs_extfree_item.c 20.9 KB
0b61f8a40   Dave Chinner   xfs: convert to S...
1
  // SPDX-License-Identifier: GPL-2.0
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
  /*
7b7187698   Nathan Scott   [XFS] Update lice...
3
4
   * Copyright (c) 2000-2001,2005 Silicon Graphics, Inc.
   * All Rights Reserved.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
5
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
6
  #include "xfs.h"
a844f4510   Nathan Scott   [XFS] Remove xfs_...
7
  #include "xfs_fs.h"
4fb6e8ade   Christoph Hellwig   xfs: merge xfs_ag...
8
  #include "xfs_format.h"
239880ef6   Dave Chinner   xfs: decouple log...
9
10
  #include "xfs_log_format.h"
  #include "xfs_trans_resv.h"
dc42375d5   Darrick J. Wong   xfs: refactor red...
11
  #include "xfs_bit.h"
5467b34bd   Darrick J. Wong   xfs: move xfs_ino...
12
  #include "xfs_shared.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
13
  #include "xfs_mount.h"
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
14
  #include "xfs_defer.h"
239880ef6   Dave Chinner   xfs: decouple log...
15
  #include "xfs_trans.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
16
17
  #include "xfs_trans_priv.h"
  #include "xfs_extfree_item.h"
1234351cb   Christoph Hellwig   xfs: introduce xl...
18
  #include "xfs_log.h"
340785cca   Darrick J. Wong   xfs: add owner fi...
19
20
  #include "xfs_btree.h"
  #include "xfs_rmap.h"
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
21
22
23
  #include "xfs_alloc.h"
  #include "xfs_bmap.h"
  #include "xfs_trace.h"
a5155b870   Darrick J. Wong   xfs: always log c...
24
  #include "xfs_error.h"
9817aa80d   Darrick J. Wong   xfs: refactor log...
25
  #include "xfs_log_priv.h"
86ffa471d   Darrick J. Wong   xfs: refactor log...
26
  #include "xfs_log_recover.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
27
28
29
  
  kmem_zone_t	*xfs_efi_zone;
  kmem_zone_t	*xfs_efd_zone;
10d0c6e06   Darrick J. Wong   xfs: refactor rec...
30
  static const struct xfs_item_ops xfs_efi_item_ops;
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
31
32
33
34
  static inline struct xfs_efi_log_item *EFI_ITEM(struct xfs_log_item *lip)
  {
  	return container_of(lip, struct xfs_efi_log_item, efi_item);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35

9817aa80d   Darrick J. Wong   xfs: refactor log...
36
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
37
38
  xfs_efi_item_free(
  	struct xfs_efi_log_item	*efip)
7d795ca34   Christoph Hellwig   [XFS] consolidate...
39
  {
b1c5ebb21   Dave Chinner   xfs: allocate log...
40
  	kmem_free(efip->efi_item.li_lv_shadow);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
41
  	if (efip->efi_format.efi_nextents > XFS_EFI_MAX_FAST_EXTENTS)
f0e2d93c2   Denys Vlasenko   [XFS] Remove unus...
42
  		kmem_free(efip);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
43
  	else
377bcd5f3   Carlos Maiolino   xfs: Remove kmem_...
44
  		kmem_cache_free(xfs_efi_zone, efip);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
45
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
46
47
  
  /*
0612d1166   Dave Chinner   xfs: fix intent u...
48
49
50
51
52
53
   * Freeing the efi requires that we remove it from the AIL if it has already
   * been placed there. However, the EFI may not yet have been placed in the AIL
   * when called by xfs_efi_release() from EFD processing due to the ordering of
   * committed vs unpin operations in bulk insert operations. Hence the reference
   * count to ensure only the last caller frees the EFI.
   */
10d0c6e06   Darrick J. Wong   xfs: refactor rec...
54
  STATIC void
0612d1166   Dave Chinner   xfs: fix intent u...
55
56
57
58
59
  xfs_efi_release(
  	struct xfs_efi_log_item	*efip)
  {
  	ASSERT(atomic_read(&efip->efi_refcount) > 0);
  	if (atomic_dec_and_test(&efip->efi_refcount)) {
655879290   Brian Foster   xfs: use delete h...
60
  		xfs_trans_ail_delete(&efip->efi_item, SHUTDOWN_LOG_IO_ERROR);
0612d1166   Dave Chinner   xfs: fix intent u...
61
62
63
64
65
  		xfs_efi_item_free(efip);
  	}
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
66
67
68
69
   * This returns the number of iovecs needed to log the given efi item.
   * We only need 1 iovec for an efi item.  It just logs the efi_log_format
   * structure.
   */
166d13688   Dave Chinner   xfs: return log i...
70
71
72
73
74
75
76
77
78
  static inline int
  xfs_efi_item_sizeof(
  	struct xfs_efi_log_item *efip)
  {
  	return sizeof(struct xfs_efi_log_format) +
  	       (efip->efi_format.efi_nextents - 1) * sizeof(xfs_extent_t);
  }
  
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
79
  xfs_efi_item_size(
166d13688   Dave Chinner   xfs: return log i...
80
81
82
  	struct xfs_log_item	*lip,
  	int			*nvecs,
  	int			*nbytes)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
83
  {
166d13688   Dave Chinner   xfs: return log i...
84
85
  	*nvecs += 1;
  	*nbytes += xfs_efi_item_sizeof(EFI_ITEM(lip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
86
87
88
89
90
91
92
93
94
95
  }
  
  /*
   * This is called to fill in the vector of log iovecs for the
   * given efi log item. We use only 1 iovec, and we point that
   * at the efi_log_format structure embedded in the efi item.
   * It is at this point that we assert that all of the extent
   * slots in the efi item have been filled.
   */
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
96
97
  xfs_efi_item_format(
  	struct xfs_log_item	*lip,
bde7cff67   Christoph Hellwig   xfs: format log i...
98
  	struct xfs_log_vec	*lv)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
99
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
100
  	struct xfs_efi_log_item	*efip = EFI_ITEM(lip);
bde7cff67   Christoph Hellwig   xfs: format log i...
101
  	struct xfs_log_iovec	*vecp = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
102

b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
103
104
  	ASSERT(atomic_read(&efip->efi_next_extent) ==
  				efip->efi_format.efi_nextents);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
105
106
  
  	efip->efi_format.efi_type = XFS_LI_EFI;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
107
  	efip->efi_format.efi_size = 1;
bde7cff67   Christoph Hellwig   xfs: format log i...
108
  	xlog_copy_iovec(lv, &vecp, XLOG_REG_TYPE_EFI_FORMAT,
1234351cb   Christoph Hellwig   xfs: introduce xl...
109
110
  			&efip->efi_format,
  			xfs_efi_item_sizeof(efip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
111
112
113
114
  }
  
  
  /*
8d99fe92f   Brian Foster   xfs: fix efi/efd ...
115
116
117
118
119
120
   * The unpin operation is the last place an EFI is manipulated in the log. It is
   * either inserted in the AIL or aborted in the event of a log I/O error. In
   * either case, the EFI transaction has been successfully committed to make it
   * this far. Therefore, we expect whoever committed the EFI to either construct
   * and commit the EFD or drop the EFD's reference in the event of error. Simply
   * drop the log's EFI reference now that the log is done with it.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
121
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
122
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
123
124
125
  xfs_efi_item_unpin(
  	struct xfs_log_item	*lip,
  	int			remove)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
127
  	struct xfs_efi_log_item	*efip = EFI_ITEM(lip);
5e4b5386a   Brian Foster   xfs: disentagle E...
128
  	xfs_efi_release(efip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
129
130
131
  }
  
  /*
8d99fe92f   Brian Foster   xfs: fix efi/efd ...
132
133
134
135
   * The EFI has been either committed or aborted if the transaction has been
   * cancelled. If the transaction was cancelled, an EFD isn't going to be
   * constructed and thus we free the EFI here directly.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
136
  STATIC void
ddf92053e   Christoph Hellwig   xfs: split iop_un...
137
  xfs_efi_item_release(
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
138
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
139
  {
ddf92053e   Christoph Hellwig   xfs: split iop_un...
140
  	xfs_efi_release(EFI_ITEM(lip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
141
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
142
143
144
  /*
   * Allocate and initialize an efi item with the given number of extents.
   */
9817aa80d   Darrick J. Wong   xfs: refactor log...
145
  STATIC struct xfs_efi_log_item *
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
146
147
148
  xfs_efi_init(
  	struct xfs_mount	*mp,
  	uint			nextents)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
149
150
  
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
151
  	struct xfs_efi_log_item	*efip;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
152
153
154
155
  	uint			size;
  
  	ASSERT(nextents > 0);
  	if (nextents > XFS_EFI_MAX_FAST_EXTENTS) {
82ff450b2   Christoph Hellwig   xfs: remove the x...
156
  		size = (uint)(sizeof(struct xfs_efi_log_item) +
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
157
  			((nextents - 1) * sizeof(xfs_extent_t)));
707e0ddaf   Tetsuo Handa   fs: xfs: Remove K...
158
  		efip = kmem_zalloc(size, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
159
  	} else {
32a2b11f4   Carlos Maiolino   xfs: Remove kmem_...
160
161
  		efip = kmem_cache_zalloc(xfs_efi_zone,
  					 GFP_KERNEL | __GFP_NOFAIL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
162
  	}
43f5efc5b   Dave Chinner   xfs: factor log i...
163
  	xfs_log_item_init(mp, &efip->efi_item, XFS_LI_EFI, &xfs_efi_item_ops);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164
  	efip->efi_format.efi_nextents = nextents;
db9d67d6b   Christoph Hellwig   xfs: remove __psi...
165
  	efip->efi_format.efi_id = (uintptr_t)(void *)efip;
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
166
  	atomic_set(&efip->efi_next_extent, 0);
666d644cd   Dave Chinner   xfs: don't free E...
167
  	atomic_set(&efip->efi_refcount, 2);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
168

7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
169
  	return efip;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
170
171
172
  }
  
  /*
6d192a9b8   Tim Shimmin   [XFS] inode items...
173
174
175
176
177
178
   * Copy an EFI format buffer from the given buf, and into the destination
   * EFI format structure.
   * The given buffer can be in 32 bit or 64 bit form (which has different padding),
   * one of which will be the native format for this kernel.
   * It will handle the conversion of formats if necessary.
   */
9817aa80d   Darrick J. Wong   xfs: refactor log...
179
  STATIC int
6d192a9b8   Tim Shimmin   [XFS] inode items...
180
181
  xfs_efi_copy_format(xfs_log_iovec_t *buf, xfs_efi_log_format_t *dst_efi_fmt)
  {
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
182
  	xfs_efi_log_format_t *src_efi_fmt = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
183
184
185
186
187
188
189
190
191
192
193
194
  	uint i;
  	uint len = sizeof(xfs_efi_log_format_t) + 
  		(src_efi_fmt->efi_nextents - 1) * sizeof(xfs_extent_t);  
  	uint len32 = sizeof(xfs_efi_log_format_32_t) + 
  		(src_efi_fmt->efi_nextents - 1) * sizeof(xfs_extent_32_t);  
  	uint len64 = sizeof(xfs_efi_log_format_64_t) + 
  		(src_efi_fmt->efi_nextents - 1) * sizeof(xfs_extent_64_t);  
  
  	if (buf->i_len == len) {
  		memcpy((char *)dst_efi_fmt, (char*)src_efi_fmt, len);
  		return 0;
  	} else if (buf->i_len == len32) {
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
195
  		xfs_efi_log_format_32_t *src_efi_fmt_32 = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
196
197
198
199
200
201
202
203
204
205
206
207
208
  
  		dst_efi_fmt->efi_type     = src_efi_fmt_32->efi_type;
  		dst_efi_fmt->efi_size     = src_efi_fmt_32->efi_size;
  		dst_efi_fmt->efi_nextents = src_efi_fmt_32->efi_nextents;
  		dst_efi_fmt->efi_id       = src_efi_fmt_32->efi_id;
  		for (i = 0; i < dst_efi_fmt->efi_nextents; i++) {
  			dst_efi_fmt->efi_extents[i].ext_start =
  				src_efi_fmt_32->efi_extents[i].ext_start;
  			dst_efi_fmt->efi_extents[i].ext_len =
  				src_efi_fmt_32->efi_extents[i].ext_len;
  		}
  		return 0;
  	} else if (buf->i_len == len64) {
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
209
  		xfs_efi_log_format_64_t *src_efi_fmt_64 = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
210
211
212
213
214
215
216
217
218
219
220
221
222
  
  		dst_efi_fmt->efi_type     = src_efi_fmt_64->efi_type;
  		dst_efi_fmt->efi_size     = src_efi_fmt_64->efi_size;
  		dst_efi_fmt->efi_nextents = src_efi_fmt_64->efi_nextents;
  		dst_efi_fmt->efi_id       = src_efi_fmt_64->efi_id;
  		for (i = 0; i < dst_efi_fmt->efi_nextents; i++) {
  			dst_efi_fmt->efi_extents[i].ext_start =
  				src_efi_fmt_64->efi_extents[i].ext_start;
  			dst_efi_fmt->efi_extents[i].ext_len =
  				src_efi_fmt_64->efi_extents[i].ext_len;
  		}
  		return 0;
  	}
a5155b870   Darrick J. Wong   xfs: always log c...
223
  	XFS_ERROR_REPORT(__func__, XFS_ERRLEVEL_LOW, NULL);
2451337dd   Dave Chinner   xfs: global error...
224
  	return -EFSCORRUPTED;
6d192a9b8   Tim Shimmin   [XFS] inode items...
225
  }
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
226
  static inline struct xfs_efd_log_item *EFD_ITEM(struct xfs_log_item *lip)
7d795ca34   Christoph Hellwig   [XFS] consolidate...
227
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
228
229
  	return container_of(lip, struct xfs_efd_log_item, efd_item);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
230

7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
231
232
233
  STATIC void
  xfs_efd_item_free(struct xfs_efd_log_item *efdp)
  {
b1c5ebb21   Dave Chinner   xfs: allocate log...
234
  	kmem_free(efdp->efd_item.li_lv_shadow);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
235
  	if (efdp->efd_format.efd_nextents > XFS_EFD_MAX_FAST_EXTENTS)
f0e2d93c2   Denys Vlasenko   [XFS] Remove unus...
236
  		kmem_free(efdp);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
237
  	else
377bcd5f3   Carlos Maiolino   xfs: Remove kmem_...
238
  		kmem_cache_free(xfs_efd_zone, efdp);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
239
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240
241
242
243
244
245
  
  /*
   * This returns the number of iovecs needed to log the given efd item.
   * We only need 1 iovec for an efd item.  It just logs the efd_log_format
   * structure.
   */
166d13688   Dave Chinner   xfs: return log i...
246
247
248
249
250
251
252
253
254
  static inline int
  xfs_efd_item_sizeof(
  	struct xfs_efd_log_item *efdp)
  {
  	return sizeof(xfs_efd_log_format_t) +
  	       (efdp->efd_format.efd_nextents - 1) * sizeof(xfs_extent_t);
  }
  
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
255
  xfs_efd_item_size(
166d13688   Dave Chinner   xfs: return log i...
256
257
258
  	struct xfs_log_item	*lip,
  	int			*nvecs,
  	int			*nbytes)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
259
  {
166d13688   Dave Chinner   xfs: return log i...
260
261
  	*nvecs += 1;
  	*nbytes += xfs_efd_item_sizeof(EFD_ITEM(lip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
262
263
264
265
266
267
268
269
270
271
  }
  
  /*
   * This is called to fill in the vector of log iovecs for the
   * given efd log item. We use only 1 iovec, and we point that
   * at the efd_log_format structure embedded in the efd item.
   * It is at this point that we assert that all of the extent
   * slots in the efd item have been filled.
   */
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
272
273
  xfs_efd_item_format(
  	struct xfs_log_item	*lip,
bde7cff67   Christoph Hellwig   xfs: format log i...
274
  	struct xfs_log_vec	*lv)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
275
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
276
  	struct xfs_efd_log_item	*efdp = EFD_ITEM(lip);
bde7cff67   Christoph Hellwig   xfs: format log i...
277
  	struct xfs_log_iovec	*vecp = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
278
279
280
281
  
  	ASSERT(efdp->efd_next_extent == efdp->efd_format.efd_nextents);
  
  	efdp->efd_format.efd_type = XFS_LI_EFD;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
282
  	efdp->efd_format.efd_size = 1;
bde7cff67   Christoph Hellwig   xfs: format log i...
283
  	xlog_copy_iovec(lv, &vecp, XLOG_REG_TYPE_EFD_FORMAT,
1234351cb   Christoph Hellwig   xfs: introduce xl...
284
285
  			&efdp->efd_format,
  			xfs_efd_item_sizeof(efdp));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
286
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
287
  /*
8d99fe92f   Brian Foster   xfs: fix efi/efd ...
288
289
290
   * The EFD is either committed or aborted if the transaction is cancelled. If
   * the transaction is cancelled, drop our reference to the EFI and free the EFD.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
291
  STATIC void
ddf92053e   Christoph Hellwig   xfs: split iop_un...
292
  xfs_efd_item_release(
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
293
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
294
  {
8d99fe92f   Brian Foster   xfs: fix efi/efd ...
295
  	struct xfs_efd_log_item	*efdp = EFD_ITEM(lip);
ddf92053e   Christoph Hellwig   xfs: split iop_un...
296
297
  	xfs_efi_release(efdp->efd_efip);
  	xfs_efd_item_free(efdp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
298
  }
272e42b21   Christoph Hellwig   xfs: constify xfs...
299
  static const struct xfs_item_ops xfs_efd_item_ops = {
9ce632a28   Christoph Hellwig   xfs: add a flag t...
300
  	.flags		= XFS_ITEM_RELEASE_WHEN_COMMITTED,
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
301
302
  	.iop_size	= xfs_efd_item_size,
  	.iop_format	= xfs_efd_item_format,
ddf92053e   Christoph Hellwig   xfs: split iop_un...
303
  	.iop_release	= xfs_efd_item_release,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
304
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
305
  /*
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
306
307
308
   * Allocate an "extent free done" log item that will hold nextents worth of
   * extents.  The caller must use all nextents extents, because we are not
   * flexible about this at all.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309
   */
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
310
  static struct xfs_efd_log_item *
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
311
312
313
314
  xfs_trans_get_efd(
  	struct xfs_trans		*tp,
  	struct xfs_efi_log_item		*efip,
  	unsigned int			nextents)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
315
  {
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
316
  	struct xfs_efd_log_item		*efdp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
317
318
  
  	ASSERT(nextents > 0);
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
319

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
320
  	if (nextents > XFS_EFD_MAX_FAST_EXTENTS) {
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
321
322
  		efdp = kmem_zalloc(sizeof(struct xfs_efd_log_item) +
  				(nextents - 1) * sizeof(struct xfs_extent),
707e0ddaf   Tetsuo Handa   fs: xfs: Remove K...
323
  				0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
324
  	} else {
32a2b11f4   Carlos Maiolino   xfs: Remove kmem_...
325
326
  		efdp = kmem_cache_zalloc(xfs_efd_zone,
  					GFP_KERNEL | __GFP_NOFAIL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327
  	}
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
328
329
  	xfs_log_item_init(tp->t_mountp, &efdp->efd_item, XFS_LI_EFD,
  			  &xfs_efd_item_ops);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
330
331
332
  	efdp->efd_efip = efip;
  	efdp->efd_format.efd_nextents = nextents;
  	efdp->efd_format.efd_efi_id = efip->efi_format.efi_id;
9c5e7c2ae   Christoph Hellwig   xfs: merge xfs_ef...
333
  	xfs_trans_add_item(tp, &efdp->efd_item);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
334
  	return efdp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
335
  }
dc42375d5   Darrick J. Wong   xfs: refactor red...
336
337
  
  /*
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
   * Free an extent and log it to the EFD. Note that the transaction is marked
   * dirty regardless of whether the extent free succeeds or fails to support the
   * EFI/EFD lifecycle rules.
   */
  static int
  xfs_trans_free_extent(
  	struct xfs_trans		*tp,
  	struct xfs_efd_log_item		*efdp,
  	xfs_fsblock_t			start_block,
  	xfs_extlen_t			ext_len,
  	const struct xfs_owner_info	*oinfo,
  	bool				skip_discard)
  {
  	struct xfs_mount		*mp = tp->t_mountp;
  	struct xfs_extent		*extp;
  	uint				next_extent;
  	xfs_agnumber_t			agno = XFS_FSB_TO_AGNO(mp, start_block);
  	xfs_agblock_t			agbno = XFS_FSB_TO_AGBNO(mp,
  								start_block);
  	int				error;
  
  	trace_xfs_bmap_free_deferred(tp->t_mountp, agno, 0, agbno, ext_len);
  
  	error = __xfs_free_extent(tp, start_block, ext_len,
  				  oinfo, XFS_AG_RESV_NONE, skip_discard);
  	/*
  	 * Mark the transaction dirty, even on error. This ensures the
  	 * transaction is aborted, which:
  	 *
  	 * 1.) releases the EFI and frees the EFD
  	 * 2.) shuts down the filesystem
  	 */
  	tp->t_flags |= XFS_TRANS_DIRTY;
  	set_bit(XFS_LI_DIRTY, &efdp->efd_item.li_flags);
  
  	next_extent = efdp->efd_next_extent;
  	ASSERT(next_extent < efdp->efd_format.efd_nextents);
  	extp = &(efdp->efd_format.efd_extents[next_extent]);
  	extp->ext_start = start_block;
  	extp->ext_len = ext_len;
  	efdp->efd_next_extent++;
  
  	return error;
  }
  
  /* Sort bmap items by AG. */
  static int
  xfs_extent_free_diff_items(
  	void				*priv,
  	struct list_head		*a,
  	struct list_head		*b)
  {
  	struct xfs_mount		*mp = priv;
  	struct xfs_extent_free_item	*ra;
  	struct xfs_extent_free_item	*rb;
  
  	ra = container_of(a, struct xfs_extent_free_item, xefi_list);
  	rb = container_of(b, struct xfs_extent_free_item, xefi_list);
  	return  XFS_FSB_TO_AGNO(mp, ra->xefi_startblock) -
  		XFS_FSB_TO_AGNO(mp, rb->xefi_startblock);
  }
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
399
400
401
402
  /* Log a free extent to the intent item. */
  STATIC void
  xfs_extent_free_log_item(
  	struct xfs_trans		*tp,
c1f09188e   Christoph Hellwig   xfs: merge the ->...
403
404
  	struct xfs_efi_log_item		*efip,
  	struct xfs_extent_free_item	*free)
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
405
  {
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
406
407
  	uint				next_extent;
  	struct xfs_extent		*extp;
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
408
409
410
411
412
413
414
415
416
417
418
419
420
421
  	tp->t_flags |= XFS_TRANS_DIRTY;
  	set_bit(XFS_LI_DIRTY, &efip->efi_item.li_flags);
  
  	/*
  	 * atomic_inc_return gives us the value after the increment;
  	 * we want to use it as an array index so we need to subtract 1 from
  	 * it.
  	 */
  	next_extent = atomic_inc_return(&efip->efi_next_extent) - 1;
  	ASSERT(next_extent < efip->efi_format.efi_nextents);
  	extp = &efip->efi_format.efi_extents[next_extent];
  	extp->ext_start = free->xefi_startblock;
  	extp->ext_len = free->xefi_blockcount;
  }
13a833333   Christoph Hellwig   xfs: turn dfp_int...
422
  static struct xfs_log_item *
c1f09188e   Christoph Hellwig   xfs: merge the ->...
423
424
425
  xfs_extent_free_create_intent(
  	struct xfs_trans		*tp,
  	struct list_head		*items,
d367a868e   Christoph Hellwig   xfs: merge the ->...
426
427
  	unsigned int			count,
  	bool				sort)
c1f09188e   Christoph Hellwig   xfs: merge the ->...
428
429
430
431
432
433
434
435
  {
  	struct xfs_mount		*mp = tp->t_mountp;
  	struct xfs_efi_log_item		*efip = xfs_efi_init(mp, count);
  	struct xfs_extent_free_item	*free;
  
  	ASSERT(count > 0);
  
  	xfs_trans_add_item(tp, &efip->efi_item);
d367a868e   Christoph Hellwig   xfs: merge the ->...
436
437
  	if (sort)
  		list_sort(mp, items, xfs_extent_free_diff_items);
c1f09188e   Christoph Hellwig   xfs: merge the ->...
438
439
  	list_for_each_entry(free, items, xefi_list)
  		xfs_extent_free_log_item(tp, efip, free);
13a833333   Christoph Hellwig   xfs: turn dfp_int...
440
  	return &efip->efi_item;
c1f09188e   Christoph Hellwig   xfs: merge the ->...
441
  }
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
442
  /* Get an EFD so we can process all the free extents. */
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
443
  static struct xfs_log_item *
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
444
445
  xfs_extent_free_create_done(
  	struct xfs_trans		*tp,
13a833333   Christoph Hellwig   xfs: turn dfp_int...
446
  	struct xfs_log_item		*intent,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
447
448
  	unsigned int			count)
  {
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
449
  	return &xfs_trans_get_efd(tp, EFI_ITEM(intent), count)->efd_item;
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
450
451
452
453
454
455
  }
  
  /* Process a free extent. */
  STATIC int
  xfs_extent_free_finish_item(
  	struct xfs_trans		*tp,
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
456
  	struct xfs_log_item		*done,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
457
  	struct list_head		*item,
3ec1b26c0   Christoph Hellwig   xfs: use a xfs_bt...
458
  	struct xfs_btree_cur		**state)
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
459
460
461
462
463
  {
  	struct xfs_extent_free_item	*free;
  	int				error;
  
  	free = container_of(item, struct xfs_extent_free_item, xefi_list);
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
464
  	error = xfs_trans_free_extent(tp, EFD_ITEM(done),
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
465
466
467
468
469
470
471
472
473
474
  			free->xefi_startblock,
  			free->xefi_blockcount,
  			&free->xefi_oinfo, free->xefi_skip_discard);
  	kmem_free(free);
  	return error;
  }
  
  /* Abort all pending EFIs. */
  STATIC void
  xfs_extent_free_abort_intent(
13a833333   Christoph Hellwig   xfs: turn dfp_int...
475
  	struct xfs_log_item		*intent)
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
476
  {
13a833333   Christoph Hellwig   xfs: turn dfp_int...
477
  	xfs_efi_release(EFI_ITEM(intent));
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
  }
  
  /* Cancel a free extent. */
  STATIC void
  xfs_extent_free_cancel_item(
  	struct list_head		*item)
  {
  	struct xfs_extent_free_item	*free;
  
  	free = container_of(item, struct xfs_extent_free_item, xefi_list);
  	kmem_free(free);
  }
  
  const struct xfs_defer_op_type xfs_extent_free_defer_type = {
  	.max_items	= XFS_EFI_MAX_FAST_EXTENTS,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
493
494
  	.create_intent	= xfs_extent_free_create_intent,
  	.abort_intent	= xfs_extent_free_abort_intent,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
495
496
497
498
499
500
501
502
503
504
505
506
  	.create_done	= xfs_extent_free_create_done,
  	.finish_item	= xfs_extent_free_finish_item,
  	.cancel_item	= xfs_extent_free_cancel_item,
  };
  
  /*
   * AGFL blocks are accounted differently in the reserve pools and are not
   * inserted into the busy extent list.
   */
  STATIC int
  xfs_agfl_free_finish_item(
  	struct xfs_trans		*tp,
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
507
  	struct xfs_log_item		*done,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
508
  	struct list_head		*item,
3ec1b26c0   Christoph Hellwig   xfs: use a xfs_bt...
509
  	struct xfs_btree_cur		**state)
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
510
511
  {
  	struct xfs_mount		*mp = tp->t_mountp;
f09d167c2   Christoph Hellwig   xfs: turn dfp_don...
512
  	struct xfs_efd_log_item		*efdp = EFD_ITEM(done);
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
  	struct xfs_extent_free_item	*free;
  	struct xfs_extent		*extp;
  	struct xfs_buf			*agbp;
  	int				error;
  	xfs_agnumber_t			agno;
  	xfs_agblock_t			agbno;
  	uint				next_extent;
  
  	free = container_of(item, struct xfs_extent_free_item, xefi_list);
  	ASSERT(free->xefi_blockcount == 1);
  	agno = XFS_FSB_TO_AGNO(mp, free->xefi_startblock);
  	agbno = XFS_FSB_TO_AGBNO(mp, free->xefi_startblock);
  
  	trace_xfs_agfl_free_deferred(mp, agno, 0, agbno, free->xefi_blockcount);
  
  	error = xfs_alloc_read_agf(mp, tp, agno, 0, &agbp);
  	if (!error)
  		error = xfs_free_agfl_block(tp, agno, agbno, agbp,
  					    &free->xefi_oinfo);
  
  	/*
  	 * Mark the transaction dirty, even on error. This ensures the
  	 * transaction is aborted, which:
  	 *
  	 * 1.) releases the EFI and frees the EFD
  	 * 2.) shuts down the filesystem
  	 */
  	tp->t_flags |= XFS_TRANS_DIRTY;
  	set_bit(XFS_LI_DIRTY, &efdp->efd_item.li_flags);
  
  	next_extent = efdp->efd_next_extent;
  	ASSERT(next_extent < efdp->efd_format.efd_nextents);
  	extp = &(efdp->efd_format.efd_extents[next_extent]);
  	extp->ext_start = free->xefi_startblock;
  	extp->ext_len = free->xefi_blockcount;
  	efdp->efd_next_extent++;
  
  	kmem_free(free);
  	return error;
  }
  
  /* sub-type with special handling for AGFL deferred frees */
  const struct xfs_defer_op_type xfs_agfl_free_defer_type = {
  	.max_items	= XFS_EFI_MAX_FAST_EXTENTS,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
557
558
  	.create_intent	= xfs_extent_free_create_intent,
  	.abort_intent	= xfs_extent_free_abort_intent,
81f400417   Christoph Hellwig   xfs: merge xfs_tr...
559
560
561
562
563
564
  	.create_done	= xfs_extent_free_create_done,
  	.finish_item	= xfs_agfl_free_finish_item,
  	.cancel_item	= xfs_extent_free_cancel_item,
  };
  
  /*
dc42375d5   Darrick J. Wong   xfs: refactor red...
565
566
567
   * Process an extent free intent item that was recovered from
   * the log.  We need to free the extents that it describes.
   */
10d0c6e06   Darrick J. Wong   xfs: refactor rec...
568
  STATIC int
96b60f826   Darrick J. Wong   xfs: refactor int...
569
570
  xfs_efi_item_recover(
  	struct xfs_log_item		*lip,
e6fff81e4   Darrick J. Wong   xfs: proper repla...
571
  	struct list_head		*capture_list)
dc42375d5   Darrick J. Wong   xfs: refactor red...
572
  {
96b60f826   Darrick J. Wong   xfs: refactor int...
573
  	struct xfs_efi_log_item		*efip = EFI_ITEM(lip);
e6fff81e4   Darrick J. Wong   xfs: proper repla...
574
  	struct xfs_mount		*mp = lip->li_mountp;
96b60f826   Darrick J. Wong   xfs: refactor int...
575
576
577
578
579
580
  	struct xfs_efd_log_item		*efdp;
  	struct xfs_trans		*tp;
  	struct xfs_extent		*extp;
  	xfs_fsblock_t			startblock_fsb;
  	int				i;
  	int				error = 0;
dc42375d5   Darrick J. Wong   xfs: refactor red...
581

dc42375d5   Darrick J. Wong   xfs: refactor red...
582
583
584
585
586
587
  	/*
  	 * First check the validity of the extents described by the
  	 * EFI.  If any are bad, then assume that all are bad and
  	 * just toss the EFI.
  	 */
  	for (i = 0; i < efip->efi_format.efi_nextents; i++) {
e127fafd1   Darrick J. Wong   xfs: remove unnec...
588
  		extp = &efip->efi_format.efi_extents[i];
dc42375d5   Darrick J. Wong   xfs: refactor red...
589
590
  		startblock_fsb = XFS_BB_TO_FSB(mp,
  				   XFS_FSB_TO_DADDR(mp, extp->ext_start));
e127fafd1   Darrick J. Wong   xfs: remove unnec...
591
592
593
  		if (startblock_fsb == 0 ||
  		    extp->ext_len == 0 ||
  		    startblock_fsb >= mp->m_sb.sb_dblocks ||
384ff09ba   Darrick J. Wong   xfs: don't releas...
594
  		    extp->ext_len >= mp->m_sb.sb_agblocks)
895e196fb   Darrick J. Wong   xfs: convert EIO ...
595
  			return -EFSCORRUPTED;
dc42375d5   Darrick J. Wong   xfs: refactor red...
596
597
598
599
600
601
602
603
  	}
  
  	error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, 0, 0, 0, &tp);
  	if (error)
  		return error;
  	efdp = xfs_trans_get_efd(tp, efip, efip->efi_format.efi_nextents);
  
  	for (i = 0; i < efip->efi_format.efi_nextents; i++) {
e127fafd1   Darrick J. Wong   xfs: remove unnec...
604
  		extp = &efip->efi_format.efi_extents[i];
dc42375d5   Darrick J. Wong   xfs: refactor red...
605
  		error = xfs_trans_free_extent(tp, efdp, extp->ext_start,
7280fedaf   Darrick J. Wong   xfs: remove xfs_r...
606
607
  					      extp->ext_len,
  					      &XFS_RMAP_OINFO_ANY_OWNER, false);
dc42375d5   Darrick J. Wong   xfs: refactor red...
608
609
610
611
  		if (error)
  			goto abort_error;
  
  	}
ff4ab5e02   Darrick J. Wong   xfs: fix an incor...
612
  	return xfs_defer_ops_capture_and_commit(tp, NULL, capture_list);
dc42375d5   Darrick J. Wong   xfs: refactor red...
613
614
615
616
617
  
  abort_error:
  	xfs_trans_cancel(tp);
  	return error;
  }
86ffa471d   Darrick J. Wong   xfs: refactor log...
618

154c733a3   Darrick J. Wong   xfs: refactor rel...
619
620
621
622
623
624
625
  STATIC bool
  xfs_efi_item_match(
  	struct xfs_log_item	*lip,
  	uint64_t		intent_id)
  {
  	return EFI_ITEM(lip)->efi_format.efi_id == intent_id;
  }
4e919af78   Darrick J. Wong   xfs: periodically...
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
  /* Relog an intent item to push the log tail forward. */
  static struct xfs_log_item *
  xfs_efi_item_relog(
  	struct xfs_log_item		*intent,
  	struct xfs_trans		*tp)
  {
  	struct xfs_efd_log_item		*efdp;
  	struct xfs_efi_log_item		*efip;
  	struct xfs_extent		*extp;
  	unsigned int			count;
  
  	count = EFI_ITEM(intent)->efi_format.efi_nextents;
  	extp = EFI_ITEM(intent)->efi_format.efi_extents;
  
  	tp->t_flags |= XFS_TRANS_DIRTY;
  	efdp = xfs_trans_get_efd(tp, EFI_ITEM(intent), count);
  	efdp->efd_next_extent = count;
  	memcpy(efdp->efd_format.efd_extents, extp, count * sizeof(*extp));
  	set_bit(XFS_LI_DIRTY, &efdp->efd_item.li_flags);
  
  	efip = xfs_efi_init(tp->t_mountp, count);
  	memcpy(efip->efi_format.efi_extents, extp, count * sizeof(*extp));
  	atomic_set(&efip->efi_next_extent, count);
  	xfs_trans_add_item(tp, &efip->efi_item);
  	set_bit(XFS_LI_DIRTY, &efip->efi_item.li_flags);
  	return &efip->efi_item;
  }
10d0c6e06   Darrick J. Wong   xfs: refactor rec...
653
654
655
656
657
658
  static const struct xfs_item_ops xfs_efi_item_ops = {
  	.iop_size	= xfs_efi_item_size,
  	.iop_format	= xfs_efi_item_format,
  	.iop_unpin	= xfs_efi_item_unpin,
  	.iop_release	= xfs_efi_item_release,
  	.iop_recover	= xfs_efi_item_recover,
154c733a3   Darrick J. Wong   xfs: refactor rel...
659
  	.iop_match	= xfs_efi_item_match,
4e919af78   Darrick J. Wong   xfs: periodically...
660
  	.iop_relog	= xfs_efi_item_relog,
10d0c6e06   Darrick J. Wong   xfs: refactor rec...
661
  };
9817aa80d   Darrick J. Wong   xfs: refactor log...
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
  /*
   * This routine is called to create an in-core extent free intent
   * item from the efi format structure which was logged on disk.
   * It allocates an in-core efi, copies the extents from the format
   * structure into it, and adds the efi to the AIL with the given
   * LSN.
   */
  STATIC int
  xlog_recover_efi_commit_pass2(
  	struct xlog			*log,
  	struct list_head		*buffer_list,
  	struct xlog_recover_item	*item,
  	xfs_lsn_t			lsn)
  {
  	struct xfs_mount		*mp = log->l_mp;
  	struct xfs_efi_log_item		*efip;
  	struct xfs_efi_log_format	*efi_formatp;
  	int				error;
  
  	efi_formatp = item->ri_buf[0].i_addr;
  
  	efip = xfs_efi_init(mp, efi_formatp->efi_nextents);
  	error = xfs_efi_copy_format(&item->ri_buf[0], &efip->efi_format);
  	if (error) {
  		xfs_efi_item_free(efip);
  		return error;
  	}
  	atomic_set(&efip->efi_next_extent, efi_formatp->efi_nextents);
9817aa80d   Darrick J. Wong   xfs: refactor log...
690
  	/*
86a371741   Darrick J. Wong   xfs: refactor add...
691
692
  	 * Insert the intent into the AIL directly and drop one reference so
  	 * that finishing or canceling the work will drop the other.
9817aa80d   Darrick J. Wong   xfs: refactor log...
693
  	 */
86a371741   Darrick J. Wong   xfs: refactor add...
694
  	xfs_trans_ail_insert(log->l_ailp, &efip->efi_item, lsn);
9817aa80d   Darrick J. Wong   xfs: refactor log...
695
696
697
  	xfs_efi_release(efip);
  	return 0;
  }
86ffa471d   Darrick J. Wong   xfs: refactor log...
698
699
  const struct xlog_recover_item_ops xlog_efi_item_ops = {
  	.item_type		= XFS_LI_EFI,
9817aa80d   Darrick J. Wong   xfs: refactor log...
700
  	.commit_pass2		= xlog_recover_efi_commit_pass2,
86ffa471d   Darrick J. Wong   xfs: refactor log...
701
  };
9817aa80d   Darrick J. Wong   xfs: refactor log...
702
703
704
705
706
707
708
709
710
711
712
713
714
715
  /*
   * This routine is called when an EFD format structure is found in a committed
   * transaction in the log. Its purpose is to cancel the corresponding EFI if it
   * was still in the log. To do this it searches the AIL for the EFI with an id
   * equal to that in the EFD format structure. If we find it we drop the EFD
   * reference, which removes the EFI from the AIL and frees it.
   */
  STATIC int
  xlog_recover_efd_commit_pass2(
  	struct xlog			*log,
  	struct list_head		*buffer_list,
  	struct xlog_recover_item	*item,
  	xfs_lsn_t			lsn)
  {
9817aa80d   Darrick J. Wong   xfs: refactor log...
716
  	struct xfs_efd_log_format	*efd_formatp;
9817aa80d   Darrick J. Wong   xfs: refactor log...
717
718
719
720
721
722
  
  	efd_formatp = item->ri_buf[0].i_addr;
  	ASSERT((item->ri_buf[0].i_len == (sizeof(xfs_efd_log_format_32_t) +
  		((efd_formatp->efd_nextents - 1) * sizeof(xfs_extent_32_t)))) ||
  	       (item->ri_buf[0].i_len == (sizeof(xfs_efd_log_format_64_t) +
  		((efd_formatp->efd_nextents - 1) * sizeof(xfs_extent_64_t)))));
9817aa80d   Darrick J. Wong   xfs: refactor log...
723

154c733a3   Darrick J. Wong   xfs: refactor rel...
724
  	xlog_recover_release_intent(log, XFS_LI_EFI, efd_formatp->efd_efi_id);
9817aa80d   Darrick J. Wong   xfs: refactor log...
725
726
  	return 0;
  }
86ffa471d   Darrick J. Wong   xfs: refactor log...
727
728
  const struct xlog_recover_item_ops xlog_efd_item_ops = {
  	.item_type		= XFS_LI_EFD,
9817aa80d   Darrick J. Wong   xfs: refactor log...
729
  	.commit_pass2		= xlog_recover_efd_commit_pass2,
86ffa471d   Darrick J. Wong   xfs: refactor log...
730
  };