aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEric Sandeen <sandeen@sandeen.net>2014-05-05 03:25:50 -0400
committerDave Chinner <david@fromorbit.com>2014-05-05 03:25:50 -0400
commit9da93f9b7cdf8ab28da6b364cdc1fafc8670b4dc (patch)
treea0315376046dce4cfcfd498af5617e4088c8e86c
parentc9eaa447e77efe77b7fa4c953bd62de8297fd6c5 (diff)
xfs: fix Q_XQUOTARM ioctl
The Q_XQUOTARM quotactl was not working properly, because we weren't passing around proper flags. The xfs_fs_set_xstate() ioctl handler used the same flags for Q_XQUOTAON/OFF as well as for Q_XQUOTARM, but Q_XQUOTAON/OFF look for XFS_UQUOTA_ACCT, XFS_UQUOTA_ENFD, XFS_GQUOTA_ACCT etc, i.e. quota type + state, while Q_XQUOTARM looks only for the type of quota, i.e. XFS_DQ_USER, XFS_DQ_GROUP etc. Unfortunately these flag spaces overlap a bit, so we got semi-random results for Q_XQUOTARM; i.e. the value for XFS_DQ_USER == XFS_UQUOTA_ACCT, etc. yeargh. Add a new quotactl op vector specifically for the QUOTARM operation, since it operates with a different flag space. This has been broken more or less forever, AFAICT. Signed-off-by: Eric Sandeen <sandeen@redhat.com> Acked-by: Jan Kara <jack@suse.cz> Reviewed-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Dave Chinner <david@fromorbit.com>
-rw-r--r--fs/quota/quota.c14
-rw-r--r--fs/xfs/xfs_quotaops.c29
-rw-r--r--include/linux/quota.h1
3 files changed, 39 insertions, 5 deletions
diff --git a/fs/quota/quota.c b/fs/quota/quota.c
index 2b363e23f36e..ff3f0b3cfdb3 100644
--- a/fs/quota/quota.c
+++ b/fs/quota/quota.c
@@ -278,6 +278,17 @@ static int quota_getxquota(struct super_block *sb, int type, qid_t id,
278 return ret; 278 return ret;
279} 279}
280 280
281static int quota_rmxquota(struct super_block *sb, void __user *addr)
282{
283 __u32 flags;
284
285 if (copy_from_user(&flags, addr, sizeof(flags)))
286 return -EFAULT;
287 if (!sb->s_qcop->rm_xquota)
288 return -ENOSYS;
289 return sb->s_qcop->rm_xquota(sb, flags);
290}
291
281/* Copy parameters and call proper function */ 292/* Copy parameters and call proper function */
282static int do_quotactl(struct super_block *sb, int type, int cmd, qid_t id, 293static int do_quotactl(struct super_block *sb, int type, int cmd, qid_t id,
283 void __user *addr, struct path *path) 294 void __user *addr, struct path *path)
@@ -316,8 +327,9 @@ static int do_quotactl(struct super_block *sb, int type, int cmd, qid_t id,
316 return sb->s_qcop->quota_sync(sb, type); 327 return sb->s_qcop->quota_sync(sb, type);
317 case Q_XQUOTAON: 328 case Q_XQUOTAON:
318 case Q_XQUOTAOFF: 329 case Q_XQUOTAOFF:
319 case Q_XQUOTARM:
320 return quota_setxstate(sb, cmd, addr); 330 return quota_setxstate(sb, cmd, addr);
331 case Q_XQUOTARM:
332 return quota_rmxquota(sb, addr);
321 case Q_XGETQSTAT: 333 case Q_XGETQSTAT:
322 return quota_getxstate(sb, addr); 334 return quota_getxstate(sb, addr);
323 case Q_XGETQSTATV: 335 case Q_XGETQSTATV:
diff --git a/fs/xfs/xfs_quotaops.c b/fs/xfs/xfs_quotaops.c
index af33cafe69b6..2ad1b9822e92 100644
--- a/fs/xfs/xfs_quotaops.c
+++ b/fs/xfs/xfs_quotaops.c
@@ -100,16 +100,36 @@ xfs_fs_set_xstate(
100 if (!XFS_IS_QUOTA_ON(mp)) 100 if (!XFS_IS_QUOTA_ON(mp))
101 return -EINVAL; 101 return -EINVAL;
102 return -xfs_qm_scall_quotaoff(mp, flags); 102 return -xfs_qm_scall_quotaoff(mp, flags);
103 case Q_XQUOTARM:
104 if (XFS_IS_QUOTA_ON(mp))
105 return -EINVAL;
106 return -xfs_qm_scall_trunc_qfiles(mp, flags);
107 } 103 }
108 104
109 return -EINVAL; 105 return -EINVAL;
110} 106}
111 107
112STATIC int 108STATIC int
109xfs_fs_rm_xquota(
110 struct super_block *sb,
111 unsigned int uflags)
112{
113 struct xfs_mount *mp = XFS_M(sb);
114 unsigned int flags = 0;
115
116 if (sb->s_flags & MS_RDONLY)
117 return -EROFS;
118
119 if (XFS_IS_QUOTA_ON(mp))
120 return -EINVAL;
121
122 if (uflags & FS_USER_QUOTA)
123 flags |= XFS_DQ_USER;
124 if (uflags & FS_GROUP_QUOTA)
125 flags |= XFS_DQ_GROUP;
126 if (uflags & FS_USER_QUOTA)
127 flags |= XFS_DQ_PROJ;
128
129 return -xfs_qm_scall_trunc_qfiles(mp, flags);
130}
131
132STATIC int
113xfs_fs_get_dqblk( 133xfs_fs_get_dqblk(
114 struct super_block *sb, 134 struct super_block *sb,
115 struct kqid qid, 135 struct kqid qid,
@@ -149,6 +169,7 @@ const struct quotactl_ops xfs_quotactl_operations = {
149 .get_xstatev = xfs_fs_get_xstatev, 169 .get_xstatev = xfs_fs_get_xstatev,
150 .get_xstate = xfs_fs_get_xstate, 170 .get_xstate = xfs_fs_get_xstate,
151 .set_xstate = xfs_fs_set_xstate, 171 .set_xstate = xfs_fs_set_xstate,
172 .rm_xquota = xfs_fs_rm_xquota,
152 .get_dqblk = xfs_fs_get_dqblk, 173 .get_dqblk = xfs_fs_get_dqblk,
153 .set_dqblk = xfs_fs_set_dqblk, 174 .set_dqblk = xfs_fs_set_dqblk,
154}; 175};
diff --git a/include/linux/quota.h b/include/linux/quota.h
index cc7494a35429..0f3c5d38da1f 100644
--- a/include/linux/quota.h
+++ b/include/linux/quota.h
@@ -329,6 +329,7 @@ struct quotactl_ops {
329 int (*get_xstate)(struct super_block *, struct fs_quota_stat *); 329 int (*get_xstate)(struct super_block *, struct fs_quota_stat *);
330 int (*set_xstate)(struct super_block *, unsigned int, int); 330 int (*set_xstate)(struct super_block *, unsigned int, int);
331 int (*get_xstatev)(struct super_block *, struct fs_quota_statv *); 331 int (*get_xstatev)(struct super_block *, struct fs_quota_statv *);
332 int (*rm_xquota)(struct super_block *, unsigned int);
332}; 333};
333 334
334struct quota_format_type { 335struct quota_format_type {