summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64/dev/mgiic.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc64/dev/mgiic.c')
-rw-r--r--sys/arch/sparc64/dev/mgiic.c334
1 files changed, 334 insertions, 0 deletions
diff --git a/sys/arch/sparc64/dev/mgiic.c b/sys/arch/sparc64/dev/mgiic.c
new file mode 100644
index 00000000000..f122477923a
--- /dev/null
+++ b/sys/arch/sparc64/dev/mgiic.c
@@ -0,0 +1,334 @@
+/* $OpenBSD: mgiic.c,v 1.1 2008/04/21 04:50:23 deraadt Exp $ */
+/*
+ * Copyright (c) 2008 Mark Kettenis <kettenis@openbsd.org>
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+#include <sys/param.h>
+#include <sys/device.h>
+#include <sys/errno.h>
+#include <sys/malloc.h>
+#include <sys/systm.h>
+#include <sys/rwlock.h>
+
+#include <uvm/uvm_extern.h>
+
+#define _SPARC_BUS_DMA_PRIVATE
+#include <machine/bus.h>
+#include <machine/autoconf.h>
+#include <machine/openfirm.h>
+
+#include <dev/i2c/i2cvar.h>
+#include <sparc64/dev/ofwi2cvar.h>
+
+#define MGIIC_SLAVEADDR 0x00
+#define MGIIC_SLAVEXADDR 0x08
+#define MGIIC_DATA 0x10
+#define MGIIC_CONTROL 0x18
+#define MGIIC_CONTROL_IEN 0x80
+#define MGIIC_CONTROL_ENAB 0x40
+#define MGIIC_CONTROL_STA 0x20
+#define MGIIC_CONTROL_STP 0x10
+#define MGIIC_CONTROL_IFLG 0x08
+#define MGIIC_CONTROL_AAK 0x04
+#define MGIIC_STATUS 0x20
+#define MGIIC_STATUS_BUSERR 0x00
+#define MGIIC_STATUS_STARTSENT 0x08
+#define MGIIC_STATUS_REPEATSTART 0x10
+#define MGIIC_STATUS_ADDR_W_ACKR 0x18
+#define MGIIC_STATUS_ADDR_W_NOACKR 0x20
+#define MGIIC_STATUS_MDATA_ACKR 0x28
+#define MGIIC_STATUS_MDATA_NOACKR 0x30
+#define MGIIC_STATUS_ARBLOST 0x38
+#define MGIIC_STATUS_ADDR_R_ACKR 0x40
+#define MGIIC_STATUS_ADDR_R_NOACKR 0x48
+#define MGIIC_STATUS_MDATA_ACKT 0x50
+#define MGIIC_STATUS_MDATA_NOACKT 0x58
+#define MGIIC_STATUS_SADDR_W_ACKT 0x60
+#define MGIIC_STATUS_ARBLOST_SLW_ACKT 0x68
+#define MGIIC_STATUS_GC_TACK 0x70
+#define MGIIC_STATUS_ARBLOST_GC_ACKT 0x78
+#define MGIIC_STATUS_IDLE 0xf8
+#define MGIIC_CLOCKCONTROL 0x28
+#define MGIIC_SOFTRESET 0x30
+
+struct mgiic_softc {
+ struct device sc_dev;
+
+ bus_space_tag_t sc_bt;
+ bus_space_handle_t sc_regh;
+
+
+ int sc_poll;
+
+ struct i2c_controller sc_i2c;
+ struct rwlock sc_lock;
+};
+
+int mgiic_match(struct device *, void *, void *);
+void mgiic_attach(struct device *, struct device *, void *);
+
+struct cfdriver mgiic_cd = {
+ NULL, "mgiic", DV_DULL
+};
+
+struct cfattach mgiic_ca = {
+ sizeof(struct mgiic_softc), mgiic_match, mgiic_attach
+};
+
+int mgiic_i2c_acquire_bus(void *, int);
+void mgiic_i2c_release_bus(void *, int);
+int mgiic_i2c_exec(void *, i2c_op_t, i2c_addr_t, const void *,
+ size_t, void *, size_t, int);
+
+int mgiic_xmit(struct mgiic_softc *, u_int8_t, const u_int8_t *, size_t);
+int mgiic_recv(struct mgiic_softc *, u_int8_t, u_int8_t *, size_t);
+volatile u_int8_t mgiic_read(struct mgiic_softc *, bus_size_t);
+volatile void mgiic_write(struct mgiic_softc *, bus_size_t, u_int8_t);
+volatile void mgiic_control(struct mgiic_softc *, u_int8_t, u_int8_t);
+int mgiic_poll(struct mgiic_softc *);
+
+int
+mgiic_match(struct device *parent, void *match, void *aux)
+{
+ struct mainbus_attach_args *ma = aux;
+ char compat[32];
+
+ if (strcmp(ma->ma_name, "i2c") != 0)
+ return (0);
+
+ if (OF_getprop(ma->ma_node, "compatible", compat, sizeof(compat)) == -1)
+ return (0);
+
+ if (strcmp(compat, "fire-i2c") == 0)
+ return (1);
+
+ return (0);
+}
+
+void
+mgiic_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct mgiic_softc *sc = (struct mgiic_softc *)self;
+ struct mainbus_attach_args *ma = aux;
+ struct i2cbus_attach_args iba;
+ int i;
+
+ sc->sc_bt = ma->ma_bustag;
+
+ if (bus_space_map(sc->sc_bt, ma->ma_reg[0].ur_paddr,
+ ma->ma_reg[0].ur_len, 0, &sc->sc_regh)) {
+ printf(": failed to map preg\n");
+ return;
+ }
+
+ printf(": ");
+ for (i = 0; i < 7; i++)
+ printf("%04x ", mgiic_read(sc, i * 8));
+ printf("\n");
+
+ rw_init(&sc->sc_lock, "iiclk");
+ sc->sc_i2c.ic_cookie = sc;
+ sc->sc_i2c.ic_acquire_bus = mgiic_i2c_acquire_bus;
+ sc->sc_i2c.ic_release_bus = mgiic_i2c_release_bus;
+ sc->sc_i2c.ic_exec = mgiic_i2c_exec;
+
+ bzero(&iba, sizeof(iba));
+ iba.iba_name = "iic";
+ iba.iba_tag = &sc->sc_i2c;
+ iba.iba_bus_scan = ofwiic_scan;
+ iba.iba_bus_scan_arg = &ma->ma_node;
+ config_found(&sc->sc_dev, &iba, iicbus_print);
+}
+
+int
+mgiic_i2c_acquire_bus(void *arg, int flags)
+{
+ struct mgiic_softc *sc = arg;
+
+ if (cold || sc->sc_poll || (flags & I2C_F_POLL))
+ return (0);
+
+ return (rw_enter(&sc->sc_lock, RW_WRITE | RW_INTR));
+}
+
+void
+mgiic_i2c_release_bus(void *arg, int flags)
+{
+ struct mgiic_softc *sc = arg;
+
+ if (cold || sc->sc_poll || (flags & I2C_F_POLL))
+ return;
+
+ rw_exit(&sc->sc_lock);
+}
+
+int
+mgiic_i2c_exec(void *arg, i2c_op_t op, i2c_addr_t addr,
+ const void *cmdbuf, size_t cmdlen, void *buf, size_t len, int flags)
+{
+ struct mgiic_softc *sc = arg;
+ int ret = 0, i = 0;
+
+ if (addr & ~0x7f)
+ return (1);
+
+ if (cold || sc->sc_poll)
+ flags |= I2C_F_POLL;
+
+ if (cmdlen > 0)
+ if (mgiic_xmit(sc, addr & 0x7f, cmdbuf, cmdlen) != 0)
+ return (1);
+
+ if (len > 0) {
+ if (I2C_OP_WRITE_P(op))
+ ret = mgiic_xmit(sc, addr & 0x7f, buf, len);
+ else
+ ret = mgiic_recv(sc, addr & 0x7f, buf, len);
+ }
+ printf("e%d\n", ret);
+ for (i = 0; i < 7; i++)
+ printf("%04x ", mgiic_read(sc, i * 8));
+ printf("\n");
+ return (ret);
+}
+
+int
+mgiic_xmit(struct mgiic_softc *sc, u_int8_t addr, const u_int8_t *buf,
+ size_t len)
+{
+ int err = 1;
+ int i;
+
+ mgiic_control(sc, MGIIC_CONTROL_STA, 0);
+
+ if (mgiic_poll(sc))
+ goto bail;
+ if (mgiic_read(sc, MGIIC_STATUS) != 0x08)
+ goto bail;
+
+ mgiic_write(sc, MGIIC_DATA, addr << 1);
+ mgiic_control(sc, 0, MGIIC_CONTROL_IFLG);
+
+ for (i = 0; i < len; i++) {
+ if (mgiic_poll(sc))
+ goto bail;
+ switch (mgiic_read(sc, MGIIC_STATUS)) {
+ case 0x18:
+ case 0x20:
+ case 0x28:
+ case 0x30:
+ mgiic_write(sc, MGIIC_DATA, buf[i]);
+ printf("w%02x ", buf[i]);
+ mgiic_control(sc, 0, MGIIC_CONTROL_IFLG);
+ break;
+ default:
+ goto bail;
+ }
+ }
+ err = 0;
+bail:
+ mgiic_control(sc, MGIIC_CONTROL_STP, MGIIC_CONTROL_IFLG);
+ while (mgiic_read(sc, MGIIC_STATUS) != 0xf8)
+ ;
+ return (err);
+}
+
+int
+mgiic_recv(struct mgiic_softc *sc, u_int8_t addr, u_int8_t *buf, size_t len)
+{
+ int err = 1;
+ int i;
+
+ mgiic_control(sc, MGIIC_CONTROL_STA, 0);
+
+ if (mgiic_poll(sc))
+ goto bail;
+ if (mgiic_read(sc, MGIIC_STATUS) != 0x08)
+ goto bail;
+
+ mgiic_write(sc, MGIIC_DATA, (addr << 1) | 0x01);
+ mgiic_control(sc, 0, MGIIC_CONTROL_IFLG);
+
+ for (i = 0; i < len; ) {
+ if (mgiic_poll(sc))
+ goto bail;
+ switch (mgiic_read(sc, MGIIC_STATUS)) {
+ case 0x40:
+ mgiic_control(sc, 0, MGIIC_CONTROL_IFLG);
+ break;
+ case 0x48:
+ mgiic_control(sc, MGIIC_CONTROL_STA, MGIIC_CONTROL_IFLG);
+ break;
+ case 0x50:
+ case 0x58:
+ buf[i] = mgiic_read(sc, MGIIC_DATA);
+ printf("r%02x ", buf[i]);
+ i++;
+ mgiic_control(sc, MGIIC_CONTROL_AAK, MGIIC_CONTROL_IFLG);
+ break;
+ default:
+ goto bail;
+ }
+ }
+ err = 0;
+bail:
+ mgiic_control(sc, MGIIC_CONTROL_STP, MGIIC_CONTROL_IFLG);
+ while (mgiic_read(sc, MGIIC_STATUS) != 0xf8)
+ ;
+ return (err);
+}
+
+volatile u_int8_t
+mgiic_read(struct mgiic_softc *sc, bus_size_t r)
+{
+ bus_space_barrier(sc->sc_bt, sc->sc_regh, r, 8,
+ BUS_SPACE_BARRIER_READ);
+ return (bus_space_read_8(sc->sc_bt, sc->sc_regh, r)) & 0xff;
+}
+
+volatile void
+mgiic_write(struct mgiic_softc *sc, bus_size_t r, u_int8_t v)
+{
+ u_int64_t val = v;
+
+ bus_space_write_8(sc->sc_bt, sc->sc_regh, r, val);
+ bus_space_barrier(sc->sc_bt, sc->sc_regh, r, 8,
+ BUS_SPACE_BARRIER_WRITE);
+}
+
+volatile void
+mgiic_control(struct mgiic_softc *sc, u_int8_t on, u_int8_t off)
+{
+ u_int8_t val;
+
+ val = (mgiic_read(sc, MGIIC_CONTROL) | on) & ~off;
+ printf("s%02xc%02x ", mgiic_read(sc, MGIIC_STATUS), val);
+ return mgiic_write(sc, MGIIC_CONTROL, val);
+}
+
+int
+mgiic_poll(struct mgiic_softc *sc)
+{
+ int i;
+
+ for (i = 0; i < 1000; i++) {
+ if (mgiic_read(sc, MGIIC_CONTROL) & MGIIC_CONTROL_IFLG) {
+ printf("s%02x ", mgiic_read(sc, MGIIC_STATUS));
+ return (0);
+ }
+ delay(100);
+ }
+ return (1);
+}