xref: /freebsd/lib/libcuse/cuse_lib.c (revision 298f5fdc242b760e70cd3494e3a4f1f50b20664d)
1 /* $FreeBSD$ */
2 /*-
3  * Copyright (c) 2010-2012 Hans Petter Selasky. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  */
26 
27 #include <stdio.h>
28 #include <stdint.h>
29 #include <pthread.h>
30 #include <signal.h>
31 #include <unistd.h>
32 #include <string.h>
33 #include <errno.h>
34 #include <stdlib.h>
35 #include <stdarg.h>
36 
37 #include <sys/types.h>
38 #include <sys/queue.h>
39 #include <sys/fcntl.h>
40 #include <sys/mman.h>
41 #include <sys/param.h>
42 
43 #include <fs/cuse/cuse_ioctl.h>
44 
45 #include "cuse.h"
46 
47 int	cuse_debug_level;
48 
49 #ifdef HAVE_DEBUG
50 static const char *cuse_cmd_str(int cmd);
51 
52 #define	DPRINTF(...) do {			\
53 	if (cuse_debug_level != 0)		\
54 		printf(__VA_ARGS__);		\
55 } while (0)
56 #else
57 #define	DPRINTF(...) do { } while (0)
58 #endif
59 
60 struct cuse_vm_allocation {
61 	uint8_t *ptr;
62 	uint32_t size;
63 };
64 
65 struct cuse_dev_entered {
66 	TAILQ_ENTRY(cuse_dev_entered) entry;
67 	pthread_t thread;
68 	void   *per_file_handle;
69 	struct cuse_dev *cdev;
70 	int	cmd;
71 	int	is_local;
72 	int	got_signal;
73 };
74 
75 struct cuse_dev {
76 	TAILQ_ENTRY(cuse_dev) entry;
77 	const struct cuse_methods *mtod;
78 	void   *priv0;
79 	void   *priv1;
80 };
81 
82 static TAILQ_HEAD(, cuse_dev) h_cuse;
83 static TAILQ_HEAD(, cuse_dev_entered) h_cuse_entered;
84 static int f_cuse = -1;
85 static pthread_mutex_t m_cuse;
86 static struct cuse_vm_allocation a_cuse[CUSE_ALLOC_UNIT_MAX];
87 
88 static void
89 cuse_lock(void)
90 {
91 	pthread_mutex_lock(&m_cuse);
92 }
93 
94 static void
95 cuse_unlock(void)
96 {
97 	pthread_mutex_unlock(&m_cuse);
98 }
99 
100 int
101 cuse_init(void)
102 {
103 	pthread_mutexattr_t attr;
104 
105 	f_cuse = open("/dev/cuse", O_RDWR);
106 	if (f_cuse < 0) {
107 		if (feature_present("cuse") == 0)
108 			return (CUSE_ERR_NOT_LOADED);
109 		else
110 			return (CUSE_ERR_INVALID);
111 	}
112 	pthread_mutexattr_init(&attr);
113 	pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
114 	pthread_mutex_init(&m_cuse, &attr);
115 
116 	TAILQ_INIT(&h_cuse);
117 	TAILQ_INIT(&h_cuse_entered);
118 
119 	return (0);
120 }
121 
122 int
123 cuse_uninit(void)
124 {
125 	int f;
126 
127 	if (f_cuse < 0)
128 		return (CUSE_ERR_INVALID);
129 
130 	f = f_cuse;
131 	f_cuse = -1;
132 
133 	close(f);
134 
135 	pthread_mutex_destroy(&m_cuse);
136 
137 	memset(a_cuse, 0, sizeof(a_cuse));
138 
139 	return (0);
140 }
141 
142 unsigned long
143 cuse_vmoffset(void *_ptr)
144 {
145 	uint8_t *ptr_min;
146 	uint8_t *ptr_max;
147 	uint8_t *ptr = _ptr;
148 	unsigned long remainder;
149 	int n;
150 
151 	cuse_lock();
152 	for (n = 0; n != CUSE_ALLOC_UNIT_MAX; n++) {
153 		if (a_cuse[n].ptr == NULL)
154 			continue;
155 
156 		ptr_min = a_cuse[n].ptr;
157 		ptr_max = a_cuse[n].ptr + a_cuse[n].size - 1;
158 
159 		if ((ptr >= ptr_min) && (ptr <= ptr_max)) {
160 
161 			cuse_unlock();
162 
163 			remainder = (ptr - ptr_min);
164 
165 			remainder -= remainder % PAGE_SIZE;
166 
167 			return ((n * PAGE_SIZE * CUSE_ALLOC_PAGES_MAX) + remainder);
168 		}
169 	}
170 	cuse_unlock();
171 
172 	return (0x80000000UL);		/* failure */
173 }
174 
175 void   *
176 cuse_vmalloc(int size)
177 {
178 	struct cuse_alloc_info info;
179 	void *ptr;
180 	int error;
181 	int n;
182 
183 	if (f_cuse < 0)
184 		return (NULL);
185 
186 	memset(&info, 0, sizeof(info));
187 
188 	if (size < 1)
189 		return (NULL);
190 
191 	info.page_count = (size + PAGE_SIZE - 1) / PAGE_SIZE;
192 
193 	cuse_lock();
194 	for (n = 0; n != CUSE_ALLOC_UNIT_MAX; n++) {
195 
196 		if (a_cuse[n].ptr != NULL)
197 			continue;
198 
199 		a_cuse[n].ptr = ((uint8_t *)1);	/* reserve */
200 		a_cuse[n].size = 0;
201 
202 		cuse_unlock();
203 
204 		info.alloc_nr = n;
205 
206 		error = ioctl(f_cuse, CUSE_IOCTL_ALLOC_MEMORY, &info);
207 
208 		if (error) {
209 
210 			cuse_lock();
211 
212 			a_cuse[n].ptr = NULL;
213 
214 			if (errno == EBUSY)
215 				continue;
216 			else
217 				break;
218 		}
219 		ptr = mmap(NULL, info.page_count * PAGE_SIZE,
220 		    PROT_READ | PROT_WRITE,
221 		    MAP_SHARED, f_cuse, CUSE_ALLOC_PAGES_MAX *
222 		    PAGE_SIZE * n);
223 
224 		if (ptr == MAP_FAILED) {
225 
226 			error = ioctl(f_cuse, CUSE_IOCTL_FREE_MEMORY, &info);
227 
228 			if (error) {
229 				/* ignore */
230 			}
231 			cuse_lock();
232 
233 			a_cuse[n].ptr = NULL;
234 
235 			break;
236 		}
237 		cuse_lock();
238 		a_cuse[n].ptr = ptr;
239 		a_cuse[n].size = size;
240 		cuse_unlock();
241 
242 		return (ptr);		/* success */
243 	}
244 	cuse_unlock();
245 	return (NULL);			/* failure */
246 }
247 
248 int
249 cuse_is_vmalloc_addr(void *ptr)
250 {
251 	int n;
252 
253 	if (f_cuse < 0 || ptr == NULL)
254 		return (0);		/* false */
255 
256 	cuse_lock();
257 	for (n = 0; n != CUSE_ALLOC_UNIT_MAX; n++) {
258 		if (a_cuse[n].ptr == ptr)
259 			break;
260 	}
261 	cuse_unlock();
262 
263 	return (n != CUSE_ALLOC_UNIT_MAX);
264 }
265 
266 void
267 cuse_vmfree(void *ptr)
268 {
269 	struct cuse_alloc_info info;
270 	int error;
271 	int n;
272 
273 	if (f_cuse < 0)
274 		return;
275 
276 	memset(&info, 0, sizeof(info));
277 
278 	cuse_lock();
279 	for (n = 0; n != CUSE_ALLOC_UNIT_MAX; n++) {
280 		if (a_cuse[n].ptr != ptr)
281 			continue;
282 
283 		cuse_unlock();
284 
285 		info.alloc_nr = n;
286 
287 		munmap(ptr, a_cuse[n].size);
288 
289 		error = ioctl(f_cuse, CUSE_IOCTL_FREE_MEMORY, &info);
290 
291 		if (error) {
292 			/* ignore */
293 		}
294 		cuse_lock();
295 
296 		a_cuse[n].ptr = NULL;
297 		a_cuse[n].size = 0;
298 
299 		break;
300 	}
301 	cuse_unlock();
302 }
303 
304 int
305 cuse_alloc_unit_number_by_id(int *pnum, int id)
306 {
307 	int error;
308 
309 	if (f_cuse < 0)
310 		return (CUSE_ERR_INVALID);
311 
312 	*pnum = (id & CUSE_ID_MASK);
313 
314 	error = ioctl(f_cuse, CUSE_IOCTL_ALLOC_UNIT_BY_ID, pnum);
315 	if (error)
316 		return (CUSE_ERR_NO_MEMORY);
317 
318 	return (0);
319 
320 }
321 
322 int
323 cuse_free_unit_number_by_id(int num, int id)
324 {
325 	int error;
326 
327 	if (f_cuse < 0)
328 		return (CUSE_ERR_INVALID);
329 
330 	if (num != -1 || id != -1)
331 		num = (id & CUSE_ID_MASK) | (num & 0xFF);
332 
333 	error = ioctl(f_cuse, CUSE_IOCTL_FREE_UNIT_BY_ID, &num);
334 	if (error)
335 		return (CUSE_ERR_NO_MEMORY);
336 
337 	return (0);
338 }
339 
340 int
341 cuse_alloc_unit_number(int *pnum)
342 {
343 	int error;
344 
345 	if (f_cuse < 0)
346 		return (CUSE_ERR_INVALID);
347 
348 	error = ioctl(f_cuse, CUSE_IOCTL_ALLOC_UNIT, pnum);
349 	if (error)
350 		return (CUSE_ERR_NO_MEMORY);
351 
352 	return (0);
353 }
354 
355 int
356 cuse_free_unit_number(int num)
357 {
358 	int error;
359 
360 	if (f_cuse < 0)
361 		return (CUSE_ERR_INVALID);
362 
363 	error = ioctl(f_cuse, CUSE_IOCTL_FREE_UNIT, &num);
364 	if (error)
365 		return (CUSE_ERR_NO_MEMORY);
366 
367 	return (0);
368 }
369 
370 struct cuse_dev *
371 cuse_dev_create(const struct cuse_methods *mtod, void *priv0, void *priv1,
372     uid_t _uid, gid_t _gid, int _perms, const char *_fmt,...)
373 {
374 	struct cuse_create_dev info;
375 	struct cuse_dev *cdev;
376 	va_list args;
377 	int error;
378 
379 	if (f_cuse < 0)
380 		return (NULL);
381 
382 	cdev = malloc(sizeof(*cdev));
383 	if (cdev == NULL)
384 		return (NULL);
385 
386 	memset(cdev, 0, sizeof(*cdev));
387 
388 	cdev->mtod = mtod;
389 	cdev->priv0 = priv0;
390 	cdev->priv1 = priv1;
391 
392 	memset(&info, 0, sizeof(info));
393 
394 	info.dev = cdev;
395 	info.user_id = _uid;
396 	info.group_id = _gid;
397 	info.permissions = _perms;
398 
399 	va_start(args, _fmt);
400 	vsnprintf(info.devname, sizeof(info.devname), _fmt, args);
401 	va_end(args);
402 
403 	error = ioctl(f_cuse, CUSE_IOCTL_CREATE_DEV, &info);
404 	if (error) {
405 		free(cdev);
406 		return (NULL);
407 	}
408 	cuse_lock();
409 	TAILQ_INSERT_TAIL(&h_cuse, cdev, entry);
410 	cuse_unlock();
411 
412 	return (cdev);
413 }
414 
415 
416 void
417 cuse_dev_destroy(struct cuse_dev *cdev)
418 {
419 	int error;
420 
421 	if (f_cuse < 0)
422 		return;
423 
424 	cuse_lock();
425 	TAILQ_REMOVE(&h_cuse, cdev, entry);
426 	cuse_unlock();
427 
428 	error = ioctl(f_cuse, CUSE_IOCTL_DESTROY_DEV, &cdev);
429 	if (error)
430 		return;
431 
432 	free(cdev);
433 }
434 
435 void   *
436 cuse_dev_get_priv0(struct cuse_dev *cdev)
437 {
438 	return (cdev->priv0);
439 }
440 
441 void   *
442 cuse_dev_get_priv1(struct cuse_dev *cdev)
443 {
444 	return (cdev->priv1);
445 }
446 
447 void
448 cuse_dev_set_priv0(struct cuse_dev *cdev, void *priv)
449 {
450 	cdev->priv0 = priv;
451 }
452 
453 void
454 cuse_dev_set_priv1(struct cuse_dev *cdev, void *priv)
455 {
456 	cdev->priv1 = priv;
457 }
458 
459 int
460 cuse_wait_and_process(void)
461 {
462 	pthread_t curr = pthread_self();
463 	struct cuse_dev_entered *pe;
464 	struct cuse_dev_entered enter;
465 	struct cuse_command info;
466 	struct cuse_dev *cdev;
467 	int error;
468 
469 	if (f_cuse < 0)
470 		return (CUSE_ERR_INVALID);
471 
472 	error = ioctl(f_cuse, CUSE_IOCTL_GET_COMMAND, &info);
473 	if (error)
474 		return (CUSE_ERR_OTHER);
475 
476 	cdev = info.dev;
477 
478 	cuse_lock();
479 	enter.thread = curr;
480 	enter.per_file_handle = (void *)info.per_file_handle;
481 	enter.cmd = info.command;
482 	enter.is_local = 0;
483 	enter.got_signal = 0;
484 	enter.cdev = cdev;
485 	TAILQ_INSERT_TAIL(&h_cuse_entered, &enter, entry);
486 	cuse_unlock();
487 
488 	DPRINTF("cuse: Command = %d = %s, flags = %d, arg = 0x%08x, ptr = 0x%08x\n",
489 	    (int)info.command, cuse_cmd_str(info.command), (int)info.fflags,
490 	    (int)info.argument, (int)info.data_pointer);
491 
492 	switch (info.command) {
493 	case CUSE_CMD_OPEN:
494 		if (cdev->mtod->cm_open != NULL)
495 			error = (cdev->mtod->cm_open) (cdev, (int)info.fflags);
496 		else
497 			error = 0;
498 		break;
499 
500 	case CUSE_CMD_CLOSE:
501 
502 		/* wait for other threads to stop */
503 
504 		while (1) {
505 
506 			error = 0;
507 
508 			cuse_lock();
509 			TAILQ_FOREACH(pe, &h_cuse_entered, entry) {
510 				if (pe->cdev != cdev)
511 					continue;
512 				if (pe->thread == curr)
513 					continue;
514 				if (pe->per_file_handle !=
515 				    enter.per_file_handle)
516 					continue;
517 				pe->got_signal = 1;
518 				pthread_kill(pe->thread, SIGHUP);
519 				error = CUSE_ERR_BUSY;
520 			}
521 			cuse_unlock();
522 
523 			if (error == 0)
524 				break;
525 			else
526 				usleep(10000);
527 		}
528 
529 		if (cdev->mtod->cm_close != NULL)
530 			error = (cdev->mtod->cm_close) (cdev, (int)info.fflags);
531 		else
532 			error = 0;
533 		break;
534 
535 	case CUSE_CMD_READ:
536 		if (cdev->mtod->cm_read != NULL) {
537 			error = (cdev->mtod->cm_read) (cdev, (int)info.fflags,
538 			    (void *)info.data_pointer, (int)info.argument);
539 		} else {
540 			error = CUSE_ERR_INVALID;
541 		}
542 		break;
543 
544 	case CUSE_CMD_WRITE:
545 		if (cdev->mtod->cm_write != NULL) {
546 			error = (cdev->mtod->cm_write) (cdev, (int)info.fflags,
547 			    (void *)info.data_pointer, (int)info.argument);
548 		} else {
549 			error = CUSE_ERR_INVALID;
550 		}
551 		break;
552 
553 	case CUSE_CMD_IOCTL:
554 		if (cdev->mtod->cm_ioctl != NULL) {
555 			error = (cdev->mtod->cm_ioctl) (cdev, (int)info.fflags,
556 			    (unsigned int)info.argument, (void *)info.data_pointer);
557 		} else {
558 			error = CUSE_ERR_INVALID;
559 		}
560 		break;
561 
562 	case CUSE_CMD_POLL:
563 		if (cdev->mtod->cm_poll != NULL) {
564 			error = (cdev->mtod->cm_poll) (cdev, (int)info.fflags,
565 			    (int)info.argument);
566 		} else {
567 			error = CUSE_POLL_ERROR;
568 		}
569 		break;
570 
571 	case CUSE_CMD_SIGNAL:
572 		cuse_lock();
573 		TAILQ_FOREACH(pe, &h_cuse_entered, entry) {
574 			if (pe->cdev != cdev)
575 				continue;
576 			if (pe->thread == curr)
577 				continue;
578 			if (pe->per_file_handle !=
579 			    enter.per_file_handle)
580 				continue;
581 			pe->got_signal = 1;
582 			pthread_kill(pe->thread, SIGHUP);
583 		}
584 		cuse_unlock();
585 		break;
586 
587 	default:
588 		error = CUSE_ERR_INVALID;
589 		break;
590 	}
591 
592 	DPRINTF("cuse: Command error = %d for %s\n",
593 	    error, cuse_cmd_str(info.command));
594 
595 	cuse_lock();
596 	TAILQ_REMOVE(&h_cuse_entered, &enter, entry);
597 	cuse_unlock();
598 
599 	/* we ignore any sync command failures */
600 	ioctl(f_cuse, CUSE_IOCTL_SYNC_COMMAND, &error);
601 
602 	return (0);
603 }
604 
605 static struct cuse_dev_entered *
606 cuse_dev_get_entered(void)
607 {
608 	struct cuse_dev_entered *pe;
609 	pthread_t curr = pthread_self();
610 
611 	cuse_lock();
612 	TAILQ_FOREACH(pe, &h_cuse_entered, entry) {
613 		if (pe->thread == curr)
614 			break;
615 	}
616 	cuse_unlock();
617 	return (pe);
618 }
619 
620 void
621 cuse_dev_set_per_file_handle(struct cuse_dev *cdev, void *handle)
622 {
623 	struct cuse_dev_entered *pe;
624 
625 	pe = cuse_dev_get_entered();
626 	if (pe == NULL || pe->cdev != cdev)
627 		return;
628 
629 	pe->per_file_handle = handle;
630 	ioctl(f_cuse, CUSE_IOCTL_SET_PFH, &handle);
631 }
632 
633 void   *
634 cuse_dev_get_per_file_handle(struct cuse_dev *cdev)
635 {
636 	struct cuse_dev_entered *pe;
637 
638 	pe = cuse_dev_get_entered();
639 	if (pe == NULL || pe->cdev != cdev)
640 		return (NULL);
641 
642 	return (pe->per_file_handle);
643 }
644 
645 void
646 cuse_set_local(int val)
647 {
648 	struct cuse_dev_entered *pe;
649 
650 	pe = cuse_dev_get_entered();
651 	if (pe == NULL)
652 		return;
653 
654 	pe->is_local = val;
655 }
656 
657 #ifdef HAVE_DEBUG
658 static const char *
659 cuse_cmd_str(int cmd)
660 {
661 	static const char *str[CUSE_CMD_MAX] = {
662 		[CUSE_CMD_NONE] = "none",
663 		[CUSE_CMD_OPEN] = "open",
664 		[CUSE_CMD_CLOSE] = "close",
665 		[CUSE_CMD_READ] = "read",
666 		[CUSE_CMD_WRITE] = "write",
667 		[CUSE_CMD_IOCTL] = "ioctl",
668 		[CUSE_CMD_POLL] = "poll",
669 		[CUSE_CMD_SIGNAL] = "signal",
670 		[CUSE_CMD_SYNC] = "sync",
671 	};
672 
673 	if ((cmd >= 0) && (cmd < CUSE_CMD_MAX) &&
674 	    (str[cmd] != NULL))
675 		return (str[cmd]);
676 	else
677 		return ("unknown");
678 }
679 
680 #endif
681 
682 int
683 cuse_get_local(void)
684 {
685 	struct cuse_dev_entered *pe;
686 
687 	pe = cuse_dev_get_entered();
688 	if (pe == NULL)
689 		return (0);
690 
691 	return (pe->is_local);
692 }
693 
694 int
695 cuse_copy_out(const void *src, void *user_dst, int len)
696 {
697 	struct cuse_data_chunk info;
698 	struct cuse_dev_entered *pe;
699 	int error;
700 
701 	if ((f_cuse < 0) || (len < 0))
702 		return (CUSE_ERR_INVALID);
703 
704 	pe = cuse_dev_get_entered();
705 	if (pe == NULL)
706 		return (CUSE_ERR_INVALID);
707 
708 	DPRINTF("cuse: copy_out(%p,%p,%d), cmd = %d = %s\n",
709 	    src, user_dst, len, pe->cmd, cuse_cmd_str(pe->cmd));
710 
711 	if (pe->is_local) {
712 		memcpy(user_dst, src, len);
713 	} else {
714 		info.local_ptr = (unsigned long)src;
715 		info.peer_ptr = (unsigned long)user_dst;
716 		info.length = len;
717 
718 		error = ioctl(f_cuse, CUSE_IOCTL_WRITE_DATA, &info);
719 		if (error) {
720 			DPRINTF("cuse: copy_out() error = %d\n", errno);
721 			return (CUSE_ERR_FAULT);
722 		}
723 	}
724 	return (0);
725 }
726 
727 int
728 cuse_copy_in(const void *user_src, void *dst, int len)
729 {
730 	struct cuse_data_chunk info;
731 	struct cuse_dev_entered *pe;
732 	int error;
733 
734 	if ((f_cuse < 0) || (len < 0))
735 		return (CUSE_ERR_INVALID);
736 
737 	pe = cuse_dev_get_entered();
738 	if (pe == NULL)
739 		return (CUSE_ERR_INVALID);
740 
741 	DPRINTF("cuse: copy_in(%p,%p,%d), cmd = %d = %s\n",
742 	    user_src, dst, len, pe->cmd, cuse_cmd_str(pe->cmd));
743 
744 	if (pe->is_local) {
745 		memcpy(dst, user_src, len);
746 	} else {
747 		info.local_ptr = (unsigned long)dst;
748 		info.peer_ptr = (unsigned long)user_src;
749 		info.length = len;
750 
751 		error = ioctl(f_cuse, CUSE_IOCTL_READ_DATA, &info);
752 		if (error) {
753 			DPRINTF("cuse: copy_in() error = %d\n", errno);
754 			return (CUSE_ERR_FAULT);
755 		}
756 	}
757 	return (0);
758 }
759 
760 struct cuse_dev *
761 cuse_dev_get_current(int *pcmd)
762 {
763 	struct cuse_dev_entered *pe;
764 
765 	pe = cuse_dev_get_entered();
766 	if (pe == NULL) {
767 		if (pcmd != NULL)
768 			*pcmd = 0;
769 		return (NULL);
770 	}
771 	if (pcmd != NULL)
772 		*pcmd = pe->cmd;
773 	return (pe->cdev);
774 }
775 
776 int
777 cuse_got_peer_signal(void)
778 {
779 	struct cuse_dev_entered *pe;
780 
781 	pe = cuse_dev_get_entered();
782 	if (pe == NULL)
783 		return (CUSE_ERR_INVALID);
784 
785 	if (pe->got_signal)
786 		return (0);
787 
788 	return (CUSE_ERR_OTHER);
789 }
790 
791 void
792 cuse_poll_wakeup(void)
793 {
794 	int error = 0;
795 
796 	if (f_cuse < 0)
797 		return;
798 
799 	ioctl(f_cuse, CUSE_IOCTL_SELWAKEUP, &error);
800 }
801