nfs4xdr.c 142.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10
/*
 *  fs/nfs/nfs4xdr.c
 *
 *  Client-side XDR for NFSv4.
 *
 *  Copyright (c) 2002 The Regents of the University of Michigan.
 *  All rights reserved.
 *
 *  Kendrick Smith <kmsmith@umich.edu>
 *  Andy Adamson   <andros@umich.edu>
11
 *
L
Linus Torvalds 已提交
12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
 *  Redistribution and use in source and binary forms, with or without
 *  modification, are permitted provided that the following conditions
 *  are met:
 *
 *  1. Redistributions of source code must retain the above copyright
 *     notice, this list of conditions and the following disclaimer.
 *  2. Redistributions in binary form must reproduce the above copyright
 *     notice, this list of conditions and the following disclaimer in the
 *     documentation and/or other materials provided with the distribution.
 *  3. Neither the name of the University nor the names of its
 *     contributors may be used to endorse or promote products derived
 *     from this software without specific prior written permission.
 *
 *  THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
 *  WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 *  MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 *  DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 *  FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 *  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 *  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
 *  BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
 *  LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 *  NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 *  SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#include <linux/param.h>
#include <linux/time.h>
#include <linux/mm.h>
#include <linux/slab.h>
#include <linux/utsname.h>
#include <linux/errno.h>
#include <linux/string.h>
#include <linux/in.h>
#include <linux/pagemap.h>
#include <linux/proc_fs.h>
#include <linux/kdev_t.h>
#include <linux/sunrpc/clnt.h>
#include <linux/nfs.h>
#include <linux/nfs4.h>
#include <linux/nfs_fs.h>
#include <linux/nfs_idmap.h>
54
#include "nfs4_fs.h"
L
Linus Torvalds 已提交
55 56 57 58 59 60

#define NFSDBG_FACILITY		NFSDBG_XDR

/* Mapping from NFS error code to "errno" error code. */
#define errno_NFSERR_IO		EIO

61
static int nfs4_stat_to_errno(int);
L
Linus Torvalds 已提交
62 63 64 65 66 67 68 69

/* NFSv4 COMPOUND tags are only wanted for debugging purposes */
#ifdef DEBUG
#define NFS4_MAXTAGLEN		20
#else
#define NFS4_MAXTAGLEN		0
#endif

70
/* lock,open owner id:
71
 * we currently use size 2 (u64) out of (NFS4_OPAQUE_LIMIT  >> 2)
L
Linus Torvalds 已提交
72
 */
73 74
#define open_owner_id_maxsz	(1 + 4)
#define lock_owner_id_maxsz	(1 + 4)
T
Trond Myklebust 已提交
75
#define decode_lockowner_maxsz	(1 + XDR_QUADLEN(IDMAP_NAMESZ))
L
Linus Torvalds 已提交
76 77 78 79
#define compound_encode_hdr_maxsz	(3 + (NFS4_MAXTAGLEN >> 2))
#define compound_decode_hdr_maxsz	(3 + (NFS4_MAXTAGLEN >> 2))
#define op_encode_hdr_maxsz	(1)
#define op_decode_hdr_maxsz	(2)
T
Trond Myklebust 已提交
80 81 82 83
#define encode_stateid_maxsz	(XDR_QUADLEN(NFS4_STATEID_SIZE))
#define decode_stateid_maxsz	(XDR_QUADLEN(NFS4_STATEID_SIZE))
#define encode_verifier_maxsz	(XDR_QUADLEN(NFS4_VERIFIER_SIZE))
#define decode_verifier_maxsz	(XDR_QUADLEN(NFS4_VERIFIER_SIZE))
L
Linus Torvalds 已提交
84 85 86 87 88 89 90 91
#define encode_putfh_maxsz	(op_encode_hdr_maxsz + 1 + \
				(NFS4_FHSIZE >> 2))
#define decode_putfh_maxsz	(op_decode_hdr_maxsz)
#define encode_putrootfh_maxsz	(op_encode_hdr_maxsz)
#define decode_putrootfh_maxsz	(op_decode_hdr_maxsz)
#define encode_getfh_maxsz      (op_encode_hdr_maxsz)
#define decode_getfh_maxsz      (op_decode_hdr_maxsz + 1 + \
				((3+NFS4_FHSIZE) >> 2))
92 93
#define nfs4_fattr_bitmap_maxsz 3
#define encode_getattr_maxsz    (op_encode_hdr_maxsz + nfs4_fattr_bitmap_maxsz)
L
Linus Torvalds 已提交
94 95
#define nfs4_name_maxsz		(1 + ((3 + NFS4_MAXNAMLEN) >> 2))
#define nfs4_path_maxsz		(1 + ((3 + NFS4_MAXPATHLEN) >> 2))
96 97
#define nfs4_owner_maxsz	(1 + XDR_QUADLEN(IDMAP_NAMESZ))
#define nfs4_group_maxsz	(1 + XDR_QUADLEN(IDMAP_NAMESZ))
98 99
/* This is based on getfattr, which uses the most attributes: */
#define nfs4_fattr_value_maxsz	(1 + (1 + 2 + 2 + 4 + 2 + 1 + 1 + 2 + 2 + \
100
				3 + 3 + 3 + nfs4_owner_maxsz + nfs4_group_maxsz))
101 102 103
#define nfs4_fattr_maxsz	(nfs4_fattr_bitmap_maxsz + \
				nfs4_fattr_value_maxsz)
#define decode_getattr_maxsz    (op_decode_hdr_maxsz + nfs4_fattr_maxsz)
T
Trond Myklebust 已提交
104 105 106 107 108
#define encode_attrs_maxsz	(nfs4_fattr_bitmap_maxsz + \
				 1 + 2 + 1 + \
				nfs4_owner_maxsz + \
				nfs4_group_maxsz + \
				4 + 4)
L
Linus Torvalds 已提交
109 110
#define encode_savefh_maxsz     (op_encode_hdr_maxsz)
#define decode_savefh_maxsz     (op_decode_hdr_maxsz)
111 112
#define encode_restorefh_maxsz  (op_encode_hdr_maxsz)
#define decode_restorefh_maxsz  (op_decode_hdr_maxsz)
F
Fred Isaman 已提交
113
#define encode_fsinfo_maxsz	(encode_getattr_maxsz)
L
Linus Torvalds 已提交
114 115 116 117 118
#define decode_fsinfo_maxsz	(op_decode_hdr_maxsz + 11)
#define encode_renew_maxsz	(op_encode_hdr_maxsz + 3)
#define decode_renew_maxsz	(op_decode_hdr_maxsz)
#define encode_setclientid_maxsz \
				(op_encode_hdr_maxsz + \
119 120 121 122 123 124
				XDR_QUADLEN(NFS4_VERIFIER_SIZE) + \
				XDR_QUADLEN(NFS4_SETCLIENTID_NAMELEN) + \
				1 /* sc_prog */ + \
				XDR_QUADLEN(RPCBIND_MAXNETIDLEN) + \
				XDR_QUADLEN(RPCBIND_MAXUADDRLEN) + \
				1) /* sc_cb_ident */
L
Linus Torvalds 已提交
125 126 127 128 129 130 131 132 133
#define decode_setclientid_maxsz \
				(op_decode_hdr_maxsz + \
				2 + \
				1024) /* large value for CLID_INUSE */
#define encode_setclientid_confirm_maxsz \
				(op_encode_hdr_maxsz + \
				3 + (NFS4_VERIFIER_SIZE >> 2))
#define decode_setclientid_confirm_maxsz \
				(op_decode_hdr_maxsz)
134 135
#define encode_lookup_maxsz	(op_encode_hdr_maxsz + nfs4_name_maxsz)
#define decode_lookup_maxsz	(op_decode_hdr_maxsz)
136 137
#define encode_share_access_maxsz \
				(2)
T
Trond Myklebust 已提交
138
#define encode_createmode_maxsz	(1 + encode_attrs_maxsz)
139 140 141 142 143 144 145 146
#define encode_opentype_maxsz	(1 + encode_createmode_maxsz)
#define encode_claim_null_maxsz	(1 + nfs4_name_maxsz)
#define encode_open_maxsz	(op_encode_hdr_maxsz + \
				2 + encode_share_access_maxsz + 2 + \
				open_owner_id_maxsz + \
				encode_opentype_maxsz + \
				encode_claim_null_maxsz)
#define decode_ace_maxsz	(3 + nfs4_owner_maxsz)
T
Trond Myklebust 已提交
147
#define decode_delegation_maxsz	(1 + decode_stateid_maxsz + 1 + \
148 149 150
				decode_ace_maxsz)
#define decode_change_info_maxsz	(5)
#define decode_open_maxsz	(op_decode_hdr_maxsz + \
T
Trond Myklebust 已提交
151
				decode_stateid_maxsz + \
152 153 154
				decode_change_info_maxsz + 1 + \
				nfs4_fattr_bitmap_maxsz + \
				decode_delegation_maxsz)
T
Trond Myklebust 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
#define encode_open_confirm_maxsz \
				(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + 1)
#define decode_open_confirm_maxsz \
				(op_decode_hdr_maxsz + \
				 decode_stateid_maxsz)
#define encode_open_downgrade_maxsz \
				(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + 1 + \
				 encode_share_access_maxsz)
#define decode_open_downgrade_maxsz \
				(op_decode_hdr_maxsz + \
				 decode_stateid_maxsz)
#define encode_close_maxsz	(op_encode_hdr_maxsz + \
				 1 + encode_stateid_maxsz)
#define decode_close_maxsz	(op_decode_hdr_maxsz + \
				 decode_stateid_maxsz)
#define encode_setattr_maxsz	(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + \
				 encode_attrs_maxsz)
#define decode_setattr_maxsz	(op_decode_hdr_maxsz + \
				 nfs4_fattr_bitmap_maxsz)
#define encode_read_maxsz	(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + 3)
#define decode_read_maxsz	(op_decode_hdr_maxsz + 2)
#define encode_readdir_maxsz	(op_encode_hdr_maxsz + \
				 2 + encode_verifier_maxsz + 5)
#define decode_readdir_maxsz	(op_decode_hdr_maxsz + \
				 decode_verifier_maxsz)
#define encode_readlink_maxsz	(op_encode_hdr_maxsz)
#define decode_readlink_maxsz	(op_decode_hdr_maxsz + 1)
#define encode_write_maxsz	(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + 4)
#define decode_write_maxsz	(op_decode_hdr_maxsz + \
				 2 + decode_verifier_maxsz)
#define encode_commit_maxsz	(op_encode_hdr_maxsz + 3)
#define decode_commit_maxsz	(op_decode_hdr_maxsz + \
				 decode_verifier_maxsz)
L
Linus Torvalds 已提交
193 194
#define encode_remove_maxsz	(op_encode_hdr_maxsz + \
				nfs4_name_maxsz)
195 196
#define decode_remove_maxsz	(op_decode_hdr_maxsz + \
				 decode_change_info_maxsz)
L
Linus Torvalds 已提交
197 198
#define encode_rename_maxsz	(op_encode_hdr_maxsz + \
				2 * nfs4_name_maxsz)
199 200 201
#define decode_rename_maxsz	(op_decode_hdr_maxsz + \
				 decode_change_info_maxsz + \
				 decode_change_info_maxsz)
L
Linus Torvalds 已提交
202 203
#define encode_link_maxsz	(op_encode_hdr_maxsz + \
				nfs4_name_maxsz)
204
#define decode_link_maxsz	(op_decode_hdr_maxsz + decode_change_info_maxsz)
T
Trond Myklebust 已提交
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221
#define encode_lock_maxsz	(op_encode_hdr_maxsz + \
				 7 + \
				 1 + encode_stateid_maxsz + 8)
#define decode_lock_denied_maxsz \
				(8 + decode_lockowner_maxsz)
#define decode_lock_maxsz	(op_decode_hdr_maxsz + \
				 decode_lock_denied_maxsz)
#define encode_lockt_maxsz	(op_encode_hdr_maxsz + 12)
#define decode_lockt_maxsz	(op_decode_hdr_maxsz + \
				 decode_lock_denied_maxsz)
#define encode_locku_maxsz	(op_encode_hdr_maxsz + 3 + \
				 encode_stateid_maxsz + \
				 4)
#define decode_locku_maxsz	(op_decode_hdr_maxsz + \
				 decode_stateid_maxsz)
#define encode_access_maxsz	(op_encode_hdr_maxsz + 1)
#define decode_access_maxsz	(op_decode_hdr_maxsz + 2)
L
Linus Torvalds 已提交
222 223
#define encode_symlink_maxsz	(op_encode_hdr_maxsz + \
				1 + nfs4_name_maxsz + \
224
				1 + \
225
				nfs4_fattr_maxsz)
L
Linus Torvalds 已提交
226 227
#define decode_symlink_maxsz	(op_decode_hdr_maxsz + 8)
#define encode_create_maxsz	(op_encode_hdr_maxsz + \
T
Trond Myklebust 已提交
228 229
				1 + 2 + nfs4_name_maxsz + \
				encode_attrs_maxsz)
230 231 232
#define decode_create_maxsz	(op_decode_hdr_maxsz + \
				decode_change_info_maxsz + \
				nfs4_fattr_bitmap_maxsz)
T
Trond Myklebust 已提交
233 234
#define encode_statfs_maxsz	(encode_getattr_maxsz)
#define decode_statfs_maxsz	(decode_getattr_maxsz)
L
Linus Torvalds 已提交
235 236
#define encode_delegreturn_maxsz (op_encode_hdr_maxsz + 4)
#define decode_delegreturn_maxsz (op_decode_hdr_maxsz)
T
Trond Myklebust 已提交
237 238 239 240 241 242
#define encode_getacl_maxsz	(encode_getattr_maxsz)
#define decode_getacl_maxsz	(op_decode_hdr_maxsz + \
				 nfs4_fattr_bitmap_maxsz + 1)
#define encode_setacl_maxsz	(op_encode_hdr_maxsz + \
				 encode_stateid_maxsz + 3)
#define decode_setacl_maxsz	(decode_setattr_maxsz)
243 244 245 246
#define encode_fs_locations_maxsz \
				(encode_getattr_maxsz)
#define decode_fs_locations_maxsz \
				(0)
247 248

#if defined(CONFIG_NFS_V4_1)
A
Andy Adamson 已提交
249 250
#define NFS4_MAX_MACHINE_NAME_LEN (64)

B
Benny Halevy 已提交
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271
#define encode_exchange_id_maxsz (op_encode_hdr_maxsz + \
				encode_verifier_maxsz + \
				1 /* co_ownerid.len */ + \
				XDR_QUADLEN(NFS4_EXCHANGE_ID_LEN) + \
				1 /* flags */ + \
				1 /* spa_how */ + \
				0 /* SP4_NONE (for now) */ + \
				1 /* zero implemetation id array */)
#define decode_exchange_id_maxsz (op_decode_hdr_maxsz + \
				2 /* eir_clientid */ + \
				1 /* eir_sequenceid */ + \
				1 /* eir_flags */ + \
				1 /* spr_how */ + \
				0 /* SP4_NONE (for now) */ + \
				2 /* eir_server_owner.so_minor_id */ + \
				/* eir_server_owner.so_major_id<> */ \
				XDR_QUADLEN(NFS4_OPAQUE_LIMIT) + 1 + \
				/* eir_server_scope<> */ \
				XDR_QUADLEN(NFS4_OPAQUE_LIMIT) + 1 + \
				1 /* eir_server_impl_id array length */ + \
				0 /* ignored eir_server_impl_id contents */)
A
Andy Adamson 已提交
272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296
#define encode_channel_attrs_maxsz  (6 + 1 /* ca_rdma_ird.len (0) */)
#define decode_channel_attrs_maxsz  (6 + \
				     1 /* ca_rdma_ird.len */ + \
				     1 /* ca_rdma_ird */)
#define encode_create_session_maxsz  (op_encode_hdr_maxsz + \
				     2 /* csa_clientid */ + \
				     1 /* csa_sequence */ + \
				     1 /* csa_flags */ + \
				     encode_channel_attrs_maxsz + \
				     encode_channel_attrs_maxsz + \
				     1 /* csa_cb_program */ + \
				     1 /* csa_sec_parms.len (1) */ + \
				     1 /* cb_secflavor (AUTH_SYS) */ + \
				     1 /* stamp */ + \
				     1 /* machinename.len */ + \
				     XDR_QUADLEN(NFS4_MAX_MACHINE_NAME_LEN) + \
				     1 /* uid */ + \
				     1 /* gid */ + \
				     1 /* gids.len (0) */)
#define decode_create_session_maxsz  (op_decode_hdr_maxsz +	\
				     XDR_QUADLEN(NFS4_MAX_SESSIONID_LEN) + \
				     1 /* csr_sequence */ + \
				     1 /* csr_flags */ + \
				     decode_channel_attrs_maxsz + \
				     decode_channel_attrs_maxsz)
A
Andy Adamson 已提交
297 298
#define encode_destroy_session_maxsz    (op_encode_hdr_maxsz + 4)
#define decode_destroy_session_maxsz    (op_decode_hdr_maxsz)
A
Andy Adamson 已提交
299 300 301 302
#define encode_sequence_maxsz	(op_encode_hdr_maxsz + \
				XDR_QUADLEN(NFS4_MAX_SESSIONID_LEN) + 4)
#define decode_sequence_maxsz	(op_decode_hdr_maxsz + \
				XDR_QUADLEN(NFS4_MAX_SESSIONID_LEN) + 5)
303 304 305 306 307
#else /* CONFIG_NFS_V4_1 */
#define encode_sequence_maxsz	0
#define decode_sequence_maxsz	0
#endif /* CONFIG_NFS_V4_1 */

L
Linus Torvalds 已提交
308 309 310
#define NFS4_enc_compound_sz	(1024)  /* XXX: large enough? */
#define NFS4_dec_compound_sz	(1024)  /* XXX: large enough? */
#define NFS4_enc_read_sz	(compound_encode_hdr_maxsz + \
311
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
312
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
313
				encode_read_maxsz)
L
Linus Torvalds 已提交
314
#define NFS4_dec_read_sz	(compound_decode_hdr_maxsz + \
315
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
316
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
317
				decode_read_maxsz)
L
Linus Torvalds 已提交
318
#define NFS4_enc_readlink_sz	(compound_encode_hdr_maxsz + \
319
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
320
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
321
				encode_readlink_maxsz)
L
Linus Torvalds 已提交
322
#define NFS4_dec_readlink_sz	(compound_decode_hdr_maxsz + \
323
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
324
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
325
				decode_readlink_maxsz)
L
Linus Torvalds 已提交
326
#define NFS4_enc_readdir_sz	(compound_encode_hdr_maxsz + \
327
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
328
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
329
				encode_readdir_maxsz)
L
Linus Torvalds 已提交
330
#define NFS4_dec_readdir_sz	(compound_decode_hdr_maxsz + \
331
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
332
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
333
				decode_readdir_maxsz)
L
Linus Torvalds 已提交
334
#define NFS4_enc_write_sz	(compound_encode_hdr_maxsz + \
335
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
336
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
337
				encode_write_maxsz + \
338
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
339
#define NFS4_dec_write_sz	(compound_decode_hdr_maxsz + \
340
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
341
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
342
				decode_write_maxsz + \
343
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
344
#define NFS4_enc_commit_sz	(compound_encode_hdr_maxsz + \
345
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
346
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
347
				encode_commit_maxsz + \
348
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
349
#define NFS4_dec_commit_sz	(compound_decode_hdr_maxsz + \
350
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
351
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
352
				decode_commit_maxsz + \
353
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
354
#define NFS4_enc_open_sz        (compound_encode_hdr_maxsz + \
355
				encode_sequence_maxsz + \
356 357 358 359 360 361 362
				encode_putfh_maxsz + \
				encode_savefh_maxsz + \
				encode_open_maxsz + \
				encode_getfh_maxsz + \
				encode_getattr_maxsz + \
				encode_restorefh_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
363
#define NFS4_dec_open_sz        (compound_decode_hdr_maxsz + \
364
				decode_sequence_maxsz + \
365 366 367 368 369 370 371
				decode_putfh_maxsz + \
				decode_savefh_maxsz + \
				decode_open_maxsz + \
				decode_getfh_maxsz + \
				decode_getattr_maxsz + \
				decode_restorefh_maxsz + \
				decode_getattr_maxsz)
T
Trond Myklebust 已提交
372 373 374 375 376 377 378 379
#define NFS4_enc_open_confirm_sz \
				(compound_encode_hdr_maxsz + \
				 encode_putfh_maxsz + \
				 encode_open_confirm_maxsz)
#define NFS4_dec_open_confirm_sz \
				(compound_decode_hdr_maxsz + \
				 decode_putfh_maxsz + \
				 decode_open_confirm_maxsz)
L
Linus Torvalds 已提交
380
#define NFS4_enc_open_noattr_sz	(compound_encode_hdr_maxsz + \
381
					encode_sequence_maxsz + \
L
Linus Torvalds 已提交
382
					encode_putfh_maxsz + \
383 384
					encode_open_maxsz + \
					encode_getattr_maxsz)
L
Linus Torvalds 已提交
385
#define NFS4_dec_open_noattr_sz	(compound_decode_hdr_maxsz + \
386
					decode_sequence_maxsz + \
L
Linus Torvalds 已提交
387
					decode_putfh_maxsz + \
388 389
					decode_open_maxsz + \
					decode_getattr_maxsz)
L
Linus Torvalds 已提交
390 391
#define NFS4_enc_open_downgrade_sz \
				(compound_encode_hdr_maxsz + \
392
				 encode_sequence_maxsz + \
T
Trond Myklebust 已提交
393 394 395
				 encode_putfh_maxsz + \
				 encode_open_downgrade_maxsz + \
				 encode_getattr_maxsz)
L
Linus Torvalds 已提交
396 397
#define NFS4_dec_open_downgrade_sz \
				(compound_decode_hdr_maxsz + \
398
				 decode_sequence_maxsz + \
T
Trond Myklebust 已提交
399 400 401 402
				 decode_putfh_maxsz + \
				 decode_open_downgrade_maxsz + \
				 decode_getattr_maxsz)
#define NFS4_enc_close_sz	(compound_encode_hdr_maxsz + \
403
				 encode_sequence_maxsz + \
T
Trond Myklebust 已提交
404 405 406 407
				 encode_putfh_maxsz + \
				 encode_close_maxsz + \
				 encode_getattr_maxsz)
#define NFS4_dec_close_sz	(compound_decode_hdr_maxsz + \
408
				 decode_sequence_maxsz + \
T
Trond Myklebust 已提交
409 410 411 412
				 decode_putfh_maxsz + \
				 decode_close_maxsz + \
				 decode_getattr_maxsz)
#define NFS4_enc_setattr_sz	(compound_encode_hdr_maxsz + \
413
				 encode_sequence_maxsz + \
T
Trond Myklebust 已提交
414 415 416 417
				 encode_putfh_maxsz + \
				 encode_setattr_maxsz + \
				 encode_getattr_maxsz)
#define NFS4_dec_setattr_sz	(compound_decode_hdr_maxsz + \
418
				 decode_sequence_maxsz + \
T
Trond Myklebust 已提交
419 420 421
				 decode_putfh_maxsz + \
				 decode_setattr_maxsz + \
				 decode_getattr_maxsz)
L
Linus Torvalds 已提交
422
#define NFS4_enc_fsinfo_sz	(compound_encode_hdr_maxsz + \
423
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
424 425 426
				encode_putfh_maxsz + \
				encode_fsinfo_maxsz)
#define NFS4_dec_fsinfo_sz	(compound_decode_hdr_maxsz + \
427
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448
				decode_putfh_maxsz + \
				decode_fsinfo_maxsz)
#define NFS4_enc_renew_sz	(compound_encode_hdr_maxsz + \
				encode_renew_maxsz)
#define NFS4_dec_renew_sz	(compound_decode_hdr_maxsz + \
				decode_renew_maxsz)
#define NFS4_enc_setclientid_sz	(compound_encode_hdr_maxsz + \
				encode_setclientid_maxsz)
#define NFS4_dec_setclientid_sz	(compound_decode_hdr_maxsz + \
				decode_setclientid_maxsz)
#define NFS4_enc_setclientid_confirm_sz \
				(compound_encode_hdr_maxsz + \
				encode_setclientid_confirm_maxsz + \
				encode_putrootfh_maxsz + \
				encode_fsinfo_maxsz)
#define NFS4_dec_setclientid_confirm_sz \
				(compound_decode_hdr_maxsz + \
				decode_setclientid_confirm_maxsz + \
				decode_putrootfh_maxsz + \
				decode_fsinfo_maxsz)
#define NFS4_enc_lock_sz        (compound_encode_hdr_maxsz + \
449
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
450
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
451
				encode_lock_maxsz)
L
Linus Torvalds 已提交
452
#define NFS4_dec_lock_sz        (compound_decode_hdr_maxsz + \
453
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
454
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
455
				decode_lock_maxsz)
L
Linus Torvalds 已提交
456
#define NFS4_enc_lockt_sz       (compound_encode_hdr_maxsz + \
457
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
458
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
459 460
				encode_lockt_maxsz)
#define NFS4_dec_lockt_sz       (compound_decode_hdr_maxsz + \
461
				 decode_sequence_maxsz + \
T
Trond Myklebust 已提交
462 463
				 decode_putfh_maxsz + \
				 decode_lockt_maxsz)
L
Linus Torvalds 已提交
464
#define NFS4_enc_locku_sz       (compound_encode_hdr_maxsz + \
465
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
466
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
467
				encode_locku_maxsz)
L
Linus Torvalds 已提交
468
#define NFS4_dec_locku_sz       (compound_decode_hdr_maxsz + \
469
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
470
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
471
				decode_locku_maxsz)
L
Linus Torvalds 已提交
472
#define NFS4_enc_access_sz	(compound_encode_hdr_maxsz + \
473
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
474
				encode_putfh_maxsz + \
475 476
				encode_access_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
477
#define NFS4_dec_access_sz	(compound_decode_hdr_maxsz + \
478
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
479
				decode_putfh_maxsz + \
480 481
				decode_access_maxsz + \
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
482
#define NFS4_enc_getattr_sz	(compound_encode_hdr_maxsz + \
483
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
484 485 486
				encode_putfh_maxsz + \
				encode_getattr_maxsz)
#define NFS4_dec_getattr_sz	(compound_decode_hdr_maxsz + \
487
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
488 489 490
				decode_putfh_maxsz + \
				decode_getattr_maxsz)
#define NFS4_enc_lookup_sz	(compound_encode_hdr_maxsz + \
491
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
492 493 494 495 496
				encode_putfh_maxsz + \
				encode_lookup_maxsz + \
				encode_getattr_maxsz + \
				encode_getfh_maxsz)
#define NFS4_dec_lookup_sz	(compound_decode_hdr_maxsz + \
497
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
498
				decode_putfh_maxsz + \
499
				decode_lookup_maxsz + \
L
Linus Torvalds 已提交
500 501 502
				decode_getattr_maxsz + \
				decode_getfh_maxsz)
#define NFS4_enc_lookup_root_sz (compound_encode_hdr_maxsz + \
503
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
504 505 506 507
				encode_putrootfh_maxsz + \
				encode_getattr_maxsz + \
				encode_getfh_maxsz)
#define NFS4_dec_lookup_root_sz (compound_decode_hdr_maxsz + \
508
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
509 510 511 512
				decode_putrootfh_maxsz + \
				decode_getattr_maxsz + \
				decode_getfh_maxsz)
#define NFS4_enc_remove_sz	(compound_encode_hdr_maxsz + \
513
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
514
				encode_putfh_maxsz + \
515 516
				encode_remove_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
517
#define NFS4_dec_remove_sz	(compound_decode_hdr_maxsz + \
518
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
519
				decode_putfh_maxsz + \
520
				decode_remove_maxsz + \
521
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
522
#define NFS4_enc_rename_sz	(compound_encode_hdr_maxsz + \
523
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
524 525 526
				encode_putfh_maxsz + \
				encode_savefh_maxsz + \
				encode_putfh_maxsz + \
527 528 529 530
				encode_rename_maxsz + \
				encode_getattr_maxsz + \
				encode_restorefh_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
531
#define NFS4_dec_rename_sz	(compound_decode_hdr_maxsz + \
532
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
533 534 535
				decode_putfh_maxsz + \
				decode_savefh_maxsz + \
				decode_putfh_maxsz + \
536 537 538 539
				decode_rename_maxsz + \
				decode_getattr_maxsz + \
				decode_restorefh_maxsz + \
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
540
#define NFS4_enc_link_sz	(compound_encode_hdr_maxsz + \
541
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
542 543 544
				encode_putfh_maxsz + \
				encode_savefh_maxsz + \
				encode_putfh_maxsz + \
545 546 547 548
				encode_link_maxsz + \
				decode_getattr_maxsz + \
				encode_restorefh_maxsz + \
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
549
#define NFS4_dec_link_sz	(compound_decode_hdr_maxsz + \
550
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
551 552 553
				decode_putfh_maxsz + \
				decode_savefh_maxsz + \
				decode_putfh_maxsz + \
554 555 556 557
				decode_link_maxsz + \
				decode_getattr_maxsz + \
				decode_restorefh_maxsz + \
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
558
#define NFS4_enc_symlink_sz	(compound_encode_hdr_maxsz + \
559
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
560 561 562 563 564
				encode_putfh_maxsz + \
				encode_symlink_maxsz + \
				encode_getattr_maxsz + \
				encode_getfh_maxsz)
#define NFS4_dec_symlink_sz	(compound_decode_hdr_maxsz + \
565
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
566 567 568 569 570
				decode_putfh_maxsz + \
				decode_symlink_maxsz + \
				decode_getattr_maxsz + \
				decode_getfh_maxsz)
#define NFS4_enc_create_sz	(compound_encode_hdr_maxsz + \
571
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
572
				encode_putfh_maxsz + \
573
				encode_savefh_maxsz + \
L
Linus Torvalds 已提交
574
				encode_create_maxsz + \
575
				encode_getfh_maxsz + \
L
Linus Torvalds 已提交
576
				encode_getattr_maxsz + \
577 578
				encode_restorefh_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
579
#define NFS4_dec_create_sz	(compound_decode_hdr_maxsz + \
580
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
581
				decode_putfh_maxsz + \
582
				decode_savefh_maxsz + \
L
Linus Torvalds 已提交
583
				decode_create_maxsz + \
584
				decode_getfh_maxsz + \
L
Linus Torvalds 已提交
585
				decode_getattr_maxsz + \
586 587
				decode_restorefh_maxsz + \
				decode_getattr_maxsz)
L
Linus Torvalds 已提交
588
#define NFS4_enc_pathconf_sz	(compound_encode_hdr_maxsz + \
589
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
590 591 592
				encode_putfh_maxsz + \
				encode_getattr_maxsz)
#define NFS4_dec_pathconf_sz	(compound_decode_hdr_maxsz + \
593
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
594 595 596
				decode_putfh_maxsz + \
				decode_getattr_maxsz)
#define NFS4_enc_statfs_sz	(compound_encode_hdr_maxsz + \
597
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
598
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
599
				encode_statfs_maxsz)
L
Linus Torvalds 已提交
600
#define NFS4_dec_statfs_sz	(compound_decode_hdr_maxsz + \
601
				decode_sequence_maxsz + \
L
Linus Torvalds 已提交
602
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
603
				decode_statfs_maxsz)
L
Linus Torvalds 已提交
604
#define NFS4_enc_server_caps_sz (compound_encode_hdr_maxsz + \
605
				encode_sequence_maxsz + \
606
				encode_putfh_maxsz + \
L
Linus Torvalds 已提交
607 608
				encode_getattr_maxsz)
#define NFS4_dec_server_caps_sz (compound_decode_hdr_maxsz + \
609
				decode_sequence_maxsz + \
610
				decode_putfh_maxsz + \
L
Linus Torvalds 已提交
611 612
				decode_getattr_maxsz)
#define NFS4_enc_delegreturn_sz	(compound_encode_hdr_maxsz + \
613
				encode_sequence_maxsz + \
L
Linus Torvalds 已提交
614
				encode_putfh_maxsz + \
615 616
				encode_delegreturn_maxsz + \
				encode_getattr_maxsz)
L
Linus Torvalds 已提交
617
#define NFS4_dec_delegreturn_sz (compound_decode_hdr_maxsz + \
618
				decode_sequence_maxsz + \
619 620
				decode_delegreturn_maxsz + \
				decode_getattr_maxsz)
621
#define NFS4_enc_getacl_sz	(compound_encode_hdr_maxsz + \
622
				encode_sequence_maxsz + \
623
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
624
				encode_getacl_maxsz)
625
#define NFS4_dec_getacl_sz	(compound_decode_hdr_maxsz + \
626
				decode_sequence_maxsz + \
627
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
628
				decode_getacl_maxsz)
629
#define NFS4_enc_setacl_sz	(compound_encode_hdr_maxsz + \
630
				encode_sequence_maxsz + \
631
				encode_putfh_maxsz + \
T
Trond Myklebust 已提交
632
				encode_setacl_maxsz)
633
#define NFS4_dec_setacl_sz	(compound_decode_hdr_maxsz + \
634
				decode_sequence_maxsz + \
635
				decode_putfh_maxsz + \
T
Trond Myklebust 已提交
636
				decode_setacl_maxsz)
637 638
#define NFS4_enc_fs_locations_sz \
				(compound_encode_hdr_maxsz + \
639
				 encode_sequence_maxsz + \
640
				 encode_putfh_maxsz + \
641 642
				 encode_lookup_maxsz + \
				 encode_fs_locations_maxsz)
643 644
#define NFS4_dec_fs_locations_sz \
				(compound_decode_hdr_maxsz + \
645
				 decode_sequence_maxsz + \
646
				 decode_putfh_maxsz + \
647 648
				 decode_lookup_maxsz + \
				 decode_fs_locations_maxsz)
B
Benny Halevy 已提交
649 650 651 652 653 654 655
#if defined(CONFIG_NFS_V4_1)
#define NFS4_enc_exchange_id_sz \
				(compound_encode_hdr_maxsz + \
				 encode_exchange_id_maxsz)
#define NFS4_dec_exchange_id_sz \
				(compound_decode_hdr_maxsz + \
				 decode_exchange_id_maxsz)
A
Andy Adamson 已提交
656 657 658 659 660 661
#define NFS4_enc_create_session_sz \
				(compound_encode_hdr_maxsz + \
				 encode_create_session_maxsz)
#define NFS4_dec_create_session_sz \
				(compound_decode_hdr_maxsz + \
				 decode_create_session_maxsz)
A
Andy Adamson 已提交
662 663 664 665
#define NFS4_enc_destroy_session_sz	(compound_encode_hdr_maxsz + \
					 encode_destroy_session_maxsz)
#define NFS4_dec_destroy_session_sz	(compound_decode_hdr_maxsz + \
					 decode_destroy_session_maxsz)
A
Andy Adamson 已提交
666 667 668 669 670 671
#define NFS4_enc_sequence_sz \
				(compound_decode_hdr_maxsz + \
				 encode_sequence_maxsz)
#define NFS4_dec_sequence_sz \
				(compound_decode_hdr_maxsz + \
				 decode_sequence_maxsz)
A
Andy Adamson 已提交
672 673 674 675 676 677 678 679
#define NFS4_enc_get_lease_time_sz	(compound_encode_hdr_maxsz + \
					 encode_sequence_maxsz + \
					 encode_putrootfh_maxsz + \
					 encode_fsinfo_maxsz)
#define NFS4_dec_get_lease_time_sz	(compound_decode_hdr_maxsz + \
					 decode_sequence_maxsz + \
					 decode_putrootfh_maxsz + \
					 decode_fsinfo_maxsz)
B
Benny Halevy 已提交
680
#endif /* CONFIG_NFS_V4_1 */
L
Linus Torvalds 已提交
681

682 683 684 685 686 687 688 689 690 691 692
static const umode_t nfs_type2fmt[] = {
	[NF4BAD] = 0,
	[NF4REG] = S_IFREG,
	[NF4DIR] = S_IFDIR,
	[NF4BLK] = S_IFBLK,
	[NF4CHR] = S_IFCHR,
	[NF4LNK] = S_IFLNK,
	[NF4SOCK] = S_IFSOCK,
	[NF4FIFO] = S_IFIFO,
	[NF4ATTRDIR] = 0,
	[NF4NAMEDATTR] = 0,
L
Linus Torvalds 已提交
693 694 695 696 697
};

struct compound_hdr {
	int32_t		status;
	uint32_t	nops;
698
	__be32 *	nops_p;
L
Linus Torvalds 已提交
699 700
	uint32_t	taglen;
	char *		tag;
701
	uint32_t	replen;		/* expected reply words */
702
	u32		minorversion;
L
Linus Torvalds 已提交
703 704
};

705 706 707 708 709 710
static __be32 *reserve_space(struct xdr_stream *xdr, size_t nbytes)
{
	__be32 *p = xdr_reserve_space(xdr, nbytes);
	BUG_ON(!p);
	return p;
}
L
Linus Torvalds 已提交
711 712 713

static void encode_string(struct xdr_stream *xdr, unsigned int len, const char *str)
{
A
Al Viro 已提交
714
	__be32 *p;
L
Linus Torvalds 已提交
715 716 717 718 719 720

	p = xdr_reserve_space(xdr, 4 + len);
	BUG_ON(p == NULL);
	xdr_encode_opaque(p, str, len);
}

721 722 723
static void encode_compound_hdr(struct xdr_stream *xdr,
				struct rpc_rqst *req,
				struct compound_hdr *hdr)
L
Linus Torvalds 已提交
724
{
A
Al Viro 已提交
725
	__be32 *p;
726 727 728 729 730 731
	struct rpc_auth *auth = req->rq_task->tk_msg.rpc_cred->cr_auth;

	/* initialize running count of expected bytes in reply.
	 * NOTE: the replied tag SHOULD be the same is the one sent,
	 * but this is not required as a MUST for the server to do so. */
	hdr->replen = RPC_REPHDRSIZE + auth->au_rslack + 3 + hdr->taglen;
L
Linus Torvalds 已提交
732 733 734

	dprintk("encode_compound: tag=%.*s\n", (int)hdr->taglen, hdr->tag);
	BUG_ON(hdr->taglen > NFS4_MAXTAGLEN);
735
	p = reserve_space(xdr, 12 + hdr->taglen);
B
Benny Halevy 已提交
736
	*p++ = cpu_to_be32(hdr->taglen);
B
Benny Halevy 已提交
737
	p = xdr_encode_opaque_fixed(p, hdr->tag, hdr->taglen);
B
Benny Halevy 已提交
738
	*p++ = cpu_to_be32(hdr->minorversion);
739
	hdr->nops_p = p;
B
Benny Halevy 已提交
740
	*p++ = cpu_to_be32(hdr->nops);
741 742 743 744
}

static void encode_nops(struct compound_hdr *hdr)
{
A
Andy Adamson 已提交
745
	BUG_ON(hdr->nops > NFS4_MAX_OPS);
746
	*hdr->nops_p = htonl(hdr->nops);
L
Linus Torvalds 已提交
747 748 749 750
}

static void encode_nfs4_verifier(struct xdr_stream *xdr, const nfs4_verifier *verf)
{
A
Al Viro 已提交
751
	__be32 *p;
L
Linus Torvalds 已提交
752 753 754 755 756 757

	p = xdr_reserve_space(xdr, NFS4_VERIFIER_SIZE);
	BUG_ON(p == NULL);
	xdr_encode_opaque_fixed(p, verf->data, NFS4_VERIFIER_SIZE);
}

758
static void encode_attrs(struct xdr_stream *xdr, const struct iattr *iap, const struct nfs_server *server)
L
Linus Torvalds 已提交
759 760 761 762 763
{
	char owner_name[IDMAP_NAMESZ];
	char owner_group[IDMAP_NAMESZ];
	int owner_namelen = 0;
	int owner_grouplen = 0;
A
Al Viro 已提交
764 765
	__be32 *p;
	__be32 *q;
L
Linus Torvalds 已提交
766 767 768 769 770 771 772 773 774
	int len;
	uint32_t bmval0 = 0;
	uint32_t bmval1 = 0;

	/*
	 * We reserve enough space to write the entire attribute buffer at once.
	 * In the worst-case, this would be
	 *   12(bitmap) + 4(attrlen) + 8(size) + 4(mode) + 4(atime) + 4(mtime)
	 *          = 36 bytes, plus any contribution from variable-length fields
775
	 *            such as owner/group.
L
Linus Torvalds 已提交
776 777 778 779 780 781 782 783 784
	 */
	len = 16;

	/* Sigh */
	if (iap->ia_valid & ATTR_SIZE)
		len += 8;
	if (iap->ia_valid & ATTR_MODE)
		len += 4;
	if (iap->ia_valid & ATTR_UID) {
785
		owner_namelen = nfs_map_uid_to_name(server->nfs_client, iap->ia_uid, owner_name);
L
Linus Torvalds 已提交
786
		if (owner_namelen < 0) {
787 788
			dprintk("nfs: couldn't resolve uid %d to string\n",
					iap->ia_uid);
L
Linus Torvalds 已提交
789 790 791 792 793 794 795 796
			/* XXX */
			strcpy(owner_name, "nobody");
			owner_namelen = sizeof("nobody") - 1;
			/* goto out; */
		}
		len += 4 + (XDR_QUADLEN(owner_namelen) << 2);
	}
	if (iap->ia_valid & ATTR_GID) {
797
		owner_grouplen = nfs_map_gid_to_group(server->nfs_client, iap->ia_gid, owner_group);
L
Linus Torvalds 已提交
798
		if (owner_grouplen < 0) {
799 800
			dprintk("nfs: couldn't resolve gid %d to string\n",
					iap->ia_gid);
L
Linus Torvalds 已提交
801 802 803 804 805 806 807 808 809 810 811 812 813 814
			strcpy(owner_group, "nobody");
			owner_grouplen = sizeof("nobody") - 1;
			/* goto out; */
		}
		len += 4 + (XDR_QUADLEN(owner_grouplen) << 2);
	}
	if (iap->ia_valid & ATTR_ATIME_SET)
		len += 16;
	else if (iap->ia_valid & ATTR_ATIME)
		len += 4;
	if (iap->ia_valid & ATTR_MTIME_SET)
		len += 16;
	else if (iap->ia_valid & ATTR_MTIME)
		len += 4;
815
	p = reserve_space(xdr, len);
L
Linus Torvalds 已提交
816 817 818 819 820

	/*
	 * We write the bitmap length now, but leave the bitmap and the attribute
	 * buffer length to be backfilled at the end of this routine.
	 */
B
Benny Halevy 已提交
821
	*p++ = cpu_to_be32(2);
L
Linus Torvalds 已提交
822 823 824 825 826
	q = p;
	p += 3;

	if (iap->ia_valid & ATTR_SIZE) {
		bmval0 |= FATTR4_WORD0_SIZE;
B
Benny Halevy 已提交
827
		p = xdr_encode_hyper(p, iap->ia_size);
L
Linus Torvalds 已提交
828 829 830
	}
	if (iap->ia_valid & ATTR_MODE) {
		bmval1 |= FATTR4_WORD1_MODE;
B
Benny Halevy 已提交
831
		*p++ = cpu_to_be32(iap->ia_mode & S_IALLUGO);
L
Linus Torvalds 已提交
832 833 834
	}
	if (iap->ia_valid & ATTR_UID) {
		bmval1 |= FATTR4_WORD1_OWNER;
B
Benny Halevy 已提交
835
		*p++ = cpu_to_be32(owner_namelen);
B
Benny Halevy 已提交
836
		p = xdr_encode_opaque_fixed(p, owner_name, owner_namelen);
L
Linus Torvalds 已提交
837 838 839
	}
	if (iap->ia_valid & ATTR_GID) {
		bmval1 |= FATTR4_WORD1_OWNER_GROUP;
B
Benny Halevy 已提交
840
		*p++ = cpu_to_be32(owner_grouplen);
B
Benny Halevy 已提交
841
		p = xdr_encode_opaque_fixed(p, owner_group, owner_grouplen);
L
Linus Torvalds 已提交
842 843 844
	}
	if (iap->ia_valid & ATTR_ATIME_SET) {
		bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET;
B
Benny Halevy 已提交
845 846 847 848
		*p++ = cpu_to_be32(NFS4_SET_TO_CLIENT_TIME);
		*p++ = cpu_to_be32(0);
		*p++ = cpu_to_be32(iap->ia_mtime.tv_sec);
		*p++ = cpu_to_be32(iap->ia_mtime.tv_nsec);
L
Linus Torvalds 已提交
849 850 851
	}
	else if (iap->ia_valid & ATTR_ATIME) {
		bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET;
B
Benny Halevy 已提交
852
		*p++ = cpu_to_be32(NFS4_SET_TO_SERVER_TIME);
L
Linus Torvalds 已提交
853 854 855
	}
	if (iap->ia_valid & ATTR_MTIME_SET) {
		bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET;
B
Benny Halevy 已提交
856 857 858 859
		*p++ = cpu_to_be32(NFS4_SET_TO_CLIENT_TIME);
		*p++ = cpu_to_be32(0);
		*p++ = cpu_to_be32(iap->ia_mtime.tv_sec);
		*p++ = cpu_to_be32(iap->ia_mtime.tv_nsec);
L
Linus Torvalds 已提交
860 861 862
	}
	else if (iap->ia_valid & ATTR_MTIME) {
		bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET;
B
Benny Halevy 已提交
863
		*p++ = cpu_to_be32(NFS4_SET_TO_SERVER_TIME);
L
Linus Torvalds 已提交
864
	}
865

L
Linus Torvalds 已提交
866 867 868 869
	/*
	 * Now we backfill the bitmap and the attribute buffer length.
	 */
	if (len != ((char *)p - (char *)q) + 4) {
870
		printk(KERN_ERR "nfs: Attr length error, %u != %Zu\n",
L
Linus Torvalds 已提交
871 872 873 874 875 876 877 878 879 880 881
				len, ((char *)p - (char *)q) + 4);
		BUG();
	}
	len = (char *)p - (char *)q - 12;
	*q++ = htonl(bmval0);
	*q++ = htonl(bmval1);
	*q++ = htonl(len);

/* out: */
}

882
static void encode_access(struct xdr_stream *xdr, u32 access, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
883
{
A
Al Viro 已提交
884
	__be32 *p;
L
Linus Torvalds 已提交
885

886
	p = reserve_space(xdr, 8);
B
Benny Halevy 已提交
887 888
	*p++ = cpu_to_be32(OP_ACCESS);
	*p++ = cpu_to_be32(access);
889
	hdr->nops++;
890
	hdr->replen += decode_access_maxsz;
L
Linus Torvalds 已提交
891 892
}

893
static void encode_close(struct xdr_stream *xdr, const struct nfs_closeargs *arg, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
894
{
A
Al Viro 已提交
895
	__be32 *p;
L
Linus Torvalds 已提交
896

897
	p = reserve_space(xdr, 8+NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
898 899
	*p++ = cpu_to_be32(OP_CLOSE);
	*p++ = cpu_to_be32(arg->seqid->sequence->counter);
B
Benny Halevy 已提交
900
	p = xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE);
901
	hdr->nops++;
902
	hdr->replen += decode_close_maxsz;
L
Linus Torvalds 已提交
903 904
}

905
static void encode_commit(struct xdr_stream *xdr, const struct nfs_writeargs *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
906
{
A
Al Viro 已提交
907
	__be32 *p;
908

909
	p = reserve_space(xdr, 16);
B
Benny Halevy 已提交
910
	*p++ = cpu_to_be32(OP_COMMIT);
B
Benny Halevy 已提交
911
	p = xdr_encode_hyper(p, args->offset);
B
Benny Halevy 已提交
912
	*p++ = cpu_to_be32(args->count);
913
	hdr->nops++;
914
	hdr->replen += decode_commit_maxsz;
L
Linus Torvalds 已提交
915 916
}

917
static void encode_create(struct xdr_stream *xdr, const struct nfs4_create_arg *create, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
918
{
A
Al Viro 已提交
919
	__be32 *p;
920

921
	p = reserve_space(xdr, 8);
B
Benny Halevy 已提交
922 923
	*p++ = cpu_to_be32(OP_CREATE);
	*p++ = cpu_to_be32(create->ftype);
L
Linus Torvalds 已提交
924 925 926

	switch (create->ftype) {
	case NF4LNK:
927
		p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
928
		*p++ = cpu_to_be32(create->u.symlink.len);
929
		xdr_write_pages(xdr, create->u.symlink.pages, 0, create->u.symlink.len);
L
Linus Torvalds 已提交
930 931 932
		break;

	case NF4BLK: case NF4CHR:
933
		p = reserve_space(xdr, 8);
B
Benny Halevy 已提交
934 935
		*p++ = cpu_to_be32(create->u.device.specdata1);
		*p++ = cpu_to_be32(create->u.device.specdata2);
L
Linus Torvalds 已提交
936 937 938 939 940 941
		break;

	default:
		break;
	}

942
	p = reserve_space(xdr, 4 + create->name->len);
B
Benny Halevy 已提交
943
	*p++ = cpu_to_be32(create->name->len);
B
Benny Halevy 已提交
944
	p = xdr_encode_opaque_fixed(p, create->name->name, create->name->len);
945
	hdr->nops++;
946
	hdr->replen += decode_create_maxsz;
L
Linus Torvalds 已提交
947

948
	encode_attrs(xdr, create->attrs, create->server);
L
Linus Torvalds 已提交
949 950
}

951
static void encode_getattr_one(struct xdr_stream *xdr, uint32_t bitmap, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
952
{
A
Andy Adamson 已提交
953
	__be32 *p;
L
Linus Torvalds 已提交
954

955
	p = reserve_space(xdr, 12);
B
Benny Halevy 已提交
956 957 958
	*p++ = cpu_to_be32(OP_GETATTR);
	*p++ = cpu_to_be32(1);
	*p++ = cpu_to_be32(bitmap);
959
	hdr->nops++;
960
	hdr->replen += decode_getattr_maxsz;
L
Linus Torvalds 已提交
961 962
}

963
static void encode_getattr_two(struct xdr_stream *xdr, uint32_t bm0, uint32_t bm1, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
964
{
A
Andy Adamson 已提交
965
	__be32 *p;
L
Linus Torvalds 已提交
966

967
	p = reserve_space(xdr, 16);
B
Benny Halevy 已提交
968 969 970 971
	*p++ = cpu_to_be32(OP_GETATTR);
	*p++ = cpu_to_be32(2);
	*p++ = cpu_to_be32(bm0);
	*p++ = cpu_to_be32(bm1);
972
	hdr->nops++;
973
	hdr->replen += decode_getattr_maxsz;
L
Linus Torvalds 已提交
974 975
}

976
static void encode_getfattr(struct xdr_stream *xdr, const u32* bitmask, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
977
{
978 979
	encode_getattr_two(xdr, bitmask[0] & nfs4_fattr_bitmap[0],
			   bitmask[1] & nfs4_fattr_bitmap[1], hdr);
L
Linus Torvalds 已提交
980 981
}

982
static void encode_fsinfo(struct xdr_stream *xdr, const u32* bitmask, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
983
{
984 985
	encode_getattr_two(xdr, bitmask[0] & nfs4_fsinfo_bitmap[0],
			   bitmask[1] & nfs4_fsinfo_bitmap[1], hdr);
L
Linus Torvalds 已提交
986 987
}

988
static void encode_fs_locations(struct xdr_stream *xdr, const u32* bitmask, struct compound_hdr *hdr)
989
{
990 991
	encode_getattr_two(xdr, bitmask[0] & nfs4_fs_locations_bitmap[0],
			   bitmask[1] & nfs4_fs_locations_bitmap[1], hdr);
992 993
}

994
static void encode_getfh(struct xdr_stream *xdr, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
995
{
A
Al Viro 已提交
996
	__be32 *p;
L
Linus Torvalds 已提交
997

998
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
999
	*p++ = cpu_to_be32(OP_GETFH);
1000
	hdr->nops++;
1001
	hdr->replen += decode_getfh_maxsz;
L
Linus Torvalds 已提交
1002 1003
}

1004
static void encode_link(struct xdr_stream *xdr, const struct qstr *name, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1005
{
A
Al Viro 已提交
1006
	__be32 *p;
L
Linus Torvalds 已提交
1007

1008
	p = reserve_space(xdr, 8 + name->len);
B
Benny Halevy 已提交
1009 1010
	*p++ = cpu_to_be32(OP_LINK);
	*p++ = cpu_to_be32(name->len);
B
Benny Halevy 已提交
1011
	p = xdr_encode_opaque_fixed(p, name->name, name->len);
1012
	hdr->nops++;
1013
	hdr->replen += decode_link_maxsz;
L
Linus Torvalds 已提交
1014 1015
}

T
Trond Myklebust 已提交
1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029
static inline int nfs4_lock_type(struct file_lock *fl, int block)
{
	if ((fl->fl_type & (F_RDLCK|F_WRLCK|F_UNLCK)) == F_RDLCK)
		return block ? NFS4_READW_LT : NFS4_READ_LT;
	return block ? NFS4_WRITEW_LT : NFS4_WRITE_LT;
}

static inline uint64_t nfs4_lock_length(struct file_lock *fl)
{
	if (fl->fl_end == OFFSET_MAX)
		return ~(uint64_t)0;
	return fl->fl_end - fl->fl_start + 1;
}

L
Linus Torvalds 已提交
1030 1031 1032 1033
/*
 * opcode,type,reclaim,offset,length,new_lock_owner = 32
 * open_seqid,open_stateid,lock_seqid,lock_owner.clientid, lock_owner.id = 40
 */
1034
static void encode_lock(struct xdr_stream *xdr, const struct nfs_lock_args *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1035
{
A
Al Viro 已提交
1036
	__be32 *p;
L
Linus Torvalds 已提交
1037

1038
	p = reserve_space(xdr, 32);
B
Benny Halevy 已提交
1039 1040 1041
	*p++ = cpu_to_be32(OP_LOCK);
	*p++ = cpu_to_be32(nfs4_lock_type(args->fl, args->block));
	*p++ = cpu_to_be32(args->reclaim);
B
Benny Halevy 已提交
1042 1043
	p = xdr_encode_hyper(p, args->fl->fl_start);
	p = xdr_encode_hyper(p, nfs4_lock_length(args->fl));
B
Benny Halevy 已提交
1044
	*p++ = cpu_to_be32(args->new_lock_owner);
T
Trond Myklebust 已提交
1045
	if (args->new_lock_owner){
1046
		p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+32);
B
Benny Halevy 已提交
1047
		*p++ = cpu_to_be32(args->open_seqid->sequence->counter);
B
Benny Halevy 已提交
1048
		p = xdr_encode_opaque_fixed(p, args->open_stateid->data, NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1049
		*p++ = cpu_to_be32(args->lock_seqid->sequence->counter);
B
Benny Halevy 已提交
1050
		p = xdr_encode_hyper(p, args->lock_owner.clientid);
B
Benny Halevy 已提交
1051
		*p++ = cpu_to_be32(16);
B
Benny Halevy 已提交
1052
		p = xdr_encode_opaque_fixed(p, "lock id:", 8);
B
Benny Halevy 已提交
1053
		p = xdr_encode_hyper(p, args->lock_owner.id);
L
Linus Torvalds 已提交
1054 1055
	}
	else {
1056
		p = reserve_space(xdr, NFS4_STATEID_SIZE+4);
B
Benny Halevy 已提交
1057
		p = xdr_encode_opaque_fixed(p, args->lock_stateid->data, NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1058
		*p++ = cpu_to_be32(args->lock_seqid->sequence->counter);
L
Linus Torvalds 已提交
1059
	}
1060
	hdr->nops++;
1061
	hdr->replen += decode_lock_maxsz;
L
Linus Torvalds 已提交
1062 1063
}

1064
static void encode_lockt(struct xdr_stream *xdr, const struct nfs_lockt_args *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1065
{
A
Al Viro 已提交
1066
	__be32 *p;
L
Linus Torvalds 已提交
1067

1068
	p = reserve_space(xdr, 52);
B
Benny Halevy 已提交
1069 1070
	*p++ = cpu_to_be32(OP_LOCKT);
	*p++ = cpu_to_be32(nfs4_lock_type(args->fl, 0));
B
Benny Halevy 已提交
1071 1072 1073
	p = xdr_encode_hyper(p, args->fl->fl_start);
	p = xdr_encode_hyper(p, nfs4_lock_length(args->fl));
	p = xdr_encode_hyper(p, args->lock_owner.clientid);
B
Benny Halevy 已提交
1074
	*p++ = cpu_to_be32(16);
B
Benny Halevy 已提交
1075
	p = xdr_encode_opaque_fixed(p, "lock id:", 8);
B
Benny Halevy 已提交
1076
	p = xdr_encode_hyper(p, args->lock_owner.id);
1077
	hdr->nops++;
1078
	hdr->replen += decode_lockt_maxsz;
L
Linus Torvalds 已提交
1079 1080
}

1081
static void encode_locku(struct xdr_stream *xdr, const struct nfs_locku_args *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1082
{
A
Al Viro 已提交
1083
	__be32 *p;
L
Linus Torvalds 已提交
1084

1085
	p = reserve_space(xdr, 12+NFS4_STATEID_SIZE+16);
B
Benny Halevy 已提交
1086 1087 1088
	*p++ = cpu_to_be32(OP_LOCKU);
	*p++ = cpu_to_be32(nfs4_lock_type(args->fl, 0));
	*p++ = cpu_to_be32(args->seqid->sequence->counter);
B
Benny Halevy 已提交
1089
	p = xdr_encode_opaque_fixed(p, args->stateid->data, NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1090 1091
	p = xdr_encode_hyper(p, args->fl->fl_start);
	p = xdr_encode_hyper(p, nfs4_lock_length(args->fl));
1092
	hdr->nops++;
1093
	hdr->replen += decode_locku_maxsz;
L
Linus Torvalds 已提交
1094 1095
}

1096
static void encode_lookup(struct xdr_stream *xdr, const struct qstr *name, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1097 1098
{
	int len = name->len;
A
Al Viro 已提交
1099
	__be32 *p;
L
Linus Torvalds 已提交
1100

1101
	p = reserve_space(xdr, 8 + len);
B
Benny Halevy 已提交
1102 1103
	*p++ = cpu_to_be32(OP_LOOKUP);
	*p++ = cpu_to_be32(len);
B
Benny Halevy 已提交
1104
	p = xdr_encode_opaque_fixed(p, name->name, len);
1105
	hdr->nops++;
1106
	hdr->replen += decode_lookup_maxsz;
L
Linus Torvalds 已提交
1107 1108
}

1109
static void encode_share_access(struct xdr_stream *xdr, fmode_t fmode)
L
Linus Torvalds 已提交
1110
{
A
Al Viro 已提交
1111
	__be32 *p;
L
Linus Torvalds 已提交
1112

1113
	p = reserve_space(xdr, 8);
1114
	switch (fmode & (FMODE_READ|FMODE_WRITE)) {
A
Andy Adamson 已提交
1115
	case FMODE_READ:
B
Benny Halevy 已提交
1116
		*p++ = cpu_to_be32(NFS4_SHARE_ACCESS_READ);
A
Andy Adamson 已提交
1117 1118
		break;
	case FMODE_WRITE:
B
Benny Halevy 已提交
1119
		*p++ = cpu_to_be32(NFS4_SHARE_ACCESS_WRITE);
A
Andy Adamson 已提交
1120 1121
		break;
	case FMODE_READ|FMODE_WRITE:
B
Benny Halevy 已提交
1122
		*p++ = cpu_to_be32(NFS4_SHARE_ACCESS_BOTH);
A
Andy Adamson 已提交
1123 1124
		break;
	default:
B
Benny Halevy 已提交
1125
		*p++ = cpu_to_be32(0);
L
Linus Torvalds 已提交
1126
	}
B
Benny Halevy 已提交
1127
	*p++ = cpu_to_be32(0);		/* for linux, share_deny = 0 always */
L
Linus Torvalds 已提交
1128 1129 1130 1131
}

static inline void encode_openhdr(struct xdr_stream *xdr, const struct nfs_openargs *arg)
{
A
Al Viro 已提交
1132
	__be32 *p;
L
Linus Torvalds 已提交
1133 1134 1135 1136
 /*
 * opcode 4, seqid 4, share_access 4, share_deny 4, clientid 8, ownerlen 4,
 * owner 4 = 32
 */
1137
	p = reserve_space(xdr, 8);
B
Benny Halevy 已提交
1138 1139
	*p++ = cpu_to_be32(OP_OPEN);
	*p++ = cpu_to_be32(arg->seqid->sequence->counter);
1140
	encode_share_access(xdr, arg->fmode);
1141
	p = reserve_space(xdr, 28);
B
Benny Halevy 已提交
1142
	p = xdr_encode_hyper(p, arg->clientid);
B
Benny Halevy 已提交
1143
	*p++ = cpu_to_be32(16);
B
Benny Halevy 已提交
1144
	p = xdr_encode_opaque_fixed(p, "open id:", 8);
B
Benny Halevy 已提交
1145
	p = xdr_encode_hyper(p, arg->id);
L
Linus Torvalds 已提交
1146 1147 1148 1149
}

static inline void encode_createmode(struct xdr_stream *xdr, const struct nfs_openargs *arg)
{
A
Al Viro 已提交
1150
	__be32 *p;
L
Linus Torvalds 已提交
1151

1152
	p = reserve_space(xdr, 4);
L
Linus Torvalds 已提交
1153
	switch(arg->open_flags & O_EXCL) {
A
Andy Adamson 已提交
1154
	case 0:
B
Benny Halevy 已提交
1155
		*p++ = cpu_to_be32(NFS4_CREATE_UNCHECKED);
A
Andy Adamson 已提交
1156 1157 1158
		encode_attrs(xdr, arg->u.attrs, arg->server);
		break;
	default:
B
Benny Halevy 已提交
1159
		*p++ = cpu_to_be32(NFS4_CREATE_EXCLUSIVE);
A
Andy Adamson 已提交
1160
		encode_nfs4_verifier(xdr, &arg->u.verifier);
L
Linus Torvalds 已提交
1161 1162 1163 1164 1165
	}
}

static void encode_opentype(struct xdr_stream *xdr, const struct nfs_openargs *arg)
{
A
Al Viro 已提交
1166
	__be32 *p;
L
Linus Torvalds 已提交
1167

1168
	p = reserve_space(xdr, 4);
L
Linus Torvalds 已提交
1169
	switch (arg->open_flags & O_CREAT) {
A
Andy Adamson 已提交
1170
	case 0:
B
Benny Halevy 已提交
1171
		*p++ = cpu_to_be32(NFS4_OPEN_NOCREATE);
A
Andy Adamson 已提交
1172 1173 1174
		break;
	default:
		BUG_ON(arg->claim != NFS4_OPEN_CLAIM_NULL);
B
Benny Halevy 已提交
1175
		*p++ = cpu_to_be32(NFS4_OPEN_CREATE);
A
Andy Adamson 已提交
1176
		encode_createmode(xdr, arg);
L
Linus Torvalds 已提交
1177 1178 1179
	}
}

1180
static inline void encode_delegation_type(struct xdr_stream *xdr, fmode_t delegation_type)
L
Linus Torvalds 已提交
1181
{
A
Al Viro 已提交
1182
	__be32 *p;
L
Linus Torvalds 已提交
1183

1184
	p = reserve_space(xdr, 4);
L
Linus Torvalds 已提交
1185
	switch (delegation_type) {
A
Andy Adamson 已提交
1186
	case 0:
B
Benny Halevy 已提交
1187
		*p++ = cpu_to_be32(NFS4_OPEN_DELEGATE_NONE);
A
Andy Adamson 已提交
1188 1189
		break;
	case FMODE_READ:
B
Benny Halevy 已提交
1190
		*p++ = cpu_to_be32(NFS4_OPEN_DELEGATE_READ);
A
Andy Adamson 已提交
1191 1192
		break;
	case FMODE_WRITE|FMODE_READ:
B
Benny Halevy 已提交
1193
		*p++ = cpu_to_be32(NFS4_OPEN_DELEGATE_WRITE);
A
Andy Adamson 已提交
1194 1195 1196
		break;
	default:
		BUG();
L
Linus Torvalds 已提交
1197 1198 1199 1200 1201
	}
}

static inline void encode_claim_null(struct xdr_stream *xdr, const struct qstr *name)
{
A
Al Viro 已提交
1202
	__be32 *p;
L
Linus Torvalds 已提交
1203

1204
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1205
	*p++ = cpu_to_be32(NFS4_OPEN_CLAIM_NULL);
L
Linus Torvalds 已提交
1206 1207 1208
	encode_string(xdr, name->len, name->name);
}

1209
static inline void encode_claim_previous(struct xdr_stream *xdr, fmode_t type)
L
Linus Torvalds 已提交
1210
{
A
Al Viro 已提交
1211
	__be32 *p;
L
Linus Torvalds 已提交
1212

1213
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1214
	*p++ = cpu_to_be32(NFS4_OPEN_CLAIM_PREVIOUS);
L
Linus Torvalds 已提交
1215 1216 1217 1218 1219
	encode_delegation_type(xdr, type);
}

static inline void encode_claim_delegate_cur(struct xdr_stream *xdr, const struct qstr *name, const nfs4_stateid *stateid)
{
A
Al Viro 已提交
1220
	__be32 *p;
L
Linus Torvalds 已提交
1221

1222
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1223
	*p++ = cpu_to_be32(NFS4_OPEN_CLAIM_DELEGATE_CUR);
B
Benny Halevy 已提交
1224
	p = xdr_encode_opaque_fixed(p, stateid->data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
1225 1226 1227
	encode_string(xdr, name->len, name->name);
}

1228
static void encode_open(struct xdr_stream *xdr, const struct nfs_openargs *arg, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1229 1230 1231 1232
{
	encode_openhdr(xdr, arg);
	encode_opentype(xdr, arg);
	switch (arg->claim) {
A
Andy Adamson 已提交
1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243
	case NFS4_OPEN_CLAIM_NULL:
		encode_claim_null(xdr, arg->name);
		break;
	case NFS4_OPEN_CLAIM_PREVIOUS:
		encode_claim_previous(xdr, arg->u.delegation_type);
		break;
	case NFS4_OPEN_CLAIM_DELEGATE_CUR:
		encode_claim_delegate_cur(xdr, arg->name, &arg->u.delegation);
		break;
	default:
		BUG();
L
Linus Torvalds 已提交
1244
	}
1245
	hdr->nops++;
1246
	hdr->replen += decode_open_maxsz;
L
Linus Torvalds 已提交
1247 1248
}

1249
static void encode_open_confirm(struct xdr_stream *xdr, const struct nfs_open_confirmargs *arg, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1250
{
A
Al Viro 已提交
1251
	__be32 *p;
L
Linus Torvalds 已提交
1252

1253
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+4);
B
Benny Halevy 已提交
1254
	*p++ = cpu_to_be32(OP_OPEN_CONFIRM);
B
Benny Halevy 已提交
1255
	p = xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1256
	*p++ = cpu_to_be32(arg->seqid->sequence->counter);
1257
	hdr->nops++;
1258
	hdr->replen += decode_open_confirm_maxsz;
L
Linus Torvalds 已提交
1259 1260
}

1261
static void encode_open_downgrade(struct xdr_stream *xdr, const struct nfs_closeargs *arg, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1262
{
A
Al Viro 已提交
1263
	__be32 *p;
L
Linus Torvalds 已提交
1264

1265
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+4);
B
Benny Halevy 已提交
1266
	*p++ = cpu_to_be32(OP_OPEN_DOWNGRADE);
B
Benny Halevy 已提交
1267
	p = xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1268
	*p++ = cpu_to_be32(arg->seqid->sequence->counter);
1269
	encode_share_access(xdr, arg->fmode);
1270
	hdr->nops++;
1271
	hdr->replen += decode_open_downgrade_maxsz;
L
Linus Torvalds 已提交
1272 1273
}

1274
static void
1275
encode_putfh(struct xdr_stream *xdr, const struct nfs_fh *fh, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1276 1277
{
	int len = fh->size;
A
Al Viro 已提交
1278
	__be32 *p;
L
Linus Torvalds 已提交
1279

1280
	p = reserve_space(xdr, 8 + len);
B
Benny Halevy 已提交
1281 1282
	*p++ = cpu_to_be32(OP_PUTFH);
	*p++ = cpu_to_be32(len);
B
Benny Halevy 已提交
1283
	p = xdr_encode_opaque_fixed(p, fh->data, len);
1284
	hdr->nops++;
1285
	hdr->replen += decode_putfh_maxsz;
L
Linus Torvalds 已提交
1286 1287
}

1288
static void encode_putrootfh(struct xdr_stream *xdr, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1289
{
A
Andy Adamson 已提交
1290
	__be32 *p;
1291

1292
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1293
	*p++ = cpu_to_be32(OP_PUTROOTFH);
1294
	hdr->nops++;
1295
	hdr->replen += decode_putrootfh_maxsz;
L
Linus Torvalds 已提交
1296 1297 1298 1299 1300
}

static void encode_stateid(struct xdr_stream *xdr, const struct nfs_open_context *ctx)
{
	nfs4_stateid stateid;
A
Al Viro 已提交
1301
	__be32 *p;
L
Linus Torvalds 已提交
1302

1303
	p = reserve_space(xdr, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
1304 1305
	if (ctx->state != NULL) {
		nfs4_copy_stateid(&stateid, ctx->state, ctx->lockowner);
B
Benny Halevy 已提交
1306
		p = xdr_encode_opaque_fixed(p, stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
1307
	} else
B
Benny Halevy 已提交
1308
		p = xdr_encode_opaque_fixed(p, zero_stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
1309 1310
}

1311
static void encode_read(struct xdr_stream *xdr, const struct nfs_readargs *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1312
{
A
Al Viro 已提交
1313
	__be32 *p;
L
Linus Torvalds 已提交
1314

1315
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1316
	*p++ = cpu_to_be32(OP_READ);
L
Linus Torvalds 已提交
1317 1318 1319

	encode_stateid(xdr, args->context);

1320
	p = reserve_space(xdr, 12);
B
Benny Halevy 已提交
1321
	p = xdr_encode_hyper(p, args->offset);
B
Benny Halevy 已提交
1322
	*p++ = cpu_to_be32(args->count);
1323
	hdr->nops++;
1324
	hdr->replen += decode_read_maxsz;
L
Linus Torvalds 已提交
1325 1326
}

1327
static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg *readdir, struct rpc_rqst *req, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1328
{
1329 1330 1331 1332
	uint32_t attrs[2] = {
		FATTR4_WORD0_RDATTR_ERROR|FATTR4_WORD0_FILEID,
		FATTR4_WORD1_MOUNTED_ON_FILEID,
	};
A
Al Viro 已提交
1333
	__be32 *p;
L
Linus Torvalds 已提交
1334

1335
	p = reserve_space(xdr, 12+NFS4_VERIFIER_SIZE+20);
B
Benny Halevy 已提交
1336
	*p++ = cpu_to_be32(OP_READDIR);
B
Benny Halevy 已提交
1337
	p = xdr_encode_hyper(p, readdir->cookie);
B
Benny Halevy 已提交
1338
	p = xdr_encode_opaque_fixed(p, readdir->verifier.data, NFS4_VERIFIER_SIZE);
B
Benny Halevy 已提交
1339 1340 1341
	*p++ = cpu_to_be32(readdir->count >> 1);  /* We're not doing readdirplus */
	*p++ = cpu_to_be32(readdir->count);
	*p++ = cpu_to_be32(2);
1342 1343 1344 1345 1346
	/* Switch to mounted_on_fileid if the server supports it */
	if (readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)
		attrs[0] &= ~FATTR4_WORD0_FILEID;
	else
		attrs[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID;
B
Benny Halevy 已提交
1347 1348
	*p++ = cpu_to_be32(attrs[0] & readdir->bitmask[0]);
	*p++ = cpu_to_be32(attrs[1] & readdir->bitmask[1]);
1349
	hdr->nops++;
1350
	hdr->replen += decode_readdir_maxsz;
1351 1352
	dprintk("%s: cookie = %Lu, verifier = %08x:%08x, bitmap = %08x:%08x\n",
			__func__,
1353 1354 1355 1356 1357
			(unsigned long long)readdir->cookie,
			((u32 *)readdir->verifier.data)[0],
			((u32 *)readdir->verifier.data)[1],
			attrs[0] & readdir->bitmask[0],
			attrs[1] & readdir->bitmask[1]);
L
Linus Torvalds 已提交
1358 1359
}

1360
static void encode_readlink(struct xdr_stream *xdr, const struct nfs4_readlink *readlink, struct rpc_rqst *req, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1361
{
A
Al Viro 已提交
1362
	__be32 *p;
L
Linus Torvalds 已提交
1363

1364
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1365
	*p++ = cpu_to_be32(OP_READLINK);
1366
	hdr->nops++;
1367
	hdr->replen += decode_readlink_maxsz;
L
Linus Torvalds 已提交
1368 1369
}

1370
static void encode_remove(struct xdr_stream *xdr, const struct qstr *name, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1371
{
A
Al Viro 已提交
1372
	__be32 *p;
L
Linus Torvalds 已提交
1373

1374
	p = reserve_space(xdr, 8 + name->len);
B
Benny Halevy 已提交
1375 1376
	*p++ = cpu_to_be32(OP_REMOVE);
	*p++ = cpu_to_be32(name->len);
B
Benny Halevy 已提交
1377
	p = xdr_encode_opaque_fixed(p, name->name, name->len);
1378
	hdr->nops++;
1379
	hdr->replen += decode_remove_maxsz;
L
Linus Torvalds 已提交
1380 1381
}

1382
static void encode_rename(struct xdr_stream *xdr, const struct qstr *oldname, const struct qstr *newname, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1383
{
A
Al Viro 已提交
1384
	__be32 *p;
L
Linus Torvalds 已提交
1385

1386
	p = reserve_space(xdr, 8 + oldname->len);
B
Benny Halevy 已提交
1387 1388
	*p++ = cpu_to_be32(OP_RENAME);
	*p++ = cpu_to_be32(oldname->len);
B
Benny Halevy 已提交
1389
	p = xdr_encode_opaque_fixed(p, oldname->name, oldname->len);
1390

1391
	p = reserve_space(xdr, 4 + newname->len);
B
Benny Halevy 已提交
1392
	*p++ = cpu_to_be32(newname->len);
B
Benny Halevy 已提交
1393
	p = xdr_encode_opaque_fixed(p, newname->name, newname->len);
1394
	hdr->nops++;
1395
	hdr->replen += decode_rename_maxsz;
L
Linus Torvalds 已提交
1396 1397
}

1398
static void encode_renew(struct xdr_stream *xdr, const struct nfs_client *client_stateid, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1399
{
A
Al Viro 已提交
1400
	__be32 *p;
L
Linus Torvalds 已提交
1401

1402
	p = reserve_space(xdr, 12);
B
Benny Halevy 已提交
1403
	*p++ = cpu_to_be32(OP_RENEW);
B
Benny Halevy 已提交
1404
	p = xdr_encode_hyper(p, client_stateid->cl_clientid);
1405
	hdr->nops++;
1406
	hdr->replen += decode_renew_maxsz;
L
Linus Torvalds 已提交
1407 1408
}

1409
static void
1410
encode_restorefh(struct xdr_stream *xdr, struct compound_hdr *hdr)
1411
{
A
Al Viro 已提交
1412
	__be32 *p;
1413

1414
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1415
	*p++ = cpu_to_be32(OP_RESTOREFH);
1416
	hdr->nops++;
1417
	hdr->replen += decode_restorefh_maxsz;
1418 1419
}

1420
static int
1421
encode_setacl(struct xdr_stream *xdr, struct nfs_setaclargs *arg, struct compound_hdr *hdr)
1422
{
A
Al Viro 已提交
1423
	__be32 *p;
1424

1425
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1426
	*p++ = cpu_to_be32(OP_SETATTR);
B
Benny Halevy 已提交
1427
	p = xdr_encode_opaque_fixed(p, zero_stateid.data, NFS4_STATEID_SIZE);
1428
	p = reserve_space(xdr, 2*4);
B
Benny Halevy 已提交
1429 1430
	*p++ = cpu_to_be32(1);
	*p++ = cpu_to_be32(FATTR4_WORD0_ACL);
1431 1432
	if (arg->acl_len % 4)
		return -EINVAL;
1433
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1434
	*p++ = cpu_to_be32(arg->acl_len);
1435
	xdr_write_pages(xdr, arg->acl_pages, arg->acl_pgbase, arg->acl_len);
1436
	hdr->nops++;
1437
	hdr->replen += decode_setacl_maxsz;
1438 1439 1440
	return 0;
}

1441
static void
1442
encode_savefh(struct xdr_stream *xdr, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1443
{
A
Al Viro 已提交
1444
	__be32 *p;
L
Linus Torvalds 已提交
1445

1446
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1447
	*p++ = cpu_to_be32(OP_SAVEFH);
1448
	hdr->nops++;
1449
	hdr->replen += decode_savefh_maxsz;
L
Linus Torvalds 已提交
1450 1451
}

1452
static void encode_setattr(struct xdr_stream *xdr, const struct nfs_setattrargs *arg, const struct nfs_server *server, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1453
{
A
Al Viro 已提交
1454
	__be32 *p;
1455

1456
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE);
B
Benny Halevy 已提交
1457
	*p++ = cpu_to_be32(OP_SETATTR);
B
Benny Halevy 已提交
1458
	p = xdr_encode_opaque_fixed(p, arg->stateid.data, NFS4_STATEID_SIZE);
1459
	hdr->nops++;
1460
	hdr->replen += decode_setattr_maxsz;
1461
	encode_attrs(xdr, arg->iap, server);
L
Linus Torvalds 已提交
1462 1463
}

1464
static void encode_setclientid(struct xdr_stream *xdr, const struct nfs4_setclientid *setclientid, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1465
{
A
Al Viro 已提交
1466
	__be32 *p;
L
Linus Torvalds 已提交
1467

1468
	p = reserve_space(xdr, 4 + NFS4_VERIFIER_SIZE);
B
Benny Halevy 已提交
1469
	*p++ = cpu_to_be32(OP_SETCLIENTID);
B
Benny Halevy 已提交
1470
	p = xdr_encode_opaque_fixed(p, setclientid->sc_verifier->data, NFS4_VERIFIER_SIZE);
L
Linus Torvalds 已提交
1471 1472

	encode_string(xdr, setclientid->sc_name_len, setclientid->sc_name);
1473
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1474
	*p++ = cpu_to_be32(setclientid->sc_prog);
L
Linus Torvalds 已提交
1475 1476
	encode_string(xdr, setclientid->sc_netid_len, setclientid->sc_netid);
	encode_string(xdr, setclientid->sc_uaddr_len, setclientid->sc_uaddr);
1477
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1478
	*p++ = cpu_to_be32(setclientid->sc_cb_ident);
1479
	hdr->nops++;
1480
	hdr->replen += decode_setclientid_maxsz;
L
Linus Torvalds 已提交
1481 1482
}

1483
static void encode_setclientid_confirm(struct xdr_stream *xdr, const struct nfs_client *client_state, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1484
{
A
Andy Adamson 已提交
1485
	__be32 *p;
L
Linus Torvalds 已提交
1486

1487
	p = reserve_space(xdr, 12 + NFS4_VERIFIER_SIZE);
B
Benny Halevy 已提交
1488
	*p++ = cpu_to_be32(OP_SETCLIENTID_CONFIRM);
B
Benny Halevy 已提交
1489
	p = xdr_encode_hyper(p, client_state->cl_clientid);
B
Benny Halevy 已提交
1490
	p = xdr_encode_opaque_fixed(p, client_state->cl_confirm.data, NFS4_VERIFIER_SIZE);
1491
	hdr->nops++;
1492
	hdr->replen += decode_setclientid_confirm_maxsz;
L
Linus Torvalds 已提交
1493 1494
}

1495
static void encode_write(struct xdr_stream *xdr, const struct nfs_writeargs *args, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1496
{
A
Al Viro 已提交
1497
	__be32 *p;
L
Linus Torvalds 已提交
1498

1499
	p = reserve_space(xdr, 4);
B
Benny Halevy 已提交
1500
	*p++ = cpu_to_be32(OP_WRITE);
L
Linus Torvalds 已提交
1501 1502 1503

	encode_stateid(xdr, args->context);

1504
	p = reserve_space(xdr, 16);
B
Benny Halevy 已提交
1505
	p = xdr_encode_hyper(p, args->offset);
B
Benny Halevy 已提交
1506 1507
	*p++ = cpu_to_be32(args->stable);
	*p++ = cpu_to_be32(args->count);
L
Linus Torvalds 已提交
1508 1509

	xdr_write_pages(xdr, args->pages, args->pgbase, args->count);
1510
	hdr->nops++;
1511
	hdr->replen += decode_write_maxsz;
L
Linus Torvalds 已提交
1512 1513
}

1514
static void encode_delegreturn(struct xdr_stream *xdr, const nfs4_stateid *stateid, struct compound_hdr *hdr)
L
Linus Torvalds 已提交
1515
{
A
Al Viro 已提交
1516
	__be32 *p;
L
Linus Torvalds 已提交
1517

1518
	p = reserve_space(xdr, 4+NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
1519

B
Benny Halevy 已提交
1520
	*p++ = cpu_to_be32(OP_DELEGRETURN);
B
Benny Halevy 已提交
1521
	p = xdr_encode_opaque_fixed(p, stateid->data, NFS4_STATEID_SIZE);
1522
	hdr->nops++;
1523
	hdr->replen += decode_delegreturn_maxsz;
L
Linus Torvalds 已提交
1524
}
1525

B
Benny Halevy 已提交
1526
#if defined(CONFIG_NFS_V4_1)
1527
/* NFSv4.1 operations */
B
Benny Halevy 已提交
1528 1529 1530 1531 1532 1533
static void encode_exchange_id(struct xdr_stream *xdr,
			       struct nfs41_exchange_id_args *args,
			       struct compound_hdr *hdr)
{
	__be32 *p;

1534
	p = reserve_space(xdr, 4 + sizeof(args->verifier->data));
B
Benny Halevy 已提交
1535
	*p++ = cpu_to_be32(OP_EXCHANGE_ID);
B
Benny Halevy 已提交
1536
	p = xdr_encode_opaque_fixed(p, args->verifier->data, sizeof(args->verifier->data));
B
Benny Halevy 已提交
1537 1538 1539

	encode_string(xdr, args->id_len, args->id);

1540
	p = reserve_space(xdr, 12);
B
Benny Halevy 已提交
1541 1542 1543
	*p++ = cpu_to_be32(args->flags);
	*p++ = cpu_to_be32(0);	/* zero length state_protect4_a */
	*p++ = cpu_to_be32(0);	/* zero length implementation id array */
B
Benny Halevy 已提交
1544 1545 1546
	hdr->nops++;
	hdr->replen += decode_exchange_id_maxsz;
}
A
Andy Adamson 已提交
1547 1548 1549 1550 1551 1552 1553 1554 1555 1556

static void encode_create_session(struct xdr_stream *xdr,
				  struct nfs41_create_session_args *args,
				  struct compound_hdr *hdr)
{
	__be32 *p;
	char machine_name[NFS4_MAX_MACHINE_NAME_LEN];
	uint32_t len;
	struct nfs_client *clp = args->client;

1557 1558
	len = scnprintf(machine_name, sizeof(machine_name), "%s",
			clp->cl_ipaddr);
A
Andy Adamson 已提交
1559

1560
	p = reserve_space(xdr, 20 + 2*28 + 20 + len + 12);
1561
	*p++ = cpu_to_be32(OP_CREATE_SESSION);
B
Benny Halevy 已提交
1562
	p = xdr_encode_hyper(p, clp->cl_ex_clid);
B
Benny Halevy 已提交
1563 1564
	*p++ = cpu_to_be32(clp->cl_seqid);			/*Sequence id */
	*p++ = cpu_to_be32(args->flags);			/*flags */
A
Andy Adamson 已提交
1565 1566

	/* Fore Channel */
B
Benny Halevy 已提交
1567 1568 1569 1570 1571 1572 1573
	*p++ = cpu_to_be32(args->fc_attrs.headerpadsz);	/* header padding size */
	*p++ = cpu_to_be32(args->fc_attrs.max_rqst_sz);	/* max req size */
	*p++ = cpu_to_be32(args->fc_attrs.max_resp_sz);	/* max resp size */
	*p++ = cpu_to_be32(args->fc_attrs.max_resp_sz_cached);	/* Max resp sz cached */
	*p++ = cpu_to_be32(args->fc_attrs.max_ops);	/* max operations */
	*p++ = cpu_to_be32(args->fc_attrs.max_reqs);	/* max requests */
	*p++ = cpu_to_be32(0);				/* rdmachannel_attrs */
A
Andy Adamson 已提交
1574 1575

	/* Back Channel */
B
Benny Halevy 已提交
1576 1577 1578 1579 1580 1581 1582
	*p++ = cpu_to_be32(args->fc_attrs.headerpadsz);	/* header padding size */
	*p++ = cpu_to_be32(args->bc_attrs.max_rqst_sz);	/* max req size */
	*p++ = cpu_to_be32(args->bc_attrs.max_resp_sz);	/* max resp size */
	*p++ = cpu_to_be32(args->bc_attrs.max_resp_sz_cached);	/* Max resp sz cached */
	*p++ = cpu_to_be32(args->bc_attrs.max_ops);	/* max operations */
	*p++ = cpu_to_be32(args->bc_attrs.max_reqs);	/* max requests */
	*p++ = cpu_to_be32(0);				/* rdmachannel_attrs */
A
Andy Adamson 已提交
1583

B
Benny Halevy 已提交
1584 1585 1586
	*p++ = cpu_to_be32(args->cb_program);		/* cb_program */
	*p++ = cpu_to_be32(1);
	*p++ = cpu_to_be32(RPC_AUTH_UNIX);			/* auth_sys */
A
Andy Adamson 已提交
1587 1588

	/* authsys_parms rfc1831 */
B
Benny Halevy 已提交
1589 1590
	*p++ = cpu_to_be32((u32)clp->cl_boot_time.tv_nsec);	/* stamp */
	*p++ = cpu_to_be32(len);
B
Benny Halevy 已提交
1591
	p = xdr_encode_opaque_fixed(p, machine_name, len);
B
Benny Halevy 已提交
1592 1593 1594
	*p++ = cpu_to_be32(0);				/* UID */
	*p++ = cpu_to_be32(0);				/* GID */
	*p++ = cpu_to_be32(0);				/* No more gids */
A
Andy Adamson 已提交
1595 1596 1597
	hdr->nops++;
	hdr->replen += decode_create_session_maxsz;
}
A
Andy Adamson 已提交
1598 1599 1600 1601 1602 1603

static void encode_destroy_session(struct xdr_stream *xdr,
				   struct nfs4_session *session,
				   struct compound_hdr *hdr)
{
	__be32 *p;
1604
	p = reserve_space(xdr, 4 + NFS4_MAX_SESSIONID_LEN);
B
Benny Halevy 已提交
1605
	*p++ = cpu_to_be32(OP_DESTROY_SESSION);
B
Benny Halevy 已提交
1606
	p = xdr_encode_opaque_fixed(p, session->sess_id.data, NFS4_MAX_SESSIONID_LEN);
A
Andy Adamson 已提交
1607 1608 1609
	hdr->nops++;
	hdr->replen += decode_destroy_session_maxsz;
}
B
Benny Halevy 已提交
1610 1611
#endif /* CONFIG_NFS_V4_1 */

1612 1613 1614 1615 1616 1617
static void encode_sequence(struct xdr_stream *xdr,
			    const struct nfs4_sequence_args *args,
			    struct compound_hdr *hdr)
{
#if defined(CONFIG_NFS_V4_1)
	struct nfs4_session *session = args->sa_session;
A
Andy Adamson 已提交
1618 1619 1620
	struct nfs4_slot_table *tp;
	struct nfs4_slot *slot;
	__be32 *p;
1621 1622 1623 1624

	if (!session)
		return;

A
Andy Adamson 已提交
1625 1626 1627 1628 1629
	tp = &session->fc_slot_table;

	WARN_ON(args->sa_slotid == NFS4_MAX_SLOT_TABLE);
	slot = tp->slots + args->sa_slotid;

1630
	p = reserve_space(xdr, 4 + NFS4_MAX_SESSIONID_LEN + 16);
B
Benny Halevy 已提交
1631
	*p++ = cpu_to_be32(OP_SEQUENCE);
A
Andy Adamson 已提交
1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644

	/*
	 * Sessionid + seqid + slotid + max slotid + cache_this
	 */
	dprintk("%s: sessionid=%u:%u:%u:%u seqid=%d slotid=%d "
		"max_slotid=%d cache_this=%d\n",
		__func__,
		((u32 *)session->sess_id.data)[0],
		((u32 *)session->sess_id.data)[1],
		((u32 *)session->sess_id.data)[2],
		((u32 *)session->sess_id.data)[3],
		slot->seq_nr, args->sa_slotid,
		tp->highest_used_slotid, args->sa_cache_this);
B
Benny Halevy 已提交
1645
	p = xdr_encode_opaque_fixed(p, session->sess_id.data, NFS4_MAX_SESSIONID_LEN);
B
Benny Halevy 已提交
1646 1647 1648 1649
	*p++ = cpu_to_be32(slot->seq_nr);
	*p++ = cpu_to_be32(args->sa_slotid);
	*p++ = cpu_to_be32(tp->highest_used_slotid);
	*p++ = cpu_to_be32(args->sa_cache_this);
1650 1651 1652 1653 1654
	hdr->nops++;
	hdr->replen += decode_sequence_maxsz;
#endif /* CONFIG_NFS_V4_1 */
}

L
Linus Torvalds 已提交
1655 1656 1657 1658
/*
 * END OF "GENERIC" ENCODE ROUTINES.
 */

1659 1660 1661 1662 1663 1664 1665 1666 1667
static u32 nfs4_xdr_minorversion(const struct nfs4_sequence_args *args)
{
#if defined(CONFIG_NFS_V4_1)
	if (args->sa_session)
		return args->sa_session->clp->cl_minorversion;
#endif /* CONFIG_NFS_V4_1 */
	return 0;
}

L
Linus Torvalds 已提交
1668 1669 1670
/*
 * Encode an ACCESS request
 */
A
Al Viro 已提交
1671
static int nfs4_xdr_enc_access(struct rpc_rqst *req, __be32 *p, const struct nfs4_accessargs *args)
L
Linus Torvalds 已提交
1672 1673 1674
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1675
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1676 1677 1678
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1679
	encode_compound_hdr(&xdr, req, &hdr);
1680
	encode_sequence(&xdr, &args->seq_args, &hdr);
1681 1682 1683
	encode_putfh(&xdr, args->fh, &hdr);
	encode_access(&xdr, args->access, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1684
	encode_nops(&hdr);
1685
	return 0;
L
Linus Torvalds 已提交
1686 1687 1688 1689 1690
}

/*
 * Encode LOOKUP request
 */
A
Al Viro 已提交
1691
static int nfs4_xdr_enc_lookup(struct rpc_rqst *req, __be32 *p, const struct nfs4_lookup_arg *args)
L
Linus Torvalds 已提交
1692 1693 1694
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1695
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1696 1697 1698
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1699
	encode_compound_hdr(&xdr, req, &hdr);
1700
	encode_sequence(&xdr, &args->seq_args, &hdr);
1701 1702 1703 1704
	encode_putfh(&xdr, args->dir_fh, &hdr);
	encode_lookup(&xdr, args->name, &hdr);
	encode_getfh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1705
	encode_nops(&hdr);
1706
	return 0;
L
Linus Torvalds 已提交
1707 1708 1709 1710 1711
}

/*
 * Encode LOOKUP_ROOT request
 */
A
Al Viro 已提交
1712
static int nfs4_xdr_enc_lookup_root(struct rpc_rqst *req, __be32 *p, const struct nfs4_lookup_root_arg *args)
L
Linus Torvalds 已提交
1713 1714 1715
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1716
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1717 1718 1719
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1720
	encode_compound_hdr(&xdr, req, &hdr);
1721
	encode_sequence(&xdr, &args->seq_args, &hdr);
1722 1723 1724
	encode_putrootfh(&xdr, &hdr);
	encode_getfh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1725
	encode_nops(&hdr);
1726
	return 0;
L
Linus Torvalds 已提交
1727 1728 1729 1730 1731
}

/*
 * Encode REMOVE request
 */
1732
static int nfs4_xdr_enc_remove(struct rpc_rqst *req, __be32 *p, const struct nfs_removeargs *args)
L
Linus Torvalds 已提交
1733 1734 1735
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1736
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1737 1738 1739
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1740
	encode_compound_hdr(&xdr, req, &hdr);
1741
	encode_sequence(&xdr, &args->seq_args, &hdr);
1742 1743 1744
	encode_putfh(&xdr, args->fh, &hdr);
	encode_remove(&xdr, &args->name, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1745
	encode_nops(&hdr);
1746
	return 0;
L
Linus Torvalds 已提交
1747 1748 1749 1750 1751
}

/*
 * Encode RENAME request
 */
A
Al Viro 已提交
1752
static int nfs4_xdr_enc_rename(struct rpc_rqst *req, __be32 *p, const struct nfs4_rename_arg *args)
L
Linus Torvalds 已提交
1753 1754 1755
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1756
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1757 1758 1759
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1760
	encode_compound_hdr(&xdr, req, &hdr);
1761
	encode_sequence(&xdr, &args->seq_args, &hdr);
1762 1763 1764 1765 1766 1767 1768
	encode_putfh(&xdr, args->old_dir, &hdr);
	encode_savefh(&xdr, &hdr);
	encode_putfh(&xdr, args->new_dir, &hdr);
	encode_rename(&xdr, args->old_name, args->new_name, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
	encode_restorefh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1769
	encode_nops(&hdr);
1770
	return 0;
L
Linus Torvalds 已提交
1771 1772 1773 1774 1775
}

/*
 * Encode LINK request
 */
A
Al Viro 已提交
1776
static int nfs4_xdr_enc_link(struct rpc_rqst *req, __be32 *p, const struct nfs4_link_arg *args)
L
Linus Torvalds 已提交
1777 1778 1779
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1780
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1781 1782 1783
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1784
	encode_compound_hdr(&xdr, req, &hdr);
1785
	encode_sequence(&xdr, &args->seq_args, &hdr);
1786 1787 1788 1789 1790 1791 1792
	encode_putfh(&xdr, args->fh, &hdr);
	encode_savefh(&xdr, &hdr);
	encode_putfh(&xdr, args->dir_fh, &hdr);
	encode_link(&xdr, args->name, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
	encode_restorefh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1793
	encode_nops(&hdr);
1794
	return 0;
L
Linus Torvalds 已提交
1795 1796 1797 1798 1799
}

/*
 * Encode CREATE request
 */
A
Al Viro 已提交
1800
static int nfs4_xdr_enc_create(struct rpc_rqst *req, __be32 *p, const struct nfs4_create_arg *args)
L
Linus Torvalds 已提交
1801 1802 1803
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1804
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1805 1806 1807
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1808
	encode_compound_hdr(&xdr, req, &hdr);
1809
	encode_sequence(&xdr, &args->seq_args, &hdr);
1810 1811 1812 1813 1814 1815 1816
	encode_putfh(&xdr, args->dir_fh, &hdr);
	encode_savefh(&xdr, &hdr);
	encode_create(&xdr, args, &hdr);
	encode_getfh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
	encode_restorefh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1817
	encode_nops(&hdr);
1818
	return 0;
L
Linus Torvalds 已提交
1819 1820 1821 1822 1823
}

/*
 * Encode SYMLINK request
 */
A
Al Viro 已提交
1824
static int nfs4_xdr_enc_symlink(struct rpc_rqst *req, __be32 *p, const struct nfs4_create_arg *args)
L
Linus Torvalds 已提交
1825 1826 1827 1828 1829 1830 1831
{
	return nfs4_xdr_enc_create(req, p, args);
}

/*
 * Encode GETATTR request
 */
A
Al Viro 已提交
1832
static int nfs4_xdr_enc_getattr(struct rpc_rqst *req, __be32 *p, const struct nfs4_getattr_arg *args)
L
Linus Torvalds 已提交
1833 1834 1835
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1836
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1837 1838 1839
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1840
	encode_compound_hdr(&xdr, req, &hdr);
1841
	encode_sequence(&xdr, &args->seq_args, &hdr);
1842 1843
	encode_putfh(&xdr, args->fh, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1844
	encode_nops(&hdr);
1845
	return 0;
L
Linus Torvalds 已提交
1846 1847 1848 1849 1850
}

/*
 * Encode a CLOSE request
 */
A
Al Viro 已提交
1851
static int nfs4_xdr_enc_close(struct rpc_rqst *req, __be32 *p, struct nfs_closeargs *args)
L
Linus Torvalds 已提交
1852
{
A
Andy Adamson 已提交
1853 1854
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1855
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
A
Andy Adamson 已提交
1856 1857 1858
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1859
	encode_compound_hdr(&xdr, req, &hdr);
1860
	encode_sequence(&xdr, &args->seq_args, &hdr);
1861 1862 1863
	encode_putfh(&xdr, args->fh, &hdr);
	encode_close(&xdr, args, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1864
	encode_nops(&hdr);
1865
	return 0;
L
Linus Torvalds 已提交
1866 1867 1868 1869 1870
}

/*
 * Encode an OPEN request
 */
A
Al Viro 已提交
1871
static int nfs4_xdr_enc_open(struct rpc_rqst *req, __be32 *p, struct nfs_openargs *args)
L
Linus Torvalds 已提交
1872 1873 1874
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1875
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1876 1877 1878
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1879
	encode_compound_hdr(&xdr, req, &hdr);
1880
	encode_sequence(&xdr, &args->seq_args, &hdr);
1881 1882 1883 1884 1885 1886 1887
	encode_putfh(&xdr, args->fh, &hdr);
	encode_savefh(&xdr, &hdr);
	encode_open(&xdr, args, &hdr);
	encode_getfh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
	encode_restorefh(&xdr, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1888
	encode_nops(&hdr);
1889
	return 0;
L
Linus Torvalds 已提交
1890 1891 1892 1893 1894
}

/*
 * Encode an OPEN_CONFIRM request
 */
A
Al Viro 已提交
1895
static int nfs4_xdr_enc_open_confirm(struct rpc_rqst *req, __be32 *p, struct nfs_open_confirmargs *args)
L
Linus Torvalds 已提交
1896 1897 1898
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1899
		.nops   = 0,
L
Linus Torvalds 已提交
1900 1901 1902
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1903
	encode_compound_hdr(&xdr, req, &hdr);
1904 1905
	encode_putfh(&xdr, args->fh, &hdr);
	encode_open_confirm(&xdr, args, &hdr);
1906
	encode_nops(&hdr);
1907
	return 0;
L
Linus Torvalds 已提交
1908 1909 1910 1911 1912
}

/*
 * Encode an OPEN request with no attributes.
 */
A
Al Viro 已提交
1913
static int nfs4_xdr_enc_open_noattr(struct rpc_rqst *req, __be32 *p, struct nfs_openargs *args)
L
Linus Torvalds 已提交
1914 1915 1916
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1917
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1918 1919 1920
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1921
	encode_compound_hdr(&xdr, req, &hdr);
1922
	encode_sequence(&xdr, &args->seq_args, &hdr);
1923 1924 1925
	encode_putfh(&xdr, args->fh, &hdr);
	encode_open(&xdr, args, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1926
	encode_nops(&hdr);
1927
	return 0;
L
Linus Torvalds 已提交
1928 1929 1930 1931 1932
}

/*
 * Encode an OPEN_DOWNGRADE request
 */
A
Al Viro 已提交
1933
static int nfs4_xdr_enc_open_downgrade(struct rpc_rqst *req, __be32 *p, struct nfs_closeargs *args)
L
Linus Torvalds 已提交
1934 1935 1936
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1937
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1938 1939 1940
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1941
	encode_compound_hdr(&xdr, req, &hdr);
1942
	encode_sequence(&xdr, &args->seq_args, &hdr);
1943 1944 1945
	encode_putfh(&xdr, args->fh, &hdr);
	encode_open_downgrade(&xdr, args, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
1946
	encode_nops(&hdr);
1947
	return 0;
L
Linus Torvalds 已提交
1948 1949 1950 1951 1952
}

/*
 * Encode a LOCK request
 */
A
Al Viro 已提交
1953
static int nfs4_xdr_enc_lock(struct rpc_rqst *req, __be32 *p, struct nfs_lock_args *args)
L
Linus Torvalds 已提交
1954 1955 1956
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1957
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1958 1959 1960
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1961
	encode_compound_hdr(&xdr, req, &hdr);
1962
	encode_sequence(&xdr, &args->seq_args, &hdr);
1963 1964
	encode_putfh(&xdr, args->fh, &hdr);
	encode_lock(&xdr, args, &hdr);
1965
	encode_nops(&hdr);
1966
	return 0;
L
Linus Torvalds 已提交
1967 1968 1969 1970 1971
}

/*
 * Encode a LOCKT request
 */
A
Al Viro 已提交
1972
static int nfs4_xdr_enc_lockt(struct rpc_rqst *req, __be32 *p, struct nfs_lockt_args *args)
L
Linus Torvalds 已提交
1973 1974 1975
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1976
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1977 1978 1979
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1980
	encode_compound_hdr(&xdr, req, &hdr);
1981
	encode_sequence(&xdr, &args->seq_args, &hdr);
1982 1983
	encode_putfh(&xdr, args->fh, &hdr);
	encode_lockt(&xdr, args, &hdr);
1984
	encode_nops(&hdr);
1985
	return 0;
L
Linus Torvalds 已提交
1986 1987 1988 1989 1990
}

/*
 * Encode a LOCKU request
 */
A
Al Viro 已提交
1991
static int nfs4_xdr_enc_locku(struct rpc_rqst *req, __be32 *p, struct nfs_locku_args *args)
L
Linus Torvalds 已提交
1992 1993 1994
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
1995
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
1996 1997 1998
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1999
	encode_compound_hdr(&xdr, req, &hdr);
2000
	encode_sequence(&xdr, &args->seq_args, &hdr);
2001 2002
	encode_putfh(&xdr, args->fh, &hdr);
	encode_locku(&xdr, args, &hdr);
2003
	encode_nops(&hdr);
2004
	return 0;
L
Linus Torvalds 已提交
2005 2006 2007 2008 2009
}

/*
 * Encode a READLINK request
 */
A
Al Viro 已提交
2010
static int nfs4_xdr_enc_readlink(struct rpc_rqst *req, __be32 *p, const struct nfs4_readlink *args)
L
Linus Torvalds 已提交
2011 2012 2013
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2014
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2015 2016 2017
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2018
	encode_compound_hdr(&xdr, req, &hdr);
2019
	encode_sequence(&xdr, &args->seq_args, &hdr);
2020 2021
	encode_putfh(&xdr, args->fh, &hdr);
	encode_readlink(&xdr, args, req, &hdr);
2022

2023
	xdr_inline_pages(&req->rq_rcv_buf, hdr.replen << 2, args->pages,
2024
			args->pgbase, args->pglen);
2025
	encode_nops(&hdr);
2026
	return 0;
L
Linus Torvalds 已提交
2027 2028 2029 2030 2031
}

/*
 * Encode a READDIR request
 */
A
Al Viro 已提交
2032
static int nfs4_xdr_enc_readdir(struct rpc_rqst *req, __be32 *p, const struct nfs4_readdir_arg *args)
L
Linus Torvalds 已提交
2033 2034 2035
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2036
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2037 2038 2039
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2040
	encode_compound_hdr(&xdr, req, &hdr);
2041
	encode_sequence(&xdr, &args->seq_args, &hdr);
2042 2043
	encode_putfh(&xdr, args->fh, &hdr);
	encode_readdir(&xdr, args, req, &hdr);
2044

2045
	xdr_inline_pages(&req->rq_rcv_buf, hdr.replen << 2, args->pages,
2046 2047
			 args->pgbase, args->count);
	dprintk("%s: inlined page args = (%u, %p, %u, %u)\n",
2048
			__func__, hdr.replen << 2, args->pages,
2049
			args->pgbase, args->count);
2050
	encode_nops(&hdr);
2051
	return 0;
L
Linus Torvalds 已提交
2052 2053 2054 2055 2056
}

/*
 * Encode a READ request
 */
A
Al Viro 已提交
2057
static int nfs4_xdr_enc_read(struct rpc_rqst *req, __be32 *p, struct nfs_readargs *args)
L
Linus Torvalds 已提交
2058 2059 2060
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2061
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2062 2063 2064
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2065
	encode_compound_hdr(&xdr, req, &hdr);
2066
	encode_sequence(&xdr, &args->seq_args, &hdr);
2067 2068
	encode_putfh(&xdr, args->fh, &hdr);
	encode_read(&xdr, args, &hdr);
L
Linus Torvalds 已提交
2069

2070
	xdr_inline_pages(&req->rq_rcv_buf, hdr.replen << 2,
L
Linus Torvalds 已提交
2071
			 args->pages, args->pgbase, args->count);
2072
	req->rq_rcv_buf.flags |= XDRBUF_READ;
2073
	encode_nops(&hdr);
2074
	return 0;
L
Linus Torvalds 已提交
2075 2076 2077 2078 2079
}

/*
 * Encode an SETATTR request
 */
A
Al Viro 已提交
2080
static int nfs4_xdr_enc_setattr(struct rpc_rqst *req, __be32 *p, struct nfs_setattrargs *args)
L
Linus Torvalds 已提交
2081
{
A
Andy Adamson 已提交
2082 2083
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2084
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
A
Andy Adamson 已提交
2085 2086 2087
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2088
	encode_compound_hdr(&xdr, req, &hdr);
2089
	encode_sequence(&xdr, &args->seq_args, &hdr);
2090 2091 2092
	encode_putfh(&xdr, args->fh, &hdr);
	encode_setattr(&xdr, args, args->server, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
2093
	encode_nops(&hdr);
2094
	return 0;
L
Linus Torvalds 已提交
2095 2096
}

2097 2098 2099 2100
/*
 * Encode a GETACL request
 */
static int
A
Al Viro 已提交
2101
nfs4_xdr_enc_getacl(struct rpc_rqst *req, __be32 *p,
2102 2103 2104 2105
		struct nfs_getaclargs *args)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2106
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
2107
	};
2108
	uint32_t replen;
2109 2110

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2111
	encode_compound_hdr(&xdr, req, &hdr);
2112
	encode_sequence(&xdr, &args->seq_args, &hdr);
2113
	encode_putfh(&xdr, args->fh, &hdr);
2114
	replen = hdr.replen + nfs4_fattr_bitmap_maxsz + 1;
2115 2116
	encode_getattr_two(&xdr, FATTR4_WORD0_ACL, 0, &hdr);

2117
	xdr_inline_pages(&req->rq_rcv_buf, replen << 2,
2118
		args->acl_pages, args->acl_pgbase, args->acl_len);
2119
	encode_nops(&hdr);
2120
	return 0;
2121 2122
}

L
Linus Torvalds 已提交
2123 2124 2125
/*
 * Encode a WRITE request
 */
A
Al Viro 已提交
2126
static int nfs4_xdr_enc_write(struct rpc_rqst *req, __be32 *p, struct nfs_writeargs *args)
L
Linus Torvalds 已提交
2127 2128 2129
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2130
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2131 2132 2133
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2134
	encode_compound_hdr(&xdr, req, &hdr);
2135
	encode_sequence(&xdr, &args->seq_args, &hdr);
2136 2137
	encode_putfh(&xdr, args->fh, &hdr);
	encode_write(&xdr, args, &hdr);
2138
	req->rq_snd_buf.flags |= XDRBUF_WRITE;
2139
	encode_getfattr(&xdr, args->bitmask, &hdr);
2140
	encode_nops(&hdr);
2141
	return 0;
L
Linus Torvalds 已提交
2142 2143 2144 2145 2146
}

/*
 *  a COMMIT request
 */
A
Al Viro 已提交
2147
static int nfs4_xdr_enc_commit(struct rpc_rqst *req, __be32 *p, struct nfs_writeargs *args)
L
Linus Torvalds 已提交
2148 2149 2150
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2151
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2152 2153 2154
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2155
	encode_compound_hdr(&xdr, req, &hdr);
2156
	encode_sequence(&xdr, &args->seq_args, &hdr);
2157 2158 2159
	encode_putfh(&xdr, args->fh, &hdr);
	encode_commit(&xdr, args, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
2160
	encode_nops(&hdr);
2161
	return 0;
L
Linus Torvalds 已提交
2162 2163 2164 2165 2166
}

/*
 * FSINFO request
 */
A
Al Viro 已提交
2167
static int nfs4_xdr_enc_fsinfo(struct rpc_rqst *req, __be32 *p, struct nfs4_fsinfo_arg *args)
L
Linus Torvalds 已提交
2168 2169 2170
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2171
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2172 2173 2174
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2175
	encode_compound_hdr(&xdr, req, &hdr);
2176
	encode_sequence(&xdr, &args->seq_args, &hdr);
2177 2178
	encode_putfh(&xdr, args->fh, &hdr);
	encode_fsinfo(&xdr, args->bitmask, &hdr);
2179
	encode_nops(&hdr);
2180
	return 0;
L
Linus Torvalds 已提交
2181 2182 2183 2184 2185
}

/*
 * a PATHCONF request
 */
A
Al Viro 已提交
2186
static int nfs4_xdr_enc_pathconf(struct rpc_rqst *req, __be32 *p, const struct nfs4_pathconf_arg *args)
L
Linus Torvalds 已提交
2187 2188 2189
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2190
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2191 2192 2193
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2194
	encode_compound_hdr(&xdr, req, &hdr);
2195
	encode_sequence(&xdr, &args->seq_args, &hdr);
2196 2197 2198
	encode_putfh(&xdr, args->fh, &hdr);
	encode_getattr_one(&xdr, args->bitmask[0] & nfs4_pathconf_bitmap[0],
			   &hdr);
2199
	encode_nops(&hdr);
2200
	return 0;
L
Linus Torvalds 已提交
2201 2202 2203 2204 2205
}

/*
 * a STATFS request
 */
A
Al Viro 已提交
2206
static int nfs4_xdr_enc_statfs(struct rpc_rqst *req, __be32 *p, const struct nfs4_statfs_arg *args)
L
Linus Torvalds 已提交
2207 2208 2209
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2210
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2211 2212 2213
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2214
	encode_compound_hdr(&xdr, req, &hdr);
2215
	encode_sequence(&xdr, &args->seq_args, &hdr);
2216 2217 2218
	encode_putfh(&xdr, args->fh, &hdr);
	encode_getattr_two(&xdr, args->bitmask[0] & nfs4_statfs_bitmap[0],
			   args->bitmask[1] & nfs4_statfs_bitmap[1], &hdr);
2219
	encode_nops(&hdr);
2220
	return 0;
L
Linus Torvalds 已提交
2221 2222 2223 2224 2225
}

/*
 * GETATTR_BITMAP request
 */
B
Benny Halevy 已提交
2226 2227
static int nfs4_xdr_enc_server_caps(struct rpc_rqst *req, __be32 *p,
				    struct nfs4_server_caps_arg *args)
L
Linus Torvalds 已提交
2228 2229 2230
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2231
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2232 2233 2234
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2235
	encode_compound_hdr(&xdr, req, &hdr);
2236
	encode_sequence(&xdr, &args->seq_args, &hdr);
B
Benny Halevy 已提交
2237
	encode_putfh(&xdr, args->fhandle, &hdr);
2238 2239 2240 2241
	encode_getattr_one(&xdr, FATTR4_WORD0_SUPPORTED_ATTRS|
			   FATTR4_WORD0_LINK_SUPPORT|
			   FATTR4_WORD0_SYMLINK_SUPPORT|
			   FATTR4_WORD0_ACLSUPPORT, &hdr);
2242
	encode_nops(&hdr);
2243
	return 0;
L
Linus Torvalds 已提交
2244 2245 2246 2247 2248
}

/*
 * a RENEW request
 */
A
Al Viro 已提交
2249
static int nfs4_xdr_enc_renew(struct rpc_rqst *req, __be32 *p, struct nfs_client *clp)
L
Linus Torvalds 已提交
2250 2251 2252
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2253
		.nops	= 0,
L
Linus Torvalds 已提交
2254 2255 2256
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2257
	encode_compound_hdr(&xdr, req, &hdr);
2258
	encode_renew(&xdr, clp, &hdr);
2259
	encode_nops(&hdr);
2260
	return 0;
L
Linus Torvalds 已提交
2261 2262 2263 2264 2265
}

/*
 * a SETCLIENTID request
 */
A
Al Viro 已提交
2266
static int nfs4_xdr_enc_setclientid(struct rpc_rqst *req, __be32 *p, struct nfs4_setclientid *sc)
L
Linus Torvalds 已提交
2267 2268 2269
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2270
		.nops	= 0,
L
Linus Torvalds 已提交
2271 2272 2273
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2274
	encode_compound_hdr(&xdr, req, &hdr);
2275
	encode_setclientid(&xdr, sc, &hdr);
2276
	encode_nops(&hdr);
2277
	return 0;
L
Linus Torvalds 已提交
2278 2279 2280 2281 2282
}

/*
 * a SETCLIENTID_CONFIRM request
 */
A
Al Viro 已提交
2283
static int nfs4_xdr_enc_setclientid_confirm(struct rpc_rqst *req, __be32 *p, struct nfs_client *clp)
L
Linus Torvalds 已提交
2284 2285 2286
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2287
		.nops	= 0,
L
Linus Torvalds 已提交
2288 2289 2290 2291
	};
	const u32 lease_bitmap[2] = { FATTR4_WORD0_LEASE_TIME, 0 };

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2292
	encode_compound_hdr(&xdr, req, &hdr);
2293 2294 2295
	encode_setclientid_confirm(&xdr, clp, &hdr);
	encode_putrootfh(&xdr, &hdr);
	encode_fsinfo(&xdr, lease_bitmap, &hdr);
2296
	encode_nops(&hdr);
2297
	return 0;
L
Linus Torvalds 已提交
2298 2299 2300 2301 2302
}

/*
 * DELEGRETURN request
 */
A
Al Viro 已提交
2303
static int nfs4_xdr_enc_delegreturn(struct rpc_rqst *req, __be32 *p, const struct nfs4_delegreturnargs *args)
L
Linus Torvalds 已提交
2304 2305 2306
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2307
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
L
Linus Torvalds 已提交
2308 2309 2310
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2311
	encode_compound_hdr(&xdr, req, &hdr);
2312
	encode_sequence(&xdr, &args->seq_args, &hdr);
2313 2314 2315
	encode_putfh(&xdr, args->fhandle, &hdr);
	encode_delegreturn(&xdr, args->stateid, &hdr);
	encode_getfattr(&xdr, args->bitmask, &hdr);
2316
	encode_nops(&hdr);
2317
	return 0;
L
Linus Torvalds 已提交
2318 2319
}

2320 2321 2322
/*
 * Encode FS_LOCATIONS request
 */
A
Al Viro 已提交
2323
static int nfs4_xdr_enc_fs_locations(struct rpc_rqst *req, __be32 *p, struct nfs4_fs_locations_arg *args)
2324 2325 2326
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
2327
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
2328
	};
2329
	uint32_t replen;
2330 2331

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
2332
	encode_compound_hdr(&xdr, req, &hdr);
2333
	encode_sequence(&xdr, &args->seq_args, &hdr);
2334 2335
	encode_putfh(&xdr, args->dir_fh, &hdr);
	encode_lookup(&xdr, args->name, &hdr);
2336
	replen = hdr.replen;	/* get the attribute into args->page */
2337 2338
	encode_fs_locations(&xdr, args->bitmask, &hdr);

2339
	xdr_inline_pages(&req->rq_rcv_buf, replen << 2, &args->page,
2340
			0, PAGE_SIZE);
2341
	encode_nops(&hdr);
2342
	return 0;
2343 2344
}

B
Benny Halevy 已提交
2345 2346 2347 2348 2349 2350 2351 2352 2353 2354 2355 2356 2357 2358 2359 2360 2361 2362
#if defined(CONFIG_NFS_V4_1)
/*
 * EXCHANGE_ID request
 */
static int nfs4_xdr_enc_exchange_id(struct rpc_rqst *req, uint32_t *p,
				    struct nfs41_exchange_id_args *args)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
		.minorversion = args->client->cl_minorversion,
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
	encode_compound_hdr(&xdr, req, &hdr);
	encode_exchange_id(&xdr, args, &hdr);
	encode_nops(&hdr);
	return 0;
}
A
Andy Adamson 已提交
2363

A
Andy Adamson 已提交
2364 2365 2366 2367 2368 2369 2370 2371 2372 2373 2374 2375 2376 2377 2378 2379 2380 2381
/*
 * a CREATE_SESSION request
 */
static int nfs4_xdr_enc_create_session(struct rpc_rqst *req, uint32_t *p,
				       struct nfs41_create_session_args *args)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
		.minorversion = args->client->cl_minorversion,
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
	encode_compound_hdr(&xdr, req, &hdr);
	encode_create_session(&xdr, args, &hdr);
	encode_nops(&hdr);
	return 0;
}

A
Andy Adamson 已提交
2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392 2393 2394 2395 2396 2397 2398 2399
/*
 * a DESTROY_SESSION request
 */
static int nfs4_xdr_enc_destroy_session(struct rpc_rqst *req, uint32_t *p,
					struct nfs4_session *session)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
		.minorversion = session->clp->cl_minorversion,
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
	encode_compound_hdr(&xdr, req, &hdr);
	encode_destroy_session(&xdr, session, &hdr);
	encode_nops(&hdr);
	return 0;
}

A
Andy Adamson 已提交
2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417
/*
 * a SEQUENCE request
 */
static int nfs4_xdr_enc_sequence(struct rpc_rqst *req, uint32_t *p,
				 struct nfs4_sequence_args *args)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
		.minorversion = nfs4_xdr_minorversion(args),
	};

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
	encode_compound_hdr(&xdr, req, &hdr);
	encode_sequence(&xdr, args, &hdr);
	encode_nops(&hdr);
	return 0;
}

A
Andy Adamson 已提交
2418 2419 2420 2421 2422 2423 2424 2425 2426 2427 2428 2429 2430 2431 2432 2433 2434 2435 2436 2437
/*
 * a GET_LEASE_TIME request
 */
static int nfs4_xdr_enc_get_lease_time(struct rpc_rqst *req, uint32_t *p,
				       struct nfs4_get_lease_time_args *args)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
		.minorversion = nfs4_xdr_minorversion(&args->la_seq_args),
	};
	const u32 lease_bitmap[2] = { FATTR4_WORD0_LEASE_TIME, 0 };

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
	encode_compound_hdr(&xdr, req, &hdr);
	encode_sequence(&xdr, &args->la_seq_args, &hdr);
	encode_putrootfh(&xdr, &hdr);
	encode_fsinfo(&xdr, lease_bitmap, &hdr);
	encode_nops(&hdr);
	return 0;
}
B
Benny Halevy 已提交
2438 2439
#endif /* CONFIG_NFS_V4_1 */

L
Linus Torvalds 已提交
2440 2441 2442 2443 2444 2445 2446 2447 2448 2449 2450 2451 2452 2453 2454 2455 2456 2457 2458 2459 2460 2461 2462 2463 2464 2465 2466
/*
 * START OF "GENERIC" DECODE ROUTINES.
 *   These may look a little ugly since they are imported from a "generic"
 * set of XDR encode/decode routines which are intended to be shared by
 * all of our NFSv4 implementations (OpenBSD, MacOS X...).
 *
 * If the pain of reading these is too great, it should be a straightforward
 * task to translate them into Linux-specific versions which are more
 * consistent with the style used in NFSv2/v3...
 */
#define READ32(x)         (x) = ntohl(*p++)
#define READ64(x)         do {			\
	(x) = (u64)ntohl(*p++) << 32;		\
	(x) |= ntohl(*p++);			\
} while (0)
#define READTIME(x)       do {			\
	p++;					\
	(x.tv_sec) = ntohl(*p++);		\
	(x.tv_nsec) = ntohl(*p++);		\
} while (0)
#define COPYMEM(x,nbytes) do {			\
	memcpy((x), p, nbytes);			\
	p += XDR_QUADLEN(nbytes);		\
} while (0)

#define READ_BUF(nbytes)  do { \
	p = xdr_inline_decode(xdr, nbytes); \
2467
	if (unlikely(!p)) { \
2468
		dprintk("nfs: %s: prematurely hit end of receive" \
2469
				" buffer\n", __func__); \
2470
		dprintk("nfs: %s: xdr->p=%p, bytes=%u, xdr->end=%p\n", \
2471
				__func__, xdr->p, nbytes, xdr->end); \
L
Linus Torvalds 已提交
2472 2473 2474 2475
		return -EIO; \
	} \
} while (0)

2476
static int decode_opaque_inline(struct xdr_stream *xdr, unsigned int *len, char **string)
L
Linus Torvalds 已提交
2477
{
A
Al Viro 已提交
2478
	__be32 *p;
L
Linus Torvalds 已提交
2479 2480 2481 2482 2483 2484 2485 2486 2487 2488

	READ_BUF(4);
	READ32(*len);
	READ_BUF(*len);
	*string = (char *)p;
	return 0;
}

static int decode_compound_hdr(struct xdr_stream *xdr, struct compound_hdr *hdr)
{
A
Al Viro 已提交
2489
	__be32 *p;
L
Linus Torvalds 已提交
2490 2491 2492 2493

	READ_BUF(8);
	READ32(hdr->status);
	READ32(hdr->taglen);
2494

L
Linus Torvalds 已提交
2495 2496 2497 2498
	READ_BUF(hdr->taglen + 4);
	hdr->tag = (char *)p;
	p += XDR_QUADLEN(hdr->taglen);
	READ32(hdr->nops);
2499 2500
	if (unlikely(hdr->nops < 1))
		return nfs4_stat_to_errno(hdr->status);
L
Linus Torvalds 已提交
2501 2502 2503 2504 2505
	return 0;
}

static int decode_op_hdr(struct xdr_stream *xdr, enum nfs_opnum4 expected)
{
A
Al Viro 已提交
2506
	__be32 *p;
L
Linus Torvalds 已提交
2507 2508 2509 2510 2511 2512
	uint32_t opnum;
	int32_t nfserr;

	READ_BUF(8);
	READ32(opnum);
	if (opnum != expected) {
2513 2514
		dprintk("nfs: Server returned operation"
			" %d but we issued a request for %d\n",
L
Linus Torvalds 已提交
2515 2516 2517 2518 2519
				opnum, expected);
		return -EIO;
	}
	READ32(nfserr);
	if (nfserr != NFS_OK)
2520
		return nfs4_stat_to_errno(nfserr);
L
Linus Torvalds 已提交
2521 2522 2523 2524
	return 0;
}

/* Dummy routine */
2525
static int decode_ace(struct xdr_stream *xdr, void *ace, struct nfs_client *clp)
L
Linus Torvalds 已提交
2526
{
A
Al Viro 已提交
2527
	__be32 *p;
2528
	unsigned int strlen;
L
Linus Torvalds 已提交
2529 2530 2531 2532 2533 2534 2535 2536
	char *str;

	READ_BUF(12);
	return decode_opaque_inline(xdr, &strlen, &str);
}

static int decode_attr_bitmap(struct xdr_stream *xdr, uint32_t *bitmap)
{
A
Al Viro 已提交
2537 2538
	uint32_t bmlen;
	__be32 *p;
L
Linus Torvalds 已提交
2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552

	READ_BUF(4);
	READ32(bmlen);

	bitmap[0] = bitmap[1] = 0;
	READ_BUF((bmlen << 2));
	if (bmlen > 0) {
		READ32(bitmap[0]);
		if (bmlen > 1)
			READ32(bitmap[1]);
	}
	return 0;
}

A
Al Viro 已提交
2553
static inline int decode_attr_length(struct xdr_stream *xdr, uint32_t *attrlen, __be32 **savep)
L
Linus Torvalds 已提交
2554
{
A
Al Viro 已提交
2555
	__be32 *p;
L
Linus Torvalds 已提交
2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569

	READ_BUF(4);
	READ32(*attrlen);
	*savep = xdr->p;
	return 0;
}

static int decode_attr_supported(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *bitmask)
{
	if (likely(bitmap[0] & FATTR4_WORD0_SUPPORTED_ATTRS)) {
		decode_attr_bitmap(xdr, bitmask);
		bitmap[0] &= ~FATTR4_WORD0_SUPPORTED_ATTRS;
	} else
		bitmask[0] = bitmask[1] = 0;
2570
	dprintk("%s: bitmask=%08x:%08x\n", __func__, bitmask[0], bitmask[1]);
L
Linus Torvalds 已提交
2571 2572 2573 2574 2575
	return 0;
}

static int decode_attr_type(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *type)
{
A
Al Viro 已提交
2576
	__be32 *p;
2577
	int ret = 0;
L
Linus Torvalds 已提交
2578 2579 2580 2581 2582 2583 2584 2585

	*type = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_TYPE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_TYPE)) {
		READ_BUF(4);
		READ32(*type);
		if (*type < NF4REG || *type > NF4NAMEDATTR) {
2586
			dprintk("%s: bad type %d\n", __func__, *type);
L
Linus Torvalds 已提交
2587 2588 2589
			return -EIO;
		}
		bitmap[0] &= ~FATTR4_WORD0_TYPE;
2590
		ret = NFS_ATTR_FATTR_TYPE;
L
Linus Torvalds 已提交
2591
	}
2592
	dprintk("%s: type=0%o\n", __func__, nfs_type2fmt[*type]);
2593
	return ret;
L
Linus Torvalds 已提交
2594 2595 2596 2597
}

static int decode_attr_change(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *change)
{
A
Al Viro 已提交
2598
	__be32 *p;
2599
	int ret = 0;
L
Linus Torvalds 已提交
2600 2601 2602 2603 2604 2605 2606 2607

	*change = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_CHANGE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_CHANGE)) {
		READ_BUF(8);
		READ64(*change);
		bitmap[0] &= ~FATTR4_WORD0_CHANGE;
2608
		ret = NFS_ATTR_FATTR_CHANGE;
L
Linus Torvalds 已提交
2609
	}
2610
	dprintk("%s: change attribute=%Lu\n", __func__,
L
Linus Torvalds 已提交
2611
			(unsigned long long)*change);
2612
	return ret;
L
Linus Torvalds 已提交
2613 2614 2615 2616
}

static int decode_attr_size(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *size)
{
A
Al Viro 已提交
2617
	__be32 *p;
2618
	int ret = 0;
L
Linus Torvalds 已提交
2619 2620 2621 2622 2623 2624 2625 2626

	*size = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_SIZE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_SIZE)) {
		READ_BUF(8);
		READ64(*size);
		bitmap[0] &= ~FATTR4_WORD0_SIZE;
2627
		ret = NFS_ATTR_FATTR_SIZE;
L
Linus Torvalds 已提交
2628
	}
2629
	dprintk("%s: file size=%Lu\n", __func__, (unsigned long long)*size);
2630
	return ret;
L
Linus Torvalds 已提交
2631 2632 2633 2634
}

static int decode_attr_link_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2635
	__be32 *p;
L
Linus Torvalds 已提交
2636 2637 2638 2639 2640 2641 2642 2643 2644

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_LINK_SUPPORT - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_LINK_SUPPORT)) {
		READ_BUF(4);
		READ32(*res);
		bitmap[0] &= ~FATTR4_WORD0_LINK_SUPPORT;
	}
2645
	dprintk("%s: link support=%s\n", __func__, *res == 0 ? "false" : "true");
L
Linus Torvalds 已提交
2646 2647 2648 2649 2650
	return 0;
}

static int decode_attr_symlink_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2651
	__be32 *p;
L
Linus Torvalds 已提交
2652 2653 2654 2655 2656 2657 2658 2659 2660

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_SYMLINK_SUPPORT - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_SYMLINK_SUPPORT)) {
		READ_BUF(4);
		READ32(*res);
		bitmap[0] &= ~FATTR4_WORD0_SYMLINK_SUPPORT;
	}
2661
	dprintk("%s: symlink support=%s\n", __func__, *res == 0 ? "false" : "true");
L
Linus Torvalds 已提交
2662 2663 2664
	return 0;
}

2665
static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_fsid *fsid)
L
Linus Torvalds 已提交
2666
{
A
Al Viro 已提交
2667
	__be32 *p;
2668
	int ret = 0;
L
Linus Torvalds 已提交
2669 2670 2671 2672 2673 2674 2675 2676 2677 2678

	fsid->major = 0;
	fsid->minor = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_FSID - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_FSID)) {
		READ_BUF(16);
		READ64(fsid->major);
		READ64(fsid->minor);
		bitmap[0] &= ~FATTR4_WORD0_FSID;
2679
		ret = NFS_ATTR_FATTR_FSID;
L
Linus Torvalds 已提交
2680
	}
2681
	dprintk("%s: fsid=(0x%Lx/0x%Lx)\n", __func__,
L
Linus Torvalds 已提交
2682 2683
			(unsigned long long)fsid->major,
			(unsigned long long)fsid->minor);
2684
	return ret;
L
Linus Torvalds 已提交
2685 2686 2687 2688
}

static int decode_attr_lease_time(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2689
	__be32 *p;
L
Linus Torvalds 已提交
2690 2691 2692 2693 2694 2695 2696 2697 2698

	*res = 60;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_LEASE_TIME - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_LEASE_TIME)) {
		READ_BUF(4);
		READ32(*res);
		bitmap[0] &= ~FATTR4_WORD0_LEASE_TIME;
	}
2699
	dprintk("%s: file size=%u\n", __func__, (unsigned int)*res);
L
Linus Torvalds 已提交
2700 2701 2702 2703 2704
	return 0;
}

static int decode_attr_aclsupport(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2705
	__be32 *p;
L
Linus Torvalds 已提交
2706 2707 2708 2709 2710 2711 2712 2713 2714

	*res = ACL4_SUPPORT_ALLOW_ACL|ACL4_SUPPORT_DENY_ACL;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_ACLSUPPORT - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_ACLSUPPORT)) {
		READ_BUF(4);
		READ32(*res);
		bitmap[0] &= ~FATTR4_WORD0_ACLSUPPORT;
	}
2715
	dprintk("%s: ACLs supported=%u\n", __func__, (unsigned int)*res);
L
Linus Torvalds 已提交
2716 2717 2718 2719 2720
	return 0;
}

static int decode_attr_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid)
{
A
Al Viro 已提交
2721
	__be32 *p;
2722
	int ret = 0;
L
Linus Torvalds 已提交
2723 2724 2725 2726 2727 2728 2729 2730

	*fileid = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_FILEID - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_FILEID)) {
		READ_BUF(8);
		READ64(*fileid);
		bitmap[0] &= ~FATTR4_WORD0_FILEID;
2731
		ret = NFS_ATTR_FATTR_FILEID;
L
Linus Torvalds 已提交
2732
	}
2733
	dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid);
2734
	return ret;
L
Linus Torvalds 已提交
2735 2736
}

2737 2738
static int decode_attr_mounted_on_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid)
{
A
Al Viro 已提交
2739
	__be32 *p;
2740
	int ret = 0;
2741 2742 2743 2744 2745 2746 2747 2748

	*fileid = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_MOUNTED_ON_FILEID - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)) {
		READ_BUF(8);
		READ64(*fileid);
		bitmap[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID;
2749
		ret = NFS_ATTR_FATTR_FILEID;
2750
	}
2751
	dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid);
2752
	return ret;
2753 2754
}

L
Linus Torvalds 已提交
2755 2756
static int decode_attr_files_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
2757
	__be32 *p;
L
Linus Torvalds 已提交
2758 2759 2760 2761 2762 2763 2764 2765 2766 2767
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_AVAIL - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_FILES_AVAIL)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[0] &= ~FATTR4_WORD0_FILES_AVAIL;
	}
2768
	dprintk("%s: files avail=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
2769 2770 2771 2772 2773
	return status;
}

static int decode_attr_files_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
2774
	__be32 *p;
L
Linus Torvalds 已提交
2775 2776 2777 2778 2779 2780 2781 2782 2783 2784
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_FREE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_FILES_FREE)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[0] &= ~FATTR4_WORD0_FILES_FREE;
	}
2785
	dprintk("%s: files free=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
2786 2787 2788 2789 2790
	return status;
}

static int decode_attr_files_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
2791
	__be32 *p;
L
Linus Torvalds 已提交
2792 2793 2794 2795 2796 2797 2798 2799 2800 2801
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_TOTAL - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_FILES_TOTAL)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[0] &= ~FATTR4_WORD0_FILES_TOTAL;
	}
2802
	dprintk("%s: files total=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
2803 2804 2805
	return status;
}

2806 2807
static int decode_pathname(struct xdr_stream *xdr, struct nfs4_pathname *path)
{
2808
	u32 n;
A
Al Viro 已提交
2809
	__be32 *p;
2810 2811 2812 2813
	int status = 0;

	READ_BUF(4);
	READ32(n);
2814 2815
	if (n == 0)
		goto root_path;
2816 2817 2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831 2832 2833 2834 2835
	dprintk("path ");
	path->ncomponents = 0;
	while (path->ncomponents < n) {
		struct nfs4_string *component = &path->components[path->ncomponents];
		status = decode_opaque_inline(xdr, &component->len, &component->data);
		if (unlikely(status != 0))
			goto out_eio;
		if (path->ncomponents != n)
			dprintk("/");
		dprintk("%s", component->data);
		if (path->ncomponents < NFS4_PATHNAME_MAXCOMPONENTS)
			path->ncomponents++;
		else {
			dprintk("cannot parse %d components in path\n", n);
			goto out_eio;
		}
	}
out:
	dprintk("\n");
	return status;
2836 2837 2838 2839 2840 2841 2842
root_path:
/* a root pathname is sent as a zero component4 */
	path->ncomponents = 1;
	path->components[0].len=0;
	path->components[0].data=NULL;
	dprintk("path /\n");
	goto out;
2843 2844 2845 2846 2847 2848 2849
out_eio:
	dprintk(" status %d", status);
	status = -EIO;
	goto out;
}

static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_fs_locations *res)
2850 2851
{
	int n;
A
Al Viro 已提交
2852
	__be32 *p;
2853 2854 2855 2856 2857 2858 2859
	int status = -EIO;

	if (unlikely(bitmap[0] & (FATTR4_WORD0_FS_LOCATIONS -1U)))
		goto out;
	status = 0;
	if (unlikely(!(bitmap[0] & FATTR4_WORD0_FS_LOCATIONS)))
		goto out;
2860
	dprintk("%s: fsroot ", __func__);
2861
	status = decode_pathname(xdr, &res->fs_path);
2862 2863 2864 2865 2866 2867 2868 2869
	if (unlikely(status != 0))
		goto out;
	READ_BUF(4);
	READ32(n);
	if (n <= 0)
		goto out_eio;
	res->nlocations = 0;
	while (res->nlocations < n) {
2870
		u32 m;
2871
		struct nfs4_fs_location *loc = &res->locations[res->nlocations];
2872

2873 2874 2875 2876
		READ_BUF(4);
		READ32(m);

		loc->nservers = 0;
2877
		dprintk("%s: servers ", __func__);
2878 2879 2880 2881 2882 2883 2884 2885 2886
		while (loc->nservers < m) {
			struct nfs4_string *server = &loc->servers[loc->nservers];
			status = decode_opaque_inline(xdr, &server->len, &server->data);
			if (unlikely(status != 0))
				goto out_eio;
			dprintk("%s ", server->data);
			if (loc->nservers < NFS4_FS_LOCATION_MAXSERVERS)
				loc->nservers++;
			else {
2887 2888 2889
				unsigned int i;
				dprintk("%s: using first %u of %u servers "
					"returned for location %u\n",
2890
						__func__,
2891 2892
						NFS4_FS_LOCATION_MAXSERVERS,
						m, res->nlocations);
2893
				for (i = loc->nservers; i < m; i++) {
T
Trond Myklebust 已提交
2894
					unsigned int len;
2895 2896 2897 2898 2899 2900 2901 2902
					char *data;
					status = decode_opaque_inline(xdr, &len, &data);
					if (unlikely(status != 0))
						goto out_eio;
				}
			}
		}
		status = decode_pathname(xdr, &loc->rootpath);
2903 2904
		if (unlikely(status != 0))
			goto out_eio;
2905
		if (res->nlocations < NFS4_FS_LOCATIONS_MAXENTRIES)
2906 2907
			res->nlocations++;
	}
2908 2909
	if (res->nlocations != 0)
		status = NFS_ATTR_FATTR_V4_REFERRAL;
2910
out:
2911
	dprintk("%s: fs_locations done, error = %d\n", __func__, status);
2912 2913 2914 2915 2916 2917
	return status;
out_eio:
	status = -EIO;
	goto out;
}

L
Linus Torvalds 已提交
2918 2919
static int decode_attr_maxfilesize(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
2920
	__be32 *p;
L
Linus Torvalds 已提交
2921 2922 2923 2924 2925 2926 2927 2928 2929 2930
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXFILESIZE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_MAXFILESIZE)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[0] &= ~FATTR4_WORD0_MAXFILESIZE;
	}
2931
	dprintk("%s: maxfilesize=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
2932 2933 2934 2935 2936
	return status;
}

static int decode_attr_maxlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxlink)
{
A
Al Viro 已提交
2937
	__be32 *p;
L
Linus Torvalds 已提交
2938 2939 2940 2941 2942 2943 2944 2945 2946 2947
	int status = 0;

	*maxlink = 1;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXLINK - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_MAXLINK)) {
		READ_BUF(4);
		READ32(*maxlink);
		bitmap[0] &= ~FATTR4_WORD0_MAXLINK;
	}
2948
	dprintk("%s: maxlink=%u\n", __func__, *maxlink);
L
Linus Torvalds 已提交
2949 2950 2951 2952 2953
	return status;
}

static int decode_attr_maxname(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxname)
{
A
Al Viro 已提交
2954
	__be32 *p;
L
Linus Torvalds 已提交
2955 2956 2957 2958 2959 2960 2961 2962 2963 2964
	int status = 0;

	*maxname = 1024;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXNAME - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_MAXNAME)) {
		READ_BUF(4);
		READ32(*maxname);
		bitmap[0] &= ~FATTR4_WORD0_MAXNAME;
	}
2965
	dprintk("%s: maxname=%u\n", __func__, *maxname);
L
Linus Torvalds 已提交
2966 2967 2968 2969 2970
	return status;
}

static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2971
	__be32 *p;
L
Linus Torvalds 已提交
2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985
	int status = 0;

	*res = 1024;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXREAD - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_MAXREAD)) {
		uint64_t maxread;
		READ_BUF(8);
		READ64(maxread);
		if (maxread > 0x7FFFFFFF)
			maxread = 0x7FFFFFFF;
		*res = (uint32_t)maxread;
		bitmap[0] &= ~FATTR4_WORD0_MAXREAD;
	}
2986
	dprintk("%s: maxread=%lu\n", __func__, (unsigned long)*res);
L
Linus Torvalds 已提交
2987 2988 2989 2990 2991
	return status;
}

static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
{
A
Al Viro 已提交
2992
	__be32 *p;
L
Linus Torvalds 已提交
2993 2994 2995 2996 2997 2998 2999 3000 3001 3002 3003 3004 3005 3006
	int status = 0;

	*res = 1024;
	if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXWRITE - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_MAXWRITE)) {
		uint64_t maxwrite;
		READ_BUF(8);
		READ64(maxwrite);
		if (maxwrite > 0x7FFFFFFF)
			maxwrite = 0x7FFFFFFF;
		*res = (uint32_t)maxwrite;
		bitmap[0] &= ~FATTR4_WORD0_MAXWRITE;
	}
3007
	dprintk("%s: maxwrite=%lu\n", __func__, (unsigned long)*res);
L
Linus Torvalds 已提交
3008 3009 3010
	return status;
}

3011
static int decode_attr_mode(struct xdr_stream *xdr, uint32_t *bitmap, umode_t *mode)
L
Linus Torvalds 已提交
3012
{
3013
	uint32_t tmp;
A
Al Viro 已提交
3014
	__be32 *p;
3015
	int ret = 0;
L
Linus Torvalds 已提交
3016 3017 3018 3019 3020 3021

	*mode = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_MODE - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_MODE)) {
		READ_BUF(4);
3022 3023
		READ32(tmp);
		*mode = tmp & ~S_IFMT;
L
Linus Torvalds 已提交
3024
		bitmap[1] &= ~FATTR4_WORD1_MODE;
3025
		ret = NFS_ATTR_FATTR_MODE;
L
Linus Torvalds 已提交
3026
	}
3027
	dprintk("%s: file mode=0%o\n", __func__, (unsigned int)*mode);
3028
	return ret;
L
Linus Torvalds 已提交
3029 3030 3031 3032
}

static int decode_attr_nlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *nlink)
{
A
Al Viro 已提交
3033
	__be32 *p;
3034
	int ret = 0;
L
Linus Torvalds 已提交
3035 3036 3037 3038 3039 3040 3041 3042

	*nlink = 1;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_NUMLINKS - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_NUMLINKS)) {
		READ_BUF(4);
		READ32(*nlink);
		bitmap[1] &= ~FATTR4_WORD1_NUMLINKS;
3043
		ret = NFS_ATTR_FATTR_NLINK;
L
Linus Torvalds 已提交
3044
	}
3045
	dprintk("%s: nlink=%u\n", __func__, (unsigned int)*nlink);
3046
	return ret;
L
Linus Torvalds 已提交
3047 3048
}

T
Trond Myklebust 已提交
3049
static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_client *clp, uint32_t *uid)
L
Linus Torvalds 已提交
3050
{
A
Al Viro 已提交
3051 3052
	uint32_t len;
	__be32 *p;
3053
	int ret = 0;
L
Linus Torvalds 已提交
3054 3055 3056 3057 3058 3059 3060 3061 3062

	*uid = -2;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_OWNER)) {
		READ_BUF(4);
		READ32(len);
		READ_BUF(len);
		if (len < XDR_MAX_NETOBJ) {
3063 3064 3065
			if (nfs_map_name_to_uid(clp, (char *)p, len, uid) == 0)
				ret = NFS_ATTR_FATTR_OWNER;
			else
L
Linus Torvalds 已提交
3066
				dprintk("%s: nfs_map_name_to_uid failed!\n",
3067
						__func__);
L
Linus Torvalds 已提交
3068
		} else
3069
			dprintk("%s: name too long (%u)!\n",
3070
					__func__, len);
L
Linus Torvalds 已提交
3071 3072
		bitmap[1] &= ~FATTR4_WORD1_OWNER;
	}
3073
	dprintk("%s: uid=%d\n", __func__, (int)*uid);
3074
	return ret;
L
Linus Torvalds 已提交
3075 3076
}

T
Trond Myklebust 已提交
3077
static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_client *clp, uint32_t *gid)
L
Linus Torvalds 已提交
3078
{
A
Al Viro 已提交
3079 3080
	uint32_t len;
	__be32 *p;
3081
	int ret = 0;
L
Linus Torvalds 已提交
3082 3083 3084 3085 3086 3087 3088 3089 3090

	*gid = -2;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER_GROUP - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_OWNER_GROUP)) {
		READ_BUF(4);
		READ32(len);
		READ_BUF(len);
		if (len < XDR_MAX_NETOBJ) {
3091 3092 3093
			if (nfs_map_group_to_gid(clp, (char *)p, len, gid) == 0)
				ret = NFS_ATTR_FATTR_GROUP;
			else
L
Linus Torvalds 已提交
3094
				dprintk("%s: nfs_map_group_to_gid failed!\n",
3095
						__func__);
L
Linus Torvalds 已提交
3096
		} else
3097
			dprintk("%s: name too long (%u)!\n",
3098
					__func__, len);
L
Linus Torvalds 已提交
3099 3100
		bitmap[1] &= ~FATTR4_WORD1_OWNER_GROUP;
	}
3101
	dprintk("%s: gid=%d\n", __func__, (int)*gid);
3102
	return ret;
L
Linus Torvalds 已提交
3103 3104 3105 3106
}

static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rdev)
{
A
Al Viro 已提交
3107 3108
	uint32_t major = 0, minor = 0;
	__be32 *p;
3109
	int ret = 0;
L
Linus Torvalds 已提交
3110 3111 3112 3113 3114 3115 3116 3117 3118 3119 3120 3121 3122 3123

	*rdev = MKDEV(0,0);
	if (unlikely(bitmap[1] & (FATTR4_WORD1_RAWDEV - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_RAWDEV)) {
		dev_t tmp;

		READ_BUF(8);
		READ32(major);
		READ32(minor);
		tmp = MKDEV(major, minor);
		if (MAJOR(tmp) == major && MINOR(tmp) == minor)
			*rdev = tmp;
		bitmap[1] &= ~ FATTR4_WORD1_RAWDEV;
3124
		ret = NFS_ATTR_FATTR_RDEV;
L
Linus Torvalds 已提交
3125
	}
3126
	dprintk("%s: rdev=(0x%x:0x%x)\n", __func__, major, minor);
3127
	return ret;
L
Linus Torvalds 已提交
3128 3129 3130 3131
}

static int decode_attr_space_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
3132
	__be32 *p;
L
Linus Torvalds 已提交
3133 3134 3135 3136 3137 3138 3139 3140 3141 3142
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_AVAIL - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_SPACE_AVAIL)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[1] &= ~FATTR4_WORD1_SPACE_AVAIL;
	}
3143
	dprintk("%s: space avail=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
3144 3145 3146 3147 3148
	return status;
}

static int decode_attr_space_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
3149
	__be32 *p;
L
Linus Torvalds 已提交
3150 3151 3152 3153 3154 3155 3156 3157 3158 3159
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_FREE - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_SPACE_FREE)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[1] &= ~FATTR4_WORD1_SPACE_FREE;
	}
3160
	dprintk("%s: space free=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
3161 3162 3163 3164 3165
	return status;
}

static int decode_attr_space_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
{
A
Al Viro 已提交
3166
	__be32 *p;
L
Linus Torvalds 已提交
3167 3168 3169 3170 3171 3172 3173 3174 3175 3176
	int status = 0;

	*res = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_TOTAL - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_SPACE_TOTAL)) {
		READ_BUF(8);
		READ64(*res);
		bitmap[1] &= ~FATTR4_WORD1_SPACE_TOTAL;
	}
3177
	dprintk("%s: space total=%Lu\n", __func__, (unsigned long long)*res);
L
Linus Torvalds 已提交
3178 3179 3180 3181 3182
	return status;
}

static int decode_attr_space_used(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *used)
{
A
Al Viro 已提交
3183
	__be32 *p;
3184
	int ret = 0;
L
Linus Torvalds 已提交
3185 3186 3187 3188 3189 3190 3191 3192

	*used = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_USED - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_SPACE_USED)) {
		READ_BUF(8);
		READ64(*used);
		bitmap[1] &= ~FATTR4_WORD1_SPACE_USED;
3193
		ret = NFS_ATTR_FATTR_SPACE_USED;
L
Linus Torvalds 已提交
3194
	}
3195
	dprintk("%s: space used=%Lu\n", __func__,
L
Linus Torvalds 已提交
3196
			(unsigned long long)*used);
3197
	return ret;
L
Linus Torvalds 已提交
3198 3199 3200 3201
}

static int decode_attr_time(struct xdr_stream *xdr, struct timespec *time)
{
A
Al Viro 已提交
3202
	__be32 *p;
L
Linus Torvalds 已提交
3203 3204 3205 3206 3207 3208 3209 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 3220 3221 3222 3223
	uint64_t sec;
	uint32_t nsec;

	READ_BUF(12);
	READ64(sec);
	READ32(nsec);
	time->tv_sec = (time_t)sec;
	time->tv_nsec = (long)nsec;
	return 0;
}

static int decode_attr_time_access(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
{
	int status = 0;

	time->tv_sec = 0;
	time->tv_nsec = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_ACCESS - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_TIME_ACCESS)) {
		status = decode_attr_time(xdr, time);
3224 3225
		if (status == 0)
			status = NFS_ATTR_FATTR_ATIME;
L
Linus Torvalds 已提交
3226 3227
		bitmap[1] &= ~FATTR4_WORD1_TIME_ACCESS;
	}
3228
	dprintk("%s: atime=%ld\n", __func__, (long)time->tv_sec);
L
Linus Torvalds 已提交
3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241
	return status;
}

static int decode_attr_time_metadata(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
{
	int status = 0;

	time->tv_sec = 0;
	time->tv_nsec = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_METADATA - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_TIME_METADATA)) {
		status = decode_attr_time(xdr, time);
3242 3243
		if (status == 0)
			status = NFS_ATTR_FATTR_CTIME;
L
Linus Torvalds 已提交
3244 3245
		bitmap[1] &= ~FATTR4_WORD1_TIME_METADATA;
	}
3246
	dprintk("%s: ctime=%ld\n", __func__, (long)time->tv_sec);
L
Linus Torvalds 已提交
3247 3248 3249 3250 3251 3252 3253 3254 3255 3256 3257 3258 3259
	return status;
}

static int decode_attr_time_modify(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
{
	int status = 0;

	time->tv_sec = 0;
	time->tv_nsec = 0;
	if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_MODIFY - 1U)))
		return -EIO;
	if (likely(bitmap[1] & FATTR4_WORD1_TIME_MODIFY)) {
		status = decode_attr_time(xdr, time);
3260 3261
		if (status == 0)
			status = NFS_ATTR_FATTR_MTIME;
L
Linus Torvalds 已提交
3262 3263
		bitmap[1] &= ~FATTR4_WORD1_TIME_MODIFY;
	}
3264
	dprintk("%s: mtime=%ld\n", __func__, (long)time->tv_sec);
L
Linus Torvalds 已提交
3265 3266 3267
	return status;
}

A
Al Viro 已提交
3268
static int verify_attr_len(struct xdr_stream *xdr, __be32 *savep, uint32_t attrlen)
L
Linus Torvalds 已提交
3269 3270 3271 3272 3273
{
	unsigned int attrwords = XDR_QUADLEN(attrlen);
	unsigned int nwords = xdr->p - savep;

	if (unlikely(attrwords != nwords)) {
3274 3275
		dprintk("%s: server returned incorrect attribute length: "
			"%u %c %u\n",
3276
				__func__,
L
Linus Torvalds 已提交
3277 3278 3279 3280 3281 3282 3283 3284 3285 3286
				attrwords << 2,
				(attrwords < nwords) ? '<' : '>',
				nwords << 2);
		return -EIO;
	}
	return 0;
}

static int decode_change_info(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
{
A
Al Viro 已提交
3287
	__be32 *p;
L
Linus Torvalds 已提交
3288 3289 3290 3291 3292 3293 3294 3295 3296 3297

	READ_BUF(20);
	READ32(cinfo->atomic);
	READ64(cinfo->before);
	READ64(cinfo->after);
	return 0;
}

static int decode_access(struct xdr_stream *xdr, struct nfs4_accessres *access)
{
A
Al Viro 已提交
3298
	__be32 *p;
L
Linus Torvalds 已提交
3299 3300 3301 3302 3303 3304 3305 3306 3307 3308 3309 3310 3311 3312 3313 3314
	uint32_t supp, acc;
	int status;

	status = decode_op_hdr(xdr, OP_ACCESS);
	if (status)
		return status;
	READ_BUF(8);
	READ32(supp);
	READ32(acc);
	access->supported = supp;
	access->access = acc;
	return 0;
}

static int decode_close(struct xdr_stream *xdr, struct nfs_closeres *res)
{
A
Al Viro 已提交
3315
	__be32 *p;
L
Linus Torvalds 已提交
3316 3317 3318
	int status;

	status = decode_op_hdr(xdr, OP_CLOSE);
3319 3320
	if (status != -EIO)
		nfs_increment_open_seqid(status, res->seqid);
L
Linus Torvalds 已提交
3321 3322
	if (status)
		return status;
3323 3324
	READ_BUF(NFS4_STATEID_SIZE);
	COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3325 3326 3327 3328 3329
	return 0;
}

static int decode_commit(struct xdr_stream *xdr, struct nfs_writeres *res)
{
A
Al Viro 已提交
3330
	__be32 *p;
L
Linus Torvalds 已提交
3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342
	int status;

	status = decode_op_hdr(xdr, OP_COMMIT);
	if (status)
		return status;
	READ_BUF(8);
	COPYMEM(res->verf->verifier, 8);
	return 0;
}

static int decode_create(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
{
A
Al Viro 已提交
3343
	__be32 *p;
L
Linus Torvalds 已提交
3344 3345 3346 3347 3348 3349 3350 3351 3352 3353 3354 3355 3356 3357 3358 3359
	uint32_t bmlen;
	int status;

	status = decode_op_hdr(xdr, OP_CREATE);
	if (status)
		return status;
	if ((status = decode_change_info(xdr, cinfo)))
		return status;
	READ_BUF(4);
	READ32(bmlen);
	READ_BUF(bmlen << 2);
	return 0;
}

static int decode_server_caps(struct xdr_stream *xdr, struct nfs4_server_caps_res *res)
{
A
Al Viro 已提交
3360
	__be32 *savep;
3361
	uint32_t attrlen, bitmap[2] = {0};
L
Linus Torvalds 已提交
3362 3363 3364 3365 3366 3367 3368 3369 3370 3371 3372 3373 3374 3375 3376 3377 3378 3379
	int status;

	if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
		goto xdr_error;
	if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
		goto xdr_error;
	if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
		goto xdr_error;
	if ((status = decode_attr_supported(xdr, bitmap, res->attr_bitmask)) != 0)
		goto xdr_error;
	if ((status = decode_attr_link_support(xdr, bitmap, &res->has_links)) != 0)
		goto xdr_error;
	if ((status = decode_attr_symlink_support(xdr, bitmap, &res->has_symlinks)) != 0)
		goto xdr_error;
	if ((status = decode_attr_aclsupport(xdr, bitmap, &res->acl_bitmask)) != 0)
		goto xdr_error;
	status = verify_attr_len(xdr, savep, attrlen);
xdr_error:
3380
	dprintk("%s: xdr returned %d!\n", __func__, -status);
L
Linus Torvalds 已提交
3381 3382
	return status;
}
3383

L
Linus Torvalds 已提交
3384 3385
static int decode_statfs(struct xdr_stream *xdr, struct nfs_fsstat *fsstat)
{
A
Al Viro 已提交
3386
	__be32 *savep;
3387
	uint32_t attrlen, bitmap[2] = {0};
L
Linus Torvalds 已提交
3388
	int status;
3389

L
Linus Torvalds 已提交
3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404 3405 3406 3407 3408 3409 3410 3411
	if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
		goto xdr_error;
	if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
		goto xdr_error;
	if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
		goto xdr_error;

	if ((status = decode_attr_files_avail(xdr, bitmap, &fsstat->afiles)) != 0)
		goto xdr_error;
	if ((status = decode_attr_files_free(xdr, bitmap, &fsstat->ffiles)) != 0)
		goto xdr_error;
	if ((status = decode_attr_files_total(xdr, bitmap, &fsstat->tfiles)) != 0)
		goto xdr_error;
	if ((status = decode_attr_space_avail(xdr, bitmap, &fsstat->abytes)) != 0)
		goto xdr_error;
	if ((status = decode_attr_space_free(xdr, bitmap, &fsstat->fbytes)) != 0)
		goto xdr_error;
	if ((status = decode_attr_space_total(xdr, bitmap, &fsstat->tbytes)) != 0)
		goto xdr_error;

	status = verify_attr_len(xdr, savep, attrlen);
xdr_error:
3412
	dprintk("%s: xdr returned %d!\n", __func__, -status);
L
Linus Torvalds 已提交
3413 3414 3415 3416 3417
	return status;
}

static int decode_pathconf(struct xdr_stream *xdr, struct nfs_pathconf *pathconf)
{
A
Al Viro 已提交
3418
	__be32 *savep;
3419
	uint32_t attrlen, bitmap[2] = {0};
L
Linus Torvalds 已提交
3420
	int status;
3421

L
Linus Torvalds 已提交
3422 3423 3424 3425 3426 3427 3428 3429 3430 3431 3432 3433 3434 3435
	if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
		goto xdr_error;
	if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
		goto xdr_error;
	if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
		goto xdr_error;

	if ((status = decode_attr_maxlink(xdr, bitmap, &pathconf->max_link)) != 0)
		goto xdr_error;
	if ((status = decode_attr_maxname(xdr, bitmap, &pathconf->max_namelen)) != 0)
		goto xdr_error;

	status = verify_attr_len(xdr, savep, attrlen);
xdr_error:
3436
	dprintk("%s: xdr returned %d!\n", __func__, -status);
L
Linus Torvalds 已提交
3437 3438 3439 3440 3441
	return status;
}

static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr, const struct nfs_server *server)
{
A
Al Viro 已提交
3442
	__be32 *savep;
L
Linus Torvalds 已提交
3443 3444 3445
	uint32_t attrlen,
		 bitmap[2] = {0},
		 type;
3446 3447
	int status;
	umode_t fmode = 0;
3448
	uint64_t fileid;
L
Linus Torvalds 已提交
3449

3450 3451
	status = decode_op_hdr(xdr, OP_GETATTR);
	if (status < 0)
L
Linus Torvalds 已提交
3452
		goto xdr_error;
3453 3454 3455

	status = decode_attr_bitmap(xdr, bitmap);
	if (status < 0)
L
Linus Torvalds 已提交
3456 3457
		goto xdr_error;

3458 3459
	status = decode_attr_length(xdr, &attrlen, &savep);
	if (status < 0)
L
Linus Torvalds 已提交
3460 3461 3462
		goto xdr_error;


3463 3464
	status = decode_attr_type(xdr, bitmap, &type);
	if (status < 0)
L
Linus Torvalds 已提交
3465
		goto xdr_error;
3466 3467 3468 3469 3470
	fattr->mode = 0;
	if (status != 0) {
		fattr->mode |= nfs_type2fmt[type];
		fattr->valid |= status;
	}
L
Linus Torvalds 已提交
3471

3472 3473
	status = decode_attr_change(xdr, bitmap, &fattr->change_attr);
	if (status < 0)
L
Linus Torvalds 已提交
3474
		goto xdr_error;
3475
	fattr->valid |= status;
3476 3477 3478

	status = decode_attr_size(xdr, bitmap, &fattr->size);
	if (status < 0)
L
Linus Torvalds 已提交
3479
		goto xdr_error;
3480
	fattr->valid |= status;
3481 3482 3483

	status = decode_attr_fsid(xdr, bitmap, &fattr->fsid);
	if (status < 0)
L
Linus Torvalds 已提交
3484
		goto xdr_error;
3485
	fattr->valid |= status;
3486 3487 3488

	status = decode_attr_fileid(xdr, bitmap, &fattr->fileid);
	if (status < 0)
L
Linus Torvalds 已提交
3489
		goto xdr_error;
3490
	fattr->valid |= status;
3491 3492

	status = decode_attr_fs_locations(xdr, bitmap, container_of(fattr,
3493
						struct nfs4_fs_locations,
3494 3495
						fattr));
	if (status < 0)
3496
		goto xdr_error;
3497
	fattr->valid |= status;
3498 3499 3500

	status = decode_attr_mode(xdr, bitmap, &fmode);
	if (status < 0)
L
Linus Torvalds 已提交
3501
		goto xdr_error;
3502 3503 3504 3505
	if (status != 0) {
		fattr->mode |= fmode;
		fattr->valid |= status;
	}
3506 3507 3508

	status = decode_attr_nlink(xdr, bitmap, &fattr->nlink);
	if (status < 0)
L
Linus Torvalds 已提交
3509
		goto xdr_error;
3510
	fattr->valid |= status;
3511 3512 3513

	status = decode_attr_owner(xdr, bitmap, server->nfs_client, &fattr->uid);
	if (status < 0)
L
Linus Torvalds 已提交
3514
		goto xdr_error;
3515
	fattr->valid |= status;
3516 3517 3518

	status = decode_attr_group(xdr, bitmap, server->nfs_client, &fattr->gid);
	if (status < 0)
L
Linus Torvalds 已提交
3519
		goto xdr_error;
3520
	fattr->valid |= status;
3521 3522 3523

	status = decode_attr_rdev(xdr, bitmap, &fattr->rdev);
	if (status < 0)
L
Linus Torvalds 已提交
3524
		goto xdr_error;
3525
	fattr->valid |= status;
3526 3527 3528

	status = decode_attr_space_used(xdr, bitmap, &fattr->du.nfs3.used);
	if (status < 0)
L
Linus Torvalds 已提交
3529
		goto xdr_error;
3530
	fattr->valid |= status;
3531 3532 3533

	status = decode_attr_time_access(xdr, bitmap, &fattr->atime);
	if (status < 0)
L
Linus Torvalds 已提交
3534
		goto xdr_error;
3535
	fattr->valid |= status;
3536 3537 3538

	status = decode_attr_time_metadata(xdr, bitmap, &fattr->ctime);
	if (status < 0)
L
Linus Torvalds 已提交
3539
		goto xdr_error;
3540
	fattr->valid |= status;
3541 3542 3543

	status = decode_attr_time_modify(xdr, bitmap, &fattr->mtime);
	if (status < 0)
L
Linus Torvalds 已提交
3544
		goto xdr_error;
3545
	fattr->valid |= status;
3546 3547 3548

	status = decode_attr_mounted_on_fileid(xdr, bitmap, &fileid);
	if (status < 0)
3549
		goto xdr_error;
3550
	if (status != 0 && !(fattr->valid & status)) {
3551
		fattr->fileid = fileid;
3552 3553
		fattr->valid |= status;
	}
3554 3555

	status = verify_attr_len(xdr, savep, attrlen);
L
Linus Torvalds 已提交
3556
xdr_error:
3557
	dprintk("%s: xdr returned %d\n", __func__, -status);
L
Linus Torvalds 已提交
3558 3559 3560 3561 3562 3563
	return status;
}


static int decode_fsinfo(struct xdr_stream *xdr, struct nfs_fsinfo *fsinfo)
{
A
Al Viro 已提交
3564
	__be32 *savep;
L
Linus Torvalds 已提交
3565 3566 3567 3568 3569 3570 3571 3572 3573 3574 3575 3576 3577 3578 3579 3580 3581 3582 3583 3584 3585 3586 3587 3588 3589
	uint32_t attrlen, bitmap[2];
	int status;

	if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
		goto xdr_error;
	if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
		goto xdr_error;
	if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
		goto xdr_error;

	fsinfo->rtmult = fsinfo->wtmult = 512;	/* ??? */

	if ((status = decode_attr_lease_time(xdr, bitmap, &fsinfo->lease_time)) != 0)
		goto xdr_error;
	if ((status = decode_attr_maxfilesize(xdr, bitmap, &fsinfo->maxfilesize)) != 0)
		goto xdr_error;
	if ((status = decode_attr_maxread(xdr, bitmap, &fsinfo->rtmax)) != 0)
		goto xdr_error;
	fsinfo->rtpref = fsinfo->dtpref = fsinfo->rtmax;
	if ((status = decode_attr_maxwrite(xdr, bitmap, &fsinfo->wtmax)) != 0)
		goto xdr_error;
	fsinfo->wtpref = fsinfo->wtmax;

	status = verify_attr_len(xdr, savep, attrlen);
xdr_error:
3590
	dprintk("%s: xdr returned %d!\n", __func__, -status);
L
Linus Torvalds 已提交
3591 3592 3593 3594 3595
	return status;
}

static int decode_getfh(struct xdr_stream *xdr, struct nfs_fh *fh)
{
A
Al Viro 已提交
3596
	__be32 *p;
L
Linus Torvalds 已提交
3597 3598 3599
	uint32_t len;
	int status;

3600 3601 3602
	/* Zero handle first to allow comparisons */
	memset(fh, 0, sizeof(*fh));

L
Linus Torvalds 已提交
3603 3604 3605 3606 3607 3608 3609 3610 3611 3612 3613 3614 3615 3616 3617 3618 3619
	status = decode_op_hdr(xdr, OP_GETFH);
	if (status)
		return status;

	READ_BUF(4);
	READ32(len);
	if (len > NFS4_FHSIZE)
		return -EIO;
	fh->size = len;
	READ_BUF(len);
	COPYMEM(fh->data, len);
	return 0;
}

static int decode_link(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
{
	int status;
3620

L
Linus Torvalds 已提交
3621 3622 3623 3624 3625 3626 3627 3628 3629
	status = decode_op_hdr(xdr, OP_LINK);
	if (status)
		return status;
	return decode_change_info(xdr, cinfo);
}

/*
 * We create the owner, so we know a proper owner.id length is 4.
 */
T
Trond Myklebust 已提交
3630
static int decode_lock_denied (struct xdr_stream *xdr, struct file_lock *fl)
L
Linus Torvalds 已提交
3631
{
T
Trond Myklebust 已提交
3632
	uint64_t offset, length, clientid;
A
Al Viro 已提交
3633
	__be32 *p;
T
Trond Myklebust 已提交
3634
	uint32_t namelen, type;
L
Linus Torvalds 已提交
3635 3636

	READ_BUF(32);
T
Trond Myklebust 已提交
3637 3638 3639 3640 3641 3642 3643 3644 3645 3646 3647 3648 3649 3650
	READ64(offset);
	READ64(length);
	READ32(type);
	if (fl != NULL) {
		fl->fl_start = (loff_t)offset;
		fl->fl_end = fl->fl_start + (loff_t)length - 1;
		if (length == ~(uint64_t)0)
			fl->fl_end = OFFSET_MAX;
		fl->fl_type = F_WRLCK;
		if (type & 1)
			fl->fl_type = F_RDLCK;
		fl->fl_pid = 0;
	}
	READ64(clientid);
L
Linus Torvalds 已提交
3651 3652 3653 3654 3655
	READ32(namelen);
	READ_BUF(namelen);
	return -NFS4ERR_DENIED;
}

T
Trond Myklebust 已提交
3656
static int decode_lock(struct xdr_stream *xdr, struct nfs_lock_res *res)
L
Linus Torvalds 已提交
3657
{
A
Al Viro 已提交
3658
	__be32 *p;
L
Linus Torvalds 已提交
3659 3660 3661
	int status;

	status = decode_op_hdr(xdr, OP_LOCK);
3662 3663
	if (status == -EIO)
		goto out;
L
Linus Torvalds 已提交
3664
	if (status == 0) {
3665 3666
		READ_BUF(NFS4_STATEID_SIZE);
		COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3667
	} else if (status == -NFS4ERR_DENIED)
3668 3669 3670 3671 3672
		status = decode_lock_denied(xdr, NULL);
	if (res->open_seqid != NULL)
		nfs_increment_open_seqid(status, res->open_seqid);
	nfs_increment_lock_seqid(status, res->lock_seqid);
out:
L
Linus Torvalds 已提交
3673 3674 3675
	return status;
}

T
Trond Myklebust 已提交
3676
static int decode_lockt(struct xdr_stream *xdr, struct nfs_lockt_res *res)
L
Linus Torvalds 已提交
3677 3678 3679 3680
{
	int status;
	status = decode_op_hdr(xdr, OP_LOCKT);
	if (status == -NFS4ERR_DENIED)
T
Trond Myklebust 已提交
3681
		return decode_lock_denied(xdr, res->denied);
L
Linus Torvalds 已提交
3682 3683 3684
	return status;
}

T
Trond Myklebust 已提交
3685
static int decode_locku(struct xdr_stream *xdr, struct nfs_locku_res *res)
L
Linus Torvalds 已提交
3686
{
A
Al Viro 已提交
3687
	__be32 *p;
L
Linus Torvalds 已提交
3688 3689 3690
	int status;

	status = decode_op_hdr(xdr, OP_LOCKU);
3691 3692
	if (status != -EIO)
		nfs_increment_lock_seqid(status, res->seqid);
L
Linus Torvalds 已提交
3693
	if (status == 0) {
3694 3695
		READ_BUF(NFS4_STATEID_SIZE);
		COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3696 3697 3698 3699 3700 3701 3702 3703 3704 3705 3706 3707
	}
	return status;
}

static int decode_lookup(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_LOOKUP);
}

/* This is too sick! */
static int decode_space_limit(struct xdr_stream *xdr, u64 *maxsize)
{
A
Andy Adamson 已提交
3708
	__be32 *p;
L
Linus Torvalds 已提交
3709 3710 3711 3712 3713
	uint32_t limit_type, nblocks, blocksize;

	READ_BUF(12);
	READ32(limit_type);
	switch (limit_type) {
A
Andy Adamson 已提交
3714 3715 3716 3717 3718 3719 3720
	case 1:
		READ64(*maxsize);
		break;
	case 2:
		READ32(nblocks);
		READ32(blocksize);
		*maxsize = (uint64_t)nblocks * (uint64_t)blocksize;
L
Linus Torvalds 已提交
3721 3722 3723 3724 3725 3726
	}
	return 0;
}

static int decode_delegation(struct xdr_stream *xdr, struct nfs_openres *res)
{
A
Andy Adamson 已提交
3727 3728
	__be32 *p;
	uint32_t delegation_type;
L
Linus Torvalds 已提交
3729 3730 3731 3732 3733 3734 3735

	READ_BUF(4);
	READ32(delegation_type);
	if (delegation_type == NFS4_OPEN_DELEGATE_NONE) {
		res->delegation_type = 0;
		return 0;
	}
3736 3737
	READ_BUF(NFS4_STATEID_SIZE+4);
	COPYMEM(res->delegation.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3738
	READ32(res->do_recall);
A
Andy Adamson 已提交
3739

L
Linus Torvalds 已提交
3740
	switch (delegation_type) {
A
Andy Adamson 已提交
3741 3742 3743 3744 3745 3746
	case NFS4_OPEN_DELEGATE_READ:
		res->delegation_type = FMODE_READ;
		break;
	case NFS4_OPEN_DELEGATE_WRITE:
		res->delegation_type = FMODE_WRITE|FMODE_READ;
		if (decode_space_limit(xdr, &res->maxsize) < 0)
L
Linus Torvalds 已提交
3747 3748
				return -EIO;
	}
3749
	return decode_ace(xdr, NULL, res->server->nfs_client);
L
Linus Torvalds 已提交
3750 3751 3752 3753
}

static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res)
{
A
Andy Adamson 已提交
3754
	__be32 *p;
3755
	uint32_t savewords, bmlen, i;
A
Andy Adamson 已提交
3756
	int status;
L
Linus Torvalds 已提交
3757

A
Andy Adamson 已提交
3758
	status = decode_op_hdr(xdr, OP_OPEN);
3759 3760
	if (status != -EIO)
		nfs_increment_open_seqid(status, res->seqid);
A
Andy Adamson 已提交
3761 3762 3763 3764
	if (status)
		return status;
	READ_BUF(NFS4_STATEID_SIZE);
	COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3765

A
Andy Adamson 已提交
3766
	decode_change_info(xdr, &res->cinfo);
L
Linus Torvalds 已提交
3767

A
Andy Adamson 已提交
3768 3769 3770 3771 3772
	READ_BUF(8);
	READ32(res->rflags);
	READ32(bmlen);
	if (bmlen > 10)
		goto xdr_error;
L
Linus Torvalds 已提交
3773

A
Andy Adamson 已提交
3774
	READ_BUF(bmlen << 2);
3775 3776 3777 3778 3779 3780
	savewords = min_t(uint32_t, bmlen, NFS4_BITMAP_SIZE);
	for (i = 0; i < savewords; ++i)
		READ32(res->attrset[i]);
	for (; i < NFS4_BITMAP_SIZE; i++)
		res->attrset[i] = 0;

L
Linus Torvalds 已提交
3781 3782
	return decode_delegation(xdr, res);
xdr_error:
3783
	dprintk("%s: Bitmap too large! Length = %u\n", __func__, bmlen);
L
Linus Torvalds 已提交
3784 3785 3786 3787 3788
	return -EIO;
}

static int decode_open_confirm(struct xdr_stream *xdr, struct nfs_open_confirmres *res)
{
A
Andy Adamson 已提交
3789
	__be32 *p;
L
Linus Torvalds 已提交
3790 3791
	int status;

A
Andy Adamson 已提交
3792
	status = decode_op_hdr(xdr, OP_OPEN_CONFIRM);
3793 3794
	if (status != -EIO)
		nfs_increment_open_seqid(status, res->seqid);
A
Andy Adamson 已提交
3795 3796 3797 3798 3799
	if (status)
		return status;
	READ_BUF(NFS4_STATEID_SIZE);
	COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
	return 0;
L
Linus Torvalds 已提交
3800 3801 3802 3803
}

static int decode_open_downgrade(struct xdr_stream *xdr, struct nfs_closeres *res)
{
A
Al Viro 已提交
3804
	__be32 *p;
L
Linus Torvalds 已提交
3805 3806 3807
	int status;

	status = decode_op_hdr(xdr, OP_OPEN_DOWNGRADE);
3808 3809
	if (status != -EIO)
		nfs_increment_open_seqid(status, res->seqid);
L
Linus Torvalds 已提交
3810 3811
	if (status)
		return status;
3812 3813
	READ_BUF(NFS4_STATEID_SIZE);
	COPYMEM(res->stateid.data, NFS4_STATEID_SIZE);
L
Linus Torvalds 已提交
3814 3815 3816 3817 3818 3819 3820 3821 3822 3823 3824 3825 3826 3827 3828 3829
	return 0;
}

static int decode_putfh(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_PUTFH);
}

static int decode_putrootfh(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_PUTROOTFH);
}

static int decode_read(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs_readres *res)
{
	struct kvec *iov = req->rq_rcv_buf.head;
A
Al Viro 已提交
3830
	__be32 *p;
L
Linus Torvalds 已提交
3831 3832 3833 3834 3835 3836 3837 3838 3839 3840 3841 3842
	uint32_t count, eof, recvd, hdrlen;
	int status;

	status = decode_op_hdr(xdr, OP_READ);
	if (status)
		return status;
	READ_BUF(8);
	READ32(eof);
	READ32(count);
	hdrlen = (u8 *) p - (u8 *) iov->iov_base;
	recvd = req->rq_rcv_buf.len - hdrlen;
	if (count > recvd) {
3843
		dprintk("NFS: server cheating in read reply: "
L
Linus Torvalds 已提交
3844 3845 3846 3847 3848 3849 3850 3851 3852 3853 3854 3855 3856 3857 3858
				"count %u > recvd %u\n", count, recvd);
		count = recvd;
		eof = 0;
	}
	xdr_read_pages(xdr, count);
	res->eof = eof;
	res->count = count;
	return 0;
}

static int decode_readdir(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs4_readdir_res *readdir)
{
	struct xdr_buf	*rcvbuf = &req->rq_rcv_buf;
	struct page	*page = *rcvbuf->pages;
	struct kvec	*iov = rcvbuf->head;
3859 3860
	size_t		hdrlen;
	u32		recvd, pglen = rcvbuf->page_len;
A
Al Viro 已提交
3861
	__be32		*end, *entry, *p, *kaddr;
3862
	unsigned int	nr = 0;
3863
	int		status;
L
Linus Torvalds 已提交
3864 3865 3866 3867 3868 3869

	status = decode_op_hdr(xdr, OP_READDIR);
	if (status)
		return status;
	READ_BUF(8);
	COPYMEM(readdir->verifier.data, 8);
3870 3871
	dprintk("%s: verifier = %08x:%08x\n",
			__func__,
3872 3873 3874
			((u32 *)readdir->verifier.data)[0],
			((u32 *)readdir->verifier.data)[1]);

L
Linus Torvalds 已提交
3875 3876 3877 3878 3879 3880 3881 3882

	hdrlen = (char *) p - (char *) iov->iov_base;
	recvd = rcvbuf->len - hdrlen;
	if (pglen > recvd)
		pglen = recvd;
	xdr_read_pages(xdr, pglen);

	BUG_ON(pglen + readdir->pgbase > PAGE_CACHE_SIZE);
A
Al Viro 已提交
3883
	kaddr = p = kmap_atomic(page, KM_USER0);
3884
	end = p + ((pglen + readdir->pgbase) >> 2);
L
Linus Torvalds 已提交
3885
	entry = p;
3886 3887 3888 3889 3890 3891

	/* Make sure the packet actually has a value_follows and EOF entry */
	if ((entry + 1) > end)
		goto short_pkt;

	for (; *p++; nr++) {
3892
		u32 len, attrlen, xlen;
3893
		if (end - p < 3)
L
Linus Torvalds 已提交
3894
			goto short_pkt;
3895
		dprintk("cookie = %Lu, ", *((unsigned long long *)p));
L
Linus Torvalds 已提交
3896 3897 3898
		p += 2;			/* cookie */
		len = ntohl(*p++);	/* filename length */
		if (len > NFS4_MAXNAMLEN) {
3899 3900
			dprintk("NFS: giant filename in readdir (len 0x%x)\n",
					len);
L
Linus Torvalds 已提交
3901 3902
			goto err_unmap;
		}
3903 3904
		xlen = XDR_QUADLEN(len);
		if (end - p < xlen + 1)
L
Linus Torvalds 已提交
3905
			goto short_pkt;
3906 3907
		dprintk("filename = %*s\n", len, (char *)p);
		p += xlen;
L
Linus Torvalds 已提交
3908
		len = ntohl(*p++);	/* bitmap length */
3909
		if (end - p < len + 1)
L
Linus Torvalds 已提交
3910
			goto short_pkt;
3911
		p += len;
L
Linus Torvalds 已提交
3912
		attrlen = XDR_QUADLEN(ntohl(*p++));
3913
		if (end - p < attrlen + 2)
L
Linus Torvalds 已提交
3914
			goto short_pkt;
3915
		p += attrlen;		/* attributes */
L
Linus Torvalds 已提交
3916 3917
		entry = p;
	}
3918 3919 3920 3921 3922 3923 3924 3925 3926
	/*
	 * Apparently some server sends responses that are a valid size, but
	 * contain no entries, and have value_follows==0 and EOF==0. For
	 * those, just set the EOF marker.
	 */
	if (!nr && entry[1] == 0) {
		dprintk("NFS: readdir reply truncated!\n");
		entry[1] = 1;
	}
3927
out:
L
Linus Torvalds 已提交
3928 3929 3930
	kunmap_atomic(kaddr, KM_USER0);
	return 0;
short_pkt:
3931 3932 3933 3934 3935 3936 3937 3938 3939
	/*
	 * When we get a short packet there are 2 possibilities. We can
	 * return an error, or fix up the response to look like a valid
	 * response and return what we have so far. If there are no
	 * entries and the packet was short, then return -EIO. If there
	 * are valid entries in the response, return them and pretend that
	 * the call was successful, but incomplete. The caller can retry the
	 * readdir starting at the last cookie.
	 */
3940
	dprintk("%s: short packet at entry %d\n", __func__, nr);
L
Linus Torvalds 已提交
3941
	entry[0] = entry[1] = 0;
3942 3943
	if (nr)
		goto out;
L
Linus Torvalds 已提交
3944 3945 3946 3947 3948 3949 3950 3951 3952
err_unmap:
	kunmap_atomic(kaddr, KM_USER0);
	return -errno_NFSERR_IO;
}

static int decode_readlink(struct xdr_stream *xdr, struct rpc_rqst *req)
{
	struct xdr_buf *rcvbuf = &req->rq_rcv_buf;
	struct kvec *iov = rcvbuf->head;
3953 3954
	size_t hdrlen;
	u32 len, recvd;
A
Al Viro 已提交
3955
	__be32 *p;
L
Linus Torvalds 已提交
3956 3957 3958 3959 3960 3961 3962 3963 3964 3965 3966
	char *kaddr;
	int status;

	status = decode_op_hdr(xdr, OP_READLINK);
	if (status)
		return status;

	/* Convert length of symlink */
	READ_BUF(4);
	READ32(len);
	if (len >= rcvbuf->page_len || len <= 0) {
3967
		dprintk("nfs: server returned giant symlink!\n");
L
Linus Torvalds 已提交
3968 3969 3970 3971 3972
		return -ENAMETOOLONG;
	}
	hdrlen = (char *) xdr->p - (char *) iov->iov_base;
	recvd = req->rq_rcv_buf.len - hdrlen;
	if (recvd < len) {
3973
		dprintk("NFS: server cheating in readlink reply: "
L
Linus Torvalds 已提交
3974 3975 3976 3977 3978 3979 3980 3981 3982 3983 3984 3985 3986 3987 3988 3989 3990 3991 3992 3993 3994 3995 3996 3997 3998 3999 4000 4001 4002 4003 4004 4005 4006 4007 4008 4009 4010 4011 4012 4013 4014 4015 4016 4017 4018 4019 4020 4021 4022
				"count %u > recvd %u\n", len, recvd);
		return -EIO;
	}
	xdr_read_pages(xdr, len);
	/*
	 * The XDR encode routine has set things up so that
	 * the link text will be copied directly into the
	 * buffer.  We just have to do overflow-checking,
	 * and and null-terminate the text (the VFS expects
	 * null-termination).
	 */
	kaddr = (char *)kmap_atomic(rcvbuf->pages[0], KM_USER0);
	kaddr[len+rcvbuf->page_base] = '\0';
	kunmap_atomic(kaddr, KM_USER0);
	return 0;
}

static int decode_remove(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
{
	int status;

	status = decode_op_hdr(xdr, OP_REMOVE);
	if (status)
		goto out;
	status = decode_change_info(xdr, cinfo);
out:
	return status;
}

static int decode_rename(struct xdr_stream *xdr, struct nfs4_change_info *old_cinfo,
	      struct nfs4_change_info *new_cinfo)
{
	int status;

	status = decode_op_hdr(xdr, OP_RENAME);
	if (status)
		goto out;
	if ((status = decode_change_info(xdr, old_cinfo)))
		goto out;
	status = decode_change_info(xdr, new_cinfo);
out:
	return status;
}

static int decode_renew(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_RENEW);
}

4023 4024 4025 4026 4027 4028
static int
decode_restorefh(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_RESTOREFH);
}

4029 4030 4031
static int decode_getacl(struct xdr_stream *xdr, struct rpc_rqst *req,
		size_t *acl_len)
{
A
Al Viro 已提交
4032
	__be32 *savep;
4033 4034 4035 4036 4037 4038 4039 4040 4041 4042 4043 4044 4045 4046 4047 4048
	uint32_t attrlen,
		 bitmap[2] = {0};
	struct kvec *iov = req->rq_rcv_buf.head;
	int status;

	*acl_len = 0;
	if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
		goto out;
	if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
		goto out;
	if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
		goto out;

	if (unlikely(bitmap[0] & (FATTR4_WORD0_ACL - 1U)))
		return -EIO;
	if (likely(bitmap[0] & FATTR4_WORD0_ACL)) {
4049 4050
		size_t hdrlen;
		u32 recvd;
4051 4052 4053 4054 4055 4056

		/* We ignore &savep and don't do consistency checks on
		 * the attr length.  Let userspace figure it out.... */
		hdrlen = (u8 *)xdr->p - (u8 *)iov->iov_base;
		recvd = req->rq_rcv_buf.len - hdrlen;
		if (attrlen > recvd) {
4057
			dprintk("NFS: server cheating in getattr"
4058 4059 4060 4061
					" acl reply: attrlen %u > recvd %u\n",
					attrlen, recvd);
			return -EINVAL;
		}
4062
		xdr_read_pages(xdr, attrlen);
4063
		*acl_len = attrlen;
J
J. Bruce Fields 已提交
4064 4065
	} else
		status = -EOPNOTSUPP;
4066 4067 4068 4069 4070

out:
	return status;
}

L
Linus Torvalds 已提交
4071 4072 4073 4074 4075 4076
static int
decode_savefh(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_SAVEFH);
}

4077
static int decode_setattr(struct xdr_stream *xdr)
L
Linus Torvalds 已提交
4078
{
A
Al Viro 已提交
4079
	__be32 *p;
L
Linus Torvalds 已提交
4080 4081 4082 4083 4084 4085 4086 4087 4088 4089 4090 4091
	uint32_t bmlen;
	int status;

	status = decode_op_hdr(xdr, OP_SETATTR);
	if (status)
		return status;
	READ_BUF(4);
	READ32(bmlen);
	READ_BUF(bmlen << 2);
	return 0;
}

4092
static int decode_setclientid(struct xdr_stream *xdr, struct nfs_client *clp)
L
Linus Torvalds 已提交
4093
{
A
Al Viro 已提交
4094
	__be32 *p;
L
Linus Torvalds 已提交
4095 4096 4097 4098 4099 4100
	uint32_t opnum;
	int32_t nfserr;

	READ_BUF(8);
	READ32(opnum);
	if (opnum != OP_SETCLIENTID) {
4101
		dprintk("nfs: decode_setclientid: Server returned operation"
4102
			" %d\n", opnum);
L
Linus Torvalds 已提交
4103 4104 4105 4106
		return -EIO;
	}
	READ32(nfserr);
	if (nfserr == NFS_OK) {
4107
		READ_BUF(8 + NFS4_VERIFIER_SIZE);
L
Linus Torvalds 已提交
4108
		READ64(clp->cl_clientid);
4109
		COPYMEM(clp->cl_confirm.data, NFS4_VERIFIER_SIZE);
L
Linus Torvalds 已提交
4110 4111 4112 4113 4114 4115 4116 4117 4118 4119 4120 4121 4122 4123
	} else if (nfserr == NFSERR_CLID_INUSE) {
		uint32_t len;

		/* skip netid string */
		READ_BUF(4);
		READ32(len);
		READ_BUF(len);

		/* skip uaddr string */
		READ_BUF(4);
		READ32(len);
		READ_BUF(len);
		return -NFSERR_CLID_INUSE;
	} else
4124
		return nfs4_stat_to_errno(nfserr);
L
Linus Torvalds 已提交
4125 4126 4127 4128 4129 4130 4131 4132 4133 4134 4135

	return 0;
}

static int decode_setclientid_confirm(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_SETCLIENTID_CONFIRM);
}

static int decode_write(struct xdr_stream *xdr, struct nfs_writeres *res)
{
A
Al Viro 已提交
4136
	__be32 *p;
L
Linus Torvalds 已提交
4137 4138 4139 4140 4141 4142 4143 4144 4145 4146 4147 4148 4149 4150 4151 4152 4153 4154
	int status;

	status = decode_op_hdr(xdr, OP_WRITE);
	if (status)
		return status;

	READ_BUF(16);
	READ32(res->count);
	READ32(res->verf->committed);
	COPYMEM(res->verf->verifier, 8);
	return 0;
}

static int decode_delegreturn(struct xdr_stream *xdr)
{
	return decode_op_hdr(xdr, OP_DELEGRETURN);
}

B
Benny Halevy 已提交
4155 4156 4157 4158 4159 4160 4161 4162 4163 4164 4165 4166 4167 4168 4169 4170 4171 4172 4173 4174 4175 4176 4177 4178 4179 4180 4181 4182 4183 4184 4185 4186 4187 4188 4189 4190 4191 4192 4193 4194 4195 4196 4197 4198
#if defined(CONFIG_NFS_V4_1)
static int decode_exchange_id(struct xdr_stream *xdr,
			      struct nfs41_exchange_id_res *res)
{
	__be32 *p;
	uint32_t dummy;
	int status;
	struct nfs_client *clp = res->client;

	status = decode_op_hdr(xdr, OP_EXCHANGE_ID);
	if (status)
		return status;

	READ_BUF(8);
	READ64(clp->cl_ex_clid);
	READ_BUF(12);
	READ32(clp->cl_seqid);
	READ32(clp->cl_exchange_flags);

	/* We ask for SP4_NONE */
	READ32(dummy);
	if (dummy != SP4_NONE)
		return -EIO;

	/* Throw away minor_id */
	READ_BUF(8);

	/* Throw away Major id */
	READ_BUF(4);
	READ32(dummy);
	READ_BUF(dummy);

	/* Throw away server_scope */
	READ_BUF(4);
	READ32(dummy);
	READ_BUF(dummy);

	/* Throw away Implementation id array */
	READ_BUF(4);
	READ32(dummy);
	READ_BUF(dummy);

	return 0;
}
A
Andy Adamson 已提交
4199 4200 4201 4202 4203 4204 4205 4206 4207 4208 4209 4210 4211 4212 4213 4214 4215 4216 4217 4218 4219 4220 4221 4222 4223 4224 4225 4226 4227 4228 4229 4230 4231 4232 4233 4234 4235 4236 4237 4238 4239 4240 4241 4242 4243 4244 4245 4246 4247 4248 4249 4250 4251

static int decode_chan_attrs(struct xdr_stream *xdr,
			     struct nfs4_channel_attrs *attrs)
{
	__be32 *p;
	u32 nr_attrs;

	READ_BUF(28);
	READ32(attrs->headerpadsz);
	READ32(attrs->max_rqst_sz);
	READ32(attrs->max_resp_sz);
	READ32(attrs->max_resp_sz_cached);
	READ32(attrs->max_ops);
	READ32(attrs->max_reqs);
	READ32(nr_attrs);
	if (unlikely(nr_attrs > 1)) {
		printk(KERN_WARNING "%s: Invalid rdma channel attrs count %u\n",
			__func__, nr_attrs);
		return -EINVAL;
	}
	if (nr_attrs == 1)
		READ_BUF(4); /* skip rdma_attrs */
	return 0;
}

static int decode_create_session(struct xdr_stream *xdr,
				 struct nfs41_create_session_res *res)
{
	__be32 *p;
	int status;
	struct nfs_client *clp = res->client;
	struct nfs4_session *session = clp->cl_session;

	status = decode_op_hdr(xdr, OP_CREATE_SESSION);

	if (status)
		return status;

	/* sessionid */
	READ_BUF(NFS4_MAX_SESSIONID_LEN);
	COPYMEM(&session->sess_id, NFS4_MAX_SESSIONID_LEN);

	/* seqid, flags */
	READ_BUF(8);
	READ32(clp->cl_seqid);
	READ32(session->flags);

	/* Channel attributes */
	status = decode_chan_attrs(xdr, &session->fc_attrs);
	if (!status)
		status = decode_chan_attrs(xdr, &session->bc_attrs);
	return status;
}
A
Andy Adamson 已提交
4252 4253 4254 4255 4256

static int decode_destroy_session(struct xdr_stream *xdr, void *dummy)
{
	return decode_op_hdr(xdr, OP_DESTROY_SESSION);
}
B
Benny Halevy 已提交
4257 4258
#endif /* CONFIG_NFS_V4_1 */

4259 4260 4261 4262 4263
static int decode_sequence(struct xdr_stream *xdr,
			   struct nfs4_sequence_res *res,
			   struct rpc_rqst *rqstp)
{
#if defined(CONFIG_NFS_V4_1)
A
Andy Adamson 已提交
4264 4265 4266 4267 4268 4269
	struct nfs4_slot *slot;
	struct nfs4_sessionid id;
	u32 dummy;
	int status;
	__be32 *p;

4270 4271 4272
	if (!res->sr_session)
		return 0;

A
Andy Adamson 已提交
4273 4274 4275
	status = decode_op_hdr(xdr, OP_SEQUENCE);
	if (status)
		goto out_err;
4276

A
Andy Adamson 已提交
4277 4278 4279 4280 4281 4282 4283 4284 4285 4286 4287 4288 4289 4290 4291 4292 4293 4294 4295 4296 4297 4298 4299 4300 4301 4302 4303 4304 4305 4306 4307 4308 4309 4310 4311 4312 4313
	/*
	 * If the server returns different values for sessionID, slotID or
	 * sequence number, the server is looney tunes.
	 */
	status = -ESERVERFAULT;

	slot = &res->sr_session->fc_slot_table.slots[res->sr_slotid];
	READ_BUF(NFS4_MAX_SESSIONID_LEN + 20);
	COPYMEM(id.data, NFS4_MAX_SESSIONID_LEN);
	if (memcmp(id.data, res->sr_session->sess_id.data,
		   NFS4_MAX_SESSIONID_LEN)) {
		dprintk("%s Invalid session id\n", __func__);
		goto out_err;
	}
	/* seqid */
	READ32(dummy);
	if (dummy != slot->seq_nr) {
		dprintk("%s Invalid sequence number\n", __func__);
		goto out_err;
	}
	/* slot id */
	READ32(dummy);
	if (dummy != res->sr_slotid) {
		dprintk("%s Invalid slot id\n", __func__);
		goto out_err;
	}
	/* highest slot id - currently not processed */
	READ32(dummy);
	/* target highest slot id - currently not processed */
	READ32(dummy);
	/* result flags - currently not processed */
	READ32(dummy);
	status = 0;
out_err:
	res->sr_status = status;
	return status;
#else  /* CONFIG_NFS_V4_1 */
4314
	return 0;
A
Andy Adamson 已提交
4315
#endif /* CONFIG_NFS_V4_1 */
4316 4317
}

4318 4319 4320 4321
/*
 * END OF "GENERIC" DECODE ROUTINES.
 */

L
Linus Torvalds 已提交
4322 4323 4324
/*
 * Decode OPEN_DOWNGRADE response
 */
A
Al Viro 已提交
4325
static int nfs4_xdr_dec_open_downgrade(struct rpc_rqst *rqstp, __be32 *p, struct nfs_closeres *res)
L
Linus Torvalds 已提交
4326
{
A
Andy Adamson 已提交
4327 4328 4329 4330 4331 4332
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4333 4334 4335
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
A
Andy Adamson 已提交
4336 4337 4338 4339 4340 4341
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_open_downgrade(&xdr, res);
4342 4343 4344
	if (status != 0)
		goto out;
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4345
out:
A
Andy Adamson 已提交
4346
	return status;
L
Linus Torvalds 已提交
4347 4348 4349 4350 4351
}

/*
 * Decode ACCESS response
 */
A
Al Viro 已提交
4352
static int nfs4_xdr_dec_access(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_accessres *res)
L
Linus Torvalds 已提交
4353 4354 4355 4356
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4357

L
Linus Torvalds 已提交
4358
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4359 4360 4361 4362 4363
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4364
		goto out;
4365 4366 4367 4368 4369 4370 4371
	status = decode_putfh(&xdr);
	if (status != 0)
		goto out;
	status = decode_access(&xdr, res);
	if (status != 0)
		goto out;
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4372 4373 4374 4375 4376 4377 4378
out:
	return status;
}

/*
 * Decode LOOKUP response
 */
A
Al Viro 已提交
4379
static int nfs4_xdr_dec_lookup(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_lookup_res *res)
L
Linus Torvalds 已提交
4380 4381 4382 4383
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4384

L
Linus Torvalds 已提交
4385
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4386 4387 4388 4389 4390
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4391 4392 4393 4394 4395 4396 4397 4398 4399 4400 4401 4402 4403 4404 4405
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	if ((status = decode_lookup(&xdr)) != 0)
		goto out;
	if ((status = decode_getfh(&xdr, res->fh)) != 0)
		goto out;
	status = decode_getfattr(&xdr, res->fattr, res->server);
out:
	return status;
}

/*
 * Decode LOOKUP_ROOT response
 */
A
Al Viro 已提交
4406
static int nfs4_xdr_dec_lookup_root(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_lookup_res *res)
L
Linus Torvalds 已提交
4407 4408 4409 4410
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4411

L
Linus Torvalds 已提交
4412
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4413 4414 4415 4416 4417
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4418 4419 4420 4421 4422 4423 4424 4425 4426 4427 4428 4429
		goto out;
	if ((status = decode_putrootfh(&xdr)) != 0)
		goto out;
	if ((status = decode_getfh(&xdr, res->fh)) == 0)
		status = decode_getfattr(&xdr, res->fattr, res->server);
out:
	return status;
}

/*
 * Decode REMOVE response
 */
4430
static int nfs4_xdr_dec_remove(struct rpc_rqst *rqstp, __be32 *p, struct nfs_removeres *res)
L
Linus Torvalds 已提交
4431 4432 4433 4434
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4435

L
Linus Torvalds 已提交
4436
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4437 4438 4439 4440 4441
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4442
		goto out;
4443 4444 4445 4446
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	if ((status = decode_remove(&xdr, &res->cinfo)) != 0)
		goto out;
4447
	decode_getfattr(&xdr, &res->dir_attr, res->server);
L
Linus Torvalds 已提交
4448 4449 4450 4451 4452 4453 4454
out:
	return status;
}

/*
 * Decode RENAME response
 */
A
Al Viro 已提交
4455
static int nfs4_xdr_dec_rename(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_rename_res *res)
L
Linus Torvalds 已提交
4456 4457 4458 4459
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4460

L
Linus Torvalds 已提交
4461
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4462 4463 4464 4465 4466
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4467 4468 4469 4470 4471 4472 4473
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	if ((status = decode_savefh(&xdr)) != 0)
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
4474 4475 4476 4477 4478 4479 4480 4481
	if ((status = decode_rename(&xdr, &res->old_cinfo, &res->new_cinfo)) != 0)
		goto out;
	/* Current FH is target directory */
	if (decode_getfattr(&xdr, res->new_fattr, res->server) != 0)
		goto out;
	if ((status = decode_restorefh(&xdr)) != 0)
		goto out;
	decode_getfattr(&xdr, res->old_fattr, res->server);
L
Linus Torvalds 已提交
4482 4483 4484 4485 4486 4487 4488
out:
	return status;
}

/*
 * Decode LINK response
 */
A
Al Viro 已提交
4489
static int nfs4_xdr_dec_link(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_link_res *res)
L
Linus Torvalds 已提交
4490 4491 4492 4493
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4494

L
Linus Torvalds 已提交
4495
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4496 4497 4498 4499 4500
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4501 4502 4503 4504 4505 4506 4507
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	if ((status = decode_savefh(&xdr)) != 0)
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
4508 4509 4510 4511 4512 4513 4514 4515 4516 4517 4518
	if ((status = decode_link(&xdr, &res->cinfo)) != 0)
		goto out;
	/*
	 * Note order: OP_LINK leaves the directory as the current
	 *             filehandle.
	 */
	if (decode_getfattr(&xdr, res->dir_attr, res->server) != 0)
		goto out;
	if ((status = decode_restorefh(&xdr)) != 0)
		goto out;
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4519 4520 4521 4522 4523 4524 4525
out:
	return status;
}

/*
 * Decode CREATE response
 */
A
Al Viro 已提交
4526
static int nfs4_xdr_dec_create(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_create_res *res)
L
Linus Torvalds 已提交
4527 4528 4529 4530
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4531

L
Linus Torvalds 已提交
4532
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4533 4534 4535 4536 4537
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
L
Linus Torvalds 已提交
4538 4539 4540
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
4541 4542
	if ((status = decode_savefh(&xdr)) != 0)
		goto out;
L
Linus Torvalds 已提交
4543 4544 4545 4546
	if ((status = decode_create(&xdr,&res->dir_cinfo)) != 0)
		goto out;
	if ((status = decode_getfh(&xdr, res->fh)) != 0)
		goto out;
4547 4548 4549 4550 4551
	if (decode_getfattr(&xdr, res->fattr, res->server) != 0)
		goto out;
	if ((status = decode_restorefh(&xdr)) != 0)
		goto out;
	decode_getfattr(&xdr, res->dir_fattr, res->server);
L
Linus Torvalds 已提交
4552 4553 4554 4555 4556 4557 4558
out:
	return status;
}

/*
 * Decode SYMLINK response
 */
A
Al Viro 已提交
4559
static int nfs4_xdr_dec_symlink(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_create_res *res)
L
Linus Torvalds 已提交
4560 4561 4562 4563 4564 4565 4566
{
	return nfs4_xdr_dec_create(rqstp, p, res);
}

/*
 * Decode GETATTR response
 */
A
Al Viro 已提交
4567
static int nfs4_xdr_dec_getattr(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_getattr_res *res)
L
Linus Torvalds 已提交
4568 4569 4570 4571
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;
4572

L
Linus Torvalds 已提交
4573 4574
	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4575 4576 4577
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4578 4579 4580 4581 4582 4583 4584 4585 4586 4587
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_getfattr(&xdr, res->fattr, res->server);
out:
	return status;
}

4588 4589 4590 4591
/*
 * Encode an SETACL request
 */
static int
A
Al Viro 已提交
4592
nfs4_xdr_enc_setacl(struct rpc_rqst *req, __be32 *p, struct nfs_setaclargs *args)
4593
{
A
Andy Adamson 已提交
4594 4595
	struct xdr_stream xdr;
	struct compound_hdr hdr = {
4596
		.minorversion = nfs4_xdr_minorversion(&args->seq_args),
A
Andy Adamson 已提交
4597 4598 4599 4600
	};
	int status;

	xdr_init_encode(&xdr, &req->rq_snd_buf, p);
4601
	encode_compound_hdr(&xdr, req, &hdr);
4602
	encode_sequence(&xdr, &args->seq_args, &hdr);
4603
	encode_putfh(&xdr, args->fh, &hdr);
4604 4605
	status = encode_setacl(&xdr, args, &hdr);
	encode_nops(&hdr);
A
Andy Adamson 已提交
4606
	return status;
4607
}
A
Andy Adamson 已提交
4608

4609 4610 4611 4612
/*
 * Decode SETACL response
 */
static int
B
Benny Halevy 已提交
4613 4614
nfs4_xdr_dec_setacl(struct rpc_rqst *rqstp, __be32 *p,
		    struct nfs_setaclres *res)
4615 4616 4617 4618 4619 4620 4621
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4622 4623 4624
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
4625 4626 4627 4628 4629
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
4630
	status = decode_setattr(&xdr);
4631 4632 4633
out:
	return status;
}
L
Linus Torvalds 已提交
4634

4635 4636 4637 4638
/*
 * Decode GETACL response
 */
static int
B
Benny Halevy 已提交
4639 4640
nfs4_xdr_dec_getacl(struct rpc_rqst *rqstp, __be32 *p,
		    struct nfs_getaclres *res)
4641 4642 4643 4644 4645 4646 4647
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4648 4649 4650
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
4651 4652 4653 4654 4655
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
B
Benny Halevy 已提交
4656
	status = decode_getacl(&xdr, rqstp, &res->acl_len);
4657 4658 4659 4660 4661

out:
	return status;
}

L
Linus Torvalds 已提交
4662 4663 4664
/*
 * Decode CLOSE response
 */
A
Al Viro 已提交
4665
static int nfs4_xdr_dec_close(struct rpc_rqst *rqstp, __be32 *p, struct nfs_closeres *res)
L
Linus Torvalds 已提交
4666
{
A
Andy Adamson 已提交
4667 4668 4669 4670 4671 4672
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4673 4674 4675
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
A
Andy Adamson 已提交
4676 4677 4678 4679 4680 4681
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_close(&xdr, res);
4682 4683 4684 4685 4686 4687 4688 4689 4690
	if (status != 0)
		goto out;
	/*
	 * Note: Server may do delete on close for this file
	 * 	in which case the getattr call will fail with
	 * 	an ESTALE error. Shouldn't be a problem,
	 * 	though, since fattr->valid will remain unset.
	 */
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4691
out:
A
Andy Adamson 已提交
4692
	return status;
L
Linus Torvalds 已提交
4693 4694 4695 4696 4697
}

/*
 * Decode OPEN response
 */
A
Al Viro 已提交
4698
static int nfs4_xdr_dec_open(struct rpc_rqst *rqstp, __be32 *p, struct nfs_openres *res)
L
Linus Torvalds 已提交
4699
{
A
Andy Adamson 已提交
4700 4701 4702 4703 4704 4705
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4706 4707 4708
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
A
Andy Adamson 已提交
4709 4710 4711 4712 4713 4714 4715 4716 4717
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_savefh(&xdr);
	if (status)
		goto out;
	status = decode_open(&xdr, res);
4718 4719
	if (status)
		goto out;
4720
	if (decode_getfh(&xdr, &res->fh) != 0)
L
Linus Torvalds 已提交
4721
		goto out;
4722 4723
	if (decode_getfattr(&xdr, res->f_attr, res->server) != 0)
		goto out;
4724
	if (decode_restorefh(&xdr) != 0)
4725 4726
		goto out;
	decode_getfattr(&xdr, res->dir_attr, res->server);
L
Linus Torvalds 已提交
4727
out:
A
Andy Adamson 已提交
4728
	return status;
L
Linus Torvalds 已提交
4729 4730 4731 4732 4733
}

/*
 * Decode OPEN_CONFIRM response
 */
A
Al Viro 已提交
4734
static int nfs4_xdr_dec_open_confirm(struct rpc_rqst *rqstp, __be32 *p, struct nfs_open_confirmres *res)
L
Linus Torvalds 已提交
4735
{
A
Andy Adamson 已提交
4736 4737 4738 4739 4740 4741 4742 4743 4744 4745 4746 4747
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_open_confirm(&xdr, res);
L
Linus Torvalds 已提交
4748
out:
A
Andy Adamson 已提交
4749
	return status;
L
Linus Torvalds 已提交
4750 4751 4752 4753 4754
}

/*
 * Decode OPEN response
 */
A
Al Viro 已提交
4755
static int nfs4_xdr_dec_open_noattr(struct rpc_rqst *rqstp, __be32 *p, struct nfs_openres *res)
L
Linus Torvalds 已提交
4756
{
A
Andy Adamson 已提交
4757 4758 4759 4760 4761 4762
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4763 4764 4765
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
A
Andy Adamson 已提交
4766 4767 4768 4769 4770 4771 4772 4773
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_open(&xdr, res);
	if (status)
		goto out;
4774
	decode_getfattr(&xdr, res->f_attr, res->server);
L
Linus Torvalds 已提交
4775
out:
A
Andy Adamson 已提交
4776
	return status;
L
Linus Torvalds 已提交
4777 4778 4779 4780 4781
}

/*
 * Decode SETATTR response
 */
A
Al Viro 已提交
4782
static int nfs4_xdr_dec_setattr(struct rpc_rqst *rqstp, __be32 *p, struct nfs_setattrres *res)
L
Linus Torvalds 已提交
4783
{
A
Andy Adamson 已提交
4784 4785 4786 4787 4788 4789
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4790 4791 4792
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
A
Andy Adamson 已提交
4793 4794 4795 4796 4797
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
4798
	status = decode_setattr(&xdr);
A
Andy Adamson 已提交
4799 4800
	if (status)
		goto out;
4801
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4802
out:
A
Andy Adamson 已提交
4803
	return status;
L
Linus Torvalds 已提交
4804 4805 4806 4807 4808
}

/*
 * Decode LOCK response
 */
A
Al Viro 已提交
4809
static int nfs4_xdr_dec_lock(struct rpc_rqst *rqstp, __be32 *p, struct nfs_lock_res *res)
L
Linus Torvalds 已提交
4810 4811 4812 4813 4814 4815 4816
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4817 4818 4819
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4820 4821 4822 4823 4824 4825 4826 4827 4828 4829 4830 4831 4832
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_lock(&xdr, res);
out:
	return status;
}

/*
 * Decode LOCKT response
 */
A
Al Viro 已提交
4833
static int nfs4_xdr_dec_lockt(struct rpc_rqst *rqstp, __be32 *p, struct nfs_lockt_res *res)
L
Linus Torvalds 已提交
4834 4835 4836 4837 4838 4839 4840
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4841 4842 4843
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4844 4845 4846 4847 4848 4849 4850 4851 4852 4853 4854 4855 4856
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_lockt(&xdr, res);
out:
	return status;
}

/*
 * Decode LOCKU response
 */
A
Al Viro 已提交
4857
static int nfs4_xdr_dec_locku(struct rpc_rqst *rqstp, __be32 *p, struct nfs_locku_res *res)
L
Linus Torvalds 已提交
4858 4859 4860 4861 4862 4863 4864
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4865 4866 4867
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4868 4869 4870 4871 4872 4873 4874 4875 4876 4877 4878 4879 4880
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_locku(&xdr, res);
out:
	return status;
}

/*
 * Decode READLINK response
 */
B
Benny Halevy 已提交
4881 4882
static int nfs4_xdr_dec_readlink(struct rpc_rqst *rqstp, __be32 *p,
				 struct nfs4_readlink_res *res)
L
Linus Torvalds 已提交
4883 4884 4885 4886 4887 4888 4889
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4890 4891 4892
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4893 4894 4895 4896 4897 4898 4899 4900 4901 4902 4903 4904 4905
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_readlink(&xdr, rqstp);
out:
	return status;
}

/*
 * Decode READDIR response
 */
A
Al Viro 已提交
4906
static int nfs4_xdr_dec_readdir(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_readdir_res *res)
L
Linus Torvalds 已提交
4907 4908 4909 4910 4911 4912 4913
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4914 4915 4916
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4917 4918 4919 4920 4921 4922 4923 4924 4925 4926 4927 4928 4929
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_readdir(&xdr, rqstp, res);
out:
	return status;
}

/*
 * Decode Read response
 */
A
Al Viro 已提交
4930
static int nfs4_xdr_dec_read(struct rpc_rqst *rqstp, __be32 *p, struct nfs_readres *res)
L
Linus Torvalds 已提交
4931 4932 4933 4934 4935 4936 4937
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4938 4939 4940
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4941 4942 4943 4944 4945 4946 4947 4948 4949 4950 4951 4952 4953 4954 4955
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_read(&xdr, rqstp, res);
	if (!status)
		status = res->count;
out:
	return status;
}

/*
 * Decode WRITE response
 */
A
Al Viro 已提交
4956
static int nfs4_xdr_dec_write(struct rpc_rqst *rqstp, __be32 *p, struct nfs_writeres *res)
L
Linus Torvalds 已提交
4957 4958 4959 4960 4961 4962 4963
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4964 4965 4966
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4967 4968 4969 4970 4971 4972
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_write(&xdr, res);
4973 4974 4975
	if (status)
		goto out;
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
4976 4977 4978 4979 4980 4981 4982 4983 4984
	if (!status)
		status = res->count;
out:
	return status;
}

/*
 * Decode COMMIT response
 */
A
Al Viro 已提交
4985
static int nfs4_xdr_dec_commit(struct rpc_rqst *rqstp, __be32 *p, struct nfs_writeres *res)
L
Linus Torvalds 已提交
4986 4987 4988 4989 4990 4991 4992
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
4993 4994 4995
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
L
Linus Torvalds 已提交
4996 4997 4998 4999 5000 5001
	if (status)
		goto out;
	status = decode_putfh(&xdr);
	if (status)
		goto out;
	status = decode_commit(&xdr, res);
5002 5003 5004
	if (status)
		goto out;
	decode_getfattr(&xdr, res->fattr, res->server);
L
Linus Torvalds 已提交
5005 5006 5007 5008 5009 5010 5011
out:
	return status;
}

/*
 * FSINFO request
 */
B
Benny Halevy 已提交
5012 5013
static int nfs4_xdr_dec_fsinfo(struct rpc_rqst *req, __be32 *p,
			       struct nfs4_fsinfo_res *res)
L
Linus Torvalds 已提交
5014 5015 5016 5017 5018 5019 5020
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
5021 5022
	if (!status)
		status = decode_sequence(&xdr, &res->seq_res, req);
L
Linus Torvalds 已提交
5023 5024 5025
	if (!status)
		status = decode_putfh(&xdr);
	if (!status)
B
Benny Halevy 已提交
5026
		status = decode_fsinfo(&xdr, res->fsinfo);
L
Linus Torvalds 已提交
5027 5028 5029 5030 5031 5032
	return status;
}

/*
 * PATHCONF request
 */
B
Benny Halevy 已提交
5033 5034
static int nfs4_xdr_dec_pathconf(struct rpc_rqst *req, __be32 *p,
				 struct nfs4_pathconf_res *res)
L
Linus Torvalds 已提交
5035 5036 5037 5038 5039 5040 5041
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
5042 5043
	if (!status)
		status = decode_sequence(&xdr, &res->seq_res, req);
L
Linus Torvalds 已提交
5044 5045 5046
	if (!status)
		status = decode_putfh(&xdr);
	if (!status)
B
Benny Halevy 已提交
5047
		status = decode_pathconf(&xdr, res->pathconf);
L
Linus Torvalds 已提交
5048 5049 5050 5051 5052 5053
	return status;
}

/*
 * STATFS request
 */
B
Benny Halevy 已提交
5054 5055
static int nfs4_xdr_dec_statfs(struct rpc_rqst *req, __be32 *p,
			       struct nfs4_statfs_res *res)
L
Linus Torvalds 已提交
5056 5057 5058 5059 5060 5061 5062
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
5063 5064
	if (!status)
		status = decode_sequence(&xdr, &res->seq_res, req);
L
Linus Torvalds 已提交
5065 5066 5067
	if (!status)
		status = decode_putfh(&xdr);
	if (!status)
B
Benny Halevy 已提交
5068
		status = decode_statfs(&xdr, res->fsstat);
L
Linus Torvalds 已提交
5069 5070 5071 5072 5073 5074
	return status;
}

/*
 * GETATTR_BITMAP request
 */
A
Al Viro 已提交
5075
static int nfs4_xdr_dec_server_caps(struct rpc_rqst *req, __be32 *p, struct nfs4_server_caps_res *res)
L
Linus Torvalds 已提交
5076 5077 5078 5079 5080 5081
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
5082 5083 5084 5085 5086
	status = decode_compound_hdr(&xdr, &hdr);
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, req);
	if (status)
L
Linus Torvalds 已提交
5087 5088 5089 5090 5091 5092 5093 5094 5095 5096 5097
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	status = decode_server_caps(&xdr, res);
out:
	return status;
}

/*
 * Decode RENEW response
 */
A
Al Viro 已提交
5098
static int nfs4_xdr_dec_renew(struct rpc_rqst *rqstp, __be32 *p, void *dummy)
L
Linus Torvalds 已提交
5099 5100 5101 5102 5103 5104 5105 5106 5107 5108 5109 5110 5111 5112 5113
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_renew(&xdr);
	return status;
}

/*
 * a SETCLIENTID request
 */
A
Al Viro 已提交
5114
static int nfs4_xdr_dec_setclientid(struct rpc_rqst *req, __be32 *p,
5115
		struct nfs_client *clp)
L
Linus Torvalds 已提交
5116 5117 5118 5119 5120 5121 5122 5123 5124 5125 5126 5127 5128 5129 5130
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_setclientid(&xdr, clp);
	return status;
}

/*
 * a SETCLIENTID_CONFIRM request
 */
A
Al Viro 已提交
5131
static int nfs4_xdr_dec_setclientid_confirm(struct rpc_rqst *req, __be32 *p, struct nfs_fsinfo *fsinfo)
L
Linus Torvalds 已提交
5132 5133 5134 5135 5136 5137 5138 5139 5140 5141 5142 5143 5144 5145 5146 5147 5148 5149 5150
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_setclientid_confirm(&xdr);
	if (!status)
		status = decode_putrootfh(&xdr);
	if (!status)
		status = decode_fsinfo(&xdr, fsinfo);
	return status;
}

/*
 * DELEGRETURN request
 */
A
Al Viro 已提交
5151
static int nfs4_xdr_dec_delegreturn(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_delegreturnres *res)
L
Linus Torvalds 已提交
5152 5153 5154 5155 5156 5157 5158
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
5159 5160 5161 5162
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, rqstp);
	if (status)
5163 5164 5165 5166 5167 5168 5169
		goto out;
	status = decode_putfh(&xdr);
	if (status != 0)
		goto out;
	status = decode_delegreturn(&xdr);
	decode_getfattr(&xdr, res->fattr, res->server);
out:
L
Linus Torvalds 已提交
5170 5171 5172
	return status;
}

5173 5174 5175
/*
 * FS_LOCATIONS request
 */
B
Benny Halevy 已提交
5176 5177
static int nfs4_xdr_dec_fs_locations(struct rpc_rqst *req, __be32 *p,
				     struct nfs4_fs_locations_res *res)
5178 5179 5180 5181 5182 5183 5184
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
5185 5186 5187 5188
	if (status)
		goto out;
	status = decode_sequence(&xdr, &res->seq_res, req);
	if (status)
5189 5190 5191 5192 5193 5194
		goto out;
	if ((status = decode_putfh(&xdr)) != 0)
		goto out;
	if ((status = decode_lookup(&xdr)) != 0)
		goto out;
	xdr_enter_page(&xdr, PAGE_SIZE);
B
Benny Halevy 已提交
5195 5196
	status = decode_getfattr(&xdr, &res->fs_locations->fattr,
				 res->fs_locations->server);
5197 5198 5199 5200
out:
	return status;
}

B
Benny Halevy 已提交
5201 5202 5203 5204 5205 5206 5207 5208 5209 5210 5211 5212 5213 5214 5215 5216 5217
#if defined(CONFIG_NFS_V4_1)
/*
 * EXCHANGE_ID request
 */
static int nfs4_xdr_dec_exchange_id(struct rpc_rqst *rqstp, uint32_t *p,
				    void *res)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_exchange_id(&xdr, res);
	return status;
}
A
Andy Adamson 已提交
5218

A
Andy Adamson 已提交
5219 5220 5221 5222 5223 5224 5225 5226 5227 5228 5229 5230 5231 5232 5233 5234 5235
/*
 * a CREATE_SESSION request
 */
static int nfs4_xdr_dec_create_session(struct rpc_rqst *rqstp, uint32_t *p,
				       struct nfs41_create_session_res *res)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_create_session(&xdr, res);
	return status;
}

A
Andy Adamson 已提交
5236 5237 5238 5239 5240 5241 5242 5243 5244 5245 5246 5247 5248 5249 5250 5251 5252
/*
 * a DESTROY_SESSION request
 */
static int nfs4_xdr_dec_destroy_session(struct rpc_rqst *rqstp, uint32_t *p,
					void *dummy)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_destroy_session(&xdr, dummy);
	return status;
}

A
Andy Adamson 已提交
5253 5254 5255 5256 5257 5258 5259 5260 5261 5262 5263 5264 5265 5266 5267 5268 5269
/*
 * a SEQUENCE request
 */
static int nfs4_xdr_dec_sequence(struct rpc_rqst *rqstp, uint32_t *p,
				 struct nfs4_sequence_res *res)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_sequence(&xdr, res, rqstp);
	return status;
}

A
Andy Adamson 已提交
5270 5271 5272 5273 5274 5275 5276 5277 5278 5279 5280 5281 5282 5283 5284 5285 5286 5287 5288 5289
/*
 * a GET_LEASE_TIME request
 */
static int nfs4_xdr_dec_get_lease_time(struct rpc_rqst *rqstp, uint32_t *p,
				       struct nfs4_get_lease_time_res *res)
{
	struct xdr_stream xdr;
	struct compound_hdr hdr;
	int status;

	xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
	status = decode_compound_hdr(&xdr, &hdr);
	if (!status)
		status = decode_sequence(&xdr, &res->lr_seq_res, rqstp);
	if (!status)
		status = decode_putrootfh(&xdr);
	if (!status)
		status = decode_fsinfo(&xdr, res->lr_fsinfo);
	return status;
}
B
Benny Halevy 已提交
5290 5291
#endif /* CONFIG_NFS_V4_1 */

5292
__be32 *nfs4_decode_dirent(__be32 *p, struct nfs_entry *entry, int plus)
L
Linus Torvalds 已提交
5293 5294 5295 5296 5297 5298 5299 5300 5301 5302 5303 5304 5305 5306 5307 5308 5309 5310 5311 5312 5313 5314 5315 5316 5317 5318 5319 5320 5321 5322 5323 5324 5325 5326
{
	uint32_t bitmap[2] = {0};
	uint32_t len;

	if (!*p++) {
		if (!*p)
			return ERR_PTR(-EAGAIN);
		entry->eof = 1;
		return ERR_PTR(-EBADCOOKIE);
	}

	entry->prev_cookie = entry->cookie;
	p = xdr_decode_hyper(p, &entry->cookie);
	entry->len = ntohl(*p++);
	entry->name = (const char *) p;
	p += XDR_QUADLEN(entry->len);

	/*
	 * In case the server doesn't return an inode number,
	 * we fake one here.  (We don't use inode number 0,
	 * since glibc seems to choke on it...)
	 */
	entry->ino = 1;

	len = ntohl(*p++);		/* bitmap length */
	if (len-- > 0) {
		bitmap[0] = ntohl(*p++);
		if (len-- > 0) {
			bitmap[1] = ntohl(*p++);
			p += len;
		}
	}
	len = XDR_QUADLEN(ntohl(*p++));	/* attribute buffer length */
	if (len > 0) {
5327 5328 5329 5330 5331 5332
		if (bitmap[0] & FATTR4_WORD0_RDATTR_ERROR) {
			bitmap[0] &= ~FATTR4_WORD0_RDATTR_ERROR;
			/* Ignore the return value of rdattr_error for now */
			p++;
			len--;
		}
L
Linus Torvalds 已提交
5333 5334 5335 5336 5337 5338 5339 5340 5341 5342 5343 5344 5345 5346 5347 5348 5349 5350 5351 5352
		if (bitmap[0] == 0 && bitmap[1] == FATTR4_WORD1_MOUNTED_ON_FILEID)
			xdr_decode_hyper(p, &entry->ino);
		else if (bitmap[0] == FATTR4_WORD0_FILEID)
			xdr_decode_hyper(p, &entry->ino);
		p += len;
	}

	entry->eof = !p[0] && p[1];
	return p;
}

/*
 * We need to translate between nfs status return values and
 * the local errno values which may not be the same.
 */
static struct {
	int stat;
	int errno;
} nfs_errtbl[] = {
	{ NFS4_OK,		0		},
5353 5354 5355 5356 5357 5358 5359 5360 5361 5362 5363 5364 5365 5366 5367 5368 5369 5370 5371 5372 5373 5374 5375 5376 5377 5378 5379 5380 5381 5382 5383 5384
	{ NFS4ERR_PERM,		-EPERM		},
	{ NFS4ERR_NOENT,	-ENOENT		},
	{ NFS4ERR_IO,		-errno_NFSERR_IO},
	{ NFS4ERR_NXIO,		-ENXIO		},
	{ NFS4ERR_ACCESS,	-EACCES		},
	{ NFS4ERR_EXIST,	-EEXIST		},
	{ NFS4ERR_XDEV,		-EXDEV		},
	{ NFS4ERR_NOTDIR,	-ENOTDIR	},
	{ NFS4ERR_ISDIR,	-EISDIR		},
	{ NFS4ERR_INVAL,	-EINVAL		},
	{ NFS4ERR_FBIG,		-EFBIG		},
	{ NFS4ERR_NOSPC,	-ENOSPC		},
	{ NFS4ERR_ROFS,		-EROFS		},
	{ NFS4ERR_MLINK,	-EMLINK		},
	{ NFS4ERR_NAMETOOLONG,	-ENAMETOOLONG	},
	{ NFS4ERR_NOTEMPTY,	-ENOTEMPTY	},
	{ NFS4ERR_DQUOT,	-EDQUOT		},
	{ NFS4ERR_STALE,	-ESTALE		},
	{ NFS4ERR_BADHANDLE,	-EBADHANDLE	},
	{ NFS4ERR_BADOWNER,	-EINVAL		},
	{ NFS4ERR_BADNAME,	-EINVAL		},
	{ NFS4ERR_BAD_COOKIE,	-EBADCOOKIE	},
	{ NFS4ERR_NOTSUPP,	-ENOTSUPP	},
	{ NFS4ERR_TOOSMALL,	-ETOOSMALL	},
	{ NFS4ERR_SERVERFAULT,	-ESERVERFAULT	},
	{ NFS4ERR_BADTYPE,	-EBADTYPE	},
	{ NFS4ERR_LOCKED,	-EAGAIN		},
	{ NFS4ERR_RESOURCE,	-EREMOTEIO	},
	{ NFS4ERR_SYMLINK,	-ELOOP		},
	{ NFS4ERR_OP_ILLEGAL,	-EOPNOTSUPP	},
	{ NFS4ERR_DEADLOCK,	-EDEADLK	},
	{ NFS4ERR_WRONGSEC,	-EPERM		}, /* FIXME: this needs
L
Linus Torvalds 已提交
5385 5386 5387
						    * to be handled by a
						    * middle-layer.
						    */
5388
	{ -1,			-EIO		}
L
Linus Torvalds 已提交
5389 5390 5391 5392 5393 5394 5395
};

/*
 * Convert an NFS error code to a local one.
 * This one is used jointly by NFSv2 and NFSv3.
 */
static int
5396
nfs4_stat_to_errno(int stat)
L
Linus Torvalds 已提交
5397 5398 5399 5400 5401 5402 5403 5404
{
	int i;
	for (i = 0; nfs_errtbl[i].stat != -1; i++) {
		if (nfs_errtbl[i].stat == stat)
			return nfs_errtbl[i].errno;
	}
	if (stat <= 10000 || stat > 10100) {
		/* The server is looney tunes. */
5405
		return -ESERVERFAULT;
L
Linus Torvalds 已提交
5406 5407 5408 5409 5410 5411
	}
	/* If we cannot translate the error, the recovery routines should
	 * handle it.
	 * Note: remaining NFSv4 error codes have values > 10000, so should
	 * not conflict with native Linux error codes.
	 */
5412
	return -stat;
L
Linus Torvalds 已提交
5413 5414 5415 5416 5417 5418 5419
}

#define PROC(proc, argtype, restype)				\
[NFSPROC4_CLNT_##proc] = {					\
	.p_proc   = NFSPROC4_COMPOUND,				\
	.p_encode = (kxdrproc_t) nfs4_xdr_##argtype,		\
	.p_decode = (kxdrproc_t) nfs4_xdr_##restype,		\
5420 5421
	.p_arglen = NFS4_##argtype##_sz,			\
	.p_replen = NFS4_##restype##_sz,			\
5422 5423
	.p_statidx = NFSPROC4_CLNT_##proc,			\
	.p_name   = #proc,					\
A
Andy Adamson 已提交
5424
}
L
Linus Torvalds 已提交
5425 5426 5427 5428 5429 5430 5431 5432 5433 5434 5435 5436 5437 5438 5439 5440 5441 5442 5443 5444 5445 5446 5447 5448 5449 5450 5451 5452 5453 5454 5455 5456 5457

struct rpc_procinfo	nfs4_procedures[] = {
  PROC(READ,		enc_read,	dec_read),
  PROC(WRITE,		enc_write,	dec_write),
  PROC(COMMIT,		enc_commit,	dec_commit),
  PROC(OPEN,		enc_open,	dec_open),
  PROC(OPEN_CONFIRM,	enc_open_confirm,	dec_open_confirm),
  PROC(OPEN_NOATTR,	enc_open_noattr,	dec_open_noattr),
  PROC(OPEN_DOWNGRADE,	enc_open_downgrade,	dec_open_downgrade),
  PROC(CLOSE,		enc_close,	dec_close),
  PROC(SETATTR,		enc_setattr,	dec_setattr),
  PROC(FSINFO,		enc_fsinfo,	dec_fsinfo),
  PROC(RENEW,		enc_renew,	dec_renew),
  PROC(SETCLIENTID,	enc_setclientid,	dec_setclientid),
  PROC(SETCLIENTID_CONFIRM,	enc_setclientid_confirm,	dec_setclientid_confirm),
  PROC(LOCK,            enc_lock,       dec_lock),
  PROC(LOCKT,           enc_lockt,      dec_lockt),
  PROC(LOCKU,           enc_locku,      dec_locku),
  PROC(ACCESS,		enc_access,	dec_access),
  PROC(GETATTR,		enc_getattr,	dec_getattr),
  PROC(LOOKUP,		enc_lookup,	dec_lookup),
  PROC(LOOKUP_ROOT,	enc_lookup_root,	dec_lookup_root),
  PROC(REMOVE,		enc_remove,	dec_remove),
  PROC(RENAME,		enc_rename,	dec_rename),
  PROC(LINK,		enc_link,	dec_link),
  PROC(SYMLINK,		enc_symlink,	dec_symlink),
  PROC(CREATE,		enc_create,	dec_create),
  PROC(PATHCONF,	enc_pathconf,	dec_pathconf),
  PROC(STATFS,		enc_statfs,	dec_statfs),
  PROC(READLINK,	enc_readlink,	dec_readlink),
  PROC(READDIR,		enc_readdir,	dec_readdir),
  PROC(SERVER_CAPS,	enc_server_caps, dec_server_caps),
  PROC(DELEGRETURN,	enc_delegreturn, dec_delegreturn),
5458
  PROC(GETACL,		enc_getacl,	dec_getacl),
5459
  PROC(SETACL,		enc_setacl,	dec_setacl),
5460
  PROC(FS_LOCATIONS,	enc_fs_locations, dec_fs_locations),
B
Benny Halevy 已提交
5461 5462
#if defined(CONFIG_NFS_V4_1)
  PROC(EXCHANGE_ID,	enc_exchange_id,	dec_exchange_id),
A
Andy Adamson 已提交
5463
  PROC(CREATE_SESSION,	enc_create_session,	dec_create_session),
A
Andy Adamson 已提交
5464
  PROC(DESTROY_SESSION,	enc_destroy_session,	dec_destroy_session),
A
Andy Adamson 已提交
5465
  PROC(SEQUENCE,	enc_sequence,	dec_sequence),
A
Andy Adamson 已提交
5466
  PROC(GET_LEASE_TIME,	enc_get_lease_time,	dec_get_lease_time),
B
Benny Halevy 已提交
5467
#endif /* CONFIG_NFS_V4_1 */
L
Linus Torvalds 已提交
5468 5469 5470 5471
};

struct rpc_version		nfs_version4 = {
	.number			= 4,
5472
	.nrprocs		= ARRAY_SIZE(nfs4_procedures),
L
Linus Torvalds 已提交
5473 5474 5475 5476 5477 5478 5479 5480
	.procs			= nfs4_procedures
};

/*
 * Local variables:
 *  c-basic-offset: 8
 * End:
 */