1 1.5 thorpej /* $NetBSD: rmixl_gpio.c,v 1.5 2021/08/07 16:18:59 thorpej Exp $ */ 2 1.5 thorpej /* $NetBSD: rmixl_gpio.c,v 1.5 2021/08/07 16:18:59 thorpej Exp $ */ 3 1.1 cliff /*- 4 1.1 cliff * Copyright (c) 2007 The NetBSD Foundation, Inc. 5 1.1 cliff * All rights reserved. 6 1.1 cliff * 7 1.1 cliff * This code is derived from software contributed to The NetBSD Foundation 8 1.1 cliff * by Matt Thomas 9 1.1 cliff * 10 1.1 cliff * Redistribution and use in source and binary forms, with or without 11 1.1 cliff * modification, are permitted provided that the following conditions 12 1.1 cliff * are met: 13 1.1 cliff * 1. Redistributions of source code must retain the above copyright 14 1.1 cliff * notice, this list of conditions and the following disclaimer. 15 1.1 cliff * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 cliff * notice, this list of conditions and the following disclaimer in the 17 1.1 cliff * documentation and/or other materials provided with the distribution. 18 1.1 cliff * 19 1.1 cliff * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 cliff * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 cliff * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 cliff * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 cliff * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 cliff * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 cliff * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 cliff * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 cliff * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 cliff * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 cliff * POSSIBILITY OF SUCH DAMAGE. 30 1.1 cliff */ 31 1.1 cliff #include <sys/cdefs.h> 32 1.5 thorpej __KERNEL_RCSID(0, "$NetBSD: rmixl_gpio.c,v 1.5 2021/08/07 16:18:59 thorpej Exp $"); 33 1.1 cliff 34 1.1 cliff #define _INTR_PRIVATE 35 1.1 cliff 36 1.1 cliff #include "locators.h" 37 1.1 cliff #include "gpio.h" 38 1.1 cliff 39 1.1 cliff #include <sys/param.h> 40 1.3 matt #include <sys/bus.h> 41 1.3 matt #include <sys/cpu.h> 42 1.1 cliff #include <sys/evcnt.h> 43 1.3 matt #include <sys/intr.h> 44 1.1 cliff #include <sys/atomic.h> 45 1.1 cliff 46 1.1 cliff #include <uvm/uvm_extern.h> 47 1.1 cliff 48 1.1 cliff #include <mips/rmi/rmixlreg.h> 49 1.1 cliff #include <mips/rmi/rmixlvar.h> 50 1.1 cliff #include <mips/rmi/rmixl_obiovar.h> 51 1.1 cliff 52 1.1 cliff #if NGPIO > 0 53 1.1 cliff #include <sys/gpio.h> 54 1.1 cliff #include <dev/gpio/gpiovar.h> 55 1.1 cliff #endif 56 1.1 cliff 57 1.1 cliff struct gpio_softc { 58 1.1 cliff device_t gpio_dev; 59 1.1 cliff void *gpio_ih; 60 1.1 cliff bus_space_tag_t gpio_memt; 61 1.1 cliff bus_space_handle_t gpio_memh; 62 1.1 cliff uint32_t gpio_enable_mask; 63 1.1 cliff uint32_t gpio_edge_mask; 64 1.1 cliff uint32_t gpio_edge_falling_mask; 65 1.1 cliff uint32_t gpio_edge_rising_mask; 66 1.1 cliff uint32_t gpio_level_mask; 67 1.1 cliff uint32_t gpio_level_hi_mask; 68 1.1 cliff uint32_t gpio_level_lo_mask; 69 1.1 cliff uint32_t gpio_inuse_mask; 70 1.1 cliff #if NGPIO > 0 71 1.1 cliff struct gpio_chipset_tag gpio_chipset; 72 1.1 cliff gpio_pin_t gpio_pins[RMIXL_GPIO_NSIGNALS]; 73 1.1 cliff #endif 74 1.1 cliff }; 75 1.1 cliff 76 1.1 cliff #define GPIO_READ(gpio, reg) \ 77 1.1 cliff bus_space_read_4((gpio)->gpio_memt, (gpio)->gpio_memh, (reg)) 78 1.1 cliff #define GPIO_WRITE(gpio, reg, val) \ 79 1.1 cliff bus_space_write_4((gpio)->gpio_memt, (gpio)->gpio_memh, (reg), (val)) 80 1.1 cliff 81 1.1 cliff static int gpio_match(device_t, cfdata_t, void *); 82 1.1 cliff static void gpio_attach(device_t, device_t, void *); 83 1.1 cliff 84 1.1 cliff CFATTACH_DECL_NEW(rmixl_gpio, 85 1.1 cliff sizeof(struct gpio_softc), 86 1.1 cliff gpio_match, gpio_attach, 87 1.1 cliff NULL, NULL); 88 1.1 cliff 89 1.1 cliff #if NGPIO > 0 90 1.1 cliff 91 1.1 cliff static int 92 1.1 cliff rmixl_gpio_pin_read(void *arg, int pin) 93 1.1 cliff { 94 1.1 cliff struct gpio_softc * const gpio = arg; 95 1.1 cliff 96 1.1 cliff KASSERT(((1 << pin) & RMIXL_GPIO_INPUT_MASK) != 0); 97 1.1 cliff return (GPIO_READ(gpio, RMIXL_GPIO_INPUT) >> pin) & 1; 98 1.1 cliff } 99 1.1 cliff 100 1.1 cliff static void 101 1.1 cliff rmixl_gpio_pin_write(void *arg, int pin, int value) 102 1.1 cliff { 103 1.1 cliff struct gpio_softc * const gpio = arg; 104 1.1 cliff uint32_t mask = 1 << pin; 105 1.1 cliff uint32_t old, new; 106 1.1 cliff 107 1.1 cliff KASSERT(((1 << pin) & RMIXL_GPIO_OUTPUT_MASK) != 0); 108 1.1 cliff 109 1.1 cliff old = GPIO_READ(gpio, RMIXL_GPIO_OUTPUT); 110 1.1 cliff if (value) 111 1.1 cliff new = old | mask; 112 1.1 cliff else 113 1.1 cliff new = old & ~mask; 114 1.1 cliff 115 1.1 cliff if (old != new) 116 1.1 cliff GPIO_WRITE(gpio, RMIXL_GPIO_OUTPUT, new); 117 1.1 cliff } 118 1.1 cliff 119 1.1 cliff static void 120 1.1 cliff rmixl_gpio_pin_ctl(void *arg, int pin, int flags) 121 1.1 cliff { 122 1.1 cliff struct gpio_softc * const gpio = arg; 123 1.1 cliff uint32_t mask = 1 << pin; 124 1.1 cliff uint32_t old, new; 125 1.1 cliff 126 1.1 cliff KASSERT((flags & (GPIO_PIN_INPUT|GPIO_PIN_OUTPUT)) != (GPIO_PIN_INPUT|GPIO_PIN_OUTPUT)); 127 1.1 cliff old = GPIO_READ(gpio, RMIXL_GPIO_IO_DIR); 128 1.1 cliff new = old; 129 1.1 cliff switch (flags & (GPIO_PIN_INPUT|GPIO_PIN_OUTPUT)) { 130 1.1 cliff case GPIO_PIN_INPUT: new &= ~mask; break; 131 1.1 cliff case GPIO_PIN_OUTPUT: new |= mask; break; 132 1.1 cliff default: return; 133 1.1 cliff } 134 1.1 cliff if (old != new) 135 1.1 cliff GPIO_WRITE(gpio, RMIXL_GPIO_IO_DIR, new); 136 1.1 cliff } 137 1.1 cliff 138 1.1 cliff static void 139 1.1 cliff gpio_defer(device_t self) 140 1.1 cliff { 141 1.1 cliff struct gpio_softc * const gpio = device_private(self); 142 1.1 cliff struct gpio_chipset_tag * const gp = &gpio->gpio_chipset; 143 1.1 cliff struct gpiobus_attach_args gba; 144 1.1 cliff gpio_pin_t *pins; 145 1.1 cliff uint32_t mask, dir, valueout, valuein; 146 1.1 cliff int pin; 147 1.1 cliff 148 1.1 cliff gp->gp_cookie = gpio; 149 1.1 cliff gp->gp_pin_read = rmixl_gpio_pin_read; 150 1.1 cliff gp->gp_pin_write = rmixl_gpio_pin_write; 151 1.1 cliff gp->gp_pin_ctl = rmixl_gpio_pin_ctl; 152 1.1 cliff 153 1.1 cliff gba.gba_gc = gp; 154 1.1 cliff gba.gba_pins = gpio->gpio_pins; 155 1.1 cliff gba.gba_npins = __arraycount(gpio->gpio_pins); 156 1.1 cliff 157 1.1 cliff dir = GPIO_READ(gpio, RMIXL_GPIO_IO_DIR); 158 1.1 cliff valueout = GPIO_READ(gpio, RMIXL_GPIO_OUTPUT); 159 1.1 cliff valuein = GPIO_READ(gpio, RMIXL_GPIO_INPUT); 160 1.1 cliff for (pin = 0, mask = 1, pins = gpio->gpio_pins; 161 1.1 cliff pin < RMIXL_GPIO_NSIGNALS; pin++, mask <<= 1, pins++) { 162 1.1 cliff pins->pin_num = pin; 163 1.1 cliff if (gpio->gpio_inuse_mask & mask) { 164 1.1 cliff if ((gpio->gpio_inuse_mask & RMIXL_GPIO_INPUT_MASK)) 165 1.1 cliff pins->pin_caps = GPIO_PIN_INPUT; 166 1.1 cliff } else { 167 1.1 cliff int caps = 0; 168 1.1 cliff if ((gpio->gpio_inuse_mask & RMIXL_GPIO_INPUT_MASK)) 169 1.1 cliff caps |= GPIO_PIN_INPUT; 170 1.1 cliff if ((gpio->gpio_inuse_mask & RMIXL_GPIO_OUTPUT_MASK)) 171 1.1 cliff caps |= GPIO_PIN_INPUT; 172 1.1 cliff pins->pin_caps = caps; 173 1.1 cliff } 174 1.1 cliff pins->pin_flags = 175 1.1 cliff (dir & mask) ? GPIO_PIN_OUTPUT : GPIO_PIN_INPUT; 176 1.1 cliff pins->pin_state = 177 1.1 cliff (((dir & mask) ? valueout : valuein) & mask) 178 1.1 cliff ? GPIO_PIN_LOW 179 1.1 cliff : GPIO_PIN_HIGH; 180 1.1 cliff } 181 1.1 cliff 182 1.5 thorpej config_found(self, &gba, gpiobus_print, CFARGS_NONE); 183 1.1 cliff } 184 1.1 cliff #endif /* NGPIO > 0 */ 185 1.1 cliff 186 1.1 cliff int 187 1.1 cliff gpio_match(device_t parent, cfdata_t cfdata, void *aux) 188 1.1 cliff { 189 1.1 cliff struct obio_attach_args *oa = aux; 190 1.1 cliff 191 1.1 cliff if (oa->obio_addr == RMIXL_IO_DEV_GPIO) /* XXX XLS */ 192 1.1 cliff return 1; 193 1.1 cliff 194 1.1 cliff return 0; 195 1.1 cliff } 196 1.1 cliff 197 1.1 cliff void 198 1.1 cliff gpio_attach(device_t parent, device_t self, void *aux) 199 1.1 cliff { 200 1.1 cliff struct obio_attach_args * const oa = aux; 201 1.1 cliff struct gpio_softc * const gpio = device_private(self); 202 1.1 cliff int error; 203 1.1 cliff 204 1.1 cliff gpio->gpio_dev = self; 205 1.1 cliff 206 1.1 cliff if (oa->obio_intr == OBIOCF_INTR_DEFAULT) 207 1.1 cliff panic("\n%s: no intr assigned", device_xname(self)); 208 1.1 cliff 209 1.1 cliff if (oa->obio_size == OBIOCF_SIZE_DEFAULT) 210 1.1 cliff oa->obio_size = 0x1000; 211 1.1 cliff 212 1.1 cliff gpio->gpio_memt = oa->obio_eb_bst; 213 1.1 cliff error = bus_space_map(oa->obio_eb_bst, oa->obio_addr, oa->obio_size, 214 1.1 cliff 0, &gpio->gpio_memh); 215 1.1 cliff 216 1.1 cliff if (error) { 217 1.1 cliff aprint_error(": failed to map register %#" 218 1.1 cliff PRIxBUSSIZE "@%#" PRIxBUSADDR ": %d\n", 219 1.1 cliff oa->obio_size, oa->obio_addr, error); 220 1.1 cliff return; 221 1.1 cliff } 222 1.1 cliff 223 1.1 cliff #ifdef NOTYET 224 1.1 cliff if (oa->obio_intr != OBIOCF_INTR_DEFAULT) { 225 1.1 cliff gpio->gpio_ih = intr_establish(oa->obio_intr, 226 1.1 cliff IPL_HIGH, IST_LEVEL, pic_handle_intr, &gpio->gpio_pic); 227 1.1 cliff KASSERT(gpio->gpio_ih != NULL); 228 1.1 cliff aprint_normal(", intr %d", oa->obio_intr); 229 1.1 cliff } 230 1.1 cliff #endif 231 1.1 cliff aprint_normal("\n"); 232 1.1 cliff #if NGPIO > 0 233 1.1 cliff config_interrupts(self, gpio_defer); 234 1.1 cliff #endif 235 1.1 cliff } 236