1*894c7bfeSvisa /* $OpenBSD: gpio.c,v 1.17 2022/04/11 14:30:05 visa Exp $ */
2aca51624Sgrange
33615eacbSgrange /*
4bf4b8d0aSmbalmer * Copyright (c) 2008 Marc Balmer <mbalmer@openbsd.org>
5aca51624Sgrange * Copyright (c) 2004, 2006 Alexander Yurchenko <grange@openbsd.org>
63615eacbSgrange *
73615eacbSgrange * Permission to use, copy, modify, and distribute this software for any
83615eacbSgrange * purpose with or without fee is hereby granted, provided that the above
93615eacbSgrange * copyright notice and this permission notice appear in all copies.
103615eacbSgrange *
113615eacbSgrange * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
123615eacbSgrange * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
133615eacbSgrange * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
143615eacbSgrange * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
153615eacbSgrange * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
163615eacbSgrange * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
173615eacbSgrange * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
183615eacbSgrange */
193615eacbSgrange
203615eacbSgrange /*
211f410c2fSgrange * General Purpose Input/Output framework.
223615eacbSgrange */
233615eacbSgrange
243615eacbSgrange #include <sys/param.h>
253615eacbSgrange #include <sys/systm.h>
263615eacbSgrange #include <sys/conf.h>
273615eacbSgrange #include <sys/device.h>
286ec0d2e1Sgrange #include <sys/fcntl.h>
293615eacbSgrange #include <sys/ioctl.h>
303615eacbSgrange #include <sys/gpio.h>
3130b09c33Sgrange #include <sys/vnode.h>
32fca40c6aSmbalmer #include <sys/malloc.h>
33fca40c6aSmbalmer #include <sys/queue.h>
343615eacbSgrange
353615eacbSgrange #include <dev/gpio/gpiovar.h>
363615eacbSgrange
373615eacbSgrange struct gpio_softc {
383615eacbSgrange struct device sc_dev;
393615eacbSgrange
40a388edcaSmbalmer gpio_chipset_tag_t sc_gc; /* GPIO controller */
413615eacbSgrange gpio_pin_t *sc_pins; /* pins array */
42a388edcaSmbalmer int sc_npins; /* number of pins */
433615eacbSgrange
443615eacbSgrange int sc_opened;
45a388edcaSmbalmer LIST_HEAD(, gpio_dev) sc_devs; /* devices */
46a388edcaSmbalmer LIST_HEAD(, gpio_name) sc_names; /* named pins */
473615eacbSgrange };
483615eacbSgrange
493615eacbSgrange int gpio_match(struct device *, void *, void *);
50fca40c6aSmbalmer int gpio_submatch(struct device *, void *, void *);
513615eacbSgrange void gpio_attach(struct device *, struct device *, void *);
52a92078b1Sreyk int gpio_detach(struct device *, int);
533615eacbSgrange int gpio_search(struct device *, void *, void *);
543615eacbSgrange int gpio_print(void *, const char *);
55a388edcaSmbalmer int gpio_pinbyname(struct gpio_softc *, char *gp_name);
56*894c7bfeSvisa int gpio_ioctl(struct gpio_softc *, u_long, caddr_t, int);
573615eacbSgrange
58471aeecfSnaddy const struct cfattach gpio_ca = {
593615eacbSgrange sizeof (struct gpio_softc),
603615eacbSgrange gpio_match,
61a92078b1Sreyk gpio_attach,
62a92078b1Sreyk gpio_detach
633615eacbSgrange };
643615eacbSgrange
653615eacbSgrange struct cfdriver gpio_cd = {
663615eacbSgrange NULL, "gpio", DV_DULL
673615eacbSgrange };
683615eacbSgrange
693615eacbSgrange int
gpio_match(struct device * parent,void * match,void * aux)703615eacbSgrange gpio_match(struct device *parent, void *match, void *aux)
713615eacbSgrange {
723615eacbSgrange struct cfdata *cf = match;
733615eacbSgrange struct gpiobus_attach_args *gba = aux;
743615eacbSgrange
75fca40c6aSmbalmer return (strcmp(gba->gba_name, cf->cf_driver->cd_name) == 0);
76fca40c6aSmbalmer }
773615eacbSgrange
78fca40c6aSmbalmer int
gpio_submatch(struct device * parent,void * match,void * aux)79fca40c6aSmbalmer gpio_submatch(struct device *parent, void *match, void *aux)
80fca40c6aSmbalmer {
81fca40c6aSmbalmer struct cfdata *cf = match;
82fca40c6aSmbalmer struct gpio_attach_args *ga = aux;
83fca40c6aSmbalmer
849585af64Smiod if (strcmp(ga->ga_dvname, cf->cf_driver->cd_name) != 0)
859585af64Smiod return (0);
869585af64Smiod
879585af64Smiod return ((*cf->cf_attach->ca_match)(parent, match, aux));
883615eacbSgrange }
893615eacbSgrange
903615eacbSgrange void
gpio_attach(struct device * parent,struct device * self,void * aux)913615eacbSgrange gpio_attach(struct device *parent, struct device *self, void *aux)
923615eacbSgrange {
933615eacbSgrange struct gpio_softc *sc = (struct gpio_softc *)self;
943615eacbSgrange struct gpiobus_attach_args *gba = aux;
953615eacbSgrange
963615eacbSgrange sc->sc_gc = gba->gba_gc;
973615eacbSgrange sc->sc_pins = gba->gba_pins;
983615eacbSgrange sc->sc_npins = gba->gba_npins;
993615eacbSgrange
1003615eacbSgrange printf(": %d pins\n", sc->sc_npins);
1013615eacbSgrange
1023615eacbSgrange /*
1033615eacbSgrange * Attach all devices that can be connected to the GPIO pins
1043615eacbSgrange * described in the kernel configuration file.
1053615eacbSgrange */
106aca51624Sgrange config_search(gpio_search, self, sc);
1073615eacbSgrange }
1083615eacbSgrange
1093615eacbSgrange int
gpio_detach(struct device * self,int flags)11030b09c33Sgrange gpio_detach(struct device *self, int flags)
111a92078b1Sreyk {
11230b09c33Sgrange int maj, mn;
11330b09c33Sgrange
11430b09c33Sgrange /* Locate the major number */
11530b09c33Sgrange for (maj = 0; maj < nchrdev; maj++)
11630b09c33Sgrange if (cdevsw[maj].d_open == gpioopen)
11730b09c33Sgrange break;
11830b09c33Sgrange
11930b09c33Sgrange /* Nuke the vnodes for any open instances (calls close) */
12030b09c33Sgrange mn = self->dv_unit;
12130b09c33Sgrange vdevgone(maj, mn, mn, VCHR);
12230b09c33Sgrange
123a92078b1Sreyk return (0);
124a92078b1Sreyk }
125a92078b1Sreyk
126a92078b1Sreyk int
gpio_search(struct device * parent,void * arg,void * aux)1273615eacbSgrange gpio_search(struct device *parent, void *arg, void *aux)
1283615eacbSgrange {
1293615eacbSgrange struct cfdata *cf = arg;
1303615eacbSgrange struct gpio_attach_args ga;
1313615eacbSgrange
132aca51624Sgrange ga.ga_gpio = aux;
133aca51624Sgrange ga.ga_offset = cf->cf_loc[0];
1343615eacbSgrange ga.ga_mask = cf->cf_loc[1];
1357d434456Smatthieu ga.ga_flags = cf->cf_loc[2];
1363615eacbSgrange
1373615eacbSgrange if (cf->cf_attach->ca_match(parent, cf, &ga) > 0)
1383615eacbSgrange config_attach(parent, cf, &ga, gpio_print);
1393615eacbSgrange
1403615eacbSgrange return (0);
1413615eacbSgrange }
1423615eacbSgrange
1433615eacbSgrange int
gpio_print(void * aux,const char * pnp)1443615eacbSgrange gpio_print(void *aux, const char *pnp)
1453615eacbSgrange {
1463615eacbSgrange struct gpio_attach_args *ga = aux;
1473615eacbSgrange int i;
1483615eacbSgrange
1493615eacbSgrange printf(" pins");
1503615eacbSgrange for (i = 0; i < 32; i++)
1513615eacbSgrange if (ga->ga_mask & (1 << i))
152aca51624Sgrange printf(" %d", ga->ga_offset + i);
1533615eacbSgrange
1543615eacbSgrange return (UNCONF);
1553615eacbSgrange }
1563615eacbSgrange
1573615eacbSgrange int
gpiobus_print(void * aux,const char * pnp)1583615eacbSgrange gpiobus_print(void *aux, const char *pnp)
1593615eacbSgrange {
1603615eacbSgrange struct gpiobus_attach_args *gba = aux;
1613615eacbSgrange
1623615eacbSgrange if (pnp != NULL)
1633615eacbSgrange printf("%s at %s", gba->gba_name, pnp);
1643615eacbSgrange
1653615eacbSgrange return (UNCONF);
1663615eacbSgrange }
1673615eacbSgrange
1683615eacbSgrange int
gpio_pin_map(void * gpio,int offset,u_int32_t mask,struct gpio_pinmap * map)169aca51624Sgrange gpio_pin_map(void *gpio, int offset, u_int32_t mask, struct gpio_pinmap *map)
170aca51624Sgrange {
171aca51624Sgrange struct gpio_softc *sc = gpio;
172aca51624Sgrange int npins, pin, i;
173aca51624Sgrange
174aca51624Sgrange npins = gpio_npins(mask);
175aca51624Sgrange if (npins > sc->sc_npins)
176aca51624Sgrange return (1);
177aca51624Sgrange
178aca51624Sgrange for (npins = 0, i = 0; i < 32; i++)
179aca51624Sgrange if (mask & (1 << i)) {
180aca51624Sgrange pin = offset + i;
181aca51624Sgrange if (pin < 0 || pin >= sc->sc_npins)
182aca51624Sgrange return (1);
183aca51624Sgrange if (sc->sc_pins[pin].pin_mapped)
184aca51624Sgrange return (1);
185aca51624Sgrange sc->sc_pins[pin].pin_mapped = 1;
186aca51624Sgrange map->pm_map[npins++] = pin;
187aca51624Sgrange }
188aca51624Sgrange map->pm_size = npins;
189aca51624Sgrange
190aca51624Sgrange return (0);
191aca51624Sgrange }
192aca51624Sgrange
193aca51624Sgrange void
gpio_pin_unmap(void * gpio,struct gpio_pinmap * map)194aca51624Sgrange gpio_pin_unmap(void *gpio, struct gpio_pinmap *map)
195aca51624Sgrange {
196aca51624Sgrange struct gpio_softc *sc = gpio;
197aca51624Sgrange int pin, i;
198aca51624Sgrange
199aca51624Sgrange for (i = 0; i < map->pm_size; i++) {
200aca51624Sgrange pin = map->pm_map[i];
201aca51624Sgrange sc->sc_pins[pin].pin_mapped = 0;
202aca51624Sgrange }
203aca51624Sgrange }
204aca51624Sgrange
205aca51624Sgrange int
gpio_pin_read(void * gpio,struct gpio_pinmap * map,int pin)206aca51624Sgrange gpio_pin_read(void *gpio, struct gpio_pinmap *map, int pin)
207aca51624Sgrange {
208aca51624Sgrange struct gpio_softc *sc = gpio;
209aca51624Sgrange
210aca51624Sgrange return (gpiobus_pin_read(sc->sc_gc, map->pm_map[pin]));
211aca51624Sgrange }
212aca51624Sgrange
213aca51624Sgrange void
gpio_pin_write(void * gpio,struct gpio_pinmap * map,int pin,int value)214aca51624Sgrange gpio_pin_write(void *gpio, struct gpio_pinmap *map, int pin, int value)
215aca51624Sgrange {
216aca51624Sgrange struct gpio_softc *sc = gpio;
217aca51624Sgrange
218aca51624Sgrange return (gpiobus_pin_write(sc->sc_gc, map->pm_map[pin], value));
219aca51624Sgrange }
220aca51624Sgrange
221aca51624Sgrange void
gpio_pin_ctl(void * gpio,struct gpio_pinmap * map,int pin,int flags)222aca51624Sgrange gpio_pin_ctl(void *gpio, struct gpio_pinmap *map, int pin, int flags)
223aca51624Sgrange {
224aca51624Sgrange struct gpio_softc *sc = gpio;
225aca51624Sgrange
226aca51624Sgrange return (gpiobus_pin_ctl(sc->sc_gc, map->pm_map[pin], flags));
227aca51624Sgrange }
228aca51624Sgrange
229aca51624Sgrange int
gpio_pin_caps(void * gpio,struct gpio_pinmap * map,int pin)230df8d0d17Sgrange gpio_pin_caps(void *gpio, struct gpio_pinmap *map, int pin)
231df8d0d17Sgrange {
232df8d0d17Sgrange struct gpio_softc *sc = gpio;
233df8d0d17Sgrange
234df8d0d17Sgrange return (sc->sc_pins[map->pm_map[pin]].pin_caps);
235df8d0d17Sgrange }
236df8d0d17Sgrange
237df8d0d17Sgrange int
gpio_npins(u_int32_t mask)238aca51624Sgrange gpio_npins(u_int32_t mask)
239aca51624Sgrange {
240aca51624Sgrange int npins, i;
241aca51624Sgrange
242aca51624Sgrange for (npins = 0, i = 0; i < 32; i++)
243aca51624Sgrange if (mask & (1 << i))
244aca51624Sgrange npins++;
245aca51624Sgrange
246aca51624Sgrange return (npins);
247aca51624Sgrange }
248aca51624Sgrange
249aca51624Sgrange int
gpioopen(dev_t dev,int flag,int mode,struct proc * p)2503615eacbSgrange gpioopen(dev_t dev, int flag, int mode, struct proc *p)
2513615eacbSgrange {
2523615eacbSgrange struct gpio_softc *sc;
253*894c7bfeSvisa int error = 0;
2543615eacbSgrange
2553615eacbSgrange sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
2563615eacbSgrange if (sc == NULL)
2573615eacbSgrange return (ENXIO);
2583615eacbSgrange
2593615eacbSgrange if (sc->sc_opened)
260*894c7bfeSvisa error = EBUSY;
261*894c7bfeSvisa else
2623615eacbSgrange sc->sc_opened = 1;
2633615eacbSgrange
264*894c7bfeSvisa device_unref(&sc->sc_dev);
265*894c7bfeSvisa
266*894c7bfeSvisa return (error);
2673615eacbSgrange }
2683615eacbSgrange
2693615eacbSgrange int
gpioclose(dev_t dev,int flag,int mode,struct proc * p)2703615eacbSgrange gpioclose(dev_t dev, int flag, int mode, struct proc *p)
2713615eacbSgrange {
2723615eacbSgrange struct gpio_softc *sc;
2733615eacbSgrange
2743615eacbSgrange sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
2753adc742dSjsg if (sc == NULL)
2763adc742dSjsg return (ENXIO);
2773adc742dSjsg
2783615eacbSgrange sc->sc_opened = 0;
2793615eacbSgrange
280*894c7bfeSvisa device_unref(&sc->sc_dev);
281*894c7bfeSvisa
2823615eacbSgrange return (0);
2833615eacbSgrange }
2843615eacbSgrange
2853615eacbSgrange int
gpio_pinbyname(struct gpio_softc * sc,char * gp_name)286a388edcaSmbalmer gpio_pinbyname(struct gpio_softc *sc, char *gp_name)
287a388edcaSmbalmer {
288a388edcaSmbalmer struct gpio_name *nm;
289a388edcaSmbalmer
290a388edcaSmbalmer LIST_FOREACH(nm, &sc->sc_names, gp_next)
291a388edcaSmbalmer if (!strcmp(nm->gp_name, gp_name))
292a388edcaSmbalmer return (nm->gp_pin);
293a388edcaSmbalmer return (-1);
294a388edcaSmbalmer }
295a388edcaSmbalmer
296a388edcaSmbalmer int
gpio_ioctl(struct gpio_softc * sc,u_long cmd,caddr_t data,int flag)297*894c7bfeSvisa gpio_ioctl(struct gpio_softc *sc, u_long cmd, caddr_t data, int flag)
2983615eacbSgrange {
2993615eacbSgrange gpio_chipset_tag_t gc;
3003615eacbSgrange struct gpio_info *info;
3013615eacbSgrange struct gpio_pin_op *op;
302fca40c6aSmbalmer struct gpio_attach *attach;
303fca40c6aSmbalmer struct gpio_attach_args ga;
304fca40c6aSmbalmer struct gpio_dev *gdev;
305a388edcaSmbalmer struct gpio_name *nm;
306a388edcaSmbalmer struct gpio_pin_set *set;
307fca40c6aSmbalmer struct device *dv;
308a388edcaSmbalmer int pin, value, flags, npins, found;
3093615eacbSgrange
3103615eacbSgrange gc = sc->sc_gc;
3113615eacbSgrange
3123615eacbSgrange switch (cmd) {
3133615eacbSgrange case GPIOINFO:
3143615eacbSgrange info = (struct gpio_info *)data;
315a388edcaSmbalmer if (securelevel < 1)
3163615eacbSgrange info->gpio_npins = sc->sc_npins;
317a388edcaSmbalmer else {
318a388edcaSmbalmer for (pin = npins = 0; pin < sc->sc_npins; pin++)
319a388edcaSmbalmer if (sc->sc_pins[pin].pin_flags & GPIO_PIN_SET)
320a388edcaSmbalmer ++npins;
321a388edcaSmbalmer info->gpio_npins = npins;
322a388edcaSmbalmer }
3233615eacbSgrange break;
3243615eacbSgrange case GPIOPINREAD:
3253615eacbSgrange op = (struct gpio_pin_op *)data;
3263615eacbSgrange
327a388edcaSmbalmer if (op->gp_name[0] != '\0') {
328a388edcaSmbalmer pin = gpio_pinbyname(sc, op->gp_name);
329a388edcaSmbalmer if (pin == -1)
330a388edcaSmbalmer return (EINVAL);
331a388edcaSmbalmer } else
3323615eacbSgrange pin = op->gp_pin;
333a388edcaSmbalmer
3343615eacbSgrange if (pin < 0 || pin >= sc->sc_npins)
3353615eacbSgrange return (EINVAL);
3363615eacbSgrange
337a388edcaSmbalmer if (!(sc->sc_pins[pin].pin_flags & GPIO_PIN_SET) &&
338a388edcaSmbalmer securelevel > 0)
339a388edcaSmbalmer return (EPERM);
340a388edcaSmbalmer
3413615eacbSgrange /* return read value */
3423615eacbSgrange op->gp_value = gpiobus_pin_read(gc, pin);
3433615eacbSgrange break;
3443615eacbSgrange case GPIOPINWRITE:
3456ec0d2e1Sgrange if ((flag & FWRITE) == 0)
3466ec0d2e1Sgrange return (EBADF);
3476ec0d2e1Sgrange
3483615eacbSgrange op = (struct gpio_pin_op *)data;
3493615eacbSgrange
350a388edcaSmbalmer if (op->gp_name[0] != '\0') {
351a388edcaSmbalmer pin = gpio_pinbyname(sc, op->gp_name);
352a388edcaSmbalmer if (pin == -1)
353a388edcaSmbalmer return (EINVAL);
354a388edcaSmbalmer } else
3553615eacbSgrange pin = op->gp_pin;
356a388edcaSmbalmer
3573615eacbSgrange if (pin < 0 || pin >= sc->sc_npins)
3583615eacbSgrange return (EINVAL);
359a388edcaSmbalmer
360aca51624Sgrange if (sc->sc_pins[pin].pin_mapped)
361aca51624Sgrange return (EBUSY);
3623615eacbSgrange
363a388edcaSmbalmer if (!(sc->sc_pins[pin].pin_flags & GPIO_PIN_SET) &&
364a388edcaSmbalmer securelevel > 0)
365a388edcaSmbalmer return (EPERM);
366a388edcaSmbalmer
3673615eacbSgrange value = op->gp_value;
3683615eacbSgrange if (value != GPIO_PIN_LOW && value != GPIO_PIN_HIGH)
3693615eacbSgrange return (EINVAL);
3703615eacbSgrange
3713615eacbSgrange gpiobus_pin_write(gc, pin, value);
3723615eacbSgrange /* return old value */
3733615eacbSgrange op->gp_value = sc->sc_pins[pin].pin_state;
3743615eacbSgrange /* update current value */
3753615eacbSgrange sc->sc_pins[pin].pin_state = value;
3763615eacbSgrange break;
3773615eacbSgrange case GPIOPINTOGGLE:
3786ec0d2e1Sgrange if ((flag & FWRITE) == 0)
3796ec0d2e1Sgrange return (EBADF);
3806ec0d2e1Sgrange
3813615eacbSgrange op = (struct gpio_pin_op *)data;
3823615eacbSgrange
383a388edcaSmbalmer if (op->gp_name[0] != '\0') {
384a388edcaSmbalmer pin = gpio_pinbyname(sc, op->gp_name);
385a388edcaSmbalmer if (pin == -1)
386a388edcaSmbalmer return (EINVAL);
387a388edcaSmbalmer } else
3883615eacbSgrange pin = op->gp_pin;
389a388edcaSmbalmer
3903615eacbSgrange if (pin < 0 || pin >= sc->sc_npins)
3913615eacbSgrange return (EINVAL);
392a388edcaSmbalmer
393aca51624Sgrange if (sc->sc_pins[pin].pin_mapped)
394aca51624Sgrange return (EBUSY);
3953615eacbSgrange
396a388edcaSmbalmer if (!(sc->sc_pins[pin].pin_flags & GPIO_PIN_SET) &&
397a388edcaSmbalmer securelevel > 0)
398a388edcaSmbalmer return (EPERM);
399a388edcaSmbalmer
4003615eacbSgrange value = (sc->sc_pins[pin].pin_state == GPIO_PIN_LOW ?
4013615eacbSgrange GPIO_PIN_HIGH : GPIO_PIN_LOW);
4023615eacbSgrange gpiobus_pin_write(gc, pin, value);
4033615eacbSgrange /* return old value */
4043615eacbSgrange op->gp_value = sc->sc_pins[pin].pin_state;
4053615eacbSgrange /* update current value */
4063615eacbSgrange sc->sc_pins[pin].pin_state = value;
4073615eacbSgrange break;
408fca40c6aSmbalmer case GPIOATTACH:
409a388edcaSmbalmer if (securelevel > 0)
410a388edcaSmbalmer return (EPERM);
411a388edcaSmbalmer
412fca40c6aSmbalmer attach = (struct gpio_attach *)data;
413fca40c6aSmbalmer bzero(&ga, sizeof(ga));
414fca40c6aSmbalmer ga.ga_gpio = sc;
415fca40c6aSmbalmer ga.ga_dvname = attach->ga_dvname;
416fca40c6aSmbalmer ga.ga_offset = attach->ga_offset;
417fca40c6aSmbalmer ga.ga_mask = attach->ga_mask;
4187d434456Smatthieu ga.ga_flags = attach->ga_flags;
419fca40c6aSmbalmer dv = config_found_sm((struct device *)sc, &ga, gpiobus_print,
420fca40c6aSmbalmer gpio_submatch);
421fca40c6aSmbalmer if (dv != NULL) {
422fb9e5ef0Sderaadt gdev = malloc(sizeof(*gdev), M_DEVBUF,
423fca40c6aSmbalmer M_WAITOK);
424fca40c6aSmbalmer gdev->sc_dev = dv;
425a388edcaSmbalmer LIST_INSERT_HEAD(&sc->sc_devs, gdev, sc_next);
426fca40c6aSmbalmer }
427fca40c6aSmbalmer break;
428fca40c6aSmbalmer case GPIODETACH:
429a388edcaSmbalmer if (securelevel > 0)
430a388edcaSmbalmer return (EPERM);
431a388edcaSmbalmer
432fca40c6aSmbalmer attach = (struct gpio_attach *)data;
433a388edcaSmbalmer LIST_FOREACH(gdev, &sc->sc_devs, sc_next) {
434fca40c6aSmbalmer if (strcmp(gdev->sc_dev->dv_xname, attach->ga_dvname)
435fca40c6aSmbalmer == 0) {
436fca40c6aSmbalmer if (config_detach(gdev->sc_dev, 0) == 0) {
437fca40c6aSmbalmer LIST_REMOVE(gdev, sc_next);
438fb9e5ef0Sderaadt free(gdev, M_DEVBUF, sizeof(*gdev));
439fca40c6aSmbalmer }
440fca40c6aSmbalmer break;
441fca40c6aSmbalmer }
442fca40c6aSmbalmer }
443fca40c6aSmbalmer break;
444a388edcaSmbalmer case GPIOPINSET:
445a388edcaSmbalmer if (securelevel > 0)
446a388edcaSmbalmer return (EPERM);
447a388edcaSmbalmer
448a388edcaSmbalmer set = (struct gpio_pin_set *)data;
449a388edcaSmbalmer
450a388edcaSmbalmer if (set->gp_name[0] != '\0') {
451a388edcaSmbalmer pin = gpio_pinbyname(sc, set->gp_name);
452a388edcaSmbalmer if (pin == -1)
453a388edcaSmbalmer return (EINVAL);
454a388edcaSmbalmer } else
455a388edcaSmbalmer pin = set->gp_pin;
456a388edcaSmbalmer if (pin < 0 || pin >= sc->sc_npins)
457a388edcaSmbalmer return (EINVAL);
458a388edcaSmbalmer flags = set->gp_flags;
459a388edcaSmbalmer /* check that the controller supports all requested flags */
460a388edcaSmbalmer if ((flags & sc->sc_pins[pin].pin_caps) != flags)
461a388edcaSmbalmer return (ENODEV);
462a388edcaSmbalmer flags = set->gp_flags | GPIO_PIN_SET;
463a388edcaSmbalmer
464a388edcaSmbalmer set->gp_caps = sc->sc_pins[pin].pin_caps;
465a388edcaSmbalmer /* return old value */
466a388edcaSmbalmer set->gp_flags = sc->sc_pins[pin].pin_flags;
467a388edcaSmbalmer if (flags > 0) {
468a388edcaSmbalmer gpiobus_pin_ctl(gc, pin, flags);
469a388edcaSmbalmer /* update current value */
470a388edcaSmbalmer sc->sc_pins[pin].pin_flags = flags;
471a388edcaSmbalmer }
472a388edcaSmbalmer
473a388edcaSmbalmer /* rename pin or new pin? */
474a388edcaSmbalmer if (set->gp_name2[0] != '\0') {
475a388edcaSmbalmer found = 0;
476a388edcaSmbalmer LIST_FOREACH(nm, &sc->sc_names, gp_next)
477a388edcaSmbalmer if (nm->gp_pin == pin) {
478a388edcaSmbalmer strlcpy(nm->gp_name, set->gp_name2,
479a388edcaSmbalmer sizeof(nm->gp_name));
480a388edcaSmbalmer found = 1;
481a388edcaSmbalmer break;
482a388edcaSmbalmer }
483a388edcaSmbalmer if (!found) {
484fb9e5ef0Sderaadt nm = malloc(sizeof(*nm), M_DEVBUF, M_WAITOK);
485a388edcaSmbalmer strlcpy(nm->gp_name, set->gp_name2,
486a388edcaSmbalmer sizeof(nm->gp_name));
487a388edcaSmbalmer nm->gp_pin = set->gp_pin;
488a388edcaSmbalmer LIST_INSERT_HEAD(&sc->sc_names, nm, gp_next);
489a388edcaSmbalmer }
490a388edcaSmbalmer }
491a388edcaSmbalmer break;
492a388edcaSmbalmer case GPIOPINUNSET:
493a388edcaSmbalmer if (securelevel > 0)
494a388edcaSmbalmer return (EPERM);
495a388edcaSmbalmer
496a388edcaSmbalmer set = (struct gpio_pin_set *)data;
497a388edcaSmbalmer if (set->gp_name[0] != '\0') {
498a388edcaSmbalmer pin = gpio_pinbyname(sc, set->gp_name);
499a388edcaSmbalmer if (pin == -1)
500a388edcaSmbalmer return (EINVAL);
501a388edcaSmbalmer } else
502a388edcaSmbalmer pin = set->gp_pin;
503a388edcaSmbalmer
504a388edcaSmbalmer if (pin < 0 || pin >= sc->sc_npins)
505a388edcaSmbalmer return (EINVAL);
506a388edcaSmbalmer if (sc->sc_pins[pin].pin_mapped)
507a388edcaSmbalmer return (EBUSY);
508a388edcaSmbalmer if (!(sc->sc_pins[pin].pin_flags & GPIO_PIN_SET))
509a388edcaSmbalmer return (EINVAL);
510a388edcaSmbalmer
511a388edcaSmbalmer LIST_FOREACH(nm, &sc->sc_names, gp_next) {
512a388edcaSmbalmer if (nm->gp_pin == pin) {
513a388edcaSmbalmer LIST_REMOVE(nm, gp_next);
514fb9e5ef0Sderaadt free(nm, M_DEVBUF, sizeof(*nm));
515a388edcaSmbalmer break;
516a388edcaSmbalmer }
517a388edcaSmbalmer }
518a388edcaSmbalmer sc->sc_pins[pin].pin_flags &= ~GPIO_PIN_SET;
519a388edcaSmbalmer break;
5203615eacbSgrange default:
5213615eacbSgrange return (ENOTTY);
5223615eacbSgrange }
5233615eacbSgrange
5243615eacbSgrange return (0);
5253615eacbSgrange }
526*894c7bfeSvisa
527*894c7bfeSvisa int
gpioioctl(dev_t dev,u_long cmd,caddr_t data,int flag,struct proc * p)528*894c7bfeSvisa gpioioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p)
529*894c7bfeSvisa {
530*894c7bfeSvisa struct gpio_softc *sc;
531*894c7bfeSvisa int error;
532*894c7bfeSvisa
533*894c7bfeSvisa sc = (struct gpio_softc *)device_lookup(&gpio_cd, minor(dev));
534*894c7bfeSvisa if (sc == NULL)
535*894c7bfeSvisa return (ENXIO);
536*894c7bfeSvisa
537*894c7bfeSvisa error = gpio_ioctl(sc, cmd, data, flag);
538*894c7bfeSvisa
539*894c7bfeSvisa device_unref(&sc->sc_dev);
540*894c7bfeSvisa
541*894c7bfeSvisa return (error);
542*894c7bfeSvisa }
543