aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
Diffstat (limited to 'fs')
-rw-r--r--fs/nfs/dir.c47
-rw-r--r--fs/nfs/nfs3proc.c2
-rw-r--r--fs/nfs/nfs4proc.c56
-rw-r--r--fs/nfs/proc.c2
4 files changed, 57 insertions, 50 deletions
diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c
index dc93d356341b..e37ffddd79c2 100644
--- a/fs/nfs/dir.c
+++ b/fs/nfs/dir.c
@@ -105,8 +105,9 @@ const struct inode_operations nfs3_dir_inode_operations = {
105#ifdef CONFIG_NFS_V4 105#ifdef CONFIG_NFS_V4
106 106
107static struct dentry *nfs_atomic_lookup(struct inode *, struct dentry *, struct nameidata *); 107static struct dentry *nfs_atomic_lookup(struct inode *, struct dentry *, struct nameidata *);
108static int nfs_open_create(struct inode *dir, struct dentry *dentry, int mode, struct nameidata *nd);
108const struct inode_operations nfs4_dir_inode_operations = { 109const struct inode_operations nfs4_dir_inode_operations = {
109 .create = nfs_create, 110 .create = nfs_open_create,
110 .lookup = nfs_atomic_lookup, 111 .lookup = nfs_atomic_lookup,
111 .link = nfs_link, 112 .link = nfs_link,
112 .unlink = nfs_unlink, 113 .unlink = nfs_unlink,
@@ -1239,6 +1240,44 @@ no_open_dput:
1239no_open: 1240no_open:
1240 return nfs_lookup_revalidate(dentry, nd); 1241 return nfs_lookup_revalidate(dentry, nd);
1241} 1242}
1243
1244static int nfs_open_create(struct inode *dir, struct dentry *dentry, int mode,
1245 struct nameidata *nd)
1246{
1247 struct nfs_open_context *ctx = NULL;
1248 struct iattr attr;
1249 int error;
1250 int open_flags = 0;
1251
1252 dfprintk(VFS, "NFS: create(%s/%ld), %s\n",
1253 dir->i_sb->s_id, dir->i_ino, dentry->d_name.name);
1254
1255 attr.ia_mode = mode;
1256 attr.ia_valid = ATTR_MODE;
1257
1258 if ((nd->flags & LOOKUP_CREATE) != 0) {
1259 open_flags = nd->intent.open.flags;
1260
1261 ctx = nameidata_to_nfs_open_context(dentry, nd);
1262 error = PTR_ERR(ctx);
1263 if (IS_ERR(ctx))
1264 goto out_err;
1265 }
1266
1267 error = NFS_PROTO(dir)->create(dir, dentry, &attr, open_flags, ctx);
1268 if (error != 0)
1269 goto out_put_ctx;
1270 if (ctx != NULL)
1271 nfs_intent_set_file(nd, ctx);
1272 return 0;
1273out_put_ctx:
1274 if (ctx != NULL)
1275 put_nfs_open_context(ctx);
1276out_err:
1277 d_drop(dentry);
1278 return error;
1279}
1280
1242#endif /* CONFIG_NFSV4 */ 1281#endif /* CONFIG_NFSV4 */
1243 1282
1244static struct dentry *nfs_readdir_lookup(nfs_readdir_descriptor_t *desc) 1283static struct dentry *nfs_readdir_lookup(nfs_readdir_descriptor_t *desc)
@@ -1369,7 +1408,6 @@ static int nfs_create(struct inode *dir, struct dentry *dentry, int mode,
1369{ 1408{
1370 struct iattr attr; 1409 struct iattr attr;
1371 int error; 1410 int error;
1372 int open_flags = 0;
1373 1411
1374 dfprintk(VFS, "NFS: create(%s/%ld), %s\n", 1412 dfprintk(VFS, "NFS: create(%s/%ld), %s\n",
1375 dir->i_sb->s_id, dir->i_ino, dentry->d_name.name); 1413 dir->i_sb->s_id, dir->i_ino, dentry->d_name.name);
@@ -1377,10 +1415,7 @@ static int nfs_create(struct inode *dir, struct dentry *dentry, int mode,
1377 attr.ia_mode = mode; 1415 attr.ia_mode = mode;
1378 attr.ia_valid = ATTR_MODE; 1416 attr.ia_valid = ATTR_MODE;
1379 1417
1380 if ((nd->flags & LOOKUP_CREATE) != 0) 1418 error = NFS_PROTO(dir)->create(dir, dentry, &attr, 0, NULL);
1381 open_flags = nd->intent.open.flags;
1382
1383 error = NFS_PROTO(dir)->create(dir, dentry, &attr, open_flags, nd);
1384 if (error != 0) 1419 if (error != 0)
1385 goto out_err; 1420 goto out_err;
1386 return 0; 1421 return 0;
diff --git a/fs/nfs/nfs3proc.c b/fs/nfs/nfs3proc.c
index fabb4f2849a1..2be4a7f59f1c 100644
--- a/fs/nfs/nfs3proc.c
+++ b/fs/nfs/nfs3proc.c
@@ -313,7 +313,7 @@ static void nfs3_free_createdata(struct nfs3_createdata *data)
313 */ 313 */
314static int 314static int
315nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr, 315nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
316 int flags, struct nameidata *nd) 316 int flags, struct nfs_open_context *ctx)
317{ 317{
318 struct nfs3_createdata *data; 318 struct nfs3_createdata *data;
319 mode_t mode = sattr->ia_mode; 319 mode_t mode = sattr->ia_mode;
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
index 83c5ef6e7cef..617b149ee16d 100644
--- a/fs/nfs/nfs4proc.c
+++ b/fs/nfs/nfs4proc.c
@@ -1998,32 +1998,6 @@ out:
1998 return status; 1998 return status;
1999} 1999}
2000 2000
2001static int nfs4_intent_set_file(struct nameidata *nd, struct path *path, struct nfs4_state *state, fmode_t fmode)
2002{
2003 struct file *filp;
2004 int ret;
2005
2006 /* If the open_intent is for execute, we have an extra check to make */
2007 if (fmode & FMODE_EXEC) {
2008 ret = nfs_may_open(state->inode,
2009 state->owner->so_cred,
2010 nd->intent.open.flags);
2011 if (ret < 0)
2012 goto out_close;
2013 }
2014 filp = lookup_instantiate_filp(nd, path->dentry, NULL);
2015 if (!IS_ERR(filp)) {
2016 struct nfs_open_context *ctx;
2017 ctx = nfs_file_open_context(filp);
2018 ctx->state = state;
2019 return 0;
2020 }
2021 ret = PTR_ERR(filp);
2022out_close:
2023 nfs4_close_sync(path, state, fmode & (FMODE_READ|FMODE_WRITE));
2024 return ret;
2025}
2026
2027struct inode * 2001struct inode *
2028nfs4_atomic_open(struct inode *dir, struct nfs_open_context *ctx, int open_flags, struct iattr *attr) 2002nfs4_atomic_open(struct inode *dir, struct nfs_open_context *ctx, int open_flags, struct iattr *attr)
2029{ 2003{
@@ -2491,36 +2465,34 @@ static int nfs4_proc_readlink(struct inode *inode, struct page *page,
2491 2465
2492static int 2466static int
2493nfs4_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr, 2467nfs4_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
2494 int flags, struct nameidata *nd) 2468 int flags, struct nfs_open_context *ctx)
2495{ 2469{
2496 struct path path = { 2470 struct path my_path = {
2497 .mnt = nd->path.mnt,
2498 .dentry = dentry, 2471 .dentry = dentry,
2499 }; 2472 };
2473 struct path *path = &my_path;
2500 struct nfs4_state *state; 2474 struct nfs4_state *state;
2501 struct rpc_cred *cred; 2475 struct rpc_cred *cred = NULL;
2502 fmode_t fmode = flags & (FMODE_READ | FMODE_WRITE); 2476 fmode_t fmode = 0;
2503 int status = 0; 2477 int status = 0;
2504 2478
2505 cred = rpc_lookup_cred(); 2479 if (ctx != NULL) {
2506 if (IS_ERR(cred)) { 2480 cred = ctx->cred;
2507 status = PTR_ERR(cred); 2481 path = &ctx->path;
2508 goto out; 2482 fmode = ctx->mode;
2509 } 2483 }
2510 state = nfs4_do_open(dir, &path, fmode, flags, sattr, cred); 2484 state = nfs4_do_open(dir, path, fmode, flags, sattr, cred);
2511 d_drop(dentry); 2485 d_drop(dentry);
2512 if (IS_ERR(state)) { 2486 if (IS_ERR(state)) {
2513 status = PTR_ERR(state); 2487 status = PTR_ERR(state);
2514 goto out_putcred; 2488 goto out;
2515 } 2489 }
2516 d_add(dentry, igrab(state->inode)); 2490 d_add(dentry, igrab(state->inode));
2517 nfs_set_verifier(dentry, nfs_save_change_attribute(dir)); 2491 nfs_set_verifier(dentry, nfs_save_change_attribute(dir));
2518 if (status == 0 && (nd->flags & LOOKUP_OPEN) != 0) 2492 if (ctx != NULL)
2519 status = nfs4_intent_set_file(nd, &path, state, fmode); 2493 ctx->state = state;
2520 else 2494 else
2521 nfs4_close_sync(&path, state, fmode); 2495 nfs4_close_sync(path, state, fmode);
2522out_putcred:
2523 put_rpccred(cred);
2524out: 2496out:
2525 return status; 2497 return status;
2526} 2498}
diff --git a/fs/nfs/proc.c b/fs/nfs/proc.c
index 611bec22f552..4ef39ae88ea5 100644
--- a/fs/nfs/proc.c
+++ b/fs/nfs/proc.c
@@ -258,7 +258,7 @@ static void nfs_free_createdata(const struct nfs_createdata *data)
258 258
259static int 259static int
260nfs_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr, 260nfs_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr,
261 int flags, struct nameidata *nd) 261 int flags, struct nfs_open_context *ctx)
262{ 262{
263 struct nfs_createdata *data; 263 struct nfs_createdata *data;
264 struct rpc_message msg = { 264 struct rpc_message msg = {