xref: /openbmc/linux/fs/xfs/xfs_dquot_item.c (revision 57e809561118a4db2e19d31282761ca062fd6014)
1c59d87c4SChristoph Hellwig /*
2c59d87c4SChristoph Hellwig  * Copyright (c) 2000-2003 Silicon Graphics, Inc.
3c59d87c4SChristoph Hellwig  * All Rights Reserved.
4c59d87c4SChristoph Hellwig  *
5c59d87c4SChristoph Hellwig  * This program is free software; you can redistribute it and/or
6c59d87c4SChristoph Hellwig  * modify it under the terms of the GNU General Public License as
7c59d87c4SChristoph Hellwig  * published by the Free Software Foundation.
8c59d87c4SChristoph Hellwig  *
9c59d87c4SChristoph Hellwig  * This program is distributed in the hope that it would be useful,
10c59d87c4SChristoph Hellwig  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11c59d87c4SChristoph Hellwig  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12c59d87c4SChristoph Hellwig  * GNU General Public License for more details.
13c59d87c4SChristoph Hellwig  *
14c59d87c4SChristoph Hellwig  * You should have received a copy of the GNU General Public License
15c59d87c4SChristoph Hellwig  * along with this program; if not, write the Free Software Foundation,
16c59d87c4SChristoph Hellwig  * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17c59d87c4SChristoph Hellwig  */
18c59d87c4SChristoph Hellwig #include "xfs.h"
19c59d87c4SChristoph Hellwig #include "xfs_fs.h"
206ca1c906SDave Chinner #include "xfs_format.h"
21239880efSDave Chinner #include "xfs_log_format.h"
22239880efSDave Chinner #include "xfs_trans_resv.h"
23c59d87c4SChristoph Hellwig #include "xfs_mount.h"
24c59d87c4SChristoph Hellwig #include "xfs_inode.h"
25a4fbe6abSDave Chinner #include "xfs_quota.h"
26c59d87c4SChristoph Hellwig #include "xfs_error.h"
27239880efSDave Chinner #include "xfs_trans.h"
28c59d87c4SChristoph Hellwig #include "xfs_buf_item.h"
29c59d87c4SChristoph Hellwig #include "xfs_trans_priv.h"
30c59d87c4SChristoph Hellwig #include "xfs_qm.h"
31239880efSDave Chinner #include "xfs_log.h"
32c59d87c4SChristoph Hellwig 
33c59d87c4SChristoph Hellwig static inline struct xfs_dq_logitem *DQUOT_ITEM(struct xfs_log_item *lip)
34c59d87c4SChristoph Hellwig {
35c59d87c4SChristoph Hellwig 	return container_of(lip, struct xfs_dq_logitem, qli_item);
36c59d87c4SChristoph Hellwig }
37c59d87c4SChristoph Hellwig 
38c59d87c4SChristoph Hellwig /*
39c59d87c4SChristoph Hellwig  * returns the number of iovecs needed to log the given dquot item.
40c59d87c4SChristoph Hellwig  */
41166d1368SDave Chinner STATIC void
42c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_size(
43166d1368SDave Chinner 	struct xfs_log_item	*lip,
44166d1368SDave Chinner 	int			*nvecs,
45166d1368SDave Chinner 	int			*nbytes)
46c59d87c4SChristoph Hellwig {
47166d1368SDave Chinner 	*nvecs += 2;
48166d1368SDave Chinner 	*nbytes += sizeof(struct xfs_dq_logformat) +
49166d1368SDave Chinner 		   sizeof(struct xfs_disk_dquot);
50c59d87c4SChristoph Hellwig }
51c59d87c4SChristoph Hellwig 
52c59d87c4SChristoph Hellwig /*
53c59d87c4SChristoph Hellwig  * fills in the vector of log iovecs for the given dquot log item.
54c59d87c4SChristoph Hellwig  */
55c59d87c4SChristoph Hellwig STATIC void
56c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_format(
57c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
58bde7cff6SChristoph Hellwig 	struct xfs_log_vec	*lv)
59c59d87c4SChristoph Hellwig {
60c59d87c4SChristoph Hellwig 	struct xfs_dq_logitem	*qlip = DQUOT_ITEM(lip);
61bde7cff6SChristoph Hellwig 	struct xfs_log_iovec	*vecp = NULL;
62ce8e9629SChristoph Hellwig 	struct xfs_dq_logformat	*qlf;
63c59d87c4SChristoph Hellwig 
64ce8e9629SChristoph Hellwig 	qlf = xlog_prepare_iovec(lv, &vecp, XLOG_REG_TYPE_QFORMAT);
65ce8e9629SChristoph Hellwig 	qlf->qlf_type = XFS_LI_DQUOT;
66ce8e9629SChristoph Hellwig 	qlf->qlf_size = 2;
67ce8e9629SChristoph Hellwig 	qlf->qlf_id = be32_to_cpu(qlip->qli_dquot->q_core.d_id);
68ce8e9629SChristoph Hellwig 	qlf->qlf_blkno = qlip->qli_dquot->q_blkno;
69ce8e9629SChristoph Hellwig 	qlf->qlf_len = 1;
70ce8e9629SChristoph Hellwig 	qlf->qlf_boffset = qlip->qli_dquot->q_bufoffset;
71ce8e9629SChristoph Hellwig 	xlog_finish_iovec(lv, vecp, sizeof(struct xfs_dq_logformat));
72bde7cff6SChristoph Hellwig 
73bde7cff6SChristoph Hellwig 	xlog_copy_iovec(lv, &vecp, XLOG_REG_TYPE_DQUOT,
74bde7cff6SChristoph Hellwig 			&qlip->qli_dquot->q_core,
75bde7cff6SChristoph Hellwig 			sizeof(struct xfs_disk_dquot));
76c59d87c4SChristoph Hellwig }
77c59d87c4SChristoph Hellwig 
78c59d87c4SChristoph Hellwig /*
79c59d87c4SChristoph Hellwig  * Increment the pin count of the given dquot.
80c59d87c4SChristoph Hellwig  */
81c59d87c4SChristoph Hellwig STATIC void
82c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_pin(
83c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip)
84c59d87c4SChristoph Hellwig {
85c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp = DQUOT_ITEM(lip)->qli_dquot;
86c59d87c4SChristoph Hellwig 
87c59d87c4SChristoph Hellwig 	ASSERT(XFS_DQ_IS_LOCKED(dqp));
88c59d87c4SChristoph Hellwig 	atomic_inc(&dqp->q_pincount);
89c59d87c4SChristoph Hellwig }
90c59d87c4SChristoph Hellwig 
91c59d87c4SChristoph Hellwig /*
92c59d87c4SChristoph Hellwig  * Decrement the pin count of the given dquot, and wake up
93c59d87c4SChristoph Hellwig  * anyone in xfs_dqwait_unpin() if the count goes to 0.	 The
94c59d87c4SChristoph Hellwig  * dquot must have been previously pinned with a call to
95c59d87c4SChristoph Hellwig  * xfs_qm_dquot_logitem_pin().
96c59d87c4SChristoph Hellwig  */
97c59d87c4SChristoph Hellwig STATIC void
98c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_unpin(
99c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
100c59d87c4SChristoph Hellwig 	int			remove)
101c59d87c4SChristoph Hellwig {
102c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp = DQUOT_ITEM(lip)->qli_dquot;
103c59d87c4SChristoph Hellwig 
104c59d87c4SChristoph Hellwig 	ASSERT(atomic_read(&dqp->q_pincount) > 0);
105c59d87c4SChristoph Hellwig 	if (atomic_dec_and_test(&dqp->q_pincount))
106c59d87c4SChristoph Hellwig 		wake_up(&dqp->q_pinwait);
107c59d87c4SChristoph Hellwig }
108c59d87c4SChristoph Hellwig 
109c59d87c4SChristoph Hellwig STATIC xfs_lsn_t
110c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_committed(
111c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
112c59d87c4SChristoph Hellwig 	xfs_lsn_t		lsn)
113c59d87c4SChristoph Hellwig {
114c59d87c4SChristoph Hellwig 	/*
115c59d87c4SChristoph Hellwig 	 * We always re-log the entire dquot when it becomes dirty,
116c59d87c4SChristoph Hellwig 	 * so, the latest copy _is_ the only one that matters.
117c59d87c4SChristoph Hellwig 	 */
118c59d87c4SChristoph Hellwig 	return lsn;
119c59d87c4SChristoph Hellwig }
120c59d87c4SChristoph Hellwig 
121c59d87c4SChristoph Hellwig /*
122c59d87c4SChristoph Hellwig  * This is called to wait for the given dquot to be unpinned.
123c59d87c4SChristoph Hellwig  * Most of these pin/unpin routines are plagiarized from inode code.
124c59d87c4SChristoph Hellwig  */
125c59d87c4SChristoph Hellwig void
126c59d87c4SChristoph Hellwig xfs_qm_dqunpin_wait(
127c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp)
128c59d87c4SChristoph Hellwig {
129c59d87c4SChristoph Hellwig 	ASSERT(XFS_DQ_IS_LOCKED(dqp));
130c59d87c4SChristoph Hellwig 	if (atomic_read(&dqp->q_pincount) == 0)
131c59d87c4SChristoph Hellwig 		return;
132c59d87c4SChristoph Hellwig 
133c59d87c4SChristoph Hellwig 	/*
134c59d87c4SChristoph Hellwig 	 * Give the log a push so we don't wait here too long.
135c59d87c4SChristoph Hellwig 	 */
136c59d87c4SChristoph Hellwig 	xfs_log_force(dqp->q_mount, 0);
137c59d87c4SChristoph Hellwig 	wait_event(dqp->q_pinwait, (atomic_read(&dqp->q_pincount) == 0));
138c59d87c4SChristoph Hellwig }
139c59d87c4SChristoph Hellwig 
140373b0589SCarlos Maiolino /*
141373b0589SCarlos Maiolino  * Callback used to mark a buffer with XFS_LI_FAILED when items in the buffer
142373b0589SCarlos Maiolino  * have been failed during writeback
143373b0589SCarlos Maiolino  *
144373b0589SCarlos Maiolino  * this informs the AIL that the dquot is already flush locked on the next push,
145373b0589SCarlos Maiolino  * and acquires a hold on the buffer to ensure that it isn't reclaimed before
146373b0589SCarlos Maiolino  * dirty data makes it to disk.
147373b0589SCarlos Maiolino  */
148373b0589SCarlos Maiolino STATIC void
149373b0589SCarlos Maiolino xfs_dquot_item_error(
150373b0589SCarlos Maiolino 	struct xfs_log_item	*lip,
151373b0589SCarlos Maiolino 	struct xfs_buf		*bp)
152373b0589SCarlos Maiolino {
15375d4a13bSDarrick J. Wong 	ASSERT(!completion_done(&DQUOT_ITEM(lip)->qli_dquot->q_flush));
154373b0589SCarlos Maiolino 	xfs_set_li_failed(lip, bp);
155373b0589SCarlos Maiolino }
156373b0589SCarlos Maiolino 
157c59d87c4SChristoph Hellwig STATIC uint
15843ff2122SChristoph Hellwig xfs_qm_dquot_logitem_push(
15943ff2122SChristoph Hellwig 	struct xfs_log_item	*lip,
160*57e80956SMatthew Wilcox 	struct list_head	*buffer_list)
161*57e80956SMatthew Wilcox 		__releases(&lip->li_ailp->ail_lock)
162*57e80956SMatthew Wilcox 		__acquires(&lip->li_ailp->ail_lock)
163c59d87c4SChristoph Hellwig {
164c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp = DQUOT_ITEM(lip)->qli_dquot;
165373b0589SCarlos Maiolino 	struct xfs_buf		*bp = lip->li_buf;
16643ff2122SChristoph Hellwig 	uint			rval = XFS_ITEM_SUCCESS;
16743ff2122SChristoph Hellwig 	int			error;
168c59d87c4SChristoph Hellwig 
169c59d87c4SChristoph Hellwig 	if (atomic_read(&dqp->q_pincount) > 0)
170c59d87c4SChristoph Hellwig 		return XFS_ITEM_PINNED;
171c59d87c4SChristoph Hellwig 
172373b0589SCarlos Maiolino 	/*
173373b0589SCarlos Maiolino 	 * The buffer containing this item failed to be written back
174373b0589SCarlos Maiolino 	 * previously. Resubmit the buffer for IO
175373b0589SCarlos Maiolino 	 */
176373b0589SCarlos Maiolino 	if (lip->li_flags & XFS_LI_FAILED) {
177373b0589SCarlos Maiolino 		if (!xfs_buf_trylock(bp))
178373b0589SCarlos Maiolino 			return XFS_ITEM_LOCKED;
179373b0589SCarlos Maiolino 
180643c8c05SCarlos Maiolino 		if (!xfs_buf_resubmit_failed_buffers(bp, buffer_list))
181373b0589SCarlos Maiolino 			rval = XFS_ITEM_FLUSHING;
182373b0589SCarlos Maiolino 
183373b0589SCarlos Maiolino 		xfs_buf_unlock(bp);
184373b0589SCarlos Maiolino 		return rval;
185373b0589SCarlos Maiolino 	}
186373b0589SCarlos Maiolino 
187800b484eSChristoph Hellwig 	if (!xfs_dqlock_nowait(dqp))
188c59d87c4SChristoph Hellwig 		return XFS_ITEM_LOCKED;
189c59d87c4SChristoph Hellwig 
190fe7257fdSChristoph Hellwig 	/*
191fe7257fdSChristoph Hellwig 	 * Re-check the pincount now that we stabilized the value by
192fe7257fdSChristoph Hellwig 	 * taking the quota lock.
193fe7257fdSChristoph Hellwig 	 */
194fe7257fdSChristoph Hellwig 	if (atomic_read(&dqp->q_pincount) > 0) {
19543ff2122SChristoph Hellwig 		rval = XFS_ITEM_PINNED;
19643ff2122SChristoph Hellwig 		goto out_unlock;
197fe7257fdSChristoph Hellwig 	}
198fe7257fdSChristoph Hellwig 
199c59d87c4SChristoph Hellwig 	/*
20043ff2122SChristoph Hellwig 	 * Someone else is already flushing the dquot.  Nothing we can do
20143ff2122SChristoph Hellwig 	 * here but wait for the flush to finish and remove the item from
20243ff2122SChristoph Hellwig 	 * the AIL.
203c59d87c4SChristoph Hellwig 	 */
20443ff2122SChristoph Hellwig 	if (!xfs_dqflock_nowait(dqp)) {
20543ff2122SChristoph Hellwig 		rval = XFS_ITEM_FLUSHING;
20643ff2122SChristoph Hellwig 		goto out_unlock;
207c59d87c4SChristoph Hellwig 	}
208c59d87c4SChristoph Hellwig 
209*57e80956SMatthew Wilcox 	spin_unlock(&lip->li_ailp->ail_lock);
21043ff2122SChristoph Hellwig 
21143ff2122SChristoph Hellwig 	error = xfs_qm_dqflush(dqp, &bp);
21243ff2122SChristoph Hellwig 	if (error) {
213c9690043SDarrick J. Wong 		xfs_warn(dqp->q_mount, "%s: push error %d on dqp "PTR_FMT,
21443ff2122SChristoph Hellwig 			__func__, error, dqp);
21543ff2122SChristoph Hellwig 	} else {
21643ff2122SChristoph Hellwig 		if (!xfs_buf_delwri_queue(bp, buffer_list))
21743ff2122SChristoph Hellwig 			rval = XFS_ITEM_FLUSHING;
21843ff2122SChristoph Hellwig 		xfs_buf_relse(bp);
21943ff2122SChristoph Hellwig 	}
22043ff2122SChristoph Hellwig 
221*57e80956SMatthew Wilcox 	spin_lock(&lip->li_ailp->ail_lock);
22243ff2122SChristoph Hellwig out_unlock:
22343ff2122SChristoph Hellwig 	xfs_dqunlock(dqp);
22443ff2122SChristoph Hellwig 	return rval;
225c59d87c4SChristoph Hellwig }
226c59d87c4SChristoph Hellwig 
227c59d87c4SChristoph Hellwig /*
228c59d87c4SChristoph Hellwig  * Unlock the dquot associated with the log item.
229c59d87c4SChristoph Hellwig  * Clear the fields of the dquot and dquot log item that
230c59d87c4SChristoph Hellwig  * are specific to the current transaction.  If the
231c59d87c4SChristoph Hellwig  * hold flags is set, do not unlock the dquot.
232c59d87c4SChristoph Hellwig  */
233c59d87c4SChristoph Hellwig STATIC void
234c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_unlock(
235c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip)
236c59d87c4SChristoph Hellwig {
237c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp = DQUOT_ITEM(lip)->qli_dquot;
238c59d87c4SChristoph Hellwig 
239c59d87c4SChristoph Hellwig 	ASSERT(XFS_DQ_IS_LOCKED(dqp));
240c59d87c4SChristoph Hellwig 
241c59d87c4SChristoph Hellwig 	/*
242c59d87c4SChristoph Hellwig 	 * Clear the transaction pointer in the dquot
243c59d87c4SChristoph Hellwig 	 */
244c59d87c4SChristoph Hellwig 	dqp->q_transp = NULL;
245c59d87c4SChristoph Hellwig 
246c59d87c4SChristoph Hellwig 	/*
247c59d87c4SChristoph Hellwig 	 * dquots are never 'held' from getting unlocked at the end of
248c59d87c4SChristoph Hellwig 	 * a transaction.  Their locking and unlocking is hidden inside the
249c59d87c4SChristoph Hellwig 	 * transaction layer, within trans_commit. Hence, no LI_HOLD flag
250c59d87c4SChristoph Hellwig 	 * for the logitem.
251c59d87c4SChristoph Hellwig 	 */
252c59d87c4SChristoph Hellwig 	xfs_dqunlock(dqp);
253c59d87c4SChristoph Hellwig }
254c59d87c4SChristoph Hellwig 
255c59d87c4SChristoph Hellwig /*
256c59d87c4SChristoph Hellwig  * this needs to stamp an lsn into the dquot, I think.
257c59d87c4SChristoph Hellwig  * rpc's that look at user dquot's would then have to
258c59d87c4SChristoph Hellwig  * push on the dependency recorded in the dquot
259c59d87c4SChristoph Hellwig  */
260c59d87c4SChristoph Hellwig STATIC void
261c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_committing(
262c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
263c59d87c4SChristoph Hellwig 	xfs_lsn_t		lsn)
264c59d87c4SChristoph Hellwig {
265c59d87c4SChristoph Hellwig }
266c59d87c4SChristoph Hellwig 
267c59d87c4SChristoph Hellwig /*
268c59d87c4SChristoph Hellwig  * This is the ops vector for dquots
269c59d87c4SChristoph Hellwig  */
270272e42b2SChristoph Hellwig static const struct xfs_item_ops xfs_dquot_item_ops = {
271c59d87c4SChristoph Hellwig 	.iop_size	= xfs_qm_dquot_logitem_size,
272c59d87c4SChristoph Hellwig 	.iop_format	= xfs_qm_dquot_logitem_format,
273c59d87c4SChristoph Hellwig 	.iop_pin	= xfs_qm_dquot_logitem_pin,
274c59d87c4SChristoph Hellwig 	.iop_unpin	= xfs_qm_dquot_logitem_unpin,
275c59d87c4SChristoph Hellwig 	.iop_unlock	= xfs_qm_dquot_logitem_unlock,
276c59d87c4SChristoph Hellwig 	.iop_committed	= xfs_qm_dquot_logitem_committed,
277c59d87c4SChristoph Hellwig 	.iop_push	= xfs_qm_dquot_logitem_push,
278373b0589SCarlos Maiolino 	.iop_committing = xfs_qm_dquot_logitem_committing,
279373b0589SCarlos Maiolino 	.iop_error	= xfs_dquot_item_error
280c59d87c4SChristoph Hellwig };
281c59d87c4SChristoph Hellwig 
282c59d87c4SChristoph Hellwig /*
283c59d87c4SChristoph Hellwig  * Initialize the dquot log item for a newly allocated dquot.
284c59d87c4SChristoph Hellwig  * The dquot isn't locked at this point, but it isn't on any of the lists
285c59d87c4SChristoph Hellwig  * either, so we don't care.
286c59d87c4SChristoph Hellwig  */
287c59d87c4SChristoph Hellwig void
288c59d87c4SChristoph Hellwig xfs_qm_dquot_logitem_init(
289c59d87c4SChristoph Hellwig 	struct xfs_dquot	*dqp)
290c59d87c4SChristoph Hellwig {
291c59d87c4SChristoph Hellwig 	struct xfs_dq_logitem	*lp = &dqp->q_logitem;
292c59d87c4SChristoph Hellwig 
293c59d87c4SChristoph Hellwig 	xfs_log_item_init(dqp->q_mount, &lp->qli_item, XFS_LI_DQUOT,
294c59d87c4SChristoph Hellwig 					&xfs_dquot_item_ops);
295c59d87c4SChristoph Hellwig 	lp->qli_dquot = dqp;
296c59d87c4SChristoph Hellwig }
297c59d87c4SChristoph Hellwig 
298c59d87c4SChristoph Hellwig /*------------------  QUOTAOFF LOG ITEMS  -------------------*/
299c59d87c4SChristoph Hellwig 
300c59d87c4SChristoph Hellwig static inline struct xfs_qoff_logitem *QOFF_ITEM(struct xfs_log_item *lip)
301c59d87c4SChristoph Hellwig {
302c59d87c4SChristoph Hellwig 	return container_of(lip, struct xfs_qoff_logitem, qql_item);
303c59d87c4SChristoph Hellwig }
304c59d87c4SChristoph Hellwig 
305c59d87c4SChristoph Hellwig 
306c59d87c4SChristoph Hellwig /*
307c59d87c4SChristoph Hellwig  * This returns the number of iovecs needed to log the given quotaoff item.
308c59d87c4SChristoph Hellwig  * We only need 1 iovec for an quotaoff item.  It just logs the
309c59d87c4SChristoph Hellwig  * quotaoff_log_format structure.
310c59d87c4SChristoph Hellwig  */
311166d1368SDave Chinner STATIC void
312c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_size(
313166d1368SDave Chinner 	struct xfs_log_item	*lip,
314166d1368SDave Chinner 	int			*nvecs,
315166d1368SDave Chinner 	int			*nbytes)
316c59d87c4SChristoph Hellwig {
317166d1368SDave Chinner 	*nvecs += 1;
318166d1368SDave Chinner 	*nbytes += sizeof(struct xfs_qoff_logitem);
319c59d87c4SChristoph Hellwig }
320c59d87c4SChristoph Hellwig 
321c59d87c4SChristoph Hellwig STATIC void
322c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_format(
323c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
324bde7cff6SChristoph Hellwig 	struct xfs_log_vec	*lv)
325c59d87c4SChristoph Hellwig {
326c59d87c4SChristoph Hellwig 	struct xfs_qoff_logitem	*qflip = QOFF_ITEM(lip);
327bde7cff6SChristoph Hellwig 	struct xfs_log_iovec	*vecp = NULL;
328ffda4e83SChristoph Hellwig 	struct xfs_qoff_logformat *qlf;
329c59d87c4SChristoph Hellwig 
330ffda4e83SChristoph Hellwig 	qlf = xlog_prepare_iovec(lv, &vecp, XLOG_REG_TYPE_QUOTAOFF);
331ffda4e83SChristoph Hellwig 	qlf->qf_type = XFS_LI_QUOTAOFF;
332ffda4e83SChristoph Hellwig 	qlf->qf_size = 1;
333ffda4e83SChristoph Hellwig 	qlf->qf_flags = qflip->qql_flags;
334ffda4e83SChristoph Hellwig 	xlog_finish_iovec(lv, vecp, sizeof(struct xfs_qoff_logitem));
335c59d87c4SChristoph Hellwig }
336c59d87c4SChristoph Hellwig 
337c59d87c4SChristoph Hellwig /*
338c59d87c4SChristoph Hellwig  * Pinning has no meaning for an quotaoff item, so just return.
339c59d87c4SChristoph Hellwig  */
340c59d87c4SChristoph Hellwig STATIC void
341c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_pin(
342c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip)
343c59d87c4SChristoph Hellwig {
344c59d87c4SChristoph Hellwig }
345c59d87c4SChristoph Hellwig 
346c59d87c4SChristoph Hellwig /*
347c59d87c4SChristoph Hellwig  * Since pinning has no meaning for an quotaoff item, unpinning does
348c59d87c4SChristoph Hellwig  * not either.
349c59d87c4SChristoph Hellwig  */
350c59d87c4SChristoph Hellwig STATIC void
351c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_unpin(
352c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
353c59d87c4SChristoph Hellwig 	int			remove)
354c59d87c4SChristoph Hellwig {
355c59d87c4SChristoph Hellwig }
356c59d87c4SChristoph Hellwig 
357c59d87c4SChristoph Hellwig /*
35843ff2122SChristoph Hellwig  * There isn't much you can do to push a quotaoff item.  It is simply
35943ff2122SChristoph Hellwig  * stuck waiting for the log to be flushed to disk.
360c59d87c4SChristoph Hellwig  */
361c59d87c4SChristoph Hellwig STATIC uint
36243ff2122SChristoph Hellwig xfs_qm_qoff_logitem_push(
36343ff2122SChristoph Hellwig 	struct xfs_log_item	*lip,
36443ff2122SChristoph Hellwig 	struct list_head	*buffer_list)
365c59d87c4SChristoph Hellwig {
366c59d87c4SChristoph Hellwig 	return XFS_ITEM_LOCKED;
367c59d87c4SChristoph Hellwig }
368c59d87c4SChristoph Hellwig 
369c59d87c4SChristoph Hellwig /*
370c59d87c4SChristoph Hellwig  * Quotaoff items have no locking or pushing, so return failure
371c59d87c4SChristoph Hellwig  * so that the caller doesn't bother with us.
372c59d87c4SChristoph Hellwig  */
373c59d87c4SChristoph Hellwig STATIC void
374c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_unlock(
375c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip)
376c59d87c4SChristoph Hellwig {
377c59d87c4SChristoph Hellwig }
378c59d87c4SChristoph Hellwig 
379c59d87c4SChristoph Hellwig /*
380c59d87c4SChristoph Hellwig  * The quotaoff-start-item is logged only once and cannot be moved in the log,
381c59d87c4SChristoph Hellwig  * so simply return the lsn at which it's been logged.
382c59d87c4SChristoph Hellwig  */
383c59d87c4SChristoph Hellwig STATIC xfs_lsn_t
384c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_committed(
385c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
386c59d87c4SChristoph Hellwig 	xfs_lsn_t		lsn)
387c59d87c4SChristoph Hellwig {
388c59d87c4SChristoph Hellwig 	return lsn;
389c59d87c4SChristoph Hellwig }
390c59d87c4SChristoph Hellwig 
391c59d87c4SChristoph Hellwig STATIC xfs_lsn_t
392c59d87c4SChristoph Hellwig xfs_qm_qoffend_logitem_committed(
393c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
394c59d87c4SChristoph Hellwig 	xfs_lsn_t		lsn)
395c59d87c4SChristoph Hellwig {
396c59d87c4SChristoph Hellwig 	struct xfs_qoff_logitem	*qfe = QOFF_ITEM(lip);
397c59d87c4SChristoph Hellwig 	struct xfs_qoff_logitem	*qfs = qfe->qql_start_lip;
398c59d87c4SChristoph Hellwig 	struct xfs_ail		*ailp = qfs->qql_item.li_ailp;
399c59d87c4SChristoph Hellwig 
400c59d87c4SChristoph Hellwig 	/*
401c59d87c4SChristoph Hellwig 	 * Delete the qoff-start logitem from the AIL.
402c59d87c4SChristoph Hellwig 	 * xfs_trans_ail_delete() drops the AIL lock.
403c59d87c4SChristoph Hellwig 	 */
404*57e80956SMatthew Wilcox 	spin_lock(&ailp->ail_lock);
40504913fddSDave Chinner 	xfs_trans_ail_delete(ailp, &qfs->qql_item, SHUTDOWN_LOG_IO_ERROR);
406c59d87c4SChristoph Hellwig 
407b1c5ebb2SDave Chinner 	kmem_free(qfs->qql_item.li_lv_shadow);
408b1c5ebb2SDave Chinner 	kmem_free(lip->li_lv_shadow);
409c59d87c4SChristoph Hellwig 	kmem_free(qfs);
410c59d87c4SChristoph Hellwig 	kmem_free(qfe);
411c59d87c4SChristoph Hellwig 	return (xfs_lsn_t)-1;
412c59d87c4SChristoph Hellwig }
413c59d87c4SChristoph Hellwig 
414c59d87c4SChristoph Hellwig /*
415c59d87c4SChristoph Hellwig  * XXX rcc - don't know quite what to do with this.  I think we can
416c59d87c4SChristoph Hellwig  * just ignore it.  The only time that isn't the case is if we allow
417c59d87c4SChristoph Hellwig  * the client to somehow see that quotas have been turned off in which
418c59d87c4SChristoph Hellwig  * we can't allow that to get back until the quotaoff hits the disk.
419c59d87c4SChristoph Hellwig  * So how would that happen?  Also, do we need different routines for
420c59d87c4SChristoph Hellwig  * quotaoff start and quotaoff end?  I suspect the answer is yes but
421c59d87c4SChristoph Hellwig  * to be sure, I need to look at the recovery code and see how quota off
422c59d87c4SChristoph Hellwig  * recovery is handled (do we roll forward or back or do something else).
423c59d87c4SChristoph Hellwig  * If we roll forwards or backwards, then we need two separate routines,
424c59d87c4SChristoph Hellwig  * one that does nothing and one that stamps in the lsn that matters
425c59d87c4SChristoph Hellwig  * (truly makes the quotaoff irrevocable).  If we do something else,
426c59d87c4SChristoph Hellwig  * then maybe we don't need two.
427c59d87c4SChristoph Hellwig  */
428c59d87c4SChristoph Hellwig STATIC void
429c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_committing(
430c59d87c4SChristoph Hellwig 	struct xfs_log_item	*lip,
431c59d87c4SChristoph Hellwig 	xfs_lsn_t		commit_lsn)
432c59d87c4SChristoph Hellwig {
433c59d87c4SChristoph Hellwig }
434c59d87c4SChristoph Hellwig 
435272e42b2SChristoph Hellwig static const struct xfs_item_ops xfs_qm_qoffend_logitem_ops = {
436c59d87c4SChristoph Hellwig 	.iop_size	= xfs_qm_qoff_logitem_size,
437c59d87c4SChristoph Hellwig 	.iop_format	= xfs_qm_qoff_logitem_format,
438c59d87c4SChristoph Hellwig 	.iop_pin	= xfs_qm_qoff_logitem_pin,
439c59d87c4SChristoph Hellwig 	.iop_unpin	= xfs_qm_qoff_logitem_unpin,
440c59d87c4SChristoph Hellwig 	.iop_unlock	= xfs_qm_qoff_logitem_unlock,
441c59d87c4SChristoph Hellwig 	.iop_committed	= xfs_qm_qoffend_logitem_committed,
442c59d87c4SChristoph Hellwig 	.iop_push	= xfs_qm_qoff_logitem_push,
443c59d87c4SChristoph Hellwig 	.iop_committing = xfs_qm_qoff_logitem_committing
444c59d87c4SChristoph Hellwig };
445c59d87c4SChristoph Hellwig 
446c59d87c4SChristoph Hellwig /*
447c59d87c4SChristoph Hellwig  * This is the ops vector shared by all quotaoff-start log items.
448c59d87c4SChristoph Hellwig  */
449272e42b2SChristoph Hellwig static const struct xfs_item_ops xfs_qm_qoff_logitem_ops = {
450c59d87c4SChristoph Hellwig 	.iop_size	= xfs_qm_qoff_logitem_size,
451c59d87c4SChristoph Hellwig 	.iop_format	= xfs_qm_qoff_logitem_format,
452c59d87c4SChristoph Hellwig 	.iop_pin	= xfs_qm_qoff_logitem_pin,
453c59d87c4SChristoph Hellwig 	.iop_unpin	= xfs_qm_qoff_logitem_unpin,
454c59d87c4SChristoph Hellwig 	.iop_unlock	= xfs_qm_qoff_logitem_unlock,
455c59d87c4SChristoph Hellwig 	.iop_committed	= xfs_qm_qoff_logitem_committed,
456c59d87c4SChristoph Hellwig 	.iop_push	= xfs_qm_qoff_logitem_push,
457c59d87c4SChristoph Hellwig 	.iop_committing = xfs_qm_qoff_logitem_committing
458c59d87c4SChristoph Hellwig };
459c59d87c4SChristoph Hellwig 
460c59d87c4SChristoph Hellwig /*
461c59d87c4SChristoph Hellwig  * Allocate and initialize an quotaoff item of the correct quota type(s).
462c59d87c4SChristoph Hellwig  */
463c59d87c4SChristoph Hellwig struct xfs_qoff_logitem *
464c59d87c4SChristoph Hellwig xfs_qm_qoff_logitem_init(
465c59d87c4SChristoph Hellwig 	struct xfs_mount	*mp,
466c59d87c4SChristoph Hellwig 	struct xfs_qoff_logitem	*start,
467c59d87c4SChristoph Hellwig 	uint			flags)
468c59d87c4SChristoph Hellwig {
469c59d87c4SChristoph Hellwig 	struct xfs_qoff_logitem	*qf;
470c59d87c4SChristoph Hellwig 
471c59d87c4SChristoph Hellwig 	qf = kmem_zalloc(sizeof(struct xfs_qoff_logitem), KM_SLEEP);
472c59d87c4SChristoph Hellwig 
473c59d87c4SChristoph Hellwig 	xfs_log_item_init(mp, &qf->qql_item, XFS_LI_QUOTAOFF, start ?
474c59d87c4SChristoph Hellwig 			&xfs_qm_qoffend_logitem_ops : &xfs_qm_qoff_logitem_ops);
475c59d87c4SChristoph Hellwig 	qf->qql_item.li_mountp = mp;
476c59d87c4SChristoph Hellwig 	qf->qql_start_lip = start;
477ffda4e83SChristoph Hellwig 	qf->qql_flags = flags;
478c59d87c4SChristoph Hellwig 	return qf;
479c59d87c4SChristoph Hellwig }
480