diff options
author | Dale Rahn <drahn@cvs.openbsd.org> | 2010-02-12 05:31:52 +0000 |
---|---|---|
committer | Dale Rahn <drahn@cvs.openbsd.org> | 2010-02-12 05:31:52 +0000 |
commit | d7628c05214d2c6dbc670d8457a4203eccdea76c (patch) | |
tree | efd99ada46dc86fe8c2b2f2a356c9a69a4bcc1cd /sys/arch/beagle/dev/omehci.c | |
parent | 1ad12adfea36d57aa632c1f6d4a526f52e4f3f35 (diff) |
Initial version of ehci shim for omap, not complete.
Diffstat (limited to 'sys/arch/beagle/dev/omehci.c')
-rw-r--r-- | sys/arch/beagle/dev/omehci.c | 269 |
1 files changed, 269 insertions, 0 deletions
diff --git a/sys/arch/beagle/dev/omehci.c b/sys/arch/beagle/dev/omehci.c new file mode 100644 index 00000000000..fea8b20e6b1 --- /dev/null +++ b/sys/arch/beagle/dev/omehci.c @@ -0,0 +1,269 @@ +/* $OpenBSD: omehci.c,v 1.1 2010/02/12 05:31:51 drahn Exp $ */ + +/* + * Copyright (c) 2005 David Gwynne <dlg@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/systm.h> +#include <sys/device.h> +#include <sys/kernel.h> +#include <sys/rwlock.h> +#include <sys/timeout.h> + +#include <machine/intr.h> +#include <machine/bus.h> + +#include <dev/usb/usb.h> +#include <dev/usb/usbdi.h> +#include <dev/usb/usbdivar.h> +#include <dev/usb/usb_mem.h> + +#include <arch/beagle/beagle/ahb.h> +#include <beagle/dev/prcmvar.h> + +#include <dev/usb/ehcireg.h> +#include <dev/usb/ehcivar.h> + +int omehci_match(struct device *, void *, void *); +void omehci_attach(struct device *, struct device *, void *); +int omehci_detach(struct device *, int); +void omehci_power(int, void *); + +struct omehci_softc { + ehci_softc_t sc; + void *sc_ih; +}; + +void omehci_enable(struct omehci_softc *); +void omehci_disable(struct omehci_softc *); + +struct cfattach omehci_ca = { + sizeof (struct omehci_softc), omehci_match, omehci_attach, + omehci_detach, ehci_activate +}; + +int +omehci_match(struct device *parent, void *match, void *aux) +{ + struct ahb_attach_args *aa = aux; + + /* XXX - 3[45]30 */ + if (aa->aa_addr != 0x48064800) + return 0; + + return (1); +} + +void +omehci_attach(struct device *parent, struct device *self, void *aux) +{ + struct omehci_softc *sc = (struct omehci_softc *)self; + struct ahb_attach_args *aa = aux; + usbd_status r; + char *devname = sc->sc.sc_bus.bdev.dv_xname; + + + sc->sc.iot = aa->aa_iot; + sc->sc.sc_bus.dmatag = aa->aa_dmat; + sc->sc.sc_size = 0; + + /* Map I/O space */ + if (bus_space_map(sc->sc.iot, aa->aa_addr, aa->aa_size, 0, + &sc->sc.ioh)) { + printf(": cannot map mem space\n"); + return; + } + sc->sc.sc_size = aa->aa_size; + + /* XXX copied from ohci_pci.c. needed? */ + bus_space_barrier(sc->sc.iot, sc->sc.ioh, 0, sc->sc.sc_size, + BUS_SPACE_BARRIER_READ|BUS_SPACE_BARRIER_WRITE); + + prcm_enableclock(PRCM_CLK_EN_USB); + +#if 0 + omehci_enable(sc); +#endif + + /* Disable interrupts, so we don't get any spurious ones. */ + sc->sc.sc_offs = EREAD1(&sc->sc, EHCI_CAPLENGTH); + EOWRITE2(&sc->sc, EHCI_USBINTR, 0); + + sc->sc_ih = intc_intr_establish(aa->aa_intr, IPL_USB, + ehci_intr, &sc->sc, devname); + if (sc->sc_ih == NULL) { + printf(": unable to establish interrupt\n"); +#if 0 + omehci_disable(sc); +#endif + +#if 0 + prcm_disableclock(PRCM_CLK_EN_USB); +#endif + + bus_space_unmap(sc->sc.iot, sc->sc.ioh, sc->sc.sc_size); + sc->sc.sc_size = 0; + return; + } + + strlcpy(sc->sc.sc_vendor, "OMAP3[45]xx", sizeof(sc->sc.sc_vendor)); + r = ehci_init(&sc->sc); + if (r != USBD_NORMAL_COMPLETION) { + printf("%s: init failed, error=%d\n", devname); + + intc_intr_disestablish(sc->sc_ih); + sc->sc_ih = NULL; + + bus_space_unmap(sc->sc.iot, sc->sc.ioh, sc->sc.sc_size); + +#if 0 + prcm_disableclock(PRCM_CLK_EN_USB); +#endif + sc->sc.sc_size = 0; + return; + } + +#if 0 + sc->sc.sc_powerhook = powerhook_establish(omehci_power, sc); + if (sc->sc.sc_powerhook == NULL) + printf("%s: cannot establish powerhook\n", devname); +#endif + sc->sc.sc_shutdownhook = shutdownhook_establish(ehci_shutdown, &sc->sc); + + sc->sc.sc_child = config_found((void *)sc, &sc->sc.sc_bus, + usbctlprint); +} + +int +omehci_detach(struct device *self, int flags) +{ + struct omehci_softc *sc = (struct omehci_softc *)self; + int rv; + + rv = ehci_detach(&sc->sc, flags); + if (rv) + return (rv); + +#if 0 + if (sc->sc.sc_powerhook != NULL) { + powerhook_disestablish(sc->sc.sc_powerhook); + sc->sc.sc_powerhook = NULL; + } +#endif + + if (sc->sc_ih != NULL) { + intc_intr_disestablish(sc->sc_ih); + sc->sc_ih = NULL; + } + + if (sc->sc.sc_size) { + bus_space_unmap(sc->sc.iot, sc->sc.ioh, sc->sc.sc_size); + sc->sc.sc_size = 0; + } + + /* stop clock */ +#if 0 + prcm_disableclock(PRCM_CLK_EN_USB); +#endif + + return (0); +} + + +#if 0 +void +omehci_power(int why, void *arg) +{ + struct omehci_softc *sc = (struct omehci_softc *)arg; + int s; + + s = splhardusb(); + sc->sc.sc_bus.use_polling++; + switch (why) { + case PWR_STANDBY: + case PWR_SUSPEND: + ohci_power(why, &sc->sc); +#if 0 + prcm_disableclock(PRCM_CLK_EN_USB); +#endif + break; + + case PWR_RESUME: + prcm_enableclock(PRCM_CLK_EN_USB); + +#if 0 + omehci_enable(sc); +#endif + ohci_power(why, &sc->sc); + break; + } + sc->sc.sc_bus.use_polling--; + splx(s); +} +#endif + +#if 0 +void +omehci_enable(struct omehci_softc *sc) +{ + u_int32_t hr; + + /* Full host reset */ + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) | USBHC_HR_FHR); + + DELAY(USBHC_RST_WAIT); + + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) & ~(USBHC_HR_FHR)); + + /* Force system bus interface reset */ + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) | USBHC_HR_FSBIR); + + while (bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR) & \ + USBHC_HR_FSBIR) + DELAY(3); + + /* Enable the ports (physically only one, only enable that one?) */ + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) & ~(USBHC_HR_SSE)); + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) & ~(USBHC_HR_SSEP2)); +} + +void +omehci_disable(struct omehci_softc *sc) +{ + u_int32_t hr; + + /* Full host reset */ + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) | USBHC_HR_FHR); + + DELAY(USBHC_RST_WAIT); + + hr = bus_space_read_4(sc->sc.iot, sc->sc.ioh, USBHC_HR); + bus_space_write_4(sc->sc.iot, sc->sc.ioh, USBHC_HR, + (hr & USBHC_HR_MASK) & ~(USBHC_HR_FHR)); +} +#endif |