blob: e909678afa2d06770946b39903e86ef0ee8cb6b4 [file] [log] [blame]
Miklos Szeredie5e55582005-09-09 13:10:28 -07001/*
2 FUSE: Filesystem in Userspace
Miklos Szeredi1729a162008-11-26 12:03:54 +01003 Copyright (C) 2001-2008 Miklos Szeredi <miklos@szeredi.hu>
Miklos Szeredie5e55582005-09-09 13:10:28 -07004
5 This program can be distributed under the terms of the GNU GPL.
6 See the file COPYING.
7*/
8
9#include "fuse_i.h"
10
11#include <linux/pagemap.h>
12#include <linux/file.h>
Miklos Szeredie5e55582005-09-09 13:10:28 -070013#include <linux/sched.h>
14#include <linux/namei.h>
Miklos Szeredi07e77dc2010-12-07 20:16:56 +010015#include <linux/slab.h>
Seth Forshee703c7362016-08-29 08:46:36 -050016#include <linux/xattr.h>
Miklos Szeredi261aaba72018-10-01 10:07:05 +020017#include <linux/iversion.h>
Seth Forshee60bcc882016-08-29 08:46:37 -050018#include <linux/posix_acl.h>
Miklos Szeredie5e55582005-09-09 13:10:28 -070019
Feng Shuo4582a4a2013-01-15 11:23:28 +080020static void fuse_advise_use_readdirplus(struct inode *dir)
21{
22 struct fuse_inode *fi = get_fuse_inode(dir);
23
24 set_bit(FUSE_I_ADVISE_RDPLUS, &fi->state);
25}
26
Miklos Szeredif75fdf22016-10-01 07:32:32 +020027union fuse_dentry {
28 u64 time;
29 struct rcu_head rcu;
30};
31
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070032static inline void fuse_dentry_settime(struct dentry *entry, u64 time)
33{
Miklos Szeredif75fdf22016-10-01 07:32:32 +020034 ((union fuse_dentry *) entry->d_fsdata)->time = time;
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070035}
36
37static inline u64 fuse_dentry_time(struct dentry *entry)
38{
Miklos Szeredif75fdf22016-10-01 07:32:32 +020039 return ((union fuse_dentry *) entry->d_fsdata)->time;
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070040}
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070041
Miklos Szeredi6f9f1182006-01-06 00:19:39 -080042/*
43 * FUSE caches dentries and attributes with separate timeout. The
44 * time in jiffies until the dentry/attributes are valid is stored in
Miklos Szeredif75fdf22016-10-01 07:32:32 +020045 * dentry->d_fsdata and fuse_inode->i_time respectively.
Miklos Szeredi6f9f1182006-01-06 00:19:39 -080046 */
47
48/*
49 * Calculate the time in jiffies until a dentry/attributes are valid
50 */
Miklos Szeredibcb6f6d2016-10-01 07:32:32 +020051static u64 time_to_jiffies(u64 sec, u32 nsec)
Miklos Szeredie5e55582005-09-09 13:10:28 -070052{
Miklos Szeredi685d16d2006-07-30 03:04:08 -070053 if (sec || nsec) {
Miklos Szeredibcb6f6d2016-10-01 07:32:32 +020054 struct timespec64 ts = {
55 sec,
David Sheets21067522017-01-13 15:58:30 +000056 min_t(u32, nsec, NSEC_PER_SEC - 1)
Miklos Szeredibcb6f6d2016-10-01 07:32:32 +020057 };
58
59 return get_jiffies_64() + timespec64_to_jiffies(&ts);
Miklos Szeredi685d16d2006-07-30 03:04:08 -070060 } else
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070061 return 0;
Miklos Szeredie5e55582005-09-09 13:10:28 -070062}
63
Miklos Szeredi6f9f1182006-01-06 00:19:39 -080064/*
65 * Set dentry and possibly attribute timeouts from the lookup/mk*
66 * replies
67 */
Miklos Szeredid123d8e2018-09-28 16:43:23 +020068void fuse_change_entry_timeout(struct dentry *entry, struct fuse_entry_out *o)
Miklos Szeredi0aa7c692006-01-06 00:19:34 -080069{
Miklos Szeredi0a0898c2006-07-30 03:04:10 -070070 fuse_dentry_settime(entry,
71 time_to_jiffies(o->entry_valid, o->entry_valid_nsec));
Miklos Szeredi1fb69e72007-10-18 03:06:58 -070072}
73
74static u64 attr_timeout(struct fuse_attr_out *o)
75{
76 return time_to_jiffies(o->attr_valid, o->attr_valid_nsec);
77}
78
Miklos Szeredid123d8e2018-09-28 16:43:23 +020079u64 entry_attr_timeout(struct fuse_entry_out *o)
Miklos Szeredi1fb69e72007-10-18 03:06:58 -070080{
81 return time_to_jiffies(o->attr_valid, o->attr_valid_nsec);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -080082}
83
Miklos Szeredi2f1e8192018-10-15 15:43:06 +020084static void fuse_invalidate_attr_mask(struct inode *inode, u32 mask)
85{
86 set_mask_bits(&get_fuse_inode(inode)->inval_mask, 0, mask);
87}
88
Miklos Szeredi6f9f1182006-01-06 00:19:39 -080089/*
90 * Mark the attributes as stale, so that at the next call to
91 * ->getattr() they will be fetched from userspace
92 */
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -080093void fuse_invalidate_attr(struct inode *inode)
94{
Miklos Szeredi2f1e8192018-10-15 15:43:06 +020095 fuse_invalidate_attr_mask(inode, STATX_BASIC_STATS);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -080096}
97
Miklos Szeredi261aaba72018-10-01 10:07:05 +020098static void fuse_dir_changed(struct inode *dir)
99{
100 fuse_invalidate_attr(dir);
101 inode_maybe_inc_iversion(dir, false);
102}
103
Andrew Gallagher451418f2013-11-05 03:55:43 -0800104/**
105 * Mark the attributes as stale due to an atime change. Avoid the invalidate if
106 * atime is not used.
107 */
108void fuse_invalidate_atime(struct inode *inode)
109{
110 if (!IS_RDONLY(inode))
Miklos Szeredi2f1e8192018-10-15 15:43:06 +0200111 fuse_invalidate_attr_mask(inode, STATX_ATIME);
Andrew Gallagher451418f2013-11-05 03:55:43 -0800112}
113
Miklos Szeredi6f9f1182006-01-06 00:19:39 -0800114/*
115 * Just mark the entry as stale, so that a next attempt to look it up
116 * will result in a new lookup call to userspace
117 *
118 * This is called when a dentry is about to become negative and the
119 * timeout is unknown (unlink, rmdir, rename and in some cases
120 * lookup)
121 */
Miklos Szeredidbd561d2008-07-25 01:49:00 -0700122void fuse_invalidate_entry_cache(struct dentry *entry)
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800123{
Miklos Szeredi0a0898c2006-07-30 03:04:10 -0700124 fuse_dentry_settime(entry, 0);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800125}
126
Miklos Szeredi6f9f1182006-01-06 00:19:39 -0800127/*
128 * Same as fuse_invalidate_entry_cache(), but also try to remove the
129 * dentry from the hash
130 */
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800131static void fuse_invalidate_entry(struct dentry *entry)
132{
133 d_invalidate(entry);
134 fuse_invalidate_entry_cache(entry);
Miklos Szeredi0aa7c692006-01-06 00:19:34 -0800135}
136
Miklos Szeredi70781872014-12-12 09:49:05 +0100137static void fuse_lookup_init(struct fuse_conn *fc, struct fuse_args *args,
Al Viro13983d02016-07-20 22:34:44 -0400138 u64 nodeid, const struct qstr *name,
Miklos Szeredie5e55582005-09-09 13:10:28 -0700139 struct fuse_entry_out *outarg)
140{
Miklos Szeredi0e9663e2007-10-18 03:07:05 -0700141 memset(outarg, 0, sizeof(struct fuse_entry_out));
Miklos Szeredi70781872014-12-12 09:49:05 +0100142 args->in.h.opcode = FUSE_LOOKUP;
143 args->in.h.nodeid = nodeid;
144 args->in.numargs = 1;
145 args->in.args[0].size = name->len + 1;
146 args->in.args[0].value = name->name;
147 args->out.numargs = 1;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100148 args->out.args[0].size = sizeof(struct fuse_entry_out);
Miklos Szeredi70781872014-12-12 09:49:05 +0100149 args->out.args[0].value = outarg;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700150}
151
Miklos Szeredi5c5c5e52008-04-30 00:54:43 -0700152u64 fuse_get_attr_version(struct fuse_conn *fc)
Miklos Szeredi7dca9fd2007-11-28 16:21:59 -0800153{
154 u64 curr_version;
155
156 /*
157 * The spin lock isn't actually needed on 64bit archs, but we
158 * don't yet care too much about such optimizations.
159 */
160 spin_lock(&fc->lock);
161 curr_version = fc->attr_version;
162 spin_unlock(&fc->lock);
163
164 return curr_version;
165}
166
Miklos Szeredi6f9f1182006-01-06 00:19:39 -0800167/*
168 * Check whether the dentry is still valid
169 *
170 * If the entry validity timeout has expired and the dentry is
171 * positive, try to redo the lookup. If the lookup results in a
172 * different inode, then let the VFS invalidate the dentry and redo
173 * the lookup once more. If the lookup results in the same inode,
174 * then refresh the attributes, timeouts and mark the dentry valid.
175 */
Al Viro0b728e12012-06-10 16:03:43 -0400176static int fuse_dentry_revalidate(struct dentry *entry, unsigned int flags)
Miklos Szeredie5e55582005-09-09 13:10:28 -0700177{
Nick Piggin34286d62011-01-07 17:49:57 +1100178 struct inode *inode;
Miklos Szeredi28420da2013-06-03 14:40:22 +0200179 struct dentry *parent;
180 struct fuse_conn *fc;
Miklos Szeredi6314efe2013-10-01 16:41:22 +0200181 struct fuse_inode *fi;
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200182 int ret;
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800183
David Howells2b0143b2015-03-17 22:25:59 +0000184 inode = d_inode_rcu(entry);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800185 if (inode && is_bad_inode(inode))
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200186 goto invalid;
Anand Avati154210c2014-06-26 20:21:57 -0400187 else if (time_before64(fuse_dentry_time(entry), get_jiffies_64()) ||
188 (flags & LOOKUP_REVAL)) {
Miklos Szeredie5e55582005-09-09 13:10:28 -0700189 struct fuse_entry_out outarg;
Miklos Szeredi70781872014-12-12 09:49:05 +0100190 FUSE_ARGS(args);
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100191 struct fuse_forget_link *forget;
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700192 u64 attr_version;
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800193
Miklos Szeredi50322fe2006-02-28 16:59:03 -0800194 /* For negative dentries, always do a fresh lookup */
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800195 if (!inode)
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200196 goto invalid;
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800197
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200198 ret = -ECHILD;
Al Viro0b728e12012-06-10 16:03:43 -0400199 if (flags & LOOKUP_RCU)
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200200 goto out;
Miklos Szeredie7c0a162011-03-21 13:58:06 +0100201
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800202 fc = get_fuse_conn(inode);
Miklos Szeredie5e55582005-09-09 13:10:28 -0700203
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100204 forget = fuse_alloc_forget();
Miklos Szeredi70781872014-12-12 09:49:05 +0100205 ret = -ENOMEM;
206 if (!forget)
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200207 goto out;
Miklos Szeredi2d510132006-11-25 11:09:20 -0800208
Miklos Szeredi7dca9fd2007-11-28 16:21:59 -0800209 attr_version = fuse_get_attr_version(fc);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700210
Miklos Szeredie956edd2006-10-17 00:10:12 -0700211 parent = dget_parent(entry);
David Howells2b0143b2015-03-17 22:25:59 +0000212 fuse_lookup_init(fc, &args, get_node_id(d_inode(parent)),
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700213 &entry->d_name, &outarg);
Miklos Szeredi70781872014-12-12 09:49:05 +0100214 ret = fuse_simple_request(fc, &args);
Miklos Szeredie956edd2006-10-17 00:10:12 -0700215 dput(parent);
Miklos Szeredi50322fe2006-02-28 16:59:03 -0800216 /* Zero nodeid is same as -ENOENT */
Miklos Szeredi70781872014-12-12 09:49:05 +0100217 if (!ret && !outarg.nodeid)
218 ret = -ENOENT;
219 if (!ret) {
Miklos Szeredi6314efe2013-10-01 16:41:22 +0200220 fi = get_fuse_inode(inode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700221 if (outarg.nodeid != get_node_id(inode)) {
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100222 fuse_queue_forget(fc, forget, outarg.nodeid, 1);
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200223 goto invalid;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700224 }
Miklos Szeredi8da5ff22006-10-17 00:10:08 -0700225 spin_lock(&fc->lock);
Miklos Szeredi1729a162008-11-26 12:03:54 +0100226 fi->nlookup++;
Miklos Szeredi8da5ff22006-10-17 00:10:08 -0700227 spin_unlock(&fc->lock);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700228 }
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100229 kfree(forget);
Miklos Szeredi70781872014-12-12 09:49:05 +0100230 if (ret == -ENOMEM)
231 goto out;
232 if (ret || (outarg.attr.mode ^ inode->i_mode) & S_IFMT)
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200233 goto invalid;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700234
Seth Forshee60bcc882016-08-29 08:46:37 -0500235 forget_all_cached_acls(inode);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700236 fuse_change_attributes(inode, &outarg.attr,
237 entry_attr_timeout(&outarg),
238 attr_version);
239 fuse_change_entry_timeout(entry, &outarg);
Miklos Szeredi28420da2013-06-03 14:40:22 +0200240 } else if (inode) {
Miklos Szeredi6314efe2013-10-01 16:41:22 +0200241 fi = get_fuse_inode(inode);
242 if (flags & LOOKUP_RCU) {
243 if (test_bit(FUSE_I_INIT_RDPLUS, &fi->state))
244 return -ECHILD;
245 } else if (test_and_clear_bit(FUSE_I_INIT_RDPLUS, &fi->state)) {
Miklos Szeredi28420da2013-06-03 14:40:22 +0200246 parent = dget_parent(entry);
David Howells2b0143b2015-03-17 22:25:59 +0000247 fuse_advise_use_readdirplus(d_inode(parent));
Miklos Szeredi28420da2013-06-03 14:40:22 +0200248 dput(parent);
249 }
Miklos Szeredie5e55582005-09-09 13:10:28 -0700250 }
Miklos Szeredie2a6b952013-09-05 11:44:43 +0200251 ret = 1;
252out:
253 return ret;
254
255invalid:
256 ret = 0;
257 goto out;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700258}
259
Miklos Szeredif75fdf22016-10-01 07:32:32 +0200260static int fuse_dentry_init(struct dentry *dentry)
261{
262 dentry->d_fsdata = kzalloc(sizeof(union fuse_dentry), GFP_KERNEL);
263
264 return dentry->d_fsdata ? 0 : -ENOMEM;
265}
266static void fuse_dentry_release(struct dentry *dentry)
267{
268 union fuse_dentry *fd = dentry->d_fsdata;
269
270 kfree_rcu(fd, rcu);
271}
272
Al Viro42695902009-02-20 05:59:13 +0000273const struct dentry_operations fuse_dentry_operations = {
Miklos Szeredie5e55582005-09-09 13:10:28 -0700274 .d_revalidate = fuse_dentry_revalidate,
Miklos Szeredif75fdf22016-10-01 07:32:32 +0200275 .d_init = fuse_dentry_init,
276 .d_release = fuse_dentry_release,
Miklos Szeredie5e55582005-09-09 13:10:28 -0700277};
278
Miklos Szeredi0ce267f2016-10-18 15:36:48 +0200279const struct dentry_operations fuse_root_dentry_operations = {
280 .d_init = fuse_dentry_init,
281 .d_release = fuse_dentry_release,
282};
283
Timo Savolaa5bfffac2007-04-08 16:04:00 -0700284int fuse_valid_type(int m)
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800285{
286 return S_ISREG(m) || S_ISDIR(m) || S_ISLNK(m) || S_ISCHR(m) ||
287 S_ISBLK(m) || S_ISFIFO(m) || S_ISSOCK(m);
288}
289
Al Viro13983d02016-07-20 22:34:44 -0400290int fuse_lookup_name(struct super_block *sb, u64 nodeid, const struct qstr *name,
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700291 struct fuse_entry_out *outarg, struct inode **inode)
292{
293 struct fuse_conn *fc = get_fuse_conn_super(sb);
Miklos Szeredi70781872014-12-12 09:49:05 +0100294 FUSE_ARGS(args);
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100295 struct fuse_forget_link *forget;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700296 u64 attr_version;
297 int err;
298
299 *inode = NULL;
300 err = -ENAMETOOLONG;
301 if (name->len > FUSE_NAME_MAX)
302 goto out;
303
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700304
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100305 forget = fuse_alloc_forget();
306 err = -ENOMEM;
Miklos Szeredi70781872014-12-12 09:49:05 +0100307 if (!forget)
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700308 goto out;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700309
310 attr_version = fuse_get_attr_version(fc);
311
Miklos Szeredi70781872014-12-12 09:49:05 +0100312 fuse_lookup_init(fc, &args, nodeid, name, outarg);
313 err = fuse_simple_request(fc, &args);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700314 /* Zero nodeid is same as -ENOENT, but with valid timeout */
315 if (err || !outarg->nodeid)
316 goto out_put_forget;
317
318 err = -EIO;
319 if (!outarg->nodeid)
320 goto out_put_forget;
321 if (!fuse_valid_type(outarg->attr.mode))
322 goto out_put_forget;
323
324 *inode = fuse_iget(sb, outarg->nodeid, outarg->generation,
325 &outarg->attr, entry_attr_timeout(outarg),
326 attr_version);
327 err = -ENOMEM;
328 if (!*inode) {
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100329 fuse_queue_forget(fc, forget, outarg->nodeid, 1);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700330 goto out;
331 }
332 err = 0;
333
334 out_put_forget:
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100335 kfree(forget);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700336 out:
337 return err;
338}
339
Miklos Szeredi0aa7c692006-01-06 00:19:34 -0800340static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry,
Al Viro00cd8dd2012-06-10 17:13:09 -0400341 unsigned int flags)
Miklos Szeredie5e55582005-09-09 13:10:28 -0700342{
343 int err;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700344 struct fuse_entry_out outarg;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700345 struct inode *inode;
Miklos Szeredi0de62562008-07-25 01:48:59 -0700346 struct dentry *newent;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700347 bool outarg_valid = true;
Miklos Szeredi63576c12018-07-26 16:13:11 +0200348 bool locked;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700349
Miklos Szeredi63576c12018-07-26 16:13:11 +0200350 locked = fuse_lock_inode(dir);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700351 err = fuse_lookup_name(dir->i_sb, get_node_id(dir), &entry->d_name,
352 &outarg, &inode);
Miklos Szeredi63576c12018-07-26 16:13:11 +0200353 fuse_unlock_inode(dir, locked);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700354 if (err == -ENOENT) {
355 outarg_valid = false;
356 err = 0;
Miklos Szeredi2d510132006-11-25 11:09:20 -0800357 }
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700358 if (err)
359 goto out_err;
Miklos Szeredi2d510132006-11-25 11:09:20 -0800360
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700361 err = -EIO;
362 if (inode && get_node_id(inode) == FUSE_ROOT_ID)
363 goto out_iput;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700364
Al Viro41d28bc2014-10-12 22:24:21 -0400365 newent = d_splice_alias(inode, entry);
Miklos Szeredi5835f332013-09-05 11:44:42 +0200366 err = PTR_ERR(newent);
367 if (IS_ERR(newent))
368 goto out_err;
Miklos Szeredid2a85162006-10-17 00:10:11 -0700369
Miklos Szeredi0de62562008-07-25 01:48:59 -0700370 entry = newent ? newent : entry;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700371 if (outarg_valid)
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700372 fuse_change_entry_timeout(entry, &outarg);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800373 else
374 fuse_invalidate_entry_cache(entry);
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700375
Feng Shuo4582a4a2013-01-15 11:23:28 +0800376 fuse_advise_use_readdirplus(dir);
Miklos Szeredi0de62562008-07-25 01:48:59 -0700377 return newent;
Miklos Szeredic180eeb2008-07-25 01:49:01 -0700378
379 out_iput:
380 iput(inode);
381 out_err:
382 return ERR_PTR(err);
Miklos Szeredie5e55582005-09-09 13:10:28 -0700383}
384
Miklos Szeredi6f9f1182006-01-06 00:19:39 -0800385/*
386 * Atomic create+open operation
387 *
388 * If the filesystem doesn't support this, then fall back to separate
389 * 'mknod' + 'open' requests.
390 */
Al Virod9585272012-06-22 12:39:14 +0400391static int fuse_create_open(struct inode *dir, struct dentry *entry,
Al Viro30d90492012-06-22 12:40:19 +0400392 struct file *file, unsigned flags,
Al Virob452a452018-06-08 13:06:28 -0400393 umode_t mode)
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800394{
395 int err;
396 struct inode *inode;
397 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100398 FUSE_ARGS(args);
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100399 struct fuse_forget_link *forget;
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200400 struct fuse_create_in inarg;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800401 struct fuse_open_out outopen;
402 struct fuse_entry_out outentry;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800403 struct fuse_file *ff;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800404
Miklos Szerediaf109bc2012-08-15 13:01:24 +0200405 /* Userspace expects S_IFREG in create mode */
406 BUG_ON((mode & S_IFMT) != S_IFREG);
407
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100408 forget = fuse_alloc_forget();
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200409 err = -ENOMEM;
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100410 if (!forget)
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200411 goto out_err;
Miklos Szeredi51eb01e2006-06-25 05:48:50 -0700412
Miklos Szeredice1d5a42006-04-10 22:54:58 -0700413 err = -ENOMEM;
Tejun Heoacf99432008-11-26 12:03:55 +0100414 ff = fuse_file_alloc(fc);
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800415 if (!ff)
Miklos Szeredi70781872014-12-12 09:49:05 +0100416 goto out_put_forget_req;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800417
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200418 if (!fc->dont_mask)
419 mode &= ~current_umask();
420
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800421 flags &= ~O_NOCTTY;
422 memset(&inarg, 0, sizeof(inarg));
Miklos Szeredi0e9663e2007-10-18 03:07:05 -0700423 memset(&outentry, 0, sizeof(outentry));
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800424 inarg.flags = flags;
425 inarg.mode = mode;
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200426 inarg.umask = current_umask();
Miklos Szeredi70781872014-12-12 09:49:05 +0100427 args.in.h.opcode = FUSE_CREATE;
428 args.in.h.nodeid = get_node_id(dir);
429 args.in.numargs = 2;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100430 args.in.args[0].size = sizeof(inarg);
Miklos Szeredi70781872014-12-12 09:49:05 +0100431 args.in.args[0].value = &inarg;
432 args.in.args[1].size = entry->d_name.len + 1;
433 args.in.args[1].value = entry->d_name.name;
434 args.out.numargs = 2;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100435 args.out.args[0].size = sizeof(outentry);
Miklos Szeredi70781872014-12-12 09:49:05 +0100436 args.out.args[0].value = &outentry;
437 args.out.args[1].size = sizeof(outopen);
438 args.out.args[1].value = &outopen;
439 err = fuse_simple_request(fc, &args);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200440 if (err)
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800441 goto out_free_ff;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800442
443 err = -EIO;
Miklos Szeredi2827d0b22005-11-28 13:44:16 -0800444 if (!S_ISREG(outentry.attr.mode) || invalid_nodeid(outentry.nodeid))
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800445 goto out_free_ff;
446
Miklos Szeredic7b71432009-04-28 16:56:37 +0200447 ff->fh = outopen.fh;
448 ff->nodeid = outentry.nodeid;
449 ff->open_flags = outopen.open_flags;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800450 inode = fuse_iget(dir->i_sb, outentry.nodeid, outentry.generation,
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700451 &outentry.attr, entry_attr_timeout(&outentry), 0);
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800452 if (!inode) {
453 flags &= ~(O_CREAT | O_EXCL | O_TRUNC);
Miklos Szeredi8b0797a2009-04-28 16:56:39 +0200454 fuse_sync_release(ff, flags);
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100455 fuse_queue_forget(fc, forget, outentry.nodeid, 1);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200456 err = -ENOMEM;
457 goto out_err;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800458 }
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100459 kfree(forget);
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800460 d_instantiate(entry, inode);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700461 fuse_change_entry_timeout(entry, &outentry);
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200462 fuse_dir_changed(dir);
Al Virobe12af32018-06-08 11:44:56 -0400463 err = finish_open(file, entry, generic_file_open);
Al Viro30d90492012-06-22 12:40:19 +0400464 if (err) {
Miklos Szeredi8b0797a2009-04-28 16:56:39 +0200465 fuse_sync_release(ff, flags);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200466 } else {
Miklos Szeredi267d8442017-02-22 20:08:25 +0100467 file->private_data = ff;
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200468 fuse_finish_open(inode, file);
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800469 }
Al Virod9585272012-06-22 12:39:14 +0400470 return err;
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800471
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200472out_free_ff:
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800473 fuse_file_free(ff);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200474out_put_forget_req:
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100475 kfree(forget);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200476out_err:
Al Virod9585272012-06-22 12:39:14 +0400477 return err;
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200478}
479
480static int fuse_mknod(struct inode *, struct dentry *, umode_t, dev_t);
Al Virod9585272012-06-22 12:39:14 +0400481static int fuse_atomic_open(struct inode *dir, struct dentry *entry,
Al Viro30d90492012-06-22 12:40:19 +0400482 struct file *file, unsigned flags,
Al Viro44907d72018-06-08 13:32:02 -0400483 umode_t mode)
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200484{
485 int err;
486 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200487 struct dentry *res = NULL;
488
Al Viro00699ad2016-07-05 09:44:53 -0400489 if (d_in_lookup(entry)) {
Al Viro00cd8dd2012-06-10 17:13:09 -0400490 res = fuse_lookup(dir, entry, 0);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200491 if (IS_ERR(res))
Al Virod9585272012-06-22 12:39:14 +0400492 return PTR_ERR(res);
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200493
494 if (res)
495 entry = res;
496 }
497
David Howells2b0143b2015-03-17 22:25:59 +0000498 if (!(flags & O_CREAT) || d_really_is_positive(entry))
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200499 goto no_open;
500
501 /* Only creates */
Al Viro73a09dd2018-06-08 13:22:02 -0400502 file->f_mode |= FMODE_CREATED;
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200503
504 if (fc->no_create)
505 goto mknod;
506
Al Virob452a452018-06-08 13:06:28 -0400507 err = fuse_create_open(dir, entry, file, flags, mode);
Al Virod9585272012-06-22 12:39:14 +0400508 if (err == -ENOSYS) {
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200509 fc->no_create = 1;
510 goto mknod;
511 }
512out_dput:
513 dput(res);
Al Virod9585272012-06-22 12:39:14 +0400514 return err;
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200515
516mknod:
517 err = fuse_mknod(dir, entry, mode, 0);
Al Virod9585272012-06-22 12:39:14 +0400518 if (err)
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200519 goto out_dput;
Miklos Szeredic8ccbe02012-06-05 15:10:22 +0200520no_open:
Al Viroe45198a2012-06-10 06:48:09 -0400521 return finish_no_open(file, res);
Miklos Szeredifd72faa2005-11-07 00:59:51 -0800522}
523
Miklos Szeredi6f9f1182006-01-06 00:19:39 -0800524/*
525 * Code shared between mknod, mkdir, symlink and link
526 */
Miklos Szeredi70781872014-12-12 09:49:05 +0100527static int create_new_entry(struct fuse_conn *fc, struct fuse_args *args,
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700528 struct inode *dir, struct dentry *entry,
Al Viro541af6a2011-07-26 03:17:33 -0400529 umode_t mode)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700530{
531 struct fuse_entry_out outarg;
532 struct inode *inode;
Al Viroc971e6a2018-05-28 18:27:19 -0400533 struct dentry *d;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700534 int err;
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100535 struct fuse_forget_link *forget;
Miklos Szeredi2d510132006-11-25 11:09:20 -0800536
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100537 forget = fuse_alloc_forget();
Miklos Szeredi70781872014-12-12 09:49:05 +0100538 if (!forget)
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100539 return -ENOMEM;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700540
Miklos Szeredi0e9663e2007-10-18 03:07:05 -0700541 memset(&outarg, 0, sizeof(outarg));
Miklos Szeredi70781872014-12-12 09:49:05 +0100542 args->in.h.nodeid = get_node_id(dir);
543 args->out.numargs = 1;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100544 args->out.args[0].size = sizeof(outarg);
Miklos Szeredi70781872014-12-12 09:49:05 +0100545 args->out.args[0].value = &outarg;
546 err = fuse_simple_request(fc, args);
Miklos Szeredi2d510132006-11-25 11:09:20 -0800547 if (err)
548 goto out_put_forget_req;
549
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800550 err = -EIO;
551 if (invalid_nodeid(outarg.nodeid))
Miklos Szeredi2d510132006-11-25 11:09:20 -0800552 goto out_put_forget_req;
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800553
554 if ((outarg.attr.mode ^ mode) & S_IFMT)
Miklos Szeredi2d510132006-11-25 11:09:20 -0800555 goto out_put_forget_req;
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800556
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700557 inode = fuse_iget(dir->i_sb, outarg.nodeid, outarg.generation,
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700558 &outarg.attr, entry_attr_timeout(&outarg), 0);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700559 if (!inode) {
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100560 fuse_queue_forget(fc, forget, outarg.nodeid, 1);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700561 return -ENOMEM;
562 }
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100563 kfree(forget);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700564
Al Viroc971e6a2018-05-28 18:27:19 -0400565 d_drop(entry);
566 d = d_splice_alias(inode, entry);
567 if (IS_ERR(d))
568 return PTR_ERR(d);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700569
Al Viroc971e6a2018-05-28 18:27:19 -0400570 if (d) {
571 fuse_change_entry_timeout(d, &outarg);
572 dput(d);
573 } else {
574 fuse_change_entry_timeout(entry, &outarg);
575 }
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200576 fuse_dir_changed(dir);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700577 return 0;
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800578
Miklos Szeredi2d510132006-11-25 11:09:20 -0800579 out_put_forget_req:
Miklos Szeredi07e77dc2010-12-07 20:16:56 +0100580 kfree(forget);
Miklos Szeredi39ee0592006-01-06 00:19:43 -0800581 return err;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700582}
583
Al Viro1a67aaf2011-07-26 01:52:52 -0400584static int fuse_mknod(struct inode *dir, struct dentry *entry, umode_t mode,
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700585 dev_t rdev)
586{
587 struct fuse_mknod_in inarg;
588 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100589 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700590
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200591 if (!fc->dont_mask)
592 mode &= ~current_umask();
593
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700594 memset(&inarg, 0, sizeof(inarg));
595 inarg.mode = mode;
596 inarg.rdev = new_encode_dev(rdev);
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200597 inarg.umask = current_umask();
Miklos Szeredi70781872014-12-12 09:49:05 +0100598 args.in.h.opcode = FUSE_MKNOD;
599 args.in.numargs = 2;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100600 args.in.args[0].size = sizeof(inarg);
Miklos Szeredi70781872014-12-12 09:49:05 +0100601 args.in.args[0].value = &inarg;
602 args.in.args[1].size = entry->d_name.len + 1;
603 args.in.args[1].value = entry->d_name.name;
604 return create_new_entry(fc, &args, dir, entry, mode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700605}
606
Al Viro4acdaf22011-07-26 01:42:34 -0400607static int fuse_create(struct inode *dir, struct dentry *entry, umode_t mode,
Al Viroebfc3b42012-06-10 18:05:36 -0400608 bool excl)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700609{
610 return fuse_mknod(dir, entry, mode, 0);
611}
612
Al Viro18bb1db2011-07-26 01:41:39 -0400613static int fuse_mkdir(struct inode *dir, struct dentry *entry, umode_t mode)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700614{
615 struct fuse_mkdir_in inarg;
616 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100617 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700618
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200619 if (!fc->dont_mask)
620 mode &= ~current_umask();
621
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700622 memset(&inarg, 0, sizeof(inarg));
623 inarg.mode = mode;
Miklos Szeredie0a43dd2009-06-30 20:12:23 +0200624 inarg.umask = current_umask();
Miklos Szeredi70781872014-12-12 09:49:05 +0100625 args.in.h.opcode = FUSE_MKDIR;
626 args.in.numargs = 2;
627 args.in.args[0].size = sizeof(inarg);
628 args.in.args[0].value = &inarg;
629 args.in.args[1].size = entry->d_name.len + 1;
630 args.in.args[1].value = entry->d_name.name;
631 return create_new_entry(fc, &args, dir, entry, S_IFDIR);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700632}
633
634static int fuse_symlink(struct inode *dir, struct dentry *entry,
635 const char *link)
636{
637 struct fuse_conn *fc = get_fuse_conn(dir);
638 unsigned len = strlen(link) + 1;
Miklos Szeredi70781872014-12-12 09:49:05 +0100639 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700640
Miklos Szeredi70781872014-12-12 09:49:05 +0100641 args.in.h.opcode = FUSE_SYMLINK;
642 args.in.numargs = 2;
643 args.in.args[0].size = entry->d_name.len + 1;
644 args.in.args[0].value = entry->d_name.name;
645 args.in.args[1].size = len;
646 args.in.args[1].value = link;
647 return create_new_entry(fc, &args, dir, entry, S_IFLNK);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700648}
649
Seth Forshee703c7362016-08-29 08:46:36 -0500650void fuse_update_ctime(struct inode *inode)
Maxim Patlasov31f32672014-04-28 14:19:24 +0200651{
652 if (!IS_NOCMTIME(inode)) {
Deepa Dinamanic2050a42016-09-14 07:48:06 -0700653 inode->i_ctime = current_time(inode);
Maxim Patlasov31f32672014-04-28 14:19:24 +0200654 mark_inode_dirty_sync(inode);
655 }
656}
657
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700658static int fuse_unlink(struct inode *dir, struct dentry *entry)
659{
660 int err;
661 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100662 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700663
Miklos Szeredi70781872014-12-12 09:49:05 +0100664 args.in.h.opcode = FUSE_UNLINK;
665 args.in.h.nodeid = get_node_id(dir);
666 args.in.numargs = 1;
667 args.in.args[0].size = entry->d_name.len + 1;
668 args.in.args[0].value = entry->d_name.name;
669 err = fuse_simple_request(fc, &args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700670 if (!err) {
David Howells2b0143b2015-03-17 22:25:59 +0000671 struct inode *inode = d_inode(entry);
Miklos Szerediac45d612012-03-05 15:48:11 +0100672 struct fuse_inode *fi = get_fuse_inode(inode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700673
Miklos Szerediac45d612012-03-05 15:48:11 +0100674 spin_lock(&fc->lock);
675 fi->attr_version = ++fc->attr_version;
Miklos Szeredidfca7ce2013-02-04 15:57:42 +0100676 /*
677 * If i_nlink == 0 then unlink doesn't make sense, yet this can
678 * happen if userspace filesystem is careless. It would be
679 * difficult to enforce correct nlink usage so just ignore this
680 * condition here
681 */
682 if (inode->i_nlink > 0)
683 drop_nlink(inode);
Miklos Szerediac45d612012-03-05 15:48:11 +0100684 spin_unlock(&fc->lock);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700685 fuse_invalidate_attr(inode);
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200686 fuse_dir_changed(dir);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800687 fuse_invalidate_entry_cache(entry);
Maxim Patlasov31f32672014-04-28 14:19:24 +0200688 fuse_update_ctime(inode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700689 } else if (err == -EINTR)
690 fuse_invalidate_entry(entry);
691 return err;
692}
693
694static int fuse_rmdir(struct inode *dir, struct dentry *entry)
695{
696 int err;
697 struct fuse_conn *fc = get_fuse_conn(dir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100698 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700699
Miklos Szeredi70781872014-12-12 09:49:05 +0100700 args.in.h.opcode = FUSE_RMDIR;
701 args.in.h.nodeid = get_node_id(dir);
702 args.in.numargs = 1;
703 args.in.args[0].size = entry->d_name.len + 1;
704 args.in.args[0].value = entry->d_name.name;
705 err = fuse_simple_request(fc, &args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700706 if (!err) {
David Howells2b0143b2015-03-17 22:25:59 +0000707 clear_nlink(d_inode(entry));
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200708 fuse_dir_changed(dir);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800709 fuse_invalidate_entry_cache(entry);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700710 } else if (err == -EINTR)
711 fuse_invalidate_entry(entry);
712 return err;
713}
714
Miklos Szeredi1560c972014-04-28 16:43:44 +0200715static int fuse_rename_common(struct inode *olddir, struct dentry *oldent,
716 struct inode *newdir, struct dentry *newent,
717 unsigned int flags, int opcode, size_t argsize)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700718{
719 int err;
Miklos Szeredi1560c972014-04-28 16:43:44 +0200720 struct fuse_rename2_in inarg;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700721 struct fuse_conn *fc = get_fuse_conn(olddir);
Miklos Szeredi70781872014-12-12 09:49:05 +0100722 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700723
Miklos Szeredi1560c972014-04-28 16:43:44 +0200724 memset(&inarg, 0, argsize);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700725 inarg.newdir = get_node_id(newdir);
Miklos Szeredi1560c972014-04-28 16:43:44 +0200726 inarg.flags = flags;
Miklos Szeredi70781872014-12-12 09:49:05 +0100727 args.in.h.opcode = opcode;
728 args.in.h.nodeid = get_node_id(olddir);
729 args.in.numargs = 3;
730 args.in.args[0].size = argsize;
731 args.in.args[0].value = &inarg;
732 args.in.args[1].size = oldent->d_name.len + 1;
733 args.in.args[1].value = oldent->d_name.name;
734 args.in.args[2].size = newent->d_name.len + 1;
735 args.in.args[2].value = newent->d_name.name;
736 err = fuse_simple_request(fc, &args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700737 if (!err) {
Miklos Szeredi08b63302007-11-28 16:22:03 -0800738 /* ctime changes */
David Howells2b0143b2015-03-17 22:25:59 +0000739 fuse_invalidate_attr(d_inode(oldent));
740 fuse_update_ctime(d_inode(oldent));
Miklos Szeredi08b63302007-11-28 16:22:03 -0800741
Miklos Szeredi1560c972014-04-28 16:43:44 +0200742 if (flags & RENAME_EXCHANGE) {
David Howells2b0143b2015-03-17 22:25:59 +0000743 fuse_invalidate_attr(d_inode(newent));
744 fuse_update_ctime(d_inode(newent));
Miklos Szeredi1560c972014-04-28 16:43:44 +0200745 }
746
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200747 fuse_dir_changed(olddir);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700748 if (olddir != newdir)
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200749 fuse_dir_changed(newdir);
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800750
751 /* newent will end up negative */
David Howells2b0143b2015-03-17 22:25:59 +0000752 if (!(flags & RENAME_EXCHANGE) && d_really_is_positive(newent)) {
753 fuse_invalidate_attr(d_inode(newent));
Miklos Szeredi8cbdf1e2006-01-06 00:19:38 -0800754 fuse_invalidate_entry_cache(newent);
David Howells2b0143b2015-03-17 22:25:59 +0000755 fuse_update_ctime(d_inode(newent));
Miklos Szeredi5219f342009-11-04 10:24:52 +0100756 }
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700757 } else if (err == -EINTR) {
758 /* If request was interrupted, DEITY only knows if the
759 rename actually took place. If the invalidation
760 fails (e.g. some process has CWD under the renamed
761 directory), then there can be inconsistency between
762 the dcache and the real filesystem. Tough luck. */
763 fuse_invalidate_entry(oldent);
David Howells2b0143b2015-03-17 22:25:59 +0000764 if (d_really_is_positive(newent))
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700765 fuse_invalidate_entry(newent);
766 }
767
768 return err;
769}
770
Miklos Szeredi1560c972014-04-28 16:43:44 +0200771static int fuse_rename2(struct inode *olddir, struct dentry *oldent,
772 struct inode *newdir, struct dentry *newent,
773 unsigned int flags)
774{
775 struct fuse_conn *fc = get_fuse_conn(olddir);
776 int err;
777
778 if (flags & ~(RENAME_NOREPLACE | RENAME_EXCHANGE))
779 return -EINVAL;
780
Miklos Szeredi4237ba42014-07-10 10:50:19 +0200781 if (flags) {
782 if (fc->no_rename2 || fc->minor < 23)
783 return -EINVAL;
Miklos Szeredi1560c972014-04-28 16:43:44 +0200784
Miklos Szeredi4237ba42014-07-10 10:50:19 +0200785 err = fuse_rename_common(olddir, oldent, newdir, newent, flags,
786 FUSE_RENAME2,
787 sizeof(struct fuse_rename2_in));
788 if (err == -ENOSYS) {
789 fc->no_rename2 = 1;
790 err = -EINVAL;
791 }
792 } else {
793 err = fuse_rename_common(olddir, oldent, newdir, newent, 0,
794 FUSE_RENAME,
795 sizeof(struct fuse_rename_in));
Miklos Szeredi1560c972014-04-28 16:43:44 +0200796 }
Miklos Szeredi1560c972014-04-28 16:43:44 +0200797
Miklos Szeredi4237ba42014-07-10 10:50:19 +0200798 return err;
799}
800
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700801static int fuse_link(struct dentry *entry, struct inode *newdir,
802 struct dentry *newent)
803{
804 int err;
805 struct fuse_link_in inarg;
David Howells2b0143b2015-03-17 22:25:59 +0000806 struct inode *inode = d_inode(entry);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700807 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +0100808 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700809
810 memset(&inarg, 0, sizeof(inarg));
811 inarg.oldnodeid = get_node_id(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +0100812 args.in.h.opcode = FUSE_LINK;
813 args.in.numargs = 2;
814 args.in.args[0].size = sizeof(inarg);
815 args.in.args[0].value = &inarg;
816 args.in.args[1].size = newent->d_name.len + 1;
817 args.in.args[1].value = newent->d_name.name;
818 err = create_new_entry(fc, &args, newdir, newent, inode->i_mode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700819 /* Contrary to "normal" filesystems it can happen that link
820 makes two "logical" inodes point to the same "physical"
821 inode. We invalidate the attributes of the old one, so it
822 will reflect changes in the backing inode (link count,
823 etc.)
824 */
Miklos Szerediac45d612012-03-05 15:48:11 +0100825 if (!err) {
826 struct fuse_inode *fi = get_fuse_inode(inode);
827
828 spin_lock(&fc->lock);
829 fi->attr_version = ++fc->attr_version;
830 inc_nlink(inode);
831 spin_unlock(&fc->lock);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700832 fuse_invalidate_attr(inode);
Maxim Patlasov31f32672014-04-28 14:19:24 +0200833 fuse_update_ctime(inode);
Miklos Szerediac45d612012-03-05 15:48:11 +0100834 } else if (err == -EINTR) {
835 fuse_invalidate_attr(inode);
836 }
Miklos Szeredi9e6268d2005-09-09 13:10:29 -0700837 return err;
838}
839
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700840static void fuse_fillattr(struct inode *inode, struct fuse_attr *attr,
841 struct kstat *stat)
842{
Miklos Szeredi203627b2012-05-10 19:49:38 +0400843 unsigned int blkbits;
Pavel Emelyanov83732002013-10-10 17:10:46 +0400844 struct fuse_conn *fc = get_fuse_conn(inode);
845
846 /* see the comment in fuse_change_attributes() */
Maxim Patlasovb0aa7602013-12-26 19:51:11 +0400847 if (fc->writeback_cache && S_ISREG(inode->i_mode)) {
Pavel Emelyanov83732002013-10-10 17:10:46 +0400848 attr->size = i_size_read(inode);
Maxim Patlasovb0aa7602013-12-26 19:51:11 +0400849 attr->mtime = inode->i_mtime.tv_sec;
850 attr->mtimensec = inode->i_mtime.tv_nsec;
Maxim Patlasov31f32672014-04-28 14:19:24 +0200851 attr->ctime = inode->i_ctime.tv_sec;
852 attr->ctimensec = inode->i_ctime.tv_nsec;
Maxim Patlasovb0aa7602013-12-26 19:51:11 +0400853 }
Miklos Szeredi203627b2012-05-10 19:49:38 +0400854
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700855 stat->dev = inode->i_sb->s_dev;
856 stat->ino = attr->ino;
857 stat->mode = (inode->i_mode & S_IFMT) | (attr->mode & 07777);
858 stat->nlink = attr->nlink;
Eric W. Biederman8cb08322018-02-21 11:18:07 -0600859 stat->uid = make_kuid(fc->user_ns, attr->uid);
860 stat->gid = make_kgid(fc->user_ns, attr->gid);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700861 stat->rdev = inode->i_rdev;
862 stat->atime.tv_sec = attr->atime;
863 stat->atime.tv_nsec = attr->atimensec;
864 stat->mtime.tv_sec = attr->mtime;
865 stat->mtime.tv_nsec = attr->mtimensec;
866 stat->ctime.tv_sec = attr->ctime;
867 stat->ctime.tv_nsec = attr->ctimensec;
868 stat->size = attr->size;
869 stat->blocks = attr->blocks;
Miklos Szeredi203627b2012-05-10 19:49:38 +0400870
871 if (attr->blksize != 0)
872 blkbits = ilog2(attr->blksize);
873 else
874 blkbits = inode->i_sb->s_blocksize_bits;
875
876 stat->blksize = 1 << blkbits;
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700877}
878
Miklos Szeredic79e3222007-10-18 03:06:59 -0700879static int fuse_do_getattr(struct inode *inode, struct kstat *stat,
880 struct file *file)
Miklos Szeredie5e55582005-09-09 13:10:28 -0700881{
882 int err;
Miklos Szeredic79e3222007-10-18 03:06:59 -0700883 struct fuse_getattr_in inarg;
884 struct fuse_attr_out outarg;
Miklos Szeredie5e55582005-09-09 13:10:28 -0700885 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +0100886 FUSE_ARGS(args);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700887 u64 attr_version;
888
Miklos Szeredi7dca9fd2007-11-28 16:21:59 -0800889 attr_version = fuse_get_attr_version(fc);
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700890
Miklos Szeredic79e3222007-10-18 03:06:59 -0700891 memset(&inarg, 0, sizeof(inarg));
Miklos Szeredi0e9663e2007-10-18 03:07:05 -0700892 memset(&outarg, 0, sizeof(outarg));
Miklos Szeredic79e3222007-10-18 03:06:59 -0700893 /* Directories have separate file-handle space */
894 if (file && S_ISREG(inode->i_mode)) {
895 struct fuse_file *ff = file->private_data;
896
897 inarg.getattr_flags |= FUSE_GETATTR_FH;
898 inarg.fh = ff->fh;
899 }
Miklos Szeredi70781872014-12-12 09:49:05 +0100900 args.in.h.opcode = FUSE_GETATTR;
901 args.in.h.nodeid = get_node_id(inode);
902 args.in.numargs = 1;
903 args.in.args[0].size = sizeof(inarg);
904 args.in.args[0].value = &inarg;
905 args.out.numargs = 1;
Miklos Szeredi21f62172015-01-06 10:45:35 +0100906 args.out.args[0].size = sizeof(outarg);
Miklos Szeredi70781872014-12-12 09:49:05 +0100907 args.out.args[0].value = &outarg;
908 err = fuse_simple_request(fc, &args);
Miklos Szeredie5e55582005-09-09 13:10:28 -0700909 if (!err) {
Miklos Szeredic79e3222007-10-18 03:06:59 -0700910 if ((inode->i_mode ^ outarg.attr.mode) & S_IFMT) {
Miklos Szeredie5e55582005-09-09 13:10:28 -0700911 make_bad_inode(inode);
912 err = -EIO;
913 } else {
Miklos Szeredic79e3222007-10-18 03:06:59 -0700914 fuse_change_attributes(inode, &outarg.attr,
915 attr_timeout(&outarg),
Miklos Szeredi1fb69e72007-10-18 03:06:58 -0700916 attr_version);
917 if (stat)
Miklos Szeredic79e3222007-10-18 03:06:59 -0700918 fuse_fillattr(inode, &outarg.attr, stat);
Miklos Szeredie5e55582005-09-09 13:10:28 -0700919 }
920 }
921 return err;
922}
923
Miklos Szeredi5b97eea2017-09-12 16:57:54 +0200924static int fuse_update_get_attr(struct inode *inode, struct file *file,
Miklos Szeredi2f1e8192018-10-15 15:43:06 +0200925 struct kstat *stat, u32 request_mask,
926 unsigned int flags)
Miklos Szeredibcb4be82007-11-28 16:21:59 -0800927{
928 struct fuse_inode *fi = get_fuse_inode(inode);
Miklos Szeredi5b97eea2017-09-12 16:57:54 +0200929 int err = 0;
Miklos Szeredibf5c1892018-03-20 17:11:44 +0100930 bool sync;
Miklos Szeredibcb4be82007-11-28 16:21:59 -0800931
Miklos Szeredibf5c1892018-03-20 17:11:44 +0100932 if (flags & AT_STATX_FORCE_SYNC)
933 sync = true;
934 else if (flags & AT_STATX_DONT_SYNC)
935 sync = false;
Miklos Szeredi2f1e8192018-10-15 15:43:06 +0200936 else if (request_mask & READ_ONCE(fi->inval_mask))
937 sync = true;
Miklos Szeredibf5c1892018-03-20 17:11:44 +0100938 else
939 sync = time_before64(fi->i_time, get_jiffies_64());
940
941 if (sync) {
Seth Forshee60bcc882016-08-29 08:46:37 -0500942 forget_all_cached_acls(inode);
Miklos Szeredibcb4be82007-11-28 16:21:59 -0800943 err = fuse_do_getattr(inode, stat, file);
Miklos Szeredi5b97eea2017-09-12 16:57:54 +0200944 } else if (stat) {
945 generic_fillattr(inode, stat);
946 stat->mode = fi->orig_i_mode;
947 stat->ino = fi->orig_ino;
Miklos Szeredibcb4be82007-11-28 16:21:59 -0800948 }
949
Miklos Szeredibcb4be82007-11-28 16:21:59 -0800950 return err;
951}
952
Miklos Szeredi5b97eea2017-09-12 16:57:54 +0200953int fuse_update_attributes(struct inode *inode, struct file *file)
954{
Miklos Szeredi802dc042018-10-15 15:43:06 +0200955 /* Do *not* need to get atime for internal purposes */
956 return fuse_update_get_attr(inode, file, NULL,
957 STATX_BASIC_STATS & ~STATX_ATIME, 0);
Miklos Szeredi5b97eea2017-09-12 16:57:54 +0200958}
959
John Muir3b463ae2009-05-31 11:13:57 -0400960int fuse_reverse_inval_entry(struct super_block *sb, u64 parent_nodeid,
John Muir451d0f52011-12-06 21:50:06 +0100961 u64 child_nodeid, struct qstr *name)
John Muir3b463ae2009-05-31 11:13:57 -0400962{
963 int err = -ENOTDIR;
964 struct inode *parent;
965 struct dentry *dir;
966 struct dentry *entry;
967
968 parent = ilookup5(sb, parent_nodeid, fuse_inode_eq, &parent_nodeid);
969 if (!parent)
970 return -ENOENT;
971
Al Viro59551022016-01-22 15:40:57 -0500972 inode_lock(parent);
John Muir3b463ae2009-05-31 11:13:57 -0400973 if (!S_ISDIR(parent->i_mode))
974 goto unlock;
975
976 err = -ENOENT;
977 dir = d_find_alias(parent);
978 if (!dir)
979 goto unlock;
980
Linus Torvalds8387ff22016-06-10 07:51:30 -0700981 name->hash = full_name_hash(dir, name->name, name->len);
John Muir3b463ae2009-05-31 11:13:57 -0400982 entry = d_lookup(dir, name);
983 dput(dir);
984 if (!entry)
985 goto unlock;
986
Miklos Szeredi261aaba72018-10-01 10:07:05 +0200987 fuse_dir_changed(parent);
John Muir3b463ae2009-05-31 11:13:57 -0400988 fuse_invalidate_entry(entry);
John Muir451d0f52011-12-06 21:50:06 +0100989
David Howells2b0143b2015-03-17 22:25:59 +0000990 if (child_nodeid != 0 && d_really_is_positive(entry)) {
Al Viro59551022016-01-22 15:40:57 -0500991 inode_lock(d_inode(entry));
David Howells2b0143b2015-03-17 22:25:59 +0000992 if (get_node_id(d_inode(entry)) != child_nodeid) {
John Muir451d0f52011-12-06 21:50:06 +0100993 err = -ENOENT;
994 goto badentry;
995 }
996 if (d_mountpoint(entry)) {
997 err = -EBUSY;
998 goto badentry;
999 }
David Howellse36cb0b2015-01-29 12:02:35 +00001000 if (d_is_dir(entry)) {
John Muir451d0f52011-12-06 21:50:06 +01001001 shrink_dcache_parent(entry);
1002 if (!simple_empty(entry)) {
1003 err = -ENOTEMPTY;
1004 goto badentry;
1005 }
David Howells2b0143b2015-03-17 22:25:59 +00001006 d_inode(entry)->i_flags |= S_DEAD;
John Muir451d0f52011-12-06 21:50:06 +01001007 }
1008 dont_mount(entry);
David Howells2b0143b2015-03-17 22:25:59 +00001009 clear_nlink(d_inode(entry));
John Muir451d0f52011-12-06 21:50:06 +01001010 err = 0;
1011 badentry:
Al Viro59551022016-01-22 15:40:57 -05001012 inode_unlock(d_inode(entry));
John Muir451d0f52011-12-06 21:50:06 +01001013 if (!err)
1014 d_delete(entry);
1015 } else {
1016 err = 0;
1017 }
John Muir3b463ae2009-05-31 11:13:57 -04001018 dput(entry);
John Muir3b463ae2009-05-31 11:13:57 -04001019
1020 unlock:
Al Viro59551022016-01-22 15:40:57 -05001021 inode_unlock(parent);
John Muir3b463ae2009-05-31 11:13:57 -04001022 iput(parent);
1023 return err;
1024}
1025
Miklos Szeredi87729a52005-09-09 13:10:34 -07001026/*
1027 * Calling into a user-controlled filesystem gives the filesystem
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001028 * daemon ptrace-like capabilities over the current process. This
Miklos Szeredi87729a52005-09-09 13:10:34 -07001029 * means, that the filesystem daemon is able to record the exact
1030 * filesystem operations performed, and can also control the behavior
1031 * of the requester process in otherwise impossible ways. For example
1032 * it can delay the operation for arbitrary length of time allowing
1033 * DoS against the requester.
1034 *
1035 * For this reason only those processes can call into the filesystem,
1036 * for which the owner of the mount has ptrace privilege. This
1037 * excludes processes started by other users, suid or sgid processes.
1038 */
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001039int fuse_allow_current_process(struct fuse_conn *fc)
Miklos Szeredi87729a52005-09-09 13:10:34 -07001040{
David Howellsc69e8d92008-11-14 10:39:19 +11001041 const struct cred *cred;
David Howellsc69e8d92008-11-14 10:39:19 +11001042
Miklos Szeredi29433a22016-10-01 07:32:32 +02001043 if (fc->allow_other)
Seth Forshee73f03c22017-12-22 15:32:33 +01001044 return current_in_userns(fc->user_ns);
Miklos Szeredi87729a52005-09-09 13:10:34 -07001045
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001046 cred = current_cred();
Eric W. Biederman499dcf22012-02-07 16:26:03 -08001047 if (uid_eq(cred->euid, fc->user_id) &&
1048 uid_eq(cred->suid, fc->user_id) &&
1049 uid_eq(cred->uid, fc->user_id) &&
1050 gid_eq(cred->egid, fc->group_id) &&
1051 gid_eq(cred->sgid, fc->group_id) &&
1052 gid_eq(cred->gid, fc->group_id))
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001053 return 1;
Miklos Szeredi87729a52005-09-09 13:10:34 -07001054
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001055 return 0;
Miklos Szeredi87729a52005-09-09 13:10:34 -07001056}
1057
Miklos Szeredi31d40d72005-11-07 00:59:50 -08001058static int fuse_access(struct inode *inode, int mask)
1059{
1060 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +01001061 FUSE_ARGS(args);
Miklos Szeredi31d40d72005-11-07 00:59:50 -08001062 struct fuse_access_in inarg;
1063 int err;
1064
Miklos Szeredi698fa1d2013-10-01 16:41:23 +02001065 BUG_ON(mask & MAY_NOT_BLOCK);
1066
Miklos Szeredi31d40d72005-11-07 00:59:50 -08001067 if (fc->no_access)
1068 return 0;
1069
Miklos Szeredi31d40d72005-11-07 00:59:50 -08001070 memset(&inarg, 0, sizeof(inarg));
Al Viroe6305c42008-07-15 21:03:57 -04001071 inarg.mask = mask & (MAY_READ | MAY_WRITE | MAY_EXEC);
Miklos Szeredi70781872014-12-12 09:49:05 +01001072 args.in.h.opcode = FUSE_ACCESS;
1073 args.in.h.nodeid = get_node_id(inode);
1074 args.in.numargs = 1;
1075 args.in.args[0].size = sizeof(inarg);
1076 args.in.args[0].value = &inarg;
1077 err = fuse_simple_request(fc, &args);
Miklos Szeredi31d40d72005-11-07 00:59:50 -08001078 if (err == -ENOSYS) {
1079 fc->no_access = 1;
1080 err = 0;
1081 }
1082 return err;
1083}
1084
Al Viro10556cb2011-06-20 19:28:19 -04001085static int fuse_perm_getattr(struct inode *inode, int mask)
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001086{
Al Viro10556cb2011-06-20 19:28:19 -04001087 if (mask & MAY_NOT_BLOCK)
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001088 return -ECHILD;
1089
Seth Forshee60bcc882016-08-29 08:46:37 -05001090 forget_all_cached_acls(inode);
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001091 return fuse_do_getattr(inode, NULL, NULL);
1092}
1093
Miklos Szeredi6f9f1182006-01-06 00:19:39 -08001094/*
1095 * Check permission. The two basic access models of FUSE are:
1096 *
1097 * 1) Local access checking ('default_permissions' mount option) based
1098 * on file mode. This is the plain old disk filesystem permission
1099 * modell.
1100 *
1101 * 2) "Remote" access checking, where server is responsible for
1102 * checking permission in each inode operation. An exception to this
1103 * is if ->permission() was invoked from sys_access() in which case an
1104 * access request is sent. Execute permission is still checked
1105 * locally based on file mode.
1106 */
Al Viro10556cb2011-06-20 19:28:19 -04001107static int fuse_permission(struct inode *inode, int mask)
Miklos Szeredie5e55582005-09-09 13:10:28 -07001108{
1109 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi244f6382007-10-16 23:31:02 -07001110 bool refreshed = false;
1111 int err = 0;
Miklos Szeredie5e55582005-09-09 13:10:28 -07001112
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001113 if (!fuse_allow_current_process(fc))
Miklos Szeredie5e55582005-09-09 13:10:28 -07001114 return -EACCES;
Miklos Szeredi244f6382007-10-16 23:31:02 -07001115
1116 /*
Miklos Szeredie8e96152007-10-16 23:31:06 -07001117 * If attributes are needed, refresh them before proceeding
Miklos Szeredi244f6382007-10-16 23:31:02 -07001118 */
Miklos Szeredi29433a22016-10-01 07:32:32 +02001119 if (fc->default_permissions ||
Miklos Szeredie8e96152007-10-16 23:31:06 -07001120 ((mask & MAY_EXEC) && S_ISREG(inode->i_mode))) {
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001121 struct fuse_inode *fi = get_fuse_inode(inode);
Miklos Szeredid233c7d2018-12-03 10:14:43 +01001122 u32 perm_mask = STATX_MODE | STATX_UID | STATX_GID;
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001123
Miklos Szeredid233c7d2018-12-03 10:14:43 +01001124 if (perm_mask & READ_ONCE(fi->inval_mask) ||
1125 time_before64(fi->i_time, get_jiffies_64())) {
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001126 refreshed = true;
1127
Al Viro10556cb2011-06-20 19:28:19 -04001128 err = fuse_perm_getattr(inode, mask);
Miklos Szeredi19690dd2011-03-21 13:58:06 +01001129 if (err)
1130 return err;
1131 }
Miklos Szeredi244f6382007-10-16 23:31:02 -07001132 }
1133
Miklos Szeredi29433a22016-10-01 07:32:32 +02001134 if (fc->default_permissions) {
Al Viro2830ba72011-06-20 19:16:29 -04001135 err = generic_permission(inode, mask);
Miklos Szeredi1e9a4ed2005-09-09 13:10:31 -07001136
1137 /* If permission is denied, try to refresh file
1138 attributes. This is also needed, because the root
1139 node will at first have no permissions */
Miklos Szeredi244f6382007-10-16 23:31:02 -07001140 if (err == -EACCES && !refreshed) {
Al Viro10556cb2011-06-20 19:28:19 -04001141 err = fuse_perm_getattr(inode, mask);
Miklos Szeredi1e9a4ed2005-09-09 13:10:31 -07001142 if (!err)
Al Viro2830ba72011-06-20 19:16:29 -04001143 err = generic_permission(inode, mask);
Miklos Szeredi1e9a4ed2005-09-09 13:10:31 -07001144 }
1145
Miklos Szeredi6f9f1182006-01-06 00:19:39 -08001146 /* Note: the opposite of the above test does not
1147 exist. So if permissions are revoked this won't be
1148 noticed immediately, only after the attribute
1149 timeout has expired */
Eric Paris9cfcac82010-07-23 11:43:51 -04001150 } else if (mask & (MAY_ACCESS | MAY_CHDIR)) {
Miklos Szeredie8e96152007-10-16 23:31:06 -07001151 err = fuse_access(inode, mask);
1152 } else if ((mask & MAY_EXEC) && S_ISREG(inode->i_mode)) {
1153 if (!(inode->i_mode & S_IXUGO)) {
1154 if (refreshed)
1155 return -EACCES;
Miklos Szeredi1e9a4ed2005-09-09 13:10:31 -07001156
Al Viro10556cb2011-06-20 19:28:19 -04001157 err = fuse_perm_getattr(inode, mask);
Miklos Szeredie8e96152007-10-16 23:31:06 -07001158 if (!err && !(inode->i_mode & S_IXUGO))
1159 return -EACCES;
1160 }
Miklos Szeredie5e55582005-09-09 13:10:28 -07001161 }
Miklos Szeredi244f6382007-10-16 23:31:02 -07001162 return err;
Miklos Szeredie5e55582005-09-09 13:10:28 -07001163}
1164
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001165static int fuse_readlink_page(struct inode *inode, struct page *page)
Miklos Szeredie5e55582005-09-09 13:10:28 -07001166{
Miklos Szeredie5e55582005-09-09 13:10:28 -07001167 struct fuse_conn *fc = get_fuse_conn(inode);
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001168 struct fuse_req *req;
1169 int err;
Miklos Szeredie5e55582005-09-09 13:10:28 -07001170
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001171 req = fuse_get_req(fc, 1);
1172 if (IS_ERR(req))
1173 return PTR_ERR(req);
Al Viro6b255392015-11-17 10:20:54 -05001174
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001175 req->out.page_zeroing = 1;
1176 req->out.argpages = 1;
1177 req->num_pages = 1;
1178 req->pages[0] = page;
1179 req->page_descs[0].length = PAGE_SIZE - 1;
1180 req->in.h.opcode = FUSE_READLINK;
1181 req->in.h.nodeid = get_node_id(inode);
1182 req->out.argvar = 1;
1183 req->out.numargs = 1;
1184 req->out.args[0].size = PAGE_SIZE - 1;
1185 fuse_request_send(fc, req);
1186 err = req->out.h.error;
Miklos Szeredi70781872014-12-12 09:49:05 +01001187
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001188 if (!err) {
1189 char *link = page_address(page);
1190 size_t len = req->out.args[0].size;
1191
1192 BUG_ON(len >= PAGE_SIZE);
1193 link[len] = '\0';
Miklos Szeredi70781872014-12-12 09:49:05 +01001194 }
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001195
1196 fuse_put_request(fc, req);
Andrew Gallagher451418f2013-11-05 03:55:43 -08001197 fuse_invalidate_atime(inode);
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001198
1199 return err;
1200}
1201
1202static const char *fuse_get_link(struct dentry *dentry, struct inode *inode,
1203 struct delayed_call *callback)
1204{
1205 struct fuse_conn *fc = get_fuse_conn(inode);
1206 struct page *page;
1207 int err;
1208
1209 err = -EIO;
1210 if (is_bad_inode(inode))
1211 goto out_err;
1212
1213 if (fc->cache_symlinks)
1214 return page_get_link(dentry, inode, callback);
1215
1216 err = -ECHILD;
1217 if (!dentry)
1218 goto out_err;
1219
1220 page = alloc_page(GFP_KERNEL);
1221 err = -ENOMEM;
1222 if (!page)
1223 goto out_err;
1224
1225 err = fuse_readlink_page(inode, page);
1226 if (err) {
1227 __free_page(page);
1228 goto out_err;
1229 }
1230
1231 set_delayed_call(callback, page_put_link, page);
1232
1233 return page_address(page);
1234
1235out_err:
1236 return ERR_PTR(err);
Miklos Szeredie5e55582005-09-09 13:10:28 -07001237}
1238
Miklos Szeredie5e55582005-09-09 13:10:28 -07001239static int fuse_dir_open(struct inode *inode, struct file *file)
1240{
Miklos Szeredi91fe96b2009-04-28 16:56:37 +02001241 return fuse_open_common(inode, file, true);
Miklos Szeredie5e55582005-09-09 13:10:28 -07001242}
1243
1244static int fuse_dir_release(struct inode *inode, struct file *file)
1245{
Chad Austin2e64ff12018-12-10 10:54:52 -08001246 fuse_release_common(file, true);
Miklos Szeredi8b0797a2009-04-28 16:56:39 +02001247
1248 return 0;
Miklos Szeredie5e55582005-09-09 13:10:28 -07001249}
1250
Josef Bacik02c24a82011-07-16 20:44:56 -04001251static int fuse_dir_fsync(struct file *file, loff_t start, loff_t end,
1252 int datasync)
Miklos Szeredi82547982005-09-09 13:10:38 -07001253{
Miklos Szeredia9c2d1e2018-12-03 10:14:43 +01001254 struct inode *inode = file->f_mapping->host;
1255 struct fuse_conn *fc = get_fuse_conn(inode);
1256 int err;
1257
1258 if (is_bad_inode(inode))
1259 return -EIO;
1260
1261 if (fc->no_fsyncdir)
1262 return 0;
1263
1264 inode_lock(inode);
1265 err = fuse_fsync_common(file, start, end, datasync, FUSE_FSYNCDIR);
1266 if (err == -ENOSYS) {
1267 fc->no_fsyncdir = 1;
1268 err = 0;
1269 }
1270 inode_unlock(inode);
1271
1272 return err;
Miklos Szeredi82547982005-09-09 13:10:38 -07001273}
1274
Miklos Szeredib18da0c2011-12-13 11:58:49 +01001275static long fuse_dir_ioctl(struct file *file, unsigned int cmd,
1276 unsigned long arg)
1277{
1278 struct fuse_conn *fc = get_fuse_conn(file->f_mapping->host);
1279
1280 /* FUSE_IOCTL_DIR only supported for API version >= 7.18 */
1281 if (fc->minor < 18)
1282 return -ENOTTY;
1283
1284 return fuse_ioctl_common(file, cmd, arg, FUSE_IOCTL_DIR);
1285}
1286
1287static long fuse_dir_compat_ioctl(struct file *file, unsigned int cmd,
1288 unsigned long arg)
1289{
1290 struct fuse_conn *fc = get_fuse_conn(file->f_mapping->host);
1291
1292 if (fc->minor < 18)
1293 return -ENOTTY;
1294
1295 return fuse_ioctl_common(file, cmd, arg,
1296 FUSE_IOCTL_COMPAT | FUSE_IOCTL_DIR);
1297}
1298
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001299static bool update_mtime(unsigned ivalid, bool trust_local_mtime)
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001300{
1301 /* Always update if mtime is explicitly set */
1302 if (ivalid & ATTR_MTIME_SET)
1303 return true;
1304
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001305 /* Or if kernel i_mtime is the official one */
1306 if (trust_local_mtime)
1307 return true;
1308
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001309 /* If it's an open(O_TRUNC) or an ftruncate(), don't update */
1310 if ((ivalid & ATTR_SIZE) && (ivalid & (ATTR_OPEN | ATTR_FILE)))
1311 return false;
1312
1313 /* In all other cases update */
1314 return true;
1315}
1316
Eric W. Biederman8cb08322018-02-21 11:18:07 -06001317static void iattr_to_fattr(struct fuse_conn *fc, struct iattr *iattr,
1318 struct fuse_setattr_in *arg, bool trust_local_cmtime)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001319{
1320 unsigned ivalid = iattr->ia_valid;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001321
1322 if (ivalid & ATTR_MODE)
Miklos Szeredibefc6492005-11-07 00:59:52 -08001323 arg->valid |= FATTR_MODE, arg->mode = iattr->ia_mode;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001324 if (ivalid & ATTR_UID)
Eric W. Biederman8cb08322018-02-21 11:18:07 -06001325 arg->valid |= FATTR_UID, arg->uid = from_kuid(fc->user_ns, iattr->ia_uid);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001326 if (ivalid & ATTR_GID)
Eric W. Biederman8cb08322018-02-21 11:18:07 -06001327 arg->valid |= FATTR_GID, arg->gid = from_kgid(fc->user_ns, iattr->ia_gid);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001328 if (ivalid & ATTR_SIZE)
Miklos Szeredibefc6492005-11-07 00:59:52 -08001329 arg->valid |= FATTR_SIZE, arg->size = iattr->ia_size;
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001330 if (ivalid & ATTR_ATIME) {
1331 arg->valid |= FATTR_ATIME;
Miklos Szeredibefc6492005-11-07 00:59:52 -08001332 arg->atime = iattr->ia_atime.tv_sec;
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001333 arg->atimensec = iattr->ia_atime.tv_nsec;
1334 if (!(ivalid & ATTR_ATIME_SET))
1335 arg->valid |= FATTR_ATIME_NOW;
1336 }
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001337 if ((ivalid & ATTR_MTIME) && update_mtime(ivalid, trust_local_cmtime)) {
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001338 arg->valid |= FATTR_MTIME;
Miklos Szeredibefc6492005-11-07 00:59:52 -08001339 arg->mtime = iattr->ia_mtime.tv_sec;
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001340 arg->mtimensec = iattr->ia_mtime.tv_nsec;
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001341 if (!(ivalid & ATTR_MTIME_SET) && !trust_local_cmtime)
Miklos Szeredi17637cb2007-10-18 03:07:01 -07001342 arg->valid |= FATTR_MTIME_NOW;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001343 }
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001344 if ((ivalid & ATTR_CTIME) && trust_local_cmtime) {
1345 arg->valid |= FATTR_CTIME;
1346 arg->ctime = iattr->ia_ctime.tv_sec;
1347 arg->ctimensec = iattr->ia_ctime.tv_nsec;
1348 }
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001349}
1350
Miklos Szeredi6f9f1182006-01-06 00:19:39 -08001351/*
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001352 * Prevent concurrent writepages on inode
1353 *
1354 * This is done by adding a negative bias to the inode write counter
1355 * and waiting for all pending writes to finish.
1356 */
1357void fuse_set_nowrite(struct inode *inode)
1358{
1359 struct fuse_conn *fc = get_fuse_conn(inode);
1360 struct fuse_inode *fi = get_fuse_inode(inode);
1361
Al Viro59551022016-01-22 15:40:57 -05001362 BUG_ON(!inode_is_locked(inode));
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001363
1364 spin_lock(&fc->lock);
1365 BUG_ON(fi->writectr < 0);
1366 fi->writectr += FUSE_NOWRITE;
1367 spin_unlock(&fc->lock);
1368 wait_event(fi->page_waitq, fi->writectr == FUSE_NOWRITE);
1369}
1370
1371/*
1372 * Allow writepages on inode
1373 *
1374 * Remove the bias from the writecounter and send any queued
1375 * writepages.
1376 */
1377static void __fuse_release_nowrite(struct inode *inode)
1378{
1379 struct fuse_inode *fi = get_fuse_inode(inode);
1380
1381 BUG_ON(fi->writectr != FUSE_NOWRITE);
1382 fi->writectr = 0;
1383 fuse_flush_writepages(inode);
1384}
1385
1386void fuse_release_nowrite(struct inode *inode)
1387{
1388 struct fuse_conn *fc = get_fuse_conn(inode);
1389
1390 spin_lock(&fc->lock);
1391 __fuse_release_nowrite(inode);
1392 spin_unlock(&fc->lock);
1393}
1394
Miklos Szeredi70781872014-12-12 09:49:05 +01001395static void fuse_setattr_fill(struct fuse_conn *fc, struct fuse_args *args,
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001396 struct inode *inode,
1397 struct fuse_setattr_in *inarg_p,
1398 struct fuse_attr_out *outarg_p)
1399{
Miklos Szeredi70781872014-12-12 09:49:05 +01001400 args->in.h.opcode = FUSE_SETATTR;
1401 args->in.h.nodeid = get_node_id(inode);
1402 args->in.numargs = 1;
1403 args->in.args[0].size = sizeof(*inarg_p);
1404 args->in.args[0].value = inarg_p;
1405 args->out.numargs = 1;
Miklos Szeredi21f62172015-01-06 10:45:35 +01001406 args->out.args[0].size = sizeof(*outarg_p);
Miklos Szeredi70781872014-12-12 09:49:05 +01001407 args->out.args[0].value = outarg_p;
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001408}
1409
1410/*
1411 * Flush inode->i_mtime to the server
1412 */
Maxim Patlasovab9e13f2014-04-28 14:19:24 +02001413int fuse_flush_times(struct inode *inode, struct fuse_file *ff)
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001414{
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001415 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +01001416 FUSE_ARGS(args);
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001417 struct fuse_setattr_in inarg;
1418 struct fuse_attr_out outarg;
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001419
1420 memset(&inarg, 0, sizeof(inarg));
1421 memset(&outarg, 0, sizeof(outarg));
1422
Maxim Patlasovab9e13f2014-04-28 14:19:24 +02001423 inarg.valid = FATTR_MTIME;
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001424 inarg.mtime = inode->i_mtime.tv_sec;
1425 inarg.mtimensec = inode->i_mtime.tv_nsec;
Maxim Patlasovab9e13f2014-04-28 14:19:24 +02001426 if (fc->minor >= 23) {
1427 inarg.valid |= FATTR_CTIME;
1428 inarg.ctime = inode->i_ctime.tv_sec;
1429 inarg.ctimensec = inode->i_ctime.tv_nsec;
1430 }
Miklos Szeredi1e18bda2014-04-28 14:19:23 +02001431 if (ff) {
1432 inarg.valid |= FATTR_FH;
1433 inarg.fh = ff->fh;
1434 }
Miklos Szeredi70781872014-12-12 09:49:05 +01001435 fuse_setattr_fill(fc, &args, inode, &inarg, &outarg);
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001436
Miklos Szeredi70781872014-12-12 09:49:05 +01001437 return fuse_simple_request(fc, &args);
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001438}
1439
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001440/*
Miklos Szeredi6f9f1182006-01-06 00:19:39 -08001441 * Set attributes, and at the same time refresh them.
1442 *
1443 * Truncation is slightly complicated, because the 'truncate' request
1444 * may fail, in which case we don't want to touch the mapping.
Miklos Szeredi9ffbb912006-10-17 00:10:06 -07001445 * vmtruncate() doesn't allow for this case, so do the rlimit checking
1446 * and the actual truncation by hand.
Miklos Szeredi6f9f1182006-01-06 00:19:39 -08001447 */
Jan Kara62490332016-05-26 17:12:41 +02001448int fuse_do_setattr(struct dentry *dentry, struct iattr *attr,
Maxim Patlasovefb9fa92012-12-18 14:05:08 +04001449 struct file *file)
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001450{
Jan Kara62490332016-05-26 17:12:41 +02001451 struct inode *inode = d_inode(dentry);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001452 struct fuse_conn *fc = get_fuse_conn(inode);
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001453 struct fuse_inode *fi = get_fuse_inode(inode);
Miklos Szeredi70781872014-12-12 09:49:05 +01001454 FUSE_ARGS(args);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001455 struct fuse_setattr_in inarg;
1456 struct fuse_attr_out outarg;
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001457 bool is_truncate = false;
Pavel Emelyanov83732002013-10-10 17:10:46 +04001458 bool is_wb = fc->writeback_cache;
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001459 loff_t oldsize;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001460 int err;
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001461 bool trust_local_cmtime = is_wb && S_ISREG(inode->i_mode);
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001462
Miklos Szeredi29433a22016-10-01 07:32:32 +02001463 if (!fc->default_permissions)
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001464 attr->ia_valid |= ATTR_FORCE;
1465
Jan Kara31051c82016-05-26 16:55:18 +02001466 err = setattr_prepare(dentry, attr);
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001467 if (err)
1468 return err;
Miklos Szeredi1e9a4ed2005-09-09 13:10:31 -07001469
Miklos Szeredi8d56add2011-02-25 14:44:58 +01001470 if (attr->ia_valid & ATTR_OPEN) {
Miklos Szeredidf0e91d2018-02-08 15:17:38 +01001471 /* This is coming from open(..., ... | O_TRUNC); */
1472 WARN_ON(!(attr->ia_valid & ATTR_SIZE));
1473 WARN_ON(attr->ia_size != 0);
1474 if (fc->atomic_o_trunc) {
1475 /*
1476 * No need to send request to userspace, since actual
1477 * truncation has already been done by OPEN. But still
1478 * need to truncate page cache.
1479 */
1480 i_size_write(inode, 0);
1481 truncate_pagecache(inode, 0);
Miklos Szeredi8d56add2011-02-25 14:44:58 +01001482 return 0;
Miklos Szeredidf0e91d2018-02-08 15:17:38 +01001483 }
Miklos Szeredi8d56add2011-02-25 14:44:58 +01001484 file = NULL;
1485 }
Miklos Szeredi6ff958e2007-10-18 03:07:02 -07001486
Miklos Szerediab2257e2018-10-01 10:07:05 +02001487 if (attr->ia_valid & ATTR_SIZE) {
1488 if (WARN_ON(!S_ISREG(inode->i_mode)))
1489 return -EIO;
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001490 is_truncate = true;
Miklos Szerediab2257e2018-10-01 10:07:05 +02001491 }
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001492
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001493 if (is_truncate) {
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001494 fuse_set_nowrite(inode);
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001495 set_bit(FUSE_I_SIZE_UNSTABLE, &fi->state);
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001496 if (trust_local_cmtime && attr->ia_size != inode->i_size)
1497 attr->ia_valid |= ATTR_MTIME | ATTR_CTIME;
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001498 }
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001499
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001500 memset(&inarg, 0, sizeof(inarg));
Miklos Szeredi0e9663e2007-10-18 03:07:05 -07001501 memset(&outarg, 0, sizeof(outarg));
Eric W. Biederman8cb08322018-02-21 11:18:07 -06001502 iattr_to_fattr(fc, attr, &inarg, trust_local_cmtime);
Miklos Szeredi49d49142007-10-18 03:07:00 -07001503 if (file) {
1504 struct fuse_file *ff = file->private_data;
1505 inarg.valid |= FATTR_FH;
1506 inarg.fh = ff->fh;
1507 }
Miklos Szeredif3332112007-10-18 03:07:04 -07001508 if (attr->ia_valid & ATTR_SIZE) {
1509 /* For mandatory locking in truncate */
1510 inarg.valid |= FATTR_LOCKOWNER;
1511 inarg.lock_owner = fuse_lock_owner_id(fc, current->files);
1512 }
Miklos Szeredi70781872014-12-12 09:49:05 +01001513 fuse_setattr_fill(fc, &args, inode, &inarg, &outarg);
1514 err = fuse_simple_request(fc, &args);
Miklos Szeredie00d2c22007-10-16 23:31:01 -07001515 if (err) {
1516 if (err == -EINTR)
1517 fuse_invalidate_attr(inode);
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001518 goto error;
Miklos Szeredie00d2c22007-10-16 23:31:01 -07001519 }
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001520
Miklos Szeredie00d2c22007-10-16 23:31:01 -07001521 if ((inode->i_mode ^ outarg.attr.mode) & S_IFMT) {
1522 make_bad_inode(inode);
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001523 err = -EIO;
1524 goto error;
Miklos Szeredie00d2c22007-10-16 23:31:01 -07001525 }
1526
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001527 spin_lock(&fc->lock);
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001528 /* the kernel maintains i_mtime locally */
Maxim Patlasov3ad22c62014-04-28 14:19:25 +02001529 if (trust_local_cmtime) {
1530 if (attr->ia_valid & ATTR_MTIME)
1531 inode->i_mtime = attr->ia_mtime;
1532 if (attr->ia_valid & ATTR_CTIME)
1533 inode->i_ctime = attr->ia_ctime;
Miklos Szeredi1e18bda2014-04-28 14:19:23 +02001534 /* FIXME: clear I_DIRTY_SYNC? */
Maxim Patlasovb0aa7602013-12-26 19:51:11 +04001535 }
1536
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001537 fuse_change_attributes_common(inode, &outarg.attr,
1538 attr_timeout(&outarg));
1539 oldsize = inode->i_size;
Pavel Emelyanov83732002013-10-10 17:10:46 +04001540 /* see the comment in fuse_change_attributes() */
1541 if (!is_wb || is_truncate || !S_ISREG(inode->i_mode))
1542 i_size_write(inode, outarg.attr.size);
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001543
1544 if (is_truncate) {
1545 /* NOTE: this may release/reacquire fc->lock */
1546 __fuse_release_nowrite(inode);
1547 }
1548 spin_unlock(&fc->lock);
1549
1550 /*
1551 * Only call invalidate_inode_pages2() after removing
1552 * FUSE_NOWRITE, otherwise fuse_launder_page() would deadlock.
1553 */
Pavel Emelyanov83732002013-10-10 17:10:46 +04001554 if ((is_truncate || !is_wb) &&
1555 S_ISREG(inode->i_mode) && oldsize != outarg.attr.size) {
Kirill A. Shutemov7caef262013-09-12 15:13:56 -07001556 truncate_pagecache(inode, outarg.attr.size);
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001557 invalidate_inode_pages2(inode->i_mapping);
1558 }
1559
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001560 clear_bit(FUSE_I_SIZE_UNSTABLE, &fi->state);
Miklos Szeredie00d2c22007-10-16 23:31:01 -07001561 return 0;
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001562
1563error:
1564 if (is_truncate)
1565 fuse_release_nowrite(inode);
1566
Maxim Patlasov06a7c3c2013-08-30 17:06:04 +04001567 clear_bit(FUSE_I_SIZE_UNSTABLE, &fi->state);
Miklos Szeredi3be5a522008-04-30 00:54:41 -07001568 return err;
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001569}
1570
Miklos Szeredi49d49142007-10-18 03:07:00 -07001571static int fuse_setattr(struct dentry *entry, struct iattr *attr)
1572{
David Howells2b0143b2015-03-17 22:25:59 +00001573 struct inode *inode = d_inode(entry);
Miklos Szeredi5e940c12016-10-01 07:32:32 +02001574 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredia09f99e2016-10-01 07:32:32 +02001575 struct file *file = (attr->ia_valid & ATTR_FILE) ? attr->ia_file : NULL;
Miklos Szeredi5e2b8822016-10-01 07:32:32 +02001576 int ret;
Maxim Patlasovefb9fa92012-12-18 14:05:08 +04001577
1578 if (!fuse_allow_current_process(get_fuse_conn(inode)))
1579 return -EACCES;
1580
Miklos Szeredia09f99e2016-10-01 07:32:32 +02001581 if (attr->ia_valid & (ATTR_KILL_SUID | ATTR_KILL_SGID)) {
Miklos Szeredia09f99e2016-10-01 07:32:32 +02001582 attr->ia_valid &= ~(ATTR_KILL_SUID | ATTR_KILL_SGID |
1583 ATTR_MODE);
Miklos Szeredia09f99e2016-10-01 07:32:32 +02001584
Miklos Szeredi5e940c12016-10-01 07:32:32 +02001585 /*
1586 * The only sane way to reliably kill suid/sgid is to do it in
1587 * the userspace filesystem
1588 *
1589 * This should be done on write(), truncate() and chown().
1590 */
1591 if (!fc->handle_killpriv) {
Miklos Szeredi5e940c12016-10-01 07:32:32 +02001592 /*
1593 * ia_mode calculation may have used stale i_mode.
1594 * Refresh and recalculate.
1595 */
1596 ret = fuse_do_getattr(inode, NULL, file);
1597 if (ret)
1598 return ret;
1599
1600 attr->ia_mode = inode->i_mode;
Miklos Szeredic01638f2016-12-06 16:18:45 +01001601 if (inode->i_mode & S_ISUID) {
Miklos Szeredi5e940c12016-10-01 07:32:32 +02001602 attr->ia_valid |= ATTR_MODE;
1603 attr->ia_mode &= ~S_ISUID;
1604 }
Miklos Szeredic01638f2016-12-06 16:18:45 +01001605 if ((inode->i_mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP)) {
Miklos Szeredi5e940c12016-10-01 07:32:32 +02001606 attr->ia_valid |= ATTR_MODE;
1607 attr->ia_mode &= ~S_ISGID;
1608 }
Miklos Szeredia09f99e2016-10-01 07:32:32 +02001609 }
1610 }
1611 if (!attr->ia_valid)
1612 return 0;
1613
Linus Torvaldsabb5a142016-10-10 13:04:49 -07001614 ret = fuse_do_setattr(entry, attr, file);
Miklos Szeredi5e2b8822016-10-01 07:32:32 +02001615 if (!ret) {
Seth Forshee60bcc882016-08-29 08:46:37 -05001616 /*
1617 * If filesystem supports acls it may have updated acl xattrs in
1618 * the filesystem, so forget cached acls for the inode.
1619 */
1620 if (fc->posix_acl)
1621 forget_all_cached_acls(inode);
1622
Miklos Szeredi5e2b8822016-10-01 07:32:32 +02001623 /* Directory mode changed, may need to revalidate access */
1624 if (d_is_dir(entry) && (attr->ia_valid & ATTR_MODE))
1625 fuse_invalidate_entry_cache(entry);
1626 }
1627 return ret;
Miklos Szeredi49d49142007-10-18 03:07:00 -07001628}
1629
David Howellsa528d352017-01-31 16:46:22 +00001630static int fuse_getattr(const struct path *path, struct kstat *stat,
1631 u32 request_mask, unsigned int flags)
Miklos Szeredie5e55582005-09-09 13:10:28 -07001632{
David Howellsa528d352017-01-31 16:46:22 +00001633 struct inode *inode = d_inode(path->dentry);
Miklos Szeredi244f6382007-10-16 23:31:02 -07001634 struct fuse_conn *fc = get_fuse_conn(inode);
Miklos Szeredi244f6382007-10-16 23:31:02 -07001635
Anatol Pomozovc2132c12013-01-14 22:30:00 -08001636 if (!fuse_allow_current_process(fc))
Miklos Szeredi244f6382007-10-16 23:31:02 -07001637 return -EACCES;
1638
Miklos Szeredi2f1e8192018-10-15 15:43:06 +02001639 return fuse_update_get_attr(inode, NULL, stat, request_mask, flags);
Miklos Szeredie5e55582005-09-09 13:10:28 -07001640}
1641
Arjan van de Ven754661f2007-02-12 00:55:38 -08001642static const struct inode_operations fuse_dir_inode_operations = {
Miklos Szeredie5e55582005-09-09 13:10:28 -07001643 .lookup = fuse_lookup,
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001644 .mkdir = fuse_mkdir,
1645 .symlink = fuse_symlink,
1646 .unlink = fuse_unlink,
1647 .rmdir = fuse_rmdir,
Miklos Szeredi2773bf02016-09-27 11:03:58 +02001648 .rename = fuse_rename2,
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001649 .link = fuse_link,
1650 .setattr = fuse_setattr,
1651 .create = fuse_create,
Miklos Szeredic8ccbe02012-06-05 15:10:22 +02001652 .atomic_open = fuse_atomic_open,
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001653 .mknod = fuse_mknod,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001654 .permission = fuse_permission,
1655 .getattr = fuse_getattr,
Miklos Szeredi92a87802005-09-09 13:10:31 -07001656 .listxattr = fuse_listxattr,
Seth Forshee60bcc882016-08-29 08:46:37 -05001657 .get_acl = fuse_get_acl,
1658 .set_acl = fuse_set_acl,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001659};
1660
Arjan van de Ven4b6f5d22006-03-28 01:56:42 -08001661static const struct file_operations fuse_dir_operations = {
Miklos Szeredib6aeade2005-09-09 13:10:30 -07001662 .llseek = generic_file_llseek,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001663 .read = generic_read_dir,
Al Virod9b3dbd2016-04-20 17:30:32 -04001664 .iterate_shared = fuse_readdir,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001665 .open = fuse_dir_open,
1666 .release = fuse_dir_release,
Miklos Szeredi82547982005-09-09 13:10:38 -07001667 .fsync = fuse_dir_fsync,
Miklos Szeredib18da0c2011-12-13 11:58:49 +01001668 .unlocked_ioctl = fuse_dir_ioctl,
1669 .compat_ioctl = fuse_dir_compat_ioctl,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001670};
1671
Arjan van de Ven754661f2007-02-12 00:55:38 -08001672static const struct inode_operations fuse_common_inode_operations = {
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001673 .setattr = fuse_setattr,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001674 .permission = fuse_permission,
1675 .getattr = fuse_getattr,
Miklos Szeredi92a87802005-09-09 13:10:31 -07001676 .listxattr = fuse_listxattr,
Seth Forshee60bcc882016-08-29 08:46:37 -05001677 .get_acl = fuse_get_acl,
1678 .set_acl = fuse_set_acl,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001679};
1680
Arjan van de Ven754661f2007-02-12 00:55:38 -08001681static const struct inode_operations fuse_symlink_inode_operations = {
Miklos Szeredi9e6268d2005-09-09 13:10:29 -07001682 .setattr = fuse_setattr,
Al Viro6b255392015-11-17 10:20:54 -05001683 .get_link = fuse_get_link,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001684 .getattr = fuse_getattr,
Miklos Szeredi92a87802005-09-09 13:10:31 -07001685 .listxattr = fuse_listxattr,
Miklos Szeredie5e55582005-09-09 13:10:28 -07001686};
1687
1688void fuse_init_common(struct inode *inode)
1689{
1690 inode->i_op = &fuse_common_inode_operations;
1691}
1692
1693void fuse_init_dir(struct inode *inode)
1694{
Miklos Szerediab2257e2018-10-01 10:07:05 +02001695 struct fuse_inode *fi = get_fuse_inode(inode);
1696
Miklos Szeredie5e55582005-09-09 13:10:28 -07001697 inode->i_op = &fuse_dir_inode_operations;
1698 inode->i_fop = &fuse_dir_operations;
Miklos Szerediab2257e2018-10-01 10:07:05 +02001699
1700 spin_lock_init(&fi->rdc.lock);
1701 fi->rdc.cached = false;
1702 fi->rdc.size = 0;
1703 fi->rdc.pos = 0;
1704 fi->rdc.version = 0;
Miklos Szeredie5e55582005-09-09 13:10:28 -07001705}
1706
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001707static int fuse_symlink_readpage(struct file *null, struct page *page)
1708{
1709 int err = fuse_readlink_page(page->mapping->host, page);
1710
1711 if (!err)
1712 SetPageUptodate(page);
1713
1714 unlock_page(page);
1715
1716 return err;
1717}
1718
1719static const struct address_space_operations fuse_symlink_aops = {
1720 .readpage = fuse_symlink_readpage,
1721};
1722
Miklos Szeredie5e55582005-09-09 13:10:28 -07001723void fuse_init_symlink(struct inode *inode)
1724{
1725 inode->i_op = &fuse_symlink_inode_operations;
Dan Schatzberg5571f1e2018-10-11 08:17:00 -07001726 inode->i_data.a_ops = &fuse_symlink_aops;
1727 inode_nohighmem(inode);
Miklos Szeredie5e55582005-09-09 13:10:28 -07001728}