NetBSD/sys/dev/gpio/gpio.c
dyoung 417afa7caa Bug fix: record the new state (high/low) set by gpio_pin_write().
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.
2007-12-14 01:49:23 +00:00

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);
}