123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * ACRN HSM irqfd: use eventfd objects to inject virtual interrupts
- *
- * Copyright (C) 2020 Intel Corporation. All rights reserved.
- *
- * Authors:
- * Shuo Liu <[email protected]>
- * Yakui Zhao <[email protected]>
- */
- #include <linux/eventfd.h>
- #include <linux/file.h>
- #include <linux/poll.h>
- #include <linux/slab.h>
- #include "acrn_drv.h"
- static LIST_HEAD(acrn_irqfd_clients);
- /**
- * struct hsm_irqfd - Properties of HSM irqfd
- * @vm: Associated VM pointer
- * @wait: Entry of wait-queue
- * @shutdown: Async shutdown work
- * @eventfd: Associated eventfd
- * @list: Entry within &acrn_vm.irqfds of irqfds of a VM
- * @pt: Structure for select/poll on the associated eventfd
- * @msi: MSI data
- */
- struct hsm_irqfd {
- struct acrn_vm *vm;
- wait_queue_entry_t wait;
- struct work_struct shutdown;
- struct eventfd_ctx *eventfd;
- struct list_head list;
- poll_table pt;
- struct acrn_msi_entry msi;
- };
- static void acrn_irqfd_inject(struct hsm_irqfd *irqfd)
- {
- struct acrn_vm *vm = irqfd->vm;
- acrn_msi_inject(vm, irqfd->msi.msi_addr,
- irqfd->msi.msi_data);
- }
- static void hsm_irqfd_shutdown(struct hsm_irqfd *irqfd)
- {
- u64 cnt;
- lockdep_assert_held(&irqfd->vm->irqfds_lock);
- /* remove from wait queue */
- list_del_init(&irqfd->list);
- eventfd_ctx_remove_wait_queue(irqfd->eventfd, &irqfd->wait, &cnt);
- eventfd_ctx_put(irqfd->eventfd);
- kfree(irqfd);
- }
- static void hsm_irqfd_shutdown_work(struct work_struct *work)
- {
- struct hsm_irqfd *irqfd;
- struct acrn_vm *vm;
- irqfd = container_of(work, struct hsm_irqfd, shutdown);
- vm = irqfd->vm;
- mutex_lock(&vm->irqfds_lock);
- if (!list_empty(&irqfd->list))
- hsm_irqfd_shutdown(irqfd);
- mutex_unlock(&vm->irqfds_lock);
- }
- /* Called with wqh->lock held and interrupts disabled */
- static int hsm_irqfd_wakeup(wait_queue_entry_t *wait, unsigned int mode,
- int sync, void *key)
- {
- unsigned long poll_bits = (unsigned long)key;
- struct hsm_irqfd *irqfd;
- struct acrn_vm *vm;
- irqfd = container_of(wait, struct hsm_irqfd, wait);
- vm = irqfd->vm;
- if (poll_bits & POLLIN)
- /* An event has been signaled, inject an interrupt */
- acrn_irqfd_inject(irqfd);
- if (poll_bits & POLLHUP)
- /* Do shutdown work in thread to hold wqh->lock */
- queue_work(vm->irqfd_wq, &irqfd->shutdown);
- return 0;
- }
- static void hsm_irqfd_poll_func(struct file *file, wait_queue_head_t *wqh,
- poll_table *pt)
- {
- struct hsm_irqfd *irqfd;
- irqfd = container_of(pt, struct hsm_irqfd, pt);
- add_wait_queue(wqh, &irqfd->wait);
- }
- /*
- * Assign an eventfd to a VM and create a HSM irqfd associated with the
- * eventfd. The properties of the HSM irqfd are built from a &struct
- * acrn_irqfd.
- */
- static int acrn_irqfd_assign(struct acrn_vm *vm, struct acrn_irqfd *args)
- {
- struct eventfd_ctx *eventfd = NULL;
- struct hsm_irqfd *irqfd, *tmp;
- __poll_t events;
- struct fd f;
- int ret = 0;
- irqfd = kzalloc(sizeof(*irqfd), GFP_KERNEL);
- if (!irqfd)
- return -ENOMEM;
- irqfd->vm = vm;
- memcpy(&irqfd->msi, &args->msi, sizeof(args->msi));
- INIT_LIST_HEAD(&irqfd->list);
- INIT_WORK(&irqfd->shutdown, hsm_irqfd_shutdown_work);
- f = fdget(args->fd);
- if (!f.file) {
- ret = -EBADF;
- goto out;
- }
- eventfd = eventfd_ctx_fileget(f.file);
- if (IS_ERR(eventfd)) {
- ret = PTR_ERR(eventfd);
- goto fail;
- }
- irqfd->eventfd = eventfd;
- /*
- * Install custom wake-up handling to be notified whenever underlying
- * eventfd is signaled.
- */
- init_waitqueue_func_entry(&irqfd->wait, hsm_irqfd_wakeup);
- init_poll_funcptr(&irqfd->pt, hsm_irqfd_poll_func);
- mutex_lock(&vm->irqfds_lock);
- list_for_each_entry(tmp, &vm->irqfds, list) {
- if (irqfd->eventfd != tmp->eventfd)
- continue;
- ret = -EBUSY;
- mutex_unlock(&vm->irqfds_lock);
- goto fail;
- }
- list_add_tail(&irqfd->list, &vm->irqfds);
- mutex_unlock(&vm->irqfds_lock);
- /* Check the pending event in this stage */
- events = vfs_poll(f.file, &irqfd->pt);
- if (events & EPOLLIN)
- acrn_irqfd_inject(irqfd);
- fdput(f);
- return 0;
- fail:
- if (eventfd && !IS_ERR(eventfd))
- eventfd_ctx_put(eventfd);
- fdput(f);
- out:
- kfree(irqfd);
- return ret;
- }
- static int acrn_irqfd_deassign(struct acrn_vm *vm,
- struct acrn_irqfd *args)
- {
- struct hsm_irqfd *irqfd, *tmp;
- struct eventfd_ctx *eventfd;
- eventfd = eventfd_ctx_fdget(args->fd);
- if (IS_ERR(eventfd))
- return PTR_ERR(eventfd);
- mutex_lock(&vm->irqfds_lock);
- list_for_each_entry_safe(irqfd, tmp, &vm->irqfds, list) {
- if (irqfd->eventfd == eventfd) {
- hsm_irqfd_shutdown(irqfd);
- break;
- }
- }
- mutex_unlock(&vm->irqfds_lock);
- eventfd_ctx_put(eventfd);
- return 0;
- }
- int acrn_irqfd_config(struct acrn_vm *vm, struct acrn_irqfd *args)
- {
- int ret;
- if (args->flags & ACRN_IRQFD_FLAG_DEASSIGN)
- ret = acrn_irqfd_deassign(vm, args);
- else
- ret = acrn_irqfd_assign(vm, args);
- return ret;
- }
- int acrn_irqfd_init(struct acrn_vm *vm)
- {
- INIT_LIST_HEAD(&vm->irqfds);
- mutex_init(&vm->irqfds_lock);
- vm->irqfd_wq = alloc_workqueue("acrn_irqfd-%u", 0, 0, vm->vmid);
- if (!vm->irqfd_wq)
- return -ENOMEM;
- dev_dbg(acrn_dev.this_device, "VM %u irqfd init.\n", vm->vmid);
- return 0;
- }
- void acrn_irqfd_deinit(struct acrn_vm *vm)
- {
- struct hsm_irqfd *irqfd, *next;
- dev_dbg(acrn_dev.this_device, "VM %u irqfd deinit.\n", vm->vmid);
- destroy_workqueue(vm->irqfd_wq);
- mutex_lock(&vm->irqfds_lock);
- list_for_each_entry_safe(irqfd, next, &vm->irqfds, list)
- hsm_irqfd_shutdown(irqfd);
- mutex_unlock(&vm->irqfds_lock);
- }
|