quota.c 8.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/*
 * Quota code necessary even when VFS quota support is not compiled
 * into the kernel.  The interesting stuff is over in dquot.c, here
 * we have symbols for initial quotactl(2) handling, the sysctl(2)
 * variables, etc - things needed even when quota support disabled.
 */

#include <linux/fs.h>
#include <linux/namei.h>
#include <linux/slab.h>
#include <asm/current.h>
#include <asm/uaccess.h>
#include <linux/kernel.h>
#include <linux/security.h>
#include <linux/syscalls.h>
#include <linux/buffer_head.h>
17
#include <linux/capability.h>
A
Adrian Bunk 已提交
18
#include <linux/quotaops.h>
19
#include <linux/types.h>
C
Christoph Hellwig 已提交
20
#include <linux/writeback.h>
L
Linus Torvalds 已提交
21

22 23
static int check_quotactl_permission(struct super_block *sb, int type, int cmd,
				     qid_t id)
L
Linus Torvalds 已提交
24
{
25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
	switch (cmd) {
	/* these commands do not require any special privilegues */
	case Q_GETFMT:
	case Q_SYNC:
	case Q_GETINFO:
	case Q_XGETQSTAT:
	case Q_XQUOTASYNC:
		break;
	/* allow to query information for dquots we "own" */
	case Q_GETQUOTA:
	case Q_XGETQUOTA:
		if ((type == USRQUOTA && current_euid() == id) ||
		    (type == GRPQUOTA && in_egroup_p(id)))
			break;
		/*FALLTHROUGH*/
	default:
L
Linus Torvalds 已提交
41 42 43 44
		if (!capable(CAP_SYS_ADMIN))
			return -EPERM;
	}

45
	return security_quotactl(cmd, type, id, sb);
L
Linus Torvalds 已提交
46 47
}

48
static int quota_sync_all(int type)
L
Linus Torvalds 已提交
49
{
50
	struct super_block *sb;
51 52 53 54 55 56 57
	int ret;

	if (type >= MAXQUOTAS)
		return -EINVAL;
	ret = security_quotactl(Q_SYNC, type, 0, NULL);
	if (ret)
		return ret;
58 59 60 61

	spin_lock(&sb_lock);
restart:
	list_for_each_entry(sb, &super_blocks, s_list) {
62 63 64
		if (!sb->s_qcop || !sb->s_qcop->quota_sync)
			continue;

65 66 67
		sb->s_count++;
		spin_unlock(&sb_lock);
		down_read(&sb->s_umount);
68
		if (sb->s_root)
69
			sb->s_qcop->quota_sync(sb, type, 1);
70 71 72 73
		up_read(&sb->s_umount);
		spin_lock(&sb_lock);
		if (__put_super_and_need_restart(sb))
			goto restart;
L
Linus Torvalds 已提交
74
	}
75
	spin_unlock(&sb_lock);
76 77

	return 0;
L
Linus Torvalds 已提交
78 79
}

C
Christoph Hellwig 已提交
80 81
static int quota_quotaon(struct super_block *sb, int type, int cmd, qid_t id,
		         void __user *addr)
L
Linus Torvalds 已提交
82
{
C
Christoph Hellwig 已提交
83
	char *pathname;
84
	int ret = -ENOSYS;
L
Linus Torvalds 已提交
85

C
Christoph Hellwig 已提交
86 87 88
	pathname = getname(addr);
	if (IS_ERR(pathname))
		return PTR_ERR(pathname);
89 90
	if (sb->s_qcop->quota_on)
		ret = sb->s_qcop->quota_on(sb, type, id, pathname, 0);
C
Christoph Hellwig 已提交
91 92 93
	putname(pathname);
	return ret;
}
L
Linus Torvalds 已提交
94

C
Christoph Hellwig 已提交
95 96 97
static int quota_getfmt(struct super_block *sb, int type, void __user *addr)
{
	__u32 fmt;
L
Linus Torvalds 已提交
98

C
Christoph Hellwig 已提交
99 100 101 102 103 104 105 106 107 108 109
	down_read(&sb_dqopt(sb)->dqptr_sem);
	if (!sb_has_quota_active(sb, type)) {
		up_read(&sb_dqopt(sb)->dqptr_sem);
		return -ESRCH;
	}
	fmt = sb_dqopt(sb)->info[type].dqi_format->qf_fmt_id;
	up_read(&sb_dqopt(sb)->dqptr_sem);
	if (copy_to_user(addr, &fmt, sizeof(fmt)))
		return -EFAULT;
	return 0;
}
L
Linus Torvalds 已提交
110

C
Christoph Hellwig 已提交
111 112 113 114
static int quota_getinfo(struct super_block *sb, int type, void __user *addr)
{
	struct if_dqinfo info;
	int ret;
L
Linus Torvalds 已提交
115

116 117 118 119
	if (!sb_has_quota_active(sb, type))
		return -ESRCH;
	if (!sb->s_qcop->get_info)
		return -ENOSYS;
C
Christoph Hellwig 已提交
120 121 122 123 124
	ret = sb->s_qcop->get_info(sb, type, &info);
	if (!ret && copy_to_user(addr, &info, sizeof(info)))
		return -EFAULT;
	return ret;
}
L
Linus Torvalds 已提交
125

C
Christoph Hellwig 已提交
126 127 128
static int quota_setinfo(struct super_block *sb, int type, void __user *addr)
{
	struct if_dqinfo info;
L
Linus Torvalds 已提交
129

C
Christoph Hellwig 已提交
130 131
	if (copy_from_user(&info, addr, sizeof(info)))
		return -EFAULT;
132 133 134 135
	if (!sb_has_quota_active(sb, type))
		return -ESRCH;
	if (!sb->s_qcop->set_info)
		return -ENOSYS;
C
Christoph Hellwig 已提交
136 137 138
	return sb->s_qcop->set_info(sb, type, &info);
}

C
Christoph Hellwig 已提交
139 140 141 142 143 144 145 146 147 148 149 150 151
static void copy_to_if_dqblk(struct if_dqblk *dst, struct fs_disk_quota *src)
{
	dst->dqb_bhardlimit = src->d_blk_hardlimit;
	dst->dqb_bsoftlimit = src->d_blk_softlimit;
	dst->dqb_curspace = src->d_bcount;
	dst->dqb_ihardlimit = src->d_ino_hardlimit;
	dst->dqb_isoftlimit = src->d_ino_softlimit;
	dst->dqb_curinodes = src->d_icount;
	dst->dqb_btime = src->d_btimer;
	dst->dqb_itime = src->d_itimer;
	dst->dqb_valid = QIF_ALL;
}

C
Christoph Hellwig 已提交
152 153 154
static int quota_getquota(struct super_block *sb, int type, qid_t id,
			  void __user *addr)
{
C
Christoph Hellwig 已提交
155
	struct fs_disk_quota fdq;
C
Christoph Hellwig 已提交
156 157 158
	struct if_dqblk idq;
	int ret;

159 160
	if (!sb->s_qcop->get_dqblk)
		return -ENOSYS;
C
Christoph Hellwig 已提交
161
	ret = sb->s_qcop->get_dqblk(sb, type, id, &fdq);
C
Christoph Hellwig 已提交
162 163
	if (ret)
		return ret;
C
Christoph Hellwig 已提交
164
	copy_to_if_dqblk(&idq, &fdq);
C
Christoph Hellwig 已提交
165 166 167 168 169 170 171 172 173 174 175 176
	if (copy_to_user(addr, &idq, sizeof(idq)))
		return -EFAULT;
	return 0;
}

static int quota_setquota(struct super_block *sb, int type, qid_t id,
			  void __user *addr)
{
	struct if_dqblk idq;

	if (copy_from_user(&idq, addr, sizeof(idq)))
		return -EFAULT;
177 178 179 180
	if (!sb_has_quota_active(sb, type))
		return -ESRCH;
	if (!sb->s_qcop->set_dqblk)
		return -ENOSYS;
C
Christoph Hellwig 已提交
181 182 183 184 185 186 187 188 189
	return sb->s_qcop->set_dqblk(sb, type, id, &idq);
}

static int quota_setxstate(struct super_block *sb, int cmd, void __user *addr)
{
	__u32 flags;

	if (copy_from_user(&flags, addr, sizeof(flags)))
		return -EFAULT;
190 191
	if (!sb->s_qcop->set_xstate)
		return -ENOSYS;
C
Christoph Hellwig 已提交
192 193 194 195 196 197 198
	return sb->s_qcop->set_xstate(sb, flags, cmd);
}

static int quota_getxstate(struct super_block *sb, void __user *addr)
{
	struct fs_quota_stat fqs;
	int ret;
199 200 201

	if (!sb->s_qcop->get_xstate)
		return -ENOSYS;
C
Christoph Hellwig 已提交
202 203 204 205 206
	ret = sb->s_qcop->get_xstate(sb, &fqs);
	if (!ret && copy_to_user(addr, &fqs, sizeof(fqs)))
		return -EFAULT;
	return ret;
}
L
Linus Torvalds 已提交
207

C
Christoph Hellwig 已提交
208 209 210 211 212 213 214
static int quota_setxquota(struct super_block *sb, int type, qid_t id,
			   void __user *addr)
{
	struct fs_disk_quota fdq;

	if (copy_from_user(&fdq, addr, sizeof(fdq)))
		return -EFAULT;
215 216
	if (!sb->s_qcop->set_xquota)
		return -ENOSYS;
C
Christoph Hellwig 已提交
217 218 219 220 221 222 223 224 225
	return sb->s_qcop->set_xquota(sb, type, id, &fdq);
}

static int quota_getxquota(struct super_block *sb, int type, qid_t id,
			   void __user *addr)
{
	struct fs_disk_quota fdq;
	int ret;

C
Christoph Hellwig 已提交
226
	if (!sb->s_qcop->get_dqblk)
227
		return -ENOSYS;
C
Christoph Hellwig 已提交
228
	ret = sb->s_qcop->get_dqblk(sb, type, id, &fdq);
C
Christoph Hellwig 已提交
229 230 231 232 233 234 235 236 237
	if (!ret && copy_to_user(addr, &fdq, sizeof(fdq)))
		return -EFAULT;
	return ret;
}

/* Copy parameters and call proper function */
static int do_quotactl(struct super_block *sb, int type, int cmd, qid_t id,
		       void __user *addr)
{
238 239 240 241 242 243 244 245 246 247 248
	int ret;

	if (type >= (XQM_COMMAND(cmd) ? XQM_MAXQUOTAS : MAXQUOTAS))
		return -EINVAL;
	if (!sb->s_qcop)
		return -ENOSYS;

	ret = check_quotactl_permission(sb, type, cmd, id);
	if (ret < 0)
		return ret;

C
Christoph Hellwig 已提交
249 250 251 252
	switch (cmd) {
	case Q_QUOTAON:
		return quota_quotaon(sb, type, cmd, id, addr);
	case Q_QUOTAOFF:
253 254
		if (!sb->s_qcop->quota_off)
			return -ENOSYS;
C
Christoph Hellwig 已提交
255 256 257 258 259 260 261 262 263 264 265 266
		return sb->s_qcop->quota_off(sb, type, 0);
	case Q_GETFMT:
		return quota_getfmt(sb, type, addr);
	case Q_GETINFO:
		return quota_getinfo(sb, type, addr);
	case Q_SETINFO:
		return quota_setinfo(sb, type, addr);
	case Q_GETQUOTA:
		return quota_getquota(sb, type, id, addr);
	case Q_SETQUOTA:
		return quota_setquota(sb, type, id, addr);
	case Q_SYNC:
267 268
		if (!sb->s_qcop->quota_sync)
			return -ENOSYS;
269
		return sb->s_qcop->quota_sync(sb, type, 1);
C
Christoph Hellwig 已提交
270 271 272 273 274 275 276 277 278 279 280
	case Q_XQUOTAON:
	case Q_XQUOTAOFF:
	case Q_XQUOTARM:
		return quota_setxstate(sb, cmd, addr);
	case Q_XGETQSTAT:
		return quota_getxstate(sb, addr);
	case Q_XSETQLIM:
		return quota_setxquota(sb, type, id, addr);
	case Q_XGETQUOTA:
		return quota_getxquota(sb, type, id, addr);
	case Q_XQUOTASYNC:
C
Christoph Hellwig 已提交
281 282 283 284 285
		/* caller already holds s_umount */
		if (sb->s_flags & MS_RDONLY)
			return -EROFS;
		writeback_inodes_sb(sb);
		return 0;
C
Christoph Hellwig 已提交
286
	default:
287
		return -EINVAL;
L
Linus Torvalds 已提交
288 289 290
	}
}

291 292 293 294
/*
 * look up a superblock on which quota ops will be performed
 * - use the name of a block device to find the superblock thereon
 */
J
Jan Kara 已提交
295
static struct super_block *quotactl_block(const char __user *special)
296 297 298 299 300 301 302
{
#ifdef CONFIG_BLOCK
	struct block_device *bdev;
	struct super_block *sb;
	char *tmp = getname(special);

	if (IS_ERR(tmp))
303
		return ERR_CAST(tmp);
304 305 306
	bdev = lookup_bdev(tmp);
	putname(tmp);
	if (IS_ERR(bdev))
307
		return ERR_CAST(bdev);
308 309 310 311 312 313 314 315 316 317 318
	sb = get_super(bdev);
	bdput(bdev);
	if (!sb)
		return ERR_PTR(-ENODEV);

	return sb;
#else
	return ERR_PTR(-ENODEV);
#endif
}

L
Linus Torvalds 已提交
319 320 321 322 323 324
/*
 * This is the system call interface. This communicates with
 * the user-level programs. Currently this only supports diskquota
 * calls. Maybe we need to add the process quotas etc. in the future,
 * but we probably should use rlimits for that.
 */
325 326
SYSCALL_DEFINE4(quotactl, unsigned int, cmd, const char __user *, special,
		qid_t, id, void __user *, addr)
L
Linus Torvalds 已提交
327 328 329 330 331 332 333 334
{
	uint cmds, type;
	struct super_block *sb = NULL;
	int ret;

	cmds = cmd >> SUBCMDSHIFT;
	type = cmd & SUBCMDMASK;

335 336 337 338 339 340 341 342 343
	/*
	 * As a special case Q_SYNC can be called without a specific device.
	 * It will iterate all superblocks that have quota enabled and call
	 * the sync action on each of them.
	 */
	if (!special) {
		if (cmds == Q_SYNC)
			return quota_sync_all(type);
		return -ENODEV;
L
Linus Torvalds 已提交
344 345
	}

346 347 348 349
	sb = quotactl_block(special);
	if (IS_ERR(sb))
		return PTR_ERR(sb);

350
	ret = do_quotactl(sb, type, cmds, id, addr);
L
Linus Torvalds 已提交
351

352
	drop_super(sb);
L
Linus Torvalds 已提交
353 354
	return ret;
}