1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1989, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
* @(#)mfs_vnops.c 8.11 (Berkeley) 5/22/95
|
1999-08-28 01:08:13 +00:00
|
|
|
* $FreeBSD$
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/buf.h>
|
|
|
|
#include <sys/vnode.h>
|
1997-11-01 20:19:44 +00:00
|
|
|
#include <sys/malloc.h>
|
1999-01-21 08:29:12 +00:00
|
|
|
#include <sys/sysproto.h>
|
|
|
|
#include <sys/mman.h>
|
1999-08-08 18:43:05 +00:00
|
|
|
#include <sys/conf.h>
|
1994-05-24 10:09:53 +00:00
|
|
|
|
|
|
|
#include <ufs/mfs/mfsnode.h>
|
|
|
|
#include <ufs/mfs/mfs_extern.h>
|
|
|
|
|
1997-10-20 19:53:26 +00:00
|
|
|
static int mfs_badop __P((struct vop_generic_args *));
|
1995-12-17 21:14:36 +00:00
|
|
|
static int mfs_bmap __P((struct vop_bmap_args *));
|
|
|
|
static int mfs_close __P((struct vop_close_args *));
|
1997-10-15 13:24:07 +00:00
|
|
|
static int mfs_fsync __P((struct vop_fsync_args *));
|
1999-01-21 08:29:12 +00:00
|
|
|
static int mfs_freeblks __P((struct vop_freeblks_args *));
|
1995-12-17 21:14:36 +00:00
|
|
|
static int mfs_inactive __P((struct vop_inactive_args *)); /* XXX */
|
|
|
|
static int mfs_open __P((struct vop_open_args *));
|
1997-11-01 20:19:44 +00:00
|
|
|
static int mfs_reclaim __P((struct vop_reclaim_args *)); /* XXX */
|
1995-12-17 21:14:36 +00:00
|
|
|
static int mfs_print __P((struct vop_print_args *)); /* XXX */
|
|
|
|
static int mfs_strategy __P((struct vop_strategy_args *)); /* XXX */
|
1998-02-16 23:55:53 +00:00
|
|
|
static int mfs_getpages __P((struct vop_getpages_args *)); /* XXX */
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* mfs vnode operations.
|
|
|
|
*/
|
1995-11-09 08:17:23 +00:00
|
|
|
vop_t **mfs_vnodeop_p;
|
1995-12-17 21:14:36 +00:00
|
|
|
static struct vnodeopv_entry_desc mfs_vnodeop_entries[] = {
|
1997-10-20 19:53:26 +00:00
|
|
|
{ &vop_default_desc, (vop_t *) mfs_badop },
|
1997-10-15 09:22:02 +00:00
|
|
|
{ &vop_bmap_desc, (vop_t *) mfs_bmap },
|
1997-10-20 19:53:26 +00:00
|
|
|
{ &vop_bwrite_desc, (vop_t *) vop_defaultop },
|
1997-10-15 09:22:02 +00:00
|
|
|
{ &vop_close_desc, (vop_t *) mfs_close },
|
1999-01-21 08:29:12 +00:00
|
|
|
{ &vop_freeblks_desc, (vop_t *) mfs_freeblks },
|
1997-10-15 13:24:07 +00:00
|
|
|
{ &vop_fsync_desc, (vop_t *) mfs_fsync },
|
1998-02-16 23:55:53 +00:00
|
|
|
{ &vop_getpages_desc, (vop_t *) mfs_getpages },
|
1997-10-15 10:05:29 +00:00
|
|
|
{ &vop_inactive_desc, (vop_t *) mfs_inactive },
|
1997-10-20 19:53:26 +00:00
|
|
|
{ &vop_ioctl_desc, (vop_t *) vop_enotty },
|
|
|
|
{ &vop_islocked_desc, (vop_t *) vop_defaultop },
|
|
|
|
{ &vop_lock_desc, (vop_t *) vop_defaultop },
|
1997-10-15 09:22:02 +00:00
|
|
|
{ &vop_open_desc, (vop_t *) mfs_open },
|
|
|
|
{ &vop_print_desc, (vop_t *) mfs_print },
|
1997-11-01 20:19:44 +00:00
|
|
|
{ &vop_reclaim_desc, (vop_t *) mfs_reclaim },
|
1997-10-15 10:05:29 +00:00
|
|
|
{ &vop_strategy_desc, (vop_t *) mfs_strategy },
|
1997-10-20 19:53:26 +00:00
|
|
|
{ &vop_unlock_desc, (vop_t *) vop_defaultop },
|
1995-11-09 08:17:23 +00:00
|
|
|
{ NULL, NULL }
|
1994-05-24 10:09:53 +00:00
|
|
|
};
|
1995-12-17 21:14:36 +00:00
|
|
|
static struct vnodeopv_desc mfs_vnodeop_opv_desc =
|
1994-05-24 10:09:53 +00:00
|
|
|
{ &mfs_vnodeop_p, mfs_vnodeop_entries };
|
|
|
|
|
1994-09-21 03:47:43 +00:00
|
|
|
VNODEOP_SET(mfs_vnodeop_opv_desc);
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Vnode Operations.
|
|
|
|
*
|
|
|
|
* Open called to allow memory filesystem to initialize and
|
|
|
|
* validate before actual IO. Record our process identifier
|
|
|
|
* so we can tell when we are doing I/O to ourself.
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_open(ap)
|
|
|
|
struct vop_open_args /* {
|
|
|
|
struct vnode *a_vp;
|
|
|
|
int a_mode;
|
|
|
|
struct ucred *a_cred;
|
|
|
|
struct proc *a_p;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
|
|
|
|
if (ap->a_vp->v_type != VBLK) {
|
1995-10-31 10:45:48 +00:00
|
|
|
panic("mfs_open not VBLK");
|
1994-05-24 10:09:53 +00:00
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
1997-10-15 13:24:07 +00:00
|
|
|
static int
|
|
|
|
mfs_fsync(ap)
|
|
|
|
struct vop_fsync_args *ap;
|
|
|
|
{
|
|
|
|
|
|
|
|
return (VOCALL(spec_vnodeop_p, VOFFSET(vop_fsync), ap));
|
|
|
|
}
|
|
|
|
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
|
|
|
* mfs_freeblks() - hook to allow us to free physical memory.
|
|
|
|
*
|
2000-03-20 10:44:49 +00:00
|
|
|
* We implement the BIO_DELETE strategy. We can't just madvise()
|
1999-01-21 08:29:12 +00:00
|
|
|
* here because we have to do it in the correct order vs other bio
|
|
|
|
* requests, so we queue it.
|
The VFS/BIO subsystem contained a number of hacks in order to optimize
piecemeal, middle-of-file writes for NFS. These hacks have caused no
end of trouble, especially when combined with mmap(). I've removed
them. Instead, NFS will issue a read-before-write to fully
instantiate the struct buf containing the write. NFS does, however,
optimize piecemeal appends to files. For most common file operations,
you will not notice the difference. The sole remaining fragment in
the VFS/BIO system is b_dirtyoff/end, which NFS uses to avoid cache
coherency issues with read-merge-write style operations. NFS also
optimizes the write-covers-entire-buffer case by avoiding the
read-before-write. There is quite a bit of room for further
optimization in these areas.
The VM system marks pages fully-valid (AKA vm_page_t->valid =
VM_PAGE_BITS_ALL) in several places, most noteably in vm_fault. This
is not correct operation. The vm_pager_get_pages() code is now
responsible for marking VM pages all-valid. A number of VM helper
routines have been added to aid in zeroing-out the invalid portions of
a VM page prior to the page being marked all-valid. This operation is
necessary to properly support mmap(). The zeroing occurs most often
when dealing with file-EOF situations. Several bugs have been fixed
in the NFS subsystem, including bits handling file and directory EOF
situations and buf->b_flags consistancy issues relating to clearing
B_ERROR & B_INVAL, and handling B_DONE.
getblk() and allocbuf() have been rewritten. B_CACHE operation is now
formally defined in comments and more straightforward in
implementation. B_CACHE for VMIO buffers is based on the validity of
the backing store. B_CACHE for non-VMIO buffers is based simply on
whether the buffer is B_INVAL or not (B_CACHE set if B_INVAL clear,
and vise-versa). biodone() is now responsible for setting B_CACHE
when a successful read completes. B_CACHE is also set when a bdwrite()
is initiated and when a bwrite() is initiated. VFS VOP_BWRITE
routines (there are only two - nfs_bwrite() and bwrite()) are now
expected to set B_CACHE. This means that bowrite() and bawrite() also
set B_CACHE indirectly.
There are a number of places in the code which were previously using
buf->b_bufsize (which is DEV_BSIZE aligned) when they should have
been using buf->b_bcount. These have been fixed. getblk() now clears
B_DONE on return because the rest of the system is so bad about
dealing with B_DONE.
Major fixes to NFS/TCP have been made. A server-side bug could cause
requests to be lost by the server due to nfs_realign() overwriting
other rpc's in the same TCP mbuf chain. The server's kernel must be
recompiled to get the benefit of the fixes.
Submitted by: Matthew Dillon <dillon@apollo.backplane.com>
1999-05-02 23:57:16 +00:00
|
|
|
*
|
|
|
|
* Note: geteblk() sets B_INVAL. We leave it set to guarentee buffer
|
|
|
|
* throw-away on brelse()? XXX
|
1999-01-21 08:29:12 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
mfs_freeblks(ap)
|
|
|
|
struct vop_freeblks_args /* {
|
|
|
|
struct vnode *a_vp;
|
|
|
|
daddr_t a_addr;
|
|
|
|
daddr_t a_length;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
struct buf *bp;
|
|
|
|
struct vnode *vp;
|
|
|
|
|
|
|
|
if (!vfinddev(ap->a_vp->v_rdev, VBLK, &vp) || vp->v_usecount == 0)
|
1999-04-11 02:28:32 +00:00
|
|
|
panic("mfs_freeblks: bad dev");
|
1999-01-21 08:29:12 +00:00
|
|
|
|
|
|
|
bp = geteblk(ap->a_length);
|
2000-03-20 10:44:49 +00:00
|
|
|
bp->b_flags |= B_ASYNC;
|
|
|
|
bp->b_iocmd = BIO_DELETE;
|
1999-01-21 08:29:12 +00:00
|
|
|
bp->b_dev = ap->a_vp->v_rdev;
|
|
|
|
bp->b_blkno = ap->a_addr;
|
|
|
|
bp->b_offset = dbtob(ap->a_addr);
|
|
|
|
bp->b_bcount = ap->a_length;
|
1999-06-26 02:47:16 +00:00
|
|
|
BUF_KERNPROC(bp);
|
1999-01-21 08:29:12 +00:00
|
|
|
VOP_STRATEGY(vp, bp);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Pass I/O requests to the memory filesystem process.
|
|
|
|
*/
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_strategy(ap)
|
|
|
|
struct vop_strategy_args /* {
|
1998-07-04 20:45:42 +00:00
|
|
|
struct vnode *a_vp;
|
1994-05-24 10:09:53 +00:00
|
|
|
struct buf *a_bp;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
register struct buf *bp = ap->a_bp;
|
|
|
|
register struct mfsnode *mfsp;
|
|
|
|
struct vnode *vp;
|
|
|
|
struct proc *p = curproc; /* XXX */
|
1999-01-21 08:29:12 +00:00
|
|
|
int s;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
|
|
|
if (!vfinddev(bp->b_dev, VBLK, &vp) || vp->v_usecount == 0)
|
|
|
|
panic("mfs_strategy: bad dev");
|
|
|
|
mfsp = VTOMFS(vp);
|
1999-01-21 08:29:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* splbio required for queueing/dequeueing, in case of forwarded
|
1999-01-27 18:19:53 +00:00
|
|
|
* BPs from bio interrupts (?). It may not be necessary.
|
1999-01-21 08:29:12 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
s = splbio();
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
if (mfsp->mfs_pid == 0) {
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
2000-03-20 10:44:49 +00:00
|
|
|
* mini-root. Note: BIO_DELETE not supported at the moment,
|
1999-01-21 08:29:12 +00:00
|
|
|
* I'm not sure what kind of dataspace b_data is in.
|
|
|
|
*/
|
1994-05-24 10:09:53 +00:00
|
|
|
caddr_t base;
|
|
|
|
|
|
|
|
base = mfsp->mfs_baseoff + (bp->b_blkno << DEV_BSHIFT);
|
2000-03-20 10:44:49 +00:00
|
|
|
if (bp->b_iocmd == BIO_DELETE)
|
1999-01-21 08:29:12 +00:00
|
|
|
;
|
2000-03-20 10:44:49 +00:00
|
|
|
if (bp->b_iocmd == BIO_READ)
|
1994-05-24 10:09:53 +00:00
|
|
|
bcopy(base, bp->b_data, bp->b_bcount);
|
|
|
|
else
|
|
|
|
bcopy(bp->b_data, base, bp->b_bcount);
|
|
|
|
biodone(bp);
|
|
|
|
} else if (mfsp->mfs_pid == p->p_pid) {
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
|
|
|
* VOP to self
|
|
|
|
*/
|
|
|
|
splx(s);
|
|
|
|
mfs_doio(bp, mfsp);
|
|
|
|
s = splbio();
|
1994-05-24 10:09:53 +00:00
|
|
|
} else {
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
|
|
|
* VOP from some other process, queue to MFS process and
|
|
|
|
* wake it up.
|
|
|
|
*/
|
1997-09-21 22:10:02 +00:00
|
|
|
bufq_insert_tail(&mfsp->buf_queue, bp);
|
1994-05-24 10:09:53 +00:00
|
|
|
wakeup((caddr_t)vp);
|
|
|
|
}
|
1999-01-21 08:29:12 +00:00
|
|
|
splx(s);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Memory file system I/O.
|
|
|
|
*
|
|
|
|
* Trivial on the HP since buffer has already been mapping into KVA space.
|
1999-01-21 08:29:12 +00:00
|
|
|
*
|
|
|
|
* Read and Write are handled with a simple copyin and copyout.
|
|
|
|
*
|
2000-03-20 10:44:49 +00:00
|
|
|
* We also partially support VOP_FREEBLKS() via BIO_DELETE. We can't implement
|
1999-01-21 08:29:12 +00:00
|
|
|
* completely -- for example, on fragments or inode metadata, but we can
|
|
|
|
* implement it for page-aligned requests.
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
|
|
|
void
|
1999-01-21 08:29:12 +00:00
|
|
|
mfs_doio(bp, mfsp)
|
1994-05-24 10:09:53 +00:00
|
|
|
register struct buf *bp;
|
1999-01-21 08:29:12 +00:00
|
|
|
struct mfsnode *mfsp;
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
1999-01-21 08:29:12 +00:00
|
|
|
caddr_t base = mfsp->mfs_baseoff + (bp->b_blkno << DEV_BSHIFT);
|
|
|
|
|
2000-03-20 10:44:49 +00:00
|
|
|
if (bp->b_iocmd == BIO_DELETE) {
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
2000-03-20 10:44:49 +00:00
|
|
|
* Implement BIO_DELETE, which allows the filesystem to tell
|
1999-01-21 08:29:12 +00:00
|
|
|
* a block device when blocks are no longer needed (like when
|
|
|
|
* a file is deleted). We use the hook to MADV_FREE the VM.
|
|
|
|
* This makes an MFS filesystem work as well or better then
|
|
|
|
* a sun-style swap-mounted filesystem.
|
|
|
|
*/
|
|
|
|
int bytes = bp->b_bcount;
|
|
|
|
|
|
|
|
if ((vm_offset_t)base & PAGE_MASK) {
|
|
|
|
int n = PAGE_SIZE - ((vm_offset_t)base & PAGE_MASK);
|
|
|
|
bytes -= n;
|
|
|
|
base += n;
|
|
|
|
}
|
|
|
|
if (bytes > 0) {
|
|
|
|
struct madvise_args uap;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1999-01-21 08:29:12 +00:00
|
|
|
bytes &= ~PAGE_MASK;
|
|
|
|
if (bytes != 0) {
|
|
|
|
bzero(&uap, sizeof(uap));
|
|
|
|
uap.addr = base;
|
|
|
|
uap.len = bytes;
|
|
|
|
uap.behav = MADV_FREE;
|
|
|
|
madvise(curproc, &uap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bp->b_error = 0;
|
2000-03-20 10:44:49 +00:00
|
|
|
} else if (bp->b_iocmd == BIO_READ) {
|
1999-01-21 08:29:12 +00:00
|
|
|
/*
|
|
|
|
* Read data from our 'memory' disk
|
|
|
|
*/
|
1994-05-24 10:09:53 +00:00
|
|
|
bp->b_error = copyin(base, bp->b_data, bp->b_bcount);
|
1999-01-21 08:29:12 +00:00
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* Write data to our 'memory' disk
|
|
|
|
*/
|
1994-05-24 10:09:53 +00:00
|
|
|
bp->b_error = copyout(bp->b_data, base, bp->b_bcount);
|
1999-01-21 08:29:12 +00:00
|
|
|
}
|
1994-05-24 10:09:53 +00:00
|
|
|
if (bp->b_error)
|
|
|
|
bp->b_flags |= B_ERROR;
|
|
|
|
biodone(bp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is a noop, simply returning what one has been given.
|
|
|
|
*/
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_bmap(ap)
|
|
|
|
struct vop_bmap_args /* {
|
|
|
|
struct vnode *a_vp;
|
1997-02-10 02:22:35 +00:00
|
|
|
ufs_daddr_t a_bn;
|
1994-05-24 10:09:53 +00:00
|
|
|
struct vnode **a_vpp;
|
1997-02-10 02:22:35 +00:00
|
|
|
ufs_daddr_t *a_bnp;
|
1994-05-24 10:09:53 +00:00
|
|
|
int *a_runp;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
|
|
|
|
if (ap->a_vpp != NULL)
|
|
|
|
*ap->a_vpp = ap->a_vp;
|
|
|
|
if (ap->a_bnp != NULL)
|
|
|
|
*ap->a_bnp = ap->a_bn;
|
1995-02-03 06:46:28 +00:00
|
|
|
if (ap->a_runp != NULL)
|
|
|
|
*ap->a_runp = 0;
|
1994-05-24 10:09:53 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Memory filesystem close routine
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_close(ap)
|
|
|
|
struct vop_close_args /* {
|
|
|
|
struct vnode *a_vp;
|
|
|
|
int a_fflag;
|
|
|
|
struct ucred *a_cred;
|
|
|
|
struct proc *a_p;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
register struct vnode *vp = ap->a_vp;
|
|
|
|
register struct mfsnode *mfsp = VTOMFS(vp);
|
|
|
|
register struct buf *bp;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Finish any pending I/O requests.
|
|
|
|
*/
|
1999-01-28 00:57:57 +00:00
|
|
|
while ((bp = bufq_first(&mfsp->buf_queue)) != NULL) {
|
1997-09-21 22:10:02 +00:00
|
|
|
bufq_remove(&mfsp->buf_queue, bp);
|
1999-01-21 08:29:12 +00:00
|
|
|
mfs_doio(bp, mfsp);
|
1994-05-24 10:09:53 +00:00
|
|
|
wakeup((caddr_t)bp);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* On last close of a memory filesystem
|
|
|
|
* we must invalidate any in core blocks, so that
|
|
|
|
* we can, free up its vnode.
|
|
|
|
*/
|
1999-01-28 00:57:57 +00:00
|
|
|
if ((error = vinvalbuf(vp, 1, ap->a_cred, ap->a_p, 0, 0)) != 0)
|
1994-05-24 10:09:53 +00:00
|
|
|
return (error);
|
|
|
|
/*
|
|
|
|
* There should be no way to have any more uses of this
|
|
|
|
* vnode, so if we find any other uses, it is a panic.
|
|
|
|
*/
|
|
|
|
if (vp->v_usecount > 1)
|
|
|
|
printf("mfs_close: ref count %d > 1\n", vp->v_usecount);
|
1997-09-21 22:10:02 +00:00
|
|
|
if (vp->v_usecount > 1 || (bufq_first(&mfsp->buf_queue) != NULL))
|
1994-05-24 10:09:53 +00:00
|
|
|
panic("mfs_close");
|
|
|
|
/*
|
|
|
|
* Send a request to the filesystem server to exit.
|
|
|
|
*/
|
1996-04-08 07:54:51 +00:00
|
|
|
mfsp->mfs_active = 0;
|
1994-05-24 10:09:53 +00:00
|
|
|
wakeup((caddr_t)vp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Memory filesystem inactive routine
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_inactive(ap)
|
|
|
|
struct vop_inactive_args /* {
|
|
|
|
struct vnode *a_vp;
|
1997-02-10 02:22:35 +00:00
|
|
|
struct proc *a_p;
|
1994-05-24 10:09:53 +00:00
|
|
|
} */ *ap;
|
|
|
|
{
|
1997-02-10 02:22:35 +00:00
|
|
|
struct vnode *vp = ap->a_vp;
|
|
|
|
struct mfsnode *mfsp = VTOMFS(vp);
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1997-09-21 22:10:02 +00:00
|
|
|
if (bufq_first(&mfsp->buf_queue) != NULL)
|
1996-04-08 07:54:51 +00:00
|
|
|
panic("mfs_inactive: not inactive (next buffer %p)",
|
1997-09-21 22:10:02 +00:00
|
|
|
bufq_first(&mfsp->buf_queue));
|
1997-02-10 02:22:35 +00:00
|
|
|
VOP_UNLOCK(vp, 0, ap->a_p);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
1997-11-01 20:19:44 +00:00
|
|
|
/*
|
|
|
|
* Reclaim a memory filesystem devvp so that it can be reused.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mfs_reclaim(ap)
|
|
|
|
struct vop_reclaim_args /* {
|
|
|
|
struct vnode *a_vp;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
register struct vnode *vp = ap->a_vp;
|
|
|
|
|
|
|
|
FREE(vp->v_data, M_MFSNODE);
|
|
|
|
vp->v_data = NULL;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Print out the contents of an mfsnode.
|
|
|
|
*/
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1994-05-24 10:09:53 +00:00
|
|
|
mfs_print(ap)
|
|
|
|
struct vop_print_args /* {
|
|
|
|
struct vnode *a_vp;
|
|
|
|
} */ *ap;
|
|
|
|
{
|
|
|
|
register struct mfsnode *mfsp = VTOMFS(ap->a_vp);
|
|
|
|
|
1998-07-11 07:46:16 +00:00
|
|
|
printf("tag VT_MFS, pid %ld, base %p, size %ld\n",
|
|
|
|
(long)mfsp->mfs_pid, (void *)mfsp->mfs_baseoff, mfsp->mfs_size);
|
1994-05-24 10:09:53 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Block device bad operation
|
|
|
|
*/
|
1995-12-17 21:14:36 +00:00
|
|
|
static int
|
1997-10-20 19:53:26 +00:00
|
|
|
mfs_badop(struct vop_generic_args *ap)
|
1994-05-24 10:09:53 +00:00
|
|
|
{
|
1997-10-20 19:53:26 +00:00
|
|
|
int i;
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1997-10-20 19:53:26 +00:00
|
|
|
printf("mfs_badop[%s]\n", ap->a_desc->vdesc_name);
|
|
|
|
i = vop_defaultop(ap);
|
|
|
|
printf("mfs_badop[%s] = %d\n", ap->a_desc->vdesc_name,i);
|
|
|
|
return (i);
|
1994-05-24 10:09:53 +00:00
|
|
|
}
|
1998-02-16 23:55:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
mfs_getpages(ap)
|
|
|
|
struct vop_getpages_args *ap;
|
|
|
|
{
|
|
|
|
|
|
|
|
return (VOCALL(spec_vnodeop_p, VOFFSET(vop_getpages), ap));
|
|
|
|
}
|