blob: abc8a21e3a822955406a65aaaf6a27b3b9424849 [file] [log] [blame]
Dave Chinner0b61f8a2018-06-05 19:42:14 -07001// SPDX-License-Identifier: GPL-2.0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002/*
Nathan Scott4ce31212005-11-02 14:59:41 +11003 * Copyright (c) 2000-2005 Silicon Graphics, Inc.
4 * All Rights Reserved.
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 */
Randy Dunlap16f7e0f2006-01-11 12:17:46 -08006
7#include <linux/capability.h>
8
Linus Torvalds1da177e2005-04-16 15:20:36 -07009#include "xfs.h"
10#include "xfs_fs.h"
Dave Chinner70a98832013-10-23 10:36:05 +110011#include "xfs_shared.h"
Dave Chinner239880e2013-10-23 10:50:10 +110012#include "xfs_format.h"
13#include "xfs_log_format.h"
14#include "xfs_trans_resv.h"
Nathan Scotta844f452005-11-02 14:38:42 +110015#include "xfs_bit.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070016#include "xfs_sb.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070017#include "xfs_mount.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070018#include "xfs_inode.h"
Dave Chinner239880e2013-10-23 10:50:10 +110019#include "xfs_trans.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070020#include "xfs_error.h"
Dave Chinnera4fbe6a2013-10-23 10:51:50 +110021#include "xfs_quota.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070022#include "xfs_qm.h"
Christoph Hellwig0b1b2132009-12-14 23:14:59 +000023#include "xfs_trace.h"
Dave Chinner6d8b79c2012-10-08 21:56:09 +110024#include "xfs_icache.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070025
Linus Torvalds1da177e2005-04-16 15:20:36 -070026STATIC int xfs_qm_log_quotaoff(xfs_mount_t *, xfs_qoff_logitem_t **, uint);
27STATIC int xfs_qm_log_quotaoff_end(xfs_mount_t *, xfs_qoff_logitem_t *,
28 uint);
Linus Torvalds1da177e2005-04-16 15:20:36 -070029
30/*
Linus Torvalds1da177e2005-04-16 15:20:36 -070031 * Turn off quota accounting and/or enforcement for all udquots and/or
32 * gdquots. Called only at unmount time.
33 *
34 * This assumes that there are no dquots of this file system cached
35 * incore, and modifies the ondisk dquot directly. Therefore, for example,
36 * it is an error to call this twice, without purging the cache.
37 */
Christoph Hellwigfcafb712009-02-09 08:47:34 +010038int
Linus Torvalds1da177e2005-04-16 15:20:36 -070039xfs_qm_scall_quotaoff(
40 xfs_mount_t *mp,
Christoph Hellwigfcafb712009-02-09 08:47:34 +010041 uint flags)
Linus Torvalds1da177e2005-04-16 15:20:36 -070042{
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +100043 struct xfs_quotainfo *q = mp->m_quotainfo;
Linus Torvalds1da177e2005-04-16 15:20:36 -070044 uint dqtype;
Linus Torvalds1da177e2005-04-16 15:20:36 -070045 int error;
46 uint inactivate_flags;
47 xfs_qoff_logitem_t *qoffstart;
Linus Torvalds1da177e2005-04-16 15:20:36 -070048
Linus Torvalds1da177e2005-04-16 15:20:36 -070049 /*
50 * No file system can have quotas enabled on disk but not in core.
51 * Note that quota utilities (like quotaoff) _expect_
Dave Chinner24513372014-06-25 14:58:08 +100052 * errno == -EEXIST here.
Linus Torvalds1da177e2005-04-16 15:20:36 -070053 */
54 if ((mp->m_qflags & flags) == 0)
Dave Chinner24513372014-06-25 14:58:08 +100055 return -EEXIST;
Linus Torvalds1da177e2005-04-16 15:20:36 -070056 error = 0;
57
58 flags &= (XFS_ALL_QUOTA_ACCT | XFS_ALL_QUOTA_ENFD);
59
60 /*
61 * We don't want to deal with two quotaoffs messing up each other,
62 * so we're going to serialize it. quotaoff isn't exactly a performance
63 * critical thing.
64 * If quotaoff, then we must be dealing with the root filesystem.
65 */
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +100066 ASSERT(q);
67 mutex_lock(&q->qi_quotaofflock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070068
69 /*
70 * If we're just turning off quota enforcement, change mp and go.
71 */
72 if ((flags & XFS_ALL_QUOTA_ACCT) == 0) {
73 mp->m_qflags &= ~(flags);
74
Eric Sandeen3685c2a2007-10-11 17:42:32 +100075 spin_lock(&mp->m_sb_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070076 mp->m_sb.sb_qflags = mp->m_qflags;
Eric Sandeen3685c2a2007-10-11 17:42:32 +100077 spin_unlock(&mp->m_sb_lock);
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +100078 mutex_unlock(&q->qi_quotaofflock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070079
80 /* XXX what to do if error ? Revert back to old vals incore ? */
Dave Chinner61e63ec2015-01-22 09:10:31 +110081 return xfs_sync_sb(mp, false);
Linus Torvalds1da177e2005-04-16 15:20:36 -070082 }
83
84 dqtype = 0;
85 inactivate_flags = 0;
86 /*
87 * If accounting is off, we must turn enforcement off, clear the
88 * quota 'CHKD' certificate to make it known that we have to
89 * do a quotacheck the next time this quota is turned on.
90 */
91 if (flags & XFS_UQUOTA_ACCT) {
92 dqtype |= XFS_QMOPT_UQUOTA;
93 flags |= (XFS_UQUOTA_CHKD | XFS_UQUOTA_ENFD);
94 inactivate_flags |= XFS_UQUOTA_ACTIVE;
95 }
96 if (flags & XFS_GQUOTA_ACCT) {
97 dqtype |= XFS_QMOPT_GQUOTA;
Chandra Seetharaman83e782e2013-06-27 17:25:10 -050098 flags |= (XFS_GQUOTA_CHKD | XFS_GQUOTA_ENFD);
Linus Torvalds1da177e2005-04-16 15:20:36 -070099 inactivate_flags |= XFS_GQUOTA_ACTIVE;
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500100 }
101 if (flags & XFS_PQUOTA_ACCT) {
Nathan Scottc8ad20f2005-06-21 15:38:48 +1000102 dqtype |= XFS_QMOPT_PQUOTA;
Chandra Seetharaman83e782e2013-06-27 17:25:10 -0500103 flags |= (XFS_PQUOTA_CHKD | XFS_PQUOTA_ENFD);
Nathan Scottc8ad20f2005-06-21 15:38:48 +1000104 inactivate_flags |= XFS_PQUOTA_ACTIVE;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700105 }
106
107 /*
108 * Nothing to do? Don't complain. This happens when we're just
109 * turning off quota enforcement.
110 */
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000111 if ((mp->m_qflags & flags) == 0)
112 goto out_unlock;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700113
114 /*
115 * Write the LI_QUOTAOFF log record, and do SB changes atomically,
David Chinnercb6edc22008-04-10 12:20:45 +1000116 * and synchronously. If we fail to write, we should abort the
117 * operation as it cannot be recovered safely if we crash.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118 */
David Chinnercb6edc22008-04-10 12:20:45 +1000119 error = xfs_qm_log_quotaoff(mp, &qoffstart, flags);
120 if (error)
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000121 goto out_unlock;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122
123 /*
124 * Next we clear the XFS_MOUNT_*DQ_ACTIVE bit(s) in the mount struct
125 * to take care of the race between dqget and quotaoff. We don't take
126 * any special locks to reset these bits. All processes need to check
127 * these bits *after* taking inode lock(s) to see if the particular
128 * quota type is in the process of being turned off. If *ACTIVE, it is
129 * guaranteed that all dquot structures and all quotainode ptrs will all
130 * stay valid as long as that inode is kept locked.
131 *
132 * There is no turning back after this.
133 */
134 mp->m_qflags &= ~inactivate_flags;
135
136 /*
137 * Give back all the dquot reference(s) held by inodes.
138 * Here we go thru every single incore inode in this file system, and
139 * do a dqrele on the i_udquot/i_gdquot that it may have.
140 * Essentially, as long as somebody has an inode locked, this guarantees
141 * that quotas will not be turned off. This is handy because in a
142 * transaction once we lock the inode(s) and check for quotaon, we can
143 * depend on the quota inodes (and other things) being valid as long as
144 * we keep the lock(s).
145 */
146 xfs_qm_dqrele_all_inodes(mp, flags);
147
148 /*
149 * Next we make the changes in the quota flag in the mount struct.
150 * This isn't protected by a particular lock directly, because we
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300151 * don't want to take a mrlock every time we depend on quotas being on.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700152 */
Christoph Hellwigb84a3a92012-03-14 11:53:34 -0500153 mp->m_qflags &= ~flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700154
155 /*
156 * Go through all the dquots of this file system and purge them,
Christoph Hellwigb84a3a92012-03-14 11:53:34 -0500157 * according to what was turned off.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700158 */
Christoph Hellwigb84a3a92012-03-14 11:53:34 -0500159 xfs_qm_dqpurge_all(mp, dqtype);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700160
161 /*
162 * Transactions that had started before ACTIVE state bit was cleared
163 * could have logged many dquots, so they'd have higher LSNs than
164 * the first QUOTAOFF log record does. If we happen to crash when
165 * the tail of the log has gone past the QUOTAOFF record, but
166 * before the last dquot modification, those dquots __will__
167 * recover, and that's not good.
168 *
169 * So, we have QUOTAOFF start and end logitems; the start
170 * logitem won't get overwritten until the end logitem appears...
171 */
David Chinnercb6edc22008-04-10 12:20:45 +1000172 error = xfs_qm_log_quotaoff_end(mp, qoffstart, flags);
173 if (error) {
174 /* We're screwed now. Shutdown is the only option. */
175 xfs_force_shutdown(mp, SHUTDOWN_CORRUPT_INCORE);
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000176 goto out_unlock;
David Chinnercb6edc22008-04-10 12:20:45 +1000177 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700178
179 /*
Chandra Seetharamanc31ad432013-07-10 18:00:36 -0500180 * If all quotas are completely turned off, close shop.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700181 */
Chandra Seetharamanc31ad432013-07-10 18:00:36 -0500182 if (mp->m_qflags == 0) {
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000183 mutex_unlock(&q->qi_quotaofflock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700184 xfs_qm_destroy_quotainfo(mp);
Eric Sandeend99831f2014-06-22 15:03:54 +1000185 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700186 }
187
188 /*
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000189 * Release our quotainode references if we don't need them anymore.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700190 */
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000191 if ((dqtype & XFS_QMOPT_UQUOTA) && q->qi_uquotaip) {
192 IRELE(q->qi_uquotaip);
193 q->qi_uquotaip = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700194 }
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500195 if ((dqtype & XFS_QMOPT_GQUOTA) && q->qi_gquotaip) {
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000196 IRELE(q->qi_gquotaip);
197 q->qi_gquotaip = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700198 }
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500199 if ((dqtype & XFS_QMOPT_PQUOTA) && q->qi_pquotaip) {
200 IRELE(q->qi_pquotaip);
201 q->qi_pquotaip = NULL;
202 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700203
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000204out_unlock:
205 mutex_unlock(&q->qi_quotaofflock);
206 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700207}
208
Christoph Hellwig5d188982010-07-20 17:51:31 +1000209STATIC int
210xfs_qm_scall_trunc_qfile(
211 struct xfs_mount *mp,
212 xfs_ino_t ino)
213{
214 struct xfs_inode *ip;
215 struct xfs_trans *tp;
216 int error;
217
218 if (ino == NULLFSINO)
219 return 0;
220
221 error = xfs_iget(mp, NULL, ino, 0, 0, &ip);
222 if (error)
223 return error;
224
225 xfs_ilock(ip, XFS_IOLOCK_EXCL);
226
Christoph Hellwig253f4912016-04-06 09:19:55 +1000227 error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, 0, 0, 0, &tp);
Christoph Hellwig5d188982010-07-20 17:51:31 +1000228 if (error) {
Christoph Hellwig5d188982010-07-20 17:51:31 +1000229 xfs_iunlock(ip, XFS_IOLOCK_EXCL);
230 goto out_put;
231 }
232
233 xfs_ilock(ip, XFS_ILOCK_EXCL);
Christoph Hellwigddc34152011-09-19 15:00:54 +0000234 xfs_trans_ijoin(tp, ip, 0);
Christoph Hellwig5d188982010-07-20 17:51:31 +1000235
Christoph Hellwig673e8e52011-12-18 20:00:04 +0000236 ip->i_d.di_size = 0;
Christoph Hellwig673e8e52011-12-18 20:00:04 +0000237 xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
238
239 error = xfs_itruncate_extents(&tp, ip, XFS_DATA_FORK, 0);
Christoph Hellwig5d188982010-07-20 17:51:31 +1000240 if (error) {
Christoph Hellwig4906e212015-06-04 13:47:56 +1000241 xfs_trans_cancel(tp);
Christoph Hellwig5d188982010-07-20 17:51:31 +1000242 goto out_unlock;
243 }
244
Christoph Hellwig673e8e52011-12-18 20:00:04 +0000245 ASSERT(ip->i_d.di_nextents == 0);
246
Dave Chinnerdcd79a12010-09-28 12:27:25 +1000247 xfs_trans_ichgtime(tp, ip, XFS_ICHGTIME_MOD | XFS_ICHGTIME_CHG);
Christoph Hellwig70393312015-06-04 13:48:08 +1000248 error = xfs_trans_commit(tp);
Christoph Hellwig5d188982010-07-20 17:51:31 +1000249
250out_unlock:
251 xfs_iunlock(ip, XFS_ILOCK_EXCL | XFS_IOLOCK_EXCL);
252out_put:
253 IRELE(ip);
254 return error;
255}
256
Christoph Hellwigfcafb712009-02-09 08:47:34 +0100257int
Linus Torvalds1da177e2005-04-16 15:20:36 -0700258xfs_qm_scall_trunc_qfiles(
259 xfs_mount_t *mp,
260 uint flags)
261{
Dave Chinner24513372014-06-25 14:58:08 +1000262 int error = -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263
Eric Sandeenf58522c2014-05-05 17:27:06 +1000264 if (!xfs_sb_version_hasquota(&mp->m_sb) || flags == 0 ||
265 (flags & ~XFS_DQ_ALLTYPES)) {
Eric Sandeen08e96e12013-10-11 20:59:05 -0500266 xfs_debug(mp, "%s: flags=%x m_qflags=%x",
Dave Chinner82211122011-03-07 10:07:35 +1100267 __func__, flags, mp->m_qflags);
Dave Chinner24513372014-06-25 14:58:08 +1000268 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700269 }
270
Jie Liuc61a9e32013-11-22 14:04:00 +0800271 if (flags & XFS_DQ_USER) {
Christoph Hellwig5d188982010-07-20 17:51:31 +1000272 error = xfs_qm_scall_trunc_qfile(mp, mp->m_sb.sb_uquotino);
Jie Liuc61a9e32013-11-22 14:04:00 +0800273 if (error)
274 return error;
275 }
276 if (flags & XFS_DQ_GROUP) {
277 error = xfs_qm_scall_trunc_qfile(mp, mp->m_sb.sb_gquotino);
278 if (error)
279 return error;
280 }
Chandra Seetharamand892d582013-07-19 17:36:02 -0500281 if (flags & XFS_DQ_PROJ)
Jie Liuc61a9e32013-11-22 14:04:00 +0800282 error = xfs_qm_scall_trunc_qfile(mp, mp->m_sb.sb_pquotino);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283
Jie Liuc61a9e32013-11-22 14:04:00 +0800284 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700285}
286
Linus Torvalds1da177e2005-04-16 15:20:36 -0700287/*
288 * Switch on (a given) quota enforcement for a filesystem. This takes
289 * effect immediately.
290 * (Switching on quota accounting must be done at mount time.)
291 */
Christoph Hellwigfcafb712009-02-09 08:47:34 +0100292int
Linus Torvalds1da177e2005-04-16 15:20:36 -0700293xfs_qm_scall_quotaon(
294 xfs_mount_t *mp,
295 uint flags)
296{
297 int error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700298 uint qf;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299
Linus Torvalds1da177e2005-04-16 15:20:36 -0700300 flags &= (XFS_ALL_QUOTA_ACCT | XFS_ALL_QUOTA_ENFD);
301 /*
302 * Switching on quota accounting must be done at mount time.
303 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700304 flags &= ~(XFS_ALL_QUOTA_ACCT);
305
Linus Torvalds1da177e2005-04-16 15:20:36 -0700306 if (flags == 0) {
Eric Sandeen08e96e12013-10-11 20:59:05 -0500307 xfs_debug(mp, "%s: zero flags, m_qflags=%x",
Dave Chinner82211122011-03-07 10:07:35 +1100308 __func__, mp->m_qflags);
Dave Chinner24513372014-06-25 14:58:08 +1000309 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310 }
311
Linus Torvalds1da177e2005-04-16 15:20:36 -0700312 /*
313 * Can't enforce without accounting. We check the superblock
314 * qflags here instead of m_qflags because rootfs can have
315 * quota acct on ondisk without m_qflags' knowing.
316 */
Jan Karafbf64b32014-10-08 11:52:52 +0200317 if (((mp->m_sb.sb_qflags & XFS_UQUOTA_ACCT) == 0 &&
Chandra Seetharaman83e782e2013-06-27 17:25:10 -0500318 (flags & XFS_UQUOTA_ENFD)) ||
Jan Karafbf64b32014-10-08 11:52:52 +0200319 ((mp->m_sb.sb_qflags & XFS_GQUOTA_ACCT) == 0 &&
Chandra Seetharaman83e782e2013-06-27 17:25:10 -0500320 (flags & XFS_GQUOTA_ENFD)) ||
Jan Karafbf64b32014-10-08 11:52:52 +0200321 ((mp->m_sb.sb_qflags & XFS_PQUOTA_ACCT) == 0 &&
Chandra Seetharaman83e782e2013-06-27 17:25:10 -0500322 (flags & XFS_PQUOTA_ENFD))) {
Dave Chinner82211122011-03-07 10:07:35 +1100323 xfs_debug(mp,
Eric Sandeen08e96e12013-10-11 20:59:05 -0500324 "%s: Can't enforce without acct, flags=%x sbflags=%x",
Dave Chinner82211122011-03-07 10:07:35 +1100325 __func__, flags, mp->m_sb.sb_qflags);
Dave Chinner24513372014-06-25 14:58:08 +1000326 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700327 }
328 /*
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300329 * If everything's up to-date incore, then don't waste time.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700330 */
331 if ((mp->m_qflags & flags) == flags)
Dave Chinner24513372014-06-25 14:58:08 +1000332 return -EEXIST;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700333
334 /*
335 * Change sb_qflags on disk but not incore mp->qflags
336 * if this is the root filesystem.
337 */
Eric Sandeen3685c2a2007-10-11 17:42:32 +1000338 spin_lock(&mp->m_sb_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700339 qf = mp->m_sb.sb_qflags;
340 mp->m_sb.sb_qflags = qf | flags;
Eric Sandeen3685c2a2007-10-11 17:42:32 +1000341 spin_unlock(&mp->m_sb_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700342
343 /*
344 * There's nothing to change if it's the same.
345 */
Dave Chinner4d11a402015-01-22 09:10:26 +1100346 if ((qf & flags) == flags)
Dave Chinner24513372014-06-25 14:58:08 +1000347 return -EEXIST;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700348
Dave Chinner61e63ec2015-01-22 09:10:31 +1100349 error = xfs_sync_sb(mp, false);
350 if (error)
Eric Sandeend99831f2014-06-22 15:03:54 +1000351 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700352 /*
353 * If we aren't trying to switch on quota enforcement, we are done.
354 */
355 if (((mp->m_sb.sb_qflags & XFS_UQUOTA_ACCT) !=
356 (mp->m_qflags & XFS_UQUOTA_ACCT)) ||
Nathan Scottc8ad20f2005-06-21 15:38:48 +1000357 ((mp->m_sb.sb_qflags & XFS_PQUOTA_ACCT) !=
358 (mp->m_qflags & XFS_PQUOTA_ACCT)) ||
359 ((mp->m_sb.sb_qflags & XFS_GQUOTA_ACCT) !=
Jan Karafbf64b32014-10-08 11:52:52 +0200360 (mp->m_qflags & XFS_GQUOTA_ACCT)))
Eric Sandeend99831f2014-06-22 15:03:54 +1000361 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700362
363 if (! XFS_IS_QUOTA_RUNNING(mp))
Dave Chinner24513372014-06-25 14:58:08 +1000364 return -ESRCH;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365
366 /*
367 * Switch on quota enforcement in core.
368 */
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000369 mutex_lock(&mp->m_quotainfo->qi_quotaofflock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700370 mp->m_qflags |= (flags & XFS_ALL_QUOTA_ENFD);
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000371 mutex_unlock(&mp->m_quotainfo->qi_quotaofflock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700372
Eric Sandeend99831f2014-06-22 15:03:54 +1000373 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374}
375
Jan Kara14bf61f2014-10-09 16:03:13 +0200376#define XFS_QC_MASK \
377 (QC_LIMIT_MASK | QC_TIMER_MASK | QC_WARNS_MASK)
Christoph Hellwigc472b432010-05-06 17:05:17 -0400378
Linus Torvalds1da177e2005-04-16 15:20:36 -0700379/*
380 * Adjust quota limits, and start/stop timers accordingly.
381 */
Christoph Hellwigfcafb712009-02-09 08:47:34 +0100382int
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383xfs_qm_scall_setqlim(
Brian Fosterb1366452013-03-18 10:51:46 -0400384 struct xfs_mount *mp,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700385 xfs_dqid_t id,
386 uint type,
Jan Kara14bf61f2014-10-09 16:03:13 +0200387 struct qc_dqblk *newlim)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388{
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000389 struct xfs_quotainfo *q = mp->m_quotainfo;
Brian Fosterb1366452013-03-18 10:51:46 -0400390 struct xfs_disk_dquot *ddq;
391 struct xfs_dquot *dqp;
392 struct xfs_trans *tp;
Carlos Maiolinobe607942016-02-08 11:27:55 +1100393 struct xfs_def_quota *defq;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700394 int error;
395 xfs_qcnt_t hard, soft;
396
Jan Kara14bf61f2014-10-09 16:03:13 +0200397 if (newlim->d_fieldmask & ~XFS_QC_MASK)
Dave Chinner24513372014-06-25 14:58:08 +1000398 return -EINVAL;
Jan Kara14bf61f2014-10-09 16:03:13 +0200399 if ((newlim->d_fieldmask & XFS_QC_MASK) == 0)
Christoph Hellwigc472b432010-05-06 17:05:17 -0400400 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700401
Dave Chinnerf6481672013-05-21 18:02:00 +1000402 /*
403 * We don't want to race with a quotaoff so take the quotaoff lock.
404 * We don't hold an inode lock, so there's nothing else to stop
405 * a quotaoff from happening.
406 */
407 mutex_lock(&q->qi_quotaofflock);
408
409 /*
410 * Get the dquot (locked) before we start, as we need to do a
411 * transaction to allocate it if it doesn't exist. Once we have the
412 * dquot, unlock it so we can start the next transaction safely. We hold
413 * a reference to the dquot, so it's safe to do this unlock/lock without
414 * it being reclaimed in the mean time.
415 */
Darrick J. Wong30ab2dc2018-05-04 15:30:24 -0700416 error = xfs_qm_dqget(mp, id, type, true, &dqp);
Dave Chinnerf6481672013-05-21 18:02:00 +1000417 if (error) {
Dave Chinner24513372014-06-25 14:58:08 +1000418 ASSERT(error != -ENOENT);
Dave Chinnerf6481672013-05-21 18:02:00 +1000419 goto out_unlock;
420 }
Carlos Maiolinobe607942016-02-08 11:27:55 +1100421
422 defq = xfs_get_defquota(dqp, q);
Dave Chinnerf6481672013-05-21 18:02:00 +1000423 xfs_dqunlock(dqp);
424
Christoph Hellwig253f4912016-04-06 09:19:55 +1000425 error = xfs_trans_alloc(mp, &M_RES(mp)->tr_qm_setqlim, 0, 0, 0, &tp);
426 if (error)
Dave Chinnerf6481672013-05-21 18:02:00 +1000427 goto out_rele;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700428
Dave Chinnerf6481672013-05-21 18:02:00 +1000429 xfs_dqlock(dqp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700430 xfs_trans_dqjoin(tp, dqp);
431 ddq = &dqp->q_core;
432
433 /*
434 * Make sure that hardlimits are >= soft limits before changing.
435 */
Jan Kara14bf61f2014-10-09 16:03:13 +0200436 hard = (newlim->d_fieldmask & QC_SPC_HARD) ?
437 (xfs_qcnt_t) XFS_B_TO_FSB(mp, newlim->d_spc_hardlimit) :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100438 be64_to_cpu(ddq->d_blk_hardlimit);
Jan Kara14bf61f2014-10-09 16:03:13 +0200439 soft = (newlim->d_fieldmask & QC_SPC_SOFT) ?
440 (xfs_qcnt_t) XFS_B_TO_FSB(mp, newlim->d_spc_softlimit) :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100441 be64_to_cpu(ddq->d_blk_softlimit);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700442 if (hard == 0 || hard >= soft) {
Christoph Hellwig1149d962005-11-02 15:01:12 +1100443 ddq->d_blk_hardlimit = cpu_to_be64(hard);
444 ddq->d_blk_softlimit = cpu_to_be64(soft);
Brian Fosterb1366452013-03-18 10:51:46 -0400445 xfs_dquot_set_prealloc_limits(dqp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446 if (id == 0) {
Carlos Maiolinobe607942016-02-08 11:27:55 +1100447 defq->bhardlimit = hard;
448 defq->bsoftlimit = soft;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700449 }
450 } else {
Eric Sandeen08e96e12013-10-11 20:59:05 -0500451 xfs_debug(mp, "blkhard %Ld < blksoft %Ld", hard, soft);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700452 }
Jan Kara14bf61f2014-10-09 16:03:13 +0200453 hard = (newlim->d_fieldmask & QC_RT_SPC_HARD) ?
454 (xfs_qcnt_t) XFS_B_TO_FSB(mp, newlim->d_rt_spc_hardlimit) :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100455 be64_to_cpu(ddq->d_rtb_hardlimit);
Jan Kara14bf61f2014-10-09 16:03:13 +0200456 soft = (newlim->d_fieldmask & QC_RT_SPC_SOFT) ?
457 (xfs_qcnt_t) XFS_B_TO_FSB(mp, newlim->d_rt_spc_softlimit) :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100458 be64_to_cpu(ddq->d_rtb_softlimit);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700459 if (hard == 0 || hard >= soft) {
Christoph Hellwig1149d962005-11-02 15:01:12 +1100460 ddq->d_rtb_hardlimit = cpu_to_be64(hard);
461 ddq->d_rtb_softlimit = cpu_to_be64(soft);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700462 if (id == 0) {
Carlos Maiolinobe607942016-02-08 11:27:55 +1100463 defq->rtbhardlimit = hard;
464 defq->rtbsoftlimit = soft;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700465 }
466 } else {
Eric Sandeen08e96e12013-10-11 20:59:05 -0500467 xfs_debug(mp, "rtbhard %Ld < rtbsoft %Ld", hard, soft);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700468 }
469
Jan Kara14bf61f2014-10-09 16:03:13 +0200470 hard = (newlim->d_fieldmask & QC_INO_HARD) ?
Linus Torvalds1da177e2005-04-16 15:20:36 -0700471 (xfs_qcnt_t) newlim->d_ino_hardlimit :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100472 be64_to_cpu(ddq->d_ino_hardlimit);
Jan Kara14bf61f2014-10-09 16:03:13 +0200473 soft = (newlim->d_fieldmask & QC_INO_SOFT) ?
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474 (xfs_qcnt_t) newlim->d_ino_softlimit :
Christoph Hellwig1149d962005-11-02 15:01:12 +1100475 be64_to_cpu(ddq->d_ino_softlimit);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700476 if (hard == 0 || hard >= soft) {
Christoph Hellwig1149d962005-11-02 15:01:12 +1100477 ddq->d_ino_hardlimit = cpu_to_be64(hard);
478 ddq->d_ino_softlimit = cpu_to_be64(soft);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700479 if (id == 0) {
Carlos Maiolinobe607942016-02-08 11:27:55 +1100480 defq->ihardlimit = hard;
481 defq->isoftlimit = soft;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700482 }
483 } else {
Eric Sandeen08e96e12013-10-11 20:59:05 -0500484 xfs_debug(mp, "ihard %Ld < isoft %Ld", hard, soft);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700485 }
486
Nathan Scott754002b2005-06-21 15:49:06 +1000487 /*
488 * Update warnings counter(s) if requested
489 */
Jan Kara14bf61f2014-10-09 16:03:13 +0200490 if (newlim->d_fieldmask & QC_SPC_WARNS)
491 ddq->d_bwarns = cpu_to_be16(newlim->d_spc_warns);
492 if (newlim->d_fieldmask & QC_INO_WARNS)
493 ddq->d_iwarns = cpu_to_be16(newlim->d_ino_warns);
494 if (newlim->d_fieldmask & QC_RT_SPC_WARNS)
495 ddq->d_rtbwarns = cpu_to_be16(newlim->d_rt_spc_warns);
Nathan Scott754002b2005-06-21 15:49:06 +1000496
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497 if (id == 0) {
498 /*
499 * Timelimits for the super user set the relative time
500 * the other users can be over quota for this file system.
501 * If it is zero a default is used. Ditto for the default
Nathan Scott754002b2005-06-21 15:49:06 +1000502 * soft and hard limit values (already done, above), and
503 * for warnings.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504 */
Jan Kara14bf61f2014-10-09 16:03:13 +0200505 if (newlim->d_fieldmask & QC_SPC_TIMER) {
506 q->qi_btimelimit = newlim->d_spc_timer;
507 ddq->d_btimer = cpu_to_be32(newlim->d_spc_timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700508 }
Jan Kara14bf61f2014-10-09 16:03:13 +0200509 if (newlim->d_fieldmask & QC_INO_TIMER) {
510 q->qi_itimelimit = newlim->d_ino_timer;
511 ddq->d_itimer = cpu_to_be32(newlim->d_ino_timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512 }
Jan Kara14bf61f2014-10-09 16:03:13 +0200513 if (newlim->d_fieldmask & QC_RT_SPC_TIMER) {
514 q->qi_rtbtimelimit = newlim->d_rt_spc_timer;
515 ddq->d_rtbtimer = cpu_to_be32(newlim->d_rt_spc_timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700516 }
Jan Kara14bf61f2014-10-09 16:03:13 +0200517 if (newlim->d_fieldmask & QC_SPC_WARNS)
518 q->qi_bwarnlimit = newlim->d_spc_warns;
519 if (newlim->d_fieldmask & QC_INO_WARNS)
520 q->qi_iwarnlimit = newlim->d_ino_warns;
521 if (newlim->d_fieldmask & QC_RT_SPC_WARNS)
522 q->qi_rtbwarnlimit = newlim->d_rt_spc_warns;
Nathan Scott754002b2005-06-21 15:49:06 +1000523 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700524 /*
525 * If the user is now over quota, start the timelimit.
526 * The user will not be 'warned'.
527 * Note that we keep the timers ticking, whether enforcement
528 * is on or off. We don't really want to bother with iterating
529 * over all ondisk dquots and turning the timers on/off.
530 */
531 xfs_qm_adjust_dqtimers(mp, ddq);
532 }
533 dqp->dq_flags |= XFS_DQ_DIRTY;
534 xfs_trans_log_dquot(tp, dqp);
535
Christoph Hellwig70393312015-06-04 13:48:08 +1000536 error = xfs_trans_commit(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700537
Dave Chinnerf6481672013-05-21 18:02:00 +1000538out_rele:
539 xfs_qm_dqrele(dqp);
540out_unlock:
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000541 mutex_unlock(&q->qi_quotaofflock);
David Chinnere5720ee2008-04-10 12:21:18 +1000542 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700543}
544
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545STATIC int
546xfs_qm_log_quotaoff_end(
547 xfs_mount_t *mp,
548 xfs_qoff_logitem_t *startqoff,
549 uint flags)
550{
Nathan Scottc8ad20f2005-06-21 15:38:48 +1000551 xfs_trans_t *tp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700552 int error;
Nathan Scottc8ad20f2005-06-21 15:38:48 +1000553 xfs_qoff_logitem_t *qoffi;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700554
Christoph Hellwig253f4912016-04-06 09:19:55 +1000555 error = xfs_trans_alloc(mp, &M_RES(mp)->tr_qm_equotaoff, 0, 0, 0, &tp);
556 if (error)
Eric Sandeend99831f2014-06-22 15:03:54 +1000557 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558
559 qoffi = xfs_trans_get_qoff_item(tp, startqoff,
560 flags & XFS_ALL_QUOTA_ACCT);
561 xfs_trans_log_quotaoff_item(tp, qoffi);
562
563 /*
564 * We have to make sure that the transaction is secure on disk before we
565 * return and actually stop quota accounting. So, make it synchronous.
566 * We don't care about quotoff's performance.
567 */
568 xfs_trans_set_sync(tp);
Christoph Hellwig70393312015-06-04 13:48:08 +1000569 return xfs_trans_commit(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700570}
571
572
573STATIC int
574xfs_qm_log_quotaoff(
575 xfs_mount_t *mp,
576 xfs_qoff_logitem_t **qoffstartp,
577 uint flags)
578{
579 xfs_trans_t *tp;
580 int error;
Brian Foster5d45ee12014-11-28 14:00:53 +1100581 xfs_qoff_logitem_t *qoffi;
582
583 *qoffstartp = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700584
Christoph Hellwig253f4912016-04-06 09:19:55 +1000585 error = xfs_trans_alloc(mp, &M_RES(mp)->tr_qm_quotaoff, 0, 0, 0, &tp);
586 if (error)
Brian Foster5d45ee12014-11-28 14:00:53 +1100587 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700588
589 qoffi = xfs_trans_get_qoff_item(tp, NULL, flags & XFS_ALL_QUOTA_ACCT);
590 xfs_trans_log_quotaoff_item(tp, qoffi);
591
Eric Sandeen3685c2a2007-10-11 17:42:32 +1000592 spin_lock(&mp->m_sb_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593 mp->m_sb.sb_qflags = (mp->m_qflags & ~(flags)) & XFS_MOUNT_QUOTA_ALL;
Eric Sandeen3685c2a2007-10-11 17:42:32 +1000594 spin_unlock(&mp->m_sb_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700595
Dave Chinner61e63ec2015-01-22 09:10:31 +1100596 xfs_log_sb(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700597
598 /*
599 * We have to make sure that the transaction is secure on disk before we
600 * return and actually stop quota accounting. So, make it synchronous.
601 * We don't care about quotoff's performance.
602 */
603 xfs_trans_set_sync(tp);
Christoph Hellwig70393312015-06-04 13:48:08 +1000604 error = xfs_trans_commit(tp);
Brian Foster5d45ee12014-11-28 14:00:53 +1100605 if (error)
606 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700607
Linus Torvalds1da177e2005-04-16 15:20:36 -0700608 *qoffstartp = qoffi;
Brian Foster5d45ee12014-11-28 14:00:53 +1100609out:
Eric Sandeend99831f2014-06-22 15:03:54 +1000610 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700611}
612
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700613/* Fill out the quota context. */
614static void
615xfs_qm_scall_getquota_fill_qc(
Christoph Hellwig18535a72012-02-20 02:28:16 +0000616 struct xfs_mount *mp,
Christoph Hellwig18535a72012-02-20 02:28:16 +0000617 uint type,
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700618 const struct xfs_dquot *dqp,
619 struct qc_dqblk *dst)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700620{
621 memset(dst, 0, sizeof(*dst));
Jan Kara14bf61f2014-10-09 16:03:13 +0200622 dst->d_spc_hardlimit =
623 XFS_FSB_TO_B(mp, be64_to_cpu(dqp->q_core.d_blk_hardlimit));
624 dst->d_spc_softlimit =
625 XFS_FSB_TO_B(mp, be64_to_cpu(dqp->q_core.d_blk_softlimit));
Christoph Hellwig18535a72012-02-20 02:28:16 +0000626 dst->d_ino_hardlimit = be64_to_cpu(dqp->q_core.d_ino_hardlimit);
627 dst->d_ino_softlimit = be64_to_cpu(dqp->q_core.d_ino_softlimit);
Jan Kara14bf61f2014-10-09 16:03:13 +0200628 dst->d_space = XFS_FSB_TO_B(mp, dqp->q_res_bcount);
629 dst->d_ino_count = dqp->q_res_icount;
630 dst->d_spc_timer = be32_to_cpu(dqp->q_core.d_btimer);
631 dst->d_ino_timer = be32_to_cpu(dqp->q_core.d_itimer);
632 dst->d_ino_warns = be16_to_cpu(dqp->q_core.d_iwarns);
633 dst->d_spc_warns = be16_to_cpu(dqp->q_core.d_bwarns);
634 dst->d_rt_spc_hardlimit =
635 XFS_FSB_TO_B(mp, be64_to_cpu(dqp->q_core.d_rtb_hardlimit));
636 dst->d_rt_spc_softlimit =
637 XFS_FSB_TO_B(mp, be64_to_cpu(dqp->q_core.d_rtb_softlimit));
638 dst->d_rt_space = XFS_FSB_TO_B(mp, dqp->q_res_rtbcount);
639 dst->d_rt_spc_timer = be32_to_cpu(dqp->q_core.d_rtbtimer);
640 dst->d_rt_spc_warns = be16_to_cpu(dqp->q_core.d_rtbwarns);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700641
642 /*
643 * Internally, we don't reset all the timers when quota enforcement
Nathan Scottc41564b2006-03-29 08:55:14 +1000644 * gets turned off. No need to confuse the user level code,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700645 * so return zeroes in that case.
646 */
Chandra Seetharaman83e782e2013-06-27 17:25:10 -0500647 if ((!XFS_IS_UQUOTA_ENFORCED(mp) &&
648 dqp->q_core.d_flags == XFS_DQ_USER) ||
649 (!XFS_IS_GQUOTA_ENFORCED(mp) &&
650 dqp->q_core.d_flags == XFS_DQ_GROUP) ||
651 (!XFS_IS_PQUOTA_ENFORCED(mp) &&
652 dqp->q_core.d_flags == XFS_DQ_PROJ)) {
Jan Kara14bf61f2014-10-09 16:03:13 +0200653 dst->d_spc_timer = 0;
654 dst->d_ino_timer = 0;
655 dst->d_rt_spc_timer = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656 }
657
658#ifdef DEBUG
Jan Kara14bf61f2014-10-09 16:03:13 +0200659 if (((XFS_IS_UQUOTA_ENFORCED(mp) && type == XFS_DQ_USER) ||
660 (XFS_IS_GQUOTA_ENFORCED(mp) && type == XFS_DQ_GROUP) ||
661 (XFS_IS_PQUOTA_ENFORCED(mp) && type == XFS_DQ_PROJ)) &&
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700662 dqp->q_core.d_id != 0) {
Jan Kara14bf61f2014-10-09 16:03:13 +0200663 if ((dst->d_space > dst->d_spc_softlimit) &&
664 (dst->d_spc_softlimit > 0)) {
665 ASSERT(dst->d_spc_timer != 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700666 }
Jan Kara14bf61f2014-10-09 16:03:13 +0200667 if ((dst->d_ino_count > dst->d_ino_softlimit) &&
Linus Torvalds1da177e2005-04-16 15:20:36 -0700668 (dst->d_ino_softlimit > 0)) {
Jan Kara14bf61f2014-10-09 16:03:13 +0200669 ASSERT(dst->d_ino_timer != 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670 }
671 }
672#endif
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700673}
674
675/* Return the quota information for the dquot matching id. */
676int
677xfs_qm_scall_getquota(
678 struct xfs_mount *mp,
679 xfs_dqid_t id,
680 uint type,
681 struct qc_dqblk *dst)
682{
683 struct xfs_dquot *dqp;
684 int error;
685
686 /*
Darrick J. Wong30ab2dc2018-05-04 15:30:24 -0700687 * Try to get the dquot. We don't want it allocated on disk, so don't
688 * set doalloc. If it doesn't exist, we'll get ENOENT back.
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700689 */
Darrick J. Wong30ab2dc2018-05-04 15:30:24 -0700690 error = xfs_qm_dqget(mp, id, type, false, &dqp);
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700691 if (error)
692 return error;
693
694 /*
695 * If everything's NULL, this dquot doesn't quite exist as far as
696 * our utility programs are concerned.
697 */
698 if (XFS_IS_DQUOT_UNINITIALIZED(dqp)) {
699 error = -ENOENT;
700 goto out_put;
701 }
702
703 xfs_qm_scall_getquota_fill_qc(mp, type, dqp, dst);
704
Christoph Hellwig18535a72012-02-20 02:28:16 +0000705out_put:
706 xfs_qm_dqput(dqp);
707 return error;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700708}
709
Darrick J. Wong2e330e72018-05-04 15:30:20 -0700710/*
711 * Return the quota information for the first initialized dquot whose id
712 * is at least as high as id.
713 */
714int
715xfs_qm_scall_getquota_next(
716 struct xfs_mount *mp,
717 xfs_dqid_t *id,
718 uint type,
719 struct qc_dqblk *dst)
720{
721 struct xfs_dquot *dqp;
722 int error;
723
724 error = xfs_qm_dqget_next(mp, *id, type, &dqp);
725 if (error)
726 return error;
727
728 /* Fill in the ID we actually read from disk */
729 *id = be32_to_cpu(dqp->q_core.d_id);
730
731 xfs_qm_scall_getquota_fill_qc(mp, type, dqp, dst);
732
733 xfs_qm_dqput(dqp);
734 return error;
735}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200737STATIC int
738xfs_dqrele_inode(
739 struct xfs_inode *ip,
Brian Fostera454f742012-11-06 09:50:39 -0500740 int flags,
741 void *args)
David Chinner5b4d89a2008-10-30 17:08:03 +1100742{
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200743 /* skip quota inodes */
Christoph Hellwig8a7b8a82010-04-20 17:01:30 +1000744 if (ip == ip->i_mount->m_quotainfo->qi_uquotaip ||
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500745 ip == ip->i_mount->m_quotainfo->qi_gquotaip ||
746 ip == ip->i_mount->m_quotainfo->qi_pquotaip) {
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200747 ASSERT(ip->i_udquot == NULL);
748 ASSERT(ip->i_gdquot == NULL);
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500749 ASSERT(ip->i_pdquot == NULL);
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200750 return 0;
751 }
Dave Chinnercb4f0d12008-11-10 17:11:18 +1100752
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200753 xfs_ilock(ip, XFS_ILOCK_EXCL);
754 if ((flags & XFS_UQUOTA_ACCT) && ip->i_udquot) {
755 xfs_qm_dqrele(ip->i_udquot);
756 ip->i_udquot = NULL;
757 }
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500758 if ((flags & XFS_GQUOTA_ACCT) && ip->i_gdquot) {
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200759 xfs_qm_dqrele(ip->i_gdquot);
760 ip->i_gdquot = NULL;
761 }
Chandra Seetharaman92f8ff72013-07-11 00:00:40 -0500762 if ((flags & XFS_PQUOTA_ACCT) && ip->i_pdquot) {
763 xfs_qm_dqrele(ip->i_pdquot);
764 ip->i_pdquot = NULL;
765 }
Christoph Hellwigf2d67612010-06-24 11:52:50 +1000766 xfs_iunlock(ip, XFS_ILOCK_EXCL);
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200767 return 0;
David Chinner5b4d89a2008-10-30 17:08:03 +1100768}
769
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200770
David Chinner5b4d89a2008-10-30 17:08:03 +1100771/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700772 * Go thru all the inodes in the file system, releasing their dquots.
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200773 *
Linus Torvalds1da177e2005-04-16 15:20:36 -0700774 * Note that the mount structure gets modified to indicate that quotas are off
Christoph Hellwigfe588ed2009-06-08 15:35:27 +0200775 * AFTER this, in the case of quotaoff.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776 */
777void
778xfs_qm_dqrele_all_inodes(
779 struct xfs_mount *mp,
780 uint flags)
781{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700782 ASSERT(mp->m_quotainfo);
Brian Fostere20c8a52017-04-26 08:30:40 -0700783 xfs_inode_ag_iterator_flags(mp, xfs_dqrele_inode, flags, NULL,
784 XFS_AGITER_INEW_WAIT);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700785}