diff options
Diffstat (limited to 'share/man/man9/disk.9')
-rw-r--r-- | share/man/man9/disk.9 | 354 |
1 files changed, 354 insertions, 0 deletions
diff --git a/share/man/man9/disk.9 b/share/man/man9/disk.9 new file mode 100644 index 00000000000..41b80b5758c --- /dev/null +++ b/share/man/man9/disk.9 @@ -0,0 +1,354 @@ +.\" $NetBSD: disk.9,v 1.1 1996/01/09 00:22:04 thorpej Exp $ +.\" +.\" Copyright (c) 1995, 1996 Jason R. Thorpe. +.\" 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 for the NetBSD Project +.\" by Jason R. Thorpe. +.\" 4. The name of the author may not be used to endorse or promote products +.\" derived from this software without specific prior written permission. +.\" +.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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. +.\" +.Dd Jan 7, 1996 +.Dt DISK 9 +.Os NetBSD +.Sh NAME +.Nm disk +.Nd generic disk framework +.Sh SYNOPSIS +.Fd #include <sys/types.h> +.Fd #include <sys/disklabel.h> +.Fd #include <sys/disk.h> +.Ft void +.Fn disk_init "void" +.Ft void +.Fn disk_attach "struct disk *" +.Ft void +.Fn disk_detatch "struct disk *" +.Ft void +.Fn disk_busy "struct disk *" +.Ft void +.Fn disk_unbusy "struct disk *" +.Ft void +.Fn disk_resetstat "struct disk *" +.Ft struct disk * +.Fn disk_find "char *" +.Sh DESCRIPTION +The NetBSD generic disk framework is designed to provide flexible, +scalable, and consistent handling of disk state and metrics information. +The fundamental component of this framework is the +.Nm disk +structure, which is defined as follows: +.Bd -literal +struct disk { + TAILQ_ENTRY(disk) dk_link; /* link in global disklist */ + char *dk_name; /* disk name */ + int dk_bopenmask; /* block devices open */ + int dk_copenmask; /* character devices open */ + int dk_openmask; /* composite (bopen|copen) */ + int dk_state; /* label state */ + int dk_blkshift; /* shift to convert DEV_BSIZE to blks */ + int dk_byteshift; /* shift to convert bytes to blks */ + + /* + * Metrics data; note that some metrics may have no meaning + * on certain types of disks. + */ + int dk_busy; /* busy counter */ + u_int64_t dk_xfer; /* total number of transfers */ + u_int64_t dk_seek; /* total independent seek operations */ + u_int64_t dk_bytes; /* total bytes transfered */ + struct timeval dk_attachtime; /* time disk was attached */ + struct timeval dk_timestamp; /* timestamp of last unbusy */ + struct timeval dk_time; /* total time spent busy */ + + struct dkdriver *dk_driver; /* pointer to driver */ + + /* + * Disk label information. Storage for the in-core disk label + * must be dynamically allocated, otherwise the size of this + * structure becomes machine-dependent. + */ + daddr_t dk_labelsector; /* sector containing label */ + struct disklabel *dk_label; /* label */ + struct cpu_disklabel *dk_cpulabel; +}; +.Ed +.Pp +The system maintains a global linked-list of all disks attached to the +system. This list, called +.Nm disklist , +may grow or shrink over time as disks are dynamically added and removed +from the system. Drivers which currently make use of the detachment +capability of the framework are the +.Nm ccd +and +.Nm vnd +pseudo-device drivers. +.Pp +The following is a brief description of each function in the framework: +.Bl -tag -width "disk_resetstat()" +.It Fn disk_init +Initialize the disklist and other data structures used by the framework. +Called by +.Fn main +before autoconfiguration. +.It Fn disk_attach +Attach a disk; allocate storage for the disklabel, set the +.Dq attached time +timestamp, insert the disk into the disklist, and intrement the +system disk count. +.It Fn disk_detatch +Detatch a disk; free storage for the disklabel, remove the disk +from the disklist, and decrement the system disk count. If the count +drops below zero, panic. +.It Fn disk_busy +Increment the disk's +.Dq busy counter . +If this counter goes from 0 to 1, set the timestamp corresponding to +this transfer. +.It Fn disk_unbusy +Decrement a disk's busy counter. If the count drops below zero, panic. +Get the current time, subtract it from the disk's timestamp, and add +the difference to the disk's running total. Set the disk's timestamp +to the current time. If the provided byte count is greater than 0, +add it to the disk's running total and increment the number of transfers +performed by the disk. +.It Fn disk_resetstat +Reset the running byte, transfer, and time totals. +.It Fn disk_find +Return a pointer to the disk structure corresponding to the name provided, +or NULL if the disk does not exist. +.El +.Pp +The functions typically called by device drivers are +.Fn disk_attach , +.Fn disk_detatch , +.Fn disk_busy , +.Fn disk_unbusy , +and +.Fn disk_resetstat . +The function +.Fn disk_find +is provided as a utility function. +.Sh USING THE FRAMEWORK +This section includes a description on basic use of the framework +and example usage of its functions. Actual implementation of +a device driver which utilizes the framework may vary. +.Pp +A special routine, +.Fn disk_init , +is provided to perform basic initialization of data structures used by +the framework. It is called exactly once by the system, in +.Fn main , +before device autoconfiguration. +.Pp +Each device in the system uses a +.Dq softc +structure which contains autoconfiguration and state information for that +device. In the case of disks, the softc should also contain one instance +of the disk stucture, eg: +.Bd -literal +struct foo_softc { + struct device *sc_dev; /* generic device information */ + struct disk *sc_dk; /* generic disk information */ + [ . . . more . . . ] +}; +.Ed +.Pp +In order for the system to gather metrics data about a disk, the disk must +be registered with the system. The +.Fn disk_attach +routine performs all of the functions currently required to register a disk +with the system including allocation of disklabel storage space, +recording of the time since boot that the disk was attached, and insertion +into the disklist. Note that since this function allocates storage space +for the disklabel, it must be called before the disklabel is read from the +media or used in any other way. Before +.Fn disk_attach +is called, a portions of the disk structure must be initialized with +data specific to that disk. For example, in the +.Dq foo +disk driver, the following would be performed in the autoconfiguration +.Dq attach +routine: +.Bd -literal +void +fooattach(parent, self, aux) + struct device *parent, *self; + void *aux; +{ + struct foo_softc *sc = (struct foo_softc *)self; + [ . . . ] + + /* Initialize and attach the disk structure. */ + sc->sc_dk.dk_driver = &foodkdriver; + sc->sc_dk.dk_name = sc->sc_dev.dv_xname; + disk_attach(&sc->sc_dk); + + /* Read geometry and fill in pertinent parts of disklabel. */ + [ . . . ] +} +.Ed +.Pp +The +.Nm foodkdriver +above is the disk's +.Dq driver +switch. This switch currently includes a pointer to the disk's +.Dq strategy +routine. This switch needs to have global scope and sould be initialized +as follows: +.Bd -literal +void foostrategy __P((struct buf *)); +struct dkdriver foodkdriver = { foostrategy }; +.Ed +.Pp +Once the disk is attached, metrics may be gathered on that disk. In order +to gather metrics data, the driver must tell the framework when the disk +starts and stops operations. This functionality is provided by the +.Fn disk_busy +and +.Fn disk_unbusy +routines. The +.Fn disk_busy +routine should be called immediately before a command to the disk is +sent, eg: +.Bd -literal +void +foostart(sc) + struct foo_softc *sc; +{ + [ . . . ] + + /* Get buffer from drive's transfer queue. */ + [ . . . ] + + /* Build command to send to drive. */ + [ . . . ] + + /* Tell the disk framework we're is going busy. */ + disk_busy(&sc->sc_dk); + + /* Send command to the drive. */ + [ . . . ] +} +.Ed +.Pp +When +.Fn disk_busy +is called, a timestamp is taken if the disk's busy counter moves from +0 to 1, indicating the disk has gone from an idle to non-idle state. +Note that +.Fn disk_busy +must be called at +.Fn splbio . +At the end of a transaction, the +.Fn disk_unbusy +routine should be called. This routine performs some consistency checks, +such as ensuring that the calls to +.Fn disk_busy +and +.Fn disk_unbusy +are balanced. This routine also performs the actual metrics calculation. +A timestamp is taken, and the difference from the timestamp taken in +.Fn disk_busy +is added to the disk's total running time. The disk's timestamp is then +updated in case there is more than one pending transfer on the disk. +A byte count is also added to the disk's running total, and if greater than +zero, the number of transfers the disk has performed is incremented. +.Bd -literal +void +foodone(xfer) + struct foo_xfer *xfer; +{ + struct foo_softc = (struct foo_softc *)xfer->xf_softc; + struct buf *bp = xfer->xf_buf; + long nbytes; + [ . . . ] + + /* + * Get number of bytes transfered. If there is no buf + * associated with the xfer, we are being called at the + * end of a non-I/O command. + */ + if (bp == NULL) + nbytes = 0; + else + nbytes = bp->b_bcount - bp->b_resid; + + [ . . . ] + + /* Notify the disk framework that we've completed the transfer. */ + disk_unbusy(&sc->sc_dk, nbytes); + + [ . . . ] +} +.Ed +.Pp +Like +.Fn disk_busy , +.Fn disk_unbusy +must be called at +.Fn splbio . +.Pp +At some point a driver may wish to reset the metrics data gathered on a +particular disk. For this function, the +.Fn disk_resetstat +routine is provided. +.Sh CODE REFERENCES +This section describes places within the NetBSD source tree where actual +code implementing or utilizing the disk framework can be found. All +pathnames are relative to +.Nm /usr/src . +.Pp +The disk framework itself is implemented within the file +.Nm sys/kern/subr_disk.c . +Data structures and function prototypes for the framework are located in +.Nm sys/sys/disk.h . +.Pp +The NetBSD machine-independent SCSI disk and CD-ROM drivers utilize the +disk framework. They are located in +.Nm sys/scsi/sd.c +and +.Nm sys/scsi/cd.c . +.Pp +The NetBSD +.Nm ccd +and +.Nm vnd +drivers utilize the detachment capability of the framework. +They are located in +.Nm sys/dev/ccd.c +and +.Nm sys/dev/vnd.c . +.Sh AUTHOR +The NetBSD generic disk framework was architected and implemented by +Jason R. Thorpe <thorpej@NetBSD.ORG>. +.Sh SEE ALSO +.Xr ccd 4 , +.Xr vnd 4 , +.Xr spl 9 . +.Sh HISTORY +The NetBSD generic disk framework appeared in NetBSD 1.1A. |