2006-01-02 19:04:38 +01:00
/*
* net / tipc / handler . c : TIPC signal handling
2007-02-09 23:25:21 +09:00
*
2006-01-11 19:14:19 +01:00
* Copyright ( c ) 2000 - 2006 , Ericsson AB
2006-01-02 19:04:38 +01:00
* Copyright ( c ) 2005 , Wind River Systems
* All rights reserved .
*
2006-01-11 13:30:43 +01:00
* Redistribution and use in source and binary forms , with or without
2006-01-02 19:04:38 +01:00
* modification , are permitted provided that the following conditions are met :
*
2006-01-11 13:30:43 +01:00
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission .
2006-01-02 19:04:38 +01:00
*
2006-01-11 13:30:43 +01:00
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) version 2 as published by the Free
* Software Foundation .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS "
* AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR
* CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS
* INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN
* CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE )
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE
2006-01-02 19:04:38 +01:00
* POSSIBILITY OF SUCH DAMAGE .
*/
# include "core.h"
struct queue_item {
struct list_head next_signal ;
void ( * handler ) ( unsigned long ) ;
unsigned long data ;
} ;
2006-12-06 20:33:20 -08:00
static struct kmem_cache * tipc_queue_item_cache ;
2006-01-02 19:04:38 +01:00
static struct list_head signal_queue_head ;
2006-06-27 02:53:55 -07:00
static DEFINE_SPINLOCK ( qitem_lock ) ;
2012-08-16 12:09:12 +00:00
static int handler_enabled __read_mostly ;
2006-01-02 19:04:38 +01:00
static void process_signal_queue ( unsigned long dummy ) ;
static DECLARE_TASKLET_DISABLED ( tipc_tasklet , process_signal_queue , 0 ) ;
2006-01-18 00:38:21 +01:00
unsigned int tipc_k_signal ( Handler routine , unsigned long argument )
2006-01-02 19:04:38 +01:00
{
struct queue_item * item ;
2013-12-09 22:54:47 -08:00
spin_lock_bh ( & qitem_lock ) ;
2006-01-02 19:04:38 +01:00
if ( ! handler_enabled ) {
2013-12-09 22:54:47 -08:00
spin_unlock_bh ( & qitem_lock ) ;
2006-01-02 19:04:38 +01:00
return - ENOPROTOOPT ;
}
item = kmem_cache_alloc ( tipc_queue_item_cache , GFP_ATOMIC ) ;
if ( ! item ) {
2012-06-29 00:16:37 -04:00
pr_err ( " Signal queue out of memory \n " ) ;
2006-01-02 19:04:38 +01:00
spin_unlock_bh ( & qitem_lock ) ;
return - ENOMEM ;
}
item - > handler = routine ;
item - > data = argument ;
list_add_tail ( & item - > next_signal , & signal_queue_head ) ;
spin_unlock_bh ( & qitem_lock ) ;
tasklet_schedule ( & tipc_tasklet ) ;
return 0 ;
}
static void process_signal_queue ( unsigned long dummy )
{
struct queue_item * __volatile__ item ;
struct list_head * l , * n ;
spin_lock_bh ( & qitem_lock ) ;
list_for_each_safe ( l , n , & signal_queue_head ) {
item = list_entry ( l , struct queue_item , next_signal ) ;
list_del ( & item - > next_signal ) ;
spin_unlock_bh ( & qitem_lock ) ;
item - > handler ( item - > data ) ;
spin_lock_bh ( & qitem_lock ) ;
kmem_cache_free ( tipc_queue_item_cache , item ) ;
}
spin_unlock_bh ( & qitem_lock ) ;
}
2006-01-18 00:38:21 +01:00
int tipc_handler_start ( void )
2006-01-02 19:04:38 +01:00
{
2007-02-09 23:25:21 +09:00
tipc_queue_item_cache =
2006-01-02 19:04:38 +01:00
kmem_cache_create ( " tipc_queue_items " , sizeof ( struct queue_item ) ,
2007-07-20 10:11:58 +09:00
0 , SLAB_HWCACHE_ALIGN , NULL ) ;
2006-01-02 19:04:38 +01:00
if ( ! tipc_queue_item_cache )
return - ENOMEM ;
INIT_LIST_HEAD ( & signal_queue_head ) ;
tasklet_enable ( & tipc_tasklet ) ;
handler_enabled = 1 ;
return 0 ;
}
2006-01-18 00:38:21 +01:00
void tipc_handler_stop ( void )
2006-01-02 19:04:38 +01:00
{
struct list_head * l , * n ;
2007-02-09 23:25:21 +09:00
struct queue_item * item ;
2006-01-02 19:04:38 +01:00
2013-12-09 22:54:47 -08:00
spin_lock_bh ( & qitem_lock ) ;
if ( ! handler_enabled ) {
spin_unlock_bh ( & qitem_lock ) ;
2006-01-02 19:04:38 +01:00
return ;
2013-12-09 22:54:47 -08:00
}
2006-01-02 19:04:38 +01:00
handler_enabled = 0 ;
2013-12-09 22:54:47 -08:00
spin_unlock_bh ( & qitem_lock ) ;
2006-01-02 19:04:38 +01:00
tasklet_kill ( & tipc_tasklet ) ;
spin_lock_bh ( & qitem_lock ) ;
list_for_each_safe ( l , n , & signal_queue_head ) {
item = list_entry ( l , struct queue_item , next_signal ) ;
list_del ( & item - > next_signal ) ;
kmem_cache_free ( tipc_queue_item_cache , item ) ;
}
spin_unlock_bh ( & qitem_lock ) ;
kmem_cache_destroy ( tipc_queue_item_cache ) ;
}