quota.c 7.8 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 139 140 141 142 143 144
	return sb->s_qcop->set_info(sb, type, &info);
}

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

145 146 147 148
	if (!sb_has_quota_active(sb, type))
		return -ESRCH;
	if (!sb->s_qcop->get_dqblk)
		return -ENOSYS;
C
Christoph Hellwig 已提交
149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
	ret = sb->s_qcop->get_dqblk(sb, type, id, &idq);
	if (ret)
		return ret;
	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;
164 165 166 167
	if (!sb_has_quota_active(sb, type))
		return -ESRCH;
	if (!sb->s_qcop->set_dqblk)
		return -ENOSYS;
C
Christoph Hellwig 已提交
168 169 170 171 172 173 174 175 176
	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;
177 178
	if (!sb->s_qcop->set_xstate)
		return -ENOSYS;
C
Christoph Hellwig 已提交
179 180 181 182 183 184 185
	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;
186 187 188

	if (!sb->s_qcop->get_xstate)
		return -ENOSYS;
C
Christoph Hellwig 已提交
189 190 191 192 193
	ret = sb->s_qcop->get_xstate(sb, &fqs);
	if (!ret && copy_to_user(addr, &fqs, sizeof(fqs)))
		return -EFAULT;
	return ret;
}
L
Linus Torvalds 已提交
194

C
Christoph Hellwig 已提交
195 196 197 198 199 200 201
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;
202 203
	if (!sb->s_qcop->set_xquota)
		return -ENOSYS;
C
Christoph Hellwig 已提交
204 205 206 207 208 209 210 211 212
	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;

213 214
	if (!sb->s_qcop->get_xquota)
		return -ENOSYS;
C
Christoph Hellwig 已提交
215 216 217 218 219 220 221 222 223 224
	ret = sb->s_qcop->get_xquota(sb, type, id, &fdq);
	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)
{
225 226 227 228 229 230 231 232 233 234 235
	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 已提交
236 237 238 239
	switch (cmd) {
	case Q_QUOTAON:
		return quota_quotaon(sb, type, cmd, id, addr);
	case Q_QUOTAOFF:
240 241
		if (!sb->s_qcop->quota_off)
			return -ENOSYS;
C
Christoph Hellwig 已提交
242 243 244 245 246 247 248 249 250 251 252 253
		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:
254 255
		if (!sb->s_qcop->quota_sync)
			return -ENOSYS;
256
		return sb->s_qcop->quota_sync(sb, type, 1);
C
Christoph Hellwig 已提交
257 258 259 260 261 262 263 264 265 266 267
	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 已提交
268 269 270 271 272
		/* caller already holds s_umount */
		if (sb->s_flags & MS_RDONLY)
			return -EROFS;
		writeback_inodes_sb(sb);
		return 0;
C
Christoph Hellwig 已提交
273
	default:
274
		return -EINVAL;
L
Linus Torvalds 已提交
275 276 277
	}
}

278 279 280 281
/*
 * 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 已提交
282
static struct super_block *quotactl_block(const char __user *special)
283 284 285 286 287 288 289
{
#ifdef CONFIG_BLOCK
	struct block_device *bdev;
	struct super_block *sb;
	char *tmp = getname(special);

	if (IS_ERR(tmp))
290
		return ERR_CAST(tmp);
291 292 293
	bdev = lookup_bdev(tmp);
	putname(tmp);
	if (IS_ERR(bdev))
294
		return ERR_CAST(bdev);
295 296 297 298 299 300 301 302 303 304 305
	sb = get_super(bdev);
	bdput(bdev);
	if (!sb)
		return ERR_PTR(-ENODEV);

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

L
Linus Torvalds 已提交
306 307 308 309 310 311
/*
 * 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.
 */
312 313
SYSCALL_DEFINE4(quotactl, unsigned int, cmd, const char __user *, special,
		qid_t, id, void __user *, addr)
L
Linus Torvalds 已提交
314 315 316 317 318 319 320 321
{
	uint cmds, type;
	struct super_block *sb = NULL;
	int ret;

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

322 323 324 325 326 327 328 329 330
	/*
	 * 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 已提交
331 332
	}

333 334 335 336
	sb = quotactl_block(special);
	if (IS_ERR(sb))
		return PTR_ERR(sb);

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

339
	drop_super(sb);
L
Linus Torvalds 已提交
340 341
	return ret;
}