2  * linux/fs/nfsd/nfsctl.c
 
   4  * Syscall interface to knfsd.
 
   6  * Copyright (C) 1995, 1996 Olaf Kirch <okir@monad.swb.de>
 
   9 #include <linux/config.h>
 
  10 #include <linux/module.h>
 
  12 #include <linux/linkage.h>
 
  13 #include <linux/time.h>
 
  14 #include <linux/errno.h>
 
  16 #include <linux/fcntl.h>
 
  17 #include <linux/net.h>
 
  19 #include <linux/syscalls.h>
 
  20 #include <linux/unistd.h>
 
  21 #include <linux/slab.h>
 
  22 #include <linux/proc_fs.h>
 
  23 #include <linux/seq_file.h>
 
  24 #include <linux/pagemap.h>
 
  25 #include <linux/init.h>
 
  26 #include <linux/string.h>
 
  28 #include <linux/nfs.h>
 
  29 #include <linux/nfsd_idmap.h>
 
  30 #include <linux/sunrpc/svc.h>
 
  31 #include <linux/nfsd/nfsd.h>
 
  32 #include <linux/nfsd/cache.h>
 
  33 #include <linux/nfsd/xdr.h>
 
  34 #include <linux/nfsd/syscall.h>
 
  35 #include <linux/nfsd/interface.h>
 
  37 #include <asm/uaccess.h>
 
  39 unsigned int nfsd_versbits = ~0;
 
  42  *      We have a single directory with 9 nodes in it.
 
  58          * The below MUST come last.  Otherwise we leave a hole in nfsd_files[]
 
  59          * with !CONFIG_NFSD_V4 and simple_fill_super() goes oops
 
  68  * write() for these nodes.
 
  70 static ssize_t write_svc(struct file *file, char *buf, size_t size);
 
  71 static ssize_t write_add(struct file *file, char *buf, size_t size);
 
  72 static ssize_t write_del(struct file *file, char *buf, size_t size);
 
  73 static ssize_t write_export(struct file *file, char *buf, size_t size);
 
  74 static ssize_t write_unexport(struct file *file, char *buf, size_t size);
 
  75 static ssize_t write_getfd(struct file *file, char *buf, size_t size);
 
  76 static ssize_t write_getfs(struct file *file, char *buf, size_t size);
 
  77 static ssize_t write_filehandle(struct file *file, char *buf, size_t size);
 
  78 static ssize_t write_threads(struct file *file, char *buf, size_t size);
 
  79 static ssize_t write_versions(struct file *file, char *buf, size_t size);
 
  81 static ssize_t write_leasetime(struct file *file, char *buf, size_t size);
 
  82 static ssize_t write_recoverydir(struct file *file, char *buf, size_t size);
 
  85 static ssize_t (*write_op[])(struct file *, char *, size_t) = {
 
  86         [NFSD_Svc] = write_svc,
 
  87         [NFSD_Add] = write_add,
 
  88         [NFSD_Del] = write_del,
 
  89         [NFSD_Export] = write_export,
 
  90         [NFSD_Unexport] = write_unexport,
 
  91         [NFSD_Getfd] = write_getfd,
 
  92         [NFSD_Getfs] = write_getfs,
 
  93         [NFSD_Fh] = write_filehandle,
 
  94         [NFSD_Threads] = write_threads,
 
  95         [NFSD_Versions] = write_versions,
 
  97         [NFSD_Leasetime] = write_leasetime,
 
  98         [NFSD_RecoveryDir] = write_recoverydir,
 
 102 static ssize_t nfsctl_transaction_write(struct file *file, const char __user *buf, size_t size, loff_t *pos)
 
 104         ino_t ino =  file->f_dentry->d_inode->i_ino;
 
 108         if (ino >= sizeof(write_op)/sizeof(write_op[0]) || !write_op[ino])
 
 111         data = simple_transaction_get(file, buf, size);
 
 113                 return PTR_ERR(data);
 
 115         rv =  write_op[ino](file, data, size);
 
 117                 simple_transaction_set(file, rv);
 
 123 static ssize_t nfsctl_transaction_read(struct file *file, char __user *buf, size_t size, loff_t *pos)
 
 125         if (! file->private_data) {
 
 126                 /* An attempt to read a transaction file without writing
 
 127                  * causes a 0-byte write so that the file can return
 
 130                 ssize_t rv = nfsctl_transaction_write(file, buf, 0, pos);
 
 134         return simple_transaction_read(file, buf, size, pos);
 
 137 static struct file_operations transaction_ops = {
 
 138         .write          = nfsctl_transaction_write,
 
 139         .read           = nfsctl_transaction_read,
 
 140         .release        = simple_transaction_release,
 
 143 extern struct seq_operations nfs_exports_op;
 
 144 static int exports_open(struct inode *inode, struct file *file)
 
 146         return seq_open(file, &nfs_exports_op);
 
 149 static struct file_operations exports_operations = {
 
 150         .open           = exports_open,
 
 153         .release        = seq_release,
 
 156 /*----------------------------------------------------------------------------*/
 
 158  * payload - write methods
 
 159  * If the method has a response, the response should be put in buf,
 
 160  * and the length returned.  Otherwise return 0 or and -error.
 
 163 static ssize_t write_svc(struct file *file, char *buf, size_t size)
 
 165         struct nfsctl_svc *data;
 
 166         if (size < sizeof(*data))
 
 168         data = (struct nfsctl_svc*) buf;
 
 169         return nfsd_svc(data->svc_port, data->svc_nthreads);
 
 172 static ssize_t write_add(struct file *file, char *buf, size_t size)
 
 174         struct nfsctl_client *data;
 
 175         if (size < sizeof(*data))
 
 177         data = (struct nfsctl_client *)buf;
 
 178         return exp_addclient(data);
 
 181 static ssize_t write_del(struct file *file, char *buf, size_t size)
 
 183         struct nfsctl_client *data;
 
 184         if (size < sizeof(*data))
 
 186         data = (struct nfsctl_client *)buf;
 
 187         return exp_delclient(data);
 
 190 static ssize_t write_export(struct file *file, char *buf, size_t size)
 
 192         struct nfsctl_export *data;
 
 193         if (size < sizeof(*data))
 
 195         data = (struct nfsctl_export*)buf;
 
 196         return exp_export(data);
 
 199 static ssize_t write_unexport(struct file *file, char *buf, size_t size)
 
 201         struct nfsctl_export *data;
 
 203         if (size < sizeof(*data))
 
 205         data = (struct nfsctl_export*)buf;
 
 206         return exp_unexport(data);
 
 209 static ssize_t write_getfs(struct file *file, char *buf, size_t size)
 
 211         struct nfsctl_fsparm *data;
 
 212         struct sockaddr_in *sin;
 
 213         struct auth_domain *clp;
 
 215         struct knfsd_fh *res;
 
 217         if (size < sizeof(*data))
 
 219         data = (struct nfsctl_fsparm*)buf;
 
 220         err = -EPROTONOSUPPORT;
 
 221         if (data->gd_addr.sa_family != AF_INET)
 
 223         sin = (struct sockaddr_in *)&data->gd_addr;
 
 224         if (data->gd_maxlen > NFS3_FHSIZE)
 
 225                 data->gd_maxlen = NFS3_FHSIZE;
 
 227         res = (struct knfsd_fh*)buf;
 
 230         if (!(clp = auth_unix_lookup(sin->sin_addr)))
 
 233                 err = exp_rootfh(clp, data->gd_path, res, data->gd_maxlen);
 
 234                 auth_domain_put(clp);
 
 238                 err = res->fh_size + (int)&((struct knfsd_fh*)0)->fh_base;
 
 243 static ssize_t write_getfd(struct file *file, char *buf, size_t size)
 
 245         struct nfsctl_fdparm *data;
 
 246         struct sockaddr_in *sin;
 
 247         struct auth_domain *clp;
 
 252         if (size < sizeof(*data))
 
 254         data = (struct nfsctl_fdparm*)buf;
 
 255         err = -EPROTONOSUPPORT;
 
 256         if (data->gd_addr.sa_family != AF_INET)
 
 259         if (data->gd_version < 2 || data->gd_version > NFSSVC_MAXVERS)
 
 263         sin = (struct sockaddr_in *)&data->gd_addr;
 
 265         if (!(clp = auth_unix_lookup(sin->sin_addr)))
 
 268                 err = exp_rootfh(clp, data->gd_path, &fh, NFS_FHSIZE);
 
 269                 auth_domain_put(clp);
 
 274                 memset(res,0, NFS_FHSIZE);
 
 275                 memcpy(res, &fh.fh_base, fh.fh_size);
 
 282 static ssize_t write_filehandle(struct file *file, char *buf, size_t size)
 
 285          *   domain path maxsize
 
 289          * qword quoting is used, so filehandle will be \x....
 
 295         struct auth_domain *dom;
 
 298         if (buf[size-1] != '\n')
 
 303         len = qword_get(&mesg, dname, size);
 
 304         if (len <= 0) return -EINVAL;
 
 307         len = qword_get(&mesg, path, size);
 
 308         if (len <= 0) return -EINVAL;
 
 310         len = get_int(&mesg, &maxsize);
 
 314         if (maxsize < NFS_FHSIZE)
 
 316         if (maxsize > NFS3_FHSIZE)
 
 317                 maxsize = NFS3_FHSIZE;
 
 319         if (qword_get(&mesg, mesg, size)>0)
 
 322         /* we have all the words, they are in buf.. */
 
 323         dom = unix_domain_find(dname);
 
 327         len = exp_rootfh(dom, path, &fh,  maxsize);
 
 328         auth_domain_put(dom);
 
 332         mesg = buf; len = SIMPLE_TRANSACTION_LIMIT;
 
 333         qword_addhex(&mesg, &len, (char*)&fh.fh_base, fh.fh_size);
 
 338 extern int nfsd_nrthreads(void);
 
 340 static ssize_t write_threads(struct file *file, char *buf, size_t size)
 
 342         /* if size > 0, look for a number of threads and call nfsd_svc
 
 343          * then write out number of threads as reply
 
 349                 rv = get_int(&mesg, &newthreads);
 
 354                 rv = nfsd_svc(2049, newthreads);
 
 358         sprintf(buf, "%d\n", nfsd_nrthreads());
 
 362 static ssize_t write_versions(struct file *file, char *buf, size_t size)
 
 366          *   [-/+]vers [-/+]vers ...
 
 377                 if (buf[size-1] != '\n')
 
 382                 len = qword_get(&mesg, vers, size);
 
 383                 if (len <= 0) return -EINVAL;
 
 386                         if (sign == '+' || sign == '-')
 
 387                                 num = simple_strtol((vers+1), NULL, 0);
 
 389                                 num = simple_strtol(vers, NULL, 0);
 
 395                                         NFSCTL_VERSET(nfsd_versbits, num);
 
 397                                         NFSCTL_VERUNSET(nfsd_versbits, num);
 
 404                 } while ((len = qword_get(&mesg, vers, size)) > 0);
 
 405                 /* If all get turned off, turn them back on, as
 
 406                  * having no versions is BAD
 
 408                 if ((nfsd_versbits & NFSCTL_VERALL)==0)
 
 409                         nfsd_versbits = NFSCTL_VERALL;
 
 411         /* Now write current state into reply buffer */
 
 414         for (num=2 ; num <= 4 ; num++)
 
 415                 if (NFSCTL_VERISSET(NFSCTL_VERALL, num)) {
 
 416                         len += sprintf(buf+len, "%s%c%d", sep,
 
 417                                        NFSCTL_VERISSET(nfsd_versbits, num)?'+':'-',
 
 421         len += sprintf(buf+len, "\n");
 
 425 #ifdef CONFIG_NFSD_V4
 
 426 extern time_t nfs4_leasetime(void);
 
 428 static ssize_t write_leasetime(struct file *file, char *buf, size_t size)
 
 430         /* if size > 10 seconds, call
 
 431          * nfs4_reset_lease() then write out the new lease (seconds) as reply
 
 438                 rv = get_int(&mesg, &lease);
 
 441                 if (lease < 10 || lease > 3600)
 
 443                 nfs4_reset_lease(lease);
 
 445         sprintf(buf, "%ld\n", nfs4_lease_time());
 
 449 static ssize_t write_recoverydir(struct file *file, char *buf, size_t size)
 
 455         if (size > PATH_MAX || buf[size-1] != '\n')
 
 460         len = qword_get(&mesg, recdir, size);
 
 464         status = nfs4_reset_recoverydir(recdir);
 
 469 /*----------------------------------------------------------------------------*/
 
 471  *      populating the filesystem.
 
 474 static int nfsd_fill_super(struct super_block * sb, void * data, int silent)
 
 476         static struct tree_descr nfsd_files[] = {
 
 477                 [NFSD_Svc] = {".svc", &transaction_ops, S_IWUSR},
 
 478                 [NFSD_Add] = {".add", &transaction_ops, S_IWUSR},
 
 479                 [NFSD_Del] = {".del", &transaction_ops, S_IWUSR},
 
 480                 [NFSD_Export] = {".export", &transaction_ops, S_IWUSR},
 
 481                 [NFSD_Unexport] = {".unexport", &transaction_ops, S_IWUSR},
 
 482                 [NFSD_Getfd] = {".getfd", &transaction_ops, S_IWUSR|S_IRUSR},
 
 483                 [NFSD_Getfs] = {".getfs", &transaction_ops, S_IWUSR|S_IRUSR},
 
 484                 [NFSD_List] = {"exports", &exports_operations, S_IRUGO},
 
 485                 [NFSD_Fh] = {"filehandle", &transaction_ops, S_IWUSR|S_IRUSR},
 
 486                 [NFSD_Threads] = {"threads", &transaction_ops, S_IWUSR|S_IRUSR},
 
 487                 [NFSD_Versions] = {"versions", &transaction_ops, S_IWUSR|S_IRUSR},
 
 488 #ifdef CONFIG_NFSD_V4
 
 489                 [NFSD_Leasetime] = {"nfsv4leasetime", &transaction_ops, S_IWUSR|S_IRUSR},
 
 490                 [NFSD_RecoveryDir] = {"nfsv4recoverydir", &transaction_ops, S_IWUSR|S_IRUSR},
 
 494         return simple_fill_super(sb, 0x6e667364, nfsd_files);
 
 497 static struct super_block *nfsd_get_sb(struct file_system_type *fs_type,
 
 498         int flags, const char *dev_name, void *data)
 
 500         return get_sb_single(fs_type, flags, data, nfsd_fill_super);
 
 503 static struct file_system_type nfsd_fs_type = {
 
 504         .owner          = THIS_MODULE,
 
 506         .get_sb         = nfsd_get_sb,
 
 507         .kill_sb        = kill_litter_super,
 
 510 static int __init init_nfsd(void)
 
 513         printk(KERN_INFO "Installing knfsd (copyright (C) 1996 okir@monad.swb.de).\n");
 
 515         nfsd_stat_init();       /* Statistics */
 
 516         nfsd_cache_init();      /* RPC reply cache */
 
 517         nfsd_export_init();     /* Exports table */
 
 518         nfsd_lockd_init();      /* lockd->nfsd callbacks */
 
 519         nfs4_state_init();      /* NFSv4 locking state */
 
 520         nfsd_idmap_init();      /* Name to ID mapping */
 
 521         if (proc_mkdir("fs/nfs", NULL)) {
 
 522                 struct proc_dir_entry *entry;
 
 523                 entry = create_proc_entry("fs/nfs/exports", 0, NULL);
 
 525                         entry->proc_fops =  &exports_operations;
 
 527         retval = register_filesystem(&nfsd_fs_type);
 
 529                 nfsd_export_shutdown();
 
 530                 nfsd_cache_shutdown();
 
 531                 remove_proc_entry("fs/nfs/exports", NULL);
 
 532                 remove_proc_entry("fs/nfs", NULL);
 
 533                 nfsd_stat_shutdown();
 
 534                 nfsd_lockd_shutdown();
 
 539 static void __exit exit_nfsd(void)
 
 541         nfsd_export_shutdown();
 
 542         nfsd_cache_shutdown();
 
 543         remove_proc_entry("fs/nfs/exports", NULL);
 
 544         remove_proc_entry("fs/nfs", NULL);
 
 545         nfsd_stat_shutdown();
 
 546         nfsd_lockd_shutdown();
 
 547         nfsd_idmap_shutdown();
 
 548         unregister_filesystem(&nfsd_fs_type);
 
 551 MODULE_AUTHOR("Olaf Kirch <okir@monad.swb.de>");
 
 552 MODULE_LICENSE("GPL");
 
 553 module_init(init_nfsd)
 
 554 module_exit(exit_nfsd)