/* $Id: at91pio.c,v 1.6 2012/11/12 18:00:36 skrll Exp $ */ /* $NetBSD: at91pio.c,v 1.6 2012/11/12 18:00:36 skrll Exp $ */ /* * Copyright (c) 2007 Embedtronics Oy. All rights reserved. * * Based on arch/arm/ep93xx/epgpio.c, * Copyright (c) 2005 HAMAJIMA Katsuomi. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include __KERNEL_RCSID(0, "$NetBSD: at91pio.c,v 1.6 2012/11/12 18:00:36 skrll Exp $"); #include #include #include #include #include #include #include #include #include #include #include #include #include "gpio.h" #if NGPIO > 0 #include #endif #include "locators.h" #ifdef AT91PIO_DEBUG int at91pio_debug = AT91PIO_DEBUG; #define DPRINTFN(n,x) if (at91pio_debug>(n)) printf x; #else #define DPRINTFN(n,x) #endif #define AT91PIO_NMAXPORTS 4 #define AT91PIO_NPINS 32 struct intr_req { int (*ireq_func)(void *); void *ireq_arg; int ireq_ipl; }; #define PIO_READ(_sc, _reg) bus_space_read_4((_sc)->sc_iot, (_sc)->sc_ioh, (_reg)) #define PIO_WRITE(_sc, _reg, _val) bus_space_write_4((_sc)->sc_iot, (_sc)->sc_ioh, (_reg), (_val)) struct at91pio_softc { bus_space_tag_t sc_iot; bus_space_handle_t sc_ioh; int sc_pid; #if NGPIO > 0 struct gpio_chipset_tag gpio_chipset; gpio_pin_t pins[AT91PIO_NPINS]; #endif int irq; void *ih; struct intr_req ireq[AT91PIO_NPINS]; }; static int at91pio_match(device_t, cfdata_t, void *); static void at91pio_attach(device_t, device_t, void *); #if NGPIO > 0 static int at91piobus_print(void *, const char *); static int at91pio_pin_read(void *, int); static void at91pio_pin_write(void *, int, int); static void at91pio_pin_ctl(void *, int, int); #endif static int at91pio_search(device_t, cfdata_t, const int *, void *); static int at91pio_print(void *, const char *); static int at91pio_intr(void* arg); CFATTACH_DECL_NEW(at91pio, sizeof(struct at91pio_softc), at91pio_match, at91pio_attach, NULL, NULL); static struct at91pio_softc *at91pio_softc[AT91_PIO_COUNT]; struct at91pio_softc *at91pio_sc(at91pio_port port) { if (port < AT91_PIO_COUNT) return at91pio_softc[port]; return NULL; } static int at91pio_match(device_t parent, cfdata_t match, void *aux) { if (strcmp(match->cf_name, "at91pio") == 0) return 2; return 0; } static void at91pio_attach(device_t parent, device_t self, void *aux) { struct at91pio_softc *sc = device_private(self); struct at91bus_attach_args *sa = aux; #if NGPIO > 0 struct gpiobus_attach_args gba; uint32_t psr, osr, pin; int j, n; #endif printf("\n"); sc->sc_iot = sa->sa_iot; sc->sc_pid = sa->sa_pid; if (bus_space_map(sa->sa_iot, sa->sa_addr, sa->sa_size, 0, &sc->sc_ioh)){ printf("%s: Cannot map registers", device_xname(self)); return; } /* save descriptor: */ at91pio_port p = at91_pio_port(sa->sa_pid); if (p < AT91_PIO_COUNT && !at91pio_softc[p]) at91pio_softc[p] = sc; /* make sure peripheral is enabled: */ at91_peripheral_clock(sc->sc_pid, 1); /* initialize ports (disable interrupts) */ PIO_WRITE(sc, PIO_IDR, -1); #if NGPIO > 0 /* initialize and attach gpio(4) */ psr = PIO_READ(sc, PIO_PSR); // only ports osr = PIO_READ(sc, PIO_OSR); pin = PIO_READ(sc, PIO_PDSR); psr &= ~at91_gpio_mask(sc->sc_pid); for (j = n = 0; j < AT91PIO_NPINS; j++) { sc->pins[n].pin_num = j; if (psr & (1 << j)) sc->pins[n].pin_caps = (GPIO_PIN_INPUT | GPIO_PIN_OUTPUT | GPIO_PIN_OPENDRAIN // @@@ not all pins | GPIO_PIN_PUSHPULL | GPIO_PIN_PULLUP); else sc->pins[n].pin_caps = 0; if (osr & (1 << j)) sc->pins[n].pin_flags = GPIO_PIN_OUTPUT; else sc->pins[n].pin_flags = GPIO_PIN_INPUT; if (pin & (1 << j)) sc->pins[n].pin_state = GPIO_PIN_HIGH; else sc->pins[n].pin_state = GPIO_PIN_LOW; n++; } sc->gpio_chipset.gp_cookie = sc; sc->gpio_chipset.gp_pin_read = at91pio_pin_read; sc->gpio_chipset.gp_pin_write = at91pio_pin_write; sc->gpio_chipset.gp_pin_ctl = at91pio_pin_ctl; gba.gba_gc = &sc->gpio_chipset; gba.gba_pins = sc->pins; gba.gba_npins = n; config_found_ia(self, "gpiobus", &gba, at91piobus_print); #endif /* attach device */ config_search_ia(at91pio_search, self, "at91pio", at91pio_print); } #if NGPIO > 0 static int at91piobus_print(void *aux, const char *name) { struct gpiobus_attach_args *gba = aux; struct at91pio_softc *sc = (struct at91pio_softc *)gba->gba_gc->gp_cookie; gpiobus_print(aux, name); aprint_normal(": port %s (mask %08"PRIX32")", at91_peripheral_name(sc->sc_pid), at91_gpio_mask(sc->sc_pid)); return (UNCONF); } #endif static int at91pio_search(device_t parent, cfdata_t cf, const int *ldesc, void *aux) { struct at91pio_softc *sc = device_private(parent); struct at91pio_attach_args paa; paa.paa_sc = sc; paa.paa_iot = sc->sc_iot; paa.paa_pid = cf->cf_loc[AT91PIOCF_PID]; paa.paa_bit = cf->cf_loc[AT91PIOCF_BIT]; if (config_match(parent, cf, &paa) > 0) config_attach(parent, cf, &paa, at91pio_print); return 0; } static int at91pio_print(void *aux, const char *name) { struct at91pio_attach_args *paa = aux; aprint_normal(":"); if (paa->paa_pid > -1) aprint_normal(" port %s", at91_peripheral_name(paa->paa_pid)); if (paa->paa_bit > -1) aprint_normal(" bit %d", paa->paa_bit); return (UNCONF); } int at91pio_read(struct at91pio_softc *sc, int bit) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif return (PIO_READ(sc, PIO_PDSR) >> bit) & 1; } void at91pio_set(struct at91pio_softc *sc, int bit) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif PIO_WRITE(sc, PIO_SODR, (1U << bit)); } void at91pio_clear(struct at91pio_softc *sc, int bit) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif PIO_WRITE(sc, PIO_CODR, (1U << bit)); } void at91pio_in(struct at91pio_softc *sc, int bit) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif PIO_WRITE(sc, PIO_ODR, (1U << bit)); } void at91pio_out(struct at91pio_softc *sc, int bit) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif PIO_WRITE(sc, PIO_OER, (1U << bit)); } void at91pio_per(struct at91pio_softc *sc, int bit, int perab) { #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif switch (perab) { case -1: PIO_WRITE(sc, PIO_PER, (1U << bit)); break; case 0: PIO_WRITE(sc, PIO_ASR, (1U << bit)); PIO_WRITE(sc, PIO_PDR, (1U << bit)); break; case 1: PIO_WRITE(sc, PIO_BSR, (1U << bit)); PIO_WRITE(sc, PIO_PDR, (1U << bit)); break; default: panic("%s: perab is invalid: %i", __FUNCTION__, perab); break; } } void * at91pio_intr_establish(struct at91pio_softc *sc, int bit, int ipl, int (*ireq_func)(void *), void *arg) { struct intr_req *ireq; DPRINTFN(1, ("at91pio_intr_establish: port=%s, bit=%d\n", at91_peripheral_name(sc->sc_pid), bit)); if (bit < 0 || bit >= AT91PIO_NPINS) return 0; ireq = &sc->ireq[bit]; if (ireq->ireq_func) /* already used */ return 0; ireq->ireq_func = ireq_func; ireq->ireq_arg = arg; ireq->ireq_ipl = ipl; PIO_WRITE(sc, PIO_IDR, (1U << bit)); /* disable interrupt for now */ at91pio_in(sc, bit); /* make sure pin is input */ #if NGPIO > 0 sc->pins[bit].pin_caps = 0; #endif #if 0 if (flag & EDGE_TRIGGER) at91pio_bit_set(sc, sc->xinttype1, bit); else /* LEVEL_SENSE */ at91pio_bit_clear(sc, sc->xinttype1, bit); if (flag & RISING_EDGE) /* or HIGH_LEVEL */ at91pio_bit_set(sc, sc->xinttype2, bit); else /* FALLING_EDGE or LOW_LEVEL */ at91pio_bit_clear(sc, sc->xinttype2, bit); if (flag & DEBOUNCE) PIO_WRITE(sc, PIO_IFER, (1U << bit)); else PIO_WRITE(sc, PIO_IFDR, (1U << bit)); #endif if (!sc->ih) { // use IPL_BIO because we want lowest possible priority as // we really don't know what priority is going to be used by // the caller.. this is not really optimal but tell me a // better way sc->ih = at91_intr_establish(sc->sc_pid, IPL_BIO, INTR_HIGH_LEVEL, at91pio_intr, sc); } //(void)PIO_READ(sc, PIO_ISR); // clear interrupts PIO_WRITE(sc, PIO_IER, (1U << bit)); // enable interrupt return sc->ih; } void at91pio_intr_disestablish(struct at91pio_softc *sc, int bit, void *cookie) { struct intr_req *ireq; int i; DPRINTFN(1, ("at91pio_intr_disestablish: port=%s, bit=%d\n", at91_peripheral_name(sc->sc_pid), bit)); if (bit < 0 || bit >= AT91PIO_NPINS) return; if (cookie != sc->ih) return; ireq = &sc->ireq[bit]; if (!ireq->ireq_func) return; PIO_WRITE(sc, PIO_IDR, (1U << bit)); ireq->ireq_func = 0; ireq->ireq_arg = 0; for (i = 0; i < AT91PIO_NPINS; i++) { if (sc->ireq[i].ireq_func) break; } if (i >= AT91PIO_NPINS) { at91_intr_disestablish(sc->ih); sc->ih = 0; } } static int at91pio_intr(void *arg) { struct at91pio_softc *sc = arg; int bit; uint32_t isr; isr = (PIO_READ(sc, PIO_ISR) & PIO_READ(sc, PIO_IMR)); if (!isr) return 0; do { bit = ffs(isr) - 1; isr &= ~(1U << bit); #ifdef DIAGNOSTIC if (bit < 0) panic("%s: isr is zero (0x%X)", __FUNCTION__, isr); #endif if (sc->ireq[bit].ireq_func) { int s = _splraise(sc->ireq[bit].ireq_ipl); (*sc->ireq[bit].ireq_func)(sc->ireq[bit].ireq_arg); splx(s); } } while (isr); return 1; } #if NGPIO > 0 static int at91pio_pin_read(void *arg, int pin) { struct at91pio_softc *sc = arg; pin %= AT91PIO_NPINS; if (!sc->pins[pin].pin_caps) return 0; /* EBUSY? */ return (PIO_READ(sc, PIO_PDSR) >> pin) & 1; } static void at91pio_pin_write(void *arg, int pin, int val) { struct at91pio_softc *sc = arg; pin %= AT91PIO_NPINS; if (!sc->pins[pin].pin_caps) return; if (val) PIO_WRITE(sc, PIO_SODR, (1U << pin)); else PIO_WRITE(sc, PIO_CODR, (1U << pin)); } static void at91pio_pin_ctl(void *arg, int pin, int flags) { struct at91pio_softc *sc = arg; pin %= AT91PIO_NPINS; if (!sc->pins[pin].pin_caps) return; if (flags & GPIO_PIN_INPUT) PIO_WRITE(sc, PIO_ODR, (1U << pin)); else if (flags & GPIO_PIN_OUTPUT) PIO_WRITE(sc, PIO_OER, (1U << pin)); if (flags & GPIO_PIN_OPENDRAIN) PIO_WRITE(sc, PIO_MDER, (1U << pin)); else if (flags & GPIO_PIN_PUSHPULL) PIO_WRITE(sc, PIO_MDDR, (1U << pin)); if (flags & GPIO_PIN_PULLUP) PIO_WRITE(sc, PIO_PUER, (1U << pin)); else PIO_WRITE(sc, PIO_PUDR, (1U << pin)); } #endif