xref: /dragonfly/sys/dev/misc/gpio/gpio.c (revision 9f7604d7)
1 /*
2  * Copyright (c) 2009 The DragonFly Project.  All rights reserved.
3  *
4  * This code is derived from software contributed to The DragonFly Project
5  * by Alex Hornung <ahornung@gmail.com>
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in
15  *    the documentation and/or other materials provided with the
16  *    distribution.
17  * 3. Neither the name of The DragonFly Project nor the names of its
18  *    contributors may be used to endorse or promote products derived
19  *    from this software without specific, prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
25  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32  * SUCH DAMAGE.
33  *
34  *
35  * Copyright (c) 2008 Marc Balmer <mbalmer@openbsd.org>
36  * Copyright (c) 2004, 2006 Alexander Yurchenko <grange@openbsd.org>
37  *
38  * Permission to use, copy, modify, and distribute this software for any
39  * purpose with or without fee is hereby granted, provided that the above
40  * copyright notice and this permission notice appear in all copies.
41  *
42  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
43  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
44  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
45  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
46  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
47  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
48  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
49  */
50 /*
51  * XXX: consumer_detach stuff.
52  * XXX: userland stuff.
53  */
54 
55 #include <sys/param.h>
56 #include <sys/conf.h>
57 #include <sys/kernel.h>
58 #include <sys/systm.h>
59 #include <sys/limits.h>
60 #include <sys/thread.h>
61 #include <sys/thread2.h>
62 #include <sys/malloc.h>
63 #include <sys/ctype.h>
64 #include <sys/sbuf.h>
65 #include <sys/queue.h>
66 #include <sys/uio.h>
67 #include <sys/lock.h>
68 #include <dev/misc/gpio/gpio.h>
69 #include <sys/devfs.h>
70 
71 struct gpio_driver {
72 	char	*name;
73 	struct devfs_bitmap	unit_bitmap;
74 	LIST_ENTRY(gpio_driver)	link;
75 };
76 
77 static LIST_HEAD(, gpio_consumer) gpio_conslist = LIST_HEAD_INITIALIZER(&gpio_conslist);
78 static LIST_HEAD(, gpio_driver) gpio_driverlist = LIST_HEAD_INITIALIZER(&gpio_driverlist);
79 DEVFS_DECLARE_CLONE_BITMAP(gpio);
80 static struct lock gpio_lock;
81 
82 void
83 gpio_consumer_register(struct gpio_consumer *gcp)
84 {
85 	lockmgr(&gpio_lock, LK_EXCLUSIVE);
86 	LIST_INSERT_HEAD(&gpio_conslist, gcp, link);
87 	lockmgr(&gpio_lock, LK_RELEASE);
88 }
89 
90 void
91 gpio_consumer_unregister(struct gpio_consumer *gcp)
92 {
93 	lockmgr(&gpio_lock, LK_EXCLUSIVE);
94 	LIST_REMOVE(gcp, link);
95 	lockmgr(&gpio_lock, LK_RELEASE);
96 }
97 
98 int
99 gpio_consumer_attach(const char *consumer, void *arg, struct gpio *gp,
100 	    int pin, u_int32_t mask)
101 {
102 	struct gpio_consumer *gcp;
103 	int error = -1;
104 	int locked = 0;
105 
106 	/* Check if it is locked already. if not, we acquire the lock */
107 	if ((lockstatus(&gpio_lock, curthread)) != LK_EXCLUSIVE) {
108 		lockmgr(&gpio_lock, LK_EXCLUSIVE);
109 		locked = 1;
110 	}
111 
112 	LIST_FOREACH(gcp, &gpio_conslist, link) {
113 		if (strcmp(gcp->consumer_name, consumer) != 0)
114 			continue;
115 
116 		if (gcp->consumer_attach)
117 			error = gcp->consumer_attach(gp, arg, pin, mask);
118 		if (error) {
119 			kprintf("gpio: Attach of consumer %s to gpio %s%d pin %d failed "
120 			    "(consumer error %d)\n", consumer, gp->driver_name,
121 			    gp->driver_unit, pin, error);
122 			goto end;
123 		}
124 
125 		kprintf("gpio: Attached consumer %s to gpio %s%d pin %d\n",
126 		    consumer, gp->driver_name, gp->driver_unit, pin);
127 		goto end;
128 	}
129 
130 	kprintf("gpio: Attach of consumer %s to gpio %s%d pin %d failed "
131 	    "(unknown consumer)\n", consumer, gp->driver_name, gp->driver_unit, pin);
132 
133 end:
134 	/* If we acquired the lock, we also get rid of it */
135 	if (locked)
136 		lockmgr(&gpio_lock, LK_RELEASE);
137 	return error;
138 }
139 
140 int
141 gpio_consumer_detach(const char *consumer, struct gpio *gp,
142 		int pin)
143 {
144 	struct gpio_consumer *gcp;
145 	int error = -1;
146 	int locked = 0;
147 
148 	/* Check if it is locked already. if not, we acquire the lock */
149 	if ((lockstatus(&gpio_lock, curthread)) != LK_EXCLUSIVE) {
150 		lockmgr(&gpio_lock, LK_EXCLUSIVE);
151 		locked = 1;
152 	}
153 
154 	LIST_FOREACH(gcp, &gpio_conslist, link) {
155 		if (strcmp(gcp->consumer_name, consumer) != 0)
156 			continue;
157 
158 		if (gcp->consumer_detach)
159 			error = gcp->consumer_detach(gp, NULL, pin);
160 		if (error) {
161 			kprintf("gpio: Detach of consumer %s from gpio %s%d pin %d failed "
162 			    "(consumer error %d)\n", consumer, gp->driver_name,
163 			    gp->driver_unit, pin, error);
164 			goto end;
165 		}
166 
167 		kprintf("gpio: Detached consumer %s from gpio %s%d pin %d\n",
168 		    consumer, gp->driver_name, gp->driver_unit, pin);
169 		goto end;
170 	}
171 
172 	kprintf("gpio: Detach of consumer %s from gpio %s%d pin %d failed "
173 	    "(unknown consumer)\n", consumer, gp->driver_name, gp->driver_unit, pin);
174 
175 end:
176 	/* If we acquired the lock, we also get rid of it */
177 	if (locked)
178 		lockmgr(&gpio_lock, LK_RELEASE);
179 	return error;
180 }
181 
182 struct gpio_mapping *
183 gpio_map(struct gpio *gp, int *map, int offset, u_int32_t mask)
184 {
185 	struct gpio_mapping *gmp;
186 	int npins, pin, i;
187 	int locked = 0;
188 
189 	npins = gpio_npins(mask);
190 	if (npins > gp->npins)
191 		return NULL;
192 	if (npins == 0)
193 		return NULL;
194 
195 	/* Check if it is locked already. if not, we acquire the lock */
196 	if ((lockstatus(&gpio_lock, curthread)) != LK_EXCLUSIVE) {
197 		lockmgr(&gpio_lock, LK_EXCLUSIVE);
198 		locked = 1;
199 	}
200 
201 	gmp = kmalloc(sizeof(struct gpio_mapping), M_TEMP, M_WAITOK);
202 	gmp->gp = gp;
203 	if (map) {
204 		gmp->map = map;
205 		gmp->map_alloced = 0;
206 	} else {
207 		gmp->map = kmalloc(sizeof(int) * npins, M_TEMP, M_WAITOK);
208 		gmp->map_alloced = 1;
209 	}
210 
211 	for (npins = 0, i = 0; i < 32; i++)
212 		if (mask & (1 << i)) {
213 			pin = offset + i;
214 			if (pin < 0 || pin >= gp->npins ||
215 				gp->pins[pin].pin_mapped || gp->pins[pin].pin_opened) {
216 				if (map == NULL)
217 					kfree(gmp->map, M_TEMP);
218 				kfree(gmp, M_TEMP);
219 				/* If we acquired the lock, we also get rid of it */
220 				if (locked)
221 					lockmgr(&gpio_lock, LK_RELEASE);
222 				return NULL;
223 			}
224 			gp->pins[pin].pin_mapped = 1;
225 			gmp->map[npins++] = pin;
226 		}
227 	gmp->size = npins;
228 
229 	/* If we acquired the lock, we also get rid of it */
230 	if (locked)
231 		lockmgr(&gpio_lock, LK_RELEASE);
232 
233 	return gmp;
234 }
235 
236 void
237 gpio_unmap(struct gpio_mapping *gmp)
238 {
239 	int pin, i;
240 	int locked = 0;
241 
242 	/* Check if it is locked already. if not, we acquire the lock */
243 	if ((lockstatus(&gpio_lock, curthread)) != LK_EXCLUSIVE) {
244 		lockmgr(&gpio_lock, LK_EXCLUSIVE);
245 		locked = 1;
246 	}
247 
248 	for (i = 0; i < gmp->size; i++) {
249 		pin = gmp->map[i];
250 		gmp->gp->pins[pin].pin_mapped = 0;
251 	}
252 
253 	if (gmp->map_alloced)
254 		kfree(gmp->map, M_TEMP);
255 	kfree(gmp, M_TEMP);
256 
257 	/* If we acquired the lock, we also get rid of it */
258 	if (locked)
259 		lockmgr(&gpio_lock, LK_RELEASE);
260 }
261 
262 int
263 gpio_npins(u_int32_t mask)
264 {
265 	int npins, i;
266 
267 	for (npins = 0, i = 0; i < 32; i++)
268 		if (mask & (1 << i))
269 			npins++;
270 
271 	return (npins);
272 }
273 
274 int
275 gpio_pin_read(struct gpio *gp, struct gpio_mapping *map, int pin)
276 {
277 	return gp->pin_read(gp->arg, map->map[pin]);
278 }
279 
280 void
281 gpio_pin_write(struct gpio *gp, struct gpio_mapping *map, int pin, int data)
282 {
283 	gp->pin_write(gp->arg, map->map[pin], data);
284 }
285 
286 void
287 gpio_pin_ctl(struct gpio *gp, struct gpio_mapping *map, int pin, int flags)
288 {
289 	gp->pin_ctl(gp->arg, map->map[pin], flags);
290 }
291 
292 int
293 gpio_pin_caps(struct gpio *gp, struct gpio_mapping *map, int pin)
294 {
295 	return (gp->pins[map->map[pin]].pin_caps);
296 }
297 
298 static int
299 gpio_open(struct dev_open_args *ap)
300 {
301 	gpio_pin_t	*pin;
302 	cdev_t	dev;
303 
304 	dev = ap->a_head.a_dev;
305 	pin = dev->si_drv2;
306 
307 	if (pin->pin_opened || pin->pin_mapped)
308 		return EBUSY;
309 
310 	pin->pin_opened = 1;
311 
312 	return 0;
313 }
314 
315 static int
316 gpio_close(struct dev_close_args *ap)
317 {
318 	gpio_pin_t	*pin;
319 	cdev_t	dev;
320 
321 	dev = ap->a_head.a_dev;
322 	pin = dev->si_drv2;
323 
324 	if (pin->pin_opened)
325 		pin->pin_opened = 0;
326 
327 	return 0;
328 }
329 
330 static int
331 gpio_write(struct dev_write_args *ap)
332 {
333 	struct gpio	*gp;
334 	gpio_pin_t	*pin;
335 	cdev_t		dev;
336 	int		error;
337 	int		data = 0;
338 
339 	dev = ap->a_head.a_dev;
340 	gp = dev->si_drv1;
341 	pin = dev->si_drv2;
342 
343 	if (ap->a_uio->uio_resid > sizeof(int))
344 		return EINVAL;
345 
346 	error = uiomove((void *)&data, ap->a_uio->uio_resid, ap->a_uio);
347 	if (error)
348 		return error;
349 
350 	if (data != GPIO_PIN_LOW && data != GPIO_PIN_HIGH)
351 		return EINVAL;
352 
353 	gp->pin_write(gp->arg, pin->pin_num, data);
354 	pin->pin_state = data;
355 
356 	return 0;
357 }
358 
359 static int
360 gpio_read(struct dev_read_args *ap)
361 {
362 	struct gpio	*gp;
363 	gpio_pin_t	*pin;
364 	cdev_t		dev;
365 	int		error;
366 	int		data = 0;
367 
368 	dev = ap->a_head.a_dev;
369 	gp = dev->si_drv1;
370 	pin = dev->si_drv2;
371 
372 	if (ap->a_uio->uio_resid < sizeof(char))
373 		return EINVAL;
374 
375 	data = gp->pin_read(gp->arg, pin->pin_num);
376 
377 	error = uiomove((void *)&data,
378 	    (ap->a_uio->uio_resid > sizeof(int))?(sizeof(int)):(ap->a_uio->uio_resid),
379 		ap->a_uio);
380 
381 	return error;
382 }
383 
384 static int
385 gpio_ioctl(struct dev_ioctl_args *ap)
386 {
387 	struct gpio_pin_set_args *gpsa;
388 	struct gpio	*gp;
389 	gpio_pin_t	*pin;
390 	cdev_t		dev;
391 
392 	dev = ap->a_head.a_dev;
393 	gpsa = (struct gpio_pin_set_args *)ap->a_data;
394 	gp = dev->si_drv1;
395 	pin = dev->si_drv2;
396 
397 	switch(ap->a_cmd) {
398 	case GPIOPINSET:
399 		if (pin->pin_opened || pin->pin_mapped)
400 			return EBUSY;
401 
402 		gpsa->caps = pin->pin_caps;
403 		gpsa->flags = pin->pin_flags;
404 
405 		if ((gpsa->flags & pin->pin_caps) != gpsa->flags)
406 			return ENODEV;
407 
408 		if (gpsa->flags > 0) {
409 			gp->pin_ctl(gp->arg, pin->pin_num, gpsa->flags);
410 			pin->pin_flags = gpsa->flags | GPIO_PIN_SET;
411 		}
412 		break;
413 
414 	case GPIOPINUNSET:
415 		return EINVAL;
416 		break;
417 
418 	default:
419 		return EINVAL;
420 	}
421 	return 0;
422 }
423 
424 static int
425 gpio_master_ioctl(struct dev_ioctl_args *ap)
426 {
427 	struct gpio_pin_set_args *gpsa;
428 	struct gpio_info	*gpi;
429 	struct gpio_attach_args	*gpaa;
430 	struct gpio	*gp;
431 	cdev_t		dev;
432 	gpio_pin_t	*pin;
433 	int		error = 0;
434 
435 	dev = ap->a_head.a_dev;
436 	gp = dev->si_drv1;
437 
438 	switch(ap->a_cmd) {
439 	case GPIOINFO:
440 		gpi = (struct gpio_info *)ap->a_data;
441 		gpi->npins = gp->npins;
442 		if (gpi->pins != NULL) {
443 			error = copyout(gp->pins, gpi->pins,
444 			    sizeof(struct gpio_pin)*gp->npins);
445 		}
446 		break;
447 
448 	case GPIOATTACH:
449 		gpaa = (struct gpio_attach_args *)ap->a_data;
450 		error = gpio_consumer_attach(gpaa->consumer_name,
451 		    (gpaa->arg_type == GPIO_TYPE_INT)?
452 		    ((void *)gpaa->consumer_arg.lint):
453 		    (gpaa->consumer_arg.string),
454 		    gp, gpaa->pin_offset, gpaa->pin_mask);
455 		break;
456 
457 	case GPIODETACH:
458 		gpaa = (struct gpio_attach_args *)ap->a_data;
459 		error = gpio_consumer_detach(gpaa->consumer_name, gp,
460 		    gpaa->pin_offset);
461 		break;
462 
463 	case GPIOPINSET:
464 		gpsa = (struct gpio_pin_set_args *)ap->a_data;
465 		if (gpsa->pin < 0 || gpsa->pin >= gp->npins)
466 			return EINVAL;
467 
468 		pin = &gp->pins[gpsa->pin];
469 
470 		if (pin->pin_opened || pin->pin_mapped)
471 			return EBUSY;
472 
473 		gpsa->caps = pin->pin_caps;
474 		gpsa->flags = pin->pin_flags;
475 
476 		if ((gpsa->flags & pin->pin_caps) != gpsa->flags)
477 			return ENODEV;
478 
479 		if (gpsa->flags > 0) {
480 			gp->pin_ctl(gp->arg, gpsa->pin, gpsa->flags);
481 			pin->pin_flags = gpsa->flags | GPIO_PIN_SET;
482 		}
483 		break;
484 
485 	case GPIOPINUNSET:
486 		gpsa = (struct gpio_pin_set_args *)ap->a_data;
487 		error = EINVAL;
488 		break;
489 
490 	default:
491 		return EINVAL;
492 	}
493 
494 	return error;
495 }
496 
497 static struct dev_ops gpio_ops = {
498 	{ "gpio", 0, 0 },
499 	.d_open  =	gpio_open,
500 	.d_close =	gpio_close,
501 	.d_write = 	gpio_write,
502 	.d_read  =	gpio_read,
503 	.d_ioctl =	gpio_ioctl,
504 };
505 
506 static struct dev_ops gpio_master_ops = {
507 	{ "gpio", 0, 0 },
508 	.d_ioctl =	gpio_master_ioctl,
509 };
510 
511 void
512 gpio_register(struct gpio *gp)
513 {
514 	struct gpio_driver *gpd;
515 	int i, unit, master_unit = -1;
516 
517 	KKASSERT(gp->npins > 0);
518 	KKASSERT(gp->pins);
519 
520 	lockmgr(&gpio_lock, LK_EXCLUSIVE);
521 	LIST_FOREACH(gpd, &gpio_driverlist, link) {
522 		if (strcmp(gpd->name, gp->driver_name) != 0)
523 			continue;
524 
525 		master_unit = devfs_clone_bitmap_get(&gpd->unit_bitmap, 0);
526 		break;
527 	}
528 	if (master_unit == -1) {
529 		gpd = kmalloc(sizeof(struct gpio_driver),
530 		    M_TEMP, M_WAITOK | M_ZERO);
531 		gpd->name = kstrdup(gp->driver_name, M_TEMP);
532 		devfs_clone_bitmap_init(&gpd->unit_bitmap);
533 		master_unit = devfs_clone_bitmap_get(&gpd->unit_bitmap, 0);
534 		LIST_INSERT_HEAD(&gpio_driverlist, gpd, link);
535 	}
536 	lockmgr(&gpio_lock, LK_RELEASE);
537 
538 	gp->driver_unit = master_unit;
539 	kprintf("gpio: GPIO driver %s%d registered, npins = %d\n",
540 	    gp->driver_name, master_unit, gp->npins);
541 
542 	unit = devfs_clone_bitmap_get(&DEVFS_CLONE_BITMAP(gpio), 0);
543 	gp->master_dev = make_dev(&gpio_master_ops, unit, UID_ROOT, GID_WHEEL, 0600,
544 	    "gpio/%s%d/master", gp->driver_name, master_unit);
545 	gp->master_dev->si_drv1 = gp;
546 
547 	for (i = 0; i < gp->npins; i++) {
548 		unit = devfs_clone_bitmap_get(&DEVFS_CLONE_BITMAP(gpio), 0);
549 		gp->pins[i].dev = make_dev(&gpio_ops, unit, UID_ROOT, GID_WHEEL, 0600,
550 		    "gpio/%s%d/%d", gp->driver_name, master_unit, gp->pins[i].pin_num);
551 		gp->pins[i].dev->si_drv1 = gp;
552 		gp->pins[i].dev->si_drv2 = &gp->pins[i];
553 	}
554 }
555 
556 void
557 gpio_unregister(struct gpio *gp)
558 {
559 	struct gpio_driver *gpd;
560 	int i;
561 
562 	KKASSERT(gp->npins > 0);
563 	KKASSERT(gp->pins);
564 
565 	for (i = 0; i < gp->npins; i++) {
566 		devfs_clone_bitmap_get(&DEVFS_CLONE_BITMAP(gpio),
567 		    minor(gp->pins[i].dev));
568 		destroy_dev(gp->pins[i].dev);
569 	}
570 
571 	destroy_dev(gp->master_dev);
572 
573 	lockmgr(&gpio_lock, LK_EXCLUSIVE);
574 	LIST_FOREACH(gpd, &gpio_driverlist, link) {
575 		if (strcmp(gpd->name, gp->driver_name) != 0)
576 			continue;
577 
578 		devfs_clone_bitmap_put(&gpd->unit_bitmap, gp->driver_unit);
579 		LIST_REMOVE(gpd, link);
580 		break;
581 	}
582 	lockmgr(&gpio_lock, LK_RELEASE);
583 
584 	kprintf("gpio: GPIO driver %s%d unregistered\n",
585 	    gp->driver_name, gp->driver_unit);
586 }
587 
588 static void
589 gpio_drvinit(void *unused)
590 {
591 	lockinit(&gpio_lock, "gpio_lock", 0, 0);
592 	devfs_clone_bitmap_init(&DEVFS_CLONE_BITMAP(gpio));
593 }
594 
595 SYSINIT(gpio, SI_SUB_PRE_DRIVERS, SI_ORDER_FIRST, gpio_drvinit, NULL);
596