Quellcodebibliothek Statistik Leitseite products/sources/formale Sprachen/C/Linux/virt/lib/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 5 kB image not shown  

Quelle  irqbypass.c   Sprache: C

 
// 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.
 */


#include <linux/irqbypass.h>
#include <linux/list.h>
#include <linux/module.h>
#include <linux/mutex.h>

MODULE_LICENSE("GPL v2");
MODULE_DESCRIPTION("IRQ bypass manager utility module");

static DEFINE_XARRAY(producers);
static DEFINE_XARRAY(consumers);
static DEFINE_MUTEX(lock);

/* @lock must be held when calling connect */
static int __connect(struct irq_bypass_producer *prod,
       struct irq_bypass_consumer *cons)
{
 int ret = 0;

 if (prod->stop)
  prod->stop(prod);
 if (cons->stop)
  cons->stop(cons);

 if (prod->add_consumer)
  ret = prod->add_consumer(prod, cons);

 if (!ret) {
  ret = cons->add_producer(cons, prod);
  if (ret && prod->del_consumer)
   prod->del_consumer(prod, cons);
 }

 if (cons->start)
  cons->start(cons);
 if (prod->start)
  prod->start(prod);

 if (!ret) {
  prod->consumer = cons;
  cons->producer = prod;
 }
 return ret;
}

/* @lock must be held when calling disconnect */
static void __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)
{
 unsigned long index = (unsigned long)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;
  }
 }

 producer->eventfd = eventfd;
 return 0;
}
EXPORT_SYMBOL_GPL(irq_bypass_register_producer);

/**
 * 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)
{
 unsigned long index = (unsigned long)producer->eventfd;

 if (!producer->eventfd)
  return;

 guard(mutex)(&lock);

 if (producer->consumer)
  __disconnect(producer, producer->consumer);

 WARN_ON_ONCE(xa_erase(&producers, index) != producer);
 producer->eventfd = NULL;
}
EXPORT_SYMBOL_GPL(irq_bypass_unregister_producer);

/**
 * 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)
{
 unsigned long index = (unsigned long)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;
  }
 }

 consumer->eventfd = eventfd;
 return 0;
}
EXPORT_SYMBOL_GPL(irq_bypass_register_consumer);

/**
 * 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)
{
 unsigned long index = (unsigned long)consumer->eventfd;

 if (!consumer->eventfd)
  return;

 guard(mutex)(&lock);

 if (consumer->producer)
  __disconnect(consumer->producer, consumer);

 WARN_ON_ONCE(xa_erase(&consumers, index) != consumer);
 consumer->eventfd = NULL;
}
EXPORT_SYMBOL_GPL(irq_bypass_unregister_consumer);

Messung V0.5
C=94 H=96 G=94

¤ Dauer der Verarbeitung: 0.4 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

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.