summaryrefslogtreecommitdiff
path: root/sys/dev/atapi/atapiconf.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/atapi/atapiconf.c')
-rw-r--r--sys/dev/atapi/atapiconf.c676
1 files changed, 676 insertions, 0 deletions
diff --git a/sys/dev/atapi/atapiconf.c b/sys/dev/atapi/atapiconf.c
new file mode 100644
index 00000000000..85095a4c0a4
--- /dev/null
+++ b/sys/dev/atapi/atapiconf.c
@@ -0,0 +1,676 @@
+/* $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.
+ */
+
+#include <sys/types.h>
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/malloc.h>
+#include <sys/device.h>
+#include <sys/buf.h>
+#include <sys/proc.h>
+
+#include <dev/atapi/atapilink.h>
+#include <dev/atapi/atapi.h>
+
+#define SILENT_PRINTF(flags,string) if (!(flags & A_SILENT)) printf string
+
+#ifdef ATAPI_DEBUG_CMD
+#define ATAPI_DEBUG_CMD_PRINT(args) printf args
+#else
+#define ATAPI_DEBUG_CMD_PRINT(args)
+#endif
+
+#ifdef ATAPI_DEBUG_FCTN
+#define ATAPI_DEBUG_FCTN_PRINT(args) printf args
+#else
+#define ATAPI_DEBUG_FCTN_PRINT(args)
+#endif
+
+struct atapibus_softc {
+ struct device sc_dev;
+ struct bus_link *b_link;
+};
+
+LIST_HEAD(pkt_free_list, atapi_command_packet) pkt_free_list;
+
+static void bswap __P((char *, int));
+static void btrim __P((char *, int));
+
+int atapi_error __P((struct atapi_command_packet *));
+void atapi_sense __P((struct atapi_command_packet *, u_int8_t, u_int8_t));
+void at_print_addr __P((struct at_dev_link *, u_int8_t));
+
+int atapibusmatch __P((struct device *, void *, void *));
+void atapibusattach __P((struct device *, struct device *, void *));
+
+struct cfattach atapibus_ca = {
+ sizeof(struct atapibus_softc), atapibusmatch, atapibusattach
+};
+
+struct cfdriver atapibus_cd = {
+ NULL, "atapibus", DV_DULL
+};
+
+
+int
+atapibusmatch(parent, match, aux)
+ struct device *parent;
+ void *match, *aux;
+{
+ struct cfdata *cf = match;
+ struct bus_link *ab_link = aux;
+
+ if (ab_link == NULL)
+ return 0;
+ if (ab_link->type != BUS)
+ return 0;
+ return 1;
+}
+
+int
+atapiprint(aux, bus)
+ void *aux;
+ char *bus;
+{
+ struct at_dev_link *ad_link = aux;
+ struct atapi_identify *id = &ad_link->id;
+ char *dtype, *fixrem;
+
+ /*
+ * Figure out basic device type.
+ */
+ switch (id->config.device_type & ATAPI_DEVICE_TYPE_MASK) {
+ case ATAPI_DEVICE_TYPE_DAD:
+ dtype = "direct";
+ break;
+
+ case ATAPI_DEVICE_TYPE_CD:
+ dtype = "cdrom";
+ break;
+
+ case ATAPI_DEVICE_TYPE_OMD:
+ dtype = "optical";
+ break;
+
+ default:
+ dtype = "unknown";
+ break;
+ }
+
+ fixrem = (id->config.cmd_drq_rem & ATAPI_REMOVABLE) ?
+ "removable" : "fixed";
+
+ /*
+ * Shuffle string byte order.
+ * Mitsumi and NEC drives don't need this.
+ */
+ if (((id->model[0] == 'N' && id->model[1] == 'E') ||
+ (id->model[0] == 'F' && id->model[1] == 'X')) == 0)
+ bswap(id->model, sizeof(id->model));
+
+ /*
+ * XXX Poorly named... These appear to actually be in
+ * XXX network byte order, so bswap() is a no-op on
+ * XXX big-endian hosts. Clean me up, please.
+ */
+ bswap(id->serial_number, sizeof(id->serial_number));
+ bswap(id->firmware_revision, sizeof(id->firmware_revision));
+
+ /*
+ * Clean up the model name, serial and
+ * revision numbers.
+ */
+ btrim(id->model, sizeof(id->model));
+ btrim(id->serial_number, sizeof(id->serial_number));
+ btrim(id->firmware_revision, sizeof(id->firmware_revision));
+
+ if (bus != NULL)
+ printf("%s", bus);
+
+ printf(" drive %d: <%s, %s, %s> type %d/%s %s",
+ ad_link->drive, id->model, id->serial_number,
+ id->firmware_revision,
+ id->config.device_type & ATAPI_DEVICE_TYPE_MASK, dtype, fixrem);
+
+ return UNCONF;
+}
+
+
+void
+atapibusattach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ struct atapibus_softc *ab = (struct atapibus_softc *)self;
+ struct bus_link *ab_link_proto = aux;
+ struct atapi_identify ids;
+ struct atapi_identify *id = &ids;
+ struct at_dev_link *ad_link;
+ int drive;
+
+ printf("\n");
+
+ ab_link_proto->atapibus_softc = (caddr_t)ab;
+ ab->b_link = ab_link_proto;
+
+ for (drive = 0; drive < 2 ; drive++) {
+ if (wdc_atapi_get_params(ab_link_proto, drive, id)) {
+#ifdef ATAPI_DEBUG_PROBE
+ printf("%s drive %d: cmdsz 0x%x drqtype 0x%x\n",
+ self->dv_xname, drive,
+ id->config.cmd_drq_rem & ATAPI_PACKET_SIZE_MASK,
+ id->config.cmd_drq_rem & ATAPI_DRQ_MASK);
+#endif
+
+ /*
+ * Allocate a device link and try and attach
+ * a driver to this device. If we fail, free
+ * the link.
+ */
+ ad_link = malloc(sizeof(*ad_link), M_DEVBUF, M_NOWAIT);
+ if (ad_link == NULL) {
+ printf("%s: can't allocate link for drive %d\n",
+ self->dv_xname, drive);
+ continue;
+ }
+
+ /* Fill in link. */
+ ad_link->drive = drive;
+ if (id->config.cmd_drq_rem & ATAPI_PACKET_SIZE_16)
+ ad_link->flags |= ACAP_LEN;
+ ad_link->flags |=
+ (id->config.cmd_drq_rem & ATAPI_DRQ_MASK) << 3;
+ ad_link->bus = ab_link_proto;
+ bcopy(id, &ad_link->id, sizeof(*id));
+
+ /* Try to find a match. */
+ if (config_found(self, ad_link, atapiprint) == NULL)
+ free(ad_link, M_DEVBUF);
+ }
+ }
+}
+
+static void
+bswap (buf, len)
+ char *buf;
+ int len;
+{
+ u_int16_t *p = (u_int16_t *)(buf + len);
+
+ while (--p >= (u_int16_t *)buf)
+ *p = ntohs(*p);
+}
+
+static void
+btrim (buf, len)
+ char *buf;
+ int len;
+{
+ char *p;
+
+ /* Remove the trailing spaces. */
+ for (p = buf; p < buf + len; ++p)
+ if (*p == '\0')
+ *p = ' ';
+
+ for (p = buf + len - 1; p >= buf && *p == ' '; --p)
+ *p = '\0';
+}
+
+int
+atapi_exec_cmd(ad_link, cmd, cmd_size, databuf, datalen, rw, flags)
+ struct at_dev_link *ad_link;
+ void *cmd;
+ int cmd_size;
+ void *databuf;
+ int datalen;
+ long rw;
+ int flags;
+{
+ struct atapi_command_packet *pkt;
+ struct bus_link *b_link = ad_link->bus;
+ int status, s;
+
+ /* Allocate packet. */
+ pkt = atapi_get_pkt(ad_link, flags);
+ if (pkt == NULL)
+ return -1;
+
+ /* Fill it out. */
+ bcopy(cmd, &pkt->cmd_store, cmd_size);
+ pkt->command = &pkt->cmd_store;
+ pkt->command_size = (ad_link->flags & ACAP_LEN) ? 16 : 12;
+ pkt->databuf = databuf;
+ pkt->data_size = datalen;
+ pkt->flags = rw | (flags & 0xff) | (ad_link->flags & 0x0300);
+ pkt->drive = ad_link->drive;
+
+ /* Send it to drive. */
+ wdc_atapi_send_command_packet(b_link, pkt);
+ if ((flags & (A_POLLED | A_NOSLEEP)) == 0) {
+ ATAPI_DEBUG_CMD_PRINT(("atapi_exec_cmd: sleeping\n"));
+
+ s = splbio();
+ while ((pkt->status & ITSDONE) == 0)
+ tsleep(pkt, PRIBIO + 1,"atapicmd", 0);
+ splx(s);
+
+ ATAPI_DEBUG_CMD_PRINT(("atapi_exec_cmd: done sleeping\n"));
+
+ status = pkt->status & STATUS_MASK;
+ atapi_free_pkt(pkt);
+ } else {
+ if ((flags & A_POLLED) != 0) {
+ if ((pkt->status & ERROR) && (pkt->error)) {
+ atapi_error(pkt);
+ SILENT_PRINTF(flags,("\n"));
+ }
+ }
+ status = pkt->status & STATUS_MASK;
+ if ((flags & A_POLLED) != 0)
+ atapi_free_pkt(pkt);
+ }
+
+ return status;
+}
+
+int
+atapi_exec_io(ad_link, cmd, cmd_size, bp, flags)
+ struct at_dev_link *ad_link;
+ void *cmd;
+ int cmd_size;
+ struct buf *bp;
+ int flags;
+{
+ struct atapi_command_packet *pkt;
+ struct bus_link *b_link = ad_link->bus;
+
+ /* Allocate a packet. */
+ pkt = atapi_get_pkt(ad_link, flags);
+ if (pkt == NULL) {
+ printf("atapi_exec_io: no pkt\n");
+ return ERROR;
+ }
+
+ /* Fill it in. */
+ bcopy(cmd, &pkt->cmd_store, cmd_size);
+ pkt->command = &pkt->cmd_store;
+ pkt->command_size = (ad_link->flags & ACAP_LEN) ? 16 : 12;
+ pkt->bp = bp;
+ pkt->databuf = bp->b_data;
+ pkt->data_size = bp->b_bcount;
+ pkt->flags = bp->b_flags & (B_READ|B_WRITE) | (flags & 0xff) |
+ (ad_link->flags & 0x0300);
+ pkt->drive = ad_link->drive;
+
+ wdc_atapi_send_command_packet(b_link, pkt);
+ return (pkt->status & STATUS_MASK);
+}
+
+void
+atapi_done(acp)
+ struct atapi_command_packet *acp;
+{
+ struct at_dev_link *ad_link = acp->ad_link;
+ struct buf *bp = acp->bp;
+ int error = 0;
+
+ ATAPI_DEBUG_CMD_PRINT(("atapi_done\n"));
+
+ if ((acp->status & ERROR) && (acp->error)) {
+ atapi_error(acp);
+ if (acp->status & RETRY) {
+ if (acp->retries <ATAPI_NRETRIES) {
+ acp->retries++;
+ acp->status = 0;
+ acp->error = 0;
+ SILENT_PRINTF(acp->flags & 0xff,
+ (", retry #%d\n", acp->retries));
+ wdc_atapi_send_command_packet(ad_link->bus,
+ acp);
+ return;
+ } else
+ acp->status = ERROR;
+ }
+ SILENT_PRINTF(acp->flags & 0xff,("\n"));
+ }
+ acp->status |= ITSDONE;
+
+ if (ad_link->done) {
+ ATAPI_DEBUG_CMD_PRINT(("calling private done\n"));
+ error = (*ad_link->done)(acp);
+ if (error == EJUSTRETURN)
+ return;
+ }
+ if (acp->bp == NULL) {
+ ATAPI_DEBUG_CMD_PRINT(("atapidone: wakeup acp\n"));
+ wakeup(acp);
+ return;
+ }
+
+ ATAPI_DEBUG_CMD_PRINT(("atapi_done: status %d\n", acp->status));
+
+ switch (acp->status & 0x0f) {
+ case MEDIA_CHANGE:
+ if (ad_link->flags & ADEV_REMOVABLE)
+ ad_link->flags &= ~ADEV_MEDIA_LOADED;
+
+ error = EIO;
+ break;
+
+ case NO_ERROR:
+ error = 0;
+ break;
+
+ case ERROR:
+ case END_OF_MEDIA:
+ default:
+ error = EIO;
+ break;
+ }
+
+ switch (acp->status & 0xf0) {
+ case NOT_READY:
+ case UNIT_ATTENTION:
+ if (ad_link->flags & ADEV_REMOVABLE)
+ ad_link->flags &= ~ADEV_MEDIA_LOADED;
+
+ error = EIO;
+ break;
+ }
+
+ if (error) {
+ bp->b_error = error;
+ bp->b_flags |= B_ERROR;
+ bp->b_resid = bp->b_bcount;
+ } else {
+ bp->b_error = 0;
+ bp->b_resid = acp->data_size;
+ }
+ biodone(bp);
+ atapi_free_pkt(acp);
+}
+
+struct atapi_command_packet *
+atapi_get_pkt(ad_link, flags)
+ struct at_dev_link *ad_link;
+ int flags;
+{
+ struct atapi_command_packet *pkt;
+ int s;
+
+ s = splbio();
+ while (ad_link->openings <= 0) {
+ if (flags & A_NOSLEEP) {
+ splx(s);
+ return 0;
+ }
+
+ ATAPI_DEBUG_CMD_PRINT(("atapi_get_pkt: sleeping\n"));
+
+ ad_link->flags |= ADEV_WAITING;
+ (void)tsleep(ad_link, PRIBIO, "getpkt", 0);
+ }
+
+ ad_link->openings--;
+
+ if ((pkt = pkt_free_list.lh_first) != 0) {
+ LIST_REMOVE(pkt, free_list);
+ splx(s);
+ } else {
+ splx(s);
+ pkt = malloc(sizeof(struct atapi_command_packet), M_DEVBUF,
+ ((flags & A_NOSLEEP) != 0 ? M_NOWAIT : M_WAITOK));
+ if (pkt == NULL) {
+ printf("atapi_get_pkt: cannot allocate pkt\n");
+ ad_link->openings++;
+ return 0;
+ }
+ }
+
+ bzero(pkt, sizeof(struct atapi_command_packet));
+ pkt->ad_link = ad_link;
+ return pkt;
+}
+
+void
+atapi_free_pkt(pkt)
+ struct atapi_command_packet *pkt;
+{
+ struct at_dev_link *ad_link = pkt->ad_link;
+ int s;
+
+ s = splbio();
+ LIST_INSERT_HEAD(&pkt_free_list, pkt, free_list);
+
+ ad_link->openings++;
+
+ if ((ad_link->flags & ADEV_WAITING) != 0) {
+ ad_link->flags &= ~ADEV_WAITING;
+ wakeup(ad_link);
+ } else {
+ if (ad_link->start) {
+ ATAPI_DEBUG_CMD_PRINT(("atapi_free_pkt: calling private start\n"));
+ (*ad_link->start)((void *)ad_link->device_softc);
+ }
+ }
+ splx(s);
+}
+
+int
+atapi_test_unit_ready(ad_link, flags)
+ struct at_dev_link *ad_link;
+ int flags;
+{
+ int ret;
+ struct test_unit_ready cmd;
+
+ ATAPI_DEBUG_FCTN_PRINT(("atapi_test_unit_ready: "));
+
+ bzero(&cmd, sizeof(cmd));
+ cmd.operation_code = ATAPI_TEST_UNIT_READY;
+
+ ret = atapi_exec_cmd(ad_link, &cmd, sizeof(cmd), 0, 0, 0, flags);
+
+ ATAPI_DEBUG_FCTN_PRINT(("atapi_test_unit_ready: ret %d\n", ret));
+
+ return ret;
+}
+
+int
+atapi_start_stop(ad_link, how, flags)
+ struct at_dev_link *ad_link;
+ int how;
+ int flags;
+{
+ struct start_stop_unit cmd;
+ int ret;
+
+ ATAPI_DEBUG_FCTN_PRINT(("atapi_start_stop: "));
+
+ bzero(&cmd, sizeof(cmd));
+ cmd.operation_code = ATAPI_START_STOP_UNIT;
+ cmd.how = how;
+
+ ret = atapi_exec_cmd(ad_link, &cmd, sizeof(cmd), 0,0,0,flags);
+
+ ATAPI_DEBUG_FCTN_PRINT(("ret %d\n", ret));
+
+ return ret;
+}
+
+int
+atapi_prevent(ad_link, how)
+ struct at_dev_link *ad_link;
+ int how;
+{
+ struct prevent_allow_medium_removal cmd;
+ int ret;
+
+ ATAPI_DEBUG_FCTN_PRINT(("atapi_prevent: "));
+
+ bzero(&cmd, sizeof(cmd));
+ cmd.operation_code = ATAPI_PREVENT_ALLOW_MEDIUM_REMOVAL;
+ cmd.how = how & 0xff;
+
+ ret = atapi_exec_cmd(ad_link, &cmd, sizeof(cmd), 0,0,0,0);
+
+ ATAPI_DEBUG_FCTN_PRINT(("ret %d\n", ret));
+
+ return ret;
+}
+
+int
+atapi_error(acp)
+ struct atapi_command_packet* acp;
+{
+ int flags, error, ret = -1;
+ struct at_dev_link *ad_link = acp->ad_link;
+
+ flags = acp->flags & 0xff;
+ error = acp->error;
+
+ at_print_addr(ad_link, acp->flags & 0xff);
+
+ if (error & ATAPI_MCR) {
+ SILENT_PRINTF(flags,("media change requested"));
+ acp->status = MEDIA_CHANGE;
+ }
+
+ if (error & ATAPI_ABRT) {
+ SILENT_PRINTF(flags,("command aborted"));
+ acp->status = ERROR;
+ }
+
+ if (error & ATAPI_EOM) {
+ SILENT_PRINTF(flags,("end of media"));
+ acp->status = END_OF_MEDIA;
+ }
+
+ if (error & ATAPI_ILI) {
+ SILENT_PRINTF(flags,("illegal length indication"));
+ acp->status = ERROR;
+ }
+
+ if ((error & 0x0f) == 0)
+ ret = 0;
+
+ atapi_sense(acp, error >> 4, flags);
+
+ if (((flags & A_SILENT) == 0) && (acp->status != NO_ERROR)) {
+ int i;
+ printf(", command:");
+ for (i = 0; i < acp->command_size; i++)
+ printf(" %2x", ((u_int8_t *)acp->command)[i]);
+ }
+
+ return ret;
+}
+
+void
+atapi_sense(acp, sense_key, flags)
+ struct atapi_command_packet *acp;
+ u_int8_t sense_key;
+ u_int8_t flags;
+{
+ struct at_dev_link *ad_link = acp->ad_link;
+
+ switch (sense_key) {
+ case ATAPI_SK_NO_SENSE:
+ break;
+
+ case ATAPI_SK_REC_ERROR:
+ SILENT_PRINTF(flags,("recovered error"));
+ acp->status = 0;
+ break;
+
+ case ATAPI_SK_NOT_READY:
+ SILENT_PRINTF(flags,("not ready"));
+ acp->status = NOT_READY;
+ break;
+
+ case ATAPI_SK_MEDIUM_ERROR:
+ SILENT_PRINTF(flags,("medium error"));
+ acp->status = ERROR;
+ break;
+
+ case ATAPI_SK_HARDWARE_ERROR:
+ SILENT_PRINTF(flags,("hardware error"));
+ acp->status = ERROR;
+ break;
+
+ case ATAPI_SK_ILLEGAL_REQUEST:
+ SILENT_PRINTF(flags,("illegal request"));
+ acp->status = ERROR;
+ break;
+
+ case ATAPI_SK_UNIT_ATTENTION:
+ SILENT_PRINTF(flags,("unit attention"));
+ acp->status = UNIT_ATTENTION;
+ if (ad_link->flags & ADEV_REMOVABLE)
+ ad_link->flags &= ~ADEV_MEDIA_LOADED;
+ break;
+
+ case ATAPI_SK_DATA_PROTECT:
+ SILENT_PRINTF(flags,("data protect"));
+ acp->status = ERROR;
+ break;
+
+ case ATAPI_SK_ABORTED_COMMAND:
+ SILENT_PRINTF(flags,("aborted command"));
+ acp->status = RETRY;
+ break;
+
+ case ATAPI_SK_MISCOMPARE:
+ SILENT_PRINTF(flags,("miscompare"));
+ acp->status = ERROR;
+ break;
+
+ default:
+ SILENT_PRINTF(flags,("unexpected sense key %02x", sense_key));
+ acp->status = ERROR;
+ }
+}
+
+void
+at_print_addr(ad_link, flags)
+ struct at_dev_link *ad_link;
+ u_int8_t flags;
+{
+
+ if (flags & A_SILENT)
+ return;
+
+ printf("%s(%s:%d): ", ad_link->device_softc ?
+ ((struct device *)ad_link->device_softc)->dv_xname : "probe",
+ ((struct device *)ad_link->bus->wdc_softc)->dv_xname,
+ ad_link->drive);
+}