| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * signalfd/eventfd compatibility | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright IBM, Corp. 2008 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Authors: | 
					
						
							|  |  |  |  *  Anthony Liguori   <aliguori@us.ibm.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This work is licensed under the terms of the GNU GPL, version 2.  See | 
					
						
							|  |  |  |  * the COPYING file in the top-level directory. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-13 17:44:23 +01:00
										 |  |  |  * Contributions after 2012-01-13 are licensed under the terms of the | 
					
						
							|  |  |  |  * GNU GPL, version 2 or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-29 17:49:55 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2013-05-02 10:21:18 +02:00
										 |  |  | #include "qemu/thread.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <sys/syscall.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct sigfd_compat_info | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     sigset_t mask; | 
					
						
							|  |  |  |     int fd; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void *sigwait_compat(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct sigfd_compat_info *info = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |     while (1) { | 
					
						
							|  |  |  |         int sig; | 
					
						
							|  |  |  |         int err; | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |         err = sigwait(&info->mask, &sig); | 
					
						
							|  |  |  |         if (err != 0) { | 
					
						
							|  |  |  |             if (errno == EINTR) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             struct qemu_signalfd_siginfo buffer; | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |             size_t offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |             memset(&buffer, 0, sizeof(buffer)); | 
					
						
							|  |  |  |             buffer.ssi_signo = sig; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |             while (offset < sizeof(buffer)) { | 
					
						
							|  |  |  |                 ssize_t len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |                 len = write(info->fd, (char *)&buffer + offset, | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |                             sizeof(buffer) - offset); | 
					
						
							|  |  |  |                 if (len == -1 && errno == EINTR) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (len <= 0) { | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |                     return NULL; | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 offset += len; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-18 14:17:16 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int qemu_signalfd_compat(const sigset_t *mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct sigfd_compat_info *info; | 
					
						
							| 
									
										
										
										
											2013-05-02 10:21:18 +02:00
										 |  |  |     QemuThread thread; | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  |     int fds[2]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     info = malloc(sizeof(*info)); | 
					
						
							|  |  |  |     if (info == NULL) { | 
					
						
							|  |  |  |         errno = ENOMEM; | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pipe(fds) == -1) { | 
					
						
							|  |  |  |         free(info); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_set_cloexec(fds[0]); | 
					
						
							|  |  |  |     qemu_set_cloexec(fds[1]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memcpy(&info->mask, mask, sizeof(*mask)); | 
					
						
							|  |  |  |     info->fd = fds[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-30 10:20:32 +00:00
										 |  |  |     qemu_thread_create(&thread, "signalfd_compat", sigwait_compat, info, | 
					
						
							|  |  |  |                        QEMU_THREAD_DETACHED); | 
					
						
							| 
									
										
										
										
											2010-10-11 15:31:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return fds[0]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qemu_signalfd(const sigset_t *mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if defined(CONFIG_SIGNALFD)
 | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = syscall(SYS_signalfd, -1, mask, _NSIG / 8); | 
					
						
							|  |  |  |     if (ret != -1) { | 
					
						
							|  |  |  |         qemu_set_cloexec(ret); | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return qemu_signalfd_compat(mask); | 
					
						
							|  |  |  | } |