Blame view

fs/xfs/xfs_extfree_item.c 13.9 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
  /*
7b7187698   Nathan Scott   [XFS] Update lice...
2
3
   * Copyright (c) 2000-2001,2005 Silicon Graphics, Inc.
   * All Rights Reserved.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
4
   *
7b7187698   Nathan Scott   [XFS] Update lice...
5
6
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License as
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
7
8
   * published by the Free Software Foundation.
   *
7b7187698   Nathan Scott   [XFS] Update lice...
9
10
11
12
   * This program is distributed in the hope that it would be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   * GNU General Public License for more details.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
13
   *
7b7187698   Nathan Scott   [XFS] Update lice...
14
15
16
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write the Free Software Foundation,
   * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
17
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18
  #include "xfs.h"
a844f4510   Nathan Scott   [XFS] Remove xfs_...
19
  #include "xfs_fs.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
  #include "xfs_types.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
21
  #include "xfs_log.h"
a844f4510   Nathan Scott   [XFS] Remove xfs_...
22
  #include "xfs_inum.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
23
24
25
  #include "xfs_trans.h"
  #include "xfs_buf_item.h"
  #include "xfs_sb.h"
da353b0d6   David Chinner   [XFS] Radix tree ...
26
  #include "xfs_ag.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
27
28
29
30
31
32
33
  #include "xfs_mount.h"
  #include "xfs_trans_priv.h"
  #include "xfs_extfree_item.h"
  
  
  kmem_zone_t	*xfs_efi_zone;
  kmem_zone_t	*xfs_efd_zone;
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
34
35
36
37
  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
38

7d795ca34   Christoph Hellwig   [XFS] consolidate...
39
  void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
40
41
  xfs_efi_item_free(
  	struct xfs_efi_log_item	*efip)
7d795ca34   Christoph Hellwig   [XFS] consolidate...
42
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
43
  	if (efip->efi_format.efi_nextents > XFS_EFI_MAX_FAST_EXTENTS)
f0e2d93c2   Denys Vlasenko   [XFS] Remove unus...
44
  		kmem_free(efip);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
45
  	else
7d795ca34   Christoph Hellwig   [XFS] consolidate...
46
  		kmem_zone_free(xfs_efi_zone, efip);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
47
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
48
49
  
  /*
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
   * 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
   * test_and_clear_bit(XFS_EFI_COMMITTED) to ensure only the last caller frees
   * the EFI.
   */
  STATIC void
  __xfs_efi_release(
  	struct xfs_efi_log_item	*efip)
  {
  	struct xfs_ail		*ailp = efip->efi_item.li_ailp;
  
  	if (!test_and_clear_bit(XFS_EFI_COMMITTED, &efip->efi_flags)) {
  		spin_lock(&ailp->xa_lock);
  		/* xfs_trans_ail_delete() drops the AIL lock. */
  		xfs_trans_ail_delete(ailp, &efip->efi_item);
  		xfs_efi_item_free(efip);
  	}
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
72
73
74
75
   * 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.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76
  STATIC uint
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
77
78
  xfs_efi_item_size(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
79
80
81
82
83
84
85
86
87
88
89
90
  {
  	return 1;
  }
  
  /*
   * 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...
91
92
93
  xfs_efi_item_format(
  	struct xfs_log_item	*lip,
  	struct xfs_log_iovec	*log_vector)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
94
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
95
96
  	struct xfs_efi_log_item	*efip = EFI_ITEM(lip);
  	uint			size;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
97

b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
98
99
  	ASSERT(atomic_read(&efip->efi_next_extent) ==
  				efip->efi_format.efi_nextents);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
100
101
102
103
104
105
  
  	efip->efi_format.efi_type = XFS_LI_EFI;
  
  	size = sizeof(xfs_efi_log_format_t);
  	size += (efip->efi_format.efi_nextents - 1) * sizeof(xfs_extent_t);
  	efip->efi_format.efi_size = 1;
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
106
  	log_vector->i_addr = &efip->efi_format;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
107
  	log_vector->i_len = size;
4139b3b33   Christoph Hellwig   xfs: kill XLOG_VE...
108
  	log_vector->i_type = XLOG_REG_TYPE_EFI_FORMAT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
109
110
111
112
113
114
115
  	ASSERT(size >= sizeof(xfs_efi_log_format_t));
  }
  
  
  /*
   * Pinning has no meaning for an efi item, so just return.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
116
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
117
118
  xfs_efi_item_pin(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
119
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
120
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
121
  /*
9c5f8414e   Dave Chinner   xfs: fix EFI tran...
122
123
124
125
   * While EFIs cannot really be pinned, the unpin operation is the last place at
   * which the EFI is manipulated during a transaction.  If we are being asked to
   * remove the EFI it's because the transaction has been cancelled and by
   * definition that means the EFI cannot be in the AIL so remove it from the
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
126
127
   * transaction and free it.  Otherwise coordinate with xfs_efi_release() (via
   * XFS_EFI_COMMITTED) to determine who gets to free the EFI.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
128
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
129
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
130
131
132
  xfs_efi_item_unpin(
  	struct xfs_log_item	*lip,
  	int			remove)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
133
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
134
  	struct xfs_efi_log_item	*efip = EFI_ITEM(lip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
135

9c5f8414e   Dave Chinner   xfs: fix EFI tran...
136
137
  	if (remove) {
  		ASSERT(!(lip->li_flags & XFS_LI_IN_AIL));
e34a314c5   Dave Chinner   xfs: fix efi item...
138
139
  		if (lip->li_desc)
  			xfs_trans_del_item(lip);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
140
  		xfs_efi_item_free(efip);
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
141
  		return;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
142
  	}
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
143
  	__xfs_efi_release(efip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
144
145
146
147
148
149
150
151
152
  }
  
  /*
   * Efi items have no locking or pushing.  However, since EFIs are
   * pulled from the AIL when their corresponding EFDs are committed
   * to disk, their situation is very similar to being pinned.  Return
   * XFS_ITEM_PINNED so that the caller will eventually flush the log.
   * This should help in getting the EFI out of the AIL.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
  STATIC uint
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
154
155
  xfs_efi_item_trylock(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
156
157
158
159
160
161
162
  {
  	return XFS_ITEM_PINNED;
  }
  
  /*
   * Efi items have no locking, so just return.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
164
165
  xfs_efi_item_unlock(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
166
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
167
168
  	if (lip->li_flags & XFS_LI_ABORTED)
  		xfs_efi_item_free(EFI_ITEM(lip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
169
170
171
  }
  
  /*
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
172
173
174
175
176
   * The EFI is logged only once and cannot be moved in the log, so simply return
   * the lsn at which it's been logged.  For bulk transaction committed
   * processing, the EFI may be processed but not yet unpinned prior to the EFD
   * being processed. Set the XFS_EFI_COMMITTED flag so this case can be detected
   * when processing the EFD.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
177
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
178
  STATIC xfs_lsn_t
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
179
180
181
  xfs_efi_item_committed(
  	struct xfs_log_item	*lip,
  	xfs_lsn_t		lsn)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
182
  {
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
183
184
185
  	struct xfs_efi_log_item	*efip = EFI_ITEM(lip);
  
  	set_bit(XFS_EFI_COMMITTED, &efip->efi_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
186
187
188
189
  	return lsn;
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
190
191
192
193
   * There isn't much you can do to push on an efi item.  It is simply
   * stuck waiting for all of its corresponding efd items to be
   * committed to disk.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
194
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
195
196
  xfs_efi_item_push(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
197
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198
199
200
201
202
203
204
205
206
  }
  
  /*
   * The EFI dependency tracking op doesn't do squat.  It can't because
   * it doesn't know where the free extent is coming from.  The dependency
   * tracking has to be handled by the "enclosing" metadata object.  For
   * example, for inodes, the inode is locked throughout the extent freeing
   * so the dependency should be recorded there.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
207
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
208
209
210
  xfs_efi_item_committing(
  	struct xfs_log_item	*lip,
  	xfs_lsn_t		lsn)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
211
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
212
213
214
215
216
  }
  
  /*
   * This is the ops vector shared by all efi log items.
   */
272e42b21   Christoph Hellwig   xfs: constify xfs...
217
  static const struct xfs_item_ops xfs_efi_item_ops = {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
218
219
220
221
222
223
224
225
226
  	.iop_size	= xfs_efi_item_size,
  	.iop_format	= xfs_efi_item_format,
  	.iop_pin	= xfs_efi_item_pin,
  	.iop_unpin	= xfs_efi_item_unpin,
  	.iop_trylock	= xfs_efi_item_trylock,
  	.iop_unlock	= xfs_efi_item_unlock,
  	.iop_committed	= xfs_efi_item_committed,
  	.iop_push	= xfs_efi_item_push,
  	.iop_committing = xfs_efi_item_committing
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
227
228
229
230
231
232
  };
  
  
  /*
   * Allocate and initialize an efi item with the given number of extents.
   */
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
233
234
235
236
  struct xfs_efi_log_item *
  xfs_efi_init(
  	struct xfs_mount	*mp,
  	uint			nextents)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
237
238
  
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
239
  	struct xfs_efi_log_item	*efip;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240
241
242
243
244
245
  	uint			size;
  
  	ASSERT(nextents > 0);
  	if (nextents > XFS_EFI_MAX_FAST_EXTENTS) {
  		size = (uint)(sizeof(xfs_efi_log_item_t) +
  			((nextents - 1) * sizeof(xfs_extent_t)));
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
246
  		efip = kmem_zalloc(size, KM_SLEEP);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
247
  	} else {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
248
  		efip = kmem_zone_zalloc(xfs_efi_zone, KM_SLEEP);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
249
  	}
43f5efc5b   Dave Chinner   xfs: factor log i...
250
  	xfs_log_item_init(mp, &efip->efi_item, XFS_LI_EFI, &xfs_efi_item_ops);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
251
252
  	efip->efi_format.efi_nextents = nextents;
  	efip->efi_format.efi_id = (__psint_t)(void*)efip;
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
253
  	atomic_set(&efip->efi_next_extent, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254

7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
255
  	return efip;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
256
257
258
  }
  
  /*
6d192a9b8   Tim Shimmin   [XFS] inode items...
259
260
261
262
263
264
265
266
267
   * 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.
   */
  int
  xfs_efi_copy_format(xfs_log_iovec_t *buf, xfs_efi_log_format_t *dst_efi_fmt)
  {
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
268
  	xfs_efi_log_format_t *src_efi_fmt = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
269
270
271
272
273
274
275
276
277
278
279
280
  	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_...
281
  		xfs_efi_log_format_32_t *src_efi_fmt_32 = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
282
283
284
285
286
287
288
289
290
291
292
293
294
  
  		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_...
295
  		xfs_efi_log_format_64_t *src_efi_fmt_64 = buf->i_addr;
6d192a9b8   Tim Shimmin   [XFS] inode items...
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
  
  		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;
  	}
  	return EFSCORRUPTED;
  }
  
  /*
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
313
314
315
316
   * This is called by the efd item code below to release references to the given
   * efi item.  Each efd calls this with the number of extents that it has
   * logged, and when the sum of these reaches the total number of extents logged
   * by this efi item we can free the efi item.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
317
318
319
320
321
   */
  void
  xfs_efi_release(xfs_efi_log_item_t	*efip,
  		uint			nextents)
  {
b199c8a4b   Dave Chinner   xfs: Pull EFI/EFD...
322
323
324
  	ASSERT(atomic_read(&efip->efi_next_extent) >= nextents);
  	if (atomic_sub_and_test(nextents, &efip->efi_next_extent))
  		__xfs_efi_release(efip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
325
  }
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
326
  static inline struct xfs_efd_log_item *EFD_ITEM(struct xfs_log_item *lip)
7d795ca34   Christoph Hellwig   [XFS] consolidate...
327
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
328
329
  	return container_of(lip, struct xfs_efd_log_item, efd_item);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
330

7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
331
332
333
334
  STATIC void
  xfs_efd_item_free(struct xfs_efd_log_item *efdp)
  {
  	if (efdp->efd_format.efd_nextents > XFS_EFD_MAX_FAST_EXTENTS)
f0e2d93c2   Denys Vlasenko   [XFS] Remove unus...
335
  		kmem_free(efdp);
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
336
  	else
7d795ca34   Christoph Hellwig   [XFS] consolidate...
337
  		kmem_zone_free(xfs_efd_zone, efdp);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
338
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
339
340
341
342
343
344
  
  /*
   * 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.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
345
  STATIC uint
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
346
347
  xfs_efd_item_size(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
348
349
350
351
352
353
354
355
356
357
358
359
  {
  	return 1;
  }
  
  /*
   * 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...
360
361
362
  xfs_efd_item_format(
  	struct xfs_log_item	*lip,
  	struct xfs_log_iovec	*log_vector)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
363
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
364
365
  	struct xfs_efd_log_item	*efdp = EFD_ITEM(lip);
  	uint			size;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
366
367
368
369
370
371
372
373
  
  	ASSERT(efdp->efd_next_extent == efdp->efd_format.efd_nextents);
  
  	efdp->efd_format.efd_type = XFS_LI_EFD;
  
  	size = sizeof(xfs_efd_log_format_t);
  	size += (efdp->efd_format.efd_nextents - 1) * sizeof(xfs_extent_t);
  	efdp->efd_format.efd_size = 1;
4e0d5f926   Christoph Hellwig   xfs: fix the xfs_...
374
  	log_vector->i_addr = &efdp->efd_format;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
375
  	log_vector->i_len = size;
4139b3b33   Christoph Hellwig   xfs: kill XLOG_VE...
376
  	log_vector->i_type = XLOG_REG_TYPE_EFD_FORMAT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
377
378
  	ASSERT(size >= sizeof(xfs_efd_log_format_t));
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
379
380
381
  /*
   * Pinning has no meaning for an efd item, so just return.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
382
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
383
384
  xfs_efd_item_pin(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
385
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
386
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
387
388
389
390
  /*
   * Since pinning has no meaning for an efd item, unpinning does
   * not either.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
391
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
392
393
394
  xfs_efd_item_unpin(
  	struct xfs_log_item	*lip,
  	int			remove)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
395
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
396
397
398
399
400
  }
  
  /*
   * Efd items have no locking, so just return success.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
401
  STATIC uint
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
402
403
  xfs_efd_item_trylock(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
404
405
406
407
408
409
410
411
  {
  	return XFS_ITEM_LOCKED;
  }
  
  /*
   * Efd items have no locking or pushing, so return failure
   * so that the caller doesn't bother with us.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
412
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
413
414
  xfs_efd_item_unlock(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
415
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
416
417
  	if (lip->li_flags & XFS_LI_ABORTED)
  		xfs_efd_item_free(EFD_ITEM(lip));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
418
419
420
421
422
423
424
425
426
  }
  
  /*
   * When the efd item is committed to disk, all we need to do
   * is delete our reference to our partner efi item and then
   * free ourselves.  Since we're freeing ourselves we must
   * return -1 to keep the transaction code from further referencing
   * this item.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
427
  STATIC xfs_lsn_t
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
428
429
430
  xfs_efd_item_committed(
  	struct xfs_log_item	*lip,
  	xfs_lsn_t		lsn)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
431
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
432
  	struct xfs_efd_log_item	*efdp = EFD_ITEM(lip);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
433
434
435
436
  	/*
  	 * If we got a log I/O error, it's always the case that the LR with the
  	 * EFI got unpinned and freed before the EFD got aborted.
  	 */
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
437
  	if (!(lip->li_flags & XFS_LI_ABORTED))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
438
  		xfs_efi_release(efdp->efd_efip, efdp->efd_format.efd_nextents);
7d795ca34   Christoph Hellwig   [XFS] consolidate...
439
  	xfs_efd_item_free(efdp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
440
441
442
443
  	return (xfs_lsn_t)-1;
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
444
445
446
   * There isn't much you can do to push on an efd item.  It is simply
   * stuck waiting for the log to be flushed to disk.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
447
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
448
449
  xfs_efd_item_push(
  	struct xfs_log_item	*lip)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
450
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
451
452
453
454
455
456
457
458
459
  }
  
  /*
   * The EFD dependency tracking op doesn't do squat.  It can't because
   * it doesn't know where the free extent is coming from.  The dependency
   * tracking has to be handled by the "enclosing" metadata object.  For
   * example, for inodes, the inode is locked throughout the extent freeing
   * so the dependency should be recorded there.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
460
  STATIC void
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
461
462
463
  xfs_efd_item_committing(
  	struct xfs_log_item	*lip,
  	xfs_lsn_t		lsn)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
464
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
465
466
467
468
469
  }
  
  /*
   * This is the ops vector shared by all efd log items.
   */
272e42b21   Christoph Hellwig   xfs: constify xfs...
470
  static const struct xfs_item_ops xfs_efd_item_ops = {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
471
472
473
474
475
476
477
478
479
  	.iop_size	= xfs_efd_item_size,
  	.iop_format	= xfs_efd_item_format,
  	.iop_pin	= xfs_efd_item_pin,
  	.iop_unpin	= xfs_efd_item_unpin,
  	.iop_trylock	= xfs_efd_item_trylock,
  	.iop_unlock	= xfs_efd_item_unlock,
  	.iop_committed	= xfs_efd_item_committed,
  	.iop_push	= xfs_efd_item_push,
  	.iop_committing = xfs_efd_item_committing
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
480
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
481
482
483
  /*
   * Allocate and initialize an efd item with the given number of extents.
   */
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
484
485
486
487
488
  struct xfs_efd_log_item *
  xfs_efd_init(
  	struct xfs_mount	*mp,
  	struct xfs_efi_log_item	*efip,
  	uint			nextents)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
489
490
  
  {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
491
  	struct xfs_efd_log_item	*efdp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
492
493
494
495
496
497
  	uint			size;
  
  	ASSERT(nextents > 0);
  	if (nextents > XFS_EFD_MAX_FAST_EXTENTS) {
  		size = (uint)(sizeof(xfs_efd_log_item_t) +
  			((nextents - 1) * sizeof(xfs_extent_t)));
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
498
  		efdp = kmem_zalloc(size, KM_SLEEP);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
499
  	} else {
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
500
  		efdp = kmem_zone_zalloc(xfs_efd_zone, KM_SLEEP);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
501
  	}
43f5efc5b   Dave Chinner   xfs: factor log i...
502
  	xfs_log_item_init(mp, &efdp->efd_item, XFS_LI_EFD, &xfs_efd_item_ops);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
503
504
505
  	efdp->efd_efip = efip;
  	efdp->efd_format.efd_nextents = nextents;
  	efdp->efd_format.efd_efi_id = efip->efi_format.efi_id;
7bfa31d8e   Christoph Hellwig   xfs: give xfs_ite...
506
  	return efdp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
507
  }