1 /*- 2 * Copyright (c) 2009 Oleksandr Tymoshenko <gonzo@freebsd.org> 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24 * SUCH DAMAGE. 25 */ 26 27 #include <sys/cdefs.h> 28 __FBSDID("$FreeBSD$"); 29 30 #include <sys/param.h> 31 #include <sys/systm.h> 32 #include <sys/bus.h> 33 #include <sys/gpio.h> 34 #include <sys/kernel.h> 35 #include <sys/malloc.h> 36 #include <sys/module.h> 37 38 #include <dev/gpio/gpiobusvar.h> 39 40 #include "gpiobus_if.h" 41 42 #undef GPIOBUS_DEBUG 43 #ifdef GPIOBUS_DEBUG 44 #define dprintf printf 45 #else 46 #define dprintf(x, arg...) 47 #endif 48 49 static void gpiobus_print_pins(struct gpiobus_ivar *, char *, size_t); 50 static int gpiobus_parse_pins(struct gpiobus_softc *, device_t, int); 51 static int gpiobus_probe(device_t); 52 static int gpiobus_attach(device_t); 53 static int gpiobus_detach(device_t); 54 static int gpiobus_suspend(device_t); 55 static int gpiobus_resume(device_t); 56 static int gpiobus_print_child(device_t, device_t); 57 static int gpiobus_child_location_str(device_t, device_t, char *, size_t); 58 static int gpiobus_child_pnpinfo_str(device_t, device_t, char *, size_t); 59 static device_t gpiobus_add_child(device_t, u_int, const char *, int); 60 static void gpiobus_hinted_child(device_t, const char *, int); 61 62 /* 63 * GPIOBUS interface 64 */ 65 static int gpiobus_acquire_bus(device_t, device_t, int); 66 static void gpiobus_release_bus(device_t, device_t); 67 static int gpiobus_pin_setflags(device_t, device_t, uint32_t, uint32_t); 68 static int gpiobus_pin_getflags(device_t, device_t, uint32_t, uint32_t*); 69 static int gpiobus_pin_getcaps(device_t, device_t, uint32_t, uint32_t*); 70 static int gpiobus_pin_set(device_t, device_t, uint32_t, unsigned int); 71 static int gpiobus_pin_get(device_t, device_t, uint32_t, unsigned int*); 72 static int gpiobus_pin_toggle(device_t, device_t, uint32_t); 73 74 int 75 gpio_check_flags(uint32_t caps, uint32_t flags) 76 { 77 78 /* Check for unwanted flags. */ 79 if ((flags & caps) == 0 || (flags & caps) != flags) 80 return (EINVAL); 81 /* Cannot mix input/output together. */ 82 if (flags & GPIO_PIN_INPUT && flags & GPIO_PIN_OUTPUT) 83 return (EINVAL); 84 /* Cannot mix pull-up/pull-down together. */ 85 if (flags & GPIO_PIN_PULLUP && flags & GPIO_PIN_PULLDOWN) 86 return (EINVAL); 87 88 return (0); 89 } 90 91 static void 92 gpiobus_print_pins(struct gpiobus_ivar *devi, char *buf, size_t buflen) 93 { 94 char tmp[128]; 95 int i, range_start, range_stop, need_coma; 96 97 if (devi->npins == 0) 98 return; 99 100 need_coma = 0; 101 range_start = range_stop = devi->pins[0]; 102 for (i = 1; i < devi->npins; i++) { 103 if (devi->pins[i] != (range_stop + 1)) { 104 if (need_coma) 105 strlcat(buf, ",", buflen); 106 memset(tmp, 0, sizeof(tmp)); 107 if (range_start != range_stop) 108 snprintf(tmp, sizeof(tmp) - 1, "%d-%d", 109 range_start, range_stop); 110 else 111 snprintf(tmp, sizeof(tmp) - 1, "%d", 112 range_start); 113 strlcat(buf, tmp, buflen); 114 115 range_start = range_stop = devi->pins[i]; 116 need_coma = 1; 117 } 118 else 119 range_stop++; 120 } 121 122 if (need_coma) 123 strlcat(buf, ",", buflen); 124 memset(tmp, 0, sizeof(tmp)); 125 if (range_start != range_stop) 126 snprintf(tmp, sizeof(tmp) - 1, "%d-%d", 127 range_start, range_stop); 128 else 129 snprintf(tmp, sizeof(tmp) - 1, "%d", 130 range_start); 131 strlcat(buf, tmp, buflen); 132 } 133 134 device_t 135 gpiobus_attach_bus(device_t dev) 136 { 137 device_t busdev; 138 139 busdev = device_add_child(dev, "gpiobus", -1); 140 if (busdev == NULL) 141 return (NULL); 142 if (device_add_child(dev, "gpioc", -1) == NULL) { 143 device_delete_child(dev, busdev); 144 return (NULL); 145 } 146 #ifdef FDT 147 ofw_gpiobus_register_provider(dev); 148 #endif 149 bus_generic_attach(dev); 150 151 return (busdev); 152 } 153 154 int 155 gpiobus_detach_bus(device_t dev) 156 { 157 158 #ifdef FDT 159 ofw_gpiobus_unregister_provider(dev); 160 #endif 161 162 return (bus_generic_detach(dev)); 163 } 164 165 int 166 gpiobus_init_softc(device_t dev) 167 { 168 struct gpiobus_softc *sc; 169 170 sc = GPIOBUS_SOFTC(dev); 171 sc->sc_busdev = dev; 172 sc->sc_dev = device_get_parent(dev); 173 sc->sc_intr_rman.rm_type = RMAN_ARRAY; 174 sc->sc_intr_rman.rm_descr = "GPIO Interrupts"; 175 if (rman_init(&sc->sc_intr_rman) != 0 || 176 rman_manage_region(&sc->sc_intr_rman, 0, ~0) != 0) 177 panic("%s: failed to set up rman.", __func__); 178 179 if (GPIO_PIN_MAX(sc->sc_dev, &sc->sc_npins) != 0) 180 return (ENXIO); 181 182 KASSERT(sc->sc_npins != 0, ("GPIO device with no pins")); 183 184 /* Pins = GPIO_PIN_MAX() + 1 */ 185 sc->sc_npins++; 186 187 sc->sc_pins_mapped = malloc(sizeof(int) * sc->sc_npins, M_DEVBUF, 188 M_NOWAIT | M_ZERO); 189 if (sc->sc_pins_mapped == NULL) 190 return (ENOMEM); 191 192 /* Initialize the bus lock. */ 193 GPIOBUS_LOCK_INIT(sc); 194 195 return (0); 196 } 197 198 static int 199 gpiobus_parse_pins(struct gpiobus_softc *sc, device_t child, int mask) 200 { 201 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 202 int i, npins; 203 204 npins = 0; 205 for (i = 0; i < 32; i++) { 206 if (mask & (1 << i)) 207 npins++; 208 } 209 210 if (npins == 0) { 211 device_printf(child, "empty pin mask\n"); 212 return (EINVAL); 213 } 214 215 devi->npins = npins; 216 devi->pins = malloc(sizeof(uint32_t) * devi->npins, M_DEVBUF, 217 M_NOWAIT | M_ZERO); 218 219 if (!devi->pins) 220 return (ENOMEM); 221 222 npins = 0; 223 for (i = 0; i < 32; i++) { 224 225 if ((mask & (1 << i)) == 0) 226 continue; 227 228 if (i >= sc->sc_npins) { 229 device_printf(child, 230 "invalid pin %d, max: %d\n", i, sc->sc_npins - 1); 231 free(devi->pins, M_DEVBUF); 232 return (EINVAL); 233 } 234 235 devi->pins[npins++] = i; 236 /* 237 * Mark pin as mapped and give warning if it's already mapped 238 */ 239 if (sc->sc_pins_mapped[i]) { 240 device_printf(child, 241 "warning: pin %d is already mapped\n", i); 242 free(devi->pins, M_DEVBUF); 243 return (EINVAL); 244 } 245 sc->sc_pins_mapped[i] = 1; 246 } 247 248 return (0); 249 } 250 251 static int 252 gpiobus_probe(device_t dev) 253 { 254 device_set_desc(dev, "GPIO bus"); 255 256 return (BUS_PROBE_GENERIC); 257 } 258 259 static int 260 gpiobus_attach(device_t dev) 261 { 262 int err; 263 264 err = gpiobus_init_softc(dev); 265 if (err != 0) 266 return (err); 267 268 /* 269 * Get parent's pins and mark them as unmapped 270 */ 271 bus_generic_probe(dev); 272 bus_enumerate_hinted_children(dev); 273 274 return (bus_generic_attach(dev)); 275 } 276 277 /* 278 * Since this is not a self-enumerating bus, and since we always add 279 * children in attach, we have to always delete children here. 280 */ 281 static int 282 gpiobus_detach(device_t dev) 283 { 284 struct gpiobus_softc *sc; 285 struct gpiobus_ivar *devi; 286 device_t *devlist; 287 int i, err, ndevs; 288 289 sc = GPIOBUS_SOFTC(dev); 290 KASSERT(mtx_initialized(&sc->sc_mtx), 291 ("gpiobus mutex not initialized")); 292 GPIOBUS_LOCK_DESTROY(sc); 293 294 if ((err = bus_generic_detach(dev)) != 0) 295 return (err); 296 297 if ((err = device_get_children(dev, &devlist, &ndevs)) != 0) 298 return (err); 299 for (i = 0; i < ndevs; i++) { 300 device_delete_child(dev, devlist[i]); 301 devi = GPIOBUS_IVAR(devlist[i]); 302 if (devi->pins) { 303 free(devi->pins, M_DEVBUF); 304 devi->pins = NULL; 305 } 306 } 307 free(devlist, M_TEMP); 308 309 if (sc->sc_pins_mapped) { 310 free(sc->sc_pins_mapped, M_DEVBUF); 311 sc->sc_pins_mapped = NULL; 312 } 313 314 return (0); 315 } 316 317 static int 318 gpiobus_suspend(device_t dev) 319 { 320 321 return (bus_generic_suspend(dev)); 322 } 323 324 static int 325 gpiobus_resume(device_t dev) 326 { 327 328 return (bus_generic_resume(dev)); 329 } 330 331 static int 332 gpiobus_print_child(device_t dev, device_t child) 333 { 334 char pins[128]; 335 int retval = 0; 336 struct gpiobus_ivar *devi; 337 338 devi = GPIOBUS_IVAR(child); 339 memset(pins, 0, sizeof(pins)); 340 retval += bus_print_child_header(dev, child); 341 retval += printf(" at pin(s) "); 342 gpiobus_print_pins(devi, pins, sizeof(pins)); 343 retval += printf("%s", pins); 344 resource_list_print_type(&devi->rl, "irq", SYS_RES_IRQ, "%ld"); 345 retval += bus_print_child_footer(dev, child); 346 347 return (retval); 348 } 349 350 static int 351 gpiobus_child_location_str(device_t bus, device_t child, char *buf, 352 size_t buflen) 353 { 354 struct gpiobus_ivar *devi; 355 356 devi = GPIOBUS_IVAR(child); 357 strlcpy(buf, "pin(s)=", buflen); 358 gpiobus_print_pins(devi, buf, buflen); 359 360 return (0); 361 } 362 363 static int 364 gpiobus_child_pnpinfo_str(device_t bus, device_t child, char *buf, 365 size_t buflen) 366 { 367 368 *buf = '\0'; 369 return (0); 370 } 371 372 static device_t 373 gpiobus_add_child(device_t dev, u_int order, const char *name, int unit) 374 { 375 device_t child; 376 struct gpiobus_ivar *devi; 377 378 child = device_add_child_ordered(dev, order, name, unit); 379 if (child == NULL) 380 return (child); 381 devi = malloc(sizeof(struct gpiobus_ivar), M_DEVBUF, M_NOWAIT | M_ZERO); 382 if (devi == NULL) { 383 device_delete_child(dev, child); 384 return (0); 385 } 386 resource_list_init(&devi->rl); 387 device_set_ivars(child, devi); 388 389 return (child); 390 } 391 392 static void 393 gpiobus_hinted_child(device_t bus, const char *dname, int dunit) 394 { 395 struct gpiobus_softc *sc = GPIOBUS_SOFTC(bus); 396 struct gpiobus_ivar *devi; 397 device_t child; 398 int irq, pins; 399 400 child = BUS_ADD_CHILD(bus, 0, dname, dunit); 401 devi = GPIOBUS_IVAR(child); 402 resource_int_value(dname, dunit, "pins", &pins); 403 if (gpiobus_parse_pins(sc, child, pins)) 404 device_delete_child(bus, child); 405 if (resource_int_value(dname, dunit, "irq", &irq) == 0) { 406 if (bus_set_resource(child, SYS_RES_IRQ, 0, irq, 1) != 0) 407 device_printf(bus, 408 "warning: bus_set_resource() failed\n"); 409 } 410 } 411 412 static int 413 gpiobus_set_resource(device_t dev, device_t child, int type, int rid, 414 u_long start, u_long count) 415 { 416 struct gpiobus_ivar *devi; 417 struct resource_list_entry *rle; 418 419 dprintf("%s: entry (%p, %p, %d, %d, %p, %ld)\n", 420 __func__, dev, child, type, rid, (void *)(intptr_t)start, count); 421 devi = GPIOBUS_IVAR(child); 422 rle = resource_list_add(&devi->rl, type, rid, start, 423 start + count - 1, count); 424 if (rle == NULL) 425 return (ENXIO); 426 427 return (0); 428 } 429 430 static struct resource * 431 gpiobus_alloc_resource(device_t bus, device_t child, int type, int *rid, 432 u_long start, u_long end, u_long count, u_int flags) 433 { 434 struct gpiobus_softc *sc; 435 struct resource *rv; 436 struct resource_list *rl; 437 struct resource_list_entry *rle; 438 int isdefault; 439 440 if (type != SYS_RES_IRQ) 441 return (NULL); 442 isdefault = (start == 0UL && end == ~0UL && count == 1); 443 rle = NULL; 444 if (isdefault) { 445 rl = BUS_GET_RESOURCE_LIST(bus, child); 446 if (rl == NULL) 447 return (NULL); 448 rle = resource_list_find(rl, type, *rid); 449 if (rle == NULL) 450 return (NULL); 451 if (rle->res != NULL) 452 panic("%s: resource entry is busy", __func__); 453 start = rle->start; 454 count = rle->count; 455 end = rle->end; 456 } 457 sc = device_get_softc(bus); 458 rv = rman_reserve_resource(&sc->sc_intr_rman, start, end, count, flags, 459 child); 460 if (rv == NULL) 461 return (NULL); 462 rman_set_rid(rv, *rid); 463 if ((flags & RF_ACTIVE) != 0 && 464 bus_activate_resource(child, type, *rid, rv) != 0) { 465 rman_release_resource(rv); 466 return (NULL); 467 } 468 469 return (rv); 470 } 471 472 static int 473 gpiobus_release_resource(device_t bus __unused, device_t child, int type, 474 int rid, struct resource *r) 475 { 476 int error; 477 478 if (rman_get_flags(r) & RF_ACTIVE) { 479 error = bus_deactivate_resource(child, type, rid, r); 480 if (error) 481 return (error); 482 } 483 484 return (rman_release_resource(r)); 485 } 486 487 static struct resource_list * 488 gpiobus_get_resource_list(device_t bus __unused, device_t child) 489 { 490 struct gpiobus_ivar *ivar; 491 492 ivar = GPIOBUS_IVAR(child); 493 494 return (&ivar->rl); 495 } 496 497 static int 498 gpiobus_acquire_bus(device_t busdev, device_t child, int how) 499 { 500 struct gpiobus_softc *sc; 501 502 sc = device_get_softc(busdev); 503 GPIOBUS_ASSERT_UNLOCKED(sc); 504 GPIOBUS_LOCK(sc); 505 if (sc->sc_owner != NULL) { 506 if (how == GPIOBUS_DONTWAIT) { 507 GPIOBUS_UNLOCK(sc); 508 return (EWOULDBLOCK); 509 } 510 while (sc->sc_owner != NULL) 511 mtx_sleep(sc, &sc->sc_mtx, 0, "gpiobuswait", 0); 512 } 513 sc->sc_owner = child; 514 GPIOBUS_UNLOCK(sc); 515 516 return (0); 517 } 518 519 static void 520 gpiobus_release_bus(device_t busdev, device_t child) 521 { 522 struct gpiobus_softc *sc; 523 524 sc = device_get_softc(busdev); 525 GPIOBUS_ASSERT_UNLOCKED(sc); 526 GPIOBUS_LOCK(sc); 527 if (sc->sc_owner == NULL) 528 panic("gpiobus: releasing unowned bus."); 529 if (sc->sc_owner != child) 530 panic("gpiobus: you don't own the bus. game over."); 531 sc->sc_owner = NULL; 532 wakeup(sc); 533 GPIOBUS_UNLOCK(sc); 534 } 535 536 static int 537 gpiobus_pin_setflags(device_t dev, device_t child, uint32_t pin, 538 uint32_t flags) 539 { 540 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 541 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 542 uint32_t caps; 543 544 if (pin >= devi->npins) 545 return (EINVAL); 546 if (GPIO_PIN_GETCAPS(sc->sc_dev, devi->pins[pin], &caps) != 0) 547 return (EINVAL); 548 if (gpio_check_flags(caps, flags) != 0) 549 return (EINVAL); 550 551 return (GPIO_PIN_SETFLAGS(sc->sc_dev, devi->pins[pin], flags)); 552 } 553 554 static int 555 gpiobus_pin_getflags(device_t dev, device_t child, uint32_t pin, 556 uint32_t *flags) 557 { 558 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 559 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 560 561 if (pin >= devi->npins) 562 return (EINVAL); 563 564 return GPIO_PIN_GETFLAGS(sc->sc_dev, devi->pins[pin], flags); 565 } 566 567 static int 568 gpiobus_pin_getcaps(device_t dev, device_t child, uint32_t pin, 569 uint32_t *caps) 570 { 571 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 572 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 573 574 if (pin >= devi->npins) 575 return (EINVAL); 576 577 return GPIO_PIN_GETCAPS(sc->sc_dev, devi->pins[pin], caps); 578 } 579 580 static int 581 gpiobus_pin_set(device_t dev, device_t child, uint32_t pin, 582 unsigned int value) 583 { 584 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 585 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 586 587 if (pin >= devi->npins) 588 return (EINVAL); 589 590 return GPIO_PIN_SET(sc->sc_dev, devi->pins[pin], value); 591 } 592 593 static int 594 gpiobus_pin_get(device_t dev, device_t child, uint32_t pin, 595 unsigned int *value) 596 { 597 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 598 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 599 600 if (pin >= devi->npins) 601 return (EINVAL); 602 603 return GPIO_PIN_GET(sc->sc_dev, devi->pins[pin], value); 604 } 605 606 static int 607 gpiobus_pin_toggle(device_t dev, device_t child, uint32_t pin) 608 { 609 struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); 610 struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); 611 612 if (pin >= devi->npins) 613 return (EINVAL); 614 615 return GPIO_PIN_TOGGLE(sc->sc_dev, devi->pins[pin]); 616 } 617 618 static device_method_t gpiobus_methods[] = { 619 /* Device interface */ 620 DEVMETHOD(device_probe, gpiobus_probe), 621 DEVMETHOD(device_attach, gpiobus_attach), 622 DEVMETHOD(device_detach, gpiobus_detach), 623 DEVMETHOD(device_shutdown, bus_generic_shutdown), 624 DEVMETHOD(device_suspend, gpiobus_suspend), 625 DEVMETHOD(device_resume, gpiobus_resume), 626 627 /* Bus interface */ 628 DEVMETHOD(bus_setup_intr, bus_generic_setup_intr), 629 DEVMETHOD(bus_config_intr, bus_generic_config_intr), 630 DEVMETHOD(bus_teardown_intr, bus_generic_teardown_intr), 631 DEVMETHOD(bus_set_resource, gpiobus_set_resource), 632 DEVMETHOD(bus_alloc_resource, gpiobus_alloc_resource), 633 DEVMETHOD(bus_release_resource, gpiobus_release_resource), 634 DEVMETHOD(bus_activate_resource, bus_generic_activate_resource), 635 DEVMETHOD(bus_deactivate_resource, bus_generic_deactivate_resource), 636 DEVMETHOD(bus_get_resource_list, gpiobus_get_resource_list), 637 DEVMETHOD(bus_add_child, gpiobus_add_child), 638 DEVMETHOD(bus_print_child, gpiobus_print_child), 639 DEVMETHOD(bus_child_pnpinfo_str, gpiobus_child_pnpinfo_str), 640 DEVMETHOD(bus_child_location_str, gpiobus_child_location_str), 641 DEVMETHOD(bus_hinted_child, gpiobus_hinted_child), 642 643 /* GPIO protocol */ 644 DEVMETHOD(gpiobus_acquire_bus, gpiobus_acquire_bus), 645 DEVMETHOD(gpiobus_release_bus, gpiobus_release_bus), 646 DEVMETHOD(gpiobus_pin_getflags, gpiobus_pin_getflags), 647 DEVMETHOD(gpiobus_pin_getcaps, gpiobus_pin_getcaps), 648 DEVMETHOD(gpiobus_pin_setflags, gpiobus_pin_setflags), 649 DEVMETHOD(gpiobus_pin_get, gpiobus_pin_get), 650 DEVMETHOD(gpiobus_pin_set, gpiobus_pin_set), 651 DEVMETHOD(gpiobus_pin_toggle, gpiobus_pin_toggle), 652 653 DEVMETHOD_END 654 }; 655 656 driver_t gpiobus_driver = { 657 "gpiobus", 658 gpiobus_methods, 659 sizeof(struct gpiobus_softc) 660 }; 661 662 devclass_t gpiobus_devclass; 663 664 DRIVER_MODULE(gpiobus, gpio, gpiobus_driver, gpiobus_devclass, 0, 0); 665 MODULE_VERSION(gpiobus, 1); 666