diff options
Diffstat (limited to 'sys/dev/atapi/atapilink.h')
-rw-r--r-- | sys/dev/atapi/atapilink.h | 352 |
1 files changed, 352 insertions, 0 deletions
diff --git a/sys/dev/atapi/atapilink.h b/sys/dev/atapi/atapilink.h new file mode 100644 index 00000000000..7d01102a2a7 --- /dev/null +++ b/sys/dev/atapi/atapilink.h @@ -0,0 +1,352 @@ +/* $NetBSD: $ */ + +/* + * Copyright (c) 1996 Manuel Bouyer. 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 Manuel Bouyer. + * 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. + */ + +#undef ATAPI_DEBUG +#undef ATAPI_DEBUG_PROBE + +struct bus_link { + u_int8_t type; +#define DRIVE 0 +#define BUS 1 + caddr_t wdc_softc; + caddr_t atapibus_softc; + struct wdc_link *ctlr_link; + u_int8_t ctrl; +}; + +struct atapi_identify { + struct config_s { + u_int8_t cmd_drq_rem; +#define ATAPI_PACKET_SIZE_MASK 0x02 +#define ATAPI_PACKET_SIZE_12 0x00 +#define ATAPI_PACKET_SIZE_16 0x01 + +#define ATAPI_DRQ_MASK 0x60 +#define ATAPI_MICROPROCESSOR_DRQ 0x00 +#define ATAPI_INTERRUPT_DRQ 0x20 +#define ATAPI_ACCELERATED_DRQ 0x40 + +#define ATAPI_REMOVABLE 0x80 + + u_int8_t device_type; +#define ATAPI_DEVICE_TYPE_MASK 0x1f +#define ATAPI_DEVICE_TYPE_DAD 0x00 /* direct access device */ + /* 0x1-0x4 reserved */ +#define ATAPI_DEVICE_TYPE_CD 0x05 /* CD-ROM */ + /* 0x6 reserved */ +#define ATAPI_DEVICE_TYPE_OMD 0x07 /* optical memory device */ + /* 0x8-0x1e reserved */ +#define ATAPI_DEVICE_TYPE_UNKNOWN 0x1f + +#define ATAPI_GC_PROTOCOL_MASK 0xc0 /* mask of protocol bits */ + /* 0x00 and 0x01 are ATA */ +#define ATAPI_GC_PROTO_TYPE_ATAPI 0x80 +#define ATAPI_GC_PROTO_TYPE_RESERVED 0xc0 + } config; /* general configuration */ + + u_int8_t cylinders[2]; + u_int8_t reserved1[2]; + u_int8_t heads[2]; + u_int8_t unf_bytes_per_track[2]; + u_int8_t unf_bytes_per_sector[2]; + u_int8_t sectors_per_track[2]; + u_int8_t reserved2[6]; + char serial_number[20]; + u_int8_t buffer_type[2]; + u_int8_t buffer_size[2]; + u_int8_t ECC_bytes_available[2]; + char firmware_revision[8]; + char model[40]; + u_int8_t sector_count[2]; + u_int8_t double_word[2]; /* == 0 for CD-ROMs */ + + struct capabilities_s { + u_int8_t vendor; + u_int8_t capflags; +#define ATAPI_CAP_DMA 0x01 /* DMA supported */ +#define ATAPI_CAP_LBA 0x02 /* LBA supported */ +#define ATAPI_IORDY_DISABLE 0x04 /* IORDY can be disabled */ +#define ATAPI_IORDY 0x08 /* IORDY supported */ + } capabilities; + + u_int8_t reserved3[2]; + u_int8_t PIO_cycle_timing[2]; + u_int8_t DMA_cycle_timing[2]; + u_int8_t validity[2]; /* of words 54-58, 64-70 in this table */ + +#define ATAPI_VALID_FIRST 0x0 /* == 1 => words 54-58 are valid */ +#define ATAPI_VALID_SECOND 0x1 /* == 1 => words 64-70 are valid */ + + u_int8_t current_chs[6]; /* cylinder/head/sector */ + u_int8_t current_capacity[4]; + u_int8_t reserved4[2]; + u_int8_t user_addressable_sectors[4]; + u_int8_t singleword_DMA_mode[2]; + +#define ATAPI_SW_DMA_MODE_AVAIL 0x00ff /* Mode 0 is supported */ +#define ATAPI_SW_DMA_MODE_ACTIVE 0xff00 /* which mode is active */ + + u_int8_t multiword_DMA_mode[2]; + +#define ATAPI_MW_DMA_MODE_AVAIL 0x00ff /* Mode 0 is supported */ +#define ATAPI_MW_DMA_MODE_ACTIVE 0xff00 /* which mode is active */ + + u_int8_t enhanced_PIO_mode[2]; + +#define ATAPI_ENHANCED_PIO_AVAIL 0x0001 /* PIO Mode 3 is supported */ + + u_int8_t blind_PIO_minimum_cycles[2]; + u_int8_t mw_dma_tct[2]; /* multi-word DMA transfer cycle time */ + u_int8_t min_PIO_tct_no_flow_control[2]; + u_int8_t min_PIO_tct_with_flow_control[2]; + u_int8_t reserved5[4]; + u_int8_t reserved6[114]; + u_int8_t vendor[64]; /* vendor unique */ + u_int8_t reserved7[192]; +}; + +struct at_dev_link { + void *device_softc; + u_int8_t drive; + u_int8_t openings; + struct atapi_identify id; + struct bus_link *bus; + u_int16_t flags; +#define ADEV_REMOVABLE 0x001 /* media is removable */ +#define ADEV_MEDIA_LOADED 0x002 /* device figures are still valid */ +#define ADEV_WAITING 0x004 /* a process is waiting for this */ +#define ADEV_OPEN 0x008 /* at least 1 open session */ +#define ACAP_DRQ_MPROC 0x000 /* microprocessor DRQ */ +#define ACAP_DRQ_INTR 0x100 /* interrupt DRQ */ +#define ACAP_DRQ_ACCEL 0x200 /* accelerated DRQ */ +#define ACAP_LEN 0x400 /* 16 bit commands */ + void (*start)(); /* device start routine */ + int (*done)(); /* device done routine */ +}; + +struct atapi_command_packet { + void *ad_link; + void *command; + char cmd_store[16]; + int command_size; + struct buf *bp; + void *databuf; + int data_size; + int flags; /* handle B_READ/B_WRITE mask 0x00f00000 */ + /* controller flags maks 0x0000000f */ + /* ATAPI flags mask 0x000000f0 */ + /* Capabilities flags 0x00000f00 */ + u_int8_t drive; + u_int16_t status; +#define STATUS_MASK 0xff +#define NO_ERROR 0x00 +#define ERROR 0x01 +#define MEDIA_CHANGE 0x02 +#define END_OF_MEDIA 0x03 +#define NOT_READY 0x10 +#define UNIT_ATTENTION 0x20 +#define RETRY 0x40 +#define ITSDONE 0x100 + u_int8_t error; + u_int8_t retries; +#define ATAPI_NRETRIES 5 + LIST_ENTRY(atapi_command_packet) free_list; +}; + +int wdc_atapi_get_params __P((struct bus_link *, u_int8_t, + struct atapi_identify *)); +void wdc_atapi_send_command_packet __P((struct bus_link *, + struct atapi_command_packet*)); + +#define A_POLLED 0x10 +#define A_NOSLEEP 0x20 +#define A_SILENT 0x40 + +void atapi_done __P((struct atapi_command_packet *)); +struct atapi_command_packet *atapi_get_pkt __P((struct at_dev_link *, int)); +void atapi_free_pkt __P((struct atapi_command_packet *)); + +/* + * Functions used for reading and writing 2, 3, and 4 byte values + * in ATAPI commands. + */ + +static __inline void _lto2b __P((u_int32_t val, u_int8_t *bytes)); +static __inline void _lto3b __P((u_int32_t val, u_int8_t *bytes)); +static __inline void _lto4b __P((u_int32_t val, u_int8_t *bytes)); +static __inline u_int32_t _2btol __P((u_int8_t *bytes)); +static __inline u_int32_t _3btol __P((u_int8_t *bytes)); +static __inline u_int32_t _4btol __P((u_int8_t *bytes)); + +static __inline void _lto2l __P((u_int32_t val, u_int8_t *bytes)); +static __inline void _lto3l __P((u_int32_t val, u_int8_t *bytes)); +static __inline void _lto4l __P((u_int32_t val, u_int8_t *bytes)); +static __inline u_int32_t _2ltol __P((u_int8_t *bytes)); +static __inline u_int32_t _3ltol __P((u_int8_t *bytes)); +static __inline u_int32_t _4ltol __P((u_int8_t *bytes)); + +static __inline void +_lto2b(val, bytes) + u_int32_t val; + u_int8_t *bytes; +{ + + bytes[0] = (val >> 8) & 0xff; + bytes[1] = val & 0xff; +} + +static __inline void +_lto3b(val, bytes) + 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(val, bytes) + 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 u_int32_t +_2btol(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = (bytes[0] << 8) | + bytes[1]; + return (rv); +} + +static __inline u_int32_t +_3btol(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = (bytes[0] << 16) | + (bytes[1] << 8) | + bytes[2]; + return (rv); +} + +static __inline u_int32_t +_4btol(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = (bytes[0] << 24) | + (bytes[1] << 16) | + (bytes[2] << 8) | + bytes[3]; + return (rv); +} + +static __inline void +_lto2l(val, bytes) + u_int32_t val; + u_int8_t *bytes; +{ + + bytes[0] = val & 0xff; + bytes[1] = (val >> 8) & 0xff; +} + +static __inline void +_lto3l(val, bytes) + 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(val, bytes) + 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(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = bytes[0] | + (bytes[1] << 8); + return (rv); +} + +static __inline u_int32_t +_3ltol(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = bytes[0] | + (bytes[1] << 8) | + (bytes[2] << 16); + return (rv); +} + +static __inline u_int32_t +_4ltol(bytes) + u_int8_t *bytes; +{ + register u_int32_t rv; + + rv = bytes[0] | + (bytes[1] << 8) | + (bytes[2] << 16) | + (bytes[3] << 24); + return (rv); +} |