2006-01-02 19:04:38 +01:00
/*
* net / tipc / subscr . c : TIPC subscription service
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"
# include "dbg.h"
# include "subscr.h"
# include "name_table.h"
# include "ref.h"
/**
* struct subscriber - TIPC network topology subscriber
* @ ref : object reference to subscriber object itself
* @ lock : pointer to spinlock controlling access to subscriber object
* @ subscriber_list : adjacent subscribers in top . server ' s list of subscribers
* @ subscription_list : list of subscription objects for this subscriber
* @ port_ref : object reference to port used to communicate with subscriber
* @ swap : indicates if subscriber uses opposite endianness in its messages
*/
2007-02-09 23:25:21 +09:00
2006-01-02 19:04:38 +01:00
struct subscriber {
u32 ref ;
2007-02-09 23:25:21 +09:00
spinlock_t * lock ;
2006-01-02 19:04:38 +01:00
struct list_head subscriber_list ;
struct list_head subscription_list ;
u32 port_ref ;
int swap ;
} ;
/**
* struct top_srv - TIPC network topology subscription service
* @ user_ref : TIPC userid of subscription service
* @ setup_port : reference to TIPC port that handles subscription requests
* @ subscription_count : number of active subscriptions ( not subscribers ! )
* @ subscriber_list : list of ports subscribing to service
* @ lock : spinlock govering access to subscriber list
*/
struct top_srv {
u32 user_ref ;
u32 setup_port ;
atomic_t subscription_count ;
struct list_head subscriber_list ;
spinlock_t lock ;
} ;
static struct top_srv topsrv = { 0 } ;
/**
* htohl - convert value to endianness used by destination
* @ in : value to convert
* @ swap : non - zero if endianness must be reversed
2007-02-09 23:25:21 +09:00
*
2006-01-02 19:04:38 +01:00
* Returns converted value
*/
2006-03-20 22:37:04 -08:00
static u32 htohl ( u32 in , int swap )
2006-01-02 19:04:38 +01:00
{
char * c = ( char * ) & in ;
return swap ? ( ( c [ 3 ] < < 3 ) + ( c [ 2 ] < < 2 ) + ( c [ 1 ] < < 1 ) + c [ 0 ] ) : in ;
}
/**
* subscr_send_event - send a message containing a tipc_event to the subscriber
*/
2007-02-09 23:25:21 +09:00
static void subscr_send_event ( struct subscription * sub ,
u32 found_lower ,
2006-01-02 19:04:38 +01:00
u32 found_upper ,
2007-02-09 23:25:21 +09:00
u32 event ,
u32 port_ref ,
2006-01-02 19:04:38 +01:00
u32 node )
{
struct iovec msg_sect ;
msg_sect . iov_base = ( void * ) & sub - > evt ;
msg_sect . iov_len = sizeof ( struct tipc_event ) ;
sub - > evt . event = htohl ( event , sub - > owner - > swap ) ;
sub - > evt . found_lower = htohl ( found_lower , sub - > owner - > swap ) ;
sub - > evt . found_upper = htohl ( found_upper , sub - > owner - > swap ) ;
sub - > evt . port . ref = htohl ( port_ref , sub - > owner - > swap ) ;
sub - > evt . port . node = htohl ( node , sub - > owner - > swap ) ;
tipc_send ( sub - > owner - > port_ref , 1 , & msg_sect ) ;
}
/**
2006-01-18 00:38:21 +01:00
* tipc_subscr_overlap - test for subscription overlap with the given values
2006-01-02 19:04:38 +01:00
*
* Returns 1 if there is overlap , otherwise 0.
*/
2007-02-09 23:25:21 +09:00
int tipc_subscr_overlap ( struct subscription * sub ,
u32 found_lower ,
2006-01-18 00:38:21 +01:00
u32 found_upper )
2006-01-02 19:04:38 +01:00
{
if ( found_lower < sub - > seq . lower )
found_lower = sub - > seq . lower ;
if ( found_upper > sub - > seq . upper )
found_upper = sub - > seq . upper ;
if ( found_lower > found_upper )
return 0 ;
return 1 ;
}
/**
2006-01-18 00:38:21 +01:00
* tipc_subscr_report_overlap - issue event if there is subscription overlap
2007-02-09 23:25:21 +09:00
*
2006-01-02 19:04:38 +01:00
* Protected by nameseq . lock in name_table . c
*/
2007-02-09 23:25:21 +09:00
void tipc_subscr_report_overlap ( struct subscription * sub ,
u32 found_lower ,
2006-01-18 00:38:21 +01:00
u32 found_upper ,
2007-02-09 23:25:21 +09:00
u32 event ,
u32 port_ref ,
2006-01-18 00:38:21 +01:00
u32 node ,
int must )
2006-01-02 19:04:38 +01:00
{
dbg ( " Rep overlap %u:%u,%u<->%u,%u \n " , sub - > seq . type , sub - > seq . lower ,
sub - > seq . upper , found_lower , found_upper ) ;
2006-01-18 00:38:21 +01:00
if ( ! tipc_subscr_overlap ( sub , found_lower , found_upper ) )
2006-01-02 19:04:38 +01:00
return ;
2006-10-16 21:59:42 -07:00
if ( ! must & & ! ( sub - > filter & TIPC_SUB_PORTS ) )
2006-01-02 19:04:38 +01:00
return ;
subscr_send_event ( sub , found_lower , found_upper , event , port_ref , node ) ;
}
/**
* subscr_timeout - subscription timeout has occurred
*/
static void subscr_timeout ( struct subscription * sub )
{
struct subscriber * subscriber ;
u32 subscriber_ref ;
/* Validate subscriber reference (in case subscriber is terminating) */
subscriber_ref = sub - > owner - > ref ;
2006-01-18 00:38:21 +01:00
subscriber = ( struct subscriber * ) tipc_ref_lock ( subscriber_ref ) ;
2006-01-02 19:04:38 +01:00
if ( subscriber = = NULL )
return ;
2006-10-16 21:59:42 -07:00
/* Validate timeout (in case subscription is being cancelled) */
if ( sub - > timeout = = TIPC_WAIT_FOREVER ) {
tipc_ref_unlock ( subscriber_ref ) ;
return ;
}
2006-01-02 19:04:38 +01:00
/* Unlink subscription from name table */
2006-01-18 00:38:21 +01:00
tipc_nametbl_unsubscribe ( sub ) ;
2006-01-02 19:04:38 +01:00
/* Notify subscriber of timeout, then unlink subscription */
2007-02-09 23:25:21 +09:00
subscr_send_event ( sub ,
sub - > evt . s . seq . lower ,
2006-01-02 19:04:38 +01:00
sub - > evt . s . seq . upper ,
2007-02-09 23:25:21 +09:00
TIPC_SUBSCR_TIMEOUT ,
0 ,
2006-01-02 19:04:38 +01:00
0 ) ;
list_del ( & sub - > subscription_list ) ;
/* Now destroy subscription */
2006-01-18 00:38:21 +01:00
tipc_ref_unlock ( subscriber_ref ) ;
2006-01-02 19:04:38 +01:00
k_term_timer ( & sub - > timer ) ;
kfree ( sub ) ;
atomic_dec ( & topsrv . subscription_count ) ;
}
2006-10-16 21:59:42 -07:00
/**
* subscr_del - delete a subscription within a subscription list
*
* Called with subscriber locked .
*/
static void subscr_del ( struct subscription * sub )
{
tipc_nametbl_unsubscribe ( sub ) ;
list_del ( & sub - > subscription_list ) ;
kfree ( sub ) ;
atomic_dec ( & topsrv . subscription_count ) ;
}
2006-01-02 19:04:38 +01:00
/**
* subscr_terminate - terminate communication with a subscriber
2007-02-09 23:25:21 +09:00
*
2006-01-02 19:04:38 +01:00
* Called with subscriber locked . Routine must temporarily release this lock
2007-02-09 23:25:21 +09:00
* to enable subscription timeout routine ( s ) to finish without deadlocking ;
2006-01-02 19:04:38 +01:00
* the lock is then reclaimed to allow caller to release it upon return .
2007-02-09 23:25:21 +09:00
* ( This should work even in the unlikely event some other thread creates
2006-01-02 19:04:38 +01:00
* a new object reference in the interim that uses this lock ; this routine will
* simply wait for it to be released , then claim it . )
*/
static void subscr_terminate ( struct subscriber * subscriber )
{
struct subscription * sub ;
struct subscription * sub_temp ;
/* Invalidate subscriber reference */
2006-01-18 00:38:21 +01:00
tipc_ref_discard ( subscriber - > ref ) ;
2006-01-02 19:04:38 +01:00
spin_unlock_bh ( subscriber - > lock ) ;
/* Destroy any existing subscriptions for subscriber */
2007-02-09 23:25:21 +09:00
2006-01-02 19:04:38 +01:00
list_for_each_entry_safe ( sub , sub_temp , & subscriber - > subscription_list ,
subscription_list ) {
if ( sub - > timeout ! = TIPC_WAIT_FOREVER ) {
k_cancel_timer ( & sub - > timer ) ;
k_term_timer ( & sub - > timer ) ;
}
2006-10-16 21:59:42 -07:00
dbg ( " Term: Removing sub %u,%u,%u from subscriber %x list \n " ,
2006-01-02 19:04:38 +01:00
sub - > seq . type , sub - > seq . lower , sub - > seq . upper , subscriber ) ;
2006-10-16 21:59:42 -07:00
subscr_del ( sub ) ;
2006-01-02 19:04:38 +01:00
}
/* Sever connection to subscriber */
tipc_shutdown ( subscriber - > port_ref ) ;
tipc_deleteport ( subscriber - > port_ref ) ;
/* Remove subscriber from topology server's subscriber list */
spin_lock_bh ( & topsrv . lock ) ;
list_del ( & subscriber - > subscriber_list ) ;
spin_unlock_bh ( & topsrv . lock ) ;
/* Now destroy subscriber */
spin_lock_bh ( subscriber - > lock ) ;
kfree ( subscriber ) ;
}
2006-10-16 21:59:42 -07:00
/**
* subscr_cancel - handle subscription cancellation request
*
* Called with subscriber locked . Routine must temporarily release this lock
* to enable the subscription timeout routine to finish without deadlocking ;
* the lock is then reclaimed to allow caller to release it upon return .
*
* Note that fields of ' s ' use subscriber ' s endianness !
*/
static void subscr_cancel ( struct tipc_subscr * s ,
struct subscriber * subscriber )
{
struct subscription * sub ;
struct subscription * sub_temp ;
int found = 0 ;
/* Find first matching subscription, exit if not found */
list_for_each_entry_safe ( sub , sub_temp , & subscriber - > subscription_list ,
subscription_list ) {
if ( ! memcmp ( s , & sub - > evt . s , sizeof ( struct tipc_subscr ) ) ) {
found = 1 ;
break ;
}
}
if ( ! found )
return ;
/* Cancel subscription timer (if used), then delete subscription */
if ( sub - > timeout ! = TIPC_WAIT_FOREVER ) {
sub - > timeout = TIPC_WAIT_FOREVER ;
spin_unlock_bh ( subscriber - > lock ) ;
k_cancel_timer ( & sub - > timer ) ;
k_term_timer ( & sub - > timer ) ;
spin_lock_bh ( subscriber - > lock ) ;
}
dbg ( " Cancel: removing sub %u,%u,%u from subscriber %x list \n " ,
sub - > seq . type , sub - > seq . lower , sub - > seq . upper , subscriber ) ;
subscr_del ( sub ) ;
}
2006-01-02 19:04:38 +01:00
/**
* subscr_subscribe - create subscription for subscriber
2007-02-09 23:25:21 +09:00
*
2006-01-02 19:04:38 +01:00
* Called with subscriber locked
*/
static void subscr_subscribe ( struct tipc_subscr * s ,
struct subscriber * subscriber )
{
struct subscription * sub ;
2006-10-16 21:59:42 -07:00
/* Determine/update subscriber's endianness */
if ( s - > filter & ( TIPC_SUB_PORTS | TIPC_SUB_SERVICE ) )
subscriber - > swap = 0 ;
else
subscriber - > swap = 1 ;
/* Detect & process a subscription cancellation request */
if ( s - > filter & htohl ( TIPC_SUB_CANCEL , subscriber - > swap ) ) {
s - > filter & = ~ htohl ( TIPC_SUB_CANCEL , subscriber - > swap ) ;
subscr_cancel ( s , subscriber ) ;
return ;
}
2006-01-02 19:04:38 +01:00
/* Refuse subscription if global limit exceeded */
if ( atomic_read ( & topsrv . subscription_count ) > = tipc_max_subscriptions ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscription rejected, subscription limit reached (%u) \n " ,
tipc_max_subscriptions ) ;
2006-01-02 19:04:38 +01:00
subscr_terminate ( subscriber ) ;
return ;
}
/* Allocate subscription object */
2006-11-21 01:22:12 -02:00
sub = kzalloc ( sizeof ( * sub ) , GFP_ATOMIC ) ;
2006-06-25 23:52:17 -07:00
if ( ! sub ) {
warn ( " Subscription rejected, no memory \n " ) ;
2006-01-02 19:04:38 +01:00
subscr_terminate ( subscriber ) ;
return ;
}
/* Initialize subscription object */
sub - > seq . type = htohl ( s - > seq . type , subscriber - > swap ) ;
sub - > seq . lower = htohl ( s - > seq . lower , subscriber - > swap ) ;
sub - > seq . upper = htohl ( s - > seq . upper , subscriber - > swap ) ;
sub - > timeout = htohl ( s - > timeout , subscriber - > swap ) ;
sub - > filter = htohl ( s - > filter , subscriber - > swap ) ;
2006-10-16 21:59:42 -07:00
if ( ( ! ( sub - > filter & TIPC_SUB_PORTS )
= = ! ( sub - > filter & TIPC_SUB_SERVICE ) )
2006-01-02 19:04:38 +01:00
| | ( sub - > seq . lower > sub - > seq . upper ) ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscription rejected, illegal request \n " ) ;
2006-01-02 19:04:38 +01:00
kfree ( sub ) ;
subscr_terminate ( subscriber ) ;
return ;
}
memcpy ( & sub - > evt . s , s , sizeof ( struct tipc_subscr ) ) ;
INIT_LIST_HEAD ( & sub - > subscription_list ) ;
INIT_LIST_HEAD ( & sub - > nameseq_list ) ;
list_add ( & sub - > subscription_list , & subscriber - > subscription_list ) ;
atomic_inc ( & topsrv . subscription_count ) ;
if ( sub - > timeout ! = TIPC_WAIT_FOREVER ) {
k_init_timer ( & sub - > timer ,
( Handler ) subscr_timeout , ( unsigned long ) sub ) ;
k_start_timer ( & sub - > timer , sub - > timeout ) ;
}
sub - > owner = subscriber ;
2006-01-18 00:38:21 +01:00
tipc_nametbl_subscribe ( sub ) ;
2006-01-02 19:04:38 +01:00
}
/**
* subscr_conn_shutdown_event - handle termination request from subscriber
*/
static void subscr_conn_shutdown_event ( void * usr_handle ,
u32 portref ,
struct sk_buff * * buf ,
unsigned char const * data ,
unsigned int size ,
int reason )
{
2006-01-12 13:22:32 -08:00
struct subscriber * subscriber ;
2006-01-02 19:04:38 +01:00
spinlock_t * subscriber_lock ;
2006-01-18 00:38:21 +01:00
subscriber = tipc_ref_lock ( ( u32 ) ( unsigned long ) usr_handle ) ;
2006-01-02 19:04:38 +01:00
if ( subscriber = = NULL )
return ;
subscriber_lock = subscriber - > lock ;
subscr_terminate ( subscriber ) ;
spin_unlock_bh ( subscriber_lock ) ;
}
/**
* subscr_conn_msg_event - handle new subscription request from subscriber
*/
static void subscr_conn_msg_event ( void * usr_handle ,
u32 port_ref ,
struct sk_buff * * buf ,
const unchar * data ,
u32 size )
{
2006-01-12 13:22:32 -08:00
struct subscriber * subscriber ;
2006-01-02 19:04:38 +01:00
spinlock_t * subscriber_lock ;
2006-01-18 00:38:21 +01:00
subscriber = tipc_ref_lock ( ( u32 ) ( unsigned long ) usr_handle ) ;
2006-01-02 19:04:38 +01:00
if ( subscriber = = NULL )
return ;
subscriber_lock = subscriber - > lock ;
if ( size ! = sizeof ( struct tipc_subscr ) )
subscr_terminate ( subscriber ) ;
else
subscr_subscribe ( ( struct tipc_subscr * ) data , subscriber ) ;
2007-02-09 23:25:21 +09:00
2006-01-02 19:04:38 +01:00
spin_unlock_bh ( subscriber_lock ) ;
}
/**
* subscr_named_msg_event - handle request to establish a new subscriber
*/
static void subscr_named_msg_event ( void * usr_handle ,
u32 port_ref ,
struct sk_buff * * buf ,
const unchar * data ,
u32 size ,
2007-02-09 23:25:21 +09:00
u32 importance ,
2006-01-02 19:04:38 +01:00
struct tipc_portid const * orig ,
struct tipc_name_seq const * dest )
{
struct subscriber * subscriber ;
2006-03-20 22:36:47 -08:00
struct iovec msg_sect = { NULL , 0 } ;
2006-01-02 19:04:38 +01:00
spinlock_t * subscriber_lock ;
dbg ( " subscr_named_msg_event: orig = %x own = %x, \n " ,
orig - > node , tipc_own_addr ) ;
if ( size & & ( size ! = sizeof ( struct tipc_subscr ) ) ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscriber rejected, invalid subscription size \n " ) ;
2006-01-02 19:04:38 +01:00
return ;
}
/* Create subscriber object */
2006-07-21 14:51:30 -07:00
subscriber = kzalloc ( sizeof ( struct subscriber ) , GFP_ATOMIC ) ;
2006-01-02 19:04:38 +01:00
if ( subscriber = = NULL ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscriber rejected, no memory \n " ) ;
2006-01-02 19:04:38 +01:00
return ;
}
INIT_LIST_HEAD ( & subscriber - > subscription_list ) ;
INIT_LIST_HEAD ( & subscriber - > subscriber_list ) ;
2006-01-18 00:38:21 +01:00
subscriber - > ref = tipc_ref_acquire ( subscriber , & subscriber - > lock ) ;
2006-01-02 19:04:38 +01:00
if ( subscriber - > ref = = 0 ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscriber rejected, reference table exhausted \n " ) ;
2006-01-02 19:04:38 +01:00
kfree ( subscriber ) ;
return ;
}
/* Establish a connection to subscriber */
tipc_createport ( topsrv . user_ref ,
2006-01-12 13:22:32 -08:00
( void * ) ( unsigned long ) subscriber - > ref ,
2006-01-02 19:04:38 +01:00
importance ,
2006-03-20 22:36:47 -08:00
NULL ,
NULL ,
2006-01-02 19:04:38 +01:00
subscr_conn_shutdown_event ,
2006-03-20 22:36:47 -08:00
NULL ,
NULL ,
2006-01-02 19:04:38 +01:00
subscr_conn_msg_event ,
2006-03-20 22:36:47 -08:00
NULL ,
2006-01-02 19:04:38 +01:00
& subscriber - > port_ref ) ;
if ( subscriber - > port_ref = = 0 ) {
2006-06-25 23:52:17 -07:00
warn ( " Subscriber rejected, unable to create port \n " ) ;
2006-01-18 00:38:21 +01:00
tipc_ref_discard ( subscriber - > ref ) ;
2006-01-02 19:04:38 +01:00
kfree ( subscriber ) ;
return ;
}
tipc_connect2port ( subscriber - > port_ref , orig ) ;
/* Add subscriber to topology server's subscriber list */
2006-01-18 00:38:21 +01:00
tipc_ref_lock ( subscriber - > ref ) ;
2006-01-02 19:04:38 +01:00
spin_lock_bh ( & topsrv . lock ) ;
list_add ( & subscriber - > subscriber_list , & topsrv . subscriber_list ) ;
spin_unlock_bh ( & topsrv . lock ) ;
/*
* Subscribe now if message contains a subscription ,
* otherwise send an empty response to complete connection handshaking
*/
subscriber_lock = subscriber - > lock ;
if ( size )
subscr_subscribe ( ( struct tipc_subscr * ) data , subscriber ) ;
else
tipc_send ( subscriber - > port_ref , 1 , & msg_sect ) ;
spin_unlock_bh ( subscriber_lock ) ;
}
2006-01-18 00:38:21 +01:00
int tipc_subscr_start ( void )
2006-01-02 19:04:38 +01:00
{
struct tipc_name_seq seq = { TIPC_TOP_SRV , TIPC_TOP_SRV , TIPC_TOP_SRV } ;
int res = - 1 ;
memset ( & topsrv , 0 , sizeof ( topsrv ) ) ;
2006-06-27 02:53:55 -07:00
spin_lock_init ( & topsrv . lock ) ;
2006-01-02 19:04:38 +01:00
INIT_LIST_HEAD ( & topsrv . subscriber_list ) ;
spin_lock_bh ( & topsrv . lock ) ;
2006-03-20 22:36:47 -08:00
res = tipc_attach ( & topsrv . user_ref , NULL , NULL ) ;
2006-01-02 19:04:38 +01:00
if ( res ) {
spin_unlock_bh ( & topsrv . lock ) ;
return res ;
}
2007-02-09 23:25:21 +09:00
res = tipc_createport ( topsrv . user_ref ,
NULL ,
TIPC_CRITICAL_IMPORTANCE ,
NULL ,
NULL ,
NULL ,
NULL ,
subscr_named_msg_event ,
NULL ,
NULL ,
& topsrv . setup_port ) ;
if ( res )
2006-01-02 19:04:38 +01:00
goto failed ;
2007-02-09 23:25:21 +09:00
res = tipc_nametbl_publish_rsv ( topsrv . setup_port , TIPC_NODE_SCOPE , & seq ) ;
if ( res )
2006-01-02 19:04:38 +01:00
goto failed ;
spin_unlock_bh ( & topsrv . lock ) ;
return 0 ;
failed :
2006-01-11 13:52:51 +01:00
err ( " Failed to create subscription service \n " ) ;
2006-01-02 19:04:38 +01:00
tipc_detach ( topsrv . user_ref ) ;
topsrv . user_ref = 0 ;
spin_unlock_bh ( & topsrv . lock ) ;
return res ;
}
2006-01-18 00:38:21 +01:00
void tipc_subscr_stop ( void )
2006-01-02 19:04:38 +01:00
{
struct subscriber * subscriber ;
struct subscriber * subscriber_temp ;
spinlock_t * subscriber_lock ;
if ( topsrv . user_ref ) {
tipc_deleteport ( topsrv . setup_port ) ;
2007-02-09 23:25:21 +09:00
list_for_each_entry_safe ( subscriber , subscriber_temp ,
2006-01-02 19:04:38 +01:00
& topsrv . subscriber_list ,
subscriber_list ) {
2006-01-18 00:38:21 +01:00
tipc_ref_lock ( subscriber - > ref ) ;
2006-01-02 19:04:38 +01:00
subscriber_lock = subscriber - > lock ;
subscr_terminate ( subscriber ) ;
spin_unlock_bh ( subscriber_lock ) ;
}
tipc_detach ( topsrv . user_ref ) ;
topsrv . user_ref = 0 ;
}
}
int tipc_ispublished ( struct tipc_name const * name )
{
u32 domain = 0 ;
2006-01-18 00:38:21 +01:00
return ( tipc_nametbl_translate ( name - > type , name - > instance , & domain ) ! = 0 ) ;
2006-01-02 19:04:38 +01:00
}