~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/fs/nfsd/nfs3proc.c

Version: ~ [ linux-6.11.5 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.58 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.114 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.169 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.228 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.284 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.322 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  1 // SPDX-License-Identifier: GPL-2.0
  2 /*
  3  * Process version 3 NFS requests.
  4  *
  5  * Copyright (C) 1996, 1997, 1998 Olaf Kirch <okir@monad.swb.de>
  6  */
  7 
  8 #include <linux/fs.h>
  9 #include <linux/ext2_fs.h>
 10 #include <linux/magic.h>
 11 #include <linux/namei.h>
 12 
 13 #include "cache.h"
 14 #include "xdr3.h"
 15 #include "vfs.h"
 16 #include "filecache.h"
 17 
 18 #define NFSDDBG_FACILITY                NFSDDBG_PROC
 19 
 20 static int      nfs3_ftypes[] = {
 21         0,                      /* NF3NON */
 22         S_IFREG,                /* NF3REG */
 23         S_IFDIR,                /* NF3DIR */
 24         S_IFBLK,                /* NF3BLK */
 25         S_IFCHR,                /* NF3CHR */
 26         S_IFLNK,                /* NF3LNK */
 27         S_IFSOCK,               /* NF3SOCK */
 28         S_IFIFO,                /* NF3FIFO */
 29 };
 30 
 31 /*
 32  * NULL call.
 33  */
 34 static __be32
 35 nfsd3_proc_null(struct svc_rqst *rqstp)
 36 {
 37         return rpc_success;
 38 }
 39 
 40 /*
 41  * Get a file's attributes
 42  */
 43 static __be32
 44 nfsd3_proc_getattr(struct svc_rqst *rqstp)
 45 {
 46         struct nfsd_fhandle *argp = rqstp->rq_argp;
 47         struct nfsd3_attrstat *resp = rqstp->rq_resp;
 48 
 49         dprintk("nfsd: GETATTR(3)  %s\n",
 50                 SVCFH_fmt(&argp->fh));
 51 
 52         fh_copy(&resp->fh, &argp->fh);
 53         resp->status = fh_verify(rqstp, &resp->fh, 0,
 54                                  NFSD_MAY_NOP | NFSD_MAY_BYPASS_GSS_ON_ROOT);
 55         if (resp->status != nfs_ok)
 56                 goto out;
 57 
 58         resp->status = fh_getattr(&resp->fh, &resp->stat);
 59 out:
 60         return rpc_success;
 61 }
 62 
 63 /*
 64  * Set a file's attributes
 65  */
 66 static __be32
 67 nfsd3_proc_setattr(struct svc_rqst *rqstp)
 68 {
 69         struct nfsd3_sattrargs *argp = rqstp->rq_argp;
 70         struct nfsd3_attrstat *resp = rqstp->rq_resp;
 71         struct nfsd_attrs attrs = {
 72                 .na_iattr       = &argp->attrs,
 73         };
 74         const struct timespec64 *guardtime = NULL;
 75 
 76         dprintk("nfsd: SETATTR(3)  %s\n",
 77                                 SVCFH_fmt(&argp->fh));
 78 
 79         fh_copy(&resp->fh, &argp->fh);
 80         if (argp->check_guard)
 81                 guardtime = &argp->guardtime;
 82         resp->status = nfsd_setattr(rqstp, &resp->fh, &attrs, guardtime);
 83         return rpc_success;
 84 }
 85 
 86 /*
 87  * Look up a path name component
 88  */
 89 static __be32
 90 nfsd3_proc_lookup(struct svc_rqst *rqstp)
 91 {
 92         struct nfsd3_diropargs *argp = rqstp->rq_argp;
 93         struct nfsd3_diropres  *resp = rqstp->rq_resp;
 94 
 95         dprintk("nfsd: LOOKUP(3)   %s %.*s\n",
 96                                 SVCFH_fmt(&argp->fh),
 97                                 argp->len,
 98                                 argp->name);
 99 
100         fh_copy(&resp->dirfh, &argp->fh);
101         fh_init(&resp->fh, NFS3_FHSIZE);
102 
103         resp->status = nfsd_lookup(rqstp, &resp->dirfh,
104                                    argp->name, argp->len,
105                                    &resp->fh);
106         return rpc_success;
107 }
108 
109 /*
110  * Check file access
111  */
112 static __be32
113 nfsd3_proc_access(struct svc_rqst *rqstp)
114 {
115         struct nfsd3_accessargs *argp = rqstp->rq_argp;
116         struct nfsd3_accessres *resp = rqstp->rq_resp;
117 
118         dprintk("nfsd: ACCESS(3)   %s 0x%x\n",
119                                 SVCFH_fmt(&argp->fh),
120                                 argp->access);
121 
122         fh_copy(&resp->fh, &argp->fh);
123         resp->access = argp->access;
124         resp->status = nfsd_access(rqstp, &resp->fh, &resp->access, NULL);
125         return rpc_success;
126 }
127 
128 /*
129  * Read a symlink.
130  */
131 static __be32
132 nfsd3_proc_readlink(struct svc_rqst *rqstp)
133 {
134         struct nfsd_fhandle *argp = rqstp->rq_argp;
135         struct nfsd3_readlinkres *resp = rqstp->rq_resp;
136 
137         dprintk("nfsd: READLINK(3) %s\n", SVCFH_fmt(&argp->fh));
138 
139         /* Read the symlink. */
140         fh_copy(&resp->fh, &argp->fh);
141         resp->len = NFS3_MAXPATHLEN;
142         resp->pages = rqstp->rq_next_page++;
143         resp->status = nfsd_readlink(rqstp, &resp->fh,
144                                      page_address(*resp->pages), &resp->len);
145         return rpc_success;
146 }
147 
148 /*
149  * Read a portion of a file.
150  */
151 static __be32
152 nfsd3_proc_read(struct svc_rqst *rqstp)
153 {
154         struct nfsd3_readargs *argp = rqstp->rq_argp;
155         struct nfsd3_readres *resp = rqstp->rq_resp;
156 
157         dprintk("nfsd: READ(3) %s %lu bytes at %Lu\n",
158                                 SVCFH_fmt(&argp->fh),
159                                 (unsigned long) argp->count,
160                                 (unsigned long long) argp->offset);
161 
162         argp->count = min_t(u32, argp->count, svc_max_payload(rqstp));
163         argp->count = min_t(u32, argp->count, rqstp->rq_res.buflen);
164         if (argp->offset > (u64)OFFSET_MAX)
165                 argp->offset = (u64)OFFSET_MAX;
166         if (argp->offset + argp->count > (u64)OFFSET_MAX)
167                 argp->count = (u64)OFFSET_MAX - argp->offset;
168 
169         resp->pages = rqstp->rq_next_page;
170 
171         /* Obtain buffer pointer for payload.
172          * 1 (status) + 22 (post_op_attr) + 1 (count) + 1 (eof)
173          * + 1 (xdr opaque byte count) = 26
174          */
175         resp->count = argp->count;
176         svc_reserve_auth(rqstp, ((1 + NFS3_POST_OP_ATTR_WORDS + 3) << 2) +
177                          resp->count + 4);
178 
179         fh_copy(&resp->fh, &argp->fh);
180         resp->status = nfsd_read(rqstp, &resp->fh, argp->offset,
181                                  &resp->count, &resp->eof);
182         return rpc_success;
183 }
184 
185 /*
186  * Write data to a file
187  */
188 static __be32
189 nfsd3_proc_write(struct svc_rqst *rqstp)
190 {
191         struct nfsd3_writeargs *argp = rqstp->rq_argp;
192         struct nfsd3_writeres *resp = rqstp->rq_resp;
193         unsigned long cnt = argp->len;
194         unsigned int nvecs;
195 
196         dprintk("nfsd: WRITE(3)    %s %d bytes at %Lu%s\n",
197                                 SVCFH_fmt(&argp->fh),
198                                 argp->len,
199                                 (unsigned long long) argp->offset,
200                                 argp->stable ? " stable" : "");
201 
202         resp->status = nfserr_fbig;
203         if (argp->offset > (u64)OFFSET_MAX ||
204             argp->offset + argp->len > (u64)OFFSET_MAX)
205                 return rpc_success;
206 
207         fh_copy(&resp->fh, &argp->fh);
208         resp->committed = argp->stable;
209         nvecs = svc_fill_write_vector(rqstp, &argp->payload);
210 
211         resp->status = nfsd_write(rqstp, &resp->fh, argp->offset,
212                                   rqstp->rq_vec, nvecs, &cnt,
213                                   resp->committed, resp->verf);
214         resp->count = cnt;
215         return rpc_success;
216 }
217 
218 /*
219  * Implement NFSv3's unchecked, guarded, and exclusive CREATE
220  * semantics for regular files. Except for the created file,
221  * this operation is stateless on the server.
222  *
223  * Upon return, caller must release @fhp and @resfhp.
224  */
225 static __be32
226 nfsd3_create_file(struct svc_rqst *rqstp, struct svc_fh *fhp,
227                   struct svc_fh *resfhp, struct nfsd3_createargs *argp)
228 {
229         struct iattr *iap = &argp->attrs;
230         struct dentry *parent, *child;
231         struct nfsd_attrs attrs = {
232                 .na_iattr       = iap,
233         };
234         __u32 v_mtime, v_atime;
235         struct inode *inode;
236         __be32 status;
237         int host_err;
238 
239         if (isdotent(argp->name, argp->len))
240                 return nfserr_exist;
241         if (!(iap->ia_valid & ATTR_MODE))
242                 iap->ia_mode = 0;
243 
244         status = fh_verify(rqstp, fhp, S_IFDIR, NFSD_MAY_EXEC);
245         if (status != nfs_ok)
246                 return status;
247 
248         parent = fhp->fh_dentry;
249         inode = d_inode(parent);
250 
251         host_err = fh_want_write(fhp);
252         if (host_err)
253                 return nfserrno(host_err);
254 
255         inode_lock_nested(inode, I_MUTEX_PARENT);
256 
257         child = lookup_one_len(argp->name, parent, argp->len);
258         if (IS_ERR(child)) {
259                 status = nfserrno(PTR_ERR(child));
260                 goto out;
261         }
262 
263         if (d_really_is_negative(child)) {
264                 status = fh_verify(rqstp, fhp, S_IFDIR, NFSD_MAY_CREATE);
265                 if (status != nfs_ok)
266                         goto out;
267         }
268 
269         status = fh_compose(resfhp, fhp->fh_export, child, fhp);
270         if (status != nfs_ok)
271                 goto out;
272 
273         v_mtime = 0;
274         v_atime = 0;
275         if (argp->createmode == NFS3_CREATE_EXCLUSIVE) {
276                 u32 *verifier = (u32 *)argp->verf;
277 
278                 /*
279                  * Solaris 7 gets confused (bugid 4218508) if these have
280                  * the high bit set, as do xfs filesystems without the
281                  * "bigtime" feature. So just clear the high bits.
282                  */
283                 v_mtime = verifier[0] & 0x7fffffff;
284                 v_atime = verifier[1] & 0x7fffffff;
285         }
286 
287         if (d_really_is_positive(child)) {
288                 status = nfs_ok;
289 
290                 switch (argp->createmode) {
291                 case NFS3_CREATE_UNCHECKED:
292                         if (!d_is_reg(child))
293                                 break;
294                         iap->ia_valid &= ATTR_SIZE;
295                         goto set_attr;
296                 case NFS3_CREATE_GUARDED:
297                         status = nfserr_exist;
298                         break;
299                 case NFS3_CREATE_EXCLUSIVE:
300                         if (inode_get_mtime_sec(d_inode(child)) == v_mtime &&
301                             inode_get_atime_sec(d_inode(child)) == v_atime &&
302                             d_inode(child)->i_size == 0) {
303                                 break;
304                         }
305                         status = nfserr_exist;
306                 }
307                 goto out;
308         }
309 
310         if (!IS_POSIXACL(inode))
311                 iap->ia_mode &= ~current_umask();
312 
313         status = fh_fill_pre_attrs(fhp);
314         if (status != nfs_ok)
315                 goto out;
316         host_err = vfs_create(&nop_mnt_idmap, inode, child, iap->ia_mode, true);
317         if (host_err < 0) {
318                 status = nfserrno(host_err);
319                 goto out;
320         }
321         fh_fill_post_attrs(fhp);
322 
323         /* A newly created file already has a file size of zero. */
324         if ((iap->ia_valid & ATTR_SIZE) && (iap->ia_size == 0))
325                 iap->ia_valid &= ~ATTR_SIZE;
326         if (argp->createmode == NFS3_CREATE_EXCLUSIVE) {
327                 iap->ia_valid = ATTR_MTIME | ATTR_ATIME |
328                                 ATTR_MTIME_SET | ATTR_ATIME_SET;
329                 iap->ia_mtime.tv_sec = v_mtime;
330                 iap->ia_atime.tv_sec = v_atime;
331                 iap->ia_mtime.tv_nsec = 0;
332                 iap->ia_atime.tv_nsec = 0;
333         }
334 
335 set_attr:
336         status = nfsd_create_setattr(rqstp, fhp, resfhp, &attrs);
337 
338 out:
339         inode_unlock(inode);
340         if (child && !IS_ERR(child))
341                 dput(child);
342         fh_drop_write(fhp);
343         return status;
344 }
345 
346 static __be32
347 nfsd3_proc_create(struct svc_rqst *rqstp)
348 {
349         struct nfsd3_createargs *argp = rqstp->rq_argp;
350         struct nfsd3_diropres *resp = rqstp->rq_resp;
351         svc_fh *dirfhp, *newfhp;
352 
353         dprintk("nfsd: CREATE(3)   %s %.*s\n",
354                                 SVCFH_fmt(&argp->fh),
355                                 argp->len,
356                                 argp->name);
357 
358         dirfhp = fh_copy(&resp->dirfh, &argp->fh);
359         newfhp = fh_init(&resp->fh, NFS3_FHSIZE);
360 
361         resp->status = nfsd3_create_file(rqstp, dirfhp, newfhp, argp);
362         return rpc_success;
363 }
364 
365 /*
366  * Make directory. This operation is not idempotent.
367  */
368 static __be32
369 nfsd3_proc_mkdir(struct svc_rqst *rqstp)
370 {
371         struct nfsd3_createargs *argp = rqstp->rq_argp;
372         struct nfsd3_diropres *resp = rqstp->rq_resp;
373         struct nfsd_attrs attrs = {
374                 .na_iattr       = &argp->attrs,
375         };
376 
377         dprintk("nfsd: MKDIR(3)    %s %.*s\n",
378                                 SVCFH_fmt(&argp->fh),
379                                 argp->len,
380                                 argp->name);
381 
382         argp->attrs.ia_valid &= ~ATTR_SIZE;
383         fh_copy(&resp->dirfh, &argp->fh);
384         fh_init(&resp->fh, NFS3_FHSIZE);
385         resp->status = nfsd_create(rqstp, &resp->dirfh, argp->name, argp->len,
386                                    &attrs, S_IFDIR, 0, &resp->fh);
387         return rpc_success;
388 }
389 
390 static __be32
391 nfsd3_proc_symlink(struct svc_rqst *rqstp)
392 {
393         struct nfsd3_symlinkargs *argp = rqstp->rq_argp;
394         struct nfsd3_diropres *resp = rqstp->rq_resp;
395         struct nfsd_attrs attrs = {
396                 .na_iattr       = &argp->attrs,
397         };
398 
399         if (argp->tlen == 0) {
400                 resp->status = nfserr_inval;
401                 goto out;
402         }
403         if (argp->tlen > NFS3_MAXPATHLEN) {
404                 resp->status = nfserr_nametoolong;
405                 goto out;
406         }
407 
408         argp->tname = svc_fill_symlink_pathname(rqstp, &argp->first,
409                                                 page_address(rqstp->rq_arg.pages[0]),
410                                                 argp->tlen);
411         if (IS_ERR(argp->tname)) {
412                 resp->status = nfserrno(PTR_ERR(argp->tname));
413                 goto out;
414         }
415 
416         dprintk("nfsd: SYMLINK(3)  %s %.*s -> %.*s\n",
417                                 SVCFH_fmt(&argp->ffh),
418                                 argp->flen, argp->fname,
419                                 argp->tlen, argp->tname);
420 
421         fh_copy(&resp->dirfh, &argp->ffh);
422         fh_init(&resp->fh, NFS3_FHSIZE);
423         resp->status = nfsd_symlink(rqstp, &resp->dirfh, argp->fname,
424                                     argp->flen, argp->tname, &attrs, &resp->fh);
425         kfree(argp->tname);
426 out:
427         return rpc_success;
428 }
429 
430 /*
431  * Make socket/fifo/device.
432  */
433 static __be32
434 nfsd3_proc_mknod(struct svc_rqst *rqstp)
435 {
436         struct nfsd3_mknodargs *argp = rqstp->rq_argp;
437         struct nfsd3_diropres  *resp = rqstp->rq_resp;
438         struct nfsd_attrs attrs = {
439                 .na_iattr       = &argp->attrs,
440         };
441         int type;
442         dev_t   rdev = 0;
443 
444         dprintk("nfsd: MKNOD(3)    %s %.*s\n",
445                                 SVCFH_fmt(&argp->fh),
446                                 argp->len,
447                                 argp->name);
448 
449         fh_copy(&resp->dirfh, &argp->fh);
450         fh_init(&resp->fh, NFS3_FHSIZE);
451 
452         if (argp->ftype == NF3CHR || argp->ftype == NF3BLK) {
453                 rdev = MKDEV(argp->major, argp->minor);
454                 if (MAJOR(rdev) != argp->major ||
455                     MINOR(rdev) != argp->minor) {
456                         resp->status = nfserr_inval;
457                         goto out;
458                 }
459         } else if (argp->ftype != NF3SOCK && argp->ftype != NF3FIFO) {
460                 resp->status = nfserr_badtype;
461                 goto out;
462         }
463 
464         type = nfs3_ftypes[argp->ftype];
465         resp->status = nfsd_create(rqstp, &resp->dirfh, argp->name, argp->len,
466                                    &attrs, type, rdev, &resp->fh);
467 out:
468         return rpc_success;
469 }
470 
471 /*
472  * Remove file/fifo/socket etc.
473  */
474 static __be32
475 nfsd3_proc_remove(struct svc_rqst *rqstp)
476 {
477         struct nfsd3_diropargs *argp = rqstp->rq_argp;
478         struct nfsd3_attrstat *resp = rqstp->rq_resp;
479 
480         dprintk("nfsd: REMOVE(3)   %s %.*s\n",
481                                 SVCFH_fmt(&argp->fh),
482                                 argp->len,
483                                 argp->name);
484 
485         /* Unlink. -S_IFDIR means file must not be a directory */
486         fh_copy(&resp->fh, &argp->fh);
487         resp->status = nfsd_unlink(rqstp, &resp->fh, -S_IFDIR,
488                                    argp->name, argp->len);
489         return rpc_success;
490 }
491 
492 /*
493  * Remove a directory
494  */
495 static __be32
496 nfsd3_proc_rmdir(struct svc_rqst *rqstp)
497 {
498         struct nfsd3_diropargs *argp = rqstp->rq_argp;
499         struct nfsd3_attrstat *resp = rqstp->rq_resp;
500 
501         dprintk("nfsd: RMDIR(3)    %s %.*s\n",
502                                 SVCFH_fmt(&argp->fh),
503                                 argp->len,
504                                 argp->name);
505 
506         fh_copy(&resp->fh, &argp->fh);
507         resp->status = nfsd_unlink(rqstp, &resp->fh, S_IFDIR,
508                                    argp->name, argp->len);
509         return rpc_success;
510 }
511 
512 static __be32
513 nfsd3_proc_rename(struct svc_rqst *rqstp)
514 {
515         struct nfsd3_renameargs *argp = rqstp->rq_argp;
516         struct nfsd3_renameres *resp = rqstp->rq_resp;
517 
518         dprintk("nfsd: RENAME(3)   %s %.*s ->\n",
519                                 SVCFH_fmt(&argp->ffh),
520                                 argp->flen,
521                                 argp->fname);
522         dprintk("nfsd: -> %s %.*s\n",
523                                 SVCFH_fmt(&argp->tfh),
524                                 argp->tlen,
525                                 argp->tname);
526 
527         fh_copy(&resp->ffh, &argp->ffh);
528         fh_copy(&resp->tfh, &argp->tfh);
529         resp->status = nfsd_rename(rqstp, &resp->ffh, argp->fname, argp->flen,
530                                    &resp->tfh, argp->tname, argp->tlen);
531         return rpc_success;
532 }
533 
534 static __be32
535 nfsd3_proc_link(struct svc_rqst *rqstp)
536 {
537         struct nfsd3_linkargs *argp = rqstp->rq_argp;
538         struct nfsd3_linkres  *resp = rqstp->rq_resp;
539 
540         dprintk("nfsd: LINK(3)     %s ->\n",
541                                 SVCFH_fmt(&argp->ffh));
542         dprintk("nfsd:   -> %s %.*s\n",
543                                 SVCFH_fmt(&argp->tfh),
544                                 argp->tlen,
545                                 argp->tname);
546 
547         fh_copy(&resp->fh,  &argp->ffh);
548         fh_copy(&resp->tfh, &argp->tfh);
549         resp->status = nfsd_link(rqstp, &resp->tfh, argp->tname, argp->tlen,
550                                  &resp->fh);
551         return rpc_success;
552 }
553 
554 static void nfsd3_init_dirlist_pages(struct svc_rqst *rqstp,
555                                      struct nfsd3_readdirres *resp,
556                                      u32 count)
557 {
558         struct xdr_buf *buf = &resp->dirlist;
559         struct xdr_stream *xdr = &resp->xdr;
560         unsigned int sendbuf = min_t(unsigned int, rqstp->rq_res.buflen,
561                                      svc_max_payload(rqstp));
562 
563         memset(buf, 0, sizeof(*buf));
564 
565         /* Reserve room for the NULL ptr & eof flag (-2 words) */
566         buf->buflen = clamp(count, (u32)(XDR_UNIT * 2), sendbuf);
567         buf->buflen -= XDR_UNIT * 2;
568         buf->pages = rqstp->rq_next_page;
569         rqstp->rq_next_page += (buf->buflen + PAGE_SIZE - 1) >> PAGE_SHIFT;
570 
571         xdr_init_encode_pages(xdr, buf, buf->pages,  NULL);
572 }
573 
574 /*
575  * Read a portion of a directory.
576  */
577 static __be32
578 nfsd3_proc_readdir(struct svc_rqst *rqstp)
579 {
580         struct nfsd3_readdirargs *argp = rqstp->rq_argp;
581         struct nfsd3_readdirres  *resp = rqstp->rq_resp;
582         loff_t          offset;
583 
584         dprintk("nfsd: READDIR(3)  %s %d bytes at %d\n",
585                                 SVCFH_fmt(&argp->fh),
586                                 argp->count, (u32) argp->cookie);
587 
588         nfsd3_init_dirlist_pages(rqstp, resp, argp->count);
589 
590         fh_copy(&resp->fh, &argp->fh);
591         resp->common.err = nfs_ok;
592         resp->cookie_offset = 0;
593         resp->rqstp = rqstp;
594         offset = argp->cookie;
595         resp->status = nfsd_readdir(rqstp, &resp->fh, &offset,
596                                     &resp->common, nfs3svc_encode_entry3);
597         memcpy(resp->verf, argp->verf, 8);
598         nfs3svc_encode_cookie3(resp, offset);
599 
600         /* Recycle only pages that were part of the reply */
601         rqstp->rq_next_page = resp->xdr.page_ptr + 1;
602 
603         return rpc_success;
604 }
605 
606 /*
607  * Read a portion of a directory, including file handles and attrs.
608  * For now, we choose to ignore the dircount parameter.
609  */
610 static __be32
611 nfsd3_proc_readdirplus(struct svc_rqst *rqstp)
612 {
613         struct nfsd3_readdirargs *argp = rqstp->rq_argp;
614         struct nfsd3_readdirres  *resp = rqstp->rq_resp;
615         loff_t  offset;
616 
617         dprintk("nfsd: READDIR+(3) %s %d bytes at %d\n",
618                                 SVCFH_fmt(&argp->fh),
619                                 argp->count, (u32) argp->cookie);
620 
621         nfsd3_init_dirlist_pages(rqstp, resp, argp->count);
622 
623         fh_copy(&resp->fh, &argp->fh);
624         resp->common.err = nfs_ok;
625         resp->cookie_offset = 0;
626         resp->rqstp = rqstp;
627         offset = argp->cookie;
628 
629         resp->status = fh_verify(rqstp, &resp->fh, S_IFDIR, NFSD_MAY_NOP);
630         if (resp->status != nfs_ok)
631                 goto out;
632 
633         if (resp->fh.fh_export->ex_flags & NFSEXP_NOREADDIRPLUS) {
634                 resp->status = nfserr_notsupp;
635                 goto out;
636         }
637 
638         resp->status = nfsd_readdir(rqstp, &resp->fh, &offset,
639                                     &resp->common, nfs3svc_encode_entryplus3);
640         memcpy(resp->verf, argp->verf, 8);
641         nfs3svc_encode_cookie3(resp, offset);
642 
643         /* Recycle only pages that were part of the reply */
644         rqstp->rq_next_page = resp->xdr.page_ptr + 1;
645 
646 out:
647         return rpc_success;
648 }
649 
650 /*
651  * Get file system stats
652  */
653 static __be32
654 nfsd3_proc_fsstat(struct svc_rqst *rqstp)
655 {
656         struct nfsd_fhandle *argp = rqstp->rq_argp;
657         struct nfsd3_fsstatres *resp = rqstp->rq_resp;
658 
659         dprintk("nfsd: FSSTAT(3)   %s\n",
660                                 SVCFH_fmt(&argp->fh));
661 
662         resp->status = nfsd_statfs(rqstp, &argp->fh, &resp->stats, 0);
663         fh_put(&argp->fh);
664         return rpc_success;
665 }
666 
667 /*
668  * Get file system info
669  */
670 static __be32
671 nfsd3_proc_fsinfo(struct svc_rqst *rqstp)
672 {
673         struct nfsd_fhandle *argp = rqstp->rq_argp;
674         struct nfsd3_fsinfores *resp = rqstp->rq_resp;
675         u32     max_blocksize = svc_max_payload(rqstp);
676 
677         dprintk("nfsd: FSINFO(3)   %s\n",
678                                 SVCFH_fmt(&argp->fh));
679 
680         resp->f_rtmax  = max_blocksize;
681         resp->f_rtpref = max_blocksize;
682         resp->f_rtmult = PAGE_SIZE;
683         resp->f_wtmax  = max_blocksize;
684         resp->f_wtpref = max_blocksize;
685         resp->f_wtmult = PAGE_SIZE;
686         resp->f_dtpref = max_blocksize;
687         resp->f_maxfilesize = ~(u32) 0;
688         resp->f_properties = NFS3_FSF_DEFAULT;
689 
690         resp->status = fh_verify(rqstp, &argp->fh, 0,
691                                  NFSD_MAY_NOP | NFSD_MAY_BYPASS_GSS_ON_ROOT);
692 
693         /* Check special features of the file system. May request
694          * different read/write sizes for file systems known to have
695          * problems with large blocks */
696         if (resp->status == nfs_ok) {
697                 struct super_block *sb = argp->fh.fh_dentry->d_sb;
698 
699                 /* Note that we don't care for remote fs's here */
700                 if (sb->s_magic == MSDOS_SUPER_MAGIC) {
701                         resp->f_properties = NFS3_FSF_BILLYBOY;
702                 }
703                 resp->f_maxfilesize = sb->s_maxbytes;
704         }
705 
706         fh_put(&argp->fh);
707         return rpc_success;
708 }
709 
710 /*
711  * Get pathconf info for the specified file
712  */
713 static __be32
714 nfsd3_proc_pathconf(struct svc_rqst *rqstp)
715 {
716         struct nfsd_fhandle *argp = rqstp->rq_argp;
717         struct nfsd3_pathconfres *resp = rqstp->rq_resp;
718 
719         dprintk("nfsd: PATHCONF(3) %s\n",
720                                 SVCFH_fmt(&argp->fh));
721 
722         /* Set default pathconf */
723         resp->p_link_max = 255;         /* at least */
724         resp->p_name_max = 255;         /* at least */
725         resp->p_no_trunc = 0;
726         resp->p_chown_restricted = 1;
727         resp->p_case_insensitive = 0;
728         resp->p_case_preserving = 1;
729 
730         resp->status = fh_verify(rqstp, &argp->fh, 0, NFSD_MAY_NOP);
731 
732         if (resp->status == nfs_ok) {
733                 struct super_block *sb = argp->fh.fh_dentry->d_sb;
734 
735                 /* Note that we don't care for remote fs's here */
736                 switch (sb->s_magic) {
737                 case EXT2_SUPER_MAGIC:
738                         resp->p_link_max = EXT2_LINK_MAX;
739                         resp->p_name_max = EXT2_NAME_LEN;
740                         break;
741                 case MSDOS_SUPER_MAGIC:
742                         resp->p_case_insensitive = 1;
743                         resp->p_case_preserving  = 0;
744                         break;
745                 }
746         }
747 
748         fh_put(&argp->fh);
749         return rpc_success;
750 }
751 
752 /*
753  * Commit a file (range) to stable storage.
754  */
755 static __be32
756 nfsd3_proc_commit(struct svc_rqst *rqstp)
757 {
758         struct nfsd3_commitargs *argp = rqstp->rq_argp;
759         struct nfsd3_commitres *resp = rqstp->rq_resp;
760         struct nfsd_file *nf;
761 
762         dprintk("nfsd: COMMIT(3)   %s %u@%Lu\n",
763                                 SVCFH_fmt(&argp->fh),
764                                 argp->count,
765                                 (unsigned long long) argp->offset);
766 
767         fh_copy(&resp->fh, &argp->fh);
768         resp->status = nfsd_file_acquire_gc(rqstp, &resp->fh, NFSD_MAY_WRITE |
769                                             NFSD_MAY_NOT_BREAK_LEASE, &nf);
770         if (resp->status)
771                 goto out;
772         resp->status = nfsd_commit(rqstp, &resp->fh, nf, argp->offset,
773                                    argp->count, resp->verf);
774         nfsd_file_put(nf);
775 out:
776         return rpc_success;
777 }
778 
779 
780 /*
781  * NFSv3 Server procedures.
782  * Only the results of non-idempotent operations are cached.
783  */
784 #define nfs3svc_encode_attrstatres      nfs3svc_encode_attrstat
785 #define nfs3svc_encode_wccstatres       nfs3svc_encode_wccstat
786 #define nfsd3_mkdirargs                 nfsd3_createargs
787 #define nfsd3_readdirplusargs           nfsd3_readdirargs
788 #define nfsd3_fhandleargs               nfsd_fhandle
789 #define nfsd3_attrstatres               nfsd3_attrstat
790 #define nfsd3_wccstatres                nfsd3_attrstat
791 #define nfsd3_createres                 nfsd3_diropres
792 
793 #define ST 1            /* status*/
794 #define FH 17           /* filehandle with length */
795 #define AT 21           /* attributes */
796 #define pAT (1+AT)      /* post attributes - conditional */
797 #define WC (7+pAT)      /* WCC attributes */
798 
799 static const struct svc_procedure nfsd_procedures3[22] = {
800         [NFS3PROC_NULL] = {
801                 .pc_func = nfsd3_proc_null,
802                 .pc_decode = nfssvc_decode_voidarg,
803                 .pc_encode = nfssvc_encode_voidres,
804                 .pc_argsize = sizeof(struct nfsd_voidargs),
805                 .pc_argzero = sizeof(struct nfsd_voidargs),
806                 .pc_ressize = sizeof(struct nfsd_voidres),
807                 .pc_cachetype = RC_NOCACHE,
808                 .pc_xdrressize = ST,
809                 .pc_name = "NULL",
810         },
811         [NFS3PROC_GETATTR] = {
812                 .pc_func = nfsd3_proc_getattr,
813                 .pc_decode = nfs3svc_decode_fhandleargs,
814                 .pc_encode = nfs3svc_encode_getattrres,
815                 .pc_release = nfs3svc_release_fhandle,
816                 .pc_argsize = sizeof(struct nfsd_fhandle),
817                 .pc_argzero = sizeof(struct nfsd_fhandle),
818                 .pc_ressize = sizeof(struct nfsd3_attrstatres),
819                 .pc_cachetype = RC_NOCACHE,
820                 .pc_xdrressize = ST+AT,
821                 .pc_name = "GETATTR",
822         },
823         [NFS3PROC_SETATTR] = {
824                 .pc_func = nfsd3_proc_setattr,
825                 .pc_decode = nfs3svc_decode_sattrargs,
826                 .pc_encode = nfs3svc_encode_wccstatres,
827                 .pc_release = nfs3svc_release_fhandle,
828                 .pc_argsize = sizeof(struct nfsd3_sattrargs),
829                 .pc_argzero = sizeof(struct nfsd3_sattrargs),
830                 .pc_ressize = sizeof(struct nfsd3_wccstatres),
831                 .pc_cachetype = RC_REPLBUFF,
832                 .pc_xdrressize = ST+WC,
833                 .pc_name = "SETATTR",
834         },
835         [NFS3PROC_LOOKUP] = {
836                 .pc_func = nfsd3_proc_lookup,
837                 .pc_decode = nfs3svc_decode_diropargs,
838                 .pc_encode = nfs3svc_encode_lookupres,
839                 .pc_release = nfs3svc_release_fhandle2,
840                 .pc_argsize = sizeof(struct nfsd3_diropargs),
841                 .pc_argzero = sizeof(struct nfsd3_diropargs),
842                 .pc_ressize = sizeof(struct nfsd3_diropres),
843                 .pc_cachetype = RC_NOCACHE,
844                 .pc_xdrressize = ST+FH+pAT+pAT,
845                 .pc_name = "LOOKUP",
846         },
847         [NFS3PROC_ACCESS] = {
848                 .pc_func = nfsd3_proc_access,
849                 .pc_decode = nfs3svc_decode_accessargs,
850                 .pc_encode = nfs3svc_encode_accessres,
851                 .pc_release = nfs3svc_release_fhandle,
852                 .pc_argsize = sizeof(struct nfsd3_accessargs),
853                 .pc_argzero = sizeof(struct nfsd3_accessargs),
854                 .pc_ressize = sizeof(struct nfsd3_accessres),
855                 .pc_cachetype = RC_NOCACHE,
856                 .pc_xdrressize = ST+pAT+1,
857                 .pc_name = "ACCESS",
858         },
859         [NFS3PROC_READLINK] = {
860                 .pc_func = nfsd3_proc_readlink,
861                 .pc_decode = nfs3svc_decode_fhandleargs,
862                 .pc_encode = nfs3svc_encode_readlinkres,
863                 .pc_release = nfs3svc_release_fhandle,
864                 .pc_argsize = sizeof(struct nfsd_fhandle),
865                 .pc_argzero = sizeof(struct nfsd_fhandle),
866                 .pc_ressize = sizeof(struct nfsd3_readlinkres),
867                 .pc_cachetype = RC_NOCACHE,
868                 .pc_xdrressize = ST+pAT+1+NFS3_MAXPATHLEN/4,
869                 .pc_name = "READLINK",
870         },
871         [NFS3PROC_READ] = {
872                 .pc_func = nfsd3_proc_read,
873                 .pc_decode = nfs3svc_decode_readargs,
874                 .pc_encode = nfs3svc_encode_readres,
875                 .pc_release = nfs3svc_release_fhandle,
876                 .pc_argsize = sizeof(struct nfsd3_readargs),
877                 .pc_argzero = sizeof(struct nfsd3_readargs),
878                 .pc_ressize = sizeof(struct nfsd3_readres),
879                 .pc_cachetype = RC_NOCACHE,
880                 .pc_xdrressize = ST+pAT+4+NFSSVC_MAXBLKSIZE/4,
881                 .pc_name = "READ",
882         },
883         [NFS3PROC_WRITE] = {
884                 .pc_func = nfsd3_proc_write,
885                 .pc_decode = nfs3svc_decode_writeargs,
886                 .pc_encode = nfs3svc_encode_writeres,
887                 .pc_release = nfs3svc_release_fhandle,
888                 .pc_argsize = sizeof(struct nfsd3_writeargs),
889                 .pc_argzero = sizeof(struct nfsd3_writeargs),
890                 .pc_ressize = sizeof(struct nfsd3_writeres),
891                 .pc_cachetype = RC_REPLBUFF,
892                 .pc_xdrressize = ST+WC+4,
893                 .pc_name = "WRITE",
894         },
895         [NFS3PROC_CREATE] = {
896                 .pc_func = nfsd3_proc_create,
897                 .pc_decode = nfs3svc_decode_createargs,
898                 .pc_encode = nfs3svc_encode_createres,
899                 .pc_release = nfs3svc_release_fhandle2,
900                 .pc_argsize = sizeof(struct nfsd3_createargs),
901                 .pc_argzero = sizeof(struct nfsd3_createargs),
902                 .pc_ressize = sizeof(struct nfsd3_createres),
903                 .pc_cachetype = RC_REPLBUFF,
904                 .pc_xdrressize = ST+(1+FH+pAT)+WC,
905                 .pc_name = "CREATE",
906         },
907         [NFS3PROC_MKDIR] = {
908                 .pc_func = nfsd3_proc_mkdir,
909                 .pc_decode = nfs3svc_decode_mkdirargs,
910                 .pc_encode = nfs3svc_encode_createres,
911                 .pc_release = nfs3svc_release_fhandle2,
912                 .pc_argsize = sizeof(struct nfsd3_mkdirargs),
913                 .pc_argzero = sizeof(struct nfsd3_mkdirargs),
914                 .pc_ressize = sizeof(struct nfsd3_createres),
915                 .pc_cachetype = RC_REPLBUFF,
916                 .pc_xdrressize = ST+(1+FH+pAT)+WC,
917                 .pc_name = "MKDIR",
918         },
919         [NFS3PROC_SYMLINK] = {
920                 .pc_func = nfsd3_proc_symlink,
921                 .pc_decode = nfs3svc_decode_symlinkargs,
922                 .pc_encode = nfs3svc_encode_createres,
923                 .pc_release = nfs3svc_release_fhandle2,
924                 .pc_argsize = sizeof(struct nfsd3_symlinkargs),
925                 .pc_argzero = sizeof(struct nfsd3_symlinkargs),
926                 .pc_ressize = sizeof(struct nfsd3_createres),
927                 .pc_cachetype = RC_REPLBUFF,
928                 .pc_xdrressize = ST+(1+FH+pAT)+WC,
929                 .pc_name = "SYMLINK",
930         },
931         [NFS3PROC_MKNOD] = {
932                 .pc_func = nfsd3_proc_mknod,
933                 .pc_decode = nfs3svc_decode_mknodargs,
934                 .pc_encode = nfs3svc_encode_createres,
935                 .pc_release = nfs3svc_release_fhandle2,
936                 .pc_argsize = sizeof(struct nfsd3_mknodargs),
937                 .pc_argzero = sizeof(struct nfsd3_mknodargs),
938                 .pc_ressize = sizeof(struct nfsd3_createres),
939                 .pc_cachetype = RC_REPLBUFF,
940                 .pc_xdrressize = ST+(1+FH+pAT)+WC,
941                 .pc_name = "MKNOD",
942         },
943         [NFS3PROC_REMOVE] = {
944                 .pc_func = nfsd3_proc_remove,
945                 .pc_decode = nfs3svc_decode_diropargs,
946                 .pc_encode = nfs3svc_encode_wccstatres,
947                 .pc_release = nfs3svc_release_fhandle,
948                 .pc_argsize = sizeof(struct nfsd3_diropargs),
949                 .pc_argzero = sizeof(struct nfsd3_diropargs),
950                 .pc_ressize = sizeof(struct nfsd3_wccstatres),
951                 .pc_cachetype = RC_REPLBUFF,
952                 .pc_xdrressize = ST+WC,
953                 .pc_name = "REMOVE",
954         },
955         [NFS3PROC_RMDIR] = {
956                 .pc_func = nfsd3_proc_rmdir,
957                 .pc_decode = nfs3svc_decode_diropargs,
958                 .pc_encode = nfs3svc_encode_wccstatres,
959                 .pc_release = nfs3svc_release_fhandle,
960                 .pc_argsize = sizeof(struct nfsd3_diropargs),
961                 .pc_argzero = sizeof(struct nfsd3_diropargs),
962                 .pc_ressize = sizeof(struct nfsd3_wccstatres),
963                 .pc_cachetype = RC_REPLBUFF,
964                 .pc_xdrressize = ST+WC,
965                 .pc_name = "RMDIR",
966         },
967         [NFS3PROC_RENAME] = {
968                 .pc_func = nfsd3_proc_rename,
969                 .pc_decode = nfs3svc_decode_renameargs,
970                 .pc_encode = nfs3svc_encode_renameres,
971                 .pc_release = nfs3svc_release_fhandle2,
972                 .pc_argsize = sizeof(struct nfsd3_renameargs),
973                 .pc_argzero = sizeof(struct nfsd3_renameargs),
974                 .pc_ressize = sizeof(struct nfsd3_renameres),
975                 .pc_cachetype = RC_REPLBUFF,
976                 .pc_xdrressize = ST+WC+WC,
977                 .pc_name = "RENAME",
978         },
979         [NFS3PROC_LINK] = {
980                 .pc_func = nfsd3_proc_link,
981                 .pc_decode = nfs3svc_decode_linkargs,
982                 .pc_encode = nfs3svc_encode_linkres,
983                 .pc_release = nfs3svc_release_fhandle2,
984                 .pc_argsize = sizeof(struct nfsd3_linkargs),
985                 .pc_argzero = sizeof(struct nfsd3_linkargs),
986                 .pc_ressize = sizeof(struct nfsd3_linkres),
987                 .pc_cachetype = RC_REPLBUFF,
988                 .pc_xdrressize = ST+pAT+WC,
989                 .pc_name = "LINK",
990         },
991         [NFS3PROC_READDIR] = {
992                 .pc_func = nfsd3_proc_readdir,
993                 .pc_decode = nfs3svc_decode_readdirargs,
994                 .pc_encode = nfs3svc_encode_readdirres,
995                 .pc_release = nfs3svc_release_fhandle,
996                 .pc_argsize = sizeof(struct nfsd3_readdirargs),
997                 .pc_argzero = sizeof(struct nfsd3_readdirargs),
998                 .pc_ressize = sizeof(struct nfsd3_readdirres),
999                 .pc_cachetype = RC_NOCACHE,
1000                 .pc_name = "READDIR",
1001         },
1002         [NFS3PROC_READDIRPLUS] = {
1003                 .pc_func = nfsd3_proc_readdirplus,
1004                 .pc_decode = nfs3svc_decode_readdirplusargs,
1005                 .pc_encode = nfs3svc_encode_readdirres,
1006                 .pc_release = nfs3svc_release_fhandle,
1007                 .pc_argsize = sizeof(struct nfsd3_readdirplusargs),
1008                 .pc_argzero = sizeof(struct nfsd3_readdirplusargs),
1009                 .pc_ressize = sizeof(struct nfsd3_readdirres),
1010                 .pc_cachetype = RC_NOCACHE,
1011                 .pc_name = "READDIRPLUS",
1012         },
1013         [NFS3PROC_FSSTAT] = {
1014                 .pc_func = nfsd3_proc_fsstat,
1015                 .pc_decode = nfs3svc_decode_fhandleargs,
1016                 .pc_encode = nfs3svc_encode_fsstatres,
1017                 .pc_argsize = sizeof(struct nfsd3_fhandleargs),
1018                 .pc_argzero = sizeof(struct nfsd3_fhandleargs),
1019                 .pc_ressize = sizeof(struct nfsd3_fsstatres),
1020                 .pc_cachetype = RC_NOCACHE,
1021                 .pc_xdrressize = ST+pAT+2*6+1,
1022                 .pc_name = "FSSTAT",
1023         },
1024         [NFS3PROC_FSINFO] = {
1025                 .pc_func = nfsd3_proc_fsinfo,
1026                 .pc_decode = nfs3svc_decode_fhandleargs,
1027                 .pc_encode = nfs3svc_encode_fsinfores,
1028                 .pc_argsize = sizeof(struct nfsd3_fhandleargs),
1029                 .pc_argzero = sizeof(struct nfsd3_fhandleargs),
1030                 .pc_ressize = sizeof(struct nfsd3_fsinfores),
1031                 .pc_cachetype = RC_NOCACHE,
1032                 .pc_xdrressize = ST+pAT+12,
1033                 .pc_name = "FSINFO",
1034         },
1035         [NFS3PROC_PATHCONF] = {
1036                 .pc_func = nfsd3_proc_pathconf,
1037                 .pc_decode = nfs3svc_decode_fhandleargs,
1038                 .pc_encode = nfs3svc_encode_pathconfres,
1039                 .pc_argsize = sizeof(struct nfsd3_fhandleargs),
1040                 .pc_argzero = sizeof(struct nfsd3_fhandleargs),
1041                 .pc_ressize = sizeof(struct nfsd3_pathconfres),
1042                 .pc_cachetype = RC_NOCACHE,
1043                 .pc_xdrressize = ST+pAT+6,
1044                 .pc_name = "PATHCONF",
1045         },
1046         [NFS3PROC_COMMIT] = {
1047                 .pc_func = nfsd3_proc_commit,
1048                 .pc_decode = nfs3svc_decode_commitargs,
1049                 .pc_encode = nfs3svc_encode_commitres,
1050                 .pc_release = nfs3svc_release_fhandle,
1051                 .pc_argsize = sizeof(struct nfsd3_commitargs),
1052                 .pc_argzero = sizeof(struct nfsd3_commitargs),
1053                 .pc_ressize = sizeof(struct nfsd3_commitres),
1054                 .pc_cachetype = RC_NOCACHE,
1055                 .pc_xdrressize = ST+WC+2,
1056                 .pc_name = "COMMIT",
1057         },
1058 };
1059 
1060 static DEFINE_PER_CPU_ALIGNED(unsigned long,
1061                               nfsd_count3[ARRAY_SIZE(nfsd_procedures3)]);
1062 const struct svc_version nfsd_version3 = {
1063         .vs_vers        = 3,
1064         .vs_nproc       = ARRAY_SIZE(nfsd_procedures3),
1065         .vs_proc        = nfsd_procedures3,
1066         .vs_dispatch    = nfsd_dispatch,
1067         .vs_count       = nfsd_count3,
1068         .vs_xdrsize     = NFS3_SVC_XDRSIZE,
1069 };
1070 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php