xfs_quotaops.c 5.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * Copyright (c) 2008, Christoph Hellwig
 * All Rights Reserved.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation.
 *
 * 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.
 *
 * 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
 */
#include "xfs.h"
19
#include "xfs_format.h"
20
#include "xfs_log_format.h"
21
#include "xfs_trans_resv.h"
22
#include "xfs_mount.h"
23
#include "xfs_inode.h"
24 25
#include "xfs_quota.h"
#include "xfs_trans.h"
26 27
#include "xfs_trace.h"
#include "xfs_icache.h"
28
#include "xfs_qm.h"
29 30 31
#include <linux/quota.h>


32 33 34 35 36 37
static void
xfs_qm_fill_state(
	struct qc_type_state	*tstate,
	struct xfs_mount	*mp,
	struct xfs_inode	*ip,
	xfs_ino_t		ino)
38
{
39 40 41 42 43 44 45 46 47 48
	struct xfs_quotainfo *q = mp->m_quotainfo;
	bool tempqip = false;

	tstate->ino = ino;
	if (!ip && ino == NULLFSINO)
		return;
	if (!ip) {
		if (xfs_iget(mp, NULL, ino, 0, 0, &ip))
			return;
		tempqip = true;
49
	}
50 51 52 53 54 55 56 57 58 59 60
	tstate->flags |= QCI_SYSFILE;
	tstate->blocks = ip->i_d.di_nblocks;
	tstate->nextents = ip->i_d.di_nextents;
	tstate->spc_timelimit = q->qi_btimelimit;
	tstate->ino_timelimit = q->qi_itimelimit;
	tstate->rt_spc_timelimit = q->qi_rtbtimelimit;
	tstate->spc_warnlimit = q->qi_bwarnlimit;
	tstate->ino_warnlimit = q->qi_iwarnlimit;
	tstate->rt_spc_warnlimit = q->qi_rtbwarnlimit;
	if (tempqip)
		IRELE(ip);
61 62
}

63 64 65 66 67 68
/*
 * Return quota status information, such as enforcements, quota file inode
 * numbers etc.
 */
static int
xfs_fs_get_quota_state(
69
	struct super_block	*sb,
70
	struct qc_state		*state)
71
{
72 73
	struct xfs_mount *mp = XFS_M(sb);
	struct xfs_quotainfo *q = mp->m_quotainfo;
74

75
	memset(state, 0, sizeof(*state));
76
	if (!XFS_IS_QUOTA_RUNNING(mp))
77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
		return 0;
	state->s_incoredqs = q->qi_dquots;
	if (XFS_IS_UQUOTA_RUNNING(mp))
		state->s_state[USRQUOTA].flags |= QCI_ACCT_ENABLED;
	if (XFS_IS_UQUOTA_ENFORCED(mp))
		state->s_state[USRQUOTA].flags |= QCI_LIMITS_ENFORCED;
	if (XFS_IS_GQUOTA_RUNNING(mp))
		state->s_state[GRPQUOTA].flags |= QCI_ACCT_ENABLED;
	if (XFS_IS_GQUOTA_ENFORCED(mp))
		state->s_state[GRPQUOTA].flags |= QCI_LIMITS_ENFORCED;
	if (XFS_IS_PQUOTA_RUNNING(mp))
		state->s_state[PRJQUOTA].flags |= QCI_ACCT_ENABLED;
	if (XFS_IS_PQUOTA_ENFORCED(mp))
		state->s_state[PRJQUOTA].flags |= QCI_LIMITS_ENFORCED;

	xfs_qm_fill_state(&state->s_state[USRQUOTA], mp, q->qi_uquotaip,
			  mp->m_sb.sb_uquotino);
	xfs_qm_fill_state(&state->s_state[GRPQUOTA], mp, q->qi_gquotaip,
			  mp->m_sb.sb_gquotino);
	xfs_qm_fill_state(&state->s_state[PRJQUOTA], mp, q->qi_pquotaip,
			  mp->m_sb.sb_pquotino);
	return 0;
99 100
}

101
STATIC int
102
xfs_quota_type(int type)
103
{
104 105 106 107 108 109 110 111
	switch (type) {
	case USRQUOTA:
		return XFS_DQ_USER;
	case GRPQUOTA:
		return XFS_DQ_GROUP;
	default:
		return XFS_DQ_PROJ;
	}
112 113
}

114 115
static unsigned int
xfs_quota_flags(unsigned int uflags)
116
{
117
	unsigned int flags = 0;
118

119
	if (uflags & FS_QUOTA_UDQ_ACCT)
120
		flags |= XFS_UQUOTA_ACCT;
121
	if (uflags & FS_QUOTA_PDQ_ACCT)
122
		flags |= XFS_PQUOTA_ACCT;
123
	if (uflags & FS_QUOTA_GDQ_ACCT)
124
		flags |= XFS_GQUOTA_ACCT;
125
	if (uflags & FS_QUOTA_UDQ_ENFD)
126
		flags |= XFS_UQUOTA_ENFD;
127 128 129 130
	if (uflags & FS_QUOTA_GDQ_ENFD)
		flags |= XFS_GQUOTA_ENFD;
	if (uflags & FS_QUOTA_PDQ_ENFD)
		flags |= XFS_PQUOTA_ENFD;
131

132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162
	return flags;
}

STATIC int
xfs_quota_enable(
	struct super_block	*sb,
	unsigned int		uflags)
{
	struct xfs_mount	*mp = XFS_M(sb);

	if (sb->s_flags & MS_RDONLY)
		return -EROFS;
	if (!XFS_IS_QUOTA_RUNNING(mp))
		return -ENOSYS;

	return xfs_qm_scall_quotaon(mp, xfs_quota_flags(uflags));
}

STATIC int
xfs_quota_disable(
	struct super_block	*sb,
	unsigned int		uflags)
{
	struct xfs_mount	*mp = XFS_M(sb);

	if (sb->s_flags & MS_RDONLY)
		return -EROFS;
	if (!XFS_IS_QUOTA_RUNNING(mp))
		return -ENOSYS;
	if (!XFS_IS_QUOTA_ON(mp))
		return -EINVAL;
163

164
	return xfs_qm_scall_quotaoff(mp, xfs_quota_flags(uflags));
165 166
}

E
Eric Sandeen 已提交
167 168 169 170 171 172 173
STATIC int
xfs_fs_rm_xquota(
	struct super_block	*sb,
	unsigned int		uflags)
{
	struct xfs_mount	*mp = XFS_M(sb);
	unsigned int		flags = 0;
D
Dave Chinner 已提交
174

E
Eric Sandeen 已提交
175 176 177 178 179 180 181 182 183 184
	if (sb->s_flags & MS_RDONLY)
		return -EROFS;

	if (XFS_IS_QUOTA_ON(mp))
		return -EINVAL;

	if (uflags & FS_USER_QUOTA)
		flags |= XFS_DQ_USER;
	if (uflags & FS_GROUP_QUOTA)
		flags |= XFS_DQ_GROUP;
185
	if (uflags & FS_PROJ_QUOTA)
E
Eric Sandeen 已提交
186 187
		flags |= XFS_DQ_PROJ;

D
Dave Chinner 已提交
188 189
	return xfs_qm_scall_trunc_qfiles(mp, flags);
}
E
Eric Sandeen 已提交
190

191
STATIC int
C
Christoph Hellwig 已提交
192
xfs_fs_get_dqblk(
193
	struct super_block	*sb,
E
Eric W. Biederman 已提交
194
	struct kqid		qid,
195
	struct qc_dqblk		*qdq)
196 197 198 199 200 201 202 203
{
	struct xfs_mount	*mp = XFS_M(sb);

	if (!XFS_IS_QUOTA_RUNNING(mp))
		return -ENOSYS;
	if (!XFS_IS_QUOTA_ON(mp))
		return -ESRCH;

D
Dave Chinner 已提交
204
	return xfs_qm_scall_getquota(mp, from_kqid(&init_user_ns, qid),
205
				      xfs_quota_type(qid.type), qdq);
206 207 208
}

STATIC int
C
Christoph Hellwig 已提交
209
xfs_fs_set_dqblk(
210
	struct super_block	*sb,
E
Eric W. Biederman 已提交
211
	struct kqid		qid,
212
	struct qc_dqblk		*qdq)
213 214 215 216 217 218 219 220 221 222
{
	struct xfs_mount	*mp = XFS_M(sb);

	if (sb->s_flags & MS_RDONLY)
		return -EROFS;
	if (!XFS_IS_QUOTA_RUNNING(mp))
		return -ENOSYS;
	if (!XFS_IS_QUOTA_ON(mp))
		return -ESRCH;

D
Dave Chinner 已提交
223
	return xfs_qm_scall_setqlim(mp, from_kqid(&init_user_ns, qid),
224
				     xfs_quota_type(qid.type), qdq);
225 226
}

227
const struct quotactl_ops xfs_quotactl_operations = {
228
	.get_state		= xfs_fs_get_quota_state,
229 230
	.quota_enable		= xfs_quota_enable,
	.quota_disable		= xfs_quota_disable,
E
Eric Sandeen 已提交
231
	.rm_xquota		= xfs_fs_rm_xquota,
C
Christoph Hellwig 已提交
232
	.get_dqblk		= xfs_fs_get_dqblk,
C
Christoph Hellwig 已提交
233
	.set_dqblk		= xfs_fs_set_dqblk,
234
};