2005-01-05 23:35:00 +00:00
|
|
|
/*-
|
1998-09-02 19:09:53 +00:00
|
|
|
* Coda: an Experimental Distributed File System
|
|
|
|
* Release 3.1
|
2008-02-10 11:18:12 +00:00
|
|
|
*
|
1998-09-02 19:09:53 +00:00
|
|
|
* Copyright (c) 1987-1998 Carnegie Mellon University
|
|
|
|
* All Rights Reserved
|
2008-02-10 11:18:12 +00:00
|
|
|
*
|
1998-09-02 19:09:53 +00:00
|
|
|
* Permission to use, copy, modify and distribute this software and its
|
|
|
|
* documentation is hereby granted, provided that both the copyright
|
|
|
|
* notice and this permission notice appear in all copies of the
|
|
|
|
* software, derivative works or modified versions, and any portions
|
|
|
|
* thereof, and that both notices appear in supporting documentation, and
|
|
|
|
* that credit is given to Carnegie Mellon University in all documents
|
|
|
|
* and publicity pertaining to direct or indirect use of this code or its
|
|
|
|
* derivatives.
|
2008-02-10 11:18:12 +00:00
|
|
|
*
|
1998-09-02 19:09:53 +00:00
|
|
|
* CODA IS AN EXPERIMENTAL SOFTWARE SYSTEM AND IS KNOWN TO HAVE BUGS,
|
|
|
|
* SOME OF WHICH MAY HAVE SERIOUS CONSEQUENCES. CARNEGIE MELLON ALLOWS
|
|
|
|
* FREE USE OF THIS SOFTWARE IN ITS "AS IS" CONDITION. CARNEGIE MELLON
|
|
|
|
* DISCLAIMS ANY LIABILITY OF ANY KIND FOR ANY DAMAGES WHATSOEVER
|
|
|
|
* RESULTING DIRECTLY OR INDIRECTLY FROM THE USE OF THIS SOFTWARE OR OF
|
|
|
|
* ANY DERIVATIVE WORK.
|
2008-02-10 11:18:12 +00:00
|
|
|
*
|
1998-09-02 19:09:53 +00:00
|
|
|
* Carnegie Mellon encourages users of this software to return any
|
|
|
|
* improvements or extensions that they make, and to grant Carnegie
|
|
|
|
* Mellon the rights to redistribute these changes without encumbrance.
|
2008-02-10 11:18:12 +00:00
|
|
|
*
|
1998-09-13 13:57:59 +00:00
|
|
|
* @(#) src/sys/coda/coda_psdev.c,v 1.1.1.1 1998/08/29 21:14:52 rvb Exp $
|
1998-09-02 19:09:53 +00:00
|
|
|
*/
|
2005-01-05 23:35:00 +00:00
|
|
|
/*-
|
1998-08-29 21:14:52 +00:00
|
|
|
* Mach Operating System
|
|
|
|
* Copyright (c) 1989 Carnegie-Mellon University
|
|
|
|
* All rights reserved. The CMU software License Agreement specifies
|
|
|
|
* the terms and conditions for use and redistribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2002-05-16 21:28:32 +00:00
|
|
|
* This code was written for the Coda filesystem at Carnegie Mellon
|
1998-08-29 21:14:52 +00:00
|
|
|
* University. Contributers include David Steere, James Kistler, and
|
|
|
|
* M. Satyanarayanan. */
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* These routines define the psuedo device for communication between Coda's
|
|
|
|
* Venus and Minicache in Mach 2.6. They used to be in cfs_subr.c, but I
|
|
|
|
* moved them to make it easier to port the Minicache without porting coda.
|
|
|
|
* -- DCS 10/12/94
|
1998-08-29 21:14:52 +00:00
|
|
|
*/
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* These routines are the device entry points for Venus.
|
|
|
|
*/
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2003-06-10 21:29:12 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
1998-08-29 21:14:52 +00:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
Various further non-functional cleanups to coda:
- Rename print_vattr to coda_print_vattr and make static, rename
print_cred to coda_print_cred.
- Remove unused coda_vop_nop.
- Add XXX comment because coda_readdir forwards to the cache vnode's
readdir rather than venus_readdir, and annotate venus_readdir as
unused.
- Rename vc_nb_* to vc_*.
- Use d_open_t, d_close_t, d_read_t, d_write_t, d_ioctl_t and d_poll_t
for prototyping vc_* as that is the intent, don't use our own
definitions.
- Rename coda_nb_statfs to coda_statfs, rename NB_SFS_SIZ to
CODA_SFS_SIZ.
- Replace one more OBE reference to NetBSD with a reference to FreeBSD.
- Tidy up a little vertical whitespace here and there.
- Annotate coda_nc_zapvnode as unused.
- Remove unused vcodattach.
- Annotate VM_INTR as unused.
- Annotate that coda_fhtovp is unused and doesn't match the FreeBSD
prototype, so isn't hooked up to vfs_fhtovp. If we want NFS export of
Coda to work someday, this needs to be fixed.
- Remove unused getNewVnode.
- Remove unused coda_vget, coda_init, coda_quotactl prototypes.
MFC after: 1 month
2008-02-09 12:49:18 +00:00
|
|
|
#include <sys/conf.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/ioccom.h>
|
1998-08-29 21:14:52 +00:00
|
|
|
#include <sys/kernel.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/lock.h>
|
1998-09-02 19:09:53 +00:00
|
|
|
#include <sys/malloc.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/file.h> /* must come after sys/malloc.h */
|
1998-08-29 21:14:52 +00:00
|
|
|
#include <sys/mount.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/mutex.h>
|
1998-08-29 21:14:52 +00:00
|
|
|
#include <sys/poll.h>
|
2001-05-01 08:13:21 +00:00
|
|
|
#include <sys/proc.h>
|
2007-07-11 21:32:08 +00:00
|
|
|
#include <sys/filedesc.h>
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2007-07-12 21:04:58 +00:00
|
|
|
#include <fs/coda/coda.h>
|
|
|
|
#include <fs/coda/cnode.h>
|
|
|
|
#include <fs/coda/coda_io.h>
|
|
|
|
#include <fs/coda/coda_psdev.h>
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Variables to determine how Coda sleeps and whether or not it is
|
|
|
|
* interruptible when it does sleep waiting for Venus.
|
|
|
|
*/
|
2008-02-15 13:31:35 +00:00
|
|
|
/* #define CTL_C */
|
1998-09-02 19:09:53 +00:00
|
|
|
|
1999-09-29 15:03:48 +00:00
|
|
|
#ifdef CTL_C
|
|
|
|
#include <sys/signalvar.h>
|
|
|
|
#endif
|
|
|
|
|
1998-09-11 18:50:17 +00:00
|
|
|
int coda_psdev_print_entry = 0;
|
2008-02-10 11:18:12 +00:00
|
|
|
static int outstanding_upcalls = 0;
|
1998-10-28 20:31:13 +00:00
|
|
|
int coda_call_sleep = PZERO - 1;
|
2008-02-10 11:18:12 +00:00
|
|
|
#ifdef CTL_C
|
1998-10-28 20:31:13 +00:00
|
|
|
int coda_pcatch = PCATCH;
|
|
|
|
#else
|
|
|
|
#endif
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
#define ENTRY do { \
|
|
|
|
if (coda_psdev_print_entry) \
|
|
|
|
myprintf(("Entered %s\n", __func__)); \
|
|
|
|
} while (0)
|
1998-08-29 21:14:52 +00:00
|
|
|
|
|
|
|
struct vmsg {
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_ENTRY(vmsg) vm_chain;
|
2008-02-10 11:18:12 +00:00
|
|
|
caddr_t vm_data;
|
|
|
|
u_short vm_flags;
|
|
|
|
u_short vm_inSize; /* Size is at most 5000 bytes */
|
|
|
|
u_short vm_outSize;
|
|
|
|
u_short vm_opcode; /* Copied from data to save ptr deref */
|
|
|
|
int vm_unique;
|
|
|
|
caddr_t vm_sleep; /* Not used by Mach. */
|
1998-08-29 21:14:52 +00:00
|
|
|
};
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
#define VM_READ 1
|
|
|
|
#define VM_WRITE 2
|
|
|
|
#define VM_INTR 4 /* Unused. */
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
int
|
|
|
|
vc_open(struct cdev *dev, int flag, int mode, struct thread *td)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
struct vcomm *vcp;
|
|
|
|
struct coda_mntinfo *mnt;
|
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
mnt = dev2coda_mntinfo(dev);
|
|
|
|
KASSERT(mnt, ("Coda: tried to open uninitialized cfs device"));
|
|
|
|
vcp = &mnt->mi_vcomm;
|
|
|
|
if (VC_OPEN(vcp))
|
|
|
|
return (EBUSY);
|
|
|
|
bzero(&(vcp->vc_selproc), sizeof (struct selinfo));
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_INIT(&vcp->vc_requests);
|
|
|
|
TAILQ_INIT(&vcp->vc_replies);
|
2008-02-10 11:18:12 +00:00
|
|
|
MARK_VC_OPEN(vcp);
|
|
|
|
mnt->mi_vfsp = NULL;
|
|
|
|
mnt->mi_rootvp = NULL;
|
|
|
|
return (0);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
int
|
|
|
|
vc_close(struct cdev *dev, int flag, int mode, struct thread *td)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
struct vcomm *vcp;
|
|
|
|
struct vmsg *vmp, *nvmp = NULL;
|
|
|
|
struct coda_mntinfo *mi;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
mi = dev2coda_mntinfo(dev);
|
|
|
|
KASSERT(mi, ("Coda: closing unknown cfs device"));
|
|
|
|
vcp = &mi->mi_vcomm;
|
|
|
|
KASSERT(VC_OPEN(vcp), ("Coda: closing unopened cfs device"));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Prevent future operations on this vfs from succeeding by
|
|
|
|
* auto-unmounting any vfs mounted via this device. This frees user
|
|
|
|
* or sysadm from having to remember where all mount points are
|
|
|
|
* located. Put this before WAKEUPs to avoid queuing new messages
|
|
|
|
* between the WAKEUP and the unmount (which can happen if we're
|
|
|
|
* unlucky).
|
|
|
|
*/
|
|
|
|
if (mi->mi_rootvp == NULL) {
|
|
|
|
/*
|
|
|
|
* Just a simple open/close with no mount.
|
|
|
|
*/
|
|
|
|
MARK_VC_CLOSED(vcp);
|
|
|
|
return (0);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
1998-10-28 20:31:13 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Let unmount know this is for real.
|
|
|
|
*/
|
|
|
|
VTOC(mi->mi_rootvp)->c_flags |= C_UNMOUNTING;
|
|
|
|
coda_unmounting(mi->mi_vfsp);
|
1998-10-28 20:31:13 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Wakeup clients so they can return.
|
|
|
|
*/
|
|
|
|
outstanding_upcalls = 0;
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_FOREACH_SAFE(vmp, &vcp->vc_requests, vm_chain, nvmp) {
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Free signal request messages and don't wakeup cause no one
|
|
|
|
* is waiting.
|
|
|
|
*/
|
|
|
|
if (vmp->vm_opcode == CODA_SIGNAL) {
|
|
|
|
CODA_FREE((caddr_t)vmp->vm_data,
|
|
|
|
(u_int)VC_IN_NO_DATA);
|
|
|
|
CODA_FREE((caddr_t)vmp, (u_int)sizeof(struct vmsg));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
outstanding_upcalls++;
|
|
|
|
wakeup(&vmp->vm_sleep);
|
|
|
|
}
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_FOREACH(vmp, &vcp->vc_replies, vm_chain) {
|
2008-02-10 11:18:12 +00:00
|
|
|
outstanding_upcalls++;
|
|
|
|
wakeup(&vmp->vm_sleep);
|
|
|
|
}
|
|
|
|
MARK_VC_CLOSED(vcp);
|
|
|
|
if (outstanding_upcalls) {
|
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
printf("presleep: outstanding_upcalls = %d\n",
|
|
|
|
outstanding_upcalls);
|
1998-10-28 20:31:13 +00:00
|
|
|
#endif
|
2008-02-10 11:18:12 +00:00
|
|
|
(void) tsleep(&outstanding_upcalls, coda_call_sleep,
|
|
|
|
"coda_umount", 0);
|
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
printf("postsleep: outstanding_upcalls = %d\n",
|
|
|
|
outstanding_upcalls);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
err = dounmount(mi->mi_vfsp, flag, td);
|
|
|
|
if (err)
|
|
|
|
myprintf(("Error %d unmounting vfs in vcclose(%s)\n", err,
|
|
|
|
devtoname(dev)));
|
|
|
|
return (0);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
int
|
|
|
|
vc_read(struct cdev *dev, struct uio *uiop, int flag)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
struct vcomm *vcp;
|
|
|
|
struct vmsg *vmp;
|
|
|
|
int error = 0;
|
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
vcp = &dev2coda_mntinfo(dev)->mi_vcomm;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get message at head of request queue.
|
|
|
|
*/
|
2008-02-17 14:33:28 +00:00
|
|
|
vmp = TAILQ_FIRST(&vcp->vc_requests);
|
|
|
|
if (vmp == NULL)
|
2008-02-10 11:18:12 +00:00
|
|
|
return (0); /* Nothing to read */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Move the input args into userspace.
|
2008-02-17 14:33:28 +00:00
|
|
|
*
|
|
|
|
* XXXRW: This is not safe in the presence of >1 reader, as vmp is
|
|
|
|
* still on the head of the list.
|
2008-02-10 11:18:12 +00:00
|
|
|
*/
|
|
|
|
uiop->uio_rw = UIO_READ;
|
|
|
|
error = uiomove(vmp->vm_data, vmp->vm_inSize, uiop);
|
|
|
|
if (error) {
|
|
|
|
myprintf(("vcread: error (%d) on uiomove\n", error));
|
|
|
|
error = EINVAL;
|
|
|
|
}
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_REMOVE(&vcp->vc_requests, vmp, vm_chain);
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* If request was a signal, free up the message and don't enqueue it
|
|
|
|
* in the reply queue.
|
|
|
|
*/
|
|
|
|
if (vmp->vm_opcode == CODA_SIGNAL) {
|
|
|
|
if (codadebug)
|
|
|
|
myprintf(("vcread: signal msg (%d, %d)\n",
|
|
|
|
vmp->vm_opcode, vmp->vm_unique));
|
|
|
|
CODA_FREE((caddr_t)vmp->vm_data, (u_int)VC_IN_NO_DATA);
|
|
|
|
CODA_FREE((caddr_t)vmp, (u_int)sizeof(struct vmsg));
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
vmp->vm_flags |= VM_READ;
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_INSERT_TAIL(&vcp->vc_replies, vmp, vm_chain);
|
2008-02-10 11:18:12 +00:00
|
|
|
return (error);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2008-02-10 11:18:12 +00:00
|
|
|
vc_write(struct cdev *dev, struct uio *uiop, int flag)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
struct vcomm *vcp;
|
|
|
|
struct vmsg *vmp;
|
|
|
|
struct coda_out_hdr *out;
|
|
|
|
u_long seq;
|
|
|
|
u_long opcode;
|
|
|
|
int buf[2];
|
|
|
|
int error = 0;
|
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
vcp = &dev2coda_mntinfo(dev)->mi_vcomm;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Peek at the opcode, unique without transfering the data.
|
|
|
|
*/
|
1998-08-29 21:14:52 +00:00
|
|
|
uiop->uio_rw = UIO_WRITE;
|
2008-02-10 11:18:12 +00:00
|
|
|
error = uiomove((caddr_t)buf, sizeof(int) * 2, uiop);
|
1998-08-29 21:14:52 +00:00
|
|
|
if (error) {
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("vcwrite: error (%d) on uiomove\n", error));
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
|
|
|
opcode = buf[0];
|
|
|
|
seq = buf[1];
|
1998-09-11 18:50:17 +00:00
|
|
|
if (codadebug)
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("vcwrite got a call for %ld.%ld\n", opcode, seq));
|
|
|
|
if (DOWNCALL(opcode)) {
|
|
|
|
union outputArgs pbuf;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the rest of the data.
|
|
|
|
*/
|
|
|
|
uiop->uio_rw = UIO_WRITE;
|
|
|
|
error = uiomove((caddr_t)&pbuf.coda_purgeuser.oh.result,
|
|
|
|
sizeof(pbuf) - (sizeof(int)*2), uiop);
|
|
|
|
if (error) {
|
|
|
|
myprintf(("vcwrite: error (%d) on uiomove (Op %ld "
|
|
|
|
"seq %ld)\n", error, opcode, seq));
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
Rather than having the Coda module use its own namecache, use the global
VFS namecache, as is done by the Coda module on Linux. Unlike the Coda
namecache, the global VFS namecache isn't tagged by credential, so use
ore conservative flushing behavior (for now) when CODA_PURGEUSER is
issued by Venus.
This improves overall integration with the FreeBSD VFS, including
allowing __getcwd() to work better, procfs/procstat monitoring, and so
on. This improves shell behavior in many cases, and improves ".."
handling. It may lead to some slowdown until we've implemented a
specific access cache, which should net improve performance, but in the
mean time, lookup access control now always goes to Venus, whereas
previously it didn't.
MFC after: 1 month
2008-02-13 13:06:22 +00:00
|
|
|
return (handleDownCall(dev2coda_mntinfo(dev), opcode, &pbuf));
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Look for the message on the (waiting for) reply queue.
|
|
|
|
*/
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_FOREACH(vmp, &vcp->vc_replies, vm_chain) {
|
2008-02-10 11:18:12 +00:00
|
|
|
if (vmp->vm_unique == seq)
|
|
|
|
break;
|
|
|
|
}
|
2008-02-17 14:33:28 +00:00
|
|
|
if (vmp == NULL) {
|
2008-02-10 11:18:12 +00:00
|
|
|
if (codadebug)
|
|
|
|
myprintf(("vcwrite: msg (%ld, %ld) not found\n",
|
|
|
|
opcode, seq));
|
|
|
|
return (ESRCH);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove the message from the reply queue.
|
|
|
|
*/
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_REMOVE(&vcp->vc_replies, vmp, vm_chain);
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Move data into response buffer.
|
|
|
|
*/
|
|
|
|
out = (struct coda_out_hdr *)vmp->vm_data;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Don't need to copy opcode and uniquifier.
|
|
|
|
*
|
|
|
|
* Get the rest of the data.
|
|
|
|
*/
|
|
|
|
if (vmp->vm_outSize < uiop->uio_resid) {
|
2009-06-25 18:46:30 +00:00
|
|
|
myprintf(("vcwrite: more data than asked for (%d < %zd)\n",
|
2008-02-10 11:18:12 +00:00
|
|
|
vmp->vm_outSize, uiop->uio_resid));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Notify caller of the error.
|
|
|
|
*/
|
|
|
|
wakeup(&vmp->vm_sleep);
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Save the value.
|
|
|
|
*/
|
|
|
|
buf[0] = uiop->uio_resid;
|
|
|
|
uiop->uio_rw = UIO_WRITE;
|
|
|
|
error = uiomove((caddr_t) &out->result, vmp->vm_outSize -
|
|
|
|
(sizeof(int) * 2), uiop);
|
|
|
|
if (error) {
|
|
|
|
myprintf(("vcwrite: error (%d) on uiomove (op %ld seq %ld)\n",
|
|
|
|
error, opcode, seq));
|
|
|
|
return (EINVAL);
|
2007-07-11 21:32:08 +00:00
|
|
|
}
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* I don't think these are used, but just in case.
|
|
|
|
*
|
|
|
|
* XXX - aren't these two already correct? -bnoble
|
|
|
|
*/
|
|
|
|
out->opcode = opcode;
|
|
|
|
out->unique = seq;
|
|
|
|
vmp->vm_outSize = buf[0]; /* Amount of data transferred? */
|
|
|
|
vmp->vm_flags |= VM_WRITE;
|
|
|
|
error = 0;
|
|
|
|
if (opcode == CODA_OPEN_BY_FD) {
|
|
|
|
struct coda_open_by_fd_out *tmp =
|
|
|
|
(struct coda_open_by_fd_out *)out;
|
|
|
|
struct file *fp;
|
|
|
|
struct vnode *vp = NULL;
|
|
|
|
|
|
|
|
if (tmp->oh.result == 0) {
|
|
|
|
error = getvnode(uiop->uio_td->td_proc->p_fd,
|
|
|
|
tmp->fd, &fp);
|
|
|
|
if (!error) {
|
|
|
|
/*
|
|
|
|
* XXX: Since the whole driver runs with
|
|
|
|
* Giant, don't actually need to acquire it
|
|
|
|
* explicitly here yet.
|
|
|
|
*/
|
|
|
|
mtx_lock(&Giant);
|
|
|
|
vp = fp->f_vnode;
|
|
|
|
VREF(vp);
|
|
|
|
fdrop(fp, uiop->uio_td);
|
|
|
|
mtx_unlock(&Giant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tmp->vp = vp;
|
|
|
|
}
|
|
|
|
wakeup(&vmp->vm_sleep);
|
|
|
|
return (error);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2008-02-10 11:18:12 +00:00
|
|
|
vc_ioctl(struct cdev *dev, u_long cmd, caddr_t addr, int flag,
|
|
|
|
struct thread *t)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
switch(cmd) {
|
Rather than having the Coda module use its own namecache, use the global
VFS namecache, as is done by the Coda module on Linux. Unlike the Coda
namecache, the global VFS namecache isn't tagged by credential, so use
ore conservative flushing behavior (for now) when CODA_PURGEUSER is
issued by Venus.
This improves overall integration with the FreeBSD VFS, including
allowing __getcwd() to work better, procfs/procstat monitoring, and so
on. This improves shell behavior in many cases, and improves ".."
handling. It may lead to some slowdown until we've implemented a
specific access cache, which should net improve performance, but in the
mean time, lookup access control now always goes to Venus, whereas
previously it didn't.
MFC after: 1 month
2008-02-13 13:06:22 +00:00
|
|
|
case CODARESIZE:
|
|
|
|
return (ENODEV);
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
case CODASTATS:
|
Rather than having the Coda module use its own namecache, use the global
VFS namecache, as is done by the Coda module on Linux. Unlike the Coda
namecache, the global VFS namecache isn't tagged by credential, so use
ore conservative flushing behavior (for now) when CODA_PURGEUSER is
issued by Venus.
This improves overall integration with the FreeBSD VFS, including
allowing __getcwd() to work better, procfs/procstat monitoring, and so
on. This improves shell behavior in many cases, and improves ".."
handling. It may lead to some slowdown until we've implemented a
specific access cache, which should net improve performance, but in the
mean time, lookup access control now always goes to Venus, whereas
previously it didn't.
MFC after: 1 month
2008-02-13 13:06:22 +00:00
|
|
|
return (ENODEV);
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
case CODAPRINT:
|
Rather than having the Coda module use its own namecache, use the global
VFS namecache, as is done by the Coda module on Linux. Unlike the Coda
namecache, the global VFS namecache isn't tagged by credential, so use
ore conservative flushing behavior (for now) when CODA_PURGEUSER is
issued by Venus.
This improves overall integration with the FreeBSD VFS, including
allowing __getcwd() to work better, procfs/procstat monitoring, and so
on. This improves shell behavior in many cases, and improves ".."
handling. It may lead to some slowdown until we've implemented a
specific access cache, which should net improve performance, but in the
mean time, lookup access control now always goes to Venus, whereas
previously it didn't.
MFC after: 1 month
2008-02-13 13:06:22 +00:00
|
|
|
return (ENODEV);
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
case CIOC_KERNEL_VERSION:
|
|
|
|
switch (*(u_int *)addr) {
|
|
|
|
case 0:
|
|
|
|
*(u_int *)addr = coda_kernel_version;
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
if (coda_kernel_version != *(u_int *)addr)
|
|
|
|
return (ENOENT);
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return (ENOENT);
|
|
|
|
}
|
|
|
|
|
1998-11-11 20:32:20 +00:00
|
|
|
default:
|
2008-02-10 11:18:12 +00:00
|
|
|
return (EINVAL);
|
1998-11-11 20:32:20 +00:00
|
|
|
}
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2008-02-10 11:18:12 +00:00
|
|
|
vc_poll(struct cdev *dev, int events, struct thread *td)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
2008-02-10 11:18:12 +00:00
|
|
|
struct vcomm *vcp;
|
|
|
|
int event_msk = 0;
|
|
|
|
|
|
|
|
ENTRY;
|
|
|
|
vcp = &dev2coda_mntinfo(dev)->mi_vcomm;
|
|
|
|
event_msk = events & (POLLIN|POLLRDNORM);
|
|
|
|
if (!event_msk)
|
|
|
|
return (0);
|
2008-02-17 14:33:28 +00:00
|
|
|
if (!TAILQ_EMPTY(&vcp->vc_requests))
|
2008-02-10 11:18:12 +00:00
|
|
|
return (events & (POLLIN|POLLRDNORM));
|
|
|
|
selrecord(td, &(vcp->vc_selproc));
|
|
|
|
return (0);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2008-02-10 11:18:12 +00:00
|
|
|
* Statistics.
|
1998-08-29 21:14:52 +00:00
|
|
|
*/
|
1998-09-11 18:50:17 +00:00
|
|
|
struct coda_clstat coda_clstat;
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Key question: whether to sleep interuptably or uninteruptably when waiting
|
|
|
|
* for Venus. The former seems better (cause you can ^C a job), but then
|
|
|
|
* GNU-EMACS completion breaks. Use tsleep with no timeout, and no longjmp
|
|
|
|
* happens. But, when sleeping "uninterruptibly", we don't get told if it
|
|
|
|
* returns abnormally (e.g. kill -9).
|
1998-08-29 21:14:52 +00:00
|
|
|
*/
|
|
|
|
int
|
2008-02-10 11:18:12 +00:00
|
|
|
coda_call(struct coda_mntinfo *mntinfo, int inSize, int *outSize,
|
|
|
|
caddr_t buffer)
|
1998-08-29 21:14:52 +00:00
|
|
|
{
|
|
|
|
struct vcomm *vcp;
|
|
|
|
struct vmsg *vmp;
|
|
|
|
int error;
|
2008-02-10 11:18:12 +00:00
|
|
|
#ifdef CTL_C
|
2001-09-12 08:38:13 +00:00
|
|
|
struct thread *td = curthread;
|
|
|
|
struct proc *p = td->td_proc;
|
2000-12-06 01:45:20 +00:00
|
|
|
sigset_t psig_omask;
|
1999-09-29 15:03:48 +00:00
|
|
|
sigset_t tempset;
|
1998-08-29 21:14:52 +00:00
|
|
|
int i;
|
|
|
|
#endif
|
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Unlikely, but could be a race condition with a dying warden.
|
|
|
|
*/
|
|
|
|
if (mntinfo == NULL)
|
|
|
|
return ENODEV;
|
1998-08-29 21:14:52 +00:00
|
|
|
vcp = &(mntinfo->mi_vcomm);
|
1998-09-11 18:50:17 +00:00
|
|
|
coda_clstat.ncalls++;
|
|
|
|
coda_clstat.reqs[((struct coda_in_hdr *)buffer)->opcode]++;
|
1998-08-29 21:14:52 +00:00
|
|
|
if (!VC_OPEN(vcp))
|
2008-02-10 11:18:12 +00:00
|
|
|
return (ENODEV);
|
1998-09-11 18:50:17 +00:00
|
|
|
CODA_ALLOC(vmp,struct vmsg *,sizeof(struct vmsg));
|
2008-02-10 11:18:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Format the request message.
|
|
|
|
*/
|
1998-08-29 21:14:52 +00:00
|
|
|
vmp->vm_data = buffer;
|
|
|
|
vmp->vm_flags = 0;
|
|
|
|
vmp->vm_inSize = inSize;
|
2008-02-10 11:18:12 +00:00
|
|
|
vmp->vm_outSize
|
1998-08-29 21:14:52 +00:00
|
|
|
= *outSize ? *outSize : inSize; /* |buffer| >= inSize */
|
1998-09-11 18:50:17 +00:00
|
|
|
vmp->vm_opcode = ((struct coda_in_hdr *)buffer)->opcode;
|
1998-08-29 21:14:52 +00:00
|
|
|
vmp->vm_unique = ++vcp->vc_seq;
|
1998-09-11 18:50:17 +00:00
|
|
|
if (codadebug)
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("Doing a call for %d.%d\n", vmp->vm_opcode,
|
|
|
|
vmp->vm_unique));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fill in the common input args.
|
|
|
|
*/
|
1998-09-11 18:50:17 +00:00
|
|
|
((struct coda_in_hdr *)buffer)->unique = vmp->vm_unique;
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* Append msg to request queue and poke Venus.
|
|
|
|
*/
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_INSERT_TAIL(&vcp->vc_requests, vmp, vm_chain);
|
2003-11-09 09:17:26 +00:00
|
|
|
selwakeuppri(&(vcp->vc_selproc), coda_call_sleep);
|
1998-08-29 21:14:52 +00:00
|
|
|
|
2008-02-10 11:18:12 +00:00
|
|
|
/*
|
|
|
|
* We can be interrupted while we wait for Venus to process our
|
|
|
|
* request. If the interrupt occurs before Venus has read the
|
|
|
|
* request, we dequeue and return. If it occurs after the read but
|
|
|
|
* before the reply, we dequeue, send a signal message, and return.
|
|
|
|
* If it occurs after the reply we ignore it. In no case do we want
|
|
|
|
* to restart the syscall. If it was interrupted by a venus shutdown
|
|
|
|
* (vcclose), return ENODEV.
|
|
|
|
*
|
|
|
|
* Ignore return, we have to check anyway.
|
|
|
|
*/
|
|
|
|
#ifdef CTL_C
|
|
|
|
/*
|
|
|
|
* This is work in progress. Setting coda_pcatch lets tsleep
|
|
|
|
* reawaken on a ^c or ^z. The problem is that emacs sets certain
|
|
|
|
* interrupts as SA_RESTART. This means that we should exit sleep
|
|
|
|
* handle the "signal" and then go to sleep again. Mostly this is
|
|
|
|
* done by letting the syscall complete and be restarted. We are not
|
|
|
|
* idempotent and can not do this. A better solution is necessary.
|
1998-08-29 21:14:52 +00:00
|
|
|
*/
|
|
|
|
i = 0;
|
2000-12-06 01:45:20 +00:00
|
|
|
PROC_LOCK(p);
|
2003-03-31 22:49:17 +00:00
|
|
|
psig_omask = td->td_sigmask;
|
1998-08-29 21:14:52 +00:00
|
|
|
do {
|
2000-12-06 01:45:20 +00:00
|
|
|
error = msleep(&vmp->vm_sleep, &p->p_mtx,
|
2008-02-10 11:18:12 +00:00
|
|
|
(coda_call_sleep|coda_pcatch), "coda_call", hz*2);
|
1999-09-29 15:03:48 +00:00
|
|
|
if (error == 0)
|
|
|
|
break;
|
|
|
|
else if (error == EWOULDBLOCK) {
|
2008-02-10 11:18:12 +00:00
|
|
|
#ifdef CODA_VERBOSE
|
1999-09-29 15:03:48 +00:00
|
|
|
printf("coda_call: tsleep TIMEOUT %d sec\n", 2+2*i);
|
1998-09-25 17:38:32 +00:00
|
|
|
#endif
|
1999-09-29 15:03:48 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
SIGEMPTYSET(tempset);
|
|
|
|
SIGADDSET(tempset, SIGIO);
|
2003-03-31 22:49:17 +00:00
|
|
|
if (SIGSETEQ(td->td_siglist, tempset)) {
|
|
|
|
SIGADDSET(td->td_sigmask, SIGIO);
|
2008-02-10 11:18:12 +00:00
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
printf("coda_call: tsleep returns %d SIGIO, "
|
|
|
|
"cnt %d\n", error, i);
|
1998-10-28 20:31:13 +00:00
|
|
|
#endif
|
1999-09-29 15:03:48 +00:00
|
|
|
} else {
|
|
|
|
SIGDELSET(tempset, SIGIO);
|
|
|
|
SIGADDSET(tempset, SIGALRM);
|
2003-03-31 22:49:17 +00:00
|
|
|
if (SIGSETEQ(td->td_siglist, tempset)) {
|
|
|
|
SIGADDSET(td->td_sigmask, SIGALRM);
|
2008-02-10 11:18:12 +00:00
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
printf("coda_call: tsleep returns "
|
|
|
|
"%d SIGALRM, cnt %d\n", error, i);
|
1998-09-25 17:38:32 +00:00
|
|
|
#endif
|
2008-02-10 11:18:12 +00:00
|
|
|
} else {
|
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
printf("coda_call: tsleep returns "
|
|
|
|
"%d, cnt %d\n", error, i);
|
2003-09-10 01:41:15 +00:00
|
|
|
#endif
|
1999-09-29 15:03:48 +00:00
|
|
|
|
2005-12-04 10:06:06 +00:00
|
|
|
#ifdef notyet
|
2003-03-31 22:49:17 +00:00
|
|
|
tempset = td->td_siglist;
|
|
|
|
SIGSETNAND(tempset, td->td_sigmask);
|
2008-02-10 11:18:12 +00:00
|
|
|
printf("coda_call: siglist = %p, "
|
|
|
|
"sigmask = %p, mask %p\n",
|
|
|
|
td->td_siglist, td->td_sigmask,
|
|
|
|
tempset);
|
1999-09-29 15:03:48 +00:00
|
|
|
break;
|
2003-03-31 22:49:17 +00:00
|
|
|
SIGSETOR(td->td_sigmask, td->td_siglist);
|
|
|
|
tempset = td->td_siglist;
|
|
|
|
SIGSETNAND(tempset, td->td_sigmask);
|
2008-02-10 11:18:12 +00:00
|
|
|
printf("coda_call: new mask, "
|
|
|
|
"siglist = %p, sigmask = %p, "
|
|
|
|
"mask %p\n", td->td_siglist,
|
|
|
|
td->td_sigmask, tempset);
|
1998-09-25 17:38:32 +00:00
|
|
|
#endif
|
1999-09-29 15:03:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1998-10-28 20:31:13 +00:00
|
|
|
} while (error && i++ < 128 && VC_OPEN(vcp));
|
2003-03-31 22:49:17 +00:00
|
|
|
td->td_sigmask = psig_omask;
|
|
|
|
signotify(td);
|
2000-12-06 01:45:20 +00:00
|
|
|
PROC_UNLOCK(p);
|
1998-08-29 21:14:52 +00:00
|
|
|
#else
|
2008-02-10 11:18:12 +00:00
|
|
|
(void)tsleep(&vmp->vm_sleep, coda_call_sleep, "coda_call", 0);
|
1998-08-29 21:14:52 +00:00
|
|
|
#endif
|
2008-02-10 11:18:12 +00:00
|
|
|
if (VC_OPEN(vcp)) {
|
|
|
|
/*
|
|
|
|
* Venus is still alive.
|
|
|
|
*
|
|
|
|
* Op went through, interrupt or not...
|
|
|
|
*/
|
|
|
|
if (vmp->vm_flags & VM_WRITE) {
|
|
|
|
error = 0;
|
|
|
|
*outSize = vmp->vm_outSize;
|
|
|
|
} else if (!(vmp->vm_flags & VM_READ)) {
|
|
|
|
/* Interrupted before venus read it. */
|
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
if (1)
|
1998-09-25 17:38:32 +00:00
|
|
|
#else
|
2008-02-10 11:18:12 +00:00
|
|
|
if (codadebug)
|
1998-09-25 17:38:32 +00:00
|
|
|
#endif
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("interrupted before read: op = "
|
|
|
|
"%d.%d, flags = %x\n", vmp->vm_opcode,
|
|
|
|
vmp->vm_unique, vmp->vm_flags));
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_REMOVE(&vcp->vc_requests, vmp, vm_chain);
|
2008-02-10 11:18:12 +00:00
|
|
|
error = EINTR;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* (!(vmp->vm_flags & VM_WRITE)) means interrupted
|
|
|
|
* after upcall started.
|
|
|
|
*
|
|
|
|
* Interrupted after start of upcall, send venus a
|
|
|
|
* signal.
|
|
|
|
*/
|
|
|
|
struct coda_in_hdr *dog;
|
|
|
|
struct vmsg *svmp;
|
|
|
|
|
|
|
|
#ifdef CODA_VERBOSE
|
|
|
|
if (1)
|
1998-09-25 17:38:32 +00:00
|
|
|
#else
|
2008-02-10 11:18:12 +00:00
|
|
|
if (codadebug)
|
1998-09-25 17:38:32 +00:00
|
|
|
#endif
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("Sending Venus a signal: op = "
|
|
|
|
"%d.%d, flags = %x\n", vmp->vm_opcode,
|
|
|
|
vmp->vm_unique, vmp->vm_flags));
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_REMOVE(&vcp->vc_requests, vmp, vm_chain);
|
2008-02-10 11:18:12 +00:00
|
|
|
error = EINTR;
|
|
|
|
CODA_ALLOC(svmp, struct vmsg *, sizeof(struct vmsg));
|
|
|
|
CODA_ALLOC((svmp->vm_data), char *,
|
|
|
|
sizeof(struct coda_in_hdr));
|
|
|
|
dog = (struct coda_in_hdr *)svmp->vm_data;
|
|
|
|
svmp->vm_flags = 0;
|
|
|
|
dog->opcode = svmp->vm_opcode = CODA_SIGNAL;
|
|
|
|
dog->unique = svmp->vm_unique = vmp->vm_unique;
|
|
|
|
svmp->vm_inSize = sizeof (struct coda_in_hdr);
|
|
|
|
/*??? rvb */ svmp->vm_outSize = sizeof (struct coda_in_hdr);
|
|
|
|
if (codadebug)
|
|
|
|
myprintf(("coda_call: enqueing signal msg "
|
|
|
|
"(%d, %d)\n", svmp->vm_opcode,
|
|
|
|
svmp->vm_unique));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Insert at head of queue!
|
2008-02-17 14:33:28 +00:00
|
|
|
*
|
|
|
|
* XXXRW: Actually, the tail.
|
2008-02-10 11:18:12 +00:00
|
|
|
*/
|
2008-02-17 14:33:28 +00:00
|
|
|
TAILQ_INSERT_TAIL(&vcp->vc_requests, svmp, vm_chain);
|
2008-02-10 11:18:12 +00:00
|
|
|
selwakeuppri(&(vcp->vc_selproc), coda_call_sleep);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* If venus died (!VC_OPEN(vcp)) */
|
1998-09-11 18:50:17 +00:00
|
|
|
if (codadebug)
|
2008-02-10 11:18:12 +00:00
|
|
|
myprintf(("vcclose woke op %d.%d flags %d\n",
|
|
|
|
vmp->vm_opcode, vmp->vm_unique, vmp->vm_flags));
|
1998-08-29 21:14:52 +00:00
|
|
|
error = ENODEV;
|
|
|
|
}
|
1998-09-11 18:50:17 +00:00
|
|
|
CODA_FREE(vmp, sizeof(struct vmsg));
|
1998-10-28 20:31:13 +00:00
|
|
|
if (outstanding_upcalls > 0 && (--outstanding_upcalls == 0))
|
|
|
|
wakeup(&outstanding_upcalls);
|
1998-08-29 21:14:52 +00:00
|
|
|
if (!error)
|
1998-09-11 18:50:17 +00:00
|
|
|
error = ((struct coda_out_hdr *)buffer)->result;
|
2008-02-10 11:18:12 +00:00
|
|
|
return (error);
|
1998-08-29 21:14:52 +00:00
|
|
|
}
|