summaryrefslogtreecommitdiff
path: root/sys/dev/isa/wdc.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/isa/wdc.c')
-rw-r--r--sys/dev/isa/wdc.c41
1 files changed, 31 insertions, 10 deletions
diff --git a/sys/dev/isa/wdc.c b/sys/dev/isa/wdc.c
index 48094436200..69720b8d9cf 100644
--- a/sys/dev/isa/wdc.c
+++ b/sys/dev/isa/wdc.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: wdc.c,v 1.19 1997/07/04 17:02:04 downsj Exp $ */
+/* $OpenBSD: wdc.c,v 1.20 1997/07/04 19:17:56 downsj Exp $ */
/* $NetBSD: wd.c,v 1.150 1996/05/12 23:54:03 mycroft Exp $ */
/*
@@ -100,6 +100,8 @@ struct cfdriver wdc_cd = {
NULL, "wdc", DV_DULL
};
+enum wdcreset_mode { WDCRESET_VERBOSE, WDCRESET_SILENT };
+
#if NWD > 0
int wdc_ata_intr __P((struct wdc_softc *,struct wdc_xfer *));
void wdc_ata_start __P((struct wdc_softc *,struct wdc_xfer *));
@@ -109,7 +111,7 @@ __inline static void u_int16_to_string __P((u_int16_t *, char *, size_t));
int wait_for_phase __P((struct wdc_softc *, int));
int wait_for_unphase __P((struct wdc_softc *, int));
void wdcstart __P((struct wdc_softc *));
-int wdcreset __P((struct wdc_softc *));
+int wdcreset __P((struct wdc_softc *, enum wdcreset_mode));
void wdcrestart __P((void *arg));
void wdcunwedge __P((struct wdc_softc *));
void wdctimeout __P((void *arg));
@@ -175,10 +177,20 @@ wdcprobe(parent, match, aux)
wdc->sc_flags |= WDCF_ONESLAVE;
}
- if (wdcreset(wdc) != 0) {
- delay(500000);
- if (wdcreset(wdc) != 0)
- goto nomatch;
+ if (wdcreset(wdc, WDCRESET_SILENT) != 0) {
+ /*
+ * if the reset failed,, there is no master. test for ATAPI
+ * signature on the salve device. If no ATAPI slave, wait 5s
+ * and retry a reset.
+ */
+ bus_space_write_1(iot, ioh, wd_sdh, WDSD_IBM | 0x10); /* slave */
+ if (bus_space_read_1(iot, ioh, wd_cyl_lo) != 0x14 ||
+ bus_space_read_1(iot, ioh, wd_cyl_hi) != 0xeb) {
+ delay(500000);
+ if (wdcreset(wdc, WDCRESET_SILENT) != 0)
+ goto nomatch;
+ }
+ wdc->sc_flags |= WDCF_ONESLAVE;
}
/* Select drive 0 or ATAPI slave device */
@@ -233,6 +245,8 @@ wdcattach(parent, self, aux)
#if NWD > 0
int drive;
#endif /* NWD */
+ bus_space_tag_t iot = wdc->sc_iot;
+ bus_space_handle_t ioh = wdc->sc_ioh;
TAILQ_INIT(&wdc->sc_xfer);
wdc->sc_drq = ia->ia_drq;
@@ -268,6 +282,12 @@ wdcattach(parent, self, aux)
* Attach standard IDE/ESDI/etc. disks to the controller.
*/
for (drive = 0; drive < 2; drive++) {
+ /* test for ATAPI signature on this drive */
+ bus_space_write_1(iot, ioh, wd_sdh, WDSD_IBM | (drive << 4));
+ if (bus_space_read_1(iot, ioh, wd_cyl_lo) == 0x14 &&
+ bus_space_read_1(iot, ioh, wd_cyl_hi) == 0xeb) {
+ continue;
+ }
/* controller active while autoconf */
wdc->sc_flags |= WDCF_ACTIVE;
@@ -1126,7 +1146,7 @@ wdc_atapi_get_params(ab_link, drive, id)
wdc->sc_flags |= WDCF_ACTIVE;
error = 1;
- (void)wdcreset(wdc);
+ (void)wdcreset(wdc, WDCRESET_VERBOSE);
if ((status = wdccommand((struct wd_link*)ab_link,
ATAPI_SOFT_RESET, drive, 0, 0, 0, 0)) != 0) {
#ifdef ATAPI_DEBUG
@@ -1543,8 +1563,9 @@ wdcintr(arg)
}
int
-wdcreset(wdc)
+wdcreset(wdc, mode)
struct wdc_softc *wdc;
+ enum wdcreset_mode mode;
{
bus_space_tag_t iot = wdc->sc_iot;
bus_space_handle_t ioh = wdc->sc_ioh;
@@ -1557,7 +1578,7 @@ wdcreset(wdc)
(void) bus_space_read_1(iot, ioh, wd_error);
bus_space_write_1(iot, ioh, wd_ctlr, WDCTL_4BIT);
- if (wait_for_unbusy(wdc) < 0) {
+ if ((wait_for_unbusy(wdc) < 0) && mode != WDCRESET_SILENT) {
printf("%s: reset failed\n", wdc->sc_dev.dv_xname);
return 1;
}
@@ -1596,7 +1617,7 @@ wdcunwedge(wdc)
untimeout(wdctimeout, wdc);
wdc->sc_flags &= ~WDCF_IRQ_WAIT;
- (void) wdcreset(wdc);
+ (void) wdcreset(wdc, WDCRESET_VERBOSE);
/* Schedule recalibrate for all drives on this controller. */
for (unit = 0; unit < 2; unit++) {