When a physical I/O device is assigned to a virtual machine through facilities like VFIO and KVM, the interrupt for the device generally bounces through the host system before being injected into the VM. However, hardware technologies exist that often allow the host to be bypassed for some of these scenarios. Intel Posted Interrupts allow the specified physical edge interrupts to be directly injected into a guest when delivered to a physical processor while the vCPU is running. ARM IRQ Forwarding allows forwarded physical interrupts to be directly deactivated by the guest.
The IRQ bypass manager here is meant to provide the shim to connect interrupt producers, generally the host physical device driver, with interrupt consumers, generally the hypervisor, in order to configure these bypass mechanism. To do this, we base the connection on a shared, opaque token. For KVM-VFIO this is expected to be an eventfd_ctx since this is the connection we already use to connect an eventfd to an irqfd on the in-kernel path. When a producer and consumer with matching tokens is found, callbacks via both registered participants allow the bypass facilities to be automatically enabled.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/* * Theory of operation * * The IRQ bypass manager is a simple set of lists and callbacks that allows * IRQ producers (ex. physical interrupt sources) to be matched to IRQ * consumers (ex. virtualization hardware that allows IRQ bypass or offload) * via a shared token (ex. eventfd_ctx). Producers and consumers register * independently. When a token match is found, the optional @stop callback * will be called for each participant. The pair will then be connected via * the @add_* callbacks, and finally the optional @start callback will allow * any final coordination. When either participant is unregistered, the * process is repeated using the @del_* callbacks in place of the @add_* * callbacks. Match tokens must be unique per producer/consumer, 1:N pairings * are not supported. */
/** * irq_bypass_register_producer - register IRQ bypass producer * @producer: pointer to producer structure * * Add the provided IRQ producer to the list of producers and connect * with any matching token found on the IRQ consumers list. */ intirq_bypass_register_producer(struct irq_bypass_producer *producer) { structirq_bypass_producer *tmp; structirq_bypass_consumer *consumer;
/** * irq_bypass_register_consumer - register IRQ bypass consumer * @consumer: pointer to consumer structure * * Add the provided IRQ consumer to the list of consumers and connect * with any matching token found on the IRQ producer list. */ intirq_bypass_register_consumer(struct irq_bypass_consumer *consumer) { structirq_bypass_consumer *tmp; structirq_bypass_producer *producer;
if (!consumer->token || !consumer->add_producer || !consumer->del_producer) return -EINVAL;
/* * vmx_update_pi_irte - set IRTE for Posted-Interrupts * * @kvm: kvm * @host_irq: host irq of the interrupt * @guest_irq: gsi of the interrupt * @set: set or unset PI * returns 0 on success, < 0 on failure */ staticintvmx_update_pi_irte(struct kvm *kvm, unsignedint host_irq, uint32_t guest_irq, boolset)