From mboxrd@z Thu Jan 1 00:00:00 1970 From: Gregory Haskins Subject: [PATCH v7 1/2] qemu-kvm: add irqfd support Date: Tue, 12 May 2009 14:32:26 -0400 Message-ID: <20090512183226.26356.46296.stgit@dev.haskins.net> References: <20090512183059.26356.94857.stgit@dev.haskins.net> Mime-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: 7bit Cc: avi@redhat.com To: kvm@vger.kernel.org Return-path: Received: from victor.provo.novell.com ([137.65.250.26]:53396 "EHLO victor.provo.novell.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751506AbZELSck (ORCPT ); Tue, 12 May 2009 14:32:40 -0400 In-Reply-To: <20090512183059.26356.94857.stgit@dev.haskins.net> Sender: kvm-owner@vger.kernel.org List-ID: irqfd lets you create an eventfd based file-desriptor to inject interrupts to a kvm guest. We associate one gsi per fd for fine-grained routing. Signed-off-by: Gregory Haskins --- kvm/libkvm/libkvm.c | 66 +++++++++++++++++++++++++++++++++++++++++++++++++++ kvm/libkvm/libkvm.h | 25 +++++++++++++++++++ 2 files changed, 91 insertions(+), 0 deletions(-) diff --git a/kvm/libkvm/libkvm.c b/kvm/libkvm/libkvm.c index ba0a5d1..74a21a2 100644 --- a/kvm/libkvm/libkvm.c +++ b/kvm/libkvm/libkvm.c @@ -34,6 +34,7 @@ #include #include #include +#include #include #include "libkvm.h" @@ -1444,3 +1445,68 @@ int kvm_assign_set_msix_entry(kvm_context_t kvm, return ret; } #endif + +#ifdef KVM_CAP_EVENTFD +static int _assign_irqfd(kvm_context_t kvm, int fd, int gsi, int flags) +{ + int r; + struct kvm_irqfd data = { + .fd = fd, + .gsi = gsi, + .flags = flags, + }; + + r = ioctl(kvm->vm_fd, KVM_ASSIGN_IRQFD, &data); + if (r == -1) + r = -errno; + return r; +} + +int kvm_create_irqfd(kvm_context_t kvm, int gsi, int flags) +{ + int r; + int fd; + + if (!kvm_check_extension(kvm, KVM_CAP_EVENTFD)) + return -ENOENT; + + fd = eventfd(0, 0); + if (fd < 0) + return -errno; + + r = _assign_irqfd(kvm, fd, gsi, flags); + if (r < 0) { + close(fd); + return -errno; + } + + return fd; +} + +int kvm_destroy_irqfd(kvm_context_t kvm, int fd, int flags) +{ + int r; + __u32 data = fd; + + r = ioctl(kvm->vm_fd, KVM_DEASSIGN_IRQFD, &data); + if (r == -1) + r = -errno; + + close(fd); + + return r; +} + +#else /* KVM_CAP_EVENTFD */ + +int kvm_create_irqfd(kvm_context_t kvm, int gsi, int flags) +{ + return -ENOENT; +} + +int kvm_destroy_irqfd(kvm_context_t kvm, int fd, int flags) +{ + return -ENOENT; +} + +#endif /* KVM_CAP_EVENTFD */ diff --git a/kvm/libkvm/libkvm.h b/kvm/libkvm/libkvm.h index 4821a1e..322b4cd 100644 --- a/kvm/libkvm/libkvm.h +++ b/kvm/libkvm/libkvm.h @@ -856,6 +856,31 @@ int kvm_commit_irq_routes(kvm_context_t kvm); */ int kvm_get_irq_route_gsi(kvm_context_t kvm); +/*! + * \brief Create a file descriptor for injecting interrupts + * + * Creates an eventfd based file-descriptor that maps to a specific GSI + * in the guest. eventfd compliant signaling (write() from userspace, or + * eventfd_signal() from kernelspace) will cause the GSI to inject + * itself into the guest at the next available window. + * + * \param kvm Pointer to the current kvm_context + * \param gsi GSI to assign to this fd + * \param flags reserved, must be zero + */ +int kvm_create_irqfd(kvm_context_t kvm, int gsi, int flags); + +/*! + * \brief Destroy an irqfd file descriptor + * + * Destroys a file descriptor previously opened with kvm_create_irqfd() + * + * \param kvm Pointer to the current kvm_context + * \param fd fd to close + * \param flags reserved, must be zero + */ +int kvm_destroy_irqfd(kvm_context_t kvm, int fd, int flags); + #ifdef KVM_CAP_DEVICE_MSIX int kvm_assign_set_msix_nr(kvm_context_t kvm, struct kvm_assigned_msix_nr *msix_nr);