xref: /openbsd-src/sys/dev/gpio/gpio.c (revision 894c7bfe3ebe1305bf9e68d6f2ce049895609e00)
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