2006-01-02 19:04:38 +01:00
/*
* net / tipc / zone . c : TIPC zone management routines
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 .
*
* Redistribution and use in source and binary forms , with or without
* 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 .
*
* 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 .
2006-01-02 19:04:38 +01:00
*
* 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
* POSSIBILITY OF SUCH DAMAGE .
*/
# include "core.h"
# include "zone.h"
# include "net.h"
# include "addr.h"
# include "node_subscr.h"
# include "cluster.h"
# include "node.h"
2006-01-18 00:38:21 +01:00
struct _zone * tipc_zone_create ( u32 addr )
2006-01-02 19:04:38 +01:00
{
2006-06-25 23:52:17 -07:00
struct _zone * z_ptr ;
2006-01-02 19:04:38 +01:00
u32 z_num ;
2006-06-25 23:52:17 -07:00
if ( ! tipc_addr_domain_valid ( addr ) ) {
err ( " Zone creation failed, invalid domain 0x%x \n " , addr ) ;
2006-03-20 22:36:47 -08:00
return NULL ;
2006-06-25 23:52:17 -07:00
}
2006-01-02 19:04:38 +01:00
2006-07-21 14:51:30 -07:00
z_ptr = kzalloc ( sizeof ( * z_ptr ) , GFP_ATOMIC ) ;
2006-06-25 23:52:17 -07:00
if ( ! z_ptr ) {
warn ( " Zone creation failed, insufficient memory \n " ) ;
return NULL ;
2006-01-02 19:04:38 +01:00
}
2006-06-25 23:52:17 -07:00
z_num = tipc_zone ( addr ) ;
z_ptr - > addr = tipc_addr ( z_num , 0 , 0 ) ;
tipc_net . zones [ z_num ] = z_ptr ;
2006-01-02 19:04:38 +01:00
return z_ptr ;
}
2006-01-18 00:38:21 +01:00
void tipc_zone_delete ( struct _zone * z_ptr )
2006-01-02 19:04:38 +01:00
{
u32 c_num ;
if ( ! z_ptr )
return ;
for ( c_num = 1 ; c_num < = tipc_max_clusters ; c_num + + ) {
2006-01-18 00:38:21 +01:00
tipc_cltr_delete ( z_ptr - > clusters [ c_num ] ) ;
2006-01-02 19:04:38 +01:00
}
kfree ( z_ptr ) ;
}
2006-01-18 00:38:21 +01:00
void tipc_zone_attach_cluster ( struct _zone * z_ptr , struct cluster * c_ptr )
2006-01-02 19:04:38 +01:00
{
u32 c_num = tipc_cluster ( c_ptr - > addr ) ;
assert ( c_ptr - > addr ) ;
assert ( c_num < = tipc_max_clusters ) ;
2008-02-24 18:38:31 -08:00
assert ( z_ptr - > clusters [ c_num ] = = NULL ) ;
2006-01-02 19:04:38 +01:00
z_ptr - > clusters [ c_num ] = c_ptr ;
}
2006-01-18 00:38:21 +01:00
void tipc_zone_remove_as_router ( struct _zone * z_ptr , u32 router )
2006-01-02 19:04:38 +01:00
{
u32 c_num ;
for ( c_num = 1 ; c_num < = tipc_max_clusters ; c_num + + ) {
if ( z_ptr - > clusters [ c_num ] ) {
2007-02-09 23:25:21 +09:00
tipc_cltr_remove_as_router ( z_ptr - > clusters [ c_num ] ,
2006-01-18 00:38:21 +01:00
router ) ;
2006-01-02 19:04:38 +01:00
}
}
}
2006-01-18 00:38:21 +01:00
void tipc_zone_send_external_routes ( struct _zone * z_ptr , u32 dest )
2006-01-02 19:04:38 +01:00
{
u32 c_num ;
for ( c_num = 1 ; c_num < = tipc_max_clusters ; c_num + + ) {
if ( z_ptr - > clusters [ c_num ] ) {
if ( in_own_cluster ( z_ptr - > addr ) )
continue ;
2006-01-18 00:38:21 +01:00
tipc_cltr_send_ext_routes ( z_ptr - > clusters [ c_num ] , dest ) ;
2006-01-02 19:04:38 +01:00
}
}
}
2008-09-02 23:38:32 -07:00
struct tipc_node * tipc_zone_select_remote_node ( struct _zone * z_ptr , u32 addr , u32 ref )
2006-01-02 19:04:38 +01:00
{
struct cluster * c_ptr ;
2008-09-02 23:38:32 -07:00
struct tipc_node * n_ptr ;
2006-01-02 19:04:38 +01:00
u32 c_num ;
if ( ! z_ptr )
2006-03-20 22:36:47 -08:00
return NULL ;
2006-01-02 19:04:38 +01:00
c_ptr = z_ptr - > clusters [ tipc_cluster ( addr ) ] ;
if ( ! c_ptr )
2006-03-20 22:36:47 -08:00
return NULL ;
2006-01-18 00:38:21 +01:00
n_ptr = tipc_cltr_select_node ( c_ptr , ref ) ;
2006-01-02 19:04:38 +01:00
if ( n_ptr )
return n_ptr ;
/* Links to any other clusters within this zone ? */
for ( c_num = 1 ; c_num < = tipc_max_clusters ; c_num + + ) {
c_ptr = z_ptr - > clusters [ c_num ] ;
if ( ! c_ptr )
2006-03-20 22:36:47 -08:00
return NULL ;
2006-01-18 00:38:21 +01:00
n_ptr = tipc_cltr_select_node ( c_ptr , ref ) ;
2006-01-02 19:04:38 +01:00
if ( n_ptr )
return n_ptr ;
}
2006-03-20 22:36:47 -08:00
return NULL ;
2006-01-02 19:04:38 +01:00
}
2006-01-18 00:38:21 +01:00
u32 tipc_zone_select_router ( struct _zone * z_ptr , u32 addr , u32 ref )
2006-01-02 19:04:38 +01:00
{
struct cluster * c_ptr ;
u32 c_num ;
u32 router ;
if ( ! z_ptr )
return 0 ;
c_ptr = z_ptr - > clusters [ tipc_cluster ( addr ) ] ;
2006-01-18 00:38:21 +01:00
router = c_ptr ? tipc_cltr_select_router ( c_ptr , ref ) : 0 ;
2006-01-02 19:04:38 +01:00
if ( router )
return router ;
/* Links to any other clusters within the zone? */
for ( c_num = 1 ; c_num < = tipc_max_clusters ; c_num + + ) {
c_ptr = z_ptr - > clusters [ c_num ] ;
2006-01-18 00:38:21 +01:00
router = c_ptr ? tipc_cltr_select_router ( c_ptr , ref ) : 0 ;
2006-01-02 19:04:38 +01:00
if ( router )
return router ;
}
return 0 ;
}
2006-01-18 00:38:21 +01:00
u32 tipc_zone_next_node ( u32 addr )
2006-01-02 19:04:38 +01:00
{
2006-01-18 00:38:21 +01:00
struct cluster * c_ptr = tipc_cltr_find ( addr ) ;
2006-01-02 19:04:38 +01:00
if ( c_ptr )
2006-01-18 00:38:21 +01:00
return tipc_cltr_next_node ( c_ptr , addr ) ;
2006-01-02 19:04:38 +01:00
return 0 ;
}