1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1982, 1986, 1989, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
* (c) UNIX System Laboratories, Inc.
|
|
|
|
* All or some portions of this file are derived from material licensed
|
|
|
|
* to the University of California by American Telephone and Telegraph
|
|
|
|
* Co. or Unix System Laboratories, Inc. and are reproduced herein with
|
|
|
|
* the permission of UNIX System Laboratories, Inc.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
1996-03-11 02:04:27 +00:00
|
|
|
* @(#)buf.h 8.9 (Berkeley) 3/30/95
|
This mega-commit is meant to fix numerous interrelated problems. There
has been some bitrot and incorrect assumptions in the vfs_bio code. These
problems have manifest themselves worse on NFS type filesystems, but can
still affect local filesystems under certain circumstances. Most of
the problems have involved mmap consistancy, and as a side-effect broke
the vfs.ioopt code. This code might have been committed seperately, but
almost everything is interrelated.
1) Allow (pmap_object_init_pt) prefaulting of buffer-busy pages that
are fully valid.
2) Rather than deactivating erroneously read initial (header) pages in
kern_exec, we now free them.
3) Fix the rundown of non-VMIO buffers that are in an inconsistent
(missing vp) state.
4) Fix the disassociation of pages from buffers in brelse. The previous
code had rotted and was faulty in a couple of important circumstances.
5) Remove a gratuitious buffer wakeup in vfs_vmio_release.
6) Remove a crufty and currently unused cluster mechanism for VBLK
files in vfs_bio_awrite. When the code is functional, I'll add back
a cleaner version.
7) The page busy count wakeups assocated with the buffer cache usage were
incorrectly cleaned up in a previous commit by me. Revert to the
original, correct version, but with a cleaner implementation.
8) The cluster read code now tries to keep data associated with buffers
more aggressively (without breaking the heuristics) when it is presumed
that the read data (buffers) will be soon needed.
9) Change to filesystem lockmgr locks so that they use LK_NOPAUSE. The
delay loop waiting is not useful for filesystem locks, due to the
length of the time intervals.
10) Correct and clean-up spec_getpages.
11) Implement a fully functional nfs_getpages, nfs_putpages.
12) Fix nfs_write so that modifications are coherent with the NFS data on
the server disk (at least as well as NFS seems to allow.)
13) Properly support MS_INVALIDATE on NFS.
14) Properly pass down MS_INVALIDATE to lower levels of the VM code from
vm_map_clean.
15) Better support the notion of pages being busy but valid, so that
fewer in-transit waits occur. (use p->busy more for pageouts instead
of PG_BUSY.) Since the page is fully valid, it is still usable for
reads.
16) It is possible (in error) for cached pages to be busy. Make the
page allocation code handle that case correctly. (It should probably
be a printf or panic, but I want the system to handle coding errors
robustly. I'll probably add a printf.)
17) Correct the design and usage of vm_page_sleep. It didn't handle
consistancy problems very well, so make the design a little less
lofty. After vm_page_sleep, if it ever blocked, it is still important
to relookup the page (if the object generation count changed), and
verify it's status (always.)
18) In vm_pageout.c, vm_pageout_clean had rotted, so clean that up.
19) Push the page busy for writes and VM_PROT_READ into vm_pageout_flush.
20) Fix vm_pager_put_pages and it's descendents to support an int flag
instead of a boolean, so that we can pass down the invalidate bit.
1998-03-07 21:37:31 +00:00
|
|
|
* $Id: buf.h,v 1.45 1998/01/22 17:30:10 dyson Exp $
|
1994-05-24 10:09:53 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _SYS_BUF_H_
|
|
|
|
#define _SYS_BUF_H_
|
1996-05-01 02:16:17 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
#include <sys/queue.h>
|
|
|
|
|
|
|
|
#define NOLIST ((struct buf *)0x87654321)
|
|
|
|
|
1995-02-18 21:12:33 +00:00
|
|
|
struct buf;
|
|
|
|
|
|
|
|
struct iodone_chain {
|
|
|
|
long ic_prev_flags;
|
|
|
|
void (*ic_prev_iodone) __P((struct buf *));
|
|
|
|
void *ic_prev_iodone_chain;
|
|
|
|
struct {
|
|
|
|
long ia_long;
|
|
|
|
void *ia_ptr;
|
|
|
|
} ic_args[5];
|
|
|
|
};
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* The buffer header describes an I/O operation in the kernel.
|
|
|
|
*/
|
|
|
|
struct buf {
|
|
|
|
LIST_ENTRY(buf) b_hash; /* Hash chain. */
|
|
|
|
LIST_ENTRY(buf) b_vnbufs; /* Buffer's associated vnode. */
|
|
|
|
TAILQ_ENTRY(buf) b_freelist; /* Free list position if not active. */
|
1995-11-19 22:22:35 +00:00
|
|
|
TAILQ_ENTRY(buf) b_act; /* Device driver queue when active. *new* */
|
1994-05-24 10:09:53 +00:00
|
|
|
struct proc *b_proc; /* Associated proc; NULL if kernel. */
|
1996-09-06 05:35:00 +00:00
|
|
|
long b_flags; /* B_* flags. */
|
|
|
|
unsigned short b_qindex; /* buffer queue index */
|
1995-12-11 04:58:34 +00:00
|
|
|
unsigned char b_usecount; /* buffer use count */
|
1994-05-24 10:09:53 +00:00
|
|
|
int b_error; /* Errno value. */
|
|
|
|
long b_bufsize; /* Allocated buffer size. */
|
|
|
|
long b_bcount; /* Valid bytes in buffer. */
|
|
|
|
long b_resid; /* Remaining I/O. */
|
|
|
|
dev_t b_dev; /* Device associated with buffer. */
|
1997-12-02 21:07:20 +00:00
|
|
|
caddr_t b_data; /* Memory, superblocks, indirect etc. */
|
1996-11-30 22:41:49 +00:00
|
|
|
caddr_t b_kvabase; /* base kva for buffer */
|
|
|
|
int b_kvasize; /* size of kva for buffer */
|
1994-05-24 10:09:53 +00:00
|
|
|
daddr_t b_lblkno; /* Logical block number. */
|
|
|
|
daddr_t b_blkno; /* Underlying physical block number. */
|
|
|
|
/* Function to call upon completion. */
|
|
|
|
void (*b_iodone) __P((struct buf *));
|
1995-02-18 21:12:33 +00:00
|
|
|
/* For nested b_iodone's. */
|
|
|
|
struct iodone_chain *b_iodone_chain;
|
1994-05-24 10:09:53 +00:00
|
|
|
struct vnode *b_vp; /* Device vnode. */
|
|
|
|
int b_dirtyoff; /* Offset in buffer of dirty region. */
|
|
|
|
int b_dirtyend; /* Offset of end of dirty region. */
|
VM level code cleanups.
1) Start using TSM.
Struct procs continue to point to upages structure, after being freed.
Struct vmspace continues to point to pte object and kva space for kstack.
u_map is now superfluous.
2) vm_map's don't need to be reference counted. They always exist either
in the kernel or in a vmspace. The vmspaces are managed by reference
counts.
3) Remove the "wired" vm_map nonsense.
4) No need to keep a cache of kernel stack kva's.
5) Get rid of strange looking ++var, and change to var++.
6) Change more data structures to use our "zone" allocator. Added
struct proc, struct vmspace and struct vnode. This saves a significant
amount of kva space and physical memory. Additionally, this enables
TSM for the zone managed memory.
7) Keep ioopt disabled for now.
8) Remove the now bogus "single use" map concept.
9) Use generation counts or id's for data structures residing in TSM, where
it allows us to avoid unneeded restart overhead during traversals, where
blocking might occur.
10) Account better for memory deficits, so the pageout daemon will be able
to make enough memory available (experimental.)
11) Fix some vnode locking problems. (From Tor, I think.)
12) Add a check in ufs_lookup, to avoid lots of unneeded calls to bcmp.
(experimental.)
13) Significantly shrink, cleanup, and make slightly faster the vm_fault.c
code. Use generation counts, get rid of unneded collpase operations,
and clean up the cluster code.
14) Make vm_zone more suitable for TSM.
This commit is partially as a result of discussions and contributions from
other people, including DG, Tor Egge, PHK, and probably others that I
have forgotten to attribute (so let me know, if I forgot.)
This is not the infamous, final cleanup of the vnode stuff, but a necessary
step. Vnode mgmt should be correct, but things might still change, and
there is still some missing stuff (like ioopt, and physical backing of
non-merged cache files, debugging of layering concepts.)
1998-01-22 17:30:44 +00:00
|
|
|
int b_generation; /* Generation count of buffer */
|
1994-05-24 10:09:53 +00:00
|
|
|
struct ucred *b_rcred; /* Read credentials reference. */
|
|
|
|
struct ucred *b_wcred; /* Write credentials reference. */
|
|
|
|
int b_validoff; /* Offset in buffer of valid region. */
|
|
|
|
int b_validend; /* Offset of end of valid region. */
|
1994-05-25 09:21:21 +00:00
|
|
|
daddr_t b_pblkno; /* physical block number */
|
1996-12-29 02:45:28 +00:00
|
|
|
void *b_saveaddr; /* Original b_addr for physio. */
|
1994-05-25 09:21:21 +00:00
|
|
|
caddr_t b_savekva; /* saved kva for transfer while bouncing */
|
|
|
|
void *b_driver1; /* for private use by the driver */
|
|
|
|
void *b_driver2; /* for private use by the driver */
|
|
|
|
void *b_spc;
|
1995-11-19 19:54:31 +00:00
|
|
|
union cluster_info {
|
|
|
|
TAILQ_HEAD(cluster_list_head, buf) cluster_head;
|
|
|
|
TAILQ_ENTRY(buf) cluster_entry;
|
|
|
|
} b_cluster;
|
1996-05-03 20:56:09 +00:00
|
|
|
struct vm_page *b_pages[btoc(MAXPHYS)];
|
1994-10-05 09:48:45 +00:00
|
|
|
int b_npages;
|
1994-05-24 10:09:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These flags are kept in b_flags.
|
|
|
|
*/
|
|
|
|
#define B_AGE 0x00000001 /* Move to age queue when I/O done. */
|
1996-03-11 02:04:27 +00:00
|
|
|
#define B_NEEDCOMMIT 0x00000002 /* Append-write in progress. */
|
1994-05-24 10:09:53 +00:00
|
|
|
#define B_ASYNC 0x00000004 /* Start I/O, do not wait. */
|
|
|
|
#define B_BAD 0x00000008 /* Bad block revectoring in progress. */
|
|
|
|
#define B_BUSY 0x00000010 /* I/O in progress. */
|
|
|
|
#define B_CACHE 0x00000020 /* Bread found us in the cache. */
|
|
|
|
#define B_CALL 0x00000040 /* Call b_iodone from biodone. */
|
|
|
|
#define B_DELWRI 0x00000080 /* Delay I/O until buffer reused. */
|
|
|
|
#define B_DIRTY 0x00000100 /* Dirty page to be pushed out async. */
|
|
|
|
#define B_DONE 0x00000200 /* I/O completed. */
|
|
|
|
#define B_EINTR 0x00000400 /* I/O was interrupted */
|
|
|
|
#define B_ERROR 0x00000800 /* I/O error occurred. */
|
|
|
|
#define B_GATHERED 0x00001000 /* LFS: already in a segment. */
|
|
|
|
#define B_INVAL 0x00002000 /* Does not contain valid info. */
|
|
|
|
#define B_LOCKED 0x00004000 /* Locked in core (not reusable). */
|
|
|
|
#define B_NOCACHE 0x00008000 /* Do not cache block after use. */
|
These changes embody the support of the fully coherent merged VM buffer cache,
much higher filesystem I/O performance, and much better paging performance. It
represents the culmination of over 6 months of R&D.
The majority of the merged VM/cache work is by John Dyson.
The following highlights the most significant changes. Additionally, there are
(mostly minor) changes to the various filesystem modules (nfs, msdosfs, etc) to
support the new VM/buffer scheme.
vfs_bio.c:
Significant rewrite of most of vfs_bio to support the merged VM buffer cache
scheme. The scheme is almost fully compatible with the old filesystem
interface. Significant improvement in the number of opportunities for write
clustering.
vfs_cluster.c, vfs_subr.c
Upgrade and performance enhancements in vfs layer code to support merged
VM/buffer cache. Fixup of vfs_cluster to eliminate the bogus pagemove stuff.
vm_object.c:
Yet more improvements in the collapse code. Elimination of some windows that
can cause list corruption.
vm_pageout.c:
Fixed it, it really works better now. Somehow in 2.0, some "enhancements"
broke the code. This code has been reworked from the ground-up.
vm_fault.c, vm_page.c, pmap.c, vm_object.c
Support for small-block filesystems with merged VM/buffer cache scheme.
pmap.c vm_map.c
Dynamic kernel VM size, now we dont have to pre-allocate excessive numbers of
kernel PTs.
vm_glue.c
Much simpler and more effective swapping code. No more gratuitous swapping.
proc.h
Fixed the problem that the p_lock flag was not being cleared on a fork.
swap_pager.c, vnode_pager.c
Removal of old vfs_bio cruft to support the past pseudo-coherency. Now the
code doesn't need it anymore.
machdep.c
Changes to better support the parameter values for the merged VM/buffer cache
scheme.
machdep.c, kern_exec.c, vm_glue.c
Implemented a seperate submap for temporary exec string space and another one
to contain process upages. This eliminates all map fragmentation problems
that previously existed.
ffs_inode.c, ufs_inode.c, ufs_readwrite.c
Changes for merged VM/buffer cache. Add "bypass" support for sneaking in on
busy buffers.
Submitted by: John Dyson and David Greenman
1995-01-09 16:06:02 +00:00
|
|
|
#define B_MALLOC 0x00010000 /* malloced b_data */
|
1996-09-06 05:35:00 +00:00
|
|
|
#define B_CLUSTEROK 0x00020000 /* Pagein op, so swap() can count it. */
|
1994-05-24 10:09:53 +00:00
|
|
|
#define B_PHYS 0x00040000 /* I/O to user memory. */
|
|
|
|
#define B_RAW 0x00080000 /* Set by physio for raw transfers. */
|
|
|
|
#define B_READ 0x00100000 /* Read buffer. */
|
|
|
|
#define B_TAPE 0x00200000 /* Magnetic tape I/O. */
|
1995-04-09 06:03:56 +00:00
|
|
|
#define B_RELBUF 0x00400000 /* Release VMIO buffer. */
|
1994-05-24 10:09:53 +00:00
|
|
|
#define B_WANTED 0x00800000 /* Process wants this buffer. */
|
|
|
|
#define B_WRITE 0x00000000 /* Write buffer (pseudo flag). */
|
|
|
|
#define B_WRITEINPROG 0x01000000 /* Write in progress. */
|
|
|
|
#define B_XXX 0x02000000 /* Debugging flag. */
|
1995-04-19 10:32:11 +00:00
|
|
|
#define B_PAGING 0x04000000 /* volatile paging I/O -- bypass VMIO */
|
1996-09-06 05:35:00 +00:00
|
|
|
#define B_ORDERED 0x08000000 /* Must guarantee I/O ordering */
|
1996-12-29 02:45:28 +00:00
|
|
|
#define B_RAM 0x10000000 /* Read ahead mark (flag) */
|
1994-10-05 09:48:45 +00:00
|
|
|
#define B_VMIO 0x20000000 /* VMIO flag */
|
1994-05-25 09:21:21 +00:00
|
|
|
#define B_CLUSTER 0x40000000 /* pagein op, so swap() can count it */
|
|
|
|
#define B_BOUNCE 0x80000000 /* bounce buffer flag */
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1997-09-21 22:09:24 +00:00
|
|
|
typedef struct buf_queue_head {
|
|
|
|
TAILQ_HEAD(, buf) queue;
|
|
|
|
struct buf *insert_point;
|
|
|
|
struct buf *switch_point;
|
|
|
|
} buf_queue_head, *buf_queue_head_t;
|
|
|
|
|
|
|
|
static __inline void bufq_init __P((buf_queue_head *head));
|
|
|
|
|
|
|
|
static __inline void bufq_insert_tail __P((buf_queue_head *head,
|
|
|
|
struct buf *bp));
|
|
|
|
|
|
|
|
static __inline void bufq_remove __P((buf_queue_head *head,
|
|
|
|
struct buf *bp));
|
|
|
|
|
|
|
|
static __inline struct buf *bufq_first __P((buf_queue_head *head));
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
bufq_init(buf_queue_head *head)
|
|
|
|
{
|
|
|
|
TAILQ_INIT(&head->queue);
|
|
|
|
head->insert_point = NULL;
|
|
|
|
head->switch_point = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
bufq_insert_tail(buf_queue_head *head, struct buf *bp)
|
|
|
|
{
|
|
|
|
if ((bp->b_flags & B_ORDERED) != 0) {
|
|
|
|
head->insert_point = bp;
|
|
|
|
head->switch_point = NULL;
|
|
|
|
}
|
|
|
|
TAILQ_INSERT_TAIL(&head->queue, bp, b_act);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
bufq_remove(buf_queue_head *head, struct buf *bp)
|
|
|
|
{
|
|
|
|
if (bp == TAILQ_FIRST(&head->queue)) {
|
|
|
|
if (bp == head->insert_point)
|
|
|
|
head->insert_point = NULL;
|
|
|
|
if (TAILQ_NEXT(bp, b_act) == head->switch_point)
|
|
|
|
head->switch_point = NULL;
|
|
|
|
} else {
|
|
|
|
if (bp == head->insert_point) {
|
|
|
|
/*
|
|
|
|
* Not 100% correct (we really want the
|
|
|
|
* previous bp), but it will ensure queue
|
|
|
|
* ordering and is less expensive than
|
|
|
|
* using a CIRCLEQ.
|
|
|
|
*/
|
1997-10-23 11:32:06 +00:00
|
|
|
head->insert_point = TAILQ_NEXT(bp, b_act);
|
1997-09-21 22:09:24 +00:00
|
|
|
}
|
|
|
|
if (bp == head->switch_point) {
|
1997-10-23 11:32:06 +00:00
|
|
|
head->switch_point = TAILQ_NEXT(bp, b_act);
|
1997-09-21 22:09:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
TAILQ_REMOVE(&head->queue, bp, b_act);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline struct buf *
|
|
|
|
bufq_first(buf_queue_head *head)
|
|
|
|
{
|
|
|
|
return (TAILQ_FIRST(&head->queue));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1995-05-30 08:16:23 +00:00
|
|
|
/*
|
1994-05-26 08:45:29 +00:00
|
|
|
* number of buffer hash entries
|
|
|
|
*/
|
|
|
|
#define BUFHSZ 512
|
|
|
|
|
|
|
|
/*
|
|
|
|
* buffer hash table calculation, originally by David Greenman
|
|
|
|
*/
|
|
|
|
#define BUFHASH(vnp, bn) \
|
1995-08-24 12:57:17 +00:00
|
|
|
(&bufhashtbl[(((unsigned long)(vnp) >> 7)+(int)(bn)) % BUFHSZ])
|
1994-05-26 08:45:29 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Definitions for the buffer free lists.
|
|
|
|
*/
|
These changes embody the support of the fully coherent merged VM buffer cache,
much higher filesystem I/O performance, and much better paging performance. It
represents the culmination of over 6 months of R&D.
The majority of the merged VM/cache work is by John Dyson.
The following highlights the most significant changes. Additionally, there are
(mostly minor) changes to the various filesystem modules (nfs, msdosfs, etc) to
support the new VM/buffer scheme.
vfs_bio.c:
Significant rewrite of most of vfs_bio to support the merged VM buffer cache
scheme. The scheme is almost fully compatible with the old filesystem
interface. Significant improvement in the number of opportunities for write
clustering.
vfs_cluster.c, vfs_subr.c
Upgrade and performance enhancements in vfs layer code to support merged
VM/buffer cache. Fixup of vfs_cluster to eliminate the bogus pagemove stuff.
vm_object.c:
Yet more improvements in the collapse code. Elimination of some windows that
can cause list corruption.
vm_pageout.c:
Fixed it, it really works better now. Somehow in 2.0, some "enhancements"
broke the code. This code has been reworked from the ground-up.
vm_fault.c, vm_page.c, pmap.c, vm_object.c
Support for small-block filesystems with merged VM/buffer cache scheme.
pmap.c vm_map.c
Dynamic kernel VM size, now we dont have to pre-allocate excessive numbers of
kernel PTs.
vm_glue.c
Much simpler and more effective swapping code. No more gratuitous swapping.
proc.h
Fixed the problem that the p_lock flag was not being cleared on a fork.
swap_pager.c, vnode_pager.c
Removal of old vfs_bio cruft to support the past pseudo-coherency. Now the
code doesn't need it anymore.
machdep.c
Changes to better support the parameter values for the merged VM/buffer cache
scheme.
machdep.c, kern_exec.c, vm_glue.c
Implemented a seperate submap for temporary exec string space and another one
to contain process upages. This eliminates all map fragmentation problems
that previously existed.
ffs_inode.c, ufs_inode.c, ufs_readwrite.c
Changes for merged VM/buffer cache. Add "bypass" support for sneaking in on
busy buffers.
Submitted by: John Dyson and David Greenman
1995-01-09 16:06:02 +00:00
|
|
|
#define BUFFER_QUEUES 6 /* number of free buffer queues */
|
1994-05-26 08:45:29 +00:00
|
|
|
|
|
|
|
#define QUEUE_NONE 0 /* on no queue */
|
|
|
|
#define QUEUE_LOCKED 1 /* locked buffers */
|
|
|
|
#define QUEUE_LRU 2 /* useful buffers */
|
These changes embody the support of the fully coherent merged VM buffer cache,
much higher filesystem I/O performance, and much better paging performance. It
represents the culmination of over 6 months of R&D.
The majority of the merged VM/cache work is by John Dyson.
The following highlights the most significant changes. Additionally, there are
(mostly minor) changes to the various filesystem modules (nfs, msdosfs, etc) to
support the new VM/buffer scheme.
vfs_bio.c:
Significant rewrite of most of vfs_bio to support the merged VM buffer cache
scheme. The scheme is almost fully compatible with the old filesystem
interface. Significant improvement in the number of opportunities for write
clustering.
vfs_cluster.c, vfs_subr.c
Upgrade and performance enhancements in vfs layer code to support merged
VM/buffer cache. Fixup of vfs_cluster to eliminate the bogus pagemove stuff.
vm_object.c:
Yet more improvements in the collapse code. Elimination of some windows that
can cause list corruption.
vm_pageout.c:
Fixed it, it really works better now. Somehow in 2.0, some "enhancements"
broke the code. This code has been reworked from the ground-up.
vm_fault.c, vm_page.c, pmap.c, vm_object.c
Support for small-block filesystems with merged VM/buffer cache scheme.
pmap.c vm_map.c
Dynamic kernel VM size, now we dont have to pre-allocate excessive numbers of
kernel PTs.
vm_glue.c
Much simpler and more effective swapping code. No more gratuitous swapping.
proc.h
Fixed the problem that the p_lock flag was not being cleared on a fork.
swap_pager.c, vnode_pager.c
Removal of old vfs_bio cruft to support the past pseudo-coherency. Now the
code doesn't need it anymore.
machdep.c
Changes to better support the parameter values for the merged VM/buffer cache
scheme.
machdep.c, kern_exec.c, vm_glue.c
Implemented a seperate submap for temporary exec string space and another one
to contain process upages. This eliminates all map fragmentation problems
that previously existed.
ffs_inode.c, ufs_inode.c, ufs_readwrite.c
Changes for merged VM/buffer cache. Add "bypass" support for sneaking in on
busy buffers.
Submitted by: John Dyson and David Greenman
1995-01-09 16:06:02 +00:00
|
|
|
#define QUEUE_VMIO 3 /* VMIO buffers */
|
|
|
|
#define QUEUE_AGE 4 /* not-useful buffers */
|
|
|
|
#define QUEUE_EMPTY 5 /* empty buffer headers*/
|
1994-05-26 08:45:29 +00:00
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
/*
|
|
|
|
* Zero out the buffer's data area.
|
|
|
|
*/
|
|
|
|
#define clrbuf(bp) { \
|
1995-12-28 23:34:30 +00:00
|
|
|
bzero((bp)->b_data, (u_int)(bp)->b_bcount); \
|
1994-05-24 10:09:53 +00:00
|
|
|
(bp)->b_resid = 0; \
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flags to low-level allocation routines. */
|
|
|
|
#define B_CLRBUF 0x01 /* Request allocated buffer be cleared. */
|
|
|
|
#define B_SYNC 0x02 /* Do all allocations synchronously. */
|
|
|
|
|
|
|
|
#ifdef KERNEL
|
1994-08-18 22:36:09 +00:00
|
|
|
extern int nbuf; /* The number of buffer headers */
|
|
|
|
extern struct buf *buf; /* The buffer headers. */
|
|
|
|
extern char *buffers; /* The buffer contents. */
|
|
|
|
extern int bufpages; /* Number of memory pages in the buffer pool. */
|
|
|
|
extern struct buf *swbuf; /* Swap I/O buffer headers. */
|
|
|
|
extern int nswbuf; /* Number of swap I/O buffer headers. */
|
1997-06-15 17:56:53 +00:00
|
|
|
extern int needsbuffer, numdirtybuffers;
|
1994-08-18 22:36:09 +00:00
|
|
|
extern TAILQ_HEAD(swqueue, buf) bswlist;
|
This mega-commit is meant to fix numerous interrelated problems. There
has been some bitrot and incorrect assumptions in the vfs_bio code. These
problems have manifest themselves worse on NFS type filesystems, but can
still affect local filesystems under certain circumstances. Most of
the problems have involved mmap consistancy, and as a side-effect broke
the vfs.ioopt code. This code might have been committed seperately, but
almost everything is interrelated.
1) Allow (pmap_object_init_pt) prefaulting of buffer-busy pages that
are fully valid.
2) Rather than deactivating erroneously read initial (header) pages in
kern_exec, we now free them.
3) Fix the rundown of non-VMIO buffers that are in an inconsistent
(missing vp) state.
4) Fix the disassociation of pages from buffers in brelse. The previous
code had rotted and was faulty in a couple of important circumstances.
5) Remove a gratuitious buffer wakeup in vfs_vmio_release.
6) Remove a crufty and currently unused cluster mechanism for VBLK
files in vfs_bio_awrite. When the code is functional, I'll add back
a cleaner version.
7) The page busy count wakeups assocated with the buffer cache usage were
incorrectly cleaned up in a previous commit by me. Revert to the
original, correct version, but with a cleaner implementation.
8) The cluster read code now tries to keep data associated with buffers
more aggressively (without breaking the heuristics) when it is presumed
that the read data (buffers) will be soon needed.
9) Change to filesystem lockmgr locks so that they use LK_NOPAUSE. The
delay loop waiting is not useful for filesystem locks, due to the
length of the time intervals.
10) Correct and clean-up spec_getpages.
11) Implement a fully functional nfs_getpages, nfs_putpages.
12) Fix nfs_write so that modifications are coherent with the NFS data on
the server disk (at least as well as NFS seems to allow.)
13) Properly support MS_INVALIDATE on NFS.
14) Properly pass down MS_INVALIDATE to lower levels of the VM code from
vm_map_clean.
15) Better support the notion of pages being busy but valid, so that
fewer in-transit waits occur. (use p->busy more for pageouts instead
of PG_BUSY.) Since the page is fully valid, it is still usable for
reads.
16) It is possible (in error) for cached pages to be busy. Make the
page allocation code handle that case correctly. (It should probably
be a printf or panic, but I want the system to handle coding errors
robustly. I'll probably add a printf.)
17) Correct the design and usage of vm_page_sleep. It didn't handle
consistancy problems very well, so make the design a little less
lofty. After vm_page_sleep, if it ever blocked, it is still important
to relookup the page (if the object generation count changed), and
verify it's status (always.)
18) In vm_pageout.c, vm_pageout_clean had rotted, so clean that up.
19) Push the page busy for writes and VM_PROT_READ into vm_pageout_flush.
20) Fix vm_pager_put_pages and it's descendents to support an int flag
instead of a boolean, so that we can pass down the invalidate bit.
1998-03-07 21:37:31 +00:00
|
|
|
extern TAILQ_HEAD(bqueues, buf) bufqueues[BUFFER_QUEUES];
|
1994-05-24 10:09:53 +00:00
|
|
|
|
1994-09-25 19:34:02 +00:00
|
|
|
void bufinit __P((void));
|
|
|
|
void bremfree __P((struct buf *));
|
1994-05-24 10:09:53 +00:00
|
|
|
int bread __P((struct vnode *, daddr_t, int,
|
|
|
|
struct ucred *, struct buf **));
|
|
|
|
int breadn __P((struct vnode *, daddr_t, int, daddr_t *, int *, int,
|
|
|
|
struct ucred *, struct buf **));
|
|
|
|
int bwrite __P((struct buf *));
|
1994-09-25 19:34:02 +00:00
|
|
|
void bdwrite __P((struct buf *));
|
|
|
|
void bawrite __P((struct buf *));
|
1996-09-06 05:35:00 +00:00
|
|
|
int bowrite __P((struct buf *));
|
1994-09-25 19:34:02 +00:00
|
|
|
void brelse __P((struct buf *));
|
1996-01-19 04:00:31 +00:00
|
|
|
void bqrelse __P((struct buf *));
|
1995-12-11 04:58:34 +00:00
|
|
|
int vfs_bio_awrite __P((struct buf *));
|
1994-10-10 00:58:36 +00:00
|
|
|
struct buf * getpbuf __P((void));
|
1994-09-25 19:34:02 +00:00
|
|
|
struct buf *incore __P((struct vnode *, daddr_t));
|
1995-12-11 04:58:34 +00:00
|
|
|
struct buf *gbincore __P((struct vnode *, daddr_t));
|
1995-03-16 18:17:34 +00:00
|
|
|
int inmem __P((struct vnode *, daddr_t));
|
1994-09-25 19:34:02 +00:00
|
|
|
struct buf *getblk __P((struct vnode *, daddr_t, int, int, int));
|
|
|
|
struct buf *geteblk __P((int));
|
1996-03-11 02:04:27 +00:00
|
|
|
int allocbuf __P((struct buf *, int));
|
1994-09-25 19:34:02 +00:00
|
|
|
int biowait __P((struct buf *));
|
|
|
|
void biodone __P((struct buf *));
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
void cluster_callback __P((struct buf *));
|
|
|
|
int cluster_read __P((struct vnode *, u_quad_t, daddr_t, long,
|
1996-12-29 02:45:28 +00:00
|
|
|
struct ucred *, long, int, struct buf **));
|
1995-12-11 04:58:34 +00:00
|
|
|
int cluster_wbuild __P((struct vnode *, long, daddr_t, int));
|
1994-05-24 10:09:53 +00:00
|
|
|
void cluster_write __P((struct buf *, u_quad_t));
|
1995-11-20 12:35:16 +00:00
|
|
|
int physio __P((void (*)(struct buf *), struct buf *, dev_t,
|
|
|
|
int, u_int (*)(struct buf *), struct uio *));
|
1994-10-10 00:58:36 +00:00
|
|
|
u_int minphys __P((struct buf *));
|
1995-04-09 06:03:56 +00:00
|
|
|
void vfs_bio_clrbuf __P((struct buf *));
|
1995-03-16 18:17:34 +00:00
|
|
|
void vfs_busy_pages __P((struct buf *, int clear_modify));
|
1997-09-16 11:44:05 +00:00
|
|
|
void vfs_unbusy_pages __P((struct buf *));
|
1994-10-10 00:58:36 +00:00
|
|
|
void vwakeup __P((struct buf *));
|
|
|
|
void vmapbuf __P((struct buf *));
|
|
|
|
void vunmapbuf __P((struct buf *));
|
|
|
|
void relpbuf __P((struct buf *));
|
|
|
|
void brelvp __P((struct buf *));
|
|
|
|
void bgetvp __P((struct vnode *, struct buf *));
|
1995-03-16 18:17:34 +00:00
|
|
|
void pbgetvp __P((struct vnode *, struct buf *));
|
|
|
|
void pbrelvp __P((struct buf *));
|
1994-10-10 00:58:36 +00:00
|
|
|
void reassignbuf __P((struct buf *, struct vnode *));
|
1995-03-16 18:17:34 +00:00
|
|
|
struct buf *trypbuf __P((void));
|
1995-03-28 07:58:53 +00:00
|
|
|
void vm_bounce_alloc __P((struct buf *));
|
|
|
|
void vm_bounce_free __P((struct buf *));
|
|
|
|
vm_offset_t vm_bounce_kva_alloc __P((int));
|
|
|
|
void vm_bounce_kva_alloc_free __P((vm_offset_t, int));
|
1997-06-15 17:56:53 +00:00
|
|
|
void vfs_bio_need_satisfy __P((void));
|
1996-05-01 02:16:17 +00:00
|
|
|
#endif /* KERNEL */
|
|
|
|
|
1994-05-24 10:09:53 +00:00
|
|
|
#endif /* !_SYS_BUF_H_ */
|