inode.c 8.8 KB
Newer Older
M
Miklos Szeredi 已提交
1 2 3 4 5 6 7 8 9 10 11 12
/*
 *
 * Copyright (C) 2011 Novell Inc.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 as published by
 * the Free Software Foundation.
 */

#include <linux/fs.h>
#include <linux/slab.h>
#include <linux/xattr.h>
13
#include <linux/posix_acl.h>
M
Miklos Szeredi 已提交
14 15
#include "overlayfs.h"

16
static int ovl_copy_up_truncate(struct dentry *dentry)
M
Miklos Szeredi 已提交
17 18 19 20 21
{
	int err;
	struct dentry *parent;
	struct kstat stat;
	struct path lowerpath;
22
	const struct cred *old_cred;
M
Miklos Szeredi 已提交
23 24 25 26 27 28 29 30

	parent = dget_parent(dentry);
	err = ovl_copy_up(parent);
	if (err)
		goto out_dput_parent;

	ovl_path_lower(dentry, &lowerpath);

31 32 33 34 35 36 37
	old_cred = ovl_override_creds(dentry->d_sb);
	err = vfs_getattr(&lowerpath, &stat);
	if (!err) {
		stat.size = 0;
		err = ovl_copy_up_one(parent, dentry, &lowerpath, &stat);
	}
	revert_creds(old_cred);
M
Miklos Szeredi 已提交
38 39 40 41 42 43 44 45 46 47

out_dput_parent:
	dput(parent);
	return err;
}

int ovl_setattr(struct dentry *dentry, struct iattr *attr)
{
	int err;
	struct dentry *upperdentry;
48
	const struct cred *old_cred;
M
Miklos Szeredi 已提交
49

50 51 52 53 54 55 56 57 58
	/*
	 * Check for permissions before trying to copy-up.  This is redundant
	 * since it will be rechecked later by ->setattr() on upper dentry.  But
	 * without this, copy-up can be triggered by just about anybody.
	 *
	 * We don't initialize inode->size, which just means that
	 * inode_newsize_ok() will always check against MAX_LFS_FILESIZE and not
	 * check for a swapfile (which this won't be anyway).
	 */
59
	err = setattr_prepare(dentry, attr);
60 61 62
	if (err)
		return err;

M
Miklos Szeredi 已提交
63 64 65 66
	err = ovl_want_write(dentry);
	if (err)
		goto out;

67 68 69 70
	err = ovl_copy_up(dentry);
	if (!err) {
		upperdentry = ovl_dentry_upper(dentry);

M
Miklos Szeredi 已提交
71 72 73
		if (attr->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
			attr->ia_valid &= ~ATTR_MODE;

A
Al Viro 已提交
74
		inode_lock(upperdentry->d_inode);
75
		old_cred = ovl_override_creds(dentry->d_sb);
M
Miklos Szeredi 已提交
76
		err = notify_change(upperdentry, attr, NULL);
77
		revert_creds(old_cred);
78 79
		if (!err)
			ovl_copyattr(upperdentry->d_inode, dentry->d_inode);
A
Al Viro 已提交
80
		inode_unlock(upperdentry->d_inode);
M
Miklos Szeredi 已提交
81 82 83 84 85 86 87 88 89 90
	}
	ovl_drop_write(dentry);
out:
	return err;
}

static int ovl_getattr(struct vfsmount *mnt, struct dentry *dentry,
			 struct kstat *stat)
{
	struct path realpath;
91 92
	const struct cred *old_cred;
	int err;
M
Miklos Szeredi 已提交
93 94

	ovl_path_real(dentry, &realpath);
95 96 97 98
	old_cred = ovl_override_creds(dentry->d_sb);
	err = vfs_getattr(&realpath, stat);
	revert_creds(old_cred);
	return err;
M
Miklos Szeredi 已提交
99 100 101 102 103
}

int ovl_permission(struct inode *inode, int mask)
{
	bool is_upper;
104
	struct inode *realinode = ovl_inode_real(inode, &is_upper);
105
	const struct cred *old_cred;
M
Miklos Szeredi 已提交
106 107 108 109 110
	int err;

	/* Careful in RCU walk mode */
	if (!realinode) {
		WARN_ON(!(mask & MAY_NOT_BLOCK));
111
		return -ECHILD;
M
Miklos Szeredi 已提交
112 113
	}

114 115 116 117 118 119 120 121 122
	/*
	 * Check overlay inode with the creds of task and underlying inode
	 * with creds of mounter
	 */
	err = generic_permission(inode, mask);
	if (err)
		return err;

	old_cred = ovl_override_creds(inode->i_sb);
123
	if (!is_upper && !special_file(realinode->i_mode) && mask & MAY_WRITE) {
124
		mask &= ~(MAY_WRITE | MAY_APPEND);
125 126 127
		/* Make sure mounter can read file for copy up later */
		mask |= MAY_READ;
	}
128
	err = inode_permission(realinode, mask);
129 130 131
	revert_creds(old_cred);

	return err;
M
Miklos Szeredi 已提交
132 133
}

134
static const char *ovl_get_link(struct dentry *dentry,
135 136
				struct inode *inode,
				struct delayed_call *done)
M
Miklos Szeredi 已提交
137
{
138 139
	const struct cred *old_cred;
	const char *p;
M
Miklos Szeredi 已提交
140

141 142 143
	if (!dentry)
		return ERR_PTR(-ECHILD);

144
	old_cred = ovl_override_creds(dentry->d_sb);
M
Miklos Szeredi 已提交
145
	p = vfs_get_link(ovl_dentry_real(dentry), done);
146 147
	revert_creds(old_cred);
	return p;
M
Miklos Szeredi 已提交
148 149
}

M
Miklos Szeredi 已提交
150
bool ovl_is_private_xattr(const char *name)
M
Miklos Szeredi 已提交
151
{
A
Andreas Gruenbacher 已提交
152 153
	return strncmp(name, OVL_XATTR_PREFIX,
		       sizeof(OVL_XATTR_PREFIX) - 1) == 0;
M
Miklos Szeredi 已提交
154 155
}

156 157
int ovl_xattr_set(struct dentry *dentry, const char *name, const void *value,
		  size_t size, int flags)
M
Miklos Szeredi 已提交
158 159
{
	int err;
160 161
	struct path realpath;
	enum ovl_path_type type = ovl_path_real(dentry, &realpath);
162
	const struct cred *old_cred;
M
Miklos Szeredi 已提交
163 164 165 166 167

	err = ovl_want_write(dentry);
	if (err)
		goto out;

168 169 170 171 172 173
	if (!value && !OVL_TYPE_UPPER(type)) {
		err = vfs_getxattr(realpath.dentry, name, NULL, 0);
		if (err < 0)
			goto out_drop_write;
	}

M
Miklos Szeredi 已提交
174 175 176 177
	err = ovl_copy_up(dentry);
	if (err)
		goto out_drop_write;

178 179 180
	if (!OVL_TYPE_UPPER(type))
		ovl_path_upper(dentry, &realpath);

181
	old_cred = ovl_override_creds(dentry->d_sb);
182 183 184 185 186 187
	if (value)
		err = vfs_setxattr(realpath.dentry, name, value, size, flags);
	else {
		WARN_ON(flags != XATTR_REPLACE);
		err = vfs_removexattr(realpath.dentry, name);
	}
188
	revert_creds(old_cred);
M
Miklos Szeredi 已提交
189 190 191 192 193 194 195

out_drop_write:
	ovl_drop_write(dentry);
out:
	return err;
}

196 197
int ovl_xattr_get(struct dentry *dentry, const char *name,
		  void *value, size_t size)
M
Miklos Szeredi 已提交
198
{
M
Miklos Szeredi 已提交
199
	struct dentry *realdentry = ovl_dentry_real(dentry);
200 201
	ssize_t res;
	const struct cred *old_cred;
202

203 204 205 206
	old_cred = ovl_override_creds(dentry->d_sb);
	res = vfs_getxattr(realdentry, name, value, size);
	revert_creds(old_cred);
	return res;
M
Miklos Szeredi 已提交
207 208 209 210
}

ssize_t ovl_listxattr(struct dentry *dentry, char *list, size_t size)
{
M
Miklos Szeredi 已提交
211
	struct dentry *realdentry = ovl_dentry_real(dentry);
M
Miklos Szeredi 已提交
212
	ssize_t res;
M
Miklos Szeredi 已提交
213 214
	size_t len;
	char *s;
215
	const struct cred *old_cred;
M
Miklos Szeredi 已提交
216

217
	old_cred = ovl_override_creds(dentry->d_sb);
M
Miklos Szeredi 已提交
218
	res = vfs_listxattr(realdentry, list, size);
219
	revert_creds(old_cred);
M
Miklos Szeredi 已提交
220 221 222 223
	if (res <= 0 || size == 0)
		return res;

	/* filter out private xattrs */
M
Miklos Szeredi 已提交
224 225
	for (s = list, len = res; len;) {
		size_t slen = strnlen(s, len) + 1;
M
Miklos Szeredi 已提交
226

M
Miklos Szeredi 已提交
227 228 229
		/* underlying fs providing us with an broken xattr list? */
		if (WARN_ON(slen > len))
			return -EIO;
M
Miklos Szeredi 已提交
230

M
Miklos Szeredi 已提交
231
		len -= slen;
M
Miklos Szeredi 已提交
232 233
		if (ovl_is_private_xattr(s)) {
			res -= slen;
M
Miklos Szeredi 已提交
234
			memmove(s, s + slen, len);
M
Miklos Szeredi 已提交
235
		} else {
M
Miklos Szeredi 已提交
236
			s += slen;
M
Miklos Szeredi 已提交
237 238 239 240 241 242
		}
	}

	return res;
}

243 244
struct posix_acl *ovl_get_acl(struct inode *inode, int type)
{
245
	struct inode *realinode = ovl_inode_real(inode, NULL);
246 247
	const struct cred *old_cred;
	struct posix_acl *acl;
248

249
	if (!IS_ENABLED(CONFIG_FS_POSIX_ACL) || !IS_POSIXACL(realinode))
250 251
		return NULL;

252
	old_cred = ovl_override_creds(inode->i_sb);
253
	acl = get_acl(realinode, type);
254 255 256
	revert_creds(old_cred);

	return acl;
257 258
}

M
Miklos Szeredi 已提交
259 260 261
static bool ovl_open_need_copy_up(int flags, enum ovl_path_type type,
				  struct dentry *realdentry)
{
M
Miklos Szeredi 已提交
262
	if (OVL_TYPE_UPPER(type))
M
Miklos Szeredi 已提交
263 264 265 266 267 268 269 270 271 272 273
		return false;

	if (special_file(realdentry->d_inode->i_mode))
		return false;

	if (!(OPEN_FMODE(flags) & FMODE_WRITE) && !(flags & O_TRUNC))
		return false;

	return true;
}

274
int ovl_open_maybe_copy_up(struct dentry *dentry, unsigned int file_flags)
M
Miklos Szeredi 已提交
275
{
276
	int err = 0;
M
Miklos Szeredi 已提交
277 278 279 280
	struct path realpath;
	enum ovl_path_type type;

	type = ovl_path_real(dentry, &realpath);
281
	if (ovl_open_need_copy_up(file_flags, type, realpath.dentry)) {
M
Miklos Szeredi 已提交
282
		err = ovl_want_write(dentry);
283 284 285 286 287 288 289
		if (!err) {
			if (file_flags & O_TRUNC)
				err = ovl_copy_up_truncate(dentry);
			else
				err = ovl_copy_up(dentry);
			ovl_drop_write(dentry);
		}
M
Miklos Szeredi 已提交
290 291
	}

292
	return err;
M
Miklos Szeredi 已提交
293 294
}

M
Miklos Szeredi 已提交
295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
int ovl_update_time(struct inode *inode, struct timespec *ts, int flags)
{
	struct dentry *alias;
	struct path upperpath;

	if (!(flags & S_ATIME))
		return 0;

	alias = d_find_any_alias(inode);
	if (!alias)
		return 0;

	ovl_path_upper(alias, &upperpath);
	if (upperpath.dentry) {
		touch_atime(&upperpath);
		inode->i_atime = d_inode(upperpath.dentry)->i_atime;
	}

	dput(alias);

	return 0;
}

M
Miklos Szeredi 已提交
318 319 320 321 322
static const struct inode_operations ovl_file_inode_operations = {
	.setattr	= ovl_setattr,
	.permission	= ovl_permission,
	.getattr	= ovl_getattr,
	.listxattr	= ovl_listxattr,
323
	.get_acl	= ovl_get_acl,
M
Miklos Szeredi 已提交
324
	.update_time	= ovl_update_time,
M
Miklos Szeredi 已提交
325 326 327 328
};

static const struct inode_operations ovl_symlink_inode_operations = {
	.setattr	= ovl_setattr,
329
	.get_link	= ovl_get_link,
M
Miklos Szeredi 已提交
330
	.readlink	= generic_readlink,
M
Miklos Szeredi 已提交
331 332
	.getattr	= ovl_getattr,
	.listxattr	= ovl_listxattr,
M
Miklos Szeredi 已提交
333
	.update_time	= ovl_update_time,
M
Miklos Szeredi 已提交
334 335
};

M
Miklos Szeredi 已提交
336
static void ovl_fill_inode(struct inode *inode, umode_t mode)
M
Miklos Szeredi 已提交
337 338 339
{
	inode->i_ino = get_next_ino();
	inode->i_mode = mode;
M
Miklos Szeredi 已提交
340
	inode->i_flags |= S_NOCMTIME;
341 342 343
#ifdef CONFIG_FS_POSIX_ACL
	inode->i_acl = inode->i_default_acl = ACL_DONT_CACHE;
#endif
M
Miklos Szeredi 已提交
344

345
	mode &= S_IFMT;
M
Miklos Szeredi 已提交
346 347 348 349 350 351 352 353 354 355
	switch (mode) {
	case S_IFDIR:
		inode->i_op = &ovl_dir_inode_operations;
		inode->i_fop = &ovl_dir_operations;
		break;

	case S_IFLNK:
		inode->i_op = &ovl_symlink_inode_operations;
		break;

M
Miklos Szeredi 已提交
356 357 358 359
	default:
		WARN(1, "illegal file type: %i\n", mode);
		/* Fall through */

M
Miklos Szeredi 已提交
360 361 362 363 364 365 366
	case S_IFREG:
	case S_IFSOCK:
	case S_IFBLK:
	case S_IFCHR:
	case S_IFIFO:
		inode->i_op = &ovl_file_inode_operations;
		break;
M
Miklos Szeredi 已提交
367 368
	}
}
M
Miklos Szeredi 已提交
369

M
Miklos Szeredi 已提交
370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402
struct inode *ovl_new_inode(struct super_block *sb, umode_t mode)
{
	struct inode *inode;

	inode = new_inode(sb);
	if (inode)
		ovl_fill_inode(inode, mode);

	return inode;
}

static int ovl_inode_test(struct inode *inode, void *data)
{
	return ovl_inode_real(inode, NULL) == data;
}

static int ovl_inode_set(struct inode *inode, void *data)
{
	inode->i_private = (void *) (((unsigned long) data) | OVL_ISUPPER_MASK);
	return 0;
}

struct inode *ovl_get_inode(struct super_block *sb, struct inode *realinode)

{
	struct inode *inode;

	inode = iget5_locked(sb, (unsigned long) realinode,
			     ovl_inode_test, ovl_inode_set, realinode);
	if (inode && inode->i_state & I_NEW) {
		ovl_fill_inode(inode, realinode->i_mode);
		set_nlink(inode, realinode->i_nlink);
		unlock_new_inode(inode);
M
Miklos Szeredi 已提交
403 404 405 406
	}

	return inode;
}