mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-25 19:32:47 +00:00 
			
		
		
		
	 82e2756897
			
		
	
	
		82e2756897
		
	
	
	
	
		
			
			Otherwise the call to event_notifier_set() is a nop, which causes
the SLOF firmware on POWER to hang when booting from a virtio-scsi
device:
virtio_scsi_dataplane_start()
 virtio_scsi_vring_init()
  virtio_bus_set_host_notifier() <- assign == true
   event_notifier_init() <- active == 1
    event_notifier_set() <- fails right away if !e->initialized
Fixes: e34e47eb28 ("event_notifier: handle initialization failure better")
Cc: mlevitsk@redhat.com
Signed-off-by: Greg Kurz <groug@kaod.org>
Message-Id: <20210216120247.1293569-1-groug@kaod.org>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
		
	
			
		
			
				
	
	
		
			141 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * event notifier support
 | |
|  *
 | |
|  * Copyright Red Hat, Inc. 2010
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Michael S. Tsirkin <mst@redhat.com>
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| #include "qemu-common.h"
 | |
| #include "qemu/cutils.h"
 | |
| #include "qemu/event_notifier.h"
 | |
| #include "qemu/main-loop.h"
 | |
| 
 | |
| #ifdef CONFIG_EVENTFD
 | |
| #include <sys/eventfd.h>
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_EVENTFD
 | |
| /*
 | |
|  * Initialize @e with existing file descriptor @fd.
 | |
|  * @fd must be a genuine eventfd object, emulation with pipe won't do.
 | |
|  */
 | |
| void event_notifier_init_fd(EventNotifier *e, int fd)
 | |
| {
 | |
|     e->rfd = fd;
 | |
|     e->wfd = fd;
 | |
|     e->initialized = true;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| int event_notifier_init(EventNotifier *e, int active)
 | |
| {
 | |
|     int fds[2];
 | |
|     int ret;
 | |
| 
 | |
| #ifdef CONFIG_EVENTFD
 | |
|     ret = eventfd(0, EFD_NONBLOCK | EFD_CLOEXEC);
 | |
| #else
 | |
|     ret = -1;
 | |
|     errno = ENOSYS;
 | |
| #endif
 | |
|     if (ret >= 0) {
 | |
|         e->rfd = e->wfd = ret;
 | |
|     } else {
 | |
|         if (errno != ENOSYS) {
 | |
|             return -errno;
 | |
|         }
 | |
|         if (qemu_pipe(fds) < 0) {
 | |
|             return -errno;
 | |
|         }
 | |
|         ret = fcntl_setfl(fds[0], O_NONBLOCK);
 | |
|         if (ret < 0) {
 | |
|             ret = -errno;
 | |
|             goto fail;
 | |
|         }
 | |
|         ret = fcntl_setfl(fds[1], O_NONBLOCK);
 | |
|         if (ret < 0) {
 | |
|             ret = -errno;
 | |
|             goto fail;
 | |
|         }
 | |
|         e->rfd = fds[0];
 | |
|         e->wfd = fds[1];
 | |
|     }
 | |
|     e->initialized = true;
 | |
|     if (active) {
 | |
|         event_notifier_set(e);
 | |
|     }
 | |
|     return 0;
 | |
| 
 | |
| fail:
 | |
|     close(fds[0]);
 | |
|     close(fds[1]);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| void event_notifier_cleanup(EventNotifier *e)
 | |
| {
 | |
|     if (!e->initialized) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (e->rfd != e->wfd) {
 | |
|         close(e->rfd);
 | |
|     }
 | |
| 
 | |
|     e->rfd = -1;
 | |
|     close(e->wfd);
 | |
|     e->wfd = -1;
 | |
|     e->initialized = false;
 | |
| }
 | |
| 
 | |
| int event_notifier_get_fd(const EventNotifier *e)
 | |
| {
 | |
|     return e->rfd;
 | |
| }
 | |
| 
 | |
| int event_notifier_set(EventNotifier *e)
 | |
| {
 | |
|     static const uint64_t value = 1;
 | |
|     ssize_t ret;
 | |
| 
 | |
|     if (!e->initialized) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     do {
 | |
|         ret = write(e->wfd, &value, sizeof(value));
 | |
|     } while (ret < 0 && errno == EINTR);
 | |
| 
 | |
|     /* EAGAIN is fine, a read must be pending.  */
 | |
|     if (ret < 0 && errno != EAGAIN) {
 | |
|         return -errno;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int event_notifier_test_and_clear(EventNotifier *e)
 | |
| {
 | |
|     int value;
 | |
|     ssize_t len;
 | |
|     char buffer[512];
 | |
| 
 | |
|     if (!e->initialized) {
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     /* Drain the notify pipe.  For eventfd, only 8 bytes will be read.  */
 | |
|     value = 0;
 | |
|     do {
 | |
|         len = read(e->rfd, buffer, sizeof(buffer));
 | |
|         value |= (len > 0);
 | |
|     } while ((len == -1 && errno == EINTR) || len == sizeof(buffer));
 | |
| 
 | |
|     return value;
 | |
| }
 |