NetBSD/sys/dev/scsipi/scsipiconf.h

932 lines
30 KiB
C

/* $NetBSD: scsipiconf.h,v 1.130 2019/03/28 10:44:29 kardel Exp $ */
/*-
* Copyright (c) 1998, 1999, 2000, 2004 The NetBSD Foundation, Inc.
* All rights reserved.
*
* This code is derived from software contributed to The NetBSD Foundation
* by Charles M. Hannum; by Jason R. Thorpe of the Numerical Aerospace
* Simulation Facility, NASA Ames Research Center.
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
*/
/*
* Originally written by Julian Elischer (julian@tfs.com)
* for TRW Financial Systems for use under the MACH(2.5) operating system.
*
* TRW Financial Systems, in accordance with their agreement with Carnegie
* Mellon University, makes this software available to CMU to distribute
* or use in any manner that they see fit as long as this message is kept with
* the software. For this reason TFS also grants any other persons or
* organisations permission to use or modify this software.
*
* TFS supplies this software to be publicly redistributed
* on the understanding that TFS is not responsible for the correct
* functioning of this software in any circumstances.
*
* Ported to run under 386BSD by Julian Elischer (julian@tfs.com) Sept 1992
*/
#ifndef _DEV_SCSIPI_SCSIPICONF_H_
#define _DEV_SCSIPI_SCSIPICONF_H_
typedef int boolean;
#include <sys/callout.h>
#include <sys/queue.h>
#include <sys/condvar.h>
#include <dev/scsipi/scsi_spc.h>
#include <dev/scsipi/scsipi_debug.h>
struct buf;
struct proc;
struct device;
struct scsipi_channel;
struct scsipi_periph;
struct scsipi_xfer;
/*
* The following defines the scsipi_xfer queue.
*/
TAILQ_HEAD(scsipi_xfer_queue, scsipi_xfer);
struct scsipi_generic {
u_int8_t opcode;
u_int8_t bytes[15];
};
/*
* scsipi_async_event_t:
*
* Asynchronous events from the adapter to the mid-layer and
* peripheral.
*
* Arguments:
*
* ASYNC_EVENT_MAX_OPENINGS scsipi_max_openings * -- max
* openings, device specified in
* parameters
*
* ASYNC_EVENT_XFER_MODE scsipi_xfer_mode * -- xfer mode
* parameters changed for I_T Nexus
* ASYNC_EVENT_RESET NULL - channel has been reset
*/
typedef enum {
ASYNC_EVENT_MAX_OPENINGS, /* set max openings on periph */
ASYNC_EVENT_XFER_MODE, /* xfer mode update for I_T */
ASYNC_EVENT_RESET /* channel reset */
} scsipi_async_event_t;
/*
* scsipi_max_openings:
*
* Argument for an ASYNC_EVENT_MAX_OPENINGS event.
*/
struct scsipi_max_openings {
int mo_target; /* openings are for this target... */
int mo_lun; /* ...and this lun */
int mo_openings; /* openings value */
};
/*
* scsipi_xfer_mode:
*
* Argument for an ASYNC_EVENT_XFER_MODE event.
*/
struct scsipi_xfer_mode {
int xm_target; /* target, for I_T Nexus */
int xm_mode; /* PERIPH_CAP* bits */
int xm_period; /* sync period */
int xm_offset; /* sync offset */
};
/*
* scsipi_adapter_req_t:
*
* Requests that can be made of an adapter.
*
* Arguments:
*
* ADAPTER_REQ_RUN_XFER scsipi_xfer * -- the xfer which
* is to be run
*
* ADAPTER_REQ_GROW_RESOURCES no argument
*
* ADAPTER_REQ_SET_XFER_MODE scsipi_xfer_mode * -- set the xfer
* mode for the I_T Nexus according to
* this
*/
typedef enum {
ADAPTER_REQ_RUN_XFER, /* run a scsipi_xfer */
ADAPTER_REQ_GROW_RESOURCES, /* grow xfer execution resources */
ADAPTER_REQ_SET_XFER_MODE /* set xfer mode */
} scsipi_adapter_req_t;
#ifdef _KERNEL
/*
* scsipi_periphsw:
*
* Callbacks into periph driver from midlayer.
*
* psw_error Called by the bustype's interpret-sense routine
* to do periph-specific sense handling.
*
* psw_start Called by midlayer to restart a device once
* more command openings become available.
*
* psw_async Called by midlayer when an asynchronous event
* from the adapter occurs.
*
* psw_done Called by the midlayer when an xfer has completed.
*/
struct scsipi_periphsw {
int (*psw_error)(struct scsipi_xfer *);
void (*psw_start)(struct scsipi_periph *);
int (*psw_async)(struct scsipi_periph *,
scsipi_async_event_t, void *);
void (*psw_done)(struct scsipi_xfer *, int);
};
struct disk_parms;
struct scsipi_inquiry_pattern;
/*
* scsipi_adapter:
*
* This structure describes an instance of a SCSIPI adapter.
*
* Note that `adapt_openings' is used by (the common case of) adapters
* which have per-adapter resources. If an adapter's command resources
* are associated with a channel, then the `chan_openings' below will
* be used instead.
*
* Note that all adapter entry points take a pointer to a channel,
* as an adapter may have more than one channel, and the channel
* structure contains the channel number.
*/
struct scsipi_adapter {
device_t adapt_dev; /* pointer to adapter's device */
int adapt_nchannels; /* number of adapter channels */
volatile int adapt_refcnt; /* adapter's reference count */
int adapt_openings; /* total # of command openings */
int adapt_max_periph; /* max openings per periph */
int adapt_flags;
void (*adapt_request)(struct scsipi_channel *,
scsipi_adapter_req_t, void *);
void (*adapt_minphys)(struct buf *);
int (*adapt_ioctl)(struct scsipi_channel *, u_long,
void *, int, struct proc *);
int (*adapt_enable)(device_t, int);
int (*adapt_getgeom)(struct scsipi_periph *,
struct disk_parms *, u_long);
int (*adapt_accesschk)(struct scsipi_periph *,
struct scsipi_inquiry_pattern *);
kmutex_t adapt_mtx;
volatile int adapt_running; /* how many users of mutex */
};
/* adapt_flags */
#define SCSIPI_ADAPT_POLL_ONLY 0x01 /* Adaptor can't do interrupts. */
#define SCSIPI_ADAPT_MPSAFE 0x02 /* Adaptor doesn't need kernel lock */
void scsipi_adapter_minphys(struct scsipi_channel *, struct buf *);
void scsipi_adapter_request(struct scsipi_channel *,
scsipi_adapter_req_t, void *);
int scsipi_adapter_ioctl(struct scsipi_channel *, u_long,
void *, int, struct proc *);
int scsipi_adapter_enable(struct scsipi_adapter *, int);
#endif
/*
* scsipi_bustype:
*
* This structure describes a SCSIPI bus type.
* The bustype_type member is shared with struct ata_bustype
* (because we can ata, atapi or scsi busses to the same controller)
*/
struct scsipi_bustype {
int bustype_type; /* symbolic name of type */
void (*bustype_cmd)(struct scsipi_xfer *);
int (*bustype_interpret_sense)(struct scsipi_xfer *);
void (*bustype_printaddr)(struct scsipi_periph *);
void (*bustype_kill_pending)(struct scsipi_periph *);
void (*bustype_async_event_xfer_mode)(struct scsipi_channel *,
void *);
};
/* bustype_type */
/* type is stored in the first byte */
#define SCSIPI_BUSTYPE_TYPE_SHIFT 0
#define SCSIPI_BUSTYPE_TYPE(x) (((x) >> SCSIPI_BUSTYPE_TYPE_SHIFT) & 0xff)
#define SCSIPI_BUSTYPE_SCSI 0 /* parallel SCSI */
#define SCSIPI_BUSTYPE_ATAPI 1
/* #define SCSIPI_BUSTYPE_ATA 2 */
/* subtype is stored in the second byte */
#define SCSIPI_BUSTYPE_SUBTYPE_SHIFT 8
#define SCSIPI_BUSTYPE_SUBTYPE(x) (((x) >> SCSIPI_BUSTYPE_SUBTYPE_SHIFT) & 0xff)
#define SCSIPI_BUSTYPE_BUSTYPE(t, s) \
((t) << SCSIPI_BUSTYPE_TYPE_SHIFT | (s) << SCSIPI_BUSTYPE_SUBTYPE_SHIFT)
/* subtypes are defined in each bus type headers */
/*
* scsipi_channel:
*
* This structure describes a single channel of a SCSIPI adapter.
* An adapter may have one or more channels. See the comment above
* regarding the resource counter.
* Note: chan_bustype has to be first member, as its bustype_type member
* is shared with the aa_bustype member of struct ata_atapi_attach.
*/
#define SCSIPI_CHAN_PERIPH_BUCKETS 16
#define SCSIPI_CHAN_PERIPH_HASHMASK (SCSIPI_CHAN_PERIPH_BUCKETS - 1)
#ifdef _KERNEL
struct scsipi_channel {
const struct scsipi_bustype *chan_bustype; /* channel's bus type */
const char *chan_name; /* this channel's name */
struct scsipi_adapter *chan_adapter; /* pointer to our adapter */
/* Periphs for this channel. */
LIST_HEAD(, scsipi_periph) chan_periphtab[SCSIPI_CHAN_PERIPH_BUCKETS];
int chan_channel; /* channel number */
int chan_flags; /* channel flags */
int chan_openings; /* number of command openings */
int chan_max_periph; /* max openings per periph */
int chan_ntargets; /* number of targets */
int chan_nluns; /* number of luns */
int chan_id; /* adapter's ID for this channel */
int chan_defquirks; /* default device's quirks */
struct lwp *chan_dthread; /* discovery thread */
struct lwp *chan_thread; /* completion thread */
int chan_tflags; /* flags for the completion thread */
int chan_qfreeze; /* freeze count for queue */
/* Job queue for this channel. */
struct scsipi_xfer_queue chan_queue;
/* Completed (async) jobs. */
struct scsipi_xfer_queue chan_complete;
/* callback we may have to call from completion thread */
void (*chan_callback)(struct scsipi_channel *, void *);
void *chan_callback_arg;
/* callback we may have to call after forking the kthread */
void (*chan_init_cb)(struct scsipi_channel *, void *);
void *chan_init_cb_arg;
kcondvar_t chan_cv_comp;
kcondvar_t chan_cv_thr;
kcondvar_t chan_cv_xs;
#define chan_cv_complete(ch) (&(ch)->chan_cv_comp)
#define chan_cv_thread(ch) (&(ch)->chan_cv_thr)
};
#define chan_running(ch) ((ch)->chan_adapter->adapt_running)
#define chan_mtx(ch) (&(ch)->chan_adapter->adapt_mtx)
#endif
/* chan_flags */
#define SCSIPI_CHAN_OPENINGS 0x01 /* use chan_openings */
#define SCSIPI_CHAN_CANGROW 0x02 /* channel can grow resources */
#define SCSIPI_CHAN_NOSETTLE 0x04 /* don't wait for devices to settle */
#define SCSIPI_CHAN_TACTIVE 0x08 /* completion thread is active */
/* chan thread flags (chan_tflags) */
#define SCSIPI_CHANT_SHUTDOWN 0x01 /* channel is shutting down */
#define SCSIPI_CHANT_CALLBACK 0x02 /* has to call chan_callback() */
#define SCSIPI_CHANT_KICK 0x04 /* need to run queues */
#define SCSIPI_CHANT_GROWRES 0x08 /* call ADAPTER_REQ_GROW_RESOURCES */
#define SCSIPI_CHAN_MAX_PERIPH(chan) \
(((chan)->chan_flags & SCSIPI_CHAN_OPENINGS) ? \
(chan)->chan_max_periph : (chan)->chan_adapter->adapt_max_periph)
#define scsipi_printaddr(periph) \
(*(periph)->periph_channel->chan_bustype->bustype_printaddr)((periph))
#define scsipi_periph_bustype(periph) \
(periph)->periph_channel->chan_bustype->bustype_type
/*
* Number of tag words in a periph structure:
*
* n_tag_words = ((256 / NBBY) / sizeof(u_int32_t))
*/
#define PERIPH_NTAGWORDS ((256 / 8) / sizeof(u_int32_t))
#ifdef _KERNEL
/*
* scsipi_opcodes:
* This optionally allocated structure documents
* valid opcodes and timeout values for the respective
* opcodes overriding the requested timeouts.
* It is created when SCSI_MAINTENANCE_IN/
* RSOC_REPORT_SUPPORTED_OPCODES can provide information
* at attach time.
*/
struct scsipi_opcodes
{
struct scsipi_opinfo
{
long ti_timeout; /* timeout in seconds (> 0 => VALID) */
unsigned long ti_flags;
#define SCSIPI_TI_VALID 0x0001 /* valid op code */
#define SCSIPI_TI_LOGGED 0x8000 /* override logged during debug */
} opcode_info[0x100];
};
/*
* scsipi_periph:
*
* This structure describes the path between a peripheral device
* and an adapter. It contains a pointer to the adapter channel
* which in turn contains a pointer to the adapter.
*
* XXX Given the way NetBSD's autoconfiguration works, this is ...
* XXX nasty.
*
* Well, it's a lot nicer than it used to be, but there could
* still be an improvement.
*/
struct scsipi_periph {
device_t periph_dev; /* pointer to peripheral's device */
struct scsipi_channel *periph_channel; /* channel we're connected to */
/* link in channel's table of periphs */
LIST_ENTRY(scsipi_periph) periph_hash;
const struct scsipi_periphsw *periph_switch; /* peripheral's entry
points */
int periph_openings; /* max # of outstanding commands */
int periph_active; /* current # of outstanding commands */
int periph_sent; /* current # of commands sent to adapt*/
int periph_mode; /* operation modes, CAP bits */
int periph_period; /* sync period (factor) */
int periph_offset; /* sync offset */
/*
* Information gleaned from the inquiry data.
*/
u_int8_t periph_type; /* basic device type */
int periph_cap; /* capabilities */
int periph_quirks; /* device's quirks */
int periph_flags; /* misc. flags */
int periph_dbflags; /* debugging flags */
int periph_target; /* target ID (drive # on ATAPI) */
int periph_lun; /* LUN (not used on ATAPI) */
int periph_version; /* ANSI SCSI version */
int periph_qfreeze; /* queue freeze count */
/* available opcodes and timeout information */
struct scsipi_opcodes *periph_opcs;
/* Bitmap of free command tags. */
u_int32_t periph_freetags[PERIPH_NTAGWORDS];
/* Pending scsipi_xfers on this peripheral. */
struct scsipi_xfer_queue periph_xferq;
callout_t periph_callout;
/* xfer which has a pending CHECK_CONDITION */
struct scsipi_xfer *periph_xscheck;
kcondvar_t periph_cv;
#define periph_cv_periph(p) (&(p)->periph_cv)
#define periph_cv_active(p) (&(p)->periph_cv)
};
#endif
/*
* Macro to return the current xfer mode of a periph.
*/
#define PERIPH_XFER_MODE(periph) \
(((periph)->periph_flags & PERIPH_MODE_VALID) ? \
(periph)->periph_mode : 0)
/* periph_cap */
#define PERIPH_CAP_ANEC 0x0001 /* async event notification */
#define PERIPH_CAP_TERMIOP 0x0002 /* terminate i/o proc. messages */
#define PERIPH_CAP_RELADR 0x0004 /* relative addressing */
#define PERIPH_CAP_WIDE32 0x0008 /* wide-32 transfers */
#define PERIPH_CAP_WIDE16 0x0010 /* wide-16 transfers */
/* XXX 0x0020 reserved for ATAPI_CFG_DRQ_MASK */
/* XXX 0x0040 reserved for ATAPI_CFG_DRQ_MASK */
#define PERIPH_CAP_SYNC 0x0080 /* synchronous transfers */
#define PERIPH_CAP_LINKCMDS 0x0100 /* linked commands */
#define PERIPH_CAP_TQING 0x0200 /* tagged queueing */
#define PERIPH_CAP_SFTRESET 0x0400 /* soft RESET condition response */
#define PERIPH_CAP_CMD16 0x0800 /* 16 byte commands (ATAPI) */
#define PERIPH_CAP_DT 0x1000 /* supports DT clock */
#define PERIPH_CAP_QAS 0x2000 /* supports quick arbit. and select. */
#define PERIPH_CAP_IUS 0x4000 /* supports information unit xfers */
/* periph_flags */
#define PERIPH_REMOVABLE 0x0001 /* media is removable */
#define PERIPH_MEDIA_LOADED 0x0002 /* media is loaded */
#define PERIPH_WAITING 0x0004 /* process waiting for opening */
#define PERIPH_OPEN 0x0008 /* device is open */
#define PERIPH_WAITDRAIN 0x0010 /* waiting for pending xfers to drain */
#define PERIPH_GROW_OPENINGS 0x0020 /* allow openings to grow */
#define PERIPH_MODE_VALID 0x0040 /* periph_mode is valid */
#define PERIPH_RECOVERING 0x0080 /* periph is recovering */
#define PERIPH_RECOVERY_ACTIVE 0x0100 /* a recovery command is active */
#define PERIPH_KEEP_LABEL 0x0200 /* retain label after 'full' close */
#define PERIPH_SENSE 0x0400 /* periph has sense pending */
#define PERIPH_UNTAG 0x0800 /* untagged command running */
/* periph_quirks */
#define PQUIRK_AUTOSAVE 0x00000001 /* do implicit SAVE POINTERS */
#define PQUIRK_NOSYNC 0x00000002 /* does not grok SDTR */
#define PQUIRK_NOWIDE 0x00000004 /* does not grok WDTR */
#define PQUIRK_NOTAG 0x00000008 /* does not grok tagged cmds */
#define PQUIRK_NOLUNS 0x00000010 /* DTWT with LUNs */
#define PQUIRK_FORCELUNS 0x00000020 /* prehistoric device groks
LUNs */
#define PQUIRK_NOMODESENSE 0x00000040 /* device doesn't do MODE SENSE
properly */
#define PQUIRK_NOSYNCCACHE 0x00000100 /* do not issue SYNC CACHE */
#define PQUIRK_LITTLETOC 0x00000400 /* audio TOC is little-endian */
#define PQUIRK_NOCAPACITY 0x00000800 /* no READ CD CAPACITY */
#define PQUIRK_NOTUR 0x00001000 /* no TEST UNIT READY */
#define PQUIRK_NODOORLOCK 0x00002000 /* can't lock door */
#define PQUIRK_NOSENSE 0x00004000 /* can't REQUEST SENSE */
#define PQUIRK_ONLYBIG 0x00008000 /* only use SCSI_{R,W}_BIG */
#define PQUIRK_NOBIGMODESENSE 0x00040000 /* has no big mode-sense op */
#define PQUIRK_CAP_SYNC 0x00080000 /* SCSI device with ST sync op*/
#define PQUIRK_CAP_WIDE16 0x00100000 /* SCSI device with ST wide op*/
#define PQUIRK_CAP_NODT 0x00200000 /* signals DT, but can't. */
#define PQUIRK_START 0x00400000 /* needs start before tur */
#define PQUIRK_NOFUA 0x00800000 /* does not grok FUA */
#define PQUIRK_NOREPSUPPOPC 0x01000000 /* does not grok
REPORT SUPPORTED OPCODES
to fetch device timeouts */
/*
* Error values an adapter driver may return
*/
typedef enum {
XS_NOERROR, /* there is no error, (sense is invalid) */
XS_SENSE, /* Check the returned sense for the error */
XS_SHORTSENSE, /* Check the ATAPI sense for the error */
XS_DRIVER_STUFFUP, /* Driver failed to perform operation */
XS_RESOURCE_SHORTAGE, /* adapter resource shortage */
XS_SELTIMEOUT, /* The device timed out.. turned off? */
XS_TIMEOUT, /* The Timeout reported was caught by SW */
XS_BUSY, /* The device busy, try again later? */
XS_RESET, /* bus was reset; possible retry command */
XS_REQUEUE /* requeue this command */
} scsipi_xfer_result_t;
#ifdef _KERNEL
/*
* Each scsipi transaction is fully described by one of these structures
* It includes information about the source of the command and also the
* device and adapter for which the command is destined.
*
* Before the HBA is given this transaction, channel_q is the linkage on
* the related channel's chan_queue.
*
* When the this transaction is taken off the channel's chan_queue and
* the HBA's request entry point is called with this transaction, the
* HBA can use the channel_q tag for whatever it likes until it calls
* scsipi_done for this transaction, at which time it has to stop
* using channel_q.
*
* After scsipi_done is called with this transaction and if there was an
* error on it, channel_q then becomes the linkage on the related channel's
* chan_complete cqueue.
*
* The device_q member is maintained by the scsipi middle layer. When
* a device issues a command, the xfer is placed on that device's
* pending commands queue. When an xfer is done and freed, it is taken
* off the device's queue. This allows for a device to wait for all of
* its pending commands to complete.
*/
struct scsipi_xfer {
TAILQ_ENTRY(scsipi_xfer) channel_q; /* entry on channel queue */
TAILQ_ENTRY(scsipi_xfer) device_q; /* device's pending xfers */
callout_t xs_callout; /* callout for adapter use */
int xs_control; /* control flags */
volatile int xs_status; /* status flags */
struct scsipi_periph *xs_periph;/* peripheral doing the xfer */
int xs_retries; /* the number of times to retry */
int xs_requeuecnt; /* number of requeues */
int timeout; /* in milliseconds */
struct scsipi_generic *cmd; /* The scsipi command to execute */
int cmdlen; /* how long it is */
u_char *data; /* DMA address OR a uio address */
int datalen; /* data len (blank if uio) */
int resid; /* how much buffer was not touched */
scsipi_xfer_result_t error; /* an error value */
struct buf *bp; /* If we need to associate with */
/* a buf */
union {
struct scsi_sense_data scsi_sense; /* 32 bytes */
u_int32_t atapi_sense;
} sense;
struct scsipi_xfer *xs_sensefor;/* we are requesting sense for this */
/* xfer */
u_int8_t status; /* SCSI status */
/*
* Info for tagged command queueing. This may or may not
* be used by a given adapter driver. These are the same
* as the bytes in the tag message.
*/
u_int8_t xs_tag_type; /* tag type */
u_int8_t xs_tag_id; /* tag ID */
struct scsipi_generic cmdstore
__aligned(4); /* stash the command in here */
#define xs_cv(xs) (&(xs)->xs_periph->periph_channel->chan_cv_xs)
};
#endif
/*
* scsipi_xfer control flags
*
* To do:
*
* - figure out what to do with XS_CTL_ESCAPE
*
* - replace XS_CTL_URGENT with an `xs_priority' field?
*/
#define XS_CTL_NOSLEEP 0x00000001 /* don't sleep */
#define XS_CTL_POLL 0x00000002 /* poll for completion */
#define XS_CTL_DISCOVERY 0x00000004 /* doing device discovery */
#define XS_CTL_ASYNC 0x00000008 /* command completes
asynchronously */
#define XS_CTL_USERCMD 0x00000010 /* user issued command */
#define XS_CTL_SILENT 0x00000020 /* don't print sense info */
#define XS_CTL_IGNORE_NOT_READY 0x00000040 /* ignore NOT READY */
#define XS_CTL_IGNORE_MEDIA_CHANGE \
0x00000080 /* ignore media change */
#define XS_CTL_IGNORE_ILLEGAL_REQUEST \
0x00000100 /* ignore ILLEGAL REQUEST */
#define XS_CTL_SILENT_NODEV 0x00000200 /* don't print sense info
if sense info is nodev */
#define XS_CTL_RESET 0x00000400 /* reset the device */
#define XS_CTL_DATA_UIO 0x00000800 /* xs_data points to uio */
#define XS_CTL_DATA_IN 0x00001000 /* data coming into memory */
#define XS_CTL_DATA_OUT 0x00002000 /* data going out of memory */
#define XS_CTL_TARGET 0x00004000 /* target mode operation */
#define XS_CTL_ESCAPE 0x00008000 /* escape operation */
#define XS_CTL_URGENT 0x00010000 /* urgent (recovery)
operation */
#define XS_CTL_SIMPLE_TAG 0x00020000 /* use a Simple Tag */
#define XS_CTL_ORDERED_TAG 0x00040000 /* use an Ordered Tag */
#define XS_CTL_HEAD_TAG 0x00080000 /* use a Head of Queue Tag */
#define XS_CTL_THAW_PERIPH 0x00100000 /* thaw periph once enqueued */
#define XS_CTL_FREEZE_PERIPH 0x00200000 /* freeze periph when done */
#define XS_CTL_REQSENSE 0x00800000 /* xfer is a request sense */
#define XS_CTL_TAGMASK (XS_CTL_SIMPLE_TAG|XS_CTL_ORDERED_TAG|XS_CTL_HEAD_TAG)
#define XS_CTL_TAGTYPE(xs) ((xs)->xs_control & XS_CTL_TAGMASK)
/*
* scsipi_xfer status flags
*/
#define XS_STS_DONE 0x00000001 /* scsipi_xfer is done */
#define XS_STS_PRIVATE 0xf0000000 /* reserved for HBA's use */
/*
* This describes matching information for scsipi_inqmatch(). The more things
* match, the higher the configuration priority.
*/
struct scsipi_inquiry_pattern {
u_int8_t type;
boolean removable;
const char *vendor;
const char *product;
const char *revision;
};
/*
* This is used to pass information from the high-level configuration code
* to the device-specific drivers.
*/
struct scsipibus_attach_args {
struct scsipi_periph *sa_periph;
struct scsipi_inquiry_pattern sa_inqbuf;
struct scsipi_inquiry_data *sa_inqptr;
union { /* bus-type specific infos */
u_int8_t scsi_version; /* SCSI version */
} scsipi_info;
};
/*
* this describes a quirk entry
*/
struct scsi_quirk_inquiry_pattern {
struct scsipi_inquiry_pattern pattern;
int quirks;
};
/*
* Default number of retries, used for generic routines.
*/
#define SCSIPIRETRIES 4
#ifdef _KERNEL
void scsipi_init(void);
void scsipi_ioctl_init(void);
void scsipi_load_verbose(void);
int scsipi_command(struct scsipi_periph *, struct scsipi_generic *, int,
u_char *, int, int, int, struct buf *, int);
void scsipi_create_completion_thread(void *);
const void *scsipi_inqmatch(struct scsipi_inquiry_pattern *, const void *,
size_t, size_t, int *);
const char *scsipi_dtype(int);
int scsipi_execute_xs(struct scsipi_xfer *);
int scsipi_test_unit_ready(struct scsipi_periph *, int);
int scsipi_prevent(struct scsipi_periph *, int, int);
int scsipi_inquire(struct scsipi_periph *,
struct scsipi_inquiry_data *, int);
int scsipi_mode_select(struct scsipi_periph *, int,
struct scsi_mode_parameter_header_6 *, int, int, int, int);
int scsipi_mode_select_big(struct scsipi_periph *, int,
struct scsi_mode_parameter_header_10 *, int, int, int, int);
int scsipi_mode_sense(struct scsipi_periph *, int, int,
struct scsi_mode_parameter_header_6 *, int, int, int, int);
int scsipi_mode_sense_big(struct scsipi_periph *, int, int,
struct scsi_mode_parameter_header_10 *, int, int, int, int);
int scsipi_start(struct scsipi_periph *, int, int);
void scsipi_done(struct scsipi_xfer *);
void scsipi_user_done(struct scsipi_xfer *);
int scsipi_interpret_sense(struct scsipi_xfer *);
void scsipi_wait_drain(struct scsipi_periph *);
void scsipi_kill_pending(struct scsipi_periph *);
void scsipi_get_opcodeinfo(struct scsipi_periph *periph);
void scsipi_free_opcodeinfo(struct scsipi_periph *periph);
struct scsipi_periph *scsipi_alloc_periph(int);
void scsipi_free_periph(struct scsipi_periph *);
/* Function pointers for scsiverbose module */
extern int (*scsipi_print_sense)(struct scsipi_xfer *, int);
extern void (*scsipi_print_sense_data)(struct scsi_sense_data *, int);
int scsipi_print_sense_stub(struct scsipi_xfer *, int);
void scsipi_print_sense_data_stub(struct scsi_sense_data *, int);
extern int scsi_verbose_loaded;
void scsipi_print_cdb(struct scsipi_generic *cmd);
int scsipi_thread_call_callback(struct scsipi_channel *,
void (*callback)(struct scsipi_channel *, void *),
void *);
void scsipi_async_event(struct scsipi_channel *,
scsipi_async_event_t, void *);
int scsipi_do_ioctl(struct scsipi_periph *, dev_t, u_long, void *,
int, struct lwp *);
void scsipi_set_xfer_mode(struct scsipi_channel *, int, int);
int scsipi_channel_init(struct scsipi_channel *);
void scsipi_channel_shutdown(struct scsipi_channel *);
void scsipi_insert_periph(struct scsipi_channel *,
struct scsipi_periph *);
void scsipi_remove_periph(struct scsipi_channel *,
struct scsipi_periph *);
struct scsipi_periph *scsipi_lookup_periph(struct scsipi_channel *,
int, int);
struct scsipi_periph *scsipi_lookup_periph_locked(struct scsipi_channel *,
int, int);
int scsipi_target_detach(struct scsipi_channel *, int, int, int);
int scsipi_adapter_addref(struct scsipi_adapter *);
void scsipi_adapter_delref(struct scsipi_adapter *);
void scsipi_channel_freeze(struct scsipi_channel *, int);
void scsipi_channel_thaw(struct scsipi_channel *, int);
void scsipi_channel_timed_thaw(void *);
void scsipi_periph_freeze(struct scsipi_periph *, int);
void scsipi_periph_thaw(struct scsipi_periph *, int);
void scsipi_periph_timed_thaw(void *);
void scsipi_periph_freeze_locked(struct scsipi_periph *, int);
void scsipi_periph_thaw_locked(struct scsipi_periph *, int);
int scsipi_sync_period_to_factor(int);
int scsipi_sync_factor_to_period(int);
int scsipi_sync_factor_to_freq(int);
void show_scsipi_xs(struct scsipi_xfer *);
void show_scsipi_cmd(struct scsipi_xfer *);
void show_mem(u_char *, int);
#endif /* _KERNEL */
static __inline void
_lto2b(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = (val >> 8) & 0xff;
bytes[1] = val & 0xff;
}
static __inline void
_lto3b(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = (val >> 16) & 0xff;
bytes[1] = (val >> 8) & 0xff;
bytes[2] = val & 0xff;
}
static __inline void
_lto4b(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = (val >> 24) & 0xff;
bytes[1] = (val >> 16) & 0xff;
bytes[2] = (val >> 8) & 0xff;
bytes[3] = val & 0xff;
}
static __inline void
_lto8b(u_int64_t val, u_int8_t *bytes)
{
bytes[0] = (val >> 56) & 0xff;
bytes[1] = (val >> 48) & 0xff;
bytes[2] = (val >> 40) & 0xff;
bytes[3] = (val >> 32) & 0xff;
bytes[4] = (val >> 24) & 0xff;
bytes[5] = (val >> 16) & 0xff;
bytes[6] = (val >> 8) & 0xff;
bytes[7] = val & 0xff;
}
static __inline u_int32_t
_2btol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = (bytes[0] << 8) |
bytes[1];
return (rv);
}
static __inline u_int32_t
_3btol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = (bytes[0] << 16) |
(bytes[1] << 8) |
bytes[2];
return (rv);
}
static __inline u_int32_t
_4btol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = ((u_int32_t)bytes[0] << 24) |
((u_int32_t)bytes[1] << 16) |
((u_int32_t)bytes[2] << 8) |
(u_int32_t)bytes[3];
return (rv);
}
static __inline u_int64_t
_5btol(const u_int8_t *bytes)
{
u_int64_t rv;
rv = ((u_int64_t)bytes[0] << 32) |
((u_int64_t)bytes[1] << 24) |
((u_int64_t)bytes[2] << 16) |
((u_int64_t)bytes[3] << 8) |
(u_int64_t)bytes[4];
return (rv);
}
static __inline u_int64_t
_8btol(const u_int8_t *bytes)
{
u_int64_t rv;
rv = ((u_int64_t)bytes[0] << 56) |
((u_int64_t)bytes[1] << 48) |
((u_int64_t)bytes[2] << 40) |
((u_int64_t)bytes[3] << 32) |
((u_int64_t)bytes[4] << 24) |
((u_int64_t)bytes[5] << 16) |
((u_int64_t)bytes[6] << 8) |
(u_int64_t)bytes[7];
return (rv);
}
static __inline void
_lto2l(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = val & 0xff;
bytes[1] = (val >> 8) & 0xff;
}
static __inline void
_lto3l(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = val & 0xff;
bytes[1] = (val >> 8) & 0xff;
bytes[2] = (val >> 16) & 0xff;
}
static __inline void
_lto4l(u_int32_t val, u_int8_t *bytes)
{
bytes[0] = val & 0xff;
bytes[1] = (val >> 8) & 0xff;
bytes[2] = (val >> 16) & 0xff;
bytes[3] = (val >> 24) & 0xff;
}
static __inline u_int32_t
_2ltol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = bytes[0] |
(bytes[1] << 8);
return (rv);
}
static __inline u_int32_t
_3ltol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = bytes[0] |
(bytes[1] << 8) |
(bytes[2] << 16);
return (rv);
}
static __inline u_int32_t
_4ltol(const u_int8_t *bytes)
{
u_int32_t rv;
rv = (u_int32_t)bytes[0] |
((u_int32_t)bytes[1] << 8) |
((u_int32_t)bytes[2] << 16) |
((u_int32_t)bytes[3] << 24);
return (rv);
}
#endif /* _DEV_SCSIPI_SCSIPICONF_H_ */