diff options
author | Weston Andros Adamson <dros@netapp.com> | 2013-08-13 16:37:37 -0400 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2013-09-05 10:50:45 -0400 |
commit | 8c21c62c4452f4e66c3dac9b3f6b74474fad3e08 (patch) | |
tree | c7c5b3edd7dfa21604336c3820e6b859d3cf0fa6 /fs | |
parent | 3787d5063c52b0c38003e6293f24839508604070 (diff) |
nfs4.1: Add SP4_MACH_CRED write and commit support
WRITE and COMMIT can use the machine credential.
If WRITE is supported and COMMIT is not, make all (mach cred) writes FILE_SYNC4.
Signed-off-by: Weston Andros Adamson <dros@netapp.com>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/nfs/nfs4_fs.h | 50 | ||||
-rw-r--r-- | fs/nfs/nfs4proc.c | 10 | ||||
-rw-r--r-- | fs/nfs/write.c | 6 |
3 files changed, 57 insertions, 9 deletions
diff --git a/fs/nfs/nfs4_fs.h b/fs/nfs/nfs4_fs.h index 3559e899666e..d2db3ce07d3a 100644 --- a/fs/nfs/nfs4_fs.h +++ b/fs/nfs/nfs4_fs.h | |||
@@ -269,15 +269,9 @@ is_ds_client(struct nfs_client *clp) | |||
269 | return clp->cl_exchange_flags & EXCHGID4_FLAG_USE_PNFS_DS; | 269 | return clp->cl_exchange_flags & EXCHGID4_FLAG_USE_PNFS_DS; |
270 | } | 270 | } |
271 | 271 | ||
272 | /* | 272 | static inline bool |
273 | * Function responsible for determining if an rpc_message should use the | 273 | _nfs4_state_protect(struct nfs_client *clp, unsigned long sp4_mode, |
274 | * machine cred under SP4_MACH_CRED and if so switching the credential and | 274 | struct rpc_clnt **clntp, struct rpc_message *msg) |
275 | * authflavor (using the nfs_client's rpc_clnt which will be krb5i/p). | ||
276 | * Should be called before rpc_call_sync/rpc_call_async. | ||
277 | */ | ||
278 | static inline void | ||
279 | nfs4_state_protect(struct nfs_client *clp, unsigned long sp4_mode, | ||
280 | struct rpc_clnt **clntp, struct rpc_message *msg) | ||
281 | { | 275 | { |
282 | struct rpc_cred *newcred = NULL; | 276 | struct rpc_cred *newcred = NULL; |
283 | rpc_authflavor_t flavor; | 277 | rpc_authflavor_t flavor; |
@@ -295,7 +289,37 @@ nfs4_state_protect(struct nfs_client *clp, unsigned long sp4_mode, | |||
295 | WARN_ON(flavor != RPC_AUTH_GSS_KRB5I && | 289 | WARN_ON(flavor != RPC_AUTH_GSS_KRB5I && |
296 | flavor != RPC_AUTH_GSS_KRB5P); | 290 | flavor != RPC_AUTH_GSS_KRB5P); |
297 | *clntp = clp->cl_rpcclient; | 291 | *clntp = clp->cl_rpcclient; |
292 | |||
293 | return true; | ||
298 | } | 294 | } |
295 | return false; | ||
296 | } | ||
297 | |||
298 | /* | ||
299 | * Function responsible for determining if an rpc_message should use the | ||
300 | * machine cred under SP4_MACH_CRED and if so switching the credential and | ||
301 | * authflavor (using the nfs_client's rpc_clnt which will be krb5i/p). | ||
302 | * Should be called before rpc_call_sync/rpc_call_async. | ||
303 | */ | ||
304 | static inline void | ||
305 | nfs4_state_protect(struct nfs_client *clp, unsigned long sp4_mode, | ||
306 | struct rpc_clnt **clntp, struct rpc_message *msg) | ||
307 | { | ||
308 | _nfs4_state_protect(clp, sp4_mode, clntp, msg); | ||
309 | } | ||
310 | |||
311 | /* | ||
312 | * Special wrapper to nfs4_state_protect for write. | ||
313 | * If WRITE can use machine cred but COMMIT cannot, make sure all writes | ||
314 | * that use machine cred use NFS_FILE_SYNC. | ||
315 | */ | ||
316 | static inline void | ||
317 | nfs4_state_protect_write(struct nfs_client *clp, struct rpc_clnt **clntp, | ||
318 | struct rpc_message *msg, struct nfs_write_data *wdata) | ||
319 | { | ||
320 | if (_nfs4_state_protect(clp, NFS_SP4_MACH_CRED_WRITE, clntp, msg) && | ||
321 | !test_bit(NFS_SP4_MACH_CRED_COMMIT, &clp->cl_sp4_flags)) | ||
322 | wdata->args.stable = NFS_FILE_SYNC; | ||
299 | } | 323 | } |
300 | #else /* CONFIG_NFS_v4_1 */ | 324 | #else /* CONFIG_NFS_v4_1 */ |
301 | static inline struct nfs4_session *nfs4_get_session(const struct nfs_server *server) | 325 | static inline struct nfs4_session *nfs4_get_session(const struct nfs_server *server) |
@@ -320,6 +344,12 @@ nfs4_state_protect(struct nfs_client *clp, unsigned long sp4_flags, | |||
320 | struct rpc_clnt **clntp, struct rpc_message *msg) | 344 | struct rpc_clnt **clntp, struct rpc_message *msg) |
321 | { | 345 | { |
322 | } | 346 | } |
347 | |||
348 | static inline void | ||
349 | nfs4_state_protect_write(struct nfs_client *clp, struct rpc_clnt **clntp, | ||
350 | struct rpc_message *msg, struct nfs_write_data *wdata) | ||
351 | { | ||
352 | } | ||
323 | #endif /* CONFIG_NFS_V4_1 */ | 353 | #endif /* CONFIG_NFS_V4_1 */ |
324 | 354 | ||
325 | extern const struct nfs4_minor_version_ops *nfs_v4_minor_ops[]; | 355 | extern const struct nfs4_minor_version_ops *nfs_v4_minor_ops[]; |
@@ -455,6 +485,8 @@ static inline bool nfs4_valid_open_stateid(const struct nfs4_state *state) | |||
455 | 485 | ||
456 | #define nfs4_close_state(a, b) do { } while (0) | 486 | #define nfs4_close_state(a, b) do { } while (0) |
457 | #define nfs4_close_sync(a, b) do { } while (0) | 487 | #define nfs4_close_sync(a, b) do { } while (0) |
488 | #define nfs4_state_protect(a, b, c, d) do { } while (0) | ||
489 | #define nfs4_state_protect_write(a, b, c, d) do { } while (0) | ||
458 | 490 | ||
459 | #endif /* CONFIG_NFS_V4 */ | 491 | #endif /* CONFIG_NFS_V4 */ |
460 | #endif /* __LINUX_FS_NFS_NFS4_FS.H */ | 492 | #endif /* __LINUX_FS_NFS_NFS4_FS.H */ |
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c index 4818a38f469b..e3cdfe346ebc 100644 --- a/fs/nfs/nfs4proc.c +++ b/fs/nfs/nfs4proc.c | |||
@@ -6223,6 +6223,16 @@ static int nfs4_sp4_select_mode(struct nfs_client *clp, | |||
6223 | dfprintk(MOUNT, " stateid mode enabled\n"); | 6223 | dfprintk(MOUNT, " stateid mode enabled\n"); |
6224 | set_bit(NFS_SP4_MACH_CRED_STATEID, &clp->cl_sp4_flags); | 6224 | set_bit(NFS_SP4_MACH_CRED_STATEID, &clp->cl_sp4_flags); |
6225 | } | 6225 | } |
6226 | |||
6227 | if (test_bit(OP_WRITE, sp->allow.u.longs)) { | ||
6228 | dfprintk(MOUNT, " write mode enabled\n"); | ||
6229 | set_bit(NFS_SP4_MACH_CRED_WRITE, &clp->cl_sp4_flags); | ||
6230 | } | ||
6231 | |||
6232 | if (test_bit(OP_COMMIT, sp->allow.u.longs)) { | ||
6233 | dfprintk(MOUNT, " commit mode enabled\n"); | ||
6234 | set_bit(NFS_SP4_MACH_CRED_COMMIT, &clp->cl_sp4_flags); | ||
6235 | } | ||
6226 | } | 6236 | } |
6227 | 6237 | ||
6228 | return 0; | 6238 | return 0; |
diff --git a/fs/nfs/write.c b/fs/nfs/write.c index 379450c8d04b..40979e815434 100644 --- a/fs/nfs/write.c +++ b/fs/nfs/write.c | |||
@@ -1022,6 +1022,9 @@ int nfs_initiate_write(struct rpc_clnt *clnt, | |||
1022 | data->args.count, | 1022 | data->args.count, |
1023 | (unsigned long long)data->args.offset); | 1023 | (unsigned long long)data->args.offset); |
1024 | 1024 | ||
1025 | nfs4_state_protect_write(NFS_SERVER(inode)->nfs_client, | ||
1026 | &task_setup_data.rpc_client, &msg, data); | ||
1027 | |||
1025 | task = rpc_run_task(&task_setup_data); | 1028 | task = rpc_run_task(&task_setup_data); |
1026 | if (IS_ERR(task)) { | 1029 | if (IS_ERR(task)) { |
1027 | ret = PTR_ERR(task); | 1030 | ret = PTR_ERR(task); |
@@ -1488,6 +1491,9 @@ int nfs_initiate_commit(struct rpc_clnt *clnt, struct nfs_commit_data *data, | |||
1488 | 1491 | ||
1489 | dprintk("NFS: %5u initiated commit call\n", data->task.tk_pid); | 1492 | dprintk("NFS: %5u initiated commit call\n", data->task.tk_pid); |
1490 | 1493 | ||
1494 | nfs4_state_protect(NFS_SERVER(data->inode)->nfs_client, | ||
1495 | NFS_SP4_MACH_CRED_COMMIT, &task_setup_data.rpc_client, &msg); | ||
1496 | |||
1491 | task = rpc_run_task(&task_setup_data); | 1497 | task = rpc_run_task(&task_setup_data); |
1492 | if (IS_ERR(task)) | 1498 | if (IS_ERR(task)) |
1493 | return PTR_ERR(task); | 1499 | return PTR_ERR(task); |