2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-09-18 22:29:39 +08:00
/*
* 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 LIST_HEAD ( producers ) ;
static LIST_HEAD ( 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 ) ;
2020-07-31 14:55:33 +08:00
if ( ret )
goto err_add_consumer ;
ret = cons - > add_producer ( cons , prod ) ;
if ( ret )
goto err_add_producer ;
2015-09-18 22:29:39 +08:00
if ( cons - > start )
cons - > start ( cons ) ;
if ( prod - > start )
prod - > start ( prod ) ;
2020-07-31 14:55:33 +08:00
err_add_producer :
if ( prod - > del_consumer )
prod - > del_consumer ( prod , cons ) ;
err_add_consumer :
2015-09-18 22:29:39 +08:00
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 ) ;
}
/**
* 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 .
*/
int irq_bypass_register_producer ( struct irq_bypass_producer * producer )
{
struct irq_bypass_producer * tmp ;
struct irq_bypass_consumer * consumer ;
2019-12-06 10:53:53 +08:00
int ret ;
2015-09-18 22:29:39 +08:00
2016-05-05 11:58:29 -06:00
if ( ! producer - > token )
return - EINVAL ;
2015-09-18 22:29:39 +08:00
might_sleep ( ) ;
if ( ! try_module_get ( THIS_MODULE ) )
return - ENODEV ;
mutex_lock ( & lock ) ;
list_for_each_entry ( tmp , & producers , node ) {
if ( tmp - > token = = producer - > token ) {
2019-12-06 10:53:53 +08:00
ret = - EBUSY ;
goto out_err ;
2015-09-18 22:29:39 +08:00
}
}
list_for_each_entry ( consumer , & consumers , node ) {
if ( consumer - > token = = producer - > token ) {
2019-12-06 10:53:53 +08:00
ret = __connect ( producer , consumer ) ;
if ( ret )
goto out_err ;
2015-09-18 22:29:39 +08:00
break ;
}
}
list_add ( & producer - > node , & producers ) ;
mutex_unlock ( & lock ) ;
return 0 ;
2019-12-06 10:53:53 +08:00
out_err :
mutex_unlock ( & lock ) ;
module_put ( THIS_MODULE ) ;
return ret ;
2015-09-18 22:29:39 +08:00
}
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 from the list of producers
* and disconnect it from any connected IRQ consumer .
*/
void irq_bypass_unregister_producer ( struct irq_bypass_producer * producer )
{
struct irq_bypass_producer * tmp ;
struct irq_bypass_consumer * consumer ;
2016-05-05 11:58:29 -06:00
if ( ! producer - > token )
return ;
2015-09-18 22:29:39 +08:00
might_sleep ( ) ;
if ( ! try_module_get ( THIS_MODULE ) )
return ; /* nothing in the list anyway */
mutex_lock ( & lock ) ;
list_for_each_entry ( tmp , & producers , node ) {
if ( tmp - > token ! = producer - > token )
continue ;
list_for_each_entry ( consumer , & consumers , node ) {
if ( consumer - > token = = producer - > token ) {
__disconnect ( producer , consumer ) ;
break ;
}
}
list_del ( & producer - > node ) ;
module_put ( THIS_MODULE ) ;
break ;
}
mutex_unlock ( & lock ) ;
module_put ( THIS_MODULE ) ;
}
EXPORT_SYMBOL_GPL ( irq_bypass_unregister_producer ) ;
/**
* 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 .
*/
int irq_bypass_register_consumer ( struct irq_bypass_consumer * consumer )
{
struct irq_bypass_consumer * tmp ;
struct irq_bypass_producer * producer ;
2019-12-06 10:53:52 +08:00
int ret ;
2015-09-18 22:29:39 +08:00
2016-05-05 11:58:29 -06:00
if ( ! consumer - > token | |
! consumer - > add_producer | | ! consumer - > del_producer )
2015-09-18 22:29:39 +08:00
return - EINVAL ;
might_sleep ( ) ;
if ( ! try_module_get ( THIS_MODULE ) )
return - ENODEV ;
mutex_lock ( & lock ) ;
list_for_each_entry ( tmp , & consumers , node ) {
2017-01-05 17:39:42 -08:00
if ( tmp - > token = = consumer - > token | | tmp = = consumer ) {
2019-12-06 10:53:52 +08:00
ret = - EBUSY ;
goto out_err ;
2015-09-18 22:29:39 +08:00
}
}
list_for_each_entry ( producer , & producers , node ) {
if ( producer - > token = = consumer - > token ) {
2019-12-06 10:53:52 +08:00
ret = __connect ( producer , consumer ) ;
if ( ret )
goto out_err ;
2015-09-18 22:29:39 +08:00
break ;
}
}
list_add ( & consumer - > node , & consumers ) ;
mutex_unlock ( & lock ) ;
return 0 ;
2019-12-06 10:53:52 +08:00
out_err :
mutex_unlock ( & lock ) ;
module_put ( THIS_MODULE ) ;
return ret ;
2015-09-18 22:29:39 +08:00
}
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 from the list of consumers
* and disconnect it from any connected IRQ producer .
*/
void irq_bypass_unregister_consumer ( struct irq_bypass_consumer * consumer )
{
struct irq_bypass_consumer * tmp ;
struct irq_bypass_producer * producer ;
2016-05-05 11:58:29 -06:00
if ( ! consumer - > token )
return ;
2015-09-18 22:29:39 +08:00
might_sleep ( ) ;
if ( ! try_module_get ( THIS_MODULE ) )
return ; /* nothing in the list anyway */
mutex_lock ( & lock ) ;
list_for_each_entry ( tmp , & consumers , node ) {
2017-01-05 17:39:42 -08:00
if ( tmp ! = consumer )
2015-09-18 22:29:39 +08:00
continue ;
list_for_each_entry ( producer , & producers , node ) {
if ( producer - > token = = consumer - > token ) {
__disconnect ( producer , consumer ) ;
break ;
}
}
list_del ( & consumer - > node ) ;
module_put ( THIS_MODULE ) ;
break ;
}
mutex_unlock ( & lock ) ;
module_put ( THIS_MODULE ) ;
}
EXPORT_SYMBOL_GPL ( irq_bypass_unregister_consumer ) ;