blob: c5c75235c5b859485a5cd8b6351c1bf5d34cd479 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * fs/nfs/nfs4xdr.c
3 *
4 * Client-side XDR for NFSv4.
5 *
6 * Copyright (c) 2002 The Regents of the University of Michigan.
7 * All rights reserved.
8 *
9 * Kendrick Smith <kmsmith@umich.edu>
10 * Andy Adamson <andros@umich.edu>
11 *
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following conditions
14 * are met:
15 *
16 * 1. Redistributions of source code must retain the above copyright
17 * notice, this list of conditions and the following disclaimer.
18 * 2. Redistributions in binary form must reproduce the above copyright
19 * notice, this list of conditions and the following disclaimer in the
20 * documentation and/or other materials provided with the distribution.
21 * 3. Neither the name of the University nor the names of its
22 * contributors may be used to endorse or promote products derived
23 * from this software without specific prior written permission.
24 *
25 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED
26 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
27 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
28 * DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
30 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
31 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
32 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
33 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
34 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
35 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36 */
37
38#include <linux/param.h>
39#include <linux/time.h>
40#include <linux/mm.h>
41#include <linux/slab.h>
42#include <linux/utsname.h>
43#include <linux/errno.h>
44#include <linux/string.h>
45#include <linux/in.h>
46#include <linux/pagemap.h>
47#include <linux/proc_fs.h>
48#include <linux/kdev_t.h>
49#include <linux/sunrpc/clnt.h>
50#include <linux/nfs.h>
51#include <linux/nfs4.h>
52#include <linux/nfs_fs.h>
53#include <linux/nfs_idmap.h>
Trond Myklebust4ce79712005-06-22 17:16:21 +000054#include "nfs4_fs.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070055
56#define NFSDBG_FACILITY NFSDBG_XDR
57
58/* Mapping from NFS error code to "errno" error code. */
59#define errno_NFSERR_IO EIO
60
61static int nfs_stat_to_errno(int);
62
63/* NFSv4 COMPOUND tags are only wanted for debugging purposes */
64#ifdef DEBUG
65#define NFS4_MAXTAGLEN 20
66#else
67#define NFS4_MAXTAGLEN 0
68#endif
69
70/* lock,open owner id:
71 * we currently use size 1 (u32) out of (NFS4_OPAQUE_LIMIT >> 2)
72 */
73#define owner_id_maxsz (1 + 1)
74#define compound_encode_hdr_maxsz (3 + (NFS4_MAXTAGLEN >> 2))
75#define compound_decode_hdr_maxsz (3 + (NFS4_MAXTAGLEN >> 2))
76#define op_encode_hdr_maxsz (1)
77#define op_decode_hdr_maxsz (2)
78#define encode_putfh_maxsz (op_encode_hdr_maxsz + 1 + \
79 (NFS4_FHSIZE >> 2))
80#define decode_putfh_maxsz (op_decode_hdr_maxsz)
81#define encode_putrootfh_maxsz (op_encode_hdr_maxsz)
82#define decode_putrootfh_maxsz (op_decode_hdr_maxsz)
83#define encode_getfh_maxsz (op_encode_hdr_maxsz)
84#define decode_getfh_maxsz (op_decode_hdr_maxsz + 1 + \
85 ((3+NFS4_FHSIZE) >> 2))
J. Bruce Fields96928202005-06-22 17:16:22 +000086#define nfs4_fattr_bitmap_maxsz 3
87#define encode_getattr_maxsz (op_encode_hdr_maxsz + nfs4_fattr_bitmap_maxsz)
Linus Torvalds1da177e2005-04-16 15:20:36 -070088#define nfs4_name_maxsz (1 + ((3 + NFS4_MAXNAMLEN) >> 2))
89#define nfs4_path_maxsz (1 + ((3 + NFS4_MAXPATHLEN) >> 2))
J. Bruce Fields96928202005-06-22 17:16:22 +000090/* This is based on getfattr, which uses the most attributes: */
91#define nfs4_fattr_value_maxsz (1 + (1 + 2 + 2 + 4 + 2 + 1 + 1 + 2 + 2 + \
92 3 + 3 + 3 + 2 * nfs4_name_maxsz))
93#define nfs4_fattr_maxsz (nfs4_fattr_bitmap_maxsz + \
94 nfs4_fattr_value_maxsz)
95#define decode_getattr_maxsz (op_decode_hdr_maxsz + nfs4_fattr_maxsz)
Linus Torvalds1da177e2005-04-16 15:20:36 -070096#define encode_savefh_maxsz (op_encode_hdr_maxsz)
97#define decode_savefh_maxsz (op_decode_hdr_maxsz)
98#define encode_fsinfo_maxsz (op_encode_hdr_maxsz + 2)
99#define decode_fsinfo_maxsz (op_decode_hdr_maxsz + 11)
100#define encode_renew_maxsz (op_encode_hdr_maxsz + 3)
101#define decode_renew_maxsz (op_decode_hdr_maxsz)
102#define encode_setclientid_maxsz \
103 (op_encode_hdr_maxsz + \
104 4 /*server->ip_addr*/ + \
105 1 /*Netid*/ + \
106 6 /*uaddr*/ + \
107 6 + (NFS4_VERIFIER_SIZE >> 2))
108#define decode_setclientid_maxsz \
109 (op_decode_hdr_maxsz + \
110 2 + \
111 1024) /* large value for CLID_INUSE */
112#define encode_setclientid_confirm_maxsz \
113 (op_encode_hdr_maxsz + \
114 3 + (NFS4_VERIFIER_SIZE >> 2))
115#define decode_setclientid_confirm_maxsz \
116 (op_decode_hdr_maxsz)
117#define encode_lookup_maxsz (op_encode_hdr_maxsz + \
118 1 + ((3 + NFS4_FHSIZE) >> 2))
119#define encode_remove_maxsz (op_encode_hdr_maxsz + \
120 nfs4_name_maxsz)
121#define encode_rename_maxsz (op_encode_hdr_maxsz + \
122 2 * nfs4_name_maxsz)
123#define decode_rename_maxsz (op_decode_hdr_maxsz + 5 + 5)
124#define encode_link_maxsz (op_encode_hdr_maxsz + \
125 nfs4_name_maxsz)
126#define decode_link_maxsz (op_decode_hdr_maxsz + 5)
127#define encode_symlink_maxsz (op_encode_hdr_maxsz + \
128 1 + nfs4_name_maxsz + \
129 nfs4_path_maxsz + \
J. Bruce Fields96928202005-06-22 17:16:22 +0000130 nfs4_fattr_maxsz)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700131#define decode_symlink_maxsz (op_decode_hdr_maxsz + 8)
132#define encode_create_maxsz (op_encode_hdr_maxsz + \
133 2 + nfs4_name_maxsz + \
J. Bruce Fields96928202005-06-22 17:16:22 +0000134 nfs4_fattr_maxsz)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135#define decode_create_maxsz (op_decode_hdr_maxsz + 8)
136#define encode_delegreturn_maxsz (op_encode_hdr_maxsz + 4)
137#define decode_delegreturn_maxsz (op_decode_hdr_maxsz)
138#define NFS4_enc_compound_sz (1024) /* XXX: large enough? */
139#define NFS4_dec_compound_sz (1024) /* XXX: large enough? */
140#define NFS4_enc_read_sz (compound_encode_hdr_maxsz + \
141 encode_putfh_maxsz + \
142 op_encode_hdr_maxsz + 7)
143#define NFS4_dec_read_sz (compound_decode_hdr_maxsz + \
144 decode_putfh_maxsz + \
145 op_decode_hdr_maxsz + 2)
146#define NFS4_enc_readlink_sz (compound_encode_hdr_maxsz + \
147 encode_putfh_maxsz + \
148 op_encode_hdr_maxsz)
149#define NFS4_dec_readlink_sz (compound_decode_hdr_maxsz + \
150 decode_putfh_maxsz + \
151 op_decode_hdr_maxsz)
152#define NFS4_enc_readdir_sz (compound_encode_hdr_maxsz + \
153 encode_putfh_maxsz + \
154 op_encode_hdr_maxsz + 9)
155#define NFS4_dec_readdir_sz (compound_decode_hdr_maxsz + \
156 decode_putfh_maxsz + \
157 op_decode_hdr_maxsz + 2)
158#define NFS4_enc_write_sz (compound_encode_hdr_maxsz + \
159 encode_putfh_maxsz + \
160 op_encode_hdr_maxsz + 8)
161#define NFS4_dec_write_sz (compound_decode_hdr_maxsz + \
162 decode_putfh_maxsz + \
163 op_decode_hdr_maxsz + 4)
164#define NFS4_enc_commit_sz (compound_encode_hdr_maxsz + \
165 encode_putfh_maxsz + \
166 op_encode_hdr_maxsz + 3)
167#define NFS4_dec_commit_sz (compound_decode_hdr_maxsz + \
168 decode_putfh_maxsz + \
169 op_decode_hdr_maxsz + 2)
170#define NFS4_enc_open_sz (compound_encode_hdr_maxsz + \
171 encode_putfh_maxsz + \
172 op_encode_hdr_maxsz + \
173 13 + 3 + 2 + 64 + \
174 encode_getattr_maxsz + \
175 encode_getfh_maxsz)
176#define NFS4_dec_open_sz (compound_decode_hdr_maxsz + \
177 decode_putfh_maxsz + \
178 op_decode_hdr_maxsz + 4 + 5 + 2 + 3 + \
179 decode_getattr_maxsz + \
180 decode_getfh_maxsz)
181#define NFS4_enc_open_confirm_sz \
182 (compound_encode_hdr_maxsz + \
183 encode_putfh_maxsz + \
184 op_encode_hdr_maxsz + 5)
185#define NFS4_dec_open_confirm_sz (compound_decode_hdr_maxsz + \
186 decode_putfh_maxsz + \
187 op_decode_hdr_maxsz + 4)
188#define NFS4_enc_open_noattr_sz (compound_encode_hdr_maxsz + \
189 encode_putfh_maxsz + \
190 op_encode_hdr_maxsz + \
191 11)
192#define NFS4_dec_open_noattr_sz (compound_decode_hdr_maxsz + \
193 decode_putfh_maxsz + \
194 op_decode_hdr_maxsz + \
195 4 + 5 + 2 + 3)
196#define NFS4_enc_open_downgrade_sz \
197 (compound_encode_hdr_maxsz + \
198 encode_putfh_maxsz + \
199 op_encode_hdr_maxsz + 7)
200#define NFS4_dec_open_downgrade_sz \
201 (compound_decode_hdr_maxsz + \
202 decode_putfh_maxsz + \
203 op_decode_hdr_maxsz + 4)
204#define NFS4_enc_close_sz (compound_encode_hdr_maxsz + \
205 encode_putfh_maxsz + \
206 op_encode_hdr_maxsz + 5)
207#define NFS4_dec_close_sz (compound_decode_hdr_maxsz + \
208 decode_putfh_maxsz + \
209 op_decode_hdr_maxsz + 4)
210#define NFS4_enc_setattr_sz (compound_encode_hdr_maxsz + \
211 encode_putfh_maxsz + \
212 op_encode_hdr_maxsz + 4 + \
J. Bruce Fields96928202005-06-22 17:16:22 +0000213 nfs4_fattr_maxsz + \
Linus Torvalds1da177e2005-04-16 15:20:36 -0700214 encode_getattr_maxsz)
215#define NFS4_dec_setattr_sz (compound_decode_hdr_maxsz + \
216 decode_putfh_maxsz + \
217 op_decode_hdr_maxsz + 3)
218#define NFS4_enc_fsinfo_sz (compound_encode_hdr_maxsz + \
219 encode_putfh_maxsz + \
220 encode_fsinfo_maxsz)
221#define NFS4_dec_fsinfo_sz (compound_decode_hdr_maxsz + \
222 decode_putfh_maxsz + \
223 decode_fsinfo_maxsz)
224#define NFS4_enc_renew_sz (compound_encode_hdr_maxsz + \
225 encode_renew_maxsz)
226#define NFS4_dec_renew_sz (compound_decode_hdr_maxsz + \
227 decode_renew_maxsz)
228#define NFS4_enc_setclientid_sz (compound_encode_hdr_maxsz + \
229 encode_setclientid_maxsz)
230#define NFS4_dec_setclientid_sz (compound_decode_hdr_maxsz + \
231 decode_setclientid_maxsz)
232#define NFS4_enc_setclientid_confirm_sz \
233 (compound_encode_hdr_maxsz + \
234 encode_setclientid_confirm_maxsz + \
235 encode_putrootfh_maxsz + \
236 encode_fsinfo_maxsz)
237#define NFS4_dec_setclientid_confirm_sz \
238 (compound_decode_hdr_maxsz + \
239 decode_setclientid_confirm_maxsz + \
240 decode_putrootfh_maxsz + \
241 decode_fsinfo_maxsz)
242#define NFS4_enc_lock_sz (compound_encode_hdr_maxsz + \
243 encode_putfh_maxsz + \
244 encode_getattr_maxsz + \
245 op_encode_hdr_maxsz + \
246 1 + 1 + 2 + 2 + \
247 1 + 4 + 1 + 2 + \
248 owner_id_maxsz)
249#define NFS4_dec_lock_sz (compound_decode_hdr_maxsz + \
250 decode_putfh_maxsz + \
251 decode_getattr_maxsz + \
252 op_decode_hdr_maxsz + \
253 2 + 2 + 1 + 2 + \
254 owner_id_maxsz)
255#define NFS4_enc_lockt_sz (compound_encode_hdr_maxsz + \
256 encode_putfh_maxsz + \
257 encode_getattr_maxsz + \
258 op_encode_hdr_maxsz + \
259 1 + 2 + 2 + 2 + \
260 owner_id_maxsz)
261#define NFS4_dec_lockt_sz (NFS4_dec_lock_sz)
262#define NFS4_enc_locku_sz (compound_encode_hdr_maxsz + \
263 encode_putfh_maxsz + \
264 encode_getattr_maxsz + \
265 op_encode_hdr_maxsz + \
266 1 + 1 + 4 + 2 + 2)
267#define NFS4_dec_locku_sz (compound_decode_hdr_maxsz + \
268 decode_putfh_maxsz + \
269 decode_getattr_maxsz + \
270 op_decode_hdr_maxsz + 4)
271#define NFS4_enc_access_sz (compound_encode_hdr_maxsz + \
272 encode_putfh_maxsz + \
273 op_encode_hdr_maxsz + 1)
274#define NFS4_dec_access_sz (compound_decode_hdr_maxsz + \
275 decode_putfh_maxsz + \
276 op_decode_hdr_maxsz + 2)
277#define NFS4_enc_getattr_sz (compound_encode_hdr_maxsz + \
278 encode_putfh_maxsz + \
279 encode_getattr_maxsz)
280#define NFS4_dec_getattr_sz (compound_decode_hdr_maxsz + \
281 decode_putfh_maxsz + \
282 decode_getattr_maxsz)
283#define NFS4_enc_lookup_sz (compound_encode_hdr_maxsz + \
284 encode_putfh_maxsz + \
285 encode_lookup_maxsz + \
286 encode_getattr_maxsz + \
287 encode_getfh_maxsz)
288#define NFS4_dec_lookup_sz (compound_decode_hdr_maxsz + \
289 decode_putfh_maxsz + \
290 op_decode_hdr_maxsz + \
291 decode_getattr_maxsz + \
292 decode_getfh_maxsz)
293#define NFS4_enc_lookup_root_sz (compound_encode_hdr_maxsz + \
294 encode_putrootfh_maxsz + \
295 encode_getattr_maxsz + \
296 encode_getfh_maxsz)
297#define NFS4_dec_lookup_root_sz (compound_decode_hdr_maxsz + \
298 decode_putrootfh_maxsz + \
299 decode_getattr_maxsz + \
300 decode_getfh_maxsz)
301#define NFS4_enc_remove_sz (compound_encode_hdr_maxsz + \
302 encode_putfh_maxsz + \
303 encode_remove_maxsz)
304#define NFS4_dec_remove_sz (compound_decode_hdr_maxsz + \
305 decode_putfh_maxsz + \
306 op_decode_hdr_maxsz + 5)
307#define NFS4_enc_rename_sz (compound_encode_hdr_maxsz + \
308 encode_putfh_maxsz + \
309 encode_savefh_maxsz + \
310 encode_putfh_maxsz + \
311 encode_rename_maxsz)
312#define NFS4_dec_rename_sz (compound_decode_hdr_maxsz + \
313 decode_putfh_maxsz + \
314 decode_savefh_maxsz + \
315 decode_putfh_maxsz + \
316 decode_rename_maxsz)
317#define NFS4_enc_link_sz (compound_encode_hdr_maxsz + \
318 encode_putfh_maxsz + \
319 encode_savefh_maxsz + \
320 encode_putfh_maxsz + \
321 encode_link_maxsz)
322#define NFS4_dec_link_sz (compound_decode_hdr_maxsz + \
323 decode_putfh_maxsz + \
324 decode_savefh_maxsz + \
325 decode_putfh_maxsz + \
326 decode_link_maxsz)
327#define NFS4_enc_symlink_sz (compound_encode_hdr_maxsz + \
328 encode_putfh_maxsz + \
329 encode_symlink_maxsz + \
330 encode_getattr_maxsz + \
331 encode_getfh_maxsz)
332#define NFS4_dec_symlink_sz (compound_decode_hdr_maxsz + \
333 decode_putfh_maxsz + \
334 decode_symlink_maxsz + \
335 decode_getattr_maxsz + \
336 decode_getfh_maxsz)
337#define NFS4_enc_create_sz (compound_encode_hdr_maxsz + \
338 encode_putfh_maxsz + \
339 encode_create_maxsz + \
340 encode_getattr_maxsz + \
341 encode_getfh_maxsz)
342#define NFS4_dec_create_sz (compound_decode_hdr_maxsz + \
343 decode_putfh_maxsz + \
344 decode_create_maxsz + \
345 decode_getattr_maxsz + \
346 decode_getfh_maxsz)
347#define NFS4_enc_pathconf_sz (compound_encode_hdr_maxsz + \
348 encode_putfh_maxsz + \
349 encode_getattr_maxsz)
350#define NFS4_dec_pathconf_sz (compound_decode_hdr_maxsz + \
351 decode_putfh_maxsz + \
352 decode_getattr_maxsz)
353#define NFS4_enc_statfs_sz (compound_encode_hdr_maxsz + \
354 encode_putfh_maxsz + \
355 encode_getattr_maxsz)
356#define NFS4_dec_statfs_sz (compound_decode_hdr_maxsz + \
357 decode_putfh_maxsz + \
358 op_decode_hdr_maxsz + 12)
359#define NFS4_enc_server_caps_sz (compound_encode_hdr_maxsz + \
360 encode_getattr_maxsz)
361#define NFS4_dec_server_caps_sz (compound_decode_hdr_maxsz + \
362 decode_getattr_maxsz)
363#define NFS4_enc_delegreturn_sz (compound_encode_hdr_maxsz + \
364 encode_putfh_maxsz + \
365 encode_delegreturn_maxsz)
366#define NFS4_dec_delegreturn_sz (compound_decode_hdr_maxsz + \
367 decode_delegreturn_maxsz)
J. Bruce Fields029d1052005-06-22 17:16:22 +0000368#define NFS4_enc_getacl_sz (compound_encode_hdr_maxsz + \
369 encode_putfh_maxsz + \
370 encode_getattr_maxsz)
371#define NFS4_dec_getacl_sz (compound_decode_hdr_maxsz + \
372 decode_putfh_maxsz + \
373 op_decode_hdr_maxsz + \
374 nfs4_fattr_bitmap_maxsz + 1)
J. Bruce Fields23ec6962005-06-22 17:16:22 +0000375#define NFS4_enc_setacl_sz (compound_encode_hdr_maxsz + \
376 encode_putfh_maxsz + \
377 op_encode_hdr_maxsz + 4 + \
378 nfs4_fattr_bitmap_maxsz + 1)
379#define NFS4_dec_setacl_sz (compound_decode_hdr_maxsz + \
380 decode_putfh_maxsz + \
381 op_decode_hdr_maxsz + nfs4_fattr_bitmap_maxsz)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382
383static struct {
384 unsigned int mode;
385 unsigned int nfs2type;
386} nfs_type2fmt[] = {
387 { 0, NFNON },
388 { S_IFREG, NFREG },
389 { S_IFDIR, NFDIR },
390 { S_IFBLK, NFBLK },
391 { S_IFCHR, NFCHR },
392 { S_IFLNK, NFLNK },
393 { S_IFSOCK, NFSOCK },
394 { S_IFIFO, NFFIFO },
395 { 0, NFNON },
396 { 0, NFNON },
397};
398
399struct compound_hdr {
400 int32_t status;
401 uint32_t nops;
402 uint32_t taglen;
403 char * tag;
404};
405
406/*
407 * START OF "GENERIC" ENCODE ROUTINES.
408 * These may look a little ugly since they are imported from a "generic"
409 * set of XDR encode/decode routines which are intended to be shared by
410 * all of our NFSv4 implementations (OpenBSD, MacOS X...).
411 *
412 * If the pain of reading these is too great, it should be a straightforward
413 * task to translate them into Linux-specific versions which are more
414 * consistent with the style used in NFSv2/v3...
415 */
416#define WRITE32(n) *p++ = htonl(n)
417#define WRITE64(n) do { \
418 *p++ = htonl((uint32_t)((n) >> 32)); \
419 *p++ = htonl((uint32_t)(n)); \
420} while (0)
421#define WRITEMEM(ptr,nbytes) do { \
422 p = xdr_encode_opaque_fixed(p, ptr, nbytes); \
423} while (0)
424
425#define RESERVE_SPACE(nbytes) do { \
426 p = xdr_reserve_space(xdr, nbytes); \
427 if (!p) printk("RESERVE_SPACE(%d) failed in function %s\n", (int) (nbytes), __FUNCTION__); \
428 BUG_ON(!p); \
429} while (0)
430
431static void encode_string(struct xdr_stream *xdr, unsigned int len, const char *str)
432{
433 uint32_t *p;
434
435 p = xdr_reserve_space(xdr, 4 + len);
436 BUG_ON(p == NULL);
437 xdr_encode_opaque(p, str, len);
438}
439
440static int encode_compound_hdr(struct xdr_stream *xdr, struct compound_hdr *hdr)
441{
442 uint32_t *p;
443
444 dprintk("encode_compound: tag=%.*s\n", (int)hdr->taglen, hdr->tag);
445 BUG_ON(hdr->taglen > NFS4_MAXTAGLEN);
446 RESERVE_SPACE(12+(XDR_QUADLEN(hdr->taglen)<<2));
447 WRITE32(hdr->taglen);
448 WRITEMEM(hdr->tag, hdr->taglen);
449 WRITE32(NFS4_MINOR_VERSION);
450 WRITE32(hdr->nops);
451 return 0;
452}
453
454static void encode_nfs4_verifier(struct xdr_stream *xdr, const nfs4_verifier *verf)
455{
456 uint32_t *p;
457
458 p = xdr_reserve_space(xdr, NFS4_VERIFIER_SIZE);
459 BUG_ON(p == NULL);
460 xdr_encode_opaque_fixed(p, verf->data, NFS4_VERIFIER_SIZE);
461}
462
463static int encode_attrs(struct xdr_stream *xdr, const struct iattr *iap, const struct nfs_server *server)
464{
465 char owner_name[IDMAP_NAMESZ];
466 char owner_group[IDMAP_NAMESZ];
467 int owner_namelen = 0;
468 int owner_grouplen = 0;
469 uint32_t *p;
470 uint32_t *q;
471 int len;
472 uint32_t bmval0 = 0;
473 uint32_t bmval1 = 0;
474 int status;
475
476 /*
477 * We reserve enough space to write the entire attribute buffer at once.
478 * In the worst-case, this would be
479 * 12(bitmap) + 4(attrlen) + 8(size) + 4(mode) + 4(atime) + 4(mtime)
480 * = 36 bytes, plus any contribution from variable-length fields
J. Bruce Fields23ec6962005-06-22 17:16:22 +0000481 * such as owner/group.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700482 */
483 len = 16;
484
485 /* Sigh */
486 if (iap->ia_valid & ATTR_SIZE)
487 len += 8;
488 if (iap->ia_valid & ATTR_MODE)
489 len += 4;
490 if (iap->ia_valid & ATTR_UID) {
491 owner_namelen = nfs_map_uid_to_name(server->nfs4_state, iap->ia_uid, owner_name);
492 if (owner_namelen < 0) {
493 printk(KERN_WARNING "nfs: couldn't resolve uid %d to string\n",
494 iap->ia_uid);
495 /* XXX */
496 strcpy(owner_name, "nobody");
497 owner_namelen = sizeof("nobody") - 1;
498 /* goto out; */
499 }
500 len += 4 + (XDR_QUADLEN(owner_namelen) << 2);
501 }
502 if (iap->ia_valid & ATTR_GID) {
503 owner_grouplen = nfs_map_gid_to_group(server->nfs4_state, iap->ia_gid, owner_group);
504 if (owner_grouplen < 0) {
505 printk(KERN_WARNING "nfs4: couldn't resolve gid %d to string\n",
506 iap->ia_gid);
507 strcpy(owner_group, "nobody");
508 owner_grouplen = sizeof("nobody") - 1;
509 /* goto out; */
510 }
511 len += 4 + (XDR_QUADLEN(owner_grouplen) << 2);
512 }
513 if (iap->ia_valid & ATTR_ATIME_SET)
514 len += 16;
515 else if (iap->ia_valid & ATTR_ATIME)
516 len += 4;
517 if (iap->ia_valid & ATTR_MTIME_SET)
518 len += 16;
519 else if (iap->ia_valid & ATTR_MTIME)
520 len += 4;
521 RESERVE_SPACE(len);
522
523 /*
524 * We write the bitmap length now, but leave the bitmap and the attribute
525 * buffer length to be backfilled at the end of this routine.
526 */
527 WRITE32(2);
528 q = p;
529 p += 3;
530
531 if (iap->ia_valid & ATTR_SIZE) {
532 bmval0 |= FATTR4_WORD0_SIZE;
533 WRITE64(iap->ia_size);
534 }
535 if (iap->ia_valid & ATTR_MODE) {
536 bmval1 |= FATTR4_WORD1_MODE;
537 WRITE32(iap->ia_mode);
538 }
539 if (iap->ia_valid & ATTR_UID) {
540 bmval1 |= FATTR4_WORD1_OWNER;
541 WRITE32(owner_namelen);
542 WRITEMEM(owner_name, owner_namelen);
543 }
544 if (iap->ia_valid & ATTR_GID) {
545 bmval1 |= FATTR4_WORD1_OWNER_GROUP;
546 WRITE32(owner_grouplen);
547 WRITEMEM(owner_group, owner_grouplen);
548 }
549 if (iap->ia_valid & ATTR_ATIME_SET) {
550 bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET;
551 WRITE32(NFS4_SET_TO_CLIENT_TIME);
552 WRITE32(0);
553 WRITE32(iap->ia_mtime.tv_sec);
554 WRITE32(iap->ia_mtime.tv_nsec);
555 }
556 else if (iap->ia_valid & ATTR_ATIME) {
557 bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET;
558 WRITE32(NFS4_SET_TO_SERVER_TIME);
559 }
560 if (iap->ia_valid & ATTR_MTIME_SET) {
561 bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET;
562 WRITE32(NFS4_SET_TO_CLIENT_TIME);
563 WRITE32(0);
564 WRITE32(iap->ia_mtime.tv_sec);
565 WRITE32(iap->ia_mtime.tv_nsec);
566 }
567 else if (iap->ia_valid & ATTR_MTIME) {
568 bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET;
569 WRITE32(NFS4_SET_TO_SERVER_TIME);
570 }
571
572 /*
573 * Now we backfill the bitmap and the attribute buffer length.
574 */
575 if (len != ((char *)p - (char *)q) + 4) {
576 printk ("encode_attr: Attr length calculation error! %u != %Zu\n",
577 len, ((char *)p - (char *)q) + 4);
578 BUG();
579 }
580 len = (char *)p - (char *)q - 12;
581 *q++ = htonl(bmval0);
582 *q++ = htonl(bmval1);
583 *q++ = htonl(len);
584
585 status = 0;
586/* out: */
587 return status;
588}
589
590static int encode_access(struct xdr_stream *xdr, u32 access)
591{
592 uint32_t *p;
593
594 RESERVE_SPACE(8);
595 WRITE32(OP_ACCESS);
596 WRITE32(access);
597
598 return 0;
599}
600
601static int encode_close(struct xdr_stream *xdr, const struct nfs_closeargs *arg)
602{
603 uint32_t *p;
604
Trond Myklebust95121352005-10-18 14:20:12 -0700605 RESERVE_SPACE(8+sizeof(arg->stateid->data));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700606 WRITE32(OP_CLOSE);
Trond Myklebustcee54fc2005-10-18 14:20:12 -0700607 WRITE32(arg->seqid->sequence->counter);
Trond Myklebust95121352005-10-18 14:20:12 -0700608 WRITEMEM(arg->stateid->data, sizeof(arg->stateid->data));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700609
610 return 0;
611}
612
613static int encode_commit(struct xdr_stream *xdr, const struct nfs_writeargs *args)
614{
615 uint32_t *p;
616
617 RESERVE_SPACE(16);
618 WRITE32(OP_COMMIT);
619 WRITE64(args->offset);
620 WRITE32(args->count);
621
622 return 0;
623}
624
625static int encode_create(struct xdr_stream *xdr, const struct nfs4_create_arg *create)
626{
627 uint32_t *p;
628
629 RESERVE_SPACE(8);
630 WRITE32(OP_CREATE);
631 WRITE32(create->ftype);
632
633 switch (create->ftype) {
634 case NF4LNK:
635 RESERVE_SPACE(4 + create->u.symlink->len);
636 WRITE32(create->u.symlink->len);
637 WRITEMEM(create->u.symlink->name, create->u.symlink->len);
638 break;
639
640 case NF4BLK: case NF4CHR:
641 RESERVE_SPACE(8);
642 WRITE32(create->u.device.specdata1);
643 WRITE32(create->u.device.specdata2);
644 break;
645
646 default:
647 break;
648 }
649
650 RESERVE_SPACE(4 + create->name->len);
651 WRITE32(create->name->len);
652 WRITEMEM(create->name->name, create->name->len);
653
654 return encode_attrs(xdr, create->attrs, create->server);
655}
656
657static int encode_getattr_one(struct xdr_stream *xdr, uint32_t bitmap)
658{
659 uint32_t *p;
660
661 RESERVE_SPACE(12);
662 WRITE32(OP_GETATTR);
663 WRITE32(1);
664 WRITE32(bitmap);
665 return 0;
666}
667
668static int encode_getattr_two(struct xdr_stream *xdr, uint32_t bm0, uint32_t bm1)
669{
670 uint32_t *p;
671
672 RESERVE_SPACE(16);
673 WRITE32(OP_GETATTR);
674 WRITE32(2);
675 WRITE32(bm0);
676 WRITE32(bm1);
677 return 0;
678}
679
680static int encode_getfattr(struct xdr_stream *xdr, const u32* bitmask)
681{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700682 return encode_getattr_two(xdr,
683 bitmask[0] & nfs4_fattr_bitmap[0],
684 bitmask[1] & nfs4_fattr_bitmap[1]);
685}
686
687static int encode_fsinfo(struct xdr_stream *xdr, const u32* bitmask)
688{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700689 return encode_getattr_two(xdr, bitmask[0] & nfs4_fsinfo_bitmap[0],
690 bitmask[1] & nfs4_fsinfo_bitmap[1]);
691}
692
693static int encode_getfh(struct xdr_stream *xdr)
694{
695 uint32_t *p;
696
697 RESERVE_SPACE(4);
698 WRITE32(OP_GETFH);
699
700 return 0;
701}
702
703static int encode_link(struct xdr_stream *xdr, const struct qstr *name)
704{
705 uint32_t *p;
706
707 RESERVE_SPACE(8 + name->len);
708 WRITE32(OP_LINK);
709 WRITE32(name->len);
710 WRITEMEM(name->name, name->len);
711
712 return 0;
713}
714
715/*
716 * opcode,type,reclaim,offset,length,new_lock_owner = 32
717 * open_seqid,open_stateid,lock_seqid,lock_owner.clientid, lock_owner.id = 40
718 */
719static int encode_lock(struct xdr_stream *xdr, const struct nfs_lockargs *arg)
720{
721 uint32_t *p;
722 struct nfs_lock_opargs *opargs = arg->u.lock;
723
724 RESERVE_SPACE(32);
725 WRITE32(OP_LOCK);
726 WRITE32(arg->type);
727 WRITE32(opargs->reclaim);
728 WRITE64(arg->offset);
729 WRITE64(arg->length);
730 WRITE32(opargs->new_lock_owner);
731 if (opargs->new_lock_owner){
Linus Torvalds1da177e2005-04-16 15:20:36 -0700732 RESERVE_SPACE(40);
Trond Myklebust06735b32005-10-18 14:20:15 -0700733 WRITE32(opargs->open_seqid->sequence->counter);
734 WRITEMEM(opargs->open_stateid->data, sizeof(opargs->open_stateid->data));
735 WRITE32(opargs->lock_seqid->sequence->counter);
736 WRITE64(opargs->lock_owner.clientid);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700737 WRITE32(4);
Trond Myklebust06735b32005-10-18 14:20:15 -0700738 WRITE32(opargs->lock_owner.id);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700739 }
740 else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700741 RESERVE_SPACE(20);
Trond Myklebust06735b32005-10-18 14:20:15 -0700742 WRITEMEM(opargs->lock_stateid->data, sizeof(opargs->lock_stateid->data));
743 WRITE32(opargs->lock_seqid->sequence->counter);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700744 }
745
746 return 0;
747}
748
749static int encode_lockt(struct xdr_stream *xdr, const struct nfs_lockargs *arg)
750{
751 uint32_t *p;
752 struct nfs_lowner *opargs = arg->u.lockt;
753
754 RESERVE_SPACE(40);
755 WRITE32(OP_LOCKT);
756 WRITE32(arg->type);
757 WRITE64(arg->offset);
758 WRITE64(arg->length);
759 WRITE64(opargs->clientid);
760 WRITE32(4);
761 WRITE32(opargs->id);
762
763 return 0;
764}
765
766static int encode_locku(struct xdr_stream *xdr, const struct nfs_lockargs *arg)
767{
768 uint32_t *p;
769 struct nfs_locku_opargs *opargs = arg->u.locku;
770
771 RESERVE_SPACE(44);
772 WRITE32(OP_LOCKU);
773 WRITE32(arg->type);
Trond Myklebustcee54fc2005-10-18 14:20:12 -0700774 WRITE32(opargs->seqid->sequence->counter);
Trond Myklebustfaf5f492005-10-18 14:20:15 -0700775 WRITEMEM(opargs->stateid->data, sizeof(opargs->stateid->data));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776 WRITE64(arg->offset);
777 WRITE64(arg->length);
778
779 return 0;
780}
781
782static int encode_lookup(struct xdr_stream *xdr, const struct qstr *name)
783{
784 int len = name->len;
785 uint32_t *p;
786
787 RESERVE_SPACE(8 + len);
788 WRITE32(OP_LOOKUP);
789 WRITE32(len);
790 WRITEMEM(name->name, len);
791
792 return 0;
793}
794
795static void encode_share_access(struct xdr_stream *xdr, int open_flags)
796{
797 uint32_t *p;
798
799 RESERVE_SPACE(8);
800 switch (open_flags & (FMODE_READ|FMODE_WRITE)) {
801 case FMODE_READ:
802 WRITE32(NFS4_SHARE_ACCESS_READ);
803 break;
804 case FMODE_WRITE:
805 WRITE32(NFS4_SHARE_ACCESS_WRITE);
806 break;
807 case FMODE_READ|FMODE_WRITE:
808 WRITE32(NFS4_SHARE_ACCESS_BOTH);
809 break;
810 default:
811 BUG();
812 }
813 WRITE32(0); /* for linux, share_deny = 0 always */
814}
815
816static inline void encode_openhdr(struct xdr_stream *xdr, const struct nfs_openargs *arg)
817{
818 uint32_t *p;
819 /*
820 * opcode 4, seqid 4, share_access 4, share_deny 4, clientid 8, ownerlen 4,
821 * owner 4 = 32
822 */
823 RESERVE_SPACE(8);
824 WRITE32(OP_OPEN);
Trond Myklebustcee54fc2005-10-18 14:20:12 -0700825 WRITE32(arg->seqid->sequence->counter);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700826 encode_share_access(xdr, arg->open_flags);
827 RESERVE_SPACE(16);
828 WRITE64(arg->clientid);
829 WRITE32(4);
830 WRITE32(arg->id);
831}
832
833static inline void encode_createmode(struct xdr_stream *xdr, const struct nfs_openargs *arg)
834{
835 uint32_t *p;
836
837 RESERVE_SPACE(4);
838 switch(arg->open_flags & O_EXCL) {
839 case 0:
840 WRITE32(NFS4_CREATE_UNCHECKED);
841 encode_attrs(xdr, arg->u.attrs, arg->server);
842 break;
843 default:
844 WRITE32(NFS4_CREATE_EXCLUSIVE);
845 encode_nfs4_verifier(xdr, &arg->u.verifier);
846 }
847}
848
849static void encode_opentype(struct xdr_stream *xdr, const struct nfs_openargs *arg)
850{
851 uint32_t *p;
852
853 RESERVE_SPACE(4);
854 switch (arg->open_flags & O_CREAT) {
855 case 0:
856 WRITE32(NFS4_OPEN_NOCREATE);
857 break;
858 default:
859 BUG_ON(arg->claim != NFS4_OPEN_CLAIM_NULL);
860 WRITE32(NFS4_OPEN_CREATE);
861 encode_createmode(xdr, arg);
862 }
863}
864
865static inline void encode_delegation_type(struct xdr_stream *xdr, int delegation_type)
866{
867 uint32_t *p;
868
869 RESERVE_SPACE(4);
870 switch (delegation_type) {
871 case 0:
872 WRITE32(NFS4_OPEN_DELEGATE_NONE);
873 break;
874 case FMODE_READ:
875 WRITE32(NFS4_OPEN_DELEGATE_READ);
876 break;
877 case FMODE_WRITE|FMODE_READ:
878 WRITE32(NFS4_OPEN_DELEGATE_WRITE);
879 break;
880 default:
881 BUG();
882 }
883}
884
885static inline void encode_claim_null(struct xdr_stream *xdr, const struct qstr *name)
886{
887 uint32_t *p;
888
889 RESERVE_SPACE(4);
890 WRITE32(NFS4_OPEN_CLAIM_NULL);
891 encode_string(xdr, name->len, name->name);
892}
893
894static inline void encode_claim_previous(struct xdr_stream *xdr, int type)
895{
896 uint32_t *p;
897
898 RESERVE_SPACE(4);
899 WRITE32(NFS4_OPEN_CLAIM_PREVIOUS);
900 encode_delegation_type(xdr, type);
901}
902
903static inline void encode_claim_delegate_cur(struct xdr_stream *xdr, const struct qstr *name, const nfs4_stateid *stateid)
904{
905 uint32_t *p;
906
907 RESERVE_SPACE(4+sizeof(stateid->data));
908 WRITE32(NFS4_OPEN_CLAIM_DELEGATE_CUR);
909 WRITEMEM(stateid->data, sizeof(stateid->data));
910 encode_string(xdr, name->len, name->name);
911}
912
913static int encode_open(struct xdr_stream *xdr, const struct nfs_openargs *arg)
914{
915 encode_openhdr(xdr, arg);
916 encode_opentype(xdr, arg);
917 switch (arg->claim) {
918 case NFS4_OPEN_CLAIM_NULL:
919 encode_claim_null(xdr, arg->name);
920 break;
921 case NFS4_OPEN_CLAIM_PREVIOUS:
922 encode_claim_previous(xdr, arg->u.delegation_type);
923 break;
924 case NFS4_OPEN_CLAIM_DELEGATE_CUR:
925 encode_claim_delegate_cur(xdr, arg->name, &arg->u.delegation);
926 break;
927 default:
928 BUG();
929 }
930 return 0;
931}
932
933static int encode_open_confirm(struct xdr_stream *xdr, const struct nfs_open_confirmargs *arg)
934{
935 uint32_t *p;
936
937 RESERVE_SPACE(8+sizeof(arg->stateid.data));
938 WRITE32(OP_OPEN_CONFIRM);
939 WRITEMEM(arg->stateid.data, sizeof(arg->stateid.data));
Trond Myklebustcee54fc2005-10-18 14:20:12 -0700940 WRITE32(arg->seqid->sequence->counter);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700941
942 return 0;
943}
944
945static int encode_open_downgrade(struct xdr_stream *xdr, const struct nfs_closeargs *arg)
946{
947 uint32_t *p;
948
Trond Myklebust95121352005-10-18 14:20:12 -0700949 RESERVE_SPACE(8+sizeof(arg->stateid->data));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950 WRITE32(OP_OPEN_DOWNGRADE);
Trond Myklebust95121352005-10-18 14:20:12 -0700951 WRITEMEM(arg->stateid->data, sizeof(arg->stateid->data));
Trond Myklebustcee54fc2005-10-18 14:20:12 -0700952 WRITE32(arg->seqid->sequence->counter);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700953 encode_share_access(xdr, arg->open_flags);
954 return 0;
955}
956
957static int
958encode_putfh(struct xdr_stream *xdr, const struct nfs_fh *fh)
959{
960 int len = fh->size;
961 uint32_t *p;
962
963 RESERVE_SPACE(8 + len);
964 WRITE32(OP_PUTFH);
965 WRITE32(len);
966 WRITEMEM(fh->data, len);
967
968 return 0;
969}
970
971static int encode_putrootfh(struct xdr_stream *xdr)
972{
973 uint32_t *p;
974
975 RESERVE_SPACE(4);
976 WRITE32(OP_PUTROOTFH);
977
978 return 0;
979}
980
981static void encode_stateid(struct xdr_stream *xdr, const struct nfs_open_context *ctx)
982{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700983 nfs4_stateid stateid;
984 uint32_t *p;
985
986 RESERVE_SPACE(16);
987 if (ctx->state != NULL) {
988 nfs4_copy_stateid(&stateid, ctx->state, ctx->lockowner);
989 WRITEMEM(stateid.data, sizeof(stateid.data));
990 } else
991 WRITEMEM(zero_stateid.data, sizeof(zero_stateid.data));
992}
993
994static int encode_read(struct xdr_stream *xdr, const struct nfs_readargs *args)
995{
996 uint32_t *p;
997
998 RESERVE_SPACE(4);
999 WRITE32(OP_READ);
1000
1001 encode_stateid(xdr, args->context);
1002
1003 RESERVE_SPACE(12);
1004 WRITE64(args->offset);
1005 WRITE32(args->count);
1006
1007 return 0;
1008}
1009
1010static int encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg *readdir, struct rpc_rqst *req)
1011{
1012 struct rpc_auth *auth = req->rq_task->tk_auth;
Manoj Naik97d312d2005-06-22 17:16:39 +00001013 uint32_t attrs[2] = {
1014 FATTR4_WORD0_RDATTR_ERROR|FATTR4_WORD0_FILEID,
1015 FATTR4_WORD1_MOUNTED_ON_FILEID,
1016 };
Linus Torvalds1da177e2005-04-16 15:20:36 -07001017 int replen;
1018 uint32_t *p;
1019
1020 RESERVE_SPACE(32+sizeof(nfs4_verifier));
1021 WRITE32(OP_READDIR);
1022 WRITE64(readdir->cookie);
1023 WRITEMEM(readdir->verifier.data, sizeof(readdir->verifier.data));
1024 WRITE32(readdir->count >> 1); /* We're not doing readdirplus */
1025 WRITE32(readdir->count);
1026 WRITE32(2);
Manoj Naik97d312d2005-06-22 17:16:39 +00001027 /* Switch to mounted_on_fileid if the server supports it */
1028 if (readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)
1029 attrs[0] &= ~FATTR4_WORD0_FILEID;
1030 else
1031 attrs[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID;
1032 WRITE32(attrs[0] & readdir->bitmask[0]);
1033 WRITE32(attrs[1] & readdir->bitmask[1]);
Trond Myklebusteadf4592005-06-22 17:16:39 +00001034 dprintk("%s: cookie = %Lu, verifier = 0x%x%x, bitmap = 0x%x%x\n",
1035 __FUNCTION__,
1036 (unsigned long long)readdir->cookie,
1037 ((u32 *)readdir->verifier.data)[0],
1038 ((u32 *)readdir->verifier.data)[1],
1039 attrs[0] & readdir->bitmask[0],
1040 attrs[1] & readdir->bitmask[1]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001041
1042 /* set up reply kvec
1043 * toplevel_status + taglen + rescount + OP_PUTFH + status
1044 * + OP_READDIR + status + verifer(2) = 9
1045 */
1046 replen = (RPC_REPHDRSIZE + auth->au_rslack + 9) << 2;
1047 xdr_inline_pages(&req->rq_rcv_buf, replen, readdir->pages,
1048 readdir->pgbase, readdir->count);
Trond Myklebusteadf4592005-06-22 17:16:39 +00001049 dprintk("%s: inlined page args = (%u, %p, %u, %u)\n",
1050 __FUNCTION__, replen, readdir->pages,
1051 readdir->pgbase, readdir->count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001052
1053 return 0;
1054}
1055
1056static int encode_readlink(struct xdr_stream *xdr, const struct nfs4_readlink *readlink, struct rpc_rqst *req)
1057{
1058 struct rpc_auth *auth = req->rq_task->tk_auth;
1059 unsigned int replen;
1060 uint32_t *p;
1061
1062 RESERVE_SPACE(4);
1063 WRITE32(OP_READLINK);
1064
1065 /* set up reply kvec
1066 * toplevel_status + taglen + rescount + OP_PUTFH + status
1067 * + OP_READLINK + status + string length = 8
1068 */
1069 replen = (RPC_REPHDRSIZE + auth->au_rslack + 8) << 2;
1070 xdr_inline_pages(&req->rq_rcv_buf, replen, readlink->pages,
1071 readlink->pgbase, readlink->pglen);
1072
1073 return 0;
1074}
1075
1076static int encode_remove(struct xdr_stream *xdr, const struct qstr *name)
1077{
1078 uint32_t *p;
1079
1080 RESERVE_SPACE(8 + name->len);
1081 WRITE32(OP_REMOVE);
1082 WRITE32(name->len);
1083 WRITEMEM(name->name, name->len);
1084
1085 return 0;
1086}
1087
1088static int encode_rename(struct xdr_stream *xdr, const struct qstr *oldname, const struct qstr *newname)
1089{
1090 uint32_t *p;
1091
1092 RESERVE_SPACE(8 + oldname->len);
1093 WRITE32(OP_RENAME);
1094 WRITE32(oldname->len);
1095 WRITEMEM(oldname->name, oldname->len);
1096
1097 RESERVE_SPACE(4 + newname->len);
1098 WRITE32(newname->len);
1099 WRITEMEM(newname->name, newname->len);
1100
1101 return 0;
1102}
1103
1104static int encode_renew(struct xdr_stream *xdr, const struct nfs4_client *client_stateid)
1105{
1106 uint32_t *p;
1107
1108 RESERVE_SPACE(12);
1109 WRITE32(OP_RENEW);
1110 WRITE64(client_stateid->cl_clientid);
1111
1112 return 0;
1113}
1114
1115static int
J. Bruce Fields23ec6962005-06-22 17:16:22 +00001116encode_setacl(struct xdr_stream *xdr, struct nfs_setaclargs *arg)
1117{
1118 uint32_t *p;
1119
1120 RESERVE_SPACE(4+sizeof(zero_stateid.data));
1121 WRITE32(OP_SETATTR);
1122 WRITEMEM(zero_stateid.data, sizeof(zero_stateid.data));
1123 RESERVE_SPACE(2*4);
1124 WRITE32(1);
1125 WRITE32(FATTR4_WORD0_ACL);
1126 if (arg->acl_len % 4)
1127 return -EINVAL;
1128 RESERVE_SPACE(4);
1129 WRITE32(arg->acl_len);
1130 xdr_write_pages(xdr, arg->acl_pages, arg->acl_pgbase, arg->acl_len);
1131 return 0;
1132}
1133
1134static int
Linus Torvalds1da177e2005-04-16 15:20:36 -07001135encode_savefh(struct xdr_stream *xdr)
1136{
1137 uint32_t *p;
1138
1139 RESERVE_SPACE(4);
1140 WRITE32(OP_SAVEFH);
1141
1142 return 0;
1143}
1144
1145static int encode_setattr(struct xdr_stream *xdr, const struct nfs_setattrargs *arg, const struct nfs_server *server)
1146{
1147 int status;
1148 uint32_t *p;
1149
1150 RESERVE_SPACE(4+sizeof(arg->stateid.data));
1151 WRITE32(OP_SETATTR);
1152 WRITEMEM(arg->stateid.data, sizeof(arg->stateid.data));
1153
1154 if ((status = encode_attrs(xdr, arg->iap, server)))
1155 return status;
1156
1157 return 0;
1158}
1159
1160static int encode_setclientid(struct xdr_stream *xdr, const struct nfs4_setclientid *setclientid)
1161{
1162 uint32_t *p;
1163
1164 RESERVE_SPACE(4 + sizeof(setclientid->sc_verifier->data));
1165 WRITE32(OP_SETCLIENTID);
1166 WRITEMEM(setclientid->sc_verifier->data, sizeof(setclientid->sc_verifier->data));
1167
1168 encode_string(xdr, setclientid->sc_name_len, setclientid->sc_name);
1169 RESERVE_SPACE(4);
1170 WRITE32(setclientid->sc_prog);
1171 encode_string(xdr, setclientid->sc_netid_len, setclientid->sc_netid);
1172 encode_string(xdr, setclientid->sc_uaddr_len, setclientid->sc_uaddr);
1173 RESERVE_SPACE(4);
1174 WRITE32(setclientid->sc_cb_ident);
1175
1176 return 0;
1177}
1178
1179static int encode_setclientid_confirm(struct xdr_stream *xdr, const struct nfs4_client *client_state)
1180{
1181 uint32_t *p;
1182
1183 RESERVE_SPACE(12 + sizeof(client_state->cl_confirm.data));
1184 WRITE32(OP_SETCLIENTID_CONFIRM);
1185 WRITE64(client_state->cl_clientid);
1186 WRITEMEM(client_state->cl_confirm.data, sizeof(client_state->cl_confirm.data));
1187
1188 return 0;
1189}
1190
1191static int encode_write(struct xdr_stream *xdr, const struct nfs_writeargs *args)
1192{
1193 uint32_t *p;
1194
1195 RESERVE_SPACE(4);
1196 WRITE32(OP_WRITE);
1197
1198 encode_stateid(xdr, args->context);
1199
1200 RESERVE_SPACE(16);
1201 WRITE64(args->offset);
1202 WRITE32(args->stable);
1203 WRITE32(args->count);
1204
1205 xdr_write_pages(xdr, args->pages, args->pgbase, args->count);
1206
1207 return 0;
1208}
1209
1210static int encode_delegreturn(struct xdr_stream *xdr, const nfs4_stateid *stateid)
1211{
1212 uint32_t *p;
1213
1214 RESERVE_SPACE(20);
1215
1216 WRITE32(OP_DELEGRETURN);
1217 WRITEMEM(stateid->data, sizeof(stateid->data));
1218 return 0;
1219
1220}
1221/*
1222 * END OF "GENERIC" ENCODE ROUTINES.
1223 */
1224
1225/*
1226 * Encode an ACCESS request
1227 */
1228static int nfs4_xdr_enc_access(struct rpc_rqst *req, uint32_t *p, const struct nfs4_accessargs *args)
1229{
1230 struct xdr_stream xdr;
1231 struct compound_hdr hdr = {
1232 .nops = 2,
1233 };
1234 int status;
1235
1236 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1237 encode_compound_hdr(&xdr, &hdr);
1238 if ((status = encode_putfh(&xdr, args->fh)) == 0)
1239 status = encode_access(&xdr, args->access);
1240 return status;
1241}
1242
1243/*
1244 * Encode LOOKUP request
1245 */
1246static int nfs4_xdr_enc_lookup(struct rpc_rqst *req, uint32_t *p, const struct nfs4_lookup_arg *args)
1247{
1248 struct xdr_stream xdr;
1249 struct compound_hdr hdr = {
1250 .nops = 4,
1251 };
1252 int status;
1253
1254 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1255 encode_compound_hdr(&xdr, &hdr);
1256 if ((status = encode_putfh(&xdr, args->dir_fh)) != 0)
1257 goto out;
1258 if ((status = encode_lookup(&xdr, args->name)) != 0)
1259 goto out;
1260 if ((status = encode_getfh(&xdr)) != 0)
1261 goto out;
1262 status = encode_getfattr(&xdr, args->bitmask);
1263out:
1264 return status;
1265}
1266
1267/*
1268 * Encode LOOKUP_ROOT request
1269 */
1270static int nfs4_xdr_enc_lookup_root(struct rpc_rqst *req, uint32_t *p, const struct nfs4_lookup_root_arg *args)
1271{
1272 struct xdr_stream xdr;
1273 struct compound_hdr hdr = {
1274 .nops = 3,
1275 };
1276 int status;
1277
1278 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1279 encode_compound_hdr(&xdr, &hdr);
1280 if ((status = encode_putrootfh(&xdr)) != 0)
1281 goto out;
1282 if ((status = encode_getfh(&xdr)) == 0)
1283 status = encode_getfattr(&xdr, args->bitmask);
1284out:
1285 return status;
1286}
1287
1288/*
1289 * Encode REMOVE request
1290 */
1291static int nfs4_xdr_enc_remove(struct rpc_rqst *req, uint32_t *p, const struct nfs4_remove_arg *args)
1292{
1293 struct xdr_stream xdr;
1294 struct compound_hdr hdr = {
1295 .nops = 2,
1296 };
1297 int status;
1298
1299 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1300 encode_compound_hdr(&xdr, &hdr);
1301 if ((status = encode_putfh(&xdr, args->fh)) == 0)
1302 status = encode_remove(&xdr, args->name);
1303 return status;
1304}
1305
1306/*
1307 * Encode RENAME request
1308 */
1309static int nfs4_xdr_enc_rename(struct rpc_rqst *req, uint32_t *p, const struct nfs4_rename_arg *args)
1310{
1311 struct xdr_stream xdr;
1312 struct compound_hdr hdr = {
1313 .nops = 4,
1314 };
1315 int status;
1316
1317 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1318 encode_compound_hdr(&xdr, &hdr);
1319 if ((status = encode_putfh(&xdr, args->old_dir)) != 0)
1320 goto out;
1321 if ((status = encode_savefh(&xdr)) != 0)
1322 goto out;
1323 if ((status = encode_putfh(&xdr, args->new_dir)) != 0)
1324 goto out;
1325 status = encode_rename(&xdr, args->old_name, args->new_name);
1326out:
1327 return status;
1328}
1329
1330/*
1331 * Encode LINK request
1332 */
1333static int nfs4_xdr_enc_link(struct rpc_rqst *req, uint32_t *p, const struct nfs4_link_arg *args)
1334{
1335 struct xdr_stream xdr;
1336 struct compound_hdr hdr = {
1337 .nops = 4,
1338 };
1339 int status;
1340
1341 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1342 encode_compound_hdr(&xdr, &hdr);
1343 if ((status = encode_putfh(&xdr, args->fh)) != 0)
1344 goto out;
1345 if ((status = encode_savefh(&xdr)) != 0)
1346 goto out;
1347 if ((status = encode_putfh(&xdr, args->dir_fh)) != 0)
1348 goto out;
1349 status = encode_link(&xdr, args->name);
1350out:
1351 return status;
1352}
1353
1354/*
1355 * Encode CREATE request
1356 */
1357static int nfs4_xdr_enc_create(struct rpc_rqst *req, uint32_t *p, const struct nfs4_create_arg *args)
1358{
1359 struct xdr_stream xdr;
1360 struct compound_hdr hdr = {
1361 .nops = 4,
1362 };
1363 int status;
1364
1365 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1366 encode_compound_hdr(&xdr, &hdr);
1367 if ((status = encode_putfh(&xdr, args->dir_fh)) != 0)
1368 goto out;
1369 if ((status = encode_create(&xdr, args)) != 0)
1370 goto out;
1371 if ((status = encode_getfh(&xdr)) != 0)
1372 goto out;
1373 status = encode_getfattr(&xdr, args->bitmask);
1374out:
1375 return status;
1376}
1377
1378/*
1379 * Encode SYMLINK request
1380 */
1381static int nfs4_xdr_enc_symlink(struct rpc_rqst *req, uint32_t *p, const struct nfs4_create_arg *args)
1382{
1383 return nfs4_xdr_enc_create(req, p, args);
1384}
1385
1386/*
1387 * Encode GETATTR request
1388 */
1389static int nfs4_xdr_enc_getattr(struct rpc_rqst *req, uint32_t *p, const struct nfs4_getattr_arg *args)
1390{
1391 struct xdr_stream xdr;
1392 struct compound_hdr hdr = {
1393 .nops = 2,
1394 };
1395 int status;
1396
1397 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1398 encode_compound_hdr(&xdr, &hdr);
1399 if ((status = encode_putfh(&xdr, args->fh)) == 0)
1400 status = encode_getfattr(&xdr, args->bitmask);
1401 return status;
1402}
1403
1404/*
1405 * Encode a CLOSE request
1406 */
1407static int nfs4_xdr_enc_close(struct rpc_rqst *req, uint32_t *p, struct nfs_closeargs *args)
1408{
1409 struct xdr_stream xdr;
1410 struct compound_hdr hdr = {
1411 .nops = 2,
1412 };
1413 int status;
1414
1415 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1416 encode_compound_hdr(&xdr, &hdr);
1417 status = encode_putfh(&xdr, args->fh);
1418 if(status)
1419 goto out;
1420 status = encode_close(&xdr, args);
1421out:
1422 return status;
1423}
1424
1425/*
1426 * Encode an OPEN request
1427 */
1428static int nfs4_xdr_enc_open(struct rpc_rqst *req, uint32_t *p, struct nfs_openargs *args)
1429{
1430 struct xdr_stream xdr;
1431 struct compound_hdr hdr = {
1432 .nops = 4,
1433 };
1434 int status;
1435
Trond Myklebustcee54fc2005-10-18 14:20:12 -07001436 status = nfs_wait_on_sequence(args->seqid, req->rq_task);
1437 if (status != 0)
1438 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001439 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1440 encode_compound_hdr(&xdr, &hdr);
1441 status = encode_putfh(&xdr, args->fh);
1442 if (status)
1443 goto out;
1444 status = encode_open(&xdr, args);
1445 if (status)
1446 goto out;
1447 status = encode_getfh(&xdr);
1448 if (status)
1449 goto out;
1450 status = encode_getfattr(&xdr, args->bitmask);
1451out:
1452 return status;
1453}
1454
1455/*
1456 * Encode an OPEN_CONFIRM request
1457 */
1458static int nfs4_xdr_enc_open_confirm(struct rpc_rqst *req, uint32_t *p, struct nfs_open_confirmargs *args)
1459{
1460 struct xdr_stream xdr;
1461 struct compound_hdr hdr = {
1462 .nops = 2,
1463 };
1464 int status;
1465
Trond Myklebustcee54fc2005-10-18 14:20:12 -07001466 status = nfs_wait_on_sequence(args->seqid, req->rq_task);
1467 if (status != 0)
1468 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001469 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1470 encode_compound_hdr(&xdr, &hdr);
1471 status = encode_putfh(&xdr, args->fh);
1472 if(status)
1473 goto out;
1474 status = encode_open_confirm(&xdr, args);
1475out:
1476 return status;
1477}
1478
1479/*
1480 * Encode an OPEN request with no attributes.
1481 */
1482static int nfs4_xdr_enc_open_noattr(struct rpc_rqst *req, uint32_t *p, struct nfs_openargs *args)
1483{
1484 struct xdr_stream xdr;
1485 struct compound_hdr hdr = {
1486 .nops = 2,
1487 };
1488 int status;
1489
Trond Myklebustcee54fc2005-10-18 14:20:12 -07001490 status = nfs_wait_on_sequence(args->seqid, req->rq_task);
1491 if (status != 0)
1492 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1494 encode_compound_hdr(&xdr, &hdr);
1495 status = encode_putfh(&xdr, args->fh);
1496 if (status)
1497 goto out;
1498 status = encode_open(&xdr, args);
1499out:
1500 return status;
1501}
1502
1503/*
1504 * Encode an OPEN_DOWNGRADE request
1505 */
1506static int nfs4_xdr_enc_open_downgrade(struct rpc_rqst *req, uint32_t *p, struct nfs_closeargs *args)
1507{
1508 struct xdr_stream xdr;
1509 struct compound_hdr hdr = {
1510 .nops = 2,
1511 };
1512 int status;
1513
1514 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1515 encode_compound_hdr(&xdr, &hdr);
1516 status = encode_putfh(&xdr, args->fh);
1517 if (status)
1518 goto out;
1519 status = encode_open_downgrade(&xdr, args);
1520out:
1521 return status;
1522}
1523
1524/*
1525 * Encode a LOCK request
1526 */
1527static int nfs4_xdr_enc_lock(struct rpc_rqst *req, uint32_t *p, struct nfs_lockargs *args)
1528{
1529 struct xdr_stream xdr;
1530 struct compound_hdr hdr = {
1531 .nops = 2,
1532 };
Trond Myklebustcee54fc2005-10-18 14:20:12 -07001533 struct nfs_lock_opargs *opargs = args->u.lock;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001534 int status;
1535
Trond Myklebust06735b32005-10-18 14:20:15 -07001536 status = nfs_wait_on_sequence(opargs->lock_seqid, req->rq_task);
Trond Myklebustcee54fc2005-10-18 14:20:12 -07001537 if (status != 0)
1538 goto out;
Trond Myklebust06735b32005-10-18 14:20:15 -07001539 /* Do we need to do an open_to_lock_owner? */
1540 if (opargs->lock_seqid->sequence->flags & NFS_SEQID_CONFIRMED)
1541 opargs->new_lock_owner = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001542 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1543 encode_compound_hdr(&xdr, &hdr);
1544 status = encode_putfh(&xdr, args->fh);
1545 if(status)
1546 goto out;
1547 status = encode_lock(&xdr, args);
1548out:
1549 return status;
1550}
1551
1552/*
1553 * Encode a LOCKT request
1554 */
1555static int nfs4_xdr_enc_lockt(struct rpc_rqst *req, uint32_t *p, struct nfs_lockargs *args)
1556{
1557 struct xdr_stream xdr;
1558 struct compound_hdr hdr = {
1559 .nops = 2,
1560 };
1561 int status;
1562
1563 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1564 encode_compound_hdr(&xdr, &hdr);
1565 status = encode_putfh(&xdr, args->fh);
1566 if(status)
1567 goto out;
1568 status = encode_lockt(&xdr, args);
1569out:
1570 return status;
1571}
1572
1573/*
1574 * Encode a LOCKU request
1575 */
1576static int nfs4_xdr_enc_locku(struct rpc_rqst *req, uint32_t *p, struct nfs_lockargs *args)
1577{
1578 struct xdr_stream xdr;
1579 struct compound_hdr hdr = {
1580 .nops = 2,
1581 };
1582 int status;
1583
1584 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1585 encode_compound_hdr(&xdr, &hdr);
1586 status = encode_putfh(&xdr, args->fh);
1587 if(status)
1588 goto out;
1589 status = encode_locku(&xdr, args);
1590out:
1591 return status;
1592}
1593
1594/*
1595 * Encode a READLINK request
1596 */
1597static int nfs4_xdr_enc_readlink(struct rpc_rqst *req, uint32_t *p, const struct nfs4_readlink *args)
1598{
1599 struct xdr_stream xdr;
1600 struct compound_hdr hdr = {
1601 .nops = 2,
1602 };
1603 int status;
1604
1605 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1606 encode_compound_hdr(&xdr, &hdr);
1607 status = encode_putfh(&xdr, args->fh);
1608 if(status)
1609 goto out;
1610 status = encode_readlink(&xdr, args, req);
1611out:
1612 return status;
1613}
1614
1615/*
1616 * Encode a READDIR request
1617 */
1618static int nfs4_xdr_enc_readdir(struct rpc_rqst *req, uint32_t *p, const struct nfs4_readdir_arg *args)
1619{
1620 struct xdr_stream xdr;
1621 struct compound_hdr hdr = {
1622 .nops = 2,
1623 };
1624 int status;
1625
1626 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1627 encode_compound_hdr(&xdr, &hdr);
1628 status = encode_putfh(&xdr, args->fh);
1629 if(status)
1630 goto out;
1631 status = encode_readdir(&xdr, args, req);
1632out:
1633 return status;
1634}
1635
1636/*
1637 * Encode a READ request
1638 */
1639static int nfs4_xdr_enc_read(struct rpc_rqst *req, uint32_t *p, struct nfs_readargs *args)
1640{
1641 struct rpc_auth *auth = req->rq_task->tk_auth;
1642 struct xdr_stream xdr;
1643 struct compound_hdr hdr = {
1644 .nops = 2,
1645 };
1646 int replen, status;
1647
1648 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1649 encode_compound_hdr(&xdr, &hdr);
1650 status = encode_putfh(&xdr, args->fh);
1651 if (status)
1652 goto out;
1653 status = encode_read(&xdr, args);
1654 if (status)
1655 goto out;
1656
1657 /* set up reply kvec
1658 * toplevel status + taglen=0 + rescount + OP_PUTFH + status
1659 * + OP_READ + status + eof + datalen = 9
1660 */
1661 replen = (RPC_REPHDRSIZE + auth->au_rslack + NFS4_dec_read_sz) << 2;
1662 xdr_inline_pages(&req->rq_rcv_buf, replen,
1663 args->pages, args->pgbase, args->count);
1664out:
1665 return status;
1666}
1667
1668/*
1669 * Encode an SETATTR request
1670 */
1671static int nfs4_xdr_enc_setattr(struct rpc_rqst *req, uint32_t *p, struct nfs_setattrargs *args)
1672
1673{
1674 struct xdr_stream xdr;
1675 struct compound_hdr hdr = {
1676 .nops = 3,
1677 };
1678 int status;
1679
1680 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1681 encode_compound_hdr(&xdr, &hdr);
1682 status = encode_putfh(&xdr, args->fh);
1683 if(status)
1684 goto out;
1685 status = encode_setattr(&xdr, args, args->server);
1686 if(status)
1687 goto out;
1688 status = encode_getfattr(&xdr, args->bitmask);
1689out:
1690 return status;
1691}
1692
1693/*
J. Bruce Fields029d1052005-06-22 17:16:22 +00001694 * Encode a GETACL request
1695 */
1696static int
1697nfs4_xdr_enc_getacl(struct rpc_rqst *req, uint32_t *p,
1698 struct nfs_getaclargs *args)
1699{
1700 struct xdr_stream xdr;
1701 struct rpc_auth *auth = req->rq_task->tk_auth;
1702 struct compound_hdr hdr = {
1703 .nops = 2,
1704 };
1705 int replen, status;
1706
1707 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1708 encode_compound_hdr(&xdr, &hdr);
1709 status = encode_putfh(&xdr, args->fh);
1710 if (status)
1711 goto out;
1712 status = encode_getattr_two(&xdr, FATTR4_WORD0_ACL, 0);
1713 /* set up reply buffer: */
1714 replen = (RPC_REPHDRSIZE + auth->au_rslack + NFS4_dec_getacl_sz) << 2;
1715 xdr_inline_pages(&req->rq_rcv_buf, replen,
1716 args->acl_pages, args->acl_pgbase, args->acl_len);
1717out:
1718 return status;
1719}
1720
1721/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07001722 * Encode a WRITE request
1723 */
1724static int nfs4_xdr_enc_write(struct rpc_rqst *req, uint32_t *p, struct nfs_writeargs *args)
1725{
1726 struct xdr_stream xdr;
1727 struct compound_hdr hdr = {
1728 .nops = 2,
1729 };
1730 int status;
1731
1732 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1733 encode_compound_hdr(&xdr, &hdr);
1734 status = encode_putfh(&xdr, args->fh);
1735 if (status)
1736 goto out;
1737 status = encode_write(&xdr, args);
1738out:
1739 return status;
1740}
1741
1742/*
1743 * a COMMIT request
1744 */
1745static int nfs4_xdr_enc_commit(struct rpc_rqst *req, uint32_t *p, struct nfs_writeargs *args)
1746{
1747 struct xdr_stream xdr;
1748 struct compound_hdr hdr = {
1749 .nops = 2,
1750 };
1751 int status;
1752
1753 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1754 encode_compound_hdr(&xdr, &hdr);
1755 status = encode_putfh(&xdr, args->fh);
1756 if (status)
1757 goto out;
1758 status = encode_commit(&xdr, args);
1759out:
1760 return status;
1761}
1762
1763/*
1764 * FSINFO request
1765 */
1766static int nfs4_xdr_enc_fsinfo(struct rpc_rqst *req, uint32_t *p, struct nfs4_fsinfo_arg *args)
1767{
1768 struct xdr_stream xdr;
1769 struct compound_hdr hdr = {
1770 .nops = 2,
1771 };
1772 int status;
1773
1774 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1775 encode_compound_hdr(&xdr, &hdr);
1776 status = encode_putfh(&xdr, args->fh);
1777 if (!status)
1778 status = encode_fsinfo(&xdr, args->bitmask);
1779 return status;
1780}
1781
1782/*
1783 * a PATHCONF request
1784 */
1785static int nfs4_xdr_enc_pathconf(struct rpc_rqst *req, uint32_t *p, const struct nfs4_pathconf_arg *args)
1786{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001787 struct xdr_stream xdr;
1788 struct compound_hdr hdr = {
1789 .nops = 2,
1790 };
1791 int status;
1792
1793 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1794 encode_compound_hdr(&xdr, &hdr);
1795 status = encode_putfh(&xdr, args->fh);
1796 if (!status)
1797 status = encode_getattr_one(&xdr,
1798 args->bitmask[0] & nfs4_pathconf_bitmap[0]);
1799 return status;
1800}
1801
1802/*
1803 * a STATFS request
1804 */
1805static int nfs4_xdr_enc_statfs(struct rpc_rqst *req, uint32_t *p, const struct nfs4_statfs_arg *args)
1806{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001807 struct xdr_stream xdr;
1808 struct compound_hdr hdr = {
1809 .nops = 2,
1810 };
1811 int status;
1812
1813 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1814 encode_compound_hdr(&xdr, &hdr);
1815 status = encode_putfh(&xdr, args->fh);
1816 if (status == 0)
1817 status = encode_getattr_two(&xdr,
1818 args->bitmask[0] & nfs4_statfs_bitmap[0],
1819 args->bitmask[1] & nfs4_statfs_bitmap[1]);
1820 return status;
1821}
1822
1823/*
1824 * GETATTR_BITMAP request
1825 */
1826static int nfs4_xdr_enc_server_caps(struct rpc_rqst *req, uint32_t *p, const struct nfs_fh *fhandle)
1827{
1828 struct xdr_stream xdr;
1829 struct compound_hdr hdr = {
1830 .nops = 2,
1831 };
1832 int status;
1833
1834 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1835 encode_compound_hdr(&xdr, &hdr);
1836 status = encode_putfh(&xdr, fhandle);
1837 if (status == 0)
1838 status = encode_getattr_one(&xdr, FATTR4_WORD0_SUPPORTED_ATTRS|
1839 FATTR4_WORD0_LINK_SUPPORT|
1840 FATTR4_WORD0_SYMLINK_SUPPORT|
1841 FATTR4_WORD0_ACLSUPPORT);
1842 return status;
1843}
1844
1845/*
1846 * a RENEW request
1847 */
1848static int nfs4_xdr_enc_renew(struct rpc_rqst *req, uint32_t *p, struct nfs4_client *clp)
1849{
1850 struct xdr_stream xdr;
1851 struct compound_hdr hdr = {
1852 .nops = 1,
1853 };
1854
1855 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1856 encode_compound_hdr(&xdr, &hdr);
1857 return encode_renew(&xdr, clp);
1858}
1859
1860/*
1861 * a SETCLIENTID request
1862 */
1863static int nfs4_xdr_enc_setclientid(struct rpc_rqst *req, uint32_t *p, struct nfs4_setclientid *sc)
1864{
1865 struct xdr_stream xdr;
1866 struct compound_hdr hdr = {
1867 .nops = 1,
1868 };
1869
1870 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1871 encode_compound_hdr(&xdr, &hdr);
1872 return encode_setclientid(&xdr, sc);
1873}
1874
1875/*
1876 * a SETCLIENTID_CONFIRM request
1877 */
1878static int nfs4_xdr_enc_setclientid_confirm(struct rpc_rqst *req, uint32_t *p, struct nfs4_client *clp)
1879{
1880 struct xdr_stream xdr;
1881 struct compound_hdr hdr = {
1882 .nops = 3,
1883 };
1884 const u32 lease_bitmap[2] = { FATTR4_WORD0_LEASE_TIME, 0 };
1885 int status;
1886
1887 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1888 encode_compound_hdr(&xdr, &hdr);
1889 status = encode_setclientid_confirm(&xdr, clp);
1890 if (!status)
1891 status = encode_putrootfh(&xdr);
1892 if (!status)
1893 status = encode_fsinfo(&xdr, lease_bitmap);
1894 return status;
1895}
1896
1897/*
1898 * DELEGRETURN request
1899 */
1900static int nfs4_xdr_enc_delegreturn(struct rpc_rqst *req, uint32_t *p, const struct nfs4_delegreturnargs *args)
1901{
1902 struct xdr_stream xdr;
1903 struct compound_hdr hdr = {
1904 .nops = 2,
1905 };
1906 int status;
1907
1908 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
1909 encode_compound_hdr(&xdr, &hdr);
1910 if ((status = encode_putfh(&xdr, args->fhandle)) == 0)
1911 status = encode_delegreturn(&xdr, args->stateid);
1912 return status;
1913}
1914
1915/*
1916 * START OF "GENERIC" DECODE ROUTINES.
1917 * These may look a little ugly since they are imported from a "generic"
1918 * set of XDR encode/decode routines which are intended to be shared by
1919 * all of our NFSv4 implementations (OpenBSD, MacOS X...).
1920 *
1921 * If the pain of reading these is too great, it should be a straightforward
1922 * task to translate them into Linux-specific versions which are more
1923 * consistent with the style used in NFSv2/v3...
1924 */
1925#define READ32(x) (x) = ntohl(*p++)
1926#define READ64(x) do { \
1927 (x) = (u64)ntohl(*p++) << 32; \
1928 (x) |= ntohl(*p++); \
1929} while (0)
1930#define READTIME(x) do { \
1931 p++; \
1932 (x.tv_sec) = ntohl(*p++); \
1933 (x.tv_nsec) = ntohl(*p++); \
1934} while (0)
1935#define COPYMEM(x,nbytes) do { \
1936 memcpy((x), p, nbytes); \
1937 p += XDR_QUADLEN(nbytes); \
1938} while (0)
1939
1940#define READ_BUF(nbytes) do { \
1941 p = xdr_inline_decode(xdr, nbytes); \
1942 if (!p) { \
1943 printk(KERN_WARNING "%s: reply buffer overflowed in line %d.", \
1944 __FUNCTION__, __LINE__); \
1945 return -EIO; \
1946 } \
1947} while (0)
1948
1949static int decode_opaque_inline(struct xdr_stream *xdr, uint32_t *len, char **string)
1950{
1951 uint32_t *p;
1952
1953 READ_BUF(4);
1954 READ32(*len);
1955 READ_BUF(*len);
1956 *string = (char *)p;
1957 return 0;
1958}
1959
1960static int decode_compound_hdr(struct xdr_stream *xdr, struct compound_hdr *hdr)
1961{
1962 uint32_t *p;
1963
1964 READ_BUF(8);
1965 READ32(hdr->status);
1966 READ32(hdr->taglen);
1967
1968 READ_BUF(hdr->taglen + 4);
1969 hdr->tag = (char *)p;
1970 p += XDR_QUADLEN(hdr->taglen);
1971 READ32(hdr->nops);
1972 return 0;
1973}
1974
1975static int decode_op_hdr(struct xdr_stream *xdr, enum nfs_opnum4 expected)
1976{
1977 uint32_t *p;
1978 uint32_t opnum;
1979 int32_t nfserr;
1980
1981 READ_BUF(8);
1982 READ32(opnum);
1983 if (opnum != expected) {
1984 printk(KERN_NOTICE
1985 "nfs4_decode_op_hdr: Server returned operation"
1986 " %d but we issued a request for %d\n",
1987 opnum, expected);
1988 return -EIO;
1989 }
1990 READ32(nfserr);
1991 if (nfserr != NFS_OK)
1992 return -nfs_stat_to_errno(nfserr);
1993 return 0;
1994}
1995
1996/* Dummy routine */
1997static int decode_ace(struct xdr_stream *xdr, void *ace, struct nfs4_client *clp)
1998{
1999 uint32_t *p;
2000 uint32_t strlen;
2001 char *str;
2002
2003 READ_BUF(12);
2004 return decode_opaque_inline(xdr, &strlen, &str);
2005}
2006
2007static int decode_attr_bitmap(struct xdr_stream *xdr, uint32_t *bitmap)
2008{
2009 uint32_t bmlen, *p;
2010
2011 READ_BUF(4);
2012 READ32(bmlen);
2013
2014 bitmap[0] = bitmap[1] = 0;
2015 READ_BUF((bmlen << 2));
2016 if (bmlen > 0) {
2017 READ32(bitmap[0]);
2018 if (bmlen > 1)
2019 READ32(bitmap[1]);
2020 }
2021 return 0;
2022}
2023
2024static inline int decode_attr_length(struct xdr_stream *xdr, uint32_t *attrlen, uint32_t **savep)
2025{
2026 uint32_t *p;
2027
2028 READ_BUF(4);
2029 READ32(*attrlen);
2030 *savep = xdr->p;
2031 return 0;
2032}
2033
2034static int decode_attr_supported(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *bitmask)
2035{
2036 if (likely(bitmap[0] & FATTR4_WORD0_SUPPORTED_ATTRS)) {
2037 decode_attr_bitmap(xdr, bitmask);
2038 bitmap[0] &= ~FATTR4_WORD0_SUPPORTED_ATTRS;
2039 } else
2040 bitmask[0] = bitmask[1] = 0;
2041 dprintk("%s: bitmask=0x%x%x\n", __FUNCTION__, bitmask[0], bitmask[1]);
2042 return 0;
2043}
2044
2045static int decode_attr_type(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *type)
2046{
2047 uint32_t *p;
2048
2049 *type = 0;
2050 if (unlikely(bitmap[0] & (FATTR4_WORD0_TYPE - 1U)))
2051 return -EIO;
2052 if (likely(bitmap[0] & FATTR4_WORD0_TYPE)) {
2053 READ_BUF(4);
2054 READ32(*type);
2055 if (*type < NF4REG || *type > NF4NAMEDATTR) {
2056 dprintk("%s: bad type %d\n", __FUNCTION__, *type);
2057 return -EIO;
2058 }
2059 bitmap[0] &= ~FATTR4_WORD0_TYPE;
2060 }
2061 dprintk("%s: type=0%o\n", __FUNCTION__, nfs_type2fmt[*type].nfs2type);
2062 return 0;
2063}
2064
2065static int decode_attr_change(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *change)
2066{
2067 uint32_t *p;
2068
2069 *change = 0;
2070 if (unlikely(bitmap[0] & (FATTR4_WORD0_CHANGE - 1U)))
2071 return -EIO;
2072 if (likely(bitmap[0] & FATTR4_WORD0_CHANGE)) {
2073 READ_BUF(8);
2074 READ64(*change);
2075 bitmap[0] &= ~FATTR4_WORD0_CHANGE;
2076 }
2077 dprintk("%s: change attribute=%Lu\n", __FUNCTION__,
2078 (unsigned long long)*change);
2079 return 0;
2080}
2081
2082static int decode_attr_size(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *size)
2083{
2084 uint32_t *p;
2085
2086 *size = 0;
2087 if (unlikely(bitmap[0] & (FATTR4_WORD0_SIZE - 1U)))
2088 return -EIO;
2089 if (likely(bitmap[0] & FATTR4_WORD0_SIZE)) {
2090 READ_BUF(8);
2091 READ64(*size);
2092 bitmap[0] &= ~FATTR4_WORD0_SIZE;
2093 }
2094 dprintk("%s: file size=%Lu\n", __FUNCTION__, (unsigned long long)*size);
2095 return 0;
2096}
2097
2098static int decode_attr_link_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2099{
2100 uint32_t *p;
2101
2102 *res = 0;
2103 if (unlikely(bitmap[0] & (FATTR4_WORD0_LINK_SUPPORT - 1U)))
2104 return -EIO;
2105 if (likely(bitmap[0] & FATTR4_WORD0_LINK_SUPPORT)) {
2106 READ_BUF(4);
2107 READ32(*res);
2108 bitmap[0] &= ~FATTR4_WORD0_LINK_SUPPORT;
2109 }
2110 dprintk("%s: link support=%s\n", __FUNCTION__, *res == 0 ? "false" : "true");
2111 return 0;
2112}
2113
2114static int decode_attr_symlink_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2115{
2116 uint32_t *p;
2117
2118 *res = 0;
2119 if (unlikely(bitmap[0] & (FATTR4_WORD0_SYMLINK_SUPPORT - 1U)))
2120 return -EIO;
2121 if (likely(bitmap[0] & FATTR4_WORD0_SYMLINK_SUPPORT)) {
2122 READ_BUF(4);
2123 READ32(*res);
2124 bitmap[0] &= ~FATTR4_WORD0_SYMLINK_SUPPORT;
2125 }
2126 dprintk("%s: symlink support=%s\n", __FUNCTION__, *res == 0 ? "false" : "true");
2127 return 0;
2128}
2129
2130static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_fsid *fsid)
2131{
2132 uint32_t *p;
2133
2134 fsid->major = 0;
2135 fsid->minor = 0;
2136 if (unlikely(bitmap[0] & (FATTR4_WORD0_FSID - 1U)))
2137 return -EIO;
2138 if (likely(bitmap[0] & FATTR4_WORD0_FSID)) {
2139 READ_BUF(16);
2140 READ64(fsid->major);
2141 READ64(fsid->minor);
2142 bitmap[0] &= ~FATTR4_WORD0_FSID;
2143 }
2144 dprintk("%s: fsid=(0x%Lx/0x%Lx)\n", __FUNCTION__,
2145 (unsigned long long)fsid->major,
2146 (unsigned long long)fsid->minor);
2147 return 0;
2148}
2149
2150static int decode_attr_lease_time(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2151{
2152 uint32_t *p;
2153
2154 *res = 60;
2155 if (unlikely(bitmap[0] & (FATTR4_WORD0_LEASE_TIME - 1U)))
2156 return -EIO;
2157 if (likely(bitmap[0] & FATTR4_WORD0_LEASE_TIME)) {
2158 READ_BUF(4);
2159 READ32(*res);
2160 bitmap[0] &= ~FATTR4_WORD0_LEASE_TIME;
2161 }
2162 dprintk("%s: file size=%u\n", __FUNCTION__, (unsigned int)*res);
2163 return 0;
2164}
2165
2166static int decode_attr_aclsupport(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2167{
2168 uint32_t *p;
2169
2170 *res = ACL4_SUPPORT_ALLOW_ACL|ACL4_SUPPORT_DENY_ACL;
2171 if (unlikely(bitmap[0] & (FATTR4_WORD0_ACLSUPPORT - 1U)))
2172 return -EIO;
2173 if (likely(bitmap[0] & FATTR4_WORD0_ACLSUPPORT)) {
2174 READ_BUF(4);
2175 READ32(*res);
2176 bitmap[0] &= ~FATTR4_WORD0_ACLSUPPORT;
2177 }
2178 dprintk("%s: ACLs supported=%u\n", __FUNCTION__, (unsigned int)*res);
2179 return 0;
2180}
2181
2182static int decode_attr_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid)
2183{
2184 uint32_t *p;
2185
2186 *fileid = 0;
2187 if (unlikely(bitmap[0] & (FATTR4_WORD0_FILEID - 1U)))
2188 return -EIO;
2189 if (likely(bitmap[0] & FATTR4_WORD0_FILEID)) {
2190 READ_BUF(8);
2191 READ64(*fileid);
2192 bitmap[0] &= ~FATTR4_WORD0_FILEID;
2193 }
2194 dprintk("%s: fileid=%Lu\n", __FUNCTION__, (unsigned long long)*fileid);
2195 return 0;
2196}
2197
2198static int decode_attr_files_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2199{
2200 uint32_t *p;
2201 int status = 0;
2202
2203 *res = 0;
2204 if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_AVAIL - 1U)))
2205 return -EIO;
2206 if (likely(bitmap[0] & FATTR4_WORD0_FILES_AVAIL)) {
2207 READ_BUF(8);
2208 READ64(*res);
2209 bitmap[0] &= ~FATTR4_WORD0_FILES_AVAIL;
2210 }
2211 dprintk("%s: files avail=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2212 return status;
2213}
2214
2215static int decode_attr_files_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2216{
2217 uint32_t *p;
2218 int status = 0;
2219
2220 *res = 0;
2221 if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_FREE - 1U)))
2222 return -EIO;
2223 if (likely(bitmap[0] & FATTR4_WORD0_FILES_FREE)) {
2224 READ_BUF(8);
2225 READ64(*res);
2226 bitmap[0] &= ~FATTR4_WORD0_FILES_FREE;
2227 }
2228 dprintk("%s: files free=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2229 return status;
2230}
2231
2232static int decode_attr_files_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2233{
2234 uint32_t *p;
2235 int status = 0;
2236
2237 *res = 0;
2238 if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_TOTAL - 1U)))
2239 return -EIO;
2240 if (likely(bitmap[0] & FATTR4_WORD0_FILES_TOTAL)) {
2241 READ_BUF(8);
2242 READ64(*res);
2243 bitmap[0] &= ~FATTR4_WORD0_FILES_TOTAL;
2244 }
2245 dprintk("%s: files total=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2246 return status;
2247}
2248
2249static int decode_attr_maxfilesize(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2250{
2251 uint32_t *p;
2252 int status = 0;
2253
2254 *res = 0;
2255 if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXFILESIZE - 1U)))
2256 return -EIO;
2257 if (likely(bitmap[0] & FATTR4_WORD0_MAXFILESIZE)) {
2258 READ_BUF(8);
2259 READ64(*res);
2260 bitmap[0] &= ~FATTR4_WORD0_MAXFILESIZE;
2261 }
2262 dprintk("%s: maxfilesize=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2263 return status;
2264}
2265
2266static int decode_attr_maxlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxlink)
2267{
2268 uint32_t *p;
2269 int status = 0;
2270
2271 *maxlink = 1;
2272 if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXLINK - 1U)))
2273 return -EIO;
2274 if (likely(bitmap[0] & FATTR4_WORD0_MAXLINK)) {
2275 READ_BUF(4);
2276 READ32(*maxlink);
2277 bitmap[0] &= ~FATTR4_WORD0_MAXLINK;
2278 }
2279 dprintk("%s: maxlink=%u\n", __FUNCTION__, *maxlink);
2280 return status;
2281}
2282
2283static int decode_attr_maxname(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxname)
2284{
2285 uint32_t *p;
2286 int status = 0;
2287
2288 *maxname = 1024;
2289 if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXNAME - 1U)))
2290 return -EIO;
2291 if (likely(bitmap[0] & FATTR4_WORD0_MAXNAME)) {
2292 READ_BUF(4);
2293 READ32(*maxname);
2294 bitmap[0] &= ~FATTR4_WORD0_MAXNAME;
2295 }
2296 dprintk("%s: maxname=%u\n", __FUNCTION__, *maxname);
2297 return status;
2298}
2299
2300static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2301{
2302 uint32_t *p;
2303 int status = 0;
2304
2305 *res = 1024;
2306 if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXREAD - 1U)))
2307 return -EIO;
2308 if (likely(bitmap[0] & FATTR4_WORD0_MAXREAD)) {
2309 uint64_t maxread;
2310 READ_BUF(8);
2311 READ64(maxread);
2312 if (maxread > 0x7FFFFFFF)
2313 maxread = 0x7FFFFFFF;
2314 *res = (uint32_t)maxread;
2315 bitmap[0] &= ~FATTR4_WORD0_MAXREAD;
2316 }
2317 dprintk("%s: maxread=%lu\n", __FUNCTION__, (unsigned long)*res);
2318 return status;
2319}
2320
2321static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res)
2322{
2323 uint32_t *p;
2324 int status = 0;
2325
2326 *res = 1024;
2327 if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXWRITE - 1U)))
2328 return -EIO;
2329 if (likely(bitmap[0] & FATTR4_WORD0_MAXWRITE)) {
2330 uint64_t maxwrite;
2331 READ_BUF(8);
2332 READ64(maxwrite);
2333 if (maxwrite > 0x7FFFFFFF)
2334 maxwrite = 0x7FFFFFFF;
2335 *res = (uint32_t)maxwrite;
2336 bitmap[0] &= ~FATTR4_WORD0_MAXWRITE;
2337 }
2338 dprintk("%s: maxwrite=%lu\n", __FUNCTION__, (unsigned long)*res);
2339 return status;
2340}
2341
2342static int decode_attr_mode(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *mode)
2343{
2344 uint32_t *p;
2345
2346 *mode = 0;
2347 if (unlikely(bitmap[1] & (FATTR4_WORD1_MODE - 1U)))
2348 return -EIO;
2349 if (likely(bitmap[1] & FATTR4_WORD1_MODE)) {
2350 READ_BUF(4);
2351 READ32(*mode);
2352 *mode &= ~S_IFMT;
2353 bitmap[1] &= ~FATTR4_WORD1_MODE;
2354 }
2355 dprintk("%s: file mode=0%o\n", __FUNCTION__, (unsigned int)*mode);
2356 return 0;
2357}
2358
2359static int decode_attr_nlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *nlink)
2360{
2361 uint32_t *p;
2362
2363 *nlink = 1;
2364 if (unlikely(bitmap[1] & (FATTR4_WORD1_NUMLINKS - 1U)))
2365 return -EIO;
2366 if (likely(bitmap[1] & FATTR4_WORD1_NUMLINKS)) {
2367 READ_BUF(4);
2368 READ32(*nlink);
2369 bitmap[1] &= ~FATTR4_WORD1_NUMLINKS;
2370 }
2371 dprintk("%s: nlink=%u\n", __FUNCTION__, (unsigned int)*nlink);
2372 return 0;
2373}
2374
2375static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_client *clp, int32_t *uid)
2376{
2377 uint32_t len, *p;
2378
2379 *uid = -2;
2380 if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER - 1U)))
2381 return -EIO;
2382 if (likely(bitmap[1] & FATTR4_WORD1_OWNER)) {
2383 READ_BUF(4);
2384 READ32(len);
2385 READ_BUF(len);
2386 if (len < XDR_MAX_NETOBJ) {
2387 if (nfs_map_name_to_uid(clp, (char *)p, len, uid) != 0)
2388 dprintk("%s: nfs_map_name_to_uid failed!\n",
2389 __FUNCTION__);
2390 } else
2391 printk(KERN_WARNING "%s: name too long (%u)!\n",
2392 __FUNCTION__, len);
2393 bitmap[1] &= ~FATTR4_WORD1_OWNER;
2394 }
2395 dprintk("%s: uid=%d\n", __FUNCTION__, (int)*uid);
2396 return 0;
2397}
2398
2399static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_client *clp, int32_t *gid)
2400{
2401 uint32_t len, *p;
2402
2403 *gid = -2;
2404 if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER_GROUP - 1U)))
2405 return -EIO;
2406 if (likely(bitmap[1] & FATTR4_WORD1_OWNER_GROUP)) {
2407 READ_BUF(4);
2408 READ32(len);
2409 READ_BUF(len);
2410 if (len < XDR_MAX_NETOBJ) {
2411 if (nfs_map_group_to_gid(clp, (char *)p, len, gid) != 0)
2412 dprintk("%s: nfs_map_group_to_gid failed!\n",
2413 __FUNCTION__);
2414 } else
2415 printk(KERN_WARNING "%s: name too long (%u)!\n",
2416 __FUNCTION__, len);
2417 bitmap[1] &= ~FATTR4_WORD1_OWNER_GROUP;
2418 }
2419 dprintk("%s: gid=%d\n", __FUNCTION__, (int)*gid);
2420 return 0;
2421}
2422
2423static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rdev)
2424{
2425 uint32_t major = 0, minor = 0, *p;
2426
2427 *rdev = MKDEV(0,0);
2428 if (unlikely(bitmap[1] & (FATTR4_WORD1_RAWDEV - 1U)))
2429 return -EIO;
2430 if (likely(bitmap[1] & FATTR4_WORD1_RAWDEV)) {
2431 dev_t tmp;
2432
2433 READ_BUF(8);
2434 READ32(major);
2435 READ32(minor);
2436 tmp = MKDEV(major, minor);
2437 if (MAJOR(tmp) == major && MINOR(tmp) == minor)
2438 *rdev = tmp;
2439 bitmap[1] &= ~ FATTR4_WORD1_RAWDEV;
2440 }
2441 dprintk("%s: rdev=(0x%x:0x%x)\n", __FUNCTION__, major, minor);
2442 return 0;
2443}
2444
2445static int decode_attr_space_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2446{
2447 uint32_t *p;
2448 int status = 0;
2449
2450 *res = 0;
2451 if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_AVAIL - 1U)))
2452 return -EIO;
2453 if (likely(bitmap[1] & FATTR4_WORD1_SPACE_AVAIL)) {
2454 READ_BUF(8);
2455 READ64(*res);
2456 bitmap[1] &= ~FATTR4_WORD1_SPACE_AVAIL;
2457 }
2458 dprintk("%s: space avail=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2459 return status;
2460}
2461
2462static int decode_attr_space_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2463{
2464 uint32_t *p;
2465 int status = 0;
2466
2467 *res = 0;
2468 if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_FREE - 1U)))
2469 return -EIO;
2470 if (likely(bitmap[1] & FATTR4_WORD1_SPACE_FREE)) {
2471 READ_BUF(8);
2472 READ64(*res);
2473 bitmap[1] &= ~FATTR4_WORD1_SPACE_FREE;
2474 }
2475 dprintk("%s: space free=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2476 return status;
2477}
2478
2479static int decode_attr_space_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res)
2480{
2481 uint32_t *p;
2482 int status = 0;
2483
2484 *res = 0;
2485 if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_TOTAL - 1U)))
2486 return -EIO;
2487 if (likely(bitmap[1] & FATTR4_WORD1_SPACE_TOTAL)) {
2488 READ_BUF(8);
2489 READ64(*res);
2490 bitmap[1] &= ~FATTR4_WORD1_SPACE_TOTAL;
2491 }
2492 dprintk("%s: space total=%Lu\n", __FUNCTION__, (unsigned long long)*res);
2493 return status;
2494}
2495
2496static int decode_attr_space_used(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *used)
2497{
2498 uint32_t *p;
2499
2500 *used = 0;
2501 if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_USED - 1U)))
2502 return -EIO;
2503 if (likely(bitmap[1] & FATTR4_WORD1_SPACE_USED)) {
2504 READ_BUF(8);
2505 READ64(*used);
2506 bitmap[1] &= ~FATTR4_WORD1_SPACE_USED;
2507 }
2508 dprintk("%s: space used=%Lu\n", __FUNCTION__,
2509 (unsigned long long)*used);
2510 return 0;
2511}
2512
2513static int decode_attr_time(struct xdr_stream *xdr, struct timespec *time)
2514{
2515 uint32_t *p;
2516 uint64_t sec;
2517 uint32_t nsec;
2518
2519 READ_BUF(12);
2520 READ64(sec);
2521 READ32(nsec);
2522 time->tv_sec = (time_t)sec;
2523 time->tv_nsec = (long)nsec;
2524 return 0;
2525}
2526
2527static int decode_attr_time_access(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
2528{
2529 int status = 0;
2530
2531 time->tv_sec = 0;
2532 time->tv_nsec = 0;
2533 if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_ACCESS - 1U)))
2534 return -EIO;
2535 if (likely(bitmap[1] & FATTR4_WORD1_TIME_ACCESS)) {
2536 status = decode_attr_time(xdr, time);
2537 bitmap[1] &= ~FATTR4_WORD1_TIME_ACCESS;
2538 }
2539 dprintk("%s: atime=%ld\n", __FUNCTION__, (long)time->tv_sec);
2540 return status;
2541}
2542
2543static int decode_attr_time_metadata(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
2544{
2545 int status = 0;
2546
2547 time->tv_sec = 0;
2548 time->tv_nsec = 0;
2549 if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_METADATA - 1U)))
2550 return -EIO;
2551 if (likely(bitmap[1] & FATTR4_WORD1_TIME_METADATA)) {
2552 status = decode_attr_time(xdr, time);
2553 bitmap[1] &= ~FATTR4_WORD1_TIME_METADATA;
2554 }
2555 dprintk("%s: ctime=%ld\n", __FUNCTION__, (long)time->tv_sec);
2556 return status;
2557}
2558
2559static int decode_attr_time_modify(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time)
2560{
2561 int status = 0;
2562
2563 time->tv_sec = 0;
2564 time->tv_nsec = 0;
2565 if (unlikely(bitmap[1] & (FATTR4_WORD1_TIME_MODIFY - 1U)))
2566 return -EIO;
2567 if (likely(bitmap[1] & FATTR4_WORD1_TIME_MODIFY)) {
2568 status = decode_attr_time(xdr, time);
2569 bitmap[1] &= ~FATTR4_WORD1_TIME_MODIFY;
2570 }
2571 dprintk("%s: mtime=%ld\n", __FUNCTION__, (long)time->tv_sec);
2572 return status;
2573}
2574
2575static int verify_attr_len(struct xdr_stream *xdr, uint32_t *savep, uint32_t attrlen)
2576{
2577 unsigned int attrwords = XDR_QUADLEN(attrlen);
2578 unsigned int nwords = xdr->p - savep;
2579
2580 if (unlikely(attrwords != nwords)) {
2581 printk(KERN_WARNING "%s: server returned incorrect attribute length: %u %c %u\n",
2582 __FUNCTION__,
2583 attrwords << 2,
2584 (attrwords < nwords) ? '<' : '>',
2585 nwords << 2);
2586 return -EIO;
2587 }
2588 return 0;
2589}
2590
2591static int decode_change_info(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
2592{
2593 uint32_t *p;
2594
2595 READ_BUF(20);
2596 READ32(cinfo->atomic);
2597 READ64(cinfo->before);
2598 READ64(cinfo->after);
2599 return 0;
2600}
2601
2602static int decode_access(struct xdr_stream *xdr, struct nfs4_accessres *access)
2603{
2604 uint32_t *p;
2605 uint32_t supp, acc;
2606 int status;
2607
2608 status = decode_op_hdr(xdr, OP_ACCESS);
2609 if (status)
2610 return status;
2611 READ_BUF(8);
2612 READ32(supp);
2613 READ32(acc);
2614 access->supported = supp;
2615 access->access = acc;
2616 return 0;
2617}
2618
2619static int decode_close(struct xdr_stream *xdr, struct nfs_closeres *res)
2620{
2621 uint32_t *p;
2622 int status;
2623
2624 status = decode_op_hdr(xdr, OP_CLOSE);
2625 if (status)
2626 return status;
2627 READ_BUF(sizeof(res->stateid.data));
2628 COPYMEM(res->stateid.data, sizeof(res->stateid.data));
2629 return 0;
2630}
2631
2632static int decode_commit(struct xdr_stream *xdr, struct nfs_writeres *res)
2633{
2634 uint32_t *p;
2635 int status;
2636
2637 status = decode_op_hdr(xdr, OP_COMMIT);
2638 if (status)
2639 return status;
2640 READ_BUF(8);
2641 COPYMEM(res->verf->verifier, 8);
2642 return 0;
2643}
2644
2645static int decode_create(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
2646{
2647 uint32_t *p;
2648 uint32_t bmlen;
2649 int status;
2650
2651 status = decode_op_hdr(xdr, OP_CREATE);
2652 if (status)
2653 return status;
2654 if ((status = decode_change_info(xdr, cinfo)))
2655 return status;
2656 READ_BUF(4);
2657 READ32(bmlen);
2658 READ_BUF(bmlen << 2);
2659 return 0;
2660}
2661
2662static int decode_server_caps(struct xdr_stream *xdr, struct nfs4_server_caps_res *res)
2663{
2664 uint32_t *savep;
2665 uint32_t attrlen,
2666 bitmap[2] = {0};
2667 int status;
2668
2669 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
2670 goto xdr_error;
2671 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
2672 goto xdr_error;
2673 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
2674 goto xdr_error;
2675 if ((status = decode_attr_supported(xdr, bitmap, res->attr_bitmask)) != 0)
2676 goto xdr_error;
2677 if ((status = decode_attr_link_support(xdr, bitmap, &res->has_links)) != 0)
2678 goto xdr_error;
2679 if ((status = decode_attr_symlink_support(xdr, bitmap, &res->has_symlinks)) != 0)
2680 goto xdr_error;
2681 if ((status = decode_attr_aclsupport(xdr, bitmap, &res->acl_bitmask)) != 0)
2682 goto xdr_error;
2683 status = verify_attr_len(xdr, savep, attrlen);
2684xdr_error:
2685 if (status != 0)
2686 printk(KERN_NOTICE "%s: xdr error %d!\n", __FUNCTION__, -status);
2687 return status;
2688}
2689
2690static int decode_statfs(struct xdr_stream *xdr, struct nfs_fsstat *fsstat)
2691{
2692 uint32_t *savep;
2693 uint32_t attrlen,
2694 bitmap[2] = {0};
2695 int status;
2696
2697 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
2698 goto xdr_error;
2699 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
2700 goto xdr_error;
2701 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
2702 goto xdr_error;
2703
2704 if ((status = decode_attr_files_avail(xdr, bitmap, &fsstat->afiles)) != 0)
2705 goto xdr_error;
2706 if ((status = decode_attr_files_free(xdr, bitmap, &fsstat->ffiles)) != 0)
2707 goto xdr_error;
2708 if ((status = decode_attr_files_total(xdr, bitmap, &fsstat->tfiles)) != 0)
2709 goto xdr_error;
2710 if ((status = decode_attr_space_avail(xdr, bitmap, &fsstat->abytes)) != 0)
2711 goto xdr_error;
2712 if ((status = decode_attr_space_free(xdr, bitmap, &fsstat->fbytes)) != 0)
2713 goto xdr_error;
2714 if ((status = decode_attr_space_total(xdr, bitmap, &fsstat->tbytes)) != 0)
2715 goto xdr_error;
2716
2717 status = verify_attr_len(xdr, savep, attrlen);
2718xdr_error:
2719 if (status != 0)
2720 printk(KERN_NOTICE "%s: xdr error %d!\n", __FUNCTION__, -status);
2721 return status;
2722}
2723
2724static int decode_pathconf(struct xdr_stream *xdr, struct nfs_pathconf *pathconf)
2725{
2726 uint32_t *savep;
2727 uint32_t attrlen,
2728 bitmap[2] = {0};
2729 int status;
2730
2731 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
2732 goto xdr_error;
2733 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
2734 goto xdr_error;
2735 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
2736 goto xdr_error;
2737
2738 if ((status = decode_attr_maxlink(xdr, bitmap, &pathconf->max_link)) != 0)
2739 goto xdr_error;
2740 if ((status = decode_attr_maxname(xdr, bitmap, &pathconf->max_namelen)) != 0)
2741 goto xdr_error;
2742
2743 status = verify_attr_len(xdr, savep, attrlen);
2744xdr_error:
2745 if (status != 0)
2746 printk(KERN_NOTICE "%s: xdr error %d!\n", __FUNCTION__, -status);
2747 return status;
2748}
2749
2750static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr, const struct nfs_server *server)
2751{
2752 uint32_t *savep;
2753 uint32_t attrlen,
2754 bitmap[2] = {0},
2755 type;
2756 int status, fmode = 0;
2757
2758 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
2759 goto xdr_error;
2760 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
2761 goto xdr_error;
2762
2763 fattr->bitmap[0] = bitmap[0];
2764 fattr->bitmap[1] = bitmap[1];
2765
2766 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
2767 goto xdr_error;
2768
2769
2770 if ((status = decode_attr_type(xdr, bitmap, &type)) != 0)
2771 goto xdr_error;
2772 fattr->type = nfs_type2fmt[type].nfs2type;
2773 fmode = nfs_type2fmt[type].mode;
2774
2775 if ((status = decode_attr_change(xdr, bitmap, &fattr->change_attr)) != 0)
2776 goto xdr_error;
2777 if ((status = decode_attr_size(xdr, bitmap, &fattr->size)) != 0)
2778 goto xdr_error;
2779 if ((status = decode_attr_fsid(xdr, bitmap, &fattr->fsid_u.nfs4)) != 0)
2780 goto xdr_error;
2781 if ((status = decode_attr_fileid(xdr, bitmap, &fattr->fileid)) != 0)
2782 goto xdr_error;
2783 if ((status = decode_attr_mode(xdr, bitmap, &fattr->mode)) != 0)
2784 goto xdr_error;
2785 fattr->mode |= fmode;
2786 if ((status = decode_attr_nlink(xdr, bitmap, &fattr->nlink)) != 0)
2787 goto xdr_error;
2788 if ((status = decode_attr_owner(xdr, bitmap, server->nfs4_state, &fattr->uid)) != 0)
2789 goto xdr_error;
2790 if ((status = decode_attr_group(xdr, bitmap, server->nfs4_state, &fattr->gid)) != 0)
2791 goto xdr_error;
2792 if ((status = decode_attr_rdev(xdr, bitmap, &fattr->rdev)) != 0)
2793 goto xdr_error;
2794 if ((status = decode_attr_space_used(xdr, bitmap, &fattr->du.nfs3.used)) != 0)
2795 goto xdr_error;
2796 if ((status = decode_attr_time_access(xdr, bitmap, &fattr->atime)) != 0)
2797 goto xdr_error;
2798 if ((status = decode_attr_time_metadata(xdr, bitmap, &fattr->ctime)) != 0)
2799 goto xdr_error;
2800 if ((status = decode_attr_time_modify(xdr, bitmap, &fattr->mtime)) != 0)
2801 goto xdr_error;
2802 if ((status = verify_attr_len(xdr, savep, attrlen)) == 0) {
2803 fattr->valid = NFS_ATTR_FATTR | NFS_ATTR_FATTR_V3 | NFS_ATTR_FATTR_V4;
2804 fattr->timestamp = jiffies;
2805 }
2806xdr_error:
2807 if (status != 0)
2808 printk(KERN_NOTICE "%s: xdr error %d!\n", __FUNCTION__, -status);
2809 return status;
2810}
2811
2812
2813static int decode_fsinfo(struct xdr_stream *xdr, struct nfs_fsinfo *fsinfo)
2814{
2815 uint32_t *savep;
2816 uint32_t attrlen, bitmap[2];
2817 int status;
2818
2819 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
2820 goto xdr_error;
2821 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
2822 goto xdr_error;
2823 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
2824 goto xdr_error;
2825
2826 fsinfo->rtmult = fsinfo->wtmult = 512; /* ??? */
2827
2828 if ((status = decode_attr_lease_time(xdr, bitmap, &fsinfo->lease_time)) != 0)
2829 goto xdr_error;
2830 if ((status = decode_attr_maxfilesize(xdr, bitmap, &fsinfo->maxfilesize)) != 0)
2831 goto xdr_error;
2832 if ((status = decode_attr_maxread(xdr, bitmap, &fsinfo->rtmax)) != 0)
2833 goto xdr_error;
2834 fsinfo->rtpref = fsinfo->dtpref = fsinfo->rtmax;
2835 if ((status = decode_attr_maxwrite(xdr, bitmap, &fsinfo->wtmax)) != 0)
2836 goto xdr_error;
2837 fsinfo->wtpref = fsinfo->wtmax;
2838
2839 status = verify_attr_len(xdr, savep, attrlen);
2840xdr_error:
2841 if (status != 0)
2842 printk(KERN_NOTICE "%s: xdr error %d!\n", __FUNCTION__, -status);
2843 return status;
2844}
2845
2846static int decode_getfh(struct xdr_stream *xdr, struct nfs_fh *fh)
2847{
2848 uint32_t *p;
2849 uint32_t len;
2850 int status;
2851
2852 status = decode_op_hdr(xdr, OP_GETFH);
2853 if (status)
2854 return status;
2855 /* Zero handle first to allow comparisons */
2856 memset(fh, 0, sizeof(*fh));
2857
2858 READ_BUF(4);
2859 READ32(len);
2860 if (len > NFS4_FHSIZE)
2861 return -EIO;
2862 fh->size = len;
2863 READ_BUF(len);
2864 COPYMEM(fh->data, len);
2865 return 0;
2866}
2867
2868static int decode_link(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
2869{
2870 int status;
2871
2872 status = decode_op_hdr(xdr, OP_LINK);
2873 if (status)
2874 return status;
2875 return decode_change_info(xdr, cinfo);
2876}
2877
2878/*
2879 * We create the owner, so we know a proper owner.id length is 4.
2880 */
2881static int decode_lock_denied (struct xdr_stream *xdr, struct nfs_lock_denied *denied)
2882{
2883 uint32_t *p;
2884 uint32_t namelen;
2885
2886 READ_BUF(32);
2887 READ64(denied->offset);
2888 READ64(denied->length);
2889 READ32(denied->type);
2890 READ64(denied->owner.clientid);
2891 READ32(namelen);
2892 READ_BUF(namelen);
2893 if (namelen == 4)
2894 READ32(denied->owner.id);
2895 return -NFS4ERR_DENIED;
2896}
2897
2898static int decode_lock(struct xdr_stream *xdr, struct nfs_lockres *res)
2899{
2900 uint32_t *p;
2901 int status;
2902
2903 status = decode_op_hdr(xdr, OP_LOCK);
2904 if (status == 0) {
Trond Myklebust06735b32005-10-18 14:20:15 -07002905 READ_BUF(sizeof(res->u.stateid.data));
2906 COPYMEM(res->u.stateid.data, sizeof(res->u.stateid.data));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002907 } else if (status == -NFS4ERR_DENIED)
2908 return decode_lock_denied(xdr, &res->u.denied);
2909 return status;
2910}
2911
2912static int decode_lockt(struct xdr_stream *xdr, struct nfs_lockres *res)
2913{
2914 int status;
2915 status = decode_op_hdr(xdr, OP_LOCKT);
2916 if (status == -NFS4ERR_DENIED)
2917 return decode_lock_denied(xdr, &res->u.denied);
2918 return status;
2919}
2920
2921static int decode_locku(struct xdr_stream *xdr, struct nfs_lockres *res)
2922{
2923 uint32_t *p;
2924 int status;
2925
2926 status = decode_op_hdr(xdr, OP_LOCKU);
2927 if (status == 0) {
Trond Myklebustfaf5f492005-10-18 14:20:15 -07002928 READ_BUF(sizeof(res->u.stateid.data));
2929 COPYMEM(res->u.stateid.data, sizeof(res->u.stateid.data));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002930 }
2931 return status;
2932}
2933
2934static int decode_lookup(struct xdr_stream *xdr)
2935{
2936 return decode_op_hdr(xdr, OP_LOOKUP);
2937}
2938
2939/* This is too sick! */
2940static int decode_space_limit(struct xdr_stream *xdr, u64 *maxsize)
2941{
2942 uint32_t *p;
2943 uint32_t limit_type, nblocks, blocksize;
2944
2945 READ_BUF(12);
2946 READ32(limit_type);
2947 switch (limit_type) {
2948 case 1:
2949 READ64(*maxsize);
2950 break;
2951 case 2:
2952 READ32(nblocks);
2953 READ32(blocksize);
2954 *maxsize = (uint64_t)nblocks * (uint64_t)blocksize;
2955 }
2956 return 0;
2957}
2958
2959static int decode_delegation(struct xdr_stream *xdr, struct nfs_openres *res)
2960{
2961 uint32_t *p;
2962 uint32_t delegation_type;
2963
2964 READ_BUF(4);
2965 READ32(delegation_type);
2966 if (delegation_type == NFS4_OPEN_DELEGATE_NONE) {
2967 res->delegation_type = 0;
2968 return 0;
2969 }
2970 READ_BUF(20);
2971 COPYMEM(res->delegation.data, sizeof(res->delegation.data));
2972 READ32(res->do_recall);
2973 switch (delegation_type) {
2974 case NFS4_OPEN_DELEGATE_READ:
2975 res->delegation_type = FMODE_READ;
2976 break;
2977 case NFS4_OPEN_DELEGATE_WRITE:
2978 res->delegation_type = FMODE_WRITE|FMODE_READ;
2979 if (decode_space_limit(xdr, &res->maxsize) < 0)
2980 return -EIO;
2981 }
2982 return decode_ace(xdr, NULL, res->server->nfs4_state);
2983}
2984
2985static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res)
2986{
2987 uint32_t *p;
2988 uint32_t bmlen;
2989 int status;
2990
2991 status = decode_op_hdr(xdr, OP_OPEN);
2992 if (status)
2993 return status;
2994 READ_BUF(sizeof(res->stateid.data));
2995 COPYMEM(res->stateid.data, sizeof(res->stateid.data));
2996
2997 decode_change_info(xdr, &res->cinfo);
2998
2999 READ_BUF(8);
3000 READ32(res->rflags);
3001 READ32(bmlen);
3002 if (bmlen > 10)
3003 goto xdr_error;
3004
3005 READ_BUF(bmlen << 2);
3006 p += bmlen;
3007 return decode_delegation(xdr, res);
3008xdr_error:
3009 printk(KERN_NOTICE "%s: xdr error!\n", __FUNCTION__);
3010 return -EIO;
3011}
3012
3013static int decode_open_confirm(struct xdr_stream *xdr, struct nfs_open_confirmres *res)
3014{
3015 uint32_t *p;
3016 int status;
3017
3018 status = decode_op_hdr(xdr, OP_OPEN_CONFIRM);
3019 if (status)
3020 return status;
3021 READ_BUF(sizeof(res->stateid.data));
3022 COPYMEM(res->stateid.data, sizeof(res->stateid.data));
3023 return 0;
3024}
3025
3026static int decode_open_downgrade(struct xdr_stream *xdr, struct nfs_closeres *res)
3027{
3028 uint32_t *p;
3029 int status;
3030
3031 status = decode_op_hdr(xdr, OP_OPEN_DOWNGRADE);
3032 if (status)
3033 return status;
3034 READ_BUF(sizeof(res->stateid.data));
3035 COPYMEM(res->stateid.data, sizeof(res->stateid.data));
3036 return 0;
3037}
3038
3039static int decode_putfh(struct xdr_stream *xdr)
3040{
3041 return decode_op_hdr(xdr, OP_PUTFH);
3042}
3043
3044static int decode_putrootfh(struct xdr_stream *xdr)
3045{
3046 return decode_op_hdr(xdr, OP_PUTROOTFH);
3047}
3048
3049static int decode_read(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs_readres *res)
3050{
3051 struct kvec *iov = req->rq_rcv_buf.head;
3052 uint32_t *p;
3053 uint32_t count, eof, recvd, hdrlen;
3054 int status;
3055
3056 status = decode_op_hdr(xdr, OP_READ);
3057 if (status)
3058 return status;
3059 READ_BUF(8);
3060 READ32(eof);
3061 READ32(count);
3062 hdrlen = (u8 *) p - (u8 *) iov->iov_base;
3063 recvd = req->rq_rcv_buf.len - hdrlen;
3064 if (count > recvd) {
3065 printk(KERN_WARNING "NFS: server cheating in read reply: "
3066 "count %u > recvd %u\n", count, recvd);
3067 count = recvd;
3068 eof = 0;
3069 }
3070 xdr_read_pages(xdr, count);
3071 res->eof = eof;
3072 res->count = count;
3073 return 0;
3074}
3075
3076static int decode_readdir(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs4_readdir_res *readdir)
3077{
3078 struct xdr_buf *rcvbuf = &req->rq_rcv_buf;
3079 struct page *page = *rcvbuf->pages;
3080 struct kvec *iov = rcvbuf->head;
3081 unsigned int nr, pglen = rcvbuf->page_len;
3082 uint32_t *end, *entry, *p, *kaddr;
3083 uint32_t len, attrlen;
3084 int hdrlen, recvd, status;
3085
3086 status = decode_op_hdr(xdr, OP_READDIR);
3087 if (status)
3088 return status;
3089 READ_BUF(8);
3090 COPYMEM(readdir->verifier.data, 8);
Trond Myklebusteadf4592005-06-22 17:16:39 +00003091 dprintk("%s: verifier = 0x%x%x\n",
3092 __FUNCTION__,
3093 ((u32 *)readdir->verifier.data)[0],
3094 ((u32 *)readdir->verifier.data)[1]);
3095
Linus Torvalds1da177e2005-04-16 15:20:36 -07003096
3097 hdrlen = (char *) p - (char *) iov->iov_base;
3098 recvd = rcvbuf->len - hdrlen;
3099 if (pglen > recvd)
3100 pglen = recvd;
3101 xdr_read_pages(xdr, pglen);
3102
3103 BUG_ON(pglen + readdir->pgbase > PAGE_CACHE_SIZE);
3104 kaddr = p = (uint32_t *) kmap_atomic(page, KM_USER0);
3105 end = (uint32_t *) ((char *)p + pglen + readdir->pgbase);
3106 entry = p;
3107 for (nr = 0; *p++; nr++) {
3108 if (p + 3 > end)
3109 goto short_pkt;
Trond Myklebusteadf4592005-06-22 17:16:39 +00003110 dprintk("cookie = %Lu, ", *((unsigned long long *)p));
Linus Torvalds1da177e2005-04-16 15:20:36 -07003111 p += 2; /* cookie */
3112 len = ntohl(*p++); /* filename length */
3113 if (len > NFS4_MAXNAMLEN) {
3114 printk(KERN_WARNING "NFS: giant filename in readdir (len 0x%x)\n", len);
3115 goto err_unmap;
3116 }
Trond Myklebusteadf4592005-06-22 17:16:39 +00003117 dprintk("filename = %*s\n", len, (char *)p);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003118 p += XDR_QUADLEN(len);
3119 if (p + 1 > end)
3120 goto short_pkt;
3121 len = ntohl(*p++); /* bitmap length */
3122 p += len;
3123 if (p + 1 > end)
3124 goto short_pkt;
3125 attrlen = XDR_QUADLEN(ntohl(*p++));
3126 p += attrlen; /* attributes */
3127 if (p + 2 > end)
3128 goto short_pkt;
3129 entry = p;
3130 }
3131 if (!nr && (entry[0] != 0 || entry[1] == 0))
3132 goto short_pkt;
3133out:
3134 kunmap_atomic(kaddr, KM_USER0);
3135 return 0;
3136short_pkt:
Trond Myklebusteadf4592005-06-22 17:16:39 +00003137 dprintk("%s: short packet at entry %d\n", __FUNCTION__, nr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003138 entry[0] = entry[1] = 0;
3139 /* truncate listing ? */
3140 if (!nr) {
3141 printk(KERN_NOTICE "NFS: readdir reply truncated!\n");
3142 entry[1] = 1;
3143 }
3144 goto out;
3145err_unmap:
3146 kunmap_atomic(kaddr, KM_USER0);
3147 return -errno_NFSERR_IO;
3148}
3149
3150static int decode_readlink(struct xdr_stream *xdr, struct rpc_rqst *req)
3151{
3152 struct xdr_buf *rcvbuf = &req->rq_rcv_buf;
3153 struct kvec *iov = rcvbuf->head;
3154 int hdrlen, len, recvd;
3155 uint32_t *p;
3156 char *kaddr;
3157 int status;
3158
3159 status = decode_op_hdr(xdr, OP_READLINK);
3160 if (status)
3161 return status;
3162
3163 /* Convert length of symlink */
3164 READ_BUF(4);
3165 READ32(len);
3166 if (len >= rcvbuf->page_len || len <= 0) {
3167 dprintk(KERN_WARNING "nfs: server returned giant symlink!\n");
3168 return -ENAMETOOLONG;
3169 }
3170 hdrlen = (char *) xdr->p - (char *) iov->iov_base;
3171 recvd = req->rq_rcv_buf.len - hdrlen;
3172 if (recvd < len) {
3173 printk(KERN_WARNING "NFS: server cheating in readlink reply: "
3174 "count %u > recvd %u\n", len, recvd);
3175 return -EIO;
3176 }
3177 xdr_read_pages(xdr, len);
3178 /*
3179 * The XDR encode routine has set things up so that
3180 * the link text will be copied directly into the
3181 * buffer. We just have to do overflow-checking,
3182 * and and null-terminate the text (the VFS expects
3183 * null-termination).
3184 */
3185 kaddr = (char *)kmap_atomic(rcvbuf->pages[0], KM_USER0);
3186 kaddr[len+rcvbuf->page_base] = '\0';
3187 kunmap_atomic(kaddr, KM_USER0);
3188 return 0;
3189}
3190
3191static int decode_remove(struct xdr_stream *xdr, struct nfs4_change_info *cinfo)
3192{
3193 int status;
3194
3195 status = decode_op_hdr(xdr, OP_REMOVE);
3196 if (status)
3197 goto out;
3198 status = decode_change_info(xdr, cinfo);
3199out:
3200 return status;
3201}
3202
3203static int decode_rename(struct xdr_stream *xdr, struct nfs4_change_info *old_cinfo,
3204 struct nfs4_change_info *new_cinfo)
3205{
3206 int status;
3207
3208 status = decode_op_hdr(xdr, OP_RENAME);
3209 if (status)
3210 goto out;
3211 if ((status = decode_change_info(xdr, old_cinfo)))
3212 goto out;
3213 status = decode_change_info(xdr, new_cinfo);
3214out:
3215 return status;
3216}
3217
3218static int decode_renew(struct xdr_stream *xdr)
3219{
3220 return decode_op_hdr(xdr, OP_RENEW);
3221}
3222
J. Bruce Fields029d1052005-06-22 17:16:22 +00003223static int decode_getacl(struct xdr_stream *xdr, struct rpc_rqst *req,
3224 size_t *acl_len)
3225{
3226 uint32_t *savep;
3227 uint32_t attrlen,
3228 bitmap[2] = {0};
3229 struct kvec *iov = req->rq_rcv_buf.head;
3230 int status;
3231
3232 *acl_len = 0;
3233 if ((status = decode_op_hdr(xdr, OP_GETATTR)) != 0)
3234 goto out;
3235 if ((status = decode_attr_bitmap(xdr, bitmap)) != 0)
3236 goto out;
3237 if ((status = decode_attr_length(xdr, &attrlen, &savep)) != 0)
3238 goto out;
3239
3240 if (unlikely(bitmap[0] & (FATTR4_WORD0_ACL - 1U)))
3241 return -EIO;
3242 if (likely(bitmap[0] & FATTR4_WORD0_ACL)) {
3243 int hdrlen, recvd;
3244
3245 /* We ignore &savep and don't do consistency checks on
3246 * the attr length. Let userspace figure it out.... */
3247 hdrlen = (u8 *)xdr->p - (u8 *)iov->iov_base;
3248 recvd = req->rq_rcv_buf.len - hdrlen;
3249 if (attrlen > recvd) {
3250 printk(KERN_WARNING "NFS: server cheating in getattr"
3251 " acl reply: attrlen %u > recvd %u\n",
3252 attrlen, recvd);
3253 return -EINVAL;
3254 }
3255 if (attrlen <= *acl_len)
3256 xdr_read_pages(xdr, attrlen);
3257 *acl_len = attrlen;
3258 }
3259
3260out:
3261 return status;
3262}
3263
Linus Torvalds1da177e2005-04-16 15:20:36 -07003264static int
3265decode_savefh(struct xdr_stream *xdr)
3266{
3267 return decode_op_hdr(xdr, OP_SAVEFH);
3268}
3269
3270static int decode_setattr(struct xdr_stream *xdr, struct nfs_setattrres *res)
3271{
3272 uint32_t *p;
3273 uint32_t bmlen;
3274 int status;
3275
3276
3277 status = decode_op_hdr(xdr, OP_SETATTR);
3278 if (status)
3279 return status;
3280 READ_BUF(4);
3281 READ32(bmlen);
3282 READ_BUF(bmlen << 2);
3283 return 0;
3284}
3285
3286static int decode_setclientid(struct xdr_stream *xdr, struct nfs4_client *clp)
3287{
3288 uint32_t *p;
3289 uint32_t opnum;
3290 int32_t nfserr;
3291
3292 READ_BUF(8);
3293 READ32(opnum);
3294 if (opnum != OP_SETCLIENTID) {
3295 printk(KERN_NOTICE
3296 "nfs4_decode_setclientid: Server returned operation"
3297 " %d\n", opnum);
3298 return -EIO;
3299 }
3300 READ32(nfserr);
3301 if (nfserr == NFS_OK) {
3302 READ_BUF(8 + sizeof(clp->cl_confirm.data));
3303 READ64(clp->cl_clientid);
3304 COPYMEM(clp->cl_confirm.data, sizeof(clp->cl_confirm.data));
3305 } else if (nfserr == NFSERR_CLID_INUSE) {
3306 uint32_t len;
3307
3308 /* skip netid string */
3309 READ_BUF(4);
3310 READ32(len);
3311 READ_BUF(len);
3312
3313 /* skip uaddr string */
3314 READ_BUF(4);
3315 READ32(len);
3316 READ_BUF(len);
3317 return -NFSERR_CLID_INUSE;
3318 } else
3319 return -nfs_stat_to_errno(nfserr);
3320
3321 return 0;
3322}
3323
3324static int decode_setclientid_confirm(struct xdr_stream *xdr)
3325{
3326 return decode_op_hdr(xdr, OP_SETCLIENTID_CONFIRM);
3327}
3328
3329static int decode_write(struct xdr_stream *xdr, struct nfs_writeres *res)
3330{
3331 uint32_t *p;
3332 int status;
3333
3334 status = decode_op_hdr(xdr, OP_WRITE);
3335 if (status)
3336 return status;
3337
3338 READ_BUF(16);
3339 READ32(res->count);
3340 READ32(res->verf->committed);
3341 COPYMEM(res->verf->verifier, 8);
3342 return 0;
3343}
3344
3345static int decode_delegreturn(struct xdr_stream *xdr)
3346{
3347 return decode_op_hdr(xdr, OP_DELEGRETURN);
3348}
3349
3350/*
3351 * Decode OPEN_DOWNGRADE response
3352 */
3353static int nfs4_xdr_dec_open_downgrade(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_closeres *res)
3354{
3355 struct xdr_stream xdr;
3356 struct compound_hdr hdr;
3357 int status;
3358
3359 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3360 status = decode_compound_hdr(&xdr, &hdr);
3361 if (status)
3362 goto out;
3363 status = decode_putfh(&xdr);
3364 if (status)
3365 goto out;
3366 status = decode_open_downgrade(&xdr, res);
3367out:
3368 return status;
3369}
3370
3371/*
3372 * END OF "GENERIC" DECODE ROUTINES.
3373 */
3374
3375/*
3376 * Decode ACCESS response
3377 */
3378static int nfs4_xdr_dec_access(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_accessres *res)
3379{
3380 struct xdr_stream xdr;
3381 struct compound_hdr hdr;
3382 int status;
3383
3384 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3385 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3386 goto out;
3387 if ((status = decode_putfh(&xdr)) == 0)
3388 status = decode_access(&xdr, res);
3389out:
3390 return status;
3391}
3392
3393/*
3394 * Decode LOOKUP response
3395 */
3396static int nfs4_xdr_dec_lookup(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_lookup_res *res)
3397{
3398 struct xdr_stream xdr;
3399 struct compound_hdr hdr;
3400 int status;
3401
3402 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3403 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3404 goto out;
3405 if ((status = decode_putfh(&xdr)) != 0)
3406 goto out;
3407 if ((status = decode_lookup(&xdr)) != 0)
3408 goto out;
3409 if ((status = decode_getfh(&xdr, res->fh)) != 0)
3410 goto out;
3411 status = decode_getfattr(&xdr, res->fattr, res->server);
3412out:
3413 return status;
3414}
3415
3416/*
3417 * Decode LOOKUP_ROOT response
3418 */
3419static int nfs4_xdr_dec_lookup_root(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_lookup_res *res)
3420{
3421 struct xdr_stream xdr;
3422 struct compound_hdr hdr;
3423 int status;
3424
3425 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3426 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3427 goto out;
3428 if ((status = decode_putrootfh(&xdr)) != 0)
3429 goto out;
3430 if ((status = decode_getfh(&xdr, res->fh)) == 0)
3431 status = decode_getfattr(&xdr, res->fattr, res->server);
3432out:
3433 return status;
3434}
3435
3436/*
3437 * Decode REMOVE response
3438 */
3439static int nfs4_xdr_dec_remove(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_change_info *cinfo)
3440{
3441 struct xdr_stream xdr;
3442 struct compound_hdr hdr;
3443 int status;
3444
3445 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3446 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3447 goto out;
3448 if ((status = decode_putfh(&xdr)) == 0)
3449 status = decode_remove(&xdr, cinfo);
3450out:
3451 return status;
3452}
3453
3454/*
3455 * Decode RENAME response
3456 */
3457static int nfs4_xdr_dec_rename(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_rename_res *res)
3458{
3459 struct xdr_stream xdr;
3460 struct compound_hdr hdr;
3461 int status;
3462
3463 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3464 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3465 goto out;
3466 if ((status = decode_putfh(&xdr)) != 0)
3467 goto out;
3468 if ((status = decode_savefh(&xdr)) != 0)
3469 goto out;
3470 if ((status = decode_putfh(&xdr)) != 0)
3471 goto out;
3472 status = decode_rename(&xdr, &res->old_cinfo, &res->new_cinfo);
3473out:
3474 return status;
3475}
3476
3477/*
3478 * Decode LINK response
3479 */
3480static int nfs4_xdr_dec_link(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_change_info *cinfo)
3481{
3482 struct xdr_stream xdr;
3483 struct compound_hdr hdr;
3484 int status;
3485
3486 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3487 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3488 goto out;
3489 if ((status = decode_putfh(&xdr)) != 0)
3490 goto out;
3491 if ((status = decode_savefh(&xdr)) != 0)
3492 goto out;
3493 if ((status = decode_putfh(&xdr)) != 0)
3494 goto out;
3495 status = decode_link(&xdr, cinfo);
3496out:
3497 return status;
3498}
3499
3500/*
3501 * Decode CREATE response
3502 */
3503static int nfs4_xdr_dec_create(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_create_res *res)
3504{
3505 struct xdr_stream xdr;
3506 struct compound_hdr hdr;
3507 int status;
3508
3509 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3510 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3511 goto out;
3512 if ((status = decode_putfh(&xdr)) != 0)
3513 goto out;
3514 if ((status = decode_create(&xdr,&res->dir_cinfo)) != 0)
3515 goto out;
3516 if ((status = decode_getfh(&xdr, res->fh)) != 0)
3517 goto out;
3518 status = decode_getfattr(&xdr, res->fattr, res->server);
3519 if (status == NFS4ERR_DELAY)
3520 status = 0;
3521out:
3522 return status;
3523}
3524
3525/*
3526 * Decode SYMLINK response
3527 */
3528static int nfs4_xdr_dec_symlink(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_create_res *res)
3529{
3530 return nfs4_xdr_dec_create(rqstp, p, res);
3531}
3532
3533/*
3534 * Decode GETATTR response
3535 */
3536static int nfs4_xdr_dec_getattr(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_getattr_res *res)
3537{
3538 struct xdr_stream xdr;
3539 struct compound_hdr hdr;
3540 int status;
3541
3542 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3543 status = decode_compound_hdr(&xdr, &hdr);
3544 if (status)
3545 goto out;
3546 status = decode_putfh(&xdr);
3547 if (status)
3548 goto out;
3549 status = decode_getfattr(&xdr, res->fattr, res->server);
3550out:
3551 return status;
3552
3553}
3554
J. Bruce Fields23ec6962005-06-22 17:16:22 +00003555/*
3556 * Encode an SETACL request
3557 */
3558static int
3559nfs4_xdr_enc_setacl(struct rpc_rqst *req, uint32_t *p, struct nfs_setaclargs *args)
3560{
3561 struct xdr_stream xdr;
3562 struct compound_hdr hdr = {
3563 .nops = 2,
3564 };
3565 int status;
3566
3567 xdr_init_encode(&xdr, &req->rq_snd_buf, p);
3568 encode_compound_hdr(&xdr, &hdr);
3569 status = encode_putfh(&xdr, args->fh);
3570 if (status)
3571 goto out;
3572 status = encode_setacl(&xdr, args);
3573out:
3574 return status;
3575}
3576/*
3577 * Decode SETACL response
3578 */
3579static int
3580nfs4_xdr_dec_setacl(struct rpc_rqst *rqstp, uint32_t *p, void *res)
3581{
3582 struct xdr_stream xdr;
3583 struct compound_hdr hdr;
3584 int status;
3585
3586 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3587 status = decode_compound_hdr(&xdr, &hdr);
3588 if (status)
3589 goto out;
3590 status = decode_putfh(&xdr);
3591 if (status)
3592 goto out;
3593 status = decode_setattr(&xdr, res);
3594out:
3595 return status;
3596}
Linus Torvalds1da177e2005-04-16 15:20:36 -07003597
3598/*
J. Bruce Fields029d1052005-06-22 17:16:22 +00003599 * Decode GETACL response
3600 */
3601static int
3602nfs4_xdr_dec_getacl(struct rpc_rqst *rqstp, uint32_t *p, size_t *acl_len)
3603{
3604 struct xdr_stream xdr;
3605 struct compound_hdr hdr;
3606 int status;
3607
3608 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3609 status = decode_compound_hdr(&xdr, &hdr);
3610 if (status)
3611 goto out;
3612 status = decode_putfh(&xdr);
3613 if (status)
3614 goto out;
3615 status = decode_getacl(&xdr, rqstp, acl_len);
3616
3617out:
3618 return status;
3619}
3620
3621/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07003622 * Decode CLOSE response
3623 */
3624static int nfs4_xdr_dec_close(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_closeres *res)
3625{
3626 struct xdr_stream xdr;
3627 struct compound_hdr hdr;
3628 int status;
3629
3630 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3631 status = decode_compound_hdr(&xdr, &hdr);
3632 if (status)
3633 goto out;
3634 status = decode_putfh(&xdr);
3635 if (status)
3636 goto out;
3637 status = decode_close(&xdr, res);
3638out:
3639 return status;
3640}
3641
3642/*
3643 * Decode OPEN response
3644 */
3645static int nfs4_xdr_dec_open(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_openres *res)
3646{
3647 struct xdr_stream xdr;
3648 struct compound_hdr hdr;
3649 int status;
3650
3651 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3652 status = decode_compound_hdr(&xdr, &hdr);
3653 if (status)
3654 goto out;
3655 status = decode_putfh(&xdr);
3656 if (status)
3657 goto out;
3658 status = decode_open(&xdr, res);
3659 if (status)
3660 goto out;
3661 status = decode_getfh(&xdr, &res->fh);
3662 if (status)
3663 goto out;
3664 status = decode_getfattr(&xdr, res->f_attr, res->server);
3665 if (status == NFS4ERR_DELAY)
3666 status = 0;
3667out:
3668 return status;
3669}
3670
3671/*
3672 * Decode OPEN_CONFIRM response
3673 */
3674static int nfs4_xdr_dec_open_confirm(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_open_confirmres *res)
3675{
3676 struct xdr_stream xdr;
3677 struct compound_hdr hdr;
3678 int status;
3679
3680 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3681 status = decode_compound_hdr(&xdr, &hdr);
3682 if (status)
3683 goto out;
3684 status = decode_putfh(&xdr);
3685 if (status)
3686 goto out;
3687 status = decode_open_confirm(&xdr, res);
3688out:
3689 return status;
3690}
3691
3692/*
3693 * Decode OPEN response
3694 */
3695static int nfs4_xdr_dec_open_noattr(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_openres *res)
3696{
3697 struct xdr_stream xdr;
3698 struct compound_hdr hdr;
3699 int status;
3700
3701 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3702 status = decode_compound_hdr(&xdr, &hdr);
3703 if (status)
3704 goto out;
3705 status = decode_putfh(&xdr);
3706 if (status)
3707 goto out;
3708 status = decode_open(&xdr, res);
3709out:
3710 return status;
3711}
3712
3713/*
3714 * Decode SETATTR response
3715 */
3716static int nfs4_xdr_dec_setattr(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_setattrres *res)
3717{
3718 struct xdr_stream xdr;
3719 struct compound_hdr hdr;
3720 int status;
3721
3722 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3723 status = decode_compound_hdr(&xdr, &hdr);
3724 if (status)
3725 goto out;
3726 status = decode_putfh(&xdr);
3727 if (status)
3728 goto out;
3729 status = decode_setattr(&xdr, res);
3730 if (status)
3731 goto out;
3732 status = decode_getfattr(&xdr, res->fattr, res->server);
3733 if (status == NFS4ERR_DELAY)
3734 status = 0;
3735out:
3736 return status;
3737}
3738
3739/*
3740 * Decode LOCK response
3741 */
3742static int nfs4_xdr_dec_lock(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_lockres *res)
3743{
3744 struct xdr_stream xdr;
3745 struct compound_hdr hdr;
3746 int status;
3747
3748 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3749 status = decode_compound_hdr(&xdr, &hdr);
3750 if (status)
3751 goto out;
3752 status = decode_putfh(&xdr);
3753 if (status)
3754 goto out;
3755 status = decode_lock(&xdr, res);
3756out:
3757 return status;
3758}
3759
3760/*
3761 * Decode LOCKT response
3762 */
3763static int nfs4_xdr_dec_lockt(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_lockres *res)
3764{
3765 struct xdr_stream xdr;
3766 struct compound_hdr hdr;
3767 int status;
3768
3769 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3770 status = decode_compound_hdr(&xdr, &hdr);
3771 if (status)
3772 goto out;
3773 status = decode_putfh(&xdr);
3774 if (status)
3775 goto out;
3776 status = decode_lockt(&xdr, res);
3777out:
3778 return status;
3779}
3780
3781/*
3782 * Decode LOCKU response
3783 */
3784static int nfs4_xdr_dec_locku(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_lockres *res)
3785{
3786 struct xdr_stream xdr;
3787 struct compound_hdr hdr;
3788 int status;
3789
3790 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3791 status = decode_compound_hdr(&xdr, &hdr);
3792 if (status)
3793 goto out;
3794 status = decode_putfh(&xdr);
3795 if (status)
3796 goto out;
3797 status = decode_locku(&xdr, res);
3798out:
3799 return status;
3800}
3801
3802/*
3803 * Decode READLINK response
3804 */
3805static int nfs4_xdr_dec_readlink(struct rpc_rqst *rqstp, uint32_t *p, void *res)
3806{
3807 struct xdr_stream xdr;
3808 struct compound_hdr hdr;
3809 int status;
3810
3811 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3812 status = decode_compound_hdr(&xdr, &hdr);
3813 if (status)
3814 goto out;
3815 status = decode_putfh(&xdr);
3816 if (status)
3817 goto out;
3818 status = decode_readlink(&xdr, rqstp);
3819out:
3820 return status;
3821}
3822
3823/*
3824 * Decode READDIR response
3825 */
3826static int nfs4_xdr_dec_readdir(struct rpc_rqst *rqstp, uint32_t *p, struct nfs4_readdir_res *res)
3827{
3828 struct xdr_stream xdr;
3829 struct compound_hdr hdr;
3830 int status;
3831
3832 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3833 status = decode_compound_hdr(&xdr, &hdr);
3834 if (status)
3835 goto out;
3836 status = decode_putfh(&xdr);
3837 if (status)
3838 goto out;
3839 status = decode_readdir(&xdr, rqstp, res);
3840out:
3841 return status;
3842}
3843
3844/*
3845 * Decode Read response
3846 */
3847static int nfs4_xdr_dec_read(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_readres *res)
3848{
3849 struct xdr_stream xdr;
3850 struct compound_hdr hdr;
3851 int status;
3852
3853 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3854 status = decode_compound_hdr(&xdr, &hdr);
3855 if (status)
3856 goto out;
3857 status = decode_putfh(&xdr);
3858 if (status)
3859 goto out;
3860 status = decode_read(&xdr, rqstp, res);
3861 if (!status)
3862 status = res->count;
3863out:
3864 return status;
3865}
3866
3867/*
3868 * Decode WRITE response
3869 */
3870static int nfs4_xdr_dec_write(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_writeres *res)
3871{
3872 struct xdr_stream xdr;
3873 struct compound_hdr hdr;
3874 int status;
3875
3876 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3877 status = decode_compound_hdr(&xdr, &hdr);
3878 if (status)
3879 goto out;
3880 status = decode_putfh(&xdr);
3881 if (status)
3882 goto out;
3883 status = decode_write(&xdr, res);
3884 if (!status)
3885 status = res->count;
3886out:
3887 return status;
3888}
3889
3890/*
3891 * Decode COMMIT response
3892 */
3893static int nfs4_xdr_dec_commit(struct rpc_rqst *rqstp, uint32_t *p, struct nfs_writeres *res)
3894{
3895 struct xdr_stream xdr;
3896 struct compound_hdr hdr;
3897 int status;
3898
3899 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3900 status = decode_compound_hdr(&xdr, &hdr);
3901 if (status)
3902 goto out;
3903 status = decode_putfh(&xdr);
3904 if (status)
3905 goto out;
3906 status = decode_commit(&xdr, res);
3907out:
3908 return status;
3909}
3910
3911/*
3912 * FSINFO request
3913 */
3914static int nfs4_xdr_dec_fsinfo(struct rpc_rqst *req, uint32_t *p, struct nfs_fsinfo *fsinfo)
3915{
3916 struct xdr_stream xdr;
3917 struct compound_hdr hdr;
3918 int status;
3919
3920 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
3921 status = decode_compound_hdr(&xdr, &hdr);
3922 if (!status)
3923 status = decode_putfh(&xdr);
3924 if (!status)
3925 status = decode_fsinfo(&xdr, fsinfo);
3926 if (!status)
3927 status = -nfs_stat_to_errno(hdr.status);
3928 return status;
3929}
3930
3931/*
3932 * PATHCONF request
3933 */
3934static int nfs4_xdr_dec_pathconf(struct rpc_rqst *req, uint32_t *p, struct nfs_pathconf *pathconf)
3935{
3936 struct xdr_stream xdr;
3937 struct compound_hdr hdr;
3938 int status;
3939
3940 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
3941 status = decode_compound_hdr(&xdr, &hdr);
3942 if (!status)
3943 status = decode_putfh(&xdr);
3944 if (!status)
3945 status = decode_pathconf(&xdr, pathconf);
3946 return status;
3947}
3948
3949/*
3950 * STATFS request
3951 */
3952static int nfs4_xdr_dec_statfs(struct rpc_rqst *req, uint32_t *p, struct nfs_fsstat *fsstat)
3953{
3954 struct xdr_stream xdr;
3955 struct compound_hdr hdr;
3956 int status;
3957
3958 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
3959 status = decode_compound_hdr(&xdr, &hdr);
3960 if (!status)
3961 status = decode_putfh(&xdr);
3962 if (!status)
3963 status = decode_statfs(&xdr, fsstat);
3964 return status;
3965}
3966
3967/*
3968 * GETATTR_BITMAP request
3969 */
3970static int nfs4_xdr_dec_server_caps(struct rpc_rqst *req, uint32_t *p, struct nfs4_server_caps_res *res)
3971{
3972 struct xdr_stream xdr;
3973 struct compound_hdr hdr;
3974 int status;
3975
3976 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
3977 if ((status = decode_compound_hdr(&xdr, &hdr)) != 0)
3978 goto out;
3979 if ((status = decode_putfh(&xdr)) != 0)
3980 goto out;
3981 status = decode_server_caps(&xdr, res);
3982out:
3983 return status;
3984}
3985
3986/*
3987 * Decode RENEW response
3988 */
3989static int nfs4_xdr_dec_renew(struct rpc_rqst *rqstp, uint32_t *p, void *dummy)
3990{
3991 struct xdr_stream xdr;
3992 struct compound_hdr hdr;
3993 int status;
3994
3995 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
3996 status = decode_compound_hdr(&xdr, &hdr);
3997 if (!status)
3998 status = decode_renew(&xdr);
3999 return status;
4000}
4001
4002/*
4003 * a SETCLIENTID request
4004 */
4005static int nfs4_xdr_dec_setclientid(struct rpc_rqst *req, uint32_t *p,
4006 struct nfs4_client *clp)
4007{
4008 struct xdr_stream xdr;
4009 struct compound_hdr hdr;
4010 int status;
4011
4012 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
4013 status = decode_compound_hdr(&xdr, &hdr);
4014 if (!status)
4015 status = decode_setclientid(&xdr, clp);
4016 if (!status)
4017 status = -nfs_stat_to_errno(hdr.status);
4018 return status;
4019}
4020
4021/*
4022 * a SETCLIENTID_CONFIRM request
4023 */
4024static int nfs4_xdr_dec_setclientid_confirm(struct rpc_rqst *req, uint32_t *p, struct nfs_fsinfo *fsinfo)
4025{
4026 struct xdr_stream xdr;
4027 struct compound_hdr hdr;
4028 int status;
4029
4030 xdr_init_decode(&xdr, &req->rq_rcv_buf, p);
4031 status = decode_compound_hdr(&xdr, &hdr);
4032 if (!status)
4033 status = decode_setclientid_confirm(&xdr);
4034 if (!status)
4035 status = decode_putrootfh(&xdr);
4036 if (!status)
4037 status = decode_fsinfo(&xdr, fsinfo);
4038 if (!status)
4039 status = -nfs_stat_to_errno(hdr.status);
4040 return status;
4041}
4042
4043/*
4044 * DELEGRETURN request
4045 */
4046static int nfs4_xdr_dec_delegreturn(struct rpc_rqst *rqstp, uint32_t *p, void *dummy)
4047{
4048 struct xdr_stream xdr;
4049 struct compound_hdr hdr;
4050 int status;
4051
4052 xdr_init_decode(&xdr, &rqstp->rq_rcv_buf, p);
4053 status = decode_compound_hdr(&xdr, &hdr);
4054 if (status == 0) {
4055 status = decode_putfh(&xdr);
4056 if (status == 0)
4057 status = decode_delegreturn(&xdr);
4058 }
4059 return status;
4060}
4061
4062uint32_t *nfs4_decode_dirent(uint32_t *p, struct nfs_entry *entry, int plus)
4063{
4064 uint32_t bitmap[2] = {0};
4065 uint32_t len;
4066
4067 if (!*p++) {
4068 if (!*p)
4069 return ERR_PTR(-EAGAIN);
4070 entry->eof = 1;
4071 return ERR_PTR(-EBADCOOKIE);
4072 }
4073
4074 entry->prev_cookie = entry->cookie;
4075 p = xdr_decode_hyper(p, &entry->cookie);
4076 entry->len = ntohl(*p++);
4077 entry->name = (const char *) p;
4078 p += XDR_QUADLEN(entry->len);
4079
4080 /*
4081 * In case the server doesn't return an inode number,
4082 * we fake one here. (We don't use inode number 0,
4083 * since glibc seems to choke on it...)
4084 */
4085 entry->ino = 1;
4086
4087 len = ntohl(*p++); /* bitmap length */
4088 if (len-- > 0) {
4089 bitmap[0] = ntohl(*p++);
4090 if (len-- > 0) {
4091 bitmap[1] = ntohl(*p++);
4092 p += len;
4093 }
4094 }
4095 len = XDR_QUADLEN(ntohl(*p++)); /* attribute buffer length */
4096 if (len > 0) {
Manoj Naik97d312d2005-06-22 17:16:39 +00004097 if (bitmap[0] & FATTR4_WORD0_RDATTR_ERROR) {
4098 bitmap[0] &= ~FATTR4_WORD0_RDATTR_ERROR;
4099 /* Ignore the return value of rdattr_error for now */
4100 p++;
4101 len--;
4102 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07004103 if (bitmap[0] == 0 && bitmap[1] == FATTR4_WORD1_MOUNTED_ON_FILEID)
4104 xdr_decode_hyper(p, &entry->ino);
4105 else if (bitmap[0] == FATTR4_WORD0_FILEID)
4106 xdr_decode_hyper(p, &entry->ino);
4107 p += len;
4108 }
4109
4110 entry->eof = !p[0] && p[1];
4111 return p;
4112}
4113
4114/*
4115 * We need to translate between nfs status return values and
4116 * the local errno values which may not be the same.
4117 */
4118static struct {
4119 int stat;
4120 int errno;
4121} nfs_errtbl[] = {
4122 { NFS4_OK, 0 },
4123 { NFS4ERR_PERM, EPERM },
4124 { NFS4ERR_NOENT, ENOENT },
4125 { NFS4ERR_IO, errno_NFSERR_IO },
4126 { NFS4ERR_NXIO, ENXIO },
4127 { NFS4ERR_ACCESS, EACCES },
4128 { NFS4ERR_EXIST, EEXIST },
4129 { NFS4ERR_XDEV, EXDEV },
4130 { NFS4ERR_NOTDIR, ENOTDIR },
4131 { NFS4ERR_ISDIR, EISDIR },
4132 { NFS4ERR_INVAL, EINVAL },
4133 { NFS4ERR_FBIG, EFBIG },
4134 { NFS4ERR_NOSPC, ENOSPC },
4135 { NFS4ERR_ROFS, EROFS },
4136 { NFS4ERR_MLINK, EMLINK },
4137 { NFS4ERR_NAMETOOLONG, ENAMETOOLONG },
4138 { NFS4ERR_NOTEMPTY, ENOTEMPTY },
4139 { NFS4ERR_DQUOT, EDQUOT },
4140 { NFS4ERR_STALE, ESTALE },
4141 { NFS4ERR_BADHANDLE, EBADHANDLE },
Manoj Naik6ebf3652005-06-22 17:16:39 +00004142 { NFS4ERR_BADOWNER, EINVAL },
4143 { NFS4ERR_BADNAME, EINVAL },
Linus Torvalds1da177e2005-04-16 15:20:36 -07004144 { NFS4ERR_BAD_COOKIE, EBADCOOKIE },
4145 { NFS4ERR_NOTSUPP, ENOTSUPP },
4146 { NFS4ERR_TOOSMALL, ETOOSMALL },
4147 { NFS4ERR_SERVERFAULT, ESERVERFAULT },
4148 { NFS4ERR_BADTYPE, EBADTYPE },
4149 { NFS4ERR_LOCKED, EAGAIN },
4150 { NFS4ERR_RESOURCE, EREMOTEIO },
4151 { NFS4ERR_SYMLINK, ELOOP },
4152 { NFS4ERR_OP_ILLEGAL, EOPNOTSUPP },
4153 { NFS4ERR_DEADLOCK, EDEADLK },
4154 { NFS4ERR_WRONGSEC, EPERM }, /* FIXME: this needs
4155 * to be handled by a
4156 * middle-layer.
4157 */
4158 { -1, EIO }
4159};
4160
4161/*
4162 * Convert an NFS error code to a local one.
4163 * This one is used jointly by NFSv2 and NFSv3.
4164 */
4165static int
4166nfs_stat_to_errno(int stat)
4167{
4168 int i;
4169 for (i = 0; nfs_errtbl[i].stat != -1; i++) {
4170 if (nfs_errtbl[i].stat == stat)
4171 return nfs_errtbl[i].errno;
4172 }
4173 if (stat <= 10000 || stat > 10100) {
4174 /* The server is looney tunes. */
4175 return ESERVERFAULT;
4176 }
4177 /* If we cannot translate the error, the recovery routines should
4178 * handle it.
4179 * Note: remaining NFSv4 error codes have values > 10000, so should
4180 * not conflict with native Linux error codes.
4181 */
4182 return stat;
4183}
4184
4185#ifndef MAX
4186# define MAX(a, b) (((a) > (b))? (a) : (b))
4187#endif
4188
4189#define PROC(proc, argtype, restype) \
4190[NFSPROC4_CLNT_##proc] = { \
4191 .p_proc = NFSPROC4_COMPOUND, \
4192 .p_encode = (kxdrproc_t) nfs4_xdr_##argtype, \
4193 .p_decode = (kxdrproc_t) nfs4_xdr_##restype, \
4194 .p_bufsiz = MAX(NFS4_##argtype##_sz,NFS4_##restype##_sz) << 2, \
4195 }
4196
4197struct rpc_procinfo nfs4_procedures[] = {
4198 PROC(READ, enc_read, dec_read),
4199 PROC(WRITE, enc_write, dec_write),
4200 PROC(COMMIT, enc_commit, dec_commit),
4201 PROC(OPEN, enc_open, dec_open),
4202 PROC(OPEN_CONFIRM, enc_open_confirm, dec_open_confirm),
4203 PROC(OPEN_NOATTR, enc_open_noattr, dec_open_noattr),
4204 PROC(OPEN_DOWNGRADE, enc_open_downgrade, dec_open_downgrade),
4205 PROC(CLOSE, enc_close, dec_close),
4206 PROC(SETATTR, enc_setattr, dec_setattr),
4207 PROC(FSINFO, enc_fsinfo, dec_fsinfo),
4208 PROC(RENEW, enc_renew, dec_renew),
4209 PROC(SETCLIENTID, enc_setclientid, dec_setclientid),
4210 PROC(SETCLIENTID_CONFIRM, enc_setclientid_confirm, dec_setclientid_confirm),
4211 PROC(LOCK, enc_lock, dec_lock),
4212 PROC(LOCKT, enc_lockt, dec_lockt),
4213 PROC(LOCKU, enc_locku, dec_locku),
4214 PROC(ACCESS, enc_access, dec_access),
4215 PROC(GETATTR, enc_getattr, dec_getattr),
4216 PROC(LOOKUP, enc_lookup, dec_lookup),
4217 PROC(LOOKUP_ROOT, enc_lookup_root, dec_lookup_root),
4218 PROC(REMOVE, enc_remove, dec_remove),
4219 PROC(RENAME, enc_rename, dec_rename),
4220 PROC(LINK, enc_link, dec_link),
4221 PROC(SYMLINK, enc_symlink, dec_symlink),
4222 PROC(CREATE, enc_create, dec_create),
4223 PROC(PATHCONF, enc_pathconf, dec_pathconf),
4224 PROC(STATFS, enc_statfs, dec_statfs),
4225 PROC(READLINK, enc_readlink, dec_readlink),
4226 PROC(READDIR, enc_readdir, dec_readdir),
4227 PROC(SERVER_CAPS, enc_server_caps, dec_server_caps),
4228 PROC(DELEGRETURN, enc_delegreturn, dec_delegreturn),
J. Bruce Fields029d1052005-06-22 17:16:22 +00004229 PROC(GETACL, enc_getacl, dec_getacl),
J. Bruce Fields23ec6962005-06-22 17:16:22 +00004230 PROC(SETACL, enc_setacl, dec_setacl),
Linus Torvalds1da177e2005-04-16 15:20:36 -07004231};
4232
4233struct rpc_version nfs_version4 = {
4234 .number = 4,
4235 .nrprocs = sizeof(nfs4_procedures)/sizeof(nfs4_procedures[0]),
4236 .procs = nfs4_procedures
4237};
4238
4239/*
4240 * Local variables:
4241 * c-basic-offset: 8
4242 * End:
4243 */