summaryrefslogtreecommitdiff
path: root/sys/dev/pci/ichiic.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci/ichiic.c')
-rw-r--r--sys/dev/pci/ichiic.c59
1 files changed, 39 insertions, 20 deletions
diff --git a/sys/dev/pci/ichiic.c b/sys/dev/pci/ichiic.c
index bf4e7435070..707311a14c0 100644
--- a/sys/dev/pci/ichiic.c
+++ b/sys/dev/pci/ichiic.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ichiic.c,v 1.7 2006/01/02 08:11:25 brad Exp $ */
+/* $OpenBSD: ichiic.c,v 1.8 2006/01/05 08:16:22 grange Exp $ */
/*
* Copyright (c) 2005 Alexander Yurchenko <grange@openbsd.org>
@@ -116,6 +116,15 @@ ichiic_attach(struct device *parent, struct device *self, void *aux)
pci_intr_handle_t ih;
const char *intrstr = NULL;
+ /* Read configuration */
+ conf = pci_conf_read(pa->pa_pc, pa->pa_tag, ICH_SMB_HOSTC);
+ DPRINTF((": conf 0x%x", conf));
+
+ if ((conf & ICH_SMB_HOSTC_HSTEN) == 0) {
+ printf(": SMBus disabled\n");
+ return;
+ }
+
/* Map I/O space */
if (pci_mapreg_map(pa, ICH_SMB_BASE, PCI_MAPREG_TYPE_IO, 0,
&sc->sc_iot, &sc->sc_ioh, NULL, &iosize, 0)) {
@@ -123,10 +132,6 @@ ichiic_attach(struct device *parent, struct device *self, void *aux)
return;
}
- /* Read configuration */
- conf = pci_conf_read(pa->pa_pc, pa->pa_tag, ICH_SMB_HOSTC);
- DPRINTF((": conf 0x%x", conf));
-
if (conf & ICH_SMB_HOSTC_SMIEN) {
printf(": SMI");
sc->sc_poll = 1;
@@ -149,10 +154,6 @@ ichiic_attach(struct device *parent, struct device *self, void *aux)
printf(": %s", intrstr);
}
- /* Enable controller */
- pci_conf_write(pa->pa_pc, pa->pa_tag, ICH_SMB_HOSTC,
- conf | ICH_SMB_HOSTC_HSTEN);
-
printf("\n");
/* Attach I2C bus */
@@ -204,10 +205,15 @@ ichiic_i2c_exec(void *cookie, i2c_op_t op, i2c_addr_t addr,
u_int8_t ctl, st;
int retries;
- DPRINTF(("%s: exec op %d, addr 0x%x, cmdlen %d, len %d, "
- "flags 0x%x, status 0x%b\n", sc->sc_dev.dv_xname, op, addr,
- cmdlen, len, flags, bus_space_read_1(sc->sc_iot, sc->sc_ioh,
- ICH_SMB_HS), ICH_SMB_HS_BITS));
+ DPRINTF(("%s: exec: op %d, addr 0x%x, cmdlen %d, len %d, flags 0x%x\n",
+ sc->sc_dev.dv_xname, op, addr, cmdlen, len, flags));
+
+ /* Check if there's a transfer already running */
+ st = bus_space_read_1(sc->sc_iot, sc->sc_ioh, ICH_SMB_HS);
+ DPRINTF(("%s: exec: st 0x%b\n", sc->sc_dev.dv_xname, st,
+ ICH_SMB_HS_BITS));
+ if (st & ICH_SMB_HS_BUSY)
+ return (1);
if (cold || sc->sc_poll)
flags |= I2C_F_POLL;
@@ -268,22 +274,35 @@ ichiic_i2c_exec(void *cookie, i2c_op_t op, i2c_addr_t addr,
break;
DELAY(ICHIIC_DELAY);
}
- if (st & ICH_SMB_HS_BUSY) {
- printf("%s: timeout, status 0x%b\n",
- sc->sc_dev.dv_xname, st, ICH_SMB_HS_BITS);
- return (1);
- }
+ if (st & ICH_SMB_HS_BUSY)
+ goto timeout;
ichiic_intr(sc);
} else {
/* Wait for interrupt */
if (tsleep(sc, PRIBIO, "iicexec", ICHIIC_TIMEOUT * hz))
- return (1);
- }
+ goto timeout;
+ }
if (sc->sc_i2c_xfer.error)
return (1);
return (0);
+
+timeout:
+ /*
+ * Transfer timeout. Kill the transaction and clear status bits.
+ */
+ printf("%s: timeout, status 0x%b\n", sc->sc_dev.dv_xname, st,
+ ICH_SMB_HS_BITS);
+ bus_space_write_1(sc->sc_iot, sc->sc_ioh, ICH_SMB_HC,
+ ICH_SMB_HC_KILL);
+ DELAY(ICHIIC_DELAY);
+ st = bus_space_read_1(sc->sc_iot, sc->sc_ioh, ICH_SMB_HS);
+ if ((st & ICH_SMB_HS_FAILED) == 0)
+ printf("%s: transaction abort failed, status 0x%b\n",
+ sc->sc_dev.dv_xname, st, ICH_SMB_HS_BITS);
+ bus_space_write_1(sc->sc_iot, sc->sc_ioh, ICH_SMB_HS, st);
+ return (1);
}
int