diff options
author | Patrick Wildt <patrick@cvs.openbsd.org> | 2013-09-06 20:45:55 +0000 |
---|---|---|
committer | Patrick Wildt <patrick@cvs.openbsd.org> | 2013-09-06 20:45:55 +0000 |
commit | cbccf2bd320a00fd2d28064002c74254167e613c (patch) | |
tree | 323c94f9972ad75f3f2318d9cac70847e3400a66 /sys/arch/armv7/imx/imxahci.c | |
parent | aba4f7ffe403094c2f9d96695a052a49d4a82890 (diff) |
Support for FreeScale's i.MX6 SoC.
Diffstat (limited to 'sys/arch/armv7/imx/imxahci.c')
-rw-r--r-- | sys/arch/armv7/imx/imxahci.c | 173 |
1 files changed, 173 insertions, 0 deletions
diff --git a/sys/arch/armv7/imx/imxahci.c b/sys/arch/armv7/imx/imxahci.c new file mode 100644 index 00000000000..ffe6f88845e --- /dev/null +++ b/sys/arch/armv7/imx/imxahci.c @@ -0,0 +1,173 @@ +/* $OpenBSD: imxahci.c,v 1.1 2013/09/06 20:45:53 patrick Exp $ */ +/* + * Copyright (c) 2013 Patrick Wildt <patrick@blueri.se> + * + * 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/buf.h> +#include <sys/kernel.h> +#include <sys/malloc.h> +#include <sys/device.h> +#include <sys/queue.h> + +#include <machine/bus.h> + +#include <dev/ic/ahcivar.h> + +#include <armv7/imx/imxvar.h> +#include <armv7/imx/imxccmvar.h> +#include <armv7/imx/imxiomuxcvar.h> + +/* registers */ +#define SATA_CAP 0x000 +#define SATA_GHC 0x004 +#define SATA_IS 0x008 +#define SATA_PI 0x00C +#define SATA_VS 0x010 +#define SATA_CCC_CTL 0x014 +#define SATA_CCC_PORTS 0x018 +#define SATA_CAP2 0x024 +#define SATA_BISTAFR 0x0A0 +#define SATA_BISTCR 0x0A4 +#define SATA_BISTFCTR 0x0A8 +#define SATA_BSTSR 0x0AC +#define SATA_OOBR 0x0BC +#define SATA_GPCR 0x0D0 +#define SATA_GPSR 0x0D4 +#define SATA_TIMER1MS 0x0E0 +#define SATA_TESTR 0x0F4 +#define SATA_VERSIONR 0x0F8 +#define SATA_P0CLB 0x100 +#define SATA_P0FB 0x108 +#define SATA_P0IS 0x110 +#define SATA_P0IE 0x114 +#define SATA_P0CMD 0x118 +#define SATA_P0TFD 0x120 +#define SATA_P0SIG 0x124 +#define SATA_P0SSTS 0x128 +#define SATA_P0SCTL 0x12C +#define SATA_P0SERR 0x130 +#define SATA_P0SACT 0x134 +#define SATA_P0CI 0x138 +#define SATA_P0SNTF 0x13C +#define SATA_P0DMACR 0x170 +#define SATA_P0PHYCR 0x178 +#define SATA_P0PHYSR 0x17C + +#define SATA_CAP_SSS (1 << 27) +#define SATA_GHC_HR (1 << 0) +#define SATA_P0PHYCR_TEST_PDDQ (1 << 20) + +void imxahci_attach(struct device *, struct device *, void *); +int imxahci_detach(struct device *, int); +int imxahci_activate(struct device *, int); + +extern int ahci_intr(void *); + +struct imxahci_softc { + struct ahci_softc sc; +}; + +struct cfattach imxahci_ca = { + sizeof(struct imxahci_softc), + NULL, + imxahci_attach, + imxahci_detach, + imxahci_activate +}; + +struct cfdriver imxahci_cd = { + NULL, "ahci", DV_DULL +}; + +void +imxahci_attach(struct device *parent, struct device *self, void *args) +{ + struct imx_attach_args *ia = args; + struct imxahci_softc *imxsc = (struct imxahci_softc *) self; + struct ahci_softc *sc = &imxsc->sc; + uint32_t timeout = 0x100000; + + sc->sc_iot = ia->ia_iot; + sc->sc_ios = ia->ia_dev->mem[0].size; + sc->sc_dmat = ia->ia_dmat; + + if (bus_space_map(sc->sc_iot, ia->ia_dev->mem[0].addr, + ia->ia_dev->mem[0].size, 0, &sc->sc_ioh)) + panic("imxahci_attach: bus_space_map failed!"); + + sc->sc_ih = arm_intr_establish(ia->ia_dev->irq[0], IPL_BIO, + ahci_intr, sc, sc->sc_dev.dv_xname); + if (sc->sc_ih == NULL) { + printf(": unable to establish interrupt\n"); + goto unmap; + } + + /* power it up */ + imxccm_enable_sata(); + delay(100); + + /* power phy up */ + imxiomuxc_enable_sata(); + + /* setup */ + bus_space_write_4(sc->sc_iot, sc->sc_ioh, SATA_P0PHYCR, + bus_space_read_4(sc->sc_iot, sc->sc_ioh, SATA_P0PHYCR) & ~SATA_P0PHYCR_TEST_PDDQ); + + bus_space_write_4(sc->sc_iot, sc->sc_ioh, SATA_GHC, SATA_GHC_HR); + + while (!bus_space_read_4(sc->sc_iot, sc->sc_ioh, SATA_VERSIONR)); + + bus_space_write_4(sc->sc_iot, sc->sc_ioh, SATA_CAP, + bus_space_read_4(sc->sc_iot, sc->sc_ioh, SATA_CAP) | SATA_CAP_SSS); + + bus_space_write_4(sc->sc_iot, sc->sc_ioh, SATA_PI, 1); + + bus_space_write_4(sc->sc_iot, sc->sc_ioh, SATA_TIMER1MS, imxccm_get_ahbclk()); + + while (!(bus_space_read_4(sc->sc_iot, sc->sc_ioh, SATA_P0SSTS) & 0xF) && timeout--); + + if (ahci_attach(sc) != 0) { + /* error printed by ahci_attach */ + goto irq; + } + + return; +irq: + arm_intr_disestablish(sc->sc_ih); +unmap: + bus_space_unmap(sc->sc_iot, sc->sc_ioh, sc->sc_ios); +} + +int +imxahci_detach(struct device *self, int flags) +{ + struct imxahci_softc *imxsc = (struct imxahci_softc *) self; + struct ahci_softc *sc = &imxsc->sc; + + ahci_detach(sc, flags); + bus_space_unmap(sc->sc_iot, sc->sc_ioh, sc->sc_ios); + return 0; +} + +int +imxahci_activate(struct device *self, int act) +{ + struct imxahci_softc *imxsc = (struct imxahci_softc *) self; + struct ahci_softc *sc = &imxsc->sc; + + return ahci_activate((struct device *)sc, act); +} |