1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1989, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to Berkeley by
|
|
|
|
* Rick Macklem at The University of Guelph.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the University of
|
|
|
|
* California, Berkeley and its contributors.
|
|
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
1997-02-10 02:22:35 +00:00
|
|
|
* @(#)nfs_syscalls.c 8.5 (Berkeley) 3/30/95
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
1995-11-14 05:16:37 +00:00
|
|
|
#include <sys/sysproto.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/kernel.h>
|
1996-11-06 10:53:16 +00:00
|
|
|
#include <sys/sysctl.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/file.h>
|
1996-04-30 23:26:52 +00:00
|
|
|
#include <sys/filedesc.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/vnode.h>
|
1997-10-12 20:26:33 +00:00
|
|
|
#include <sys/malloc.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/proc.h>
|
2000-05-05 09:59:14 +00:00
|
|
|
#include <sys/bio.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/buf.h>
|
|
|
|
#include <sys/mbuf.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/socketvar.h>
|
|
|
|
#include <sys/domain.h>
|
|
|
|
#include <sys/protosw.h>
|
|
|
|
#include <sys/namei.h>
|
2001-09-18 23:32:09 +00:00
|
|
|
#include <sys/unistd.h>
|
|
|
|
#include <sys/kthread.h>
|
2001-04-17 20:45:23 +00:00
|
|
|
#include <sys/fcntl.h>
|
|
|
|
#include <sys/lockf.h>
|
2001-09-18 23:32:09 +00:00
|
|
|
#include <sys/mutex.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <netinet/tcp.h>
|
1995-06-27 11:07:30 +00:00
|
|
|
#include <nfs/xdr_subs.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <nfs/rpcv2.h>
|
1995-06-27 11:07:30 +00:00
|
|
|
#include <nfs/nfsproto.h>
|
2001-09-18 23:32:09 +00:00
|
|
|
#include <nfsclient/nfs.h>
|
|
|
|
#include <nfsclient/nfsm_subs.h>
|
|
|
|
#include <nfsclient/nfsmount.h>
|
|
|
|
#include <nfsclient/nfsnode.h>
|
|
|
|
#include <nfsclient/nfs_lock.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1997-10-12 20:26:33 +00:00
|
|
|
static MALLOC_DEFINE(M_NFSSVC, "NFS srvsock", "Nfs server structure");
|
1997-10-11 18:31:40 +00:00
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
static void nfssvc_iod(void *);
|
1994-05-24 10:09:53 +00:00
|
|
|
|
|
|
|
#define TRUE 1
|
|
|
|
#define FALSE 0
|
|
|
|
|
|
|
|
static int nfs_asyncdaemon[NFS_MAXASYNCDAEMON];
|
1995-11-21 15:51:39 +00:00
|
|
|
|
1999-02-18 09:19:41 +00:00
|
|
|
SYSCTL_DECL(_vfs_nfs);
|
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
static void
|
|
|
|
nfsiod_setup(void *dummy)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
2001-09-18 23:32:09 +00:00
|
|
|
int i;
|
1994-05-24 10:09:53 +00:00
|
|
|
int error;
|
2001-09-18 23:32:09 +00:00
|
|
|
struct proc *p;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
error = kthread_create(nfssvc_iod, NULL, &p, RFHIGHPID,
|
|
|
|
"nfsiod %d", i);
|
1994-05-24 10:09:53 +00:00
|
|
|
if (error)
|
2001-09-18 23:32:09 +00:00
|
|
|
panic("nfsiod_setup: kthread_create error %d", error);
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
|
|
|
}
|
2001-09-18 23:32:09 +00:00
|
|
|
SYSINIT(nfsiod, SI_SUB_KTHREAD_IDLE, SI_ORDER_ANY, nfsiod_setup, NULL);
|
1998-05-31 17:27:58 +00:00
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
static int nfs_defect = 0;
|
|
|
|
SYSCTL_INT(_vfs_nfs, OID_AUTO, defect, CTLFLAG_RW, &nfs_defect, 0, "");
|
1998-05-31 17:27:58 +00:00
|
|
|
|
1999-02-25 00:03:51 +00:00
|
|
|
int
|
2001-09-18 23:32:09 +00:00
|
|
|
nfsclnt(struct thread *td, struct nfsclnt_args *uap)
|
1999-02-25 00:03:51 +00:00
|
|
|
{
|
2001-09-18 23:32:09 +00:00
|
|
|
struct lockd_ans la;
|
|
|
|
int error;
|
1998-05-31 17:27:58 +00:00
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
if ((uap->flag & NFSCLNT_LOCKDANS) != 0) {
|
|
|
|
error = copyin(uap->argp, &la, sizeof(la));
|
|
|
|
return (error != 0 ? error : nfslockdans(td->td_proc, &la));
|
1998-05-31 17:27:58 +00:00
|
|
|
}
|
2001-09-18 23:32:09 +00:00
|
|
|
return EINVAL;
|
1998-05-31 17:27:58 +00:00
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Asynchronous I/O daemons for client nfs.
|
|
|
|
* They do read-ahead and write-behind operations on the block I/O cache.
|
|
|
|
* Never returns unless it fails or gets killed.
|
|
|
|
*/
|
2001-09-18 23:32:09 +00:00
|
|
|
static void
|
|
|
|
nfssvc_iod(void *dummy)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
2001-09-18 23:32:09 +00:00
|
|
|
struct buf *bp;
|
|
|
|
int i, myiod;
|
1996-11-06 10:53:16 +00:00
|
|
|
struct nfsmount *nmp;
|
1997-11-07 08:53:44 +00:00
|
|
|
int error = 0;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
2001-09-18 23:32:09 +00:00
|
|
|
mtx_lock(&Giant);
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Assign my position or return error if too many already running
|
|
|
|
*/
|
|
|
|
myiod = -1;
|
|
|
|
for (i = 0; i < NFS_MAXASYNCDAEMON; i++)
|
|
|
|
if (nfs_asyncdaemon[i] == 0) {
|
|
|
|
nfs_asyncdaemon[i]++;
|
|
|
|
myiod = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (myiod == -1)
|
2001-09-18 23:32:09 +00:00
|
|
|
return /* XXX (EBUSY) */;
|
1994-05-24 10:09:53 +00:00
|
|
|
nfs_numasync++;
|
|
|
|
/*
|
|
|
|
* Just loop around doin our stuff until SIGKILL
|
|
|
|
*/
|
|
|
|
for (;;) {
|
1996-11-06 10:53:16 +00:00
|
|
|
while (((nmp = nfs_iodmount[myiod]) == NULL
|
2001-09-18 23:32:09 +00:00
|
|
|
|| !TAILQ_FIRST(&nmp->nm_bufq))
|
1996-11-06 10:53:16 +00:00
|
|
|
&& error == 0) {
|
|
|
|
if (nmp)
|
|
|
|
nmp->nm_bufqiods--;
|
2001-09-18 23:32:09 +00:00
|
|
|
nfs_iodwant[myiod] = curthread->td_proc;
|
1996-11-06 10:53:16 +00:00
|
|
|
nfs_iodmount[myiod] = NULL;
|
1995-06-27 11:07:30 +00:00
|
|
|
error = tsleep((caddr_t)&nfs_iodwant[myiod],
|
|
|
|
PWAIT | PCATCH, "nfsidl", 0);
|
|
|
|
}
|
|
|
|
if (error) {
|
|
|
|
nfs_asyncdaemon[myiod] = 0;
|
1998-05-31 17:27:58 +00:00
|
|
|
if (nmp)
|
|
|
|
nmp->nm_bufqiods--;
|
1997-06-25 21:07:26 +00:00
|
|
|
nfs_iodwant[myiod] = NULL;
|
1996-11-06 10:53:16 +00:00
|
|
|
nfs_iodmount[myiod] = NULL;
|
1995-06-27 11:07:30 +00:00
|
|
|
nfs_numasync--;
|
2001-09-18 23:32:09 +00:00
|
|
|
return /* XXX (error) */;
|
1995-06-27 11:07:30 +00:00
|
|
|
}
|
2001-09-18 23:32:09 +00:00
|
|
|
while ((bp = TAILQ_FIRST(&nmp->nm_bufq)) != NULL) {
|
1996-11-06 10:53:16 +00:00
|
|
|
/* Take one off the front of the list */
|
|
|
|
TAILQ_REMOVE(&nmp->nm_bufq, bp, b_freelist);
|
|
|
|
nmp->nm_bufqlen--;
|
2000-01-05 05:11:37 +00:00
|
|
|
if (nmp->nm_bufqwant && nmp->nm_bufqlen <= nfs_numasync) {
|
1996-11-06 10:53:16 +00:00
|
|
|
nmp->nm_bufqwant = FALSE;
|
|
|
|
wakeup(&nmp->nm_bufq);
|
|
|
|
}
|
2000-03-20 10:44:49 +00:00
|
|
|
if (bp->b_iocmd == BIO_READ)
|
2001-09-12 08:38:13 +00:00
|
|
|
(void) nfs_doio(bp, bp->b_rcred, (struct thread *)0);
|
1996-11-06 10:53:16 +00:00
|
|
|
else
|
2001-09-12 08:38:13 +00:00
|
|
|
(void) nfs_doio(bp, bp->b_wcred, (struct thread *)0);
|
1996-11-06 10:53:16 +00:00
|
|
|
/*
|
|
|
|
* If there are more than one iod on this mount, then defect
|
|
|
|
* so that the iods can be shared out fairly between the mounts
|
|
|
|
*/
|
|
|
|
if (nfs_defect && nmp->nm_bufqiods > 1) {
|
|
|
|
NFS_DPF(ASYNCIO,
|
|
|
|
("nfssvc_iod: iod %d defecting from mount %p\n",
|
|
|
|
myiod, nmp));
|
|
|
|
nfs_iodmount[myiod] = NULL;
|
|
|
|
nmp->nm_bufqiods--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
|
|
|
}
|