417afa7caa
I don't know why we gpio(4) didn't do that before, but it seems harmless. Add naive power handling: when a gpio(4) instance is resumed, write each pin's configuration flags and output state from the softc to the hardware using gpiobus_pin_ctl() and gpiobus_pin_write(). While suspended, fail every ioctl() except for GPIOINFO with EBUSY.
401 lines
8.6 KiB
C
401 lines
8.6 KiB
C
/* $NetBSD: gpio.c,v 1.13 2007/12/14 01:49:23 dyoung Exp $ */
|
|
/* $OpenBSD: gpio.c,v 1.6 2006/01/14 12:33:49 grange Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2004, 2006 Alexander Yurchenko <grange@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/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: gpio.c,v 1.13 2007/12/14 01:49:23 dyoung Exp $");
|
|
|
|
/*
|
|
* General Purpose Input/Output framework.
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/device.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/gpio.h>
|
|
#include <sys/vnode.h>
|
|
|
|
#include <dev/gpio/gpiovar.h>
|
|
|
|
#include "locators.h"
|
|
|
|
struct gpio_softc {
|
|
struct device sc_dev;
|
|
|
|
gpio_chipset_tag_t sc_gc; /* our GPIO controller */
|
|
gpio_pin_t *sc_pins; /* pins array */
|
|
int sc_npins; /* total number of pins */
|
|
|
|
int sc_opened;
|
|
int sc_dying;
|
|
};
|
|
|
|
int gpio_match(struct device *, struct cfdata *, void *);
|
|
void gpio_attach(struct device *, struct device *, void *);
|
|
bool gpio_resume(device_t);
|
|
int gpio_detach(struct device *, int);
|
|
int gpio_activate(struct device *, enum devact);
|
|
int gpio_search(struct device *, struct cfdata *, const int *, void *);
|
|
int gpio_print(void *, const char *);
|
|
|
|
CFATTACH_DECL(gpio, sizeof(struct gpio_softc),
|
|
gpio_match, gpio_attach, gpio_detach, gpio_activate);
|
|
|
|
dev_type_open(gpioopen);
|
|
dev_type_close(gpioclose);
|
|
dev_type_ioctl(gpioioctl);
|
|
|
|
const struct cdevsw gpio_cdevsw = {
|
|
gpioopen, gpioclose, noread, nowrite, gpioioctl,
|
|
nostop, notty, nopoll, nommap, nokqfilter, D_OTHER,
|
|
};
|
|
|
|
extern struct cfdriver gpio_cd;
|
|
|
|
int
|
|
gpio_match(struct device *parent, struct cfdata *cf,
|
|
void *aux)
|
|
{
|
|
|
|
return (1);
|
|
}
|
|
|
|
bool
|
|
gpio_resume(device_t self)
|
|
{
|
|
struct gpio_softc *sc = device_private(self);
|
|
int pin;
|
|
|
|
for (pin = 0; pin < sc->sc_npins; pin++) {
|
|
gpiobus_pin_ctl(sc->sc_gc, pin, sc->sc_pins[pin].pin_flags);
|
|
gpiobus_pin_write(sc->sc_gc, pin, sc->sc_pins[pin].pin_state);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void
|
|
gpio_attach(struct device *parent, struct device *self, void *aux)
|
|
{
|
|
struct gpio_softc *sc = device_private(self);
|
|
struct gpiobus_attach_args *gba = aux;
|
|
|
|
sc->sc_gc = gba->gba_gc;
|
|
sc->sc_pins = gba->gba_pins;
|
|
sc->sc_npins = gba->gba_npins;
|
|
|
|
printf(": %d pins\n", sc->sc_npins);
|
|
|
|
if (!pmf_device_register(self, NULL, gpio_resume))
|
|
aprint_error_dev(self, "couldn't establish power handler\n");
|
|
|
|
/*
|
|
* Attach all devices that can be connected to the GPIO pins
|
|
* described in the kernel configuration file.
|
|
*/
|
|
config_search_ia(gpio_search, self, "gpio", sc);
|
|
}
|
|
|
|
int
|
|
gpio_detach(struct device *self, int flags)
|
|
{
|
|
#if 0
|
|
int maj, mn;
|
|
|
|
/* Locate the major number */
|
|
for (maj = 0; maj < nchrdev; maj++)
|
|
if (cdevsw[maj].d_open == gpioopen)
|
|
break;
|
|
|
|
/* Nuke the vnodes for any open instances (calls close) */
|
|
mn = device_unit(self);
|
|
vdevgone(maj, mn, mn, VCHR);
|
|
#endif
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
gpio_activate(struct device *self, enum devact act)
|
|
{
|
|
struct gpio_softc *sc = device_private(self);
|
|
|
|
switch (act) {
|
|
case DVACT_ACTIVATE:
|
|
return (EOPNOTSUPP);
|
|
case DVACT_DEACTIVATE:
|
|
sc->sc_dying = 1;
|
|
break;
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
gpio_search(struct device *parent, struct cfdata *cf,
|
|
const int *ldesc, void *aux)
|
|
{
|
|
struct gpio_attach_args ga;
|
|
|
|
ga.ga_gpio = aux;
|
|
ga.ga_offset = cf->cf_loc[GPIOCF_OFFSET];
|
|
ga.ga_mask = cf->cf_loc[GPIOCF_MASK];
|
|
|
|
if (config_match(parent, cf, &ga) > 0)
|
|
config_attach(parent, cf, &ga, gpio_print);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
gpio_print(void *aux, const char *pnp)
|
|
{
|
|
struct gpio_attach_args *ga = aux;
|
|
int i;
|
|
|
|
printf(" pins");
|
|
for (i = 0; i < 32; i++)
|
|
if (ga->ga_mask & (1 << i))
|
|
printf(" %d", ga->ga_offset + i);
|
|
|
|
return (UNCONF);
|
|
}
|
|
|
|
int
|
|
gpiobus_print(void *aux, const char *pnp)
|
|
{
|
|
#if 0
|
|
struct gpiobus_attach_args *gba = aux;
|
|
#endif
|
|
if (pnp != NULL)
|
|
printf("%s at %s", "gpiobus", pnp);
|
|
|
|
return (UNCONF);
|
|
}
|
|
|
|
int
|
|
gpio_pin_map(void *gpio, int offset, u_int32_t mask, struct gpio_pinmap *map)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
int npins, pin, i;
|
|
|
|
npins = gpio_npins(mask);
|
|
if (npins > sc->sc_npins)
|
|
return (1);
|
|
|
|
for (npins = 0, i = 0; i < 32; i++)
|
|
if (mask & (1 << i)) {
|
|
pin = offset + i;
|
|
if (pin < 0 || pin >= sc->sc_npins)
|
|
return (1);
|
|
if (sc->sc_pins[pin].pin_mapped)
|
|
return (1);
|
|
sc->sc_pins[pin].pin_mapped = 1;
|
|
map->pm_map[npins++] = pin;
|
|
}
|
|
map->pm_size = npins;
|
|
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
gpio_pin_unmap(void *gpio, struct gpio_pinmap *map)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
int pin, i;
|
|
|
|
for (i = 0; i < map->pm_size; i++) {
|
|
pin = map->pm_map[i];
|
|
sc->sc_pins[pin].pin_mapped = 0;
|
|
}
|
|
}
|
|
|
|
int
|
|
gpio_pin_read(void *gpio, struct gpio_pinmap *map, int pin)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
|
|
return (gpiobus_pin_read(sc->sc_gc, map->pm_map[pin]));
|
|
}
|
|
|
|
void
|
|
gpio_pin_write(void *gpio, struct gpio_pinmap *map, int pin, int value)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
|
|
gpiobus_pin_write(sc->sc_gc, map->pm_map[pin], value);
|
|
sc->sc_pins[map->pm_map[pin]].pin_state = value;
|
|
}
|
|
|
|
void
|
|
gpio_pin_ctl(void *gpio, struct gpio_pinmap *map, int pin, int flags)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
|
|
return (gpiobus_pin_ctl(sc->sc_gc, map->pm_map[pin], flags));
|
|
}
|
|
|
|
int
|
|
gpio_pin_caps(void *gpio, struct gpio_pinmap *map, int pin)
|
|
{
|
|
struct gpio_softc *sc = gpio;
|
|
|
|
return (sc->sc_pins[map->pm_map[pin]].pin_caps);
|
|
}
|
|
|
|
int
|
|
gpio_npins(u_int32_t mask)
|
|
{
|
|
int npins, i;
|
|
|
|
for (npins = 0, i = 0; i < 32; i++)
|
|
if (mask & (1 << i))
|
|
npins++;
|
|
|
|
return (npins);
|
|
}
|
|
|
|
int
|
|
gpioopen(dev_t dev, int flag, int mode,
|
|
struct lwp *l)
|
|
{
|
|
struct gpio_softc *sc;
|
|
|
|
sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
|
|
if (sc == NULL)
|
|
return (ENXIO);
|
|
|
|
if (sc->sc_opened)
|
|
return (EBUSY);
|
|
sc->sc_opened = 1;
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
gpioclose(dev_t dev, int flag, int mode,
|
|
struct lwp *l)
|
|
{
|
|
struct gpio_softc *sc;
|
|
|
|
sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
|
|
sc->sc_opened = 0;
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
gpioioctl(dev_t dev, u_long cmd, void *data, int flag,
|
|
struct lwp *l)
|
|
{
|
|
struct gpio_softc *sc;
|
|
gpio_chipset_tag_t gc;
|
|
struct gpio_info *info;
|
|
struct gpio_pin_op *op;
|
|
struct gpio_pin_ctl *ctl;
|
|
int pin, value, flags;
|
|
|
|
sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
|
|
gc = sc->sc_gc;
|
|
|
|
if (cmd != GPIOINFO && !device_is_active(&sc->sc_dev))
|
|
return EBUSY;
|
|
|
|
switch (cmd) {
|
|
case GPIOINFO:
|
|
info = (struct gpio_info *)data;
|
|
|
|
info->gpio_npins = sc->sc_npins;
|
|
break;
|
|
case GPIOPINREAD:
|
|
op = (struct gpio_pin_op *)data;
|
|
|
|
pin = op->gp_pin;
|
|
if (pin < 0 || pin >= sc->sc_npins)
|
|
return (EINVAL);
|
|
|
|
/* return read value */
|
|
op->gp_value = gpiobus_pin_read(gc, pin);
|
|
break;
|
|
case GPIOPINWRITE:
|
|
op = (struct gpio_pin_op *)data;
|
|
|
|
pin = op->gp_pin;
|
|
if (pin < 0 || pin >= sc->sc_npins)
|
|
return (EINVAL);
|
|
if (sc->sc_pins[pin].pin_mapped)
|
|
return (EBUSY);
|
|
|
|
value = op->gp_value;
|
|
if (value != GPIO_PIN_LOW && value != GPIO_PIN_HIGH)
|
|
return (EINVAL);
|
|
|
|
gpiobus_pin_write(gc, pin, value);
|
|
/* return old value */
|
|
op->gp_value = sc->sc_pins[pin].pin_state;
|
|
/* update current value */
|
|
sc->sc_pins[pin].pin_state = value;
|
|
break;
|
|
case GPIOPINTOGGLE:
|
|
op = (struct gpio_pin_op *)data;
|
|
|
|
pin = op->gp_pin;
|
|
if (pin < 0 || pin >= sc->sc_npins)
|
|
return (EINVAL);
|
|
if (sc->sc_pins[pin].pin_mapped)
|
|
return (EBUSY);
|
|
|
|
value = (sc->sc_pins[pin].pin_state == GPIO_PIN_LOW ?
|
|
GPIO_PIN_HIGH : GPIO_PIN_LOW);
|
|
gpiobus_pin_write(gc, pin, value);
|
|
/* return old value */
|
|
op->gp_value = sc->sc_pins[pin].pin_state;
|
|
/* update current value */
|
|
sc->sc_pins[pin].pin_state = value;
|
|
break;
|
|
case GPIOPINCTL:
|
|
ctl = (struct gpio_pin_ctl *)data;
|
|
|
|
pin = ctl->gp_pin;
|
|
if (pin < 0 || pin >= sc->sc_npins)
|
|
return (EINVAL);
|
|
if (sc->sc_pins[pin].pin_mapped)
|
|
return (EBUSY);
|
|
|
|
flags = ctl->gp_flags;
|
|
/* check that the controller supports all requested flags */
|
|
if ((flags & sc->sc_pins[pin].pin_caps) != flags)
|
|
return (ENODEV);
|
|
|
|
ctl->gp_caps = sc->sc_pins[pin].pin_caps;
|
|
/* return old value */
|
|
ctl->gp_flags = sc->sc_pins[pin].pin_flags;
|
|
if (flags > 0) {
|
|
gpiobus_pin_ctl(gc, pin, flags);
|
|
/* update current value */
|
|
sc->sc_pins[pin].pin_flags = flags;
|
|
}
|
|
break;
|
|
default:
|
|
return (ENOTTY);
|
|
}
|
|
|
|
return (0);
|
|
}
|