// SPDX-License-Identifier: GPL-2.0-only /* * IRQ offload/bypass manager * * Copyright (C) 2015 Red Hat, Inc. * Copyright (c) 2015 Linaro Ltd. * * Various virtualization hardware acceleration techniques allow bypassing or * offloading interrupts received from devices around the host kernel. Posted * Interrupts on Intel VT-d systems can allow interrupts to be received * directly by a virtual machine. ARM IRQ Forwarding allows forwarded physical * interrupts to be directly deactivated by the guest. This manager allows * interrupt producers and consumers to find each other to enable this sort of * bypass.
*/
/* @lock must be held when calling disconnect */ staticvoid __disconnect(struct irq_bypass_producer *prod, struct irq_bypass_consumer *cons)
{ if (prod->stop)
prod->stop(prod); if (cons->stop)
cons->stop(cons);
cons->del_producer(cons, prod);
if (prod->del_consumer)
prod->del_consumer(prod, cons);
if (cons->start)
cons->start(cons); if (prod->start)
prod->start(prod);
prod->consumer = NULL;
cons->producer = NULL;
}
/** * irq_bypass_register_producer - register IRQ bypass producer * @producer: pointer to producer structure * @eventfd: pointer to the eventfd context associated with the producer * @irq: Linux IRQ number of the underlying producer device * * Add the provided IRQ producer to the set of producers and connect with the * consumer with a matching eventfd, if one exists.
*/ int irq_bypass_register_producer(struct irq_bypass_producer *producer, struct eventfd_ctx *eventfd, int irq)
{ unsignedlong index = (unsignedlong)eventfd; struct irq_bypass_consumer *consumer; int ret;
if (WARN_ON_ONCE(producer->eventfd)) return -EINVAL;
producer->irq = irq;
guard(mutex)(&lock);
ret = xa_insert(&producers, index, producer, GFP_KERNEL); if (ret) return ret;
consumer = xa_load(&consumers, index); if (consumer) {
ret = __connect(producer, consumer); if (ret) {
WARN_ON_ONCE(xa_erase(&producers, index) != producer); return ret;
}
}
/** * irq_bypass_unregister_producer - unregister IRQ bypass producer * @producer: pointer to producer structure * * Remove a previously registered IRQ producer (note, it's safe to call this * even if registration was unsuccessful). Disconnect from the associated * consumer, if one exists.
*/ void irq_bypass_unregister_producer(struct irq_bypass_producer *producer)
{ unsignedlong index = (unsignedlong)producer->eventfd;
if (!producer->eventfd) return;
guard(mutex)(&lock);
if (producer->consumer)
__disconnect(producer, producer->consumer);
/** * irq_bypass_register_consumer - register IRQ bypass consumer * @consumer: pointer to consumer structure * @eventfd: pointer to the eventfd context associated with the consumer * * Add the provided IRQ consumer to the set of consumers and connect with the * producer with a matching eventfd, if one exists.
*/ int irq_bypass_register_consumer(struct irq_bypass_consumer *consumer, struct eventfd_ctx *eventfd)
{ unsignedlong index = (unsignedlong)eventfd; struct irq_bypass_producer *producer; int ret;
if (WARN_ON_ONCE(consumer->eventfd)) return -EINVAL;
if (!consumer->add_producer || !consumer->del_producer) return -EINVAL;
guard(mutex)(&lock);
ret = xa_insert(&consumers, index, consumer, GFP_KERNEL); if (ret) return ret;
producer = xa_load(&producers, index); if (producer) {
ret = __connect(producer, consumer); if (ret) {
WARN_ON_ONCE(xa_erase(&consumers, index) != consumer); return ret;
}
}
/** * irq_bypass_unregister_consumer - unregister IRQ bypass consumer * @consumer: pointer to consumer structure * * Remove a previously registered IRQ consumer (note, it's safe to call this * even if registration was unsuccessful). Disconnect from the associated * producer, if one exists.
*/ void irq_bypass_unregister_consumer(struct irq_bypass_consumer *consumer)
{ unsignedlong index = (unsignedlong)consumer->eventfd;
if (!consumer->eventfd) return;
guard(mutex)(&lock);
if (consumer->producer)
__disconnect(consumer->producer, consumer);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.