2005-06-21 23:43:18 +04:00
/*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
* Robert Olsson < robert . olsson @ its . uu . se > Uppsala Universitet
* & Swedish University of Agricultural Sciences .
*
2007-02-09 17:24:47 +03:00
* Jens Laas < jens . laas @ data . slu . se > Swedish University of
2005-06-21 23:43:18 +04:00
* Agricultural Sciences .
2007-02-09 17:24:47 +03:00
*
2005-06-21 23:43:18 +04:00
* Hans Liss < hans . liss @ its . uu . se > Uppsala Universitet
*
* This work is based on the LPC - trie which is originally descibed in :
2007-02-09 17:24:47 +03:00
*
2005-06-21 23:43:18 +04:00
* An experimental study of compression methods for dynamic tries
* Stefan Nilsson and Matti Tikkanen . Algorithmica , 33 ( 1 ) : 19 - 33 , 2002.
* http : //www.nada.kth.se/~snilsson/public/papers/dyntrie2/
*
*
* IP - address lookup using LC - tries . Stefan Nilsson and Gunnar Karlsson
* IEEE Journal on Selected Areas in Communications , 17 ( 6 ) : 1083 - 1092 , June 1999
*
* Version : $ Id : fib_trie . c , v 1.3 2005 / 06 / 08 14 : 20 : 01 robert Exp $
*
*
* Code from fib_hash has been reused which includes the following header :
*
*
* INET An implementation of the TCP / IP protocol suite for the LINUX
* operating system . INET is implemented using the BSD Socket
* interface as the means of communication with the user level .
*
* IPv4 FIB : lookup engine and maintenance routines .
*
*
* Authors : Alexey Kuznetsov , < kuznet @ ms2 . inr . ac . ru >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
2005-12-22 22:25:10 +03:00
*
* Substantial contributions to this work comes from :
*
* David S . Miller , < davem @ davemloft . net >
* Stephen Hemminger < shemminger @ osdl . org >
* Paul E . McKenney < paulmck @ us . ibm . com >
* Patrick McHardy < kaber @ trash . net >
2005-06-21 23:43:18 +04:00
*/
2006-04-04 23:53:35 +04:00
# define VERSION "0.407"
2005-06-21 23:43:18 +04:00
# include <asm/uaccess.h>
# include <asm/system.h>
# include <asm/bitops.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/string.h>
# include <linux/socket.h>
# include <linux/sockios.h>
# include <linux/errno.h>
# include <linux/in.h>
# include <linux/inet.h>
2006-01-04 01:38:34 +03:00
# include <linux/inetdevice.h>
2005-06-21 23:43:18 +04:00
# include <linux/netdevice.h>
# include <linux/if_arp.h>
# include <linux/proc_fs.h>
2005-08-26 00:01:29 +04:00
# include <linux/rcupdate.h>
2005-06-21 23:43:18 +04:00
# include <linux/skbuff.h>
# include <linux/netlink.h>
# include <linux/init.h>
# include <linux/list.h>
# include <net/ip.h>
# include <net/protocol.h>
# include <net/route.h>
# include <net/tcp.h>
# include <net/sock.h>
# include <net/ip_fib.h>
# include "fib_lookup.h"
# undef CONFIG_IP_FIB_TRIE_STATS
2006-03-21 08:35:01 +03:00
# define MAX_STAT_DEPTH 32
2005-06-21 23:43:18 +04:00
# define KEYLENGTH (8*sizeof(t_key))
# define MASK_PFX(k, l) (((l)==0)?0:(k >> (KEYLENGTH-l)) << (KEYLENGTH-l))
# define TKEY_GET_MASK(offset, bits) (((bits)==0)?0:((t_key)(-1) << (KEYLENGTH - bits) >> offset))
typedef unsigned int t_key ;
# define T_TNODE 0
# define T_LEAF 1
# define NODE_TYPE_MASK 0x1UL
2005-08-10 07:24:39 +04:00
# define NODE_PARENT(node) \
2005-08-26 00:01:29 +04:00
( ( struct tnode * ) rcu_dereference ( ( ( node ) - > parent & ~ NODE_TYPE_MASK ) ) )
# define NODE_TYPE(node) ((node)->parent & NODE_TYPE_MASK)
# define NODE_SET_PARENT(node, ptr) \
rcu_assign_pointer ( ( node ) - > parent , \
( ( unsigned long ) ( ptr ) ) | NODE_TYPE ( node ) )
2005-08-10 07:24:39 +04:00
# define IS_TNODE(n) (!(n->parent & T_LEAF))
# define IS_LEAF(n) (n->parent & T_LEAF)
2005-06-21 23:43:18 +04:00
struct node {
2005-08-10 07:24:39 +04:00
t_key key ;
unsigned long parent ;
2005-06-21 23:43:18 +04:00
} ;
struct leaf {
2005-08-10 07:24:39 +04:00
t_key key ;
unsigned long parent ;
2005-06-21 23:43:18 +04:00
struct hlist_head list ;
2005-08-26 00:01:29 +04:00
struct rcu_head rcu ;
2005-06-21 23:43:18 +04:00
} ;
struct leaf_info {
struct hlist_node hlist ;
2005-08-26 00:01:29 +04:00
struct rcu_head rcu ;
2005-06-21 23:43:18 +04:00
int plen ;
struct list_head falh ;
} ;
struct tnode {
2005-08-10 07:24:39 +04:00
t_key key ;
unsigned long parent ;
unsigned short pos : 5 ; /* 2log(KEYLENGTH) bits needed */
unsigned short bits : 5 ; /* 2log(KEYLENGTH) bits needed */
unsigned short full_children ; /* KEYLENGTH bits needed */
unsigned short empty_children ; /* KEYLENGTH bits needed */
2005-08-26 00:01:29 +04:00
struct rcu_head rcu ;
2005-08-10 07:24:39 +04:00
struct node * child [ 0 ] ;
2005-06-21 23:43:18 +04:00
} ;
# ifdef CONFIG_IP_FIB_TRIE_STATS
struct trie_use_stats {
unsigned int gets ;
unsigned int backtrack ;
unsigned int semantic_match_passed ;
unsigned int semantic_match_miss ;
unsigned int null_node_hit ;
2005-07-06 02:02:40 +04:00
unsigned int resize_node_skipped ;
2005-06-21 23:43:18 +04:00
} ;
# endif
struct trie_stat {
unsigned int totdepth ;
unsigned int maxdepth ;
unsigned int tnodes ;
unsigned int leaves ;
unsigned int nullpointers ;
2006-03-21 08:35:01 +03:00
unsigned int nodesizes [ MAX_STAT_DEPTH ] ;
2005-07-20 01:01:51 +04:00
} ;
2005-06-21 23:43:18 +04:00
struct trie {
2005-08-10 07:24:39 +04:00
struct node * trie ;
2005-06-21 23:43:18 +04:00
# ifdef CONFIG_IP_FIB_TRIE_STATS
struct trie_use_stats stats ;
# endif
2005-08-10 07:24:39 +04:00
int size ;
2005-06-21 23:43:18 +04:00
unsigned int revision ;
} ;
static void put_child ( struct trie * t , struct tnode * tn , int i , struct node * n ) ;
static void tnode_put_child_reorg ( struct tnode * tn , int i , struct node * n , int wasfull ) ;
static struct node * resize ( struct trie * t , struct tnode * tn ) ;
2005-08-10 07:25:06 +04:00
static struct tnode * inflate ( struct trie * t , struct tnode * tn ) ;
static struct tnode * halve ( struct trie * t , struct tnode * tn ) ;
2005-06-21 23:43:18 +04:00
static void tnode_free ( struct tnode * tn ) ;
2006-12-07 07:33:20 +03:00
static struct kmem_cache * fn_alias_kmem __read_mostly ;
2005-06-21 23:43:18 +04:00
static struct trie * trie_local = NULL , * trie_main = NULL ;
2005-08-26 00:01:29 +04:00
/* rcu_read_lock needs to be hold by caller from readside */
2005-07-20 01:01:51 +04:00
static inline struct node * tnode_get_child ( struct tnode * tn , int i )
2005-06-21 23:43:18 +04:00
{
2005-08-10 07:24:39 +04:00
BUG_ON ( i > = 1 < < tn - > bits ) ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
return rcu_dereference ( tn - > child [ i ] ) ;
2005-06-21 23:43:18 +04:00
}
2005-08-10 07:25:39 +04:00
static inline int tnode_child_length ( const struct tnode * tn )
2005-06-21 23:43:18 +04:00
{
2005-08-10 07:24:39 +04:00
return 1 < < tn - > bits ;
2005-06-21 23:43:18 +04:00
}
static inline t_key tkey_extract_bits ( t_key a , int offset , int bits )
{
2005-08-10 07:24:39 +04:00
if ( offset < KEYLENGTH )
2005-06-21 23:43:18 +04:00
return ( ( t_key ) ( a < < offset ) ) > > ( KEYLENGTH - bits ) ;
2005-08-10 07:24:39 +04:00
else
2005-06-21 23:43:18 +04:00
return 0 ;
}
static inline int tkey_equals ( t_key a , t_key b )
{
2005-07-20 01:01:51 +04:00
return a = = b ;
2005-06-21 23:43:18 +04:00
}
static inline int tkey_sub_equals ( t_key a , int offset , int bits , t_key b )
{
2005-07-20 01:01:51 +04:00
if ( bits = = 0 | | offset > = KEYLENGTH )
return 1 ;
2005-08-10 07:24:39 +04:00
bits = bits > KEYLENGTH ? KEYLENGTH : bits ;
return ( ( a ^ b ) < < offset ) > > ( KEYLENGTH - bits ) = = 0 ;
2005-07-20 01:01:51 +04:00
}
2005-06-21 23:43:18 +04:00
static inline int tkey_mismatch ( t_key a , int offset , t_key b )
{
t_key diff = a ^ b ;
int i = offset ;
2005-07-20 01:01:51 +04:00
if ( ! diff )
return 0 ;
while ( ( diff < < i ) > > ( KEYLENGTH - 1 ) = = 0 )
2005-06-21 23:43:18 +04:00
i + + ;
return i ;
}
/*
2007-02-09 17:24:47 +03:00
To understand this stuff , an understanding of keys and all their bits is
necessary . Every node in the trie has a key associated with it , but not
2005-06-21 23:43:18 +04:00
all of the bits in that key are significant .
Consider a node ' n ' and its parent ' tp ' .
2007-02-09 17:24:47 +03:00
If n is a leaf , every bit in its key is significant . Its presence is
necessitated by path compression , since during a tree traversal ( when
searching for a leaf - unless we are doing an insertion ) we will completely
ignore all skipped bits we encounter . Thus we need to verify , at the end of
a potentially successful search , that we have indeed been walking the
2005-06-21 23:43:18 +04:00
correct key path .
2007-02-09 17:24:47 +03:00
Note that we can never " miss " the correct key in the tree if present by
following the wrong path . Path compression ensures that segments of the key
that are the same for all keys with a given prefix are skipped , but the
skipped part * is * identical for each node in the subtrie below the skipped
bit ! trie_insert ( ) in this implementation takes care of that - note the
2005-06-21 23:43:18 +04:00
call to tkey_sub_equals ( ) in trie_insert ( ) .
2007-02-09 17:24:47 +03:00
if n is an internal node - a ' tnode ' here , the various parts of its key
2005-06-21 23:43:18 +04:00
have many different meanings .
2007-02-09 17:24:47 +03:00
Example :
2005-06-21 23:43:18 +04:00
_________________________________________________________________
| i | i | i | i | i | i | i | N | N | N | S | S | S | S | S | C |
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
2007-02-09 17:24:47 +03:00
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
2005-06-21 23:43:18 +04:00
_________________________________________________________________
| C | C | C | u | u | u | u | u | u | u | u | u | u | u | u | u |
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
tp - > pos = 7
tp - > bits = 3
n - > pos = 15
2005-08-10 07:24:39 +04:00
n - > bits = 4
2005-06-21 23:43:18 +04:00
2007-02-09 17:24:47 +03:00
First , let ' s just ignore the bits that come before the parent tp , that is
the bits from 0 to ( tp - > pos - 1 ) . They are * known * but at this point we do
2005-06-21 23:43:18 +04:00
not use them for anything .
The bits from ( tp - > pos ) to ( tp - > pos + tp - > bits - 1 ) - " N " , above - are the
2007-02-09 17:24:47 +03:00
index into the parent ' s child array . That is , they will be used to find
2005-06-21 23:43:18 +04:00
' n ' among tp ' s children .
The bits from ( tp - > pos + tp - > bits ) to ( n - > pos - 1 ) - " S " - are skipped bits
for the node n .
2007-02-09 17:24:47 +03:00
All the bits we have seen so far are significant to the node n . The rest
2005-06-21 23:43:18 +04:00
of the bits are really not needed or indeed known in n - > key .
2007-02-09 17:24:47 +03:00
The bits from ( n - > pos ) to ( n - > pos + n - > bits - 1 ) - " C " - are the index into
2005-06-21 23:43:18 +04:00
n ' s child array , and will of course be different for each child .
2007-02-09 17:24:47 +03:00
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
The rest of the bits , from ( n - > pos + n - > bits ) onward , are completely unknown
at this point .
*/
2005-08-24 08:59:41 +04:00
static inline void check_tnode ( const struct tnode * tn )
2005-06-21 23:43:18 +04:00
{
2005-08-24 08:59:41 +04:00
WARN_ON ( tn & & tn - > pos + tn - > bits > 32 ) ;
2005-06-21 23:43:18 +04:00
}
static int halve_threshold = 25 ;
static int inflate_threshold = 50 ;
2005-10-05 00:01:58 +04:00
static int halve_threshold_root = 15 ;
2007-02-09 17:24:47 +03:00
static int inflate_threshold_root = 25 ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
static void __alias_free_mem ( struct rcu_head * head )
2005-06-21 23:43:18 +04:00
{
2005-08-26 00:01:29 +04:00
struct fib_alias * fa = container_of ( head , struct fib_alias , rcu ) ;
kmem_cache_free ( fn_alias_kmem , fa ) ;
2005-06-21 23:43:18 +04:00
}
2005-08-26 00:01:29 +04:00
static inline void alias_free_mem_rcu ( struct fib_alias * fa )
2005-06-21 23:43:18 +04:00
{
2005-08-26 00:01:29 +04:00
call_rcu ( & fa - > rcu , __alias_free_mem ) ;
}
2005-08-10 07:24:39 +04:00
2005-08-26 00:01:29 +04:00
static void __leaf_free_rcu ( struct rcu_head * head )
{
kfree ( container_of ( head , struct leaf , rcu ) ) ;
}
2005-08-10 07:24:39 +04:00
2005-08-26 00:01:29 +04:00
static void __leaf_info_free_rcu ( struct rcu_head * head )
2005-06-21 23:43:18 +04:00
{
2005-08-26 00:01:29 +04:00
kfree ( container_of ( head , struct leaf_info , rcu ) ) ;
2005-06-21 23:43:18 +04:00
}
2005-08-26 00:01:29 +04:00
static inline void free_leaf_info ( struct leaf_info * leaf )
2005-06-21 23:43:18 +04:00
{
2005-08-26 00:01:29 +04:00
call_rcu ( & leaf - > rcu , __leaf_info_free_rcu ) ;
2005-06-21 23:43:18 +04:00
}
2005-07-06 01:44:55 +04:00
static struct tnode * tnode_alloc ( unsigned int size )
{
2005-08-26 00:01:29 +04:00
struct page * pages ;
if ( size < = PAGE_SIZE )
return kcalloc ( size , 1 , GFP_KERNEL ) ;
pages = alloc_pages ( GFP_KERNEL | __GFP_ZERO , get_order ( size ) ) ;
if ( ! pages )
return NULL ;
return page_address ( pages ) ;
2005-07-06 01:44:55 +04:00
}
2005-08-26 00:01:29 +04:00
static void __tnode_free_rcu ( struct rcu_head * head )
2005-07-06 01:44:55 +04:00
{
2005-08-26 00:01:29 +04:00
struct tnode * tn = container_of ( head , struct tnode , rcu ) ;
2005-07-06 01:44:55 +04:00
unsigned int size = sizeof ( struct tnode ) +
2005-08-26 00:01:29 +04:00
( 1 < < tn - > bits ) * sizeof ( struct node * ) ;
2005-07-06 01:44:55 +04:00
if ( size < = PAGE_SIZE )
kfree ( tn ) ;
else
free_pages ( ( unsigned long ) tn , get_order ( size ) ) ;
}
2005-08-26 00:01:29 +04:00
static inline void tnode_free ( struct tnode * tn )
{
2006-04-04 23:53:35 +04:00
if ( IS_LEAF ( tn ) ) {
struct leaf * l = ( struct leaf * ) tn ;
call_rcu_bh ( & l - > rcu , __leaf_free_rcu ) ;
}
2007-02-09 17:24:47 +03:00
else
2006-04-04 23:53:35 +04:00
call_rcu ( & tn - > rcu , __tnode_free_rcu ) ;
2005-08-26 00:01:29 +04:00
}
static struct leaf * leaf_new ( void )
{
struct leaf * l = kmalloc ( sizeof ( struct leaf ) , GFP_KERNEL ) ;
if ( l ) {
l - > parent = T_LEAF ;
INIT_HLIST_HEAD ( & l - > list ) ;
}
return l ;
}
static struct leaf_info * leaf_info_new ( int plen )
{
struct leaf_info * li = kmalloc ( sizeof ( struct leaf_info ) , GFP_KERNEL ) ;
if ( li ) {
li - > plen = plen ;
INIT_LIST_HEAD ( & li - > falh ) ;
}
return li ;
}
2005-06-21 23:43:18 +04:00
static struct tnode * tnode_new ( t_key key , int pos , int bits )
{
int nchildren = 1 < < bits ;
int sz = sizeof ( struct tnode ) + nchildren * sizeof ( struct node * ) ;
2005-07-06 01:44:55 +04:00
struct tnode * tn = tnode_alloc ( sz ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
if ( tn ) {
2005-06-21 23:43:18 +04:00
memset ( tn , 0 , sz ) ;
2005-08-26 00:01:29 +04:00
tn - > parent = T_TNODE ;
2005-06-21 23:43:18 +04:00
tn - > pos = pos ;
tn - > bits = bits ;
tn - > key = key ;
tn - > full_children = 0 ;
tn - > empty_children = 1 < < bits ;
}
2005-07-20 01:01:51 +04:00
2005-08-24 08:59:41 +04:00
pr_debug ( " AT %p s=%u %u \n " , tn , ( unsigned int ) sizeof ( struct tnode ) ,
( unsigned int ) ( sizeof ( struct node ) * 1 < < bits ) ) ;
2005-06-21 23:43:18 +04:00
return tn ;
}
/*
* Check whether a tnode ' n ' is " full " , i . e . it is an internal node
* and no bits are skipped . See discussion in dyntree paper p . 6
*/
2005-08-10 07:25:39 +04:00
static inline int tnode_full ( const struct tnode * tn , const struct node * n )
2005-06-21 23:43:18 +04:00
{
2005-07-20 01:01:51 +04:00
if ( n = = NULL | | IS_LEAF ( n ) )
2005-06-21 23:43:18 +04:00
return 0 ;
return ( ( struct tnode * ) n ) - > pos = = tn - > pos + tn - > bits ;
}
2005-07-20 01:01:51 +04:00
static inline void put_child ( struct trie * t , struct tnode * tn , int i , struct node * n )
2005-06-21 23:43:18 +04:00
{
tnode_put_child_reorg ( tn , i , n , - 1 ) ;
}
2005-07-20 01:01:51 +04:00
/*
2005-06-21 23:43:18 +04:00
* Add a child at position i overwriting the old value .
* Update the value of full_children and empty_children .
*/
2005-07-20 01:01:51 +04:00
static void tnode_put_child_reorg ( struct tnode * tn , int i , struct node * n , int wasfull )
2005-06-21 23:43:18 +04:00
{
2005-08-26 00:01:29 +04:00
struct node * chi = tn - > child [ i ] ;
2005-06-21 23:43:18 +04:00
int isfull ;
2005-08-24 08:59:41 +04:00
BUG_ON ( i > = 1 < < tn - > bits ) ;
2005-06-21 23:43:18 +04:00
/* update emptyChildren */
if ( n = = NULL & & chi ! = NULL )
tn - > empty_children + + ;
else if ( n ! = NULL & & chi = = NULL )
tn - > empty_children - - ;
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
/* update fullChildren */
2005-08-10 07:24:39 +04:00
if ( wasfull = = - 1 )
2005-06-21 23:43:18 +04:00
wasfull = tnode_full ( tn , chi ) ;
isfull = tnode_full ( tn , n ) ;
2005-07-20 01:01:51 +04:00
if ( wasfull & & ! isfull )
2005-06-21 23:43:18 +04:00
tn - > full_children - - ;
2005-07-20 01:01:51 +04:00
else if ( ! wasfull & & isfull )
2005-06-21 23:43:18 +04:00
tn - > full_children + + ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( n )
NODE_SET_PARENT ( n , tn ) ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( tn - > child [ i ] , n ) ;
2005-06-21 23:43:18 +04:00
}
2005-07-20 01:01:51 +04:00
static struct node * resize ( struct trie * t , struct tnode * tn )
2005-06-21 23:43:18 +04:00
{
int i ;
2005-07-06 02:02:40 +04:00
int err = 0 ;
2005-08-10 07:25:06 +04:00
struct tnode * old_tn ;
2005-10-05 00:01:58 +04:00
int inflate_threshold_use ;
int halve_threshold_use ;
2005-06-21 23:43:18 +04:00
2007-02-09 17:24:47 +03:00
if ( ! tn )
2005-06-21 23:43:18 +04:00
return NULL ;
2005-08-24 08:59:41 +04:00
pr_debug ( " In tnode_resize %p inflate_threshold=%d threshold=%d \n " ,
tn , inflate_threshold , halve_threshold ) ;
2005-06-21 23:43:18 +04:00
/* No children */
if ( tn - > empty_children = = tnode_child_length ( tn ) ) {
tnode_free ( tn ) ;
return NULL ;
}
/* One child */
if ( tn - > empty_children = = tnode_child_length ( tn ) - 1 )
for ( i = 0 ; i < tnode_child_length ( tn ) ; i + + ) {
2005-08-10 07:24:39 +04:00
struct node * n ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
n = tn - > child [ i ] ;
2005-08-26 00:01:29 +04:00
if ( ! n )
2005-08-10 07:24:39 +04:00
continue ;
/* compress one level */
2005-08-26 00:01:29 +04:00
NODE_SET_PARENT ( n , NULL ) ;
2005-08-10 07:24:39 +04:00
tnode_free ( tn ) ;
return n ;
2005-06-21 23:43:18 +04:00
}
2005-07-20 01:01:51 +04:00
/*
2005-06-21 23:43:18 +04:00
* Double as long as the resulting node has a number of
* nonempty nodes that are above the threshold .
*/
/*
2005-07-20 01:01:51 +04:00
* From " Implementing a dynamic compressed trie " by Stefan Nilsson of
* the Helsinki University of Technology and Matti Tikkanen of Nokia
2005-06-21 23:43:18 +04:00
* Telecommunications , page 6 :
2005-07-20 01:01:51 +04:00
* " A node is doubled if the ratio of non-empty children to all
2005-06-21 23:43:18 +04:00
* children in the * doubled * node is at least ' high ' . "
*
2005-07-20 01:01:51 +04:00
* ' high ' in this instance is the variable ' inflate_threshold ' . It
* is expressed as a percentage , so we multiply it with
* tnode_child_length ( ) and instead of multiplying by 2 ( since the
* child array will be doubled by inflate ( ) ) and multiplying
* the left - hand side by 100 ( to handle the percentage thing ) we
2005-06-21 23:43:18 +04:00
* multiply the left - hand side by 50.
2005-07-20 01:01:51 +04:00
*
* The left - hand side may look a bit weird : tnode_child_length ( tn )
* - tn - > empty_children is of course the number of non - null children
* in the current node . tn - > full_children is the number of " full "
2005-06-21 23:43:18 +04:00
* children , that is non - null tnodes with a skip value of 0.
2005-07-20 01:01:51 +04:00
* All of those will be doubled in the resulting inflated tnode , so
2005-06-21 23:43:18 +04:00
* we just count them one extra time here .
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* A clearer way to write this would be :
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* to_be_doubled = tn - > full_children ;
2005-07-20 01:01:51 +04:00
* not_to_be_doubled = tnode_child_length ( tn ) - tn - > empty_children -
2005-06-21 23:43:18 +04:00
* tn - > full_children ;
*
* new_child_length = tnode_child_length ( tn ) * 2 ;
*
2005-07-20 01:01:51 +04:00
* new_fill_factor = 100 * ( not_to_be_doubled + 2 * to_be_doubled ) /
2005-06-21 23:43:18 +04:00
* new_child_length ;
* if ( new_fill_factor > = inflate_threshold )
2005-07-20 01:01:51 +04:00
*
* . . . and so on , tho it would mess up the while ( ) loop .
*
2005-06-21 23:43:18 +04:00
* anyway ,
* 100 * ( not_to_be_doubled + 2 * to_be_doubled ) / new_child_length > =
* inflate_threshold
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* avoid a division :
* 100 * ( not_to_be_doubled + 2 * to_be_doubled ) > =
* inflate_threshold * new_child_length
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* expand not_to_be_doubled and to_be_doubled , and shorten :
2005-07-20 01:01:51 +04:00
* 100 * ( tnode_child_length ( tn ) - tn - > empty_children +
2005-08-10 07:24:39 +04:00
* tn - > full_children ) > = inflate_threshold * new_child_length
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* expand new_child_length :
2005-07-20 01:01:51 +04:00
* 100 * ( tnode_child_length ( tn ) - tn - > empty_children +
2005-08-10 07:24:39 +04:00
* tn - > full_children ) > =
2005-06-21 23:43:18 +04:00
* inflate_threshold * tnode_child_length ( tn ) * 2
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
* shorten again :
2005-07-20 01:01:51 +04:00
* 50 * ( tn - > full_children + tnode_child_length ( tn ) -
2005-08-10 07:24:39 +04:00
* tn - > empty_children ) > = inflate_threshold *
2005-06-21 23:43:18 +04:00
* tnode_child_length ( tn )
2005-07-20 01:01:51 +04:00
*
2005-06-21 23:43:18 +04:00
*/
check_tnode ( tn ) ;
2005-07-20 01:01:51 +04:00
2005-10-05 00:01:58 +04:00
/* Keep root node larger */
if ( ! tn - > parent )
inflate_threshold_use = inflate_threshold_root ;
2007-02-09 17:24:47 +03:00
else
2005-10-05 00:01:58 +04:00
inflate_threshold_use = inflate_threshold ;
2005-07-06 02:02:40 +04:00
err = 0 ;
2005-06-21 23:43:18 +04:00
while ( ( tn - > full_children > 0 & &
50 * ( tn - > full_children + tnode_child_length ( tn ) - tn - > empty_children ) > =
2005-10-05 00:01:58 +04:00
inflate_threshold_use * tnode_child_length ( tn ) ) ) {
2005-06-21 23:43:18 +04:00
2005-08-10 07:25:06 +04:00
old_tn = tn ;
tn = inflate ( t , tn ) ;
if ( IS_ERR ( tn ) ) {
tn = old_tn ;
2005-07-06 02:02:40 +04:00
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . resize_node_skipped + + ;
# endif
break ;
}
2005-06-21 23:43:18 +04:00
}
check_tnode ( tn ) ;
/*
* Halve as long as the number of empty children in this
* node is above threshold .
*/
2005-07-06 02:02:40 +04:00
2005-10-05 00:01:58 +04:00
/* Keep root node larger */
if ( ! tn - > parent )
halve_threshold_use = halve_threshold_root ;
2007-02-09 17:24:47 +03:00
else
2005-10-05 00:01:58 +04:00
halve_threshold_use = halve_threshold ;
2005-07-06 02:02:40 +04:00
err = 0 ;
2005-06-21 23:43:18 +04:00
while ( tn - > bits > 1 & &
100 * ( tnode_child_length ( tn ) - tn - > empty_children ) <
2005-10-05 00:01:58 +04:00
halve_threshold_use * tnode_child_length ( tn ) ) {
2005-07-06 02:02:40 +04:00
2005-08-10 07:25:06 +04:00
old_tn = tn ;
tn = halve ( t , tn ) ;
if ( IS_ERR ( tn ) ) {
tn = old_tn ;
2005-07-06 02:02:40 +04:00
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . resize_node_skipped + + ;
# endif
break ;
}
}
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
/* Only one child remains */
if ( tn - > empty_children = = tnode_child_length ( tn ) - 1 )
for ( i = 0 ; i < tnode_child_length ( tn ) ; i + + ) {
2005-08-10 07:24:39 +04:00
struct node * n ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
n = tn - > child [ i ] ;
2005-08-26 00:01:29 +04:00
if ( ! n )
2005-08-10 07:24:39 +04:00
continue ;
/* compress one level */
2005-08-26 00:01:29 +04:00
NODE_SET_PARENT ( n , NULL ) ;
2005-08-10 07:24:39 +04:00
tnode_free ( tn ) ;
return n ;
2005-06-21 23:43:18 +04:00
}
return ( struct node * ) tn ;
}
2005-08-10 07:25:06 +04:00
static struct tnode * inflate ( struct trie * t , struct tnode * tn )
2005-06-21 23:43:18 +04:00
{
struct tnode * inode ;
struct tnode * oldtnode = tn ;
int olen = tnode_child_length ( tn ) ;
int i ;
2005-08-24 08:59:41 +04:00
pr_debug ( " In inflate \n " ) ;
2005-06-21 23:43:18 +04:00
tn = tnode_new ( oldtnode - > key , oldtnode - > pos , oldtnode - > bits + 1 ) ;
2005-08-24 08:59:41 +04:00
if ( ! tn )
2005-08-10 07:25:06 +04:00
return ERR_PTR ( - ENOMEM ) ;
2005-07-06 02:02:40 +04:00
/*
2005-07-20 01:01:51 +04:00
* Preallocate and store tnodes before the actual work so we
* don ' t get into an inconsistent state if memory allocation
* fails . In case of failure we return the oldnode and inflate
2005-07-06 02:02:40 +04:00
* of tnode is ignored .
*/
2005-08-10 07:24:39 +04:00
for ( i = 0 ; i < olen ; i + + ) {
2005-07-06 02:02:40 +04:00
struct tnode * inode = ( struct tnode * ) tnode_get_child ( oldtnode , i ) ;
if ( inode & &
IS_TNODE ( inode ) & &
inode - > pos = = oldtnode - > pos + oldtnode - > bits & &
inode - > bits > 1 ) {
struct tnode * left , * right ;
t_key m = TKEY_GET_MASK ( inode - > pos , 1 ) ;
2005-07-20 01:01:51 +04:00
2005-07-06 02:02:40 +04:00
left = tnode_new ( inode - > key & ( ~ m ) , inode - > pos + 1 ,
inode - > bits - 1 ) ;
2005-08-10 07:25:06 +04:00
if ( ! left )
goto nomem ;
2005-08-10 07:24:39 +04:00
2005-07-06 02:02:40 +04:00
right = tnode_new ( inode - > key | m , inode - > pos + 1 ,
inode - > bits - 1 ) ;
2007-02-09 17:24:47 +03:00
if ( ! right ) {
2005-08-10 07:25:06 +04:00
tnode_free ( left ) ;
goto nomem ;
2007-02-09 17:24:47 +03:00
}
2005-07-06 02:02:40 +04:00
put_child ( t , tn , 2 * i , ( struct node * ) left ) ;
put_child ( t , tn , 2 * i + 1 , ( struct node * ) right ) ;
}
}
2005-08-10 07:24:39 +04:00
for ( i = 0 ; i < olen ; i + + ) {
2005-06-21 23:43:18 +04:00
struct node * node = tnode_get_child ( oldtnode , i ) ;
2005-08-10 07:24:39 +04:00
struct tnode * left , * right ;
int size , j ;
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
/* An empty child */
if ( node = = NULL )
continue ;
/* A leaf or an internal node with skipped bits */
2005-07-20 01:01:51 +04:00
if ( IS_LEAF ( node ) | | ( ( struct tnode * ) node ) - > pos >
2005-06-21 23:43:18 +04:00
tn - > pos + tn - > bits - 1 ) {
2005-07-20 01:01:51 +04:00
if ( tkey_extract_bits ( node - > key , oldtnode - > pos + oldtnode - > bits ,
2005-06-21 23:43:18 +04:00
1 ) = = 0 )
put_child ( t , tn , 2 * i , node ) ;
else
put_child ( t , tn , 2 * i + 1 , node ) ;
continue ;
}
/* An internal node with two children */
inode = ( struct tnode * ) node ;
if ( inode - > bits = = 1 ) {
put_child ( t , tn , 2 * i , inode - > child [ 0 ] ) ;
put_child ( t , tn , 2 * i + 1 , inode - > child [ 1 ] ) ;
tnode_free ( inode ) ;
2005-08-10 07:24:39 +04:00
continue ;
2005-06-21 23:43:18 +04:00
}
2005-08-10 07:24:39 +04:00
/* An internal node with more than two children */
/* We will replace this node 'inode' with two new
* ones , ' left ' and ' right ' , each with half of the
* original children . The two new nodes will have
* a position one bit further down the key and this
* means that the " significant " part of their keys
* ( see the discussion near the top of this file )
* will differ by one bit , which will be " 0 " in
* left ' s key and " 1 " in right ' s key . Since we are
* moving the key position by one step , the bit that
* we are moving away from - the bit at position
* ( inode - > pos ) - is the one that will differ between
* left and right . So . . . we synthesize that bit in the
* two new keys .
* The mask ' m ' below will be a single " one " bit at
* the position ( inode - > pos )
*/
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/* Use the old key, but set the new significant
* bit to zero .
*/
2005-07-06 02:02:40 +04:00
2005-08-10 07:24:39 +04:00
left = ( struct tnode * ) tnode_get_child ( tn , 2 * i ) ;
put_child ( t , tn , 2 * i , NULL ) ;
2005-07-06 02:02:40 +04:00
2005-08-10 07:24:39 +04:00
BUG_ON ( ! left ) ;
2005-07-06 02:02:40 +04:00
2005-08-10 07:24:39 +04:00
right = ( struct tnode * ) tnode_get_child ( tn , 2 * i + 1 ) ;
put_child ( t , tn , 2 * i + 1 , NULL ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
BUG_ON ( ! right ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
size = tnode_child_length ( left ) ;
for ( j = 0 ; j < size ; j + + ) {
put_child ( t , left , j , inode - > child [ j ] ) ;
put_child ( t , right , j , inode - > child [ j + size ] ) ;
2005-06-21 23:43:18 +04:00
}
2005-08-10 07:24:39 +04:00
put_child ( t , tn , 2 * i , resize ( t , left ) ) ;
put_child ( t , tn , 2 * i + 1 , resize ( t , right ) ) ;
tnode_free ( inode ) ;
2005-06-21 23:43:18 +04:00
}
tnode_free ( oldtnode ) ;
return tn ;
2005-08-10 07:25:06 +04:00
nomem :
{
int size = tnode_child_length ( tn ) ;
int j ;
2005-08-24 08:59:41 +04:00
for ( j = 0 ; j < size ; j + + )
2005-08-10 07:25:06 +04:00
if ( tn - > child [ j ] )
tnode_free ( ( struct tnode * ) tn - > child [ j ] ) ;
tnode_free ( tn ) ;
2005-08-24 08:59:41 +04:00
2005-08-10 07:25:06 +04:00
return ERR_PTR ( - ENOMEM ) ;
}
2005-06-21 23:43:18 +04:00
}
2005-08-10 07:25:06 +04:00
static struct tnode * halve ( struct trie * t , struct tnode * tn )
2005-06-21 23:43:18 +04:00
{
struct tnode * oldtnode = tn ;
struct node * left , * right ;
int i ;
int olen = tnode_child_length ( tn ) ;
2005-08-24 08:59:41 +04:00
pr_debug ( " In halve \n " ) ;
2005-07-20 01:01:51 +04:00
tn = tnode_new ( oldtnode - > key , oldtnode - > pos , oldtnode - > bits - 1 ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:25:06 +04:00
if ( ! tn )
return ERR_PTR ( - ENOMEM ) ;
2005-07-06 02:02:40 +04:00
/*
2005-07-20 01:01:51 +04:00
* Preallocate and store tnodes before the actual work so we
* don ' t get into an inconsistent state if memory allocation
* fails . In case of failure we return the oldnode and halve
2005-07-06 02:02:40 +04:00
* of tnode is ignored .
*/
2005-08-10 07:24:39 +04:00
for ( i = 0 ; i < olen ; i + = 2 ) {
2005-07-06 02:02:40 +04:00
left = tnode_get_child ( oldtnode , i ) ;
right = tnode_get_child ( oldtnode , i + 1 ) ;
2005-07-20 01:01:51 +04:00
2005-07-06 02:02:40 +04:00
/* Two nonempty children */
2005-08-24 08:59:41 +04:00
if ( left & & right ) {
2005-08-10 07:25:06 +04:00
struct tnode * newn ;
2005-08-24 08:59:41 +04:00
2005-08-10 07:25:06 +04:00
newn = tnode_new ( left - > key , tn - > pos + tn - > bits , 1 ) ;
2005-08-24 08:59:41 +04:00
if ( ! newn )
2005-08-10 07:25:06 +04:00
goto nomem ;
2005-08-24 08:59:41 +04:00
2005-08-10 07:25:06 +04:00
put_child ( t , tn , i / 2 , ( struct node * ) newn ) ;
2005-07-06 02:02:40 +04:00
}
}
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
for ( i = 0 ; i < olen ; i + = 2 ) {
struct tnode * newBinNode ;
2005-06-21 23:43:18 +04:00
left = tnode_get_child ( oldtnode , i ) ;
right = tnode_get_child ( oldtnode , i + 1 ) ;
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
/* At least one of the children is empty */
if ( left = = NULL ) {
if ( right = = NULL ) /* Both are empty */
continue ;
put_child ( t , tn , i / 2 , right ) ;
2005-08-10 07:24:39 +04:00
continue ;
2005-08-24 08:59:41 +04:00
}
2005-08-10 07:24:39 +04:00
if ( right = = NULL ) {
2005-06-21 23:43:18 +04:00
put_child ( t , tn , i / 2 , left ) ;
2005-08-10 07:24:39 +04:00
continue ;
}
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
/* Two nonempty children */
2005-08-10 07:24:39 +04:00
newBinNode = ( struct tnode * ) tnode_get_child ( tn , i / 2 ) ;
put_child ( t , tn , i / 2 , NULL ) ;
put_child ( t , newBinNode , 0 , left ) ;
put_child ( t , newBinNode , 1 , right ) ;
put_child ( t , tn , i / 2 , resize ( t , newBinNode ) ) ;
2005-06-21 23:43:18 +04:00
}
tnode_free ( oldtnode ) ;
return tn ;
2005-08-10 07:25:06 +04:00
nomem :
{
int size = tnode_child_length ( tn ) ;
int j ;
2005-08-24 08:59:41 +04:00
for ( j = 0 ; j < size ; j + + )
2005-08-10 07:25:06 +04:00
if ( tn - > child [ j ] )
tnode_free ( ( struct tnode * ) tn - > child [ j ] ) ;
tnode_free ( tn ) ;
2005-08-24 08:59:41 +04:00
2005-08-10 07:25:06 +04:00
return ERR_PTR ( - ENOMEM ) ;
}
2005-06-21 23:43:18 +04:00
}
2005-08-10 07:24:39 +04:00
static void trie_init ( struct trie * t )
2005-06-21 23:43:18 +04:00
{
2005-08-10 07:24:39 +04:00
if ( ! t )
return ;
t - > size = 0 ;
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( t - > trie , NULL ) ;
2005-08-10 07:24:39 +04:00
t - > revision = 0 ;
2005-06-21 23:43:18 +04:00
# ifdef CONFIG_IP_FIB_TRIE_STATS
2005-08-10 07:24:39 +04:00
memset ( & t - > stats , 0 , sizeof ( struct trie_use_stats ) ) ;
2005-06-21 23:43:18 +04:00
# endif
}
2005-09-20 02:31:18 +04:00
/* readside must use rcu_read_lock currently dump routines
2005-08-26 00:01:29 +04:00
via get_fa_head and dump */
2005-09-20 02:31:18 +04:00
static struct leaf_info * find_leaf_info ( struct leaf * l , int plen )
2005-06-21 23:43:18 +04:00
{
2005-09-20 02:31:18 +04:00
struct hlist_head * head = & l - > list ;
2005-06-21 23:43:18 +04:00
struct hlist_node * node ;
struct leaf_info * li ;
2005-08-26 00:01:29 +04:00
hlist_for_each_entry_rcu ( li , node , head , hlist )
2005-07-20 01:01:51 +04:00
if ( li - > plen = = plen )
2005-06-21 23:43:18 +04:00
return li ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
return NULL ;
}
static inline struct list_head * get_fa_head ( struct leaf * l , int plen )
{
2005-09-20 02:31:18 +04:00
struct leaf_info * li = find_leaf_info ( l , plen ) ;
2005-07-20 01:01:51 +04:00
2005-08-10 07:24:39 +04:00
if ( ! li )
return NULL ;
2005-07-20 01:01:51 +04:00
2005-08-10 07:24:39 +04:00
return & li - > falh ;
2005-06-21 23:43:18 +04:00
}
static void insert_leaf_info ( struct hlist_head * head , struct leaf_info * new )
{
2007-02-09 17:24:47 +03:00
struct leaf_info * li = NULL , * last = NULL ;
struct hlist_node * node ;
if ( hlist_empty ( head ) ) {
hlist_add_head_rcu ( & new - > hlist , head ) ;
} else {
hlist_for_each_entry ( li , node , head , hlist ) {
if ( new - > plen > li - > plen )
break ;
last = li ;
}
if ( last )
hlist_add_after_rcu ( & last - > hlist , & new - > hlist ) ;
else
hlist_add_before_rcu ( & new - > hlist , & li - > hlist ) ;
}
2005-06-21 23:43:18 +04:00
}
2005-08-26 00:01:29 +04:00
/* rcu_read_lock needs to be hold by caller from readside */
2005-06-21 23:43:18 +04:00
static struct leaf *
fib_find_node ( struct trie * t , u32 key )
{
int pos ;
struct tnode * tn ;
struct node * n ;
pos = 0 ;
2005-08-26 00:01:29 +04:00
n = rcu_dereference ( t - > trie ) ;
2005-06-21 23:43:18 +04:00
while ( n ! = NULL & & NODE_TYPE ( n ) = = T_TNODE ) {
tn = ( struct tnode * ) n ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
check_tnode ( tn ) ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( tkey_sub_equals ( tn - > key , pos , tn - > pos - pos , key ) ) {
2005-08-10 07:24:39 +04:00
pos = tn - > pos + tn - > bits ;
2005-06-21 23:43:18 +04:00
n = tnode_get_child ( tn , tkey_extract_bits ( key , tn - > pos , tn - > bits ) ) ;
2005-08-10 07:24:39 +04:00
} else
2005-06-21 23:43:18 +04:00
break ;
}
/* Case we have found a leaf. Compare prefixes */
2005-08-10 07:24:39 +04:00
if ( n ! = NULL & & IS_LEAF ( n ) & & tkey_equals ( key , n - > key ) )
return ( struct leaf * ) n ;
2005-06-21 23:43:18 +04:00
return NULL ;
}
static struct node * trie_rebalance ( struct trie * t , struct tnode * tn )
{
int wasfull ;
t_key cindex , key ;
struct tnode * tp = NULL ;
key = tn - > key ;
while ( tn ! = NULL & & NODE_PARENT ( tn ) ! = NULL ) {
tp = NODE_PARENT ( tn ) ;
cindex = tkey_extract_bits ( key , tp - > pos , tp - > bits ) ;
wasfull = tnode_full ( tp , tnode_get_child ( tp , cindex ) ) ;
tn = ( struct tnode * ) resize ( t , ( struct tnode * ) tn ) ;
tnode_put_child_reorg ( ( struct tnode * ) tp , cindex , ( struct node * ) tn , wasfull ) ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( ! NODE_PARENT ( tn ) )
2005-06-21 23:43:18 +04:00
break ;
tn = NODE_PARENT ( tn ) ;
}
/* Handle last (top) tnode */
2005-07-20 01:01:51 +04:00
if ( IS_TNODE ( tn ) )
2005-06-21 23:43:18 +04:00
tn = ( struct tnode * ) resize ( t , ( struct tnode * ) tn ) ;
return ( struct node * ) tn ;
}
2005-08-26 00:01:29 +04:00
/* only used from updater-side */
2005-06-29 02:00:39 +04:00
static struct list_head *
fib_insert_node ( struct trie * t , int * err , u32 key , int plen )
2005-06-21 23:43:18 +04:00
{
int pos , newpos ;
struct tnode * tp = NULL , * tn = NULL ;
struct node * n ;
struct leaf * l ;
int missbit ;
2005-07-20 01:01:51 +04:00
struct list_head * fa_head = NULL ;
2005-06-21 23:43:18 +04:00
struct leaf_info * li ;
t_key cindex ;
pos = 0 ;
2005-07-20 01:01:51 +04:00
n = t - > trie ;
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
/* If we point to NULL, stop. Either the tree is empty and we should
* just put a new leaf in if , or we have reached an empty child slot ,
2005-06-21 23:43:18 +04:00
* and we should just put our new leaf in that .
2005-07-20 01:01:51 +04:00
* If we point to a T_TNODE , check if it matches our key . Note that
* a T_TNODE might be skipping any number of bits - its ' pos ' need
2005-06-21 23:43:18 +04:00
* not be the parent ' s ' pos ' + ' bits ' !
*
2005-07-20 01:01:51 +04:00
* If it does match the current key , get pos / bits from it , extract
2005-06-21 23:43:18 +04:00
* the index from our key , push the T_TNODE and walk the tree .
*
* If it doesn ' t , we have to replace it with a new T_TNODE .
*
2005-07-20 01:01:51 +04:00
* If we point to a T_LEAF , it might or might not have the same key
* as we do . If it does , just change the value , update the T_LEAF ' s
* value , and return it .
2005-06-21 23:43:18 +04:00
* If it doesn ' t , we need to replace it with a T_TNODE .
*/
while ( n ! = NULL & & NODE_TYPE ( n ) = = T_TNODE ) {
tn = ( struct tnode * ) n ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
check_tnode ( tn ) ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( tkey_sub_equals ( tn - > key , pos , tn - > pos - pos , key ) ) {
2005-06-21 23:43:18 +04:00
tp = tn ;
2005-08-10 07:24:39 +04:00
pos = tn - > pos + tn - > bits ;
2005-06-21 23:43:18 +04:00
n = tnode_get_child ( tn , tkey_extract_bits ( key , tn - > pos , tn - > bits ) ) ;
2005-08-24 08:59:41 +04:00
BUG_ON ( n & & NODE_PARENT ( n ) ! = tn ) ;
2005-08-10 07:24:39 +04:00
} else
2005-06-21 23:43:18 +04:00
break ;
}
/*
* n - - - - > NULL , LEAF or TNODE
*
2005-07-20 01:01:51 +04:00
* tp is n ' s ( parent ) - - - - > NULL or TNODE
2005-06-21 23:43:18 +04:00
*/
2005-08-10 07:24:39 +04:00
BUG_ON ( tp & & IS_LEAF ( tp ) ) ;
2005-06-21 23:43:18 +04:00
/* Case 1: n is a leaf. Compare prefixes */
2005-07-20 01:01:51 +04:00
if ( n ! = NULL & & IS_LEAF ( n ) & & tkey_equals ( key , n - > key ) ) {
2005-08-10 07:24:39 +04:00
struct leaf * l = ( struct leaf * ) n ;
2005-06-21 23:43:18 +04:00
li = leaf_info_new ( plen ) ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( ! li ) {
2005-06-29 02:00:39 +04:00
* err = - ENOMEM ;
goto err ;
}
2005-06-21 23:43:18 +04:00
fa_head = & li - > falh ;
insert_leaf_info ( & l - > list , li ) ;
goto done ;
}
t - > size + + ;
l = leaf_new ( ) ;
2005-07-20 01:01:51 +04:00
if ( ! l ) {
2005-06-29 02:00:39 +04:00
* err = - ENOMEM ;
goto err ;
}
2005-06-21 23:43:18 +04:00
l - > key = key ;
li = leaf_info_new ( plen ) ;
2005-07-20 01:01:51 +04:00
if ( ! li ) {
2005-06-29 02:00:39 +04:00
tnode_free ( ( struct tnode * ) l ) ;
* err = - ENOMEM ;
goto err ;
}
2005-06-21 23:43:18 +04:00
fa_head = & li - > falh ;
insert_leaf_info ( & l - > list , li ) ;
if ( t - > trie & & n = = NULL ) {
2005-08-10 07:24:39 +04:00
/* Case 2: n is NULL, and will just insert a new leaf */
2005-06-21 23:43:18 +04:00
NODE_SET_PARENT ( l , tp ) ;
2005-08-10 07:24:39 +04:00
cindex = tkey_extract_bits ( key , tp - > pos , tp - > bits ) ;
put_child ( t , ( struct tnode * ) tp , cindex , ( struct node * ) l ) ;
} else {
/* Case 3: n is a LEAF or a TNODE and the key doesn't match. */
2005-07-20 01:01:51 +04:00
/*
* Add a new tnode here
2005-06-21 23:43:18 +04:00
* first tnode need some special handling
*/
if ( tp )
2005-08-10 07:24:39 +04:00
pos = tp - > pos + tp - > bits ;
2005-06-21 23:43:18 +04:00
else
2005-08-10 07:24:39 +04:00
pos = 0 ;
2005-07-20 01:01:51 +04:00
if ( n ) {
2005-06-21 23:43:18 +04:00
newpos = tkey_mismatch ( key , pos , n - > key ) ;
tn = tnode_new ( n - > key , newpos , 1 ) ;
2005-08-10 07:24:39 +04:00
} else {
2005-06-21 23:43:18 +04:00
newpos = 0 ;
2005-07-20 01:01:51 +04:00
tn = tnode_new ( key , newpos , 1 ) ; /* First tnode */
2005-06-21 23:43:18 +04:00
}
2005-07-20 01:01:51 +04:00
if ( ! tn ) {
2005-06-29 02:00:39 +04:00
free_leaf_info ( li ) ;
tnode_free ( ( struct tnode * ) l ) ;
* err = - ENOMEM ;
goto err ;
2005-08-10 07:24:39 +04:00
}
2005-06-21 23:43:18 +04:00
NODE_SET_PARENT ( tn , tp ) ;
2005-08-10 07:24:39 +04:00
missbit = tkey_extract_bits ( key , newpos , 1 ) ;
2005-06-21 23:43:18 +04:00
put_child ( t , tn , missbit , ( struct node * ) l ) ;
put_child ( t , tn , 1 - missbit , n ) ;
2005-07-20 01:01:51 +04:00
if ( tp ) {
2005-06-21 23:43:18 +04:00
cindex = tkey_extract_bits ( key , tp - > pos , tp - > bits ) ;
put_child ( t , ( struct tnode * ) tp , cindex , ( struct node * ) tn ) ;
2005-08-10 07:24:39 +04:00
} else {
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( t - > trie , ( struct node * ) tn ) ; /* First tnode */
2005-06-21 23:43:18 +04:00
tp = tn ;
}
}
2005-08-10 07:24:39 +04:00
if ( tp & & tp - > pos + tp - > bits > 32 )
2005-09-21 11:15:39 +04:00
printk ( KERN_WARNING " fib_trie tp=%p pos=%d, bits=%d, key=%0x plen=%d \n " ,
2005-06-21 23:43:18 +04:00
tp , tp - > pos , tp - > bits , key , plen ) ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
/* Rebalance the trie */
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( t - > trie , trie_rebalance ( t , tp ) ) ;
2005-06-29 02:00:39 +04:00
done :
t - > revision + + ;
2005-08-10 07:24:39 +04:00
err :
2005-06-21 23:43:18 +04:00
return fa_head ;
}
2007-03-27 01:22:22 +04:00
/*
* Caller must hold RTNL .
*/
2006-08-18 05:14:52 +04:00
static int fn_trie_insert ( struct fib_table * tb , struct fib_config * cfg )
2005-06-21 23:43:18 +04:00
{
struct trie * t = ( struct trie * ) tb - > tb_data ;
struct fib_alias * fa , * new_fa ;
2005-07-20 01:01:51 +04:00
struct list_head * fa_head = NULL ;
2005-06-21 23:43:18 +04:00
struct fib_info * fi ;
2006-08-18 05:14:52 +04:00
int plen = cfg - > fc_dst_len ;
u8 tos = cfg - > fc_tos ;
2005-06-21 23:43:18 +04:00
u32 key , mask ;
int err ;
struct leaf * l ;
if ( plen > 32 )
return - EINVAL ;
2006-08-18 05:14:52 +04:00
key = ntohl ( cfg - > fc_dst ) ;
2005-06-21 23:43:18 +04:00
2006-08-11 10:08:33 +04:00
pr_debug ( " Insert table=%u %08x/%d \n " , tb - > tb_id , key , plen ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
mask = ntohl ( inet_make_mask ( plen ) ) ;
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
if ( key & ~ mask )
2005-06-21 23:43:18 +04:00
return - EINVAL ;
key = key & mask ;
2006-08-18 05:14:52 +04:00
fi = fib_create_info ( cfg ) ;
if ( IS_ERR ( fi ) ) {
err = PTR_ERR ( fi ) ;
2005-06-21 23:43:18 +04:00
goto err ;
2006-08-18 05:14:52 +04:00
}
2005-06-21 23:43:18 +04:00
l = fib_find_node ( t , key ) ;
2005-07-20 01:01:51 +04:00
fa = NULL ;
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
if ( l ) {
2005-06-21 23:43:18 +04:00
fa_head = get_fa_head ( l , plen ) ;
fa = fib_find_alias ( fa_head , tos , fi - > fib_priority ) ;
}
/* Now fa, if non-NULL, points to the first fib alias
* with the same keys [ prefix , tos , priority ] , if such key already
* exists or to the node before which we will insert new one .
*
* If fa is NULL , we will need to allocate a new one and
* insert to the head of f .
*
* If f is NULL , no fib node matched the destination key
* and we need to allocate a new one of those as well .
*/
2005-08-10 07:24:39 +04:00
if ( fa & & fa - > fa_info - > fib_priority = = fi - > fib_priority ) {
2005-06-21 23:43:18 +04:00
struct fib_alias * fa_orig ;
err = - EEXIST ;
2006-08-18 05:14:52 +04:00
if ( cfg - > fc_nlflags & NLM_F_EXCL )
2005-06-21 23:43:18 +04:00
goto out ;
2006-08-18 05:14:52 +04:00
if ( cfg - > fc_nlflags & NLM_F_REPLACE ) {
2005-06-21 23:43:18 +04:00
struct fib_info * fi_drop ;
u8 state ;
2005-08-26 00:01:29 +04:00
err = - ENOBUFS ;
2006-12-07 07:33:17 +03:00
new_fa = kmem_cache_alloc ( fn_alias_kmem , GFP_KERNEL ) ;
2005-08-26 00:01:29 +04:00
if ( new_fa = = NULL )
goto out ;
2005-06-21 23:43:18 +04:00
fi_drop = fa - > fa_info ;
2005-08-26 00:01:29 +04:00
new_fa - > fa_tos = fa - > fa_tos ;
new_fa - > fa_info = fi ;
2006-08-18 05:14:52 +04:00
new_fa - > fa_type = cfg - > fc_type ;
new_fa - > fa_scope = cfg - > fc_scope ;
2005-06-21 23:43:18 +04:00
state = fa - > fa_state ;
2005-08-26 00:01:29 +04:00
new_fa - > fa_state & = ~ FA_S_ACCESSED ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
list_replace_rcu ( & fa - > fa_list , & new_fa - > fa_list ) ;
alias_free_mem_rcu ( fa ) ;
2005-06-21 23:43:18 +04:00
fib_release_info ( fi_drop ) ;
if ( state & FA_S_ACCESSED )
2005-08-10 07:24:39 +04:00
rt_cache_flush ( - 1 ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
goto succeeded ;
2005-06-21 23:43:18 +04:00
}
/* Error if we find a perfect match which
* uses the same scope , type , and nexthop
* information .
*/
fa_orig = fa ;
list_for_each_entry ( fa , fa_orig - > fa_list . prev , fa_list ) {
if ( fa - > fa_tos ! = tos )
break ;
if ( fa - > fa_info - > fib_priority ! = fi - > fib_priority )
break ;
2006-08-18 05:14:52 +04:00
if ( fa - > fa_type = = cfg - > fc_type & &
fa - > fa_scope = = cfg - > fc_scope & &
2005-06-21 23:43:18 +04:00
fa - > fa_info = = fi ) {
goto out ;
}
}
2006-08-18 05:14:52 +04:00
if ( ! ( cfg - > fc_nlflags & NLM_F_APPEND ) )
2005-06-21 23:43:18 +04:00
fa = fa_orig ;
}
err = - ENOENT ;
2006-08-18 05:14:52 +04:00
if ( ! ( cfg - > fc_nlflags & NLM_F_CREATE ) )
2005-06-21 23:43:18 +04:00
goto out ;
err = - ENOBUFS ;
2006-12-07 07:33:17 +03:00
new_fa = kmem_cache_alloc ( fn_alias_kmem , GFP_KERNEL ) ;
2005-06-21 23:43:18 +04:00
if ( new_fa = = NULL )
goto out ;
new_fa - > fa_info = fi ;
new_fa - > fa_tos = tos ;
2006-08-18 05:14:52 +04:00
new_fa - > fa_type = cfg - > fc_type ;
new_fa - > fa_scope = cfg - > fc_scope ;
2005-06-21 23:43:18 +04:00
new_fa - > fa_state = 0 ;
/*
* Insert new entry to the list .
*/
2005-07-20 01:01:51 +04:00
if ( ! fa_head ) {
2005-06-29 02:00:39 +04:00
err = 0 ;
2006-07-13 00:29:56 +04:00
fa_head = fib_insert_node ( t , & err , key , plen ) ;
2005-07-20 01:01:51 +04:00
if ( err )
2005-06-29 02:00:39 +04:00
goto out_free_new_fa ;
}
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
list_add_tail_rcu ( & new_fa - > fa_list ,
( fa ? & fa - > fa_list : fa_head ) ) ;
2005-06-21 23:43:18 +04:00
rt_cache_flush ( - 1 ) ;
2006-08-18 05:14:52 +04:00
rtmsg_fib ( RTM_NEWROUTE , htonl ( key ) , new_fa , plen , tb - > tb_id ,
& cfg - > fc_nlinfo ) ;
2005-06-21 23:43:18 +04:00
succeeded :
return 0 ;
2005-06-29 02:00:39 +04:00
out_free_new_fa :
kmem_cache_free ( fn_alias_kmem , new_fa ) ;
2005-06-21 23:43:18 +04:00
out :
fib_release_info ( fi ) ;
2005-08-10 07:24:39 +04:00
err :
2005-06-21 23:43:18 +04:00
return err ;
}
2005-08-26 00:01:29 +04:00
2005-09-20 02:31:18 +04:00
/* should be called with rcu_read_lock */
2005-08-24 08:59:41 +04:00
static inline int check_leaf ( struct trie * t , struct leaf * l ,
t_key key , int * plen , const struct flowi * flp ,
2005-08-24 09:06:09 +04:00
struct fib_result * res )
2005-06-21 23:43:18 +04:00
{
2005-08-24 09:06:09 +04:00
int err , i ;
2006-09-20 00:42:46 +04:00
__be32 mask ;
2005-06-21 23:43:18 +04:00
struct leaf_info * li ;
struct hlist_head * hhead = & l - > list ;
struct hlist_node * node ;
2005-07-20 01:01:51 +04:00
2005-08-26 00:01:29 +04:00
hlist_for_each_entry_rcu ( li , node , hhead , hlist ) {
2005-06-21 23:43:18 +04:00
i = li - > plen ;
2006-09-20 00:42:46 +04:00
mask = inet_make_mask ( i ) ;
if ( l - > key ! = ( key & ntohl ( mask ) ) )
2005-06-21 23:43:18 +04:00
continue ;
2006-09-20 00:42:46 +04:00
if ( ( err = fib_semantic_match ( & li - > falh , flp , res , htonl ( l - > key ) , mask , i ) ) < = 0 ) {
2005-06-21 23:43:18 +04:00
* plen = i ;
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . semantic_match_passed + + ;
# endif
2005-08-24 09:06:09 +04:00
return err ;
2005-06-21 23:43:18 +04:00
}
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . semantic_match_miss + + ;
# endif
}
2005-08-24 09:06:09 +04:00
return 1 ;
2005-06-21 23:43:18 +04:00
}
static int
fn_trie_lookup ( struct fib_table * tb , const struct flowi * flp , struct fib_result * res )
{
struct trie * t = ( struct trie * ) tb - > tb_data ;
int plen , ret = 0 ;
struct node * n ;
struct tnode * pn ;
int pos , bits ;
2005-08-10 07:24:39 +04:00
t_key key = ntohl ( flp - > fl4_dst ) ;
2005-06-21 23:43:18 +04:00
int chopped_off ;
t_key cindex = 0 ;
int current_prefix_length = KEYLENGTH ;
2005-08-10 07:24:39 +04:00
struct tnode * cn ;
t_key node_prefix , key_prefix , pref_mismatch ;
int mp ;
2005-08-26 00:01:29 +04:00
rcu_read_lock ( ) ;
2005-08-10 07:24:39 +04:00
2005-08-26 00:01:29 +04:00
n = rcu_dereference ( t - > trie ) ;
2005-07-20 01:01:51 +04:00
if ( ! n )
2005-06-21 23:43:18 +04:00
goto failed ;
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . gets + + ;
# endif
/* Just a leaf? */
if ( IS_LEAF ( n ) ) {
2005-08-24 09:06:09 +04:00
if ( ( ret = check_leaf ( t , ( struct leaf * ) n , key , & plen , flp , res ) ) < = 0 )
2005-06-21 23:43:18 +04:00
goto found ;
goto failed ;
}
pn = ( struct tnode * ) n ;
chopped_off = 0 ;
2005-07-20 01:01:51 +04:00
2005-08-10 07:24:39 +04:00
while ( pn ) {
2005-06-21 23:43:18 +04:00
pos = pn - > pos ;
bits = pn - > bits ;
2005-07-20 01:01:51 +04:00
if ( ! chopped_off )
2005-06-21 23:43:18 +04:00
cindex = tkey_extract_bits ( MASK_PFX ( key , current_prefix_length ) , pos , bits ) ;
n = tnode_get_child ( pn , cindex ) ;
if ( n = = NULL ) {
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . null_node_hit + + ;
# endif
goto backtrace ;
}
2005-08-10 07:24:39 +04:00
if ( IS_LEAF ( n ) ) {
if ( ( ret = check_leaf ( t , ( struct leaf * ) n , key , & plen , flp , res ) ) < = 0 )
goto found ;
else
goto backtrace ;
}
2005-06-21 23:43:18 +04:00
# define HL_OPTIMIZE
# ifdef HL_OPTIMIZE
2005-08-10 07:24:39 +04:00
cn = ( struct tnode * ) n ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/*
* It ' s a tnode , and we can do some extra checks here if we
* like , to avoid descending into a dead - end branch .
* This tnode is in the parent ' s child array at index
* key [ p_pos . . p_pos + p_bits ] but potentially with some bits
* chopped off , so in reality the index may be just a
* subprefix , padded with zero at the end .
* We can also take a look at any skipped bits in this
* tnode - everything up to p_pos is supposed to be ok ,
* and the non - chopped bits of the index ( se previous
* paragraph ) are also guaranteed ok , but the rest is
* considered unknown .
*
* The skipped bits are key [ pos + bits . . cn - > pos ] .
*/
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/* If current_prefix_length < pos+bits, we are already doing
* actual prefix matching , which means everything from
* pos + ( bits - chopped_off ) onward must be zero along some
* branch of this subtree - otherwise there is * no * valid
* prefix present . Here we can only check the skipped
* bits . Remember , since we have already indexed into the
* parent ' s child array , we know that the bits we chopped of
* * are * zero .
*/
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/* NOTA BENE: CHECKING ONLY SKIPPED BITS FOR THE NEW NODE HERE */
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
if ( current_prefix_length < pos + bits ) {
if ( tkey_extract_bits ( cn - > key , current_prefix_length ,
cn - > pos - current_prefix_length ) ! = 0 | |
! ( cn - > child [ 0 ] ) )
goto backtrace ;
}
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/*
* If chopped_off = 0 , the index is fully validated and we
* only need to look at the skipped bits for this , the new ,
* tnode . What we actually want to do is to find out if
* these skipped bits match our key perfectly , or if we will
* have to count on finding a matching prefix further down ,
* because if we do , we would like to have some way of
* verifying the existence of such a prefix at this point .
*/
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/* The only thing we can do at this point is to verify that
* any such matching prefix can indeed be a prefix to our
* key , and if the bits in the node we are inspecting that
* do not match our key are not ZERO , this cannot be true .
* Thus , find out where there is a mismatch ( before cn - > pos )
* and verify that all the mismatching bits are zero in the
* new tnode ' s key .
*/
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
/* Note: We aren't very concerned about the piece of the key
* that precede pn - > pos + pn - > bits , since these have already been
* checked . The bits after cn - > pos aren ' t checked since these are
* by definition " unknown " at this point . Thus , what we want to
* see is if we are about to enter the " prefix matching " state ,
* and in that case verify that the skipped bits that will prevail
* throughout this subtree are zero , as they have to be if we are
* to find a matching prefix .
*/
node_prefix = MASK_PFX ( cn - > key , cn - > pos ) ;
key_prefix = MASK_PFX ( key , cn - > pos ) ;
pref_mismatch = key_prefix ^ node_prefix ;
mp = 0 ;
/* In short: If skipped bits in this node do not match the search
* key , enter the " prefix matching " state . directly .
*/
if ( pref_mismatch ) {
while ( ! ( pref_mismatch & ( 1 < < ( KEYLENGTH - 1 ) ) ) ) {
mp + + ;
pref_mismatch = pref_mismatch < < 1 ;
}
key_prefix = tkey_extract_bits ( cn - > key , mp , cn - > pos - mp ) ;
if ( key_prefix ! = 0 )
goto backtrace ;
if ( current_prefix_length > = cn - > pos )
current_prefix_length = mp ;
2005-07-20 01:01:51 +04:00
}
2005-08-10 07:24:39 +04:00
# endif
pn = ( struct tnode * ) n ; /* Descend */
chopped_off = 0 ;
continue ;
2005-06-21 23:43:18 +04:00
backtrace :
chopped_off + + ;
/* As zero don't change the child key (cindex) */
2005-08-10 07:24:39 +04:00
while ( ( chopped_off < = pn - > bits ) & & ! ( cindex & ( 1 < < ( chopped_off - 1 ) ) ) )
2005-06-21 23:43:18 +04:00
chopped_off + + ;
/* Decrease current_... with bits chopped off */
if ( current_prefix_length > pn - > pos + pn - > bits - chopped_off )
current_prefix_length = pn - > pos + pn - > bits - chopped_off ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
/*
2005-07-20 01:01:51 +04:00
* Either we do the actual chop off according or if we have
2005-06-21 23:43:18 +04:00
* chopped off all bits in this tnode walk up to our parent .
*/
2005-08-10 07:24:39 +04:00
if ( chopped_off < = pn - > bits ) {
2005-06-21 23:43:18 +04:00
cindex & = ~ ( 1 < < ( chopped_off - 1 ) ) ;
2005-08-10 07:24:39 +04:00
} else {
2005-07-20 01:01:51 +04:00
if ( NODE_PARENT ( pn ) = = NULL )
2005-06-21 23:43:18 +04:00
goto failed ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
/* Get Child's index */
cindex = tkey_extract_bits ( pn - > key , NODE_PARENT ( pn ) - > pos , NODE_PARENT ( pn ) - > bits ) ;
pn = NODE_PARENT ( pn ) ;
chopped_off = 0 ;
# ifdef CONFIG_IP_FIB_TRIE_STATS
t - > stats . backtrack + + ;
# endif
goto backtrace ;
2005-07-20 01:01:51 +04:00
}
2005-06-21 23:43:18 +04:00
}
failed :
2005-07-20 01:01:51 +04:00
ret = 1 ;
2005-06-21 23:43:18 +04:00
found :
2005-08-26 00:01:29 +04:00
rcu_read_unlock ( ) ;
2005-06-21 23:43:18 +04:00
return ret ;
}
2005-08-26 00:01:29 +04:00
/* only called from updater side */
2005-06-21 23:43:18 +04:00
static int trie_leaf_remove ( struct trie * t , t_key key )
{
t_key cindex ;
struct tnode * tp = NULL ;
struct node * n = t - > trie ;
struct leaf * l ;
2005-08-24 08:59:41 +04:00
pr_debug ( " entering trie_leaf_remove(%p) \n " , n ) ;
2005-06-21 23:43:18 +04:00
/* Note that in the case skipped bits, those bits are *not* checked!
2005-07-20 01:01:51 +04:00
* When we finish this , we will have NULL or a T_LEAF , and the
2005-06-21 23:43:18 +04:00
* T_LEAF may or may not match our key .
*/
2005-08-10 07:24:39 +04:00
while ( n ! = NULL & & IS_TNODE ( n ) ) {
2005-06-21 23:43:18 +04:00
struct tnode * tn = ( struct tnode * ) n ;
check_tnode ( tn ) ;
n = tnode_get_child ( tn , tkey_extract_bits ( key , tn - > pos , tn - > bits ) ) ;
2005-08-24 08:59:41 +04:00
BUG_ON ( n & & NODE_PARENT ( n ) ! = tn ) ;
2005-08-10 07:24:39 +04:00
}
2005-06-21 23:43:18 +04:00
l = ( struct leaf * ) n ;
2005-07-20 01:01:51 +04:00
if ( ! n | | ! tkey_equals ( l - > key , key ) )
2005-06-21 23:43:18 +04:00
return 0 ;
2005-07-20 01:01:51 +04:00
/*
* Key found .
* Remove the leaf and rebalance the tree
2005-06-21 23:43:18 +04:00
*/
t - > revision + + ;
t - > size - - ;
tp = NODE_PARENT ( n ) ;
tnode_free ( ( struct tnode * ) n ) ;
2005-07-20 01:01:51 +04:00
if ( tp ) {
2005-06-21 23:43:18 +04:00
cindex = tkey_extract_bits ( key , tp - > pos , tp - > bits ) ;
put_child ( t , ( struct tnode * ) tp , cindex , NULL ) ;
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( t - > trie , trie_rebalance ( t , tp ) ) ;
2005-08-10 07:24:39 +04:00
} else
2005-08-26 00:01:29 +04:00
rcu_assign_pointer ( t - > trie , NULL ) ;
2005-06-21 23:43:18 +04:00
return 1 ;
}
2007-03-27 01:22:22 +04:00
/*
* Caller must hold RTNL .
*/
2006-08-18 05:14:52 +04:00
static int fn_trie_delete ( struct fib_table * tb , struct fib_config * cfg )
2005-06-21 23:43:18 +04:00
{
struct trie * t = ( struct trie * ) tb - > tb_data ;
u32 key , mask ;
2006-08-18 05:14:52 +04:00
int plen = cfg - > fc_dst_len ;
u8 tos = cfg - > fc_tos ;
2005-06-21 23:43:18 +04:00
struct fib_alias * fa , * fa_to_delete ;
struct list_head * fa_head ;
struct leaf * l ;
2005-08-10 07:24:39 +04:00
struct leaf_info * li ;
2005-07-20 01:01:51 +04:00
if ( plen > 32 )
2005-06-21 23:43:18 +04:00
return - EINVAL ;
2006-08-18 05:14:52 +04:00
key = ntohl ( cfg - > fc_dst ) ;
2005-08-10 07:24:39 +04:00
mask = ntohl ( inet_make_mask ( plen ) ) ;
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
if ( key & ~ mask )
2005-06-21 23:43:18 +04:00
return - EINVAL ;
key = key & mask ;
l = fib_find_node ( t , key ) ;
2005-07-20 01:01:51 +04:00
if ( ! l )
2005-06-21 23:43:18 +04:00
return - ESRCH ;
fa_head = get_fa_head ( l , plen ) ;
fa = fib_find_alias ( fa_head , tos , 0 ) ;
if ( ! fa )
return - ESRCH ;
2005-08-24 08:59:41 +04:00
pr_debug ( " Deleting %08x/%d tos=%d t=%p \n " , key , plen , tos , t ) ;
2005-06-21 23:43:18 +04:00
fa_to_delete = NULL ;
fa_head = fa - > fa_list . prev ;
2005-08-26 00:01:29 +04:00
2005-06-21 23:43:18 +04:00
list_for_each_entry ( fa , fa_head , fa_list ) {
struct fib_info * fi = fa - > fa_info ;
if ( fa - > fa_tos ! = tos )
break ;
2006-08-18 05:14:52 +04:00
if ( ( ! cfg - > fc_type | | fa - > fa_type = = cfg - > fc_type ) & &
( cfg - > fc_scope = = RT_SCOPE_NOWHERE | |
fa - > fa_scope = = cfg - > fc_scope ) & &
( ! cfg - > fc_protocol | |
fi - > fib_protocol = = cfg - > fc_protocol ) & &
fib_nh_match ( cfg , fi ) = = 0 ) {
2005-06-21 23:43:18 +04:00
fa_to_delete = fa ;
break ;
}
}
2005-08-10 07:24:39 +04:00
if ( ! fa_to_delete )
return - ESRCH ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
fa = fa_to_delete ;
2006-08-18 05:14:52 +04:00
rtmsg_fib ( RTM_DELROUTE , htonl ( key ) , fa , plen , tb - > tb_id ,
& cfg - > fc_nlinfo ) ;
2005-08-10 07:24:39 +04:00
l = fib_find_node ( t , key ) ;
2005-09-20 02:31:18 +04:00
li = find_leaf_info ( l , plen ) ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
list_del_rcu ( & fa - > fa_list ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
if ( list_empty ( fa_head ) ) {
2005-08-26 00:01:29 +04:00
hlist_del_rcu ( & li - > hlist ) ;
2005-08-10 07:24:39 +04:00
free_leaf_info ( li ) ;
2005-08-26 00:01:29 +04:00
}
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
if ( hlist_empty ( & l - > list ) )
trie_leaf_remove ( t , key ) ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
if ( fa - > fa_state & FA_S_ACCESSED )
rt_cache_flush ( - 1 ) ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
fib_release_info ( fa - > fa_info ) ;
alias_free_mem_rcu ( fa ) ;
2005-08-10 07:24:39 +04:00
return 0 ;
2005-06-21 23:43:18 +04:00
}
static int trie_flush_list ( struct trie * t , struct list_head * head )
{
struct fib_alias * fa , * fa_node ;
int found = 0 ;
list_for_each_entry_safe ( fa , fa_node , head , fa_list ) {
struct fib_info * fi = fa - > fa_info ;
2005-08-26 00:01:29 +04:00
if ( fi & & ( fi - > fib_flags & RTNH_F_DEAD ) ) {
list_del_rcu ( & fa - > fa_list ) ;
fib_release_info ( fa - > fa_info ) ;
alias_free_mem_rcu ( fa ) ;
2005-06-21 23:43:18 +04:00
found + + ;
}
}
return found ;
}
static int trie_flush_leaf ( struct trie * t , struct leaf * l )
{
int found = 0 ;
struct hlist_head * lih = & l - > list ;
struct hlist_node * node , * tmp ;
struct leaf_info * li = NULL ;
hlist_for_each_entry_safe ( li , node , tmp , lih , hlist ) {
found + = trie_flush_list ( t , & li - > falh ) ;
if ( list_empty ( & li - > falh ) ) {
2005-08-26 00:01:29 +04:00
hlist_del_rcu ( & li - > hlist ) ;
2005-06-21 23:43:18 +04:00
free_leaf_info ( li ) ;
}
}
return found ;
}
2005-08-26 00:01:29 +04:00
/* rcu_read_lock needs to be hold by caller from readside */
2005-06-21 23:43:18 +04:00
static struct leaf * nextleaf ( struct trie * t , struct leaf * thisleaf )
{
struct node * c = ( struct node * ) thisleaf ;
struct tnode * p ;
int idx ;
2005-08-26 00:01:29 +04:00
struct node * trie = rcu_dereference ( t - > trie ) ;
2005-06-21 23:43:18 +04:00
2005-07-20 01:01:51 +04:00
if ( c = = NULL ) {
2005-08-26 00:01:29 +04:00
if ( trie = = NULL )
2005-06-21 23:43:18 +04:00
return NULL ;
2005-08-26 00:01:29 +04:00
if ( IS_LEAF ( trie ) ) /* trie w. just a leaf */
return ( struct leaf * ) trie ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
p = ( struct tnode * ) trie ; /* Start */
2005-08-10 07:24:39 +04:00
} else
2005-06-21 23:43:18 +04:00
p = ( struct tnode * ) NODE_PARENT ( c ) ;
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
while ( p ) {
int pos , last ;
/* Find the next child of the parent */
2005-07-20 01:01:51 +04:00
if ( c )
pos = 1 + tkey_extract_bits ( c - > key , p - > pos , p - > bits ) ;
else
2005-06-21 23:43:18 +04:00
pos = 0 ;
last = 1 < < p - > bits ;
2005-08-10 07:24:39 +04:00
for ( idx = pos ; idx < last ; idx + + ) {
2005-08-26 00:01:29 +04:00
c = rcu_dereference ( p - > child [ idx ] ) ;
if ( ! c )
2005-08-10 07:24:39 +04:00
continue ;
/* Decend if tnode */
2005-08-26 00:01:29 +04:00
while ( IS_TNODE ( c ) ) {
p = ( struct tnode * ) c ;
2007-02-09 17:24:47 +03:00
idx = 0 ;
2005-08-10 07:24:39 +04:00
/* Rightmost non-NULL branch */
if ( p & & IS_TNODE ( p ) )
2005-08-26 00:01:29 +04:00
while ( ! ( c = rcu_dereference ( p - > child [ idx ] ) )
& & idx < ( 1 < < p - > bits ) ) idx + + ;
2005-08-10 07:24:39 +04:00
/* Done with this tnode? */
2005-08-26 00:01:29 +04:00
if ( idx > = ( 1 < < p - > bits ) | | ! c )
2005-08-10 07:24:39 +04:00
goto up ;
2005-06-21 23:43:18 +04:00
}
2005-08-26 00:01:29 +04:00
return ( struct leaf * ) c ;
2005-06-21 23:43:18 +04:00
}
up :
/* No more children go up one step */
2005-08-10 07:24:39 +04:00
c = ( struct node * ) p ;
2005-06-21 23:43:18 +04:00
p = ( struct tnode * ) NODE_PARENT ( p ) ;
}
return NULL ; /* Ready. Root of trie */
}
2007-03-27 01:22:22 +04:00
/*
* Caller must hold RTNL .
*/
2005-06-21 23:43:18 +04:00
static int fn_trie_flush ( struct fib_table * tb )
{
struct trie * t = ( struct trie * ) tb - > tb_data ;
struct leaf * ll = NULL , * l = NULL ;
int found = 0 , h ;
t - > revision + + ;
2005-08-10 07:24:39 +04:00
for ( h = 0 ; ( l = nextleaf ( t , l ) ) ! = NULL ; h + + ) {
2005-06-21 23:43:18 +04:00
found + = trie_flush_leaf ( t , l ) ;
if ( ll & & hlist_empty ( & ll - > list ) )
trie_leaf_remove ( t , ll - > key ) ;
ll = l ;
}
if ( ll & & hlist_empty ( & ll - > list ) )
trie_leaf_remove ( t , ll - > key ) ;
2005-08-24 08:59:41 +04:00
pr_debug ( " trie_flush found=%d \n " , found ) ;
2005-06-21 23:43:18 +04:00
return found ;
}
2005-08-10 07:24:39 +04:00
static int trie_last_dflt = - 1 ;
2005-06-21 23:43:18 +04:00
static void
fn_trie_select_default ( struct fib_table * tb , const struct flowi * flp , struct fib_result * res )
{
struct trie * t = ( struct trie * ) tb - > tb_data ;
int order , last_idx ;
struct fib_info * fi = NULL ;
struct fib_info * last_resort ;
struct fib_alias * fa = NULL ;
struct list_head * fa_head ;
struct leaf * l ;
last_idx = - 1 ;
last_resort = NULL ;
order = - 1 ;
2005-08-26 00:01:29 +04:00
rcu_read_lock ( ) ;
2005-07-20 01:01:51 +04:00
2005-06-21 23:43:18 +04:00
l = fib_find_node ( t , 0 ) ;
2005-07-20 01:01:51 +04:00
if ( ! l )
2005-06-21 23:43:18 +04:00
goto out ;
fa_head = get_fa_head ( l , 0 ) ;
2005-07-20 01:01:51 +04:00
if ( ! fa_head )
2005-06-21 23:43:18 +04:00
goto out ;
2005-07-20 01:01:51 +04:00
if ( list_empty ( fa_head ) )
2005-06-21 23:43:18 +04:00
goto out ;
2005-08-26 00:01:29 +04:00
list_for_each_entry_rcu ( fa , fa_head , fa_list ) {
2005-06-21 23:43:18 +04:00
struct fib_info * next_fi = fa - > fa_info ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
if ( fa - > fa_scope ! = res - > scope | |
fa - > fa_type ! = RTN_UNICAST )
continue ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
if ( next_fi - > fib_priority > res - > fi - > fib_priority )
break ;
if ( ! next_fi - > fib_nh [ 0 ] . nh_gw | |
next_fi - > fib_nh [ 0 ] . nh_scope ! = RT_SCOPE_LINK )
continue ;
fa - > fa_state | = FA_S_ACCESSED ;
2005-08-10 07:24:39 +04:00
2005-06-21 23:43:18 +04:00
if ( fi = = NULL ) {
if ( next_fi ! = res - > fi )
break ;
} else if ( ! fib_detect_death ( fi , order , & last_resort ,
& last_idx , & trie_last_dflt ) ) {
if ( res - > fi )
fib_info_put ( res - > fi ) ;
res - > fi = fi ;
atomic_inc ( & fi - > fib_clntref ) ;
trie_last_dflt = order ;
goto out ;
}
fi = next_fi ;
order + + ;
}
if ( order < = 0 | | fi = = NULL ) {
trie_last_dflt = - 1 ;
goto out ;
}
if ( ! fib_detect_death ( fi , order , & last_resort , & last_idx , & trie_last_dflt ) ) {
if ( res - > fi )
fib_info_put ( res - > fi ) ;
res - > fi = fi ;
atomic_inc ( & fi - > fib_clntref ) ;
trie_last_dflt = order ;
goto out ;
}
if ( last_idx > = 0 ) {
if ( res - > fi )
fib_info_put ( res - > fi ) ;
res - > fi = last_resort ;
if ( last_resort )
atomic_inc ( & last_resort - > fib_clntref ) ;
}
trie_last_dflt = last_idx ;
out : ;
2005-08-26 00:01:29 +04:00
rcu_read_unlock ( ) ;
2005-06-21 23:43:18 +04:00
}
2005-07-20 01:01:51 +04:00
static int fn_trie_dump_fa ( t_key key , int plen , struct list_head * fah , struct fib_table * tb ,
2005-06-21 23:43:18 +04:00
struct sk_buff * skb , struct netlink_callback * cb )
{
int i , s_i ;
struct fib_alias * fa ;
2006-09-27 09:21:45 +04:00
__be32 xkey = htonl ( key ) ;
2005-06-21 23:43:18 +04:00
2006-08-11 10:10:46 +04:00
s_i = cb - > args [ 4 ] ;
2005-06-21 23:43:18 +04:00
i = 0 ;
2005-08-26 00:01:29 +04:00
/* rcu_read_lock is hold by caller */
list_for_each_entry_rcu ( fa , fah , fa_list ) {
2005-06-21 23:43:18 +04:00
if ( i < s_i ) {
i + + ;
continue ;
}
2005-09-21 11:15:39 +04:00
BUG_ON ( ! fa - > fa_info ) ;
2005-06-21 23:43:18 +04:00
if ( fib_dump_info ( skb , NETLINK_CB ( cb - > skb ) . pid ,
cb - > nlh - > nlmsg_seq ,
RTM_NEWROUTE ,
tb - > tb_id ,
fa - > fa_type ,
fa - > fa_scope ,
2006-08-18 05:15:17 +04:00
xkey ,
2005-06-21 23:43:18 +04:00
plen ,
fa - > fa_tos ,
2005-06-22 01:43:28 +04:00
fa - > fa_info , 0 ) < 0 ) {
2006-08-11 10:10:46 +04:00
cb - > args [ 4 ] = i ;
2005-06-21 23:43:18 +04:00
return - 1 ;
2005-08-10 07:24:39 +04:00
}
2005-06-21 23:43:18 +04:00
i + + ;
}
2006-08-11 10:10:46 +04:00
cb - > args [ 4 ] = i ;
2005-06-21 23:43:18 +04:00
return skb - > len ;
}
2005-07-20 01:01:51 +04:00
static int fn_trie_dump_plen ( struct trie * t , int plen , struct fib_table * tb , struct sk_buff * skb ,
2005-06-21 23:43:18 +04:00
struct netlink_callback * cb )
{
int h , s_h ;
struct list_head * fa_head ;
struct leaf * l = NULL ;
2006-08-11 10:10:46 +04:00
s_h = cb - > args [ 3 ] ;
2005-06-21 23:43:18 +04:00
2005-08-10 07:24:39 +04:00
for ( h = 0 ; ( l = nextleaf ( t , l ) ) ! = NULL ; h + + ) {
2005-06-21 23:43:18 +04:00
if ( h < s_h )
continue ;
if ( h > s_h )
2006-08-11 10:10:46 +04:00
memset ( & cb - > args [ 4 ] , 0 ,
sizeof ( cb - > args ) - 4 * sizeof ( cb - > args [ 0 ] ) ) ;
2005-06-21 23:43:18 +04:00
fa_head = get_fa_head ( l , plen ) ;
2005-08-10 07:24:39 +04:00
2005-07-20 01:01:51 +04:00
if ( ! fa_head )
2005-06-21 23:43:18 +04:00
continue ;
2005-07-20 01:01:51 +04:00
if ( list_empty ( fa_head ) )
2005-06-21 23:43:18 +04:00
continue ;
if ( fn_trie_dump_fa ( l - > key , plen , fa_head , tb , skb , cb ) < 0 ) {
2006-08-11 10:10:46 +04:00
cb - > args [ 3 ] = h ;
2005-06-21 23:43:18 +04:00
return - 1 ;
}
}
2006-08-11 10:10:46 +04:00
cb - > args [ 3 ] = h ;
2005-06-21 23:43:18 +04:00
return skb - > len ;
}
static int fn_trie_dump ( struct fib_table * tb , struct sk_buff * skb , struct netlink_callback * cb )
{
int m , s_m ;
struct trie * t = ( struct trie * ) tb - > tb_data ;
2006-08-11 10:10:46 +04:00
s_m = cb - > args [ 2 ] ;
2005-06-21 23:43:18 +04:00
2005-08-26 00:01:29 +04:00
rcu_read_lock ( ) ;
2005-08-10 07:24:39 +04:00
for ( m = 0 ; m < = 32 ; m + + ) {
2005-06-21 23:43:18 +04:00
if ( m < s_m )
continue ;
if ( m > s_m )
2006-08-11 10:10:46 +04:00
memset ( & cb - > args [ 3 ] , 0 ,
sizeof ( cb - > args ) - 3 * sizeof ( cb - > args [ 0 ] ) ) ;
2005-06-21 23:43:18 +04:00
if ( fn_trie_dump_plen ( t , 32 - m , tb , skb , cb ) < 0 ) {
2006-08-11 10:10:46 +04:00
cb - > args [ 2 ] = m ;
2005-06-21 23:43:18 +04:00
goto out ;
}
}
2005-08-26 00:01:29 +04:00
rcu_read_unlock ( ) ;
2006-08-11 10:10:46 +04:00
cb - > args [ 2 ] = m ;
2005-06-21 23:43:18 +04:00
return skb - > len ;
2005-08-10 07:24:39 +04:00
out :
2005-08-26 00:01:29 +04:00
rcu_read_unlock ( ) ;
2005-06-21 23:43:18 +04:00
return - 1 ;
}
/* Fix more generic FIB names for init later */
# ifdef CONFIG_IP_MULTIPLE_TABLES
2006-08-11 10:08:33 +04:00
struct fib_table * fib_hash_init ( u32 id )
2005-06-21 23:43:18 +04:00
# else
2006-08-11 10:08:33 +04:00
struct fib_table * __init fib_hash_init ( u32 id )
2005-06-21 23:43:18 +04:00
# endif
{
struct fib_table * tb ;
struct trie * t ;
if ( fn_alias_kmem = = NULL )
fn_alias_kmem = kmem_cache_create ( " ip_fib_alias " ,
sizeof ( struct fib_alias ) ,
0 , SLAB_HWCACHE_ALIGN ,
NULL , NULL ) ;
tb = kmalloc ( sizeof ( struct fib_table ) + sizeof ( struct trie ) ,
GFP_KERNEL ) ;
if ( tb = = NULL )
return NULL ;
tb - > tb_id = id ;
tb - > tb_lookup = fn_trie_lookup ;
tb - > tb_insert = fn_trie_insert ;
tb - > tb_delete = fn_trie_delete ;
tb - > tb_flush = fn_trie_flush ;
tb - > tb_select_default = fn_trie_select_default ;
tb - > tb_dump = fn_trie_dump ;
memset ( tb - > tb_data , 0 , sizeof ( struct trie ) ) ;
t = ( struct trie * ) tb - > tb_data ;
trie_init ( t ) ;
2005-07-20 01:01:51 +04:00
if ( id = = RT_TABLE_LOCAL )
2005-08-10 07:24:39 +04:00
trie_local = t ;
2005-07-20 01:01:51 +04:00
else if ( id = = RT_TABLE_MAIN )
2005-08-10 07:24:39 +04:00
trie_main = t ;
2005-06-21 23:43:18 +04:00
if ( id = = RT_TABLE_LOCAL )
2005-09-21 11:15:39 +04:00
printk ( KERN_INFO " IPv4 FIB: Using LC-trie version %s \n " , VERSION ) ;
2005-06-21 23:43:18 +04:00
return tb ;
}
2005-09-10 00:35:42 +04:00
# ifdef CONFIG_PROC_FS
/* Depth first Trie walk iterator */
struct fib_trie_iter {
struct tnode * tnode ;
struct trie * trie ;
unsigned index ;
unsigned depth ;
} ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static struct node * fib_trie_get_next ( struct fib_trie_iter * iter )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
struct tnode * tn = iter - > tnode ;
unsigned cindex = iter - > index ;
struct tnode * p ;
2005-06-21 23:43:18 +04:00
2007-01-25 01:42:04 +03:00
/* A single entry routing table */
if ( ! tn )
return NULL ;
2005-09-10 00:35:42 +04:00
pr_debug ( " get_next iter={node=%p index=%d depth=%d} \n " ,
iter - > tnode , iter - > index , iter - > depth ) ;
rescan :
while ( cindex < ( 1 < < tn - > bits ) ) {
struct node * n = tnode_get_child ( tn , cindex ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( n ) {
if ( IS_LEAF ( n ) ) {
iter - > tnode = tn ;
iter - > index = cindex + 1 ;
} else {
/* push down one level */
iter - > tnode = ( struct tnode * ) n ;
iter - > index = 0 ;
+ + iter - > depth ;
}
return n ;
}
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
+ + cindex ;
}
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
/* Current node exhausted, pop back up */
p = NODE_PARENT ( tn ) ;
if ( p ) {
cindex = tkey_extract_bits ( tn - > key , p - > pos , p - > bits ) + 1 ;
tn = p ;
- - iter - > depth ;
goto rescan ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
/* got root? */
return NULL ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
static struct node * fib_trie_get_first ( struct fib_trie_iter * iter ,
struct trie * t )
2005-06-21 23:43:18 +04:00
{
2006-03-21 08:34:12 +03:00
struct node * n ;
if ( ! t )
return NULL ;
n = rcu_dereference ( t - > trie ) ;
if ( ! iter )
return NULL ;
2005-06-21 23:43:18 +04:00
2007-01-25 01:42:04 +03:00
if ( n ) {
if ( IS_TNODE ( n ) ) {
iter - > tnode = ( struct tnode * ) n ;
iter - > trie = t ;
iter - > index = 0 ;
iter - > depth = 1 ;
} else {
iter - > tnode = NULL ;
iter - > trie = t ;
iter - > index = 0 ;
iter - > depth = 0 ;
}
2005-09-10 00:35:42 +04:00
return n ;
2005-08-10 07:24:39 +04:00
}
2005-09-10 00:35:42 +04:00
return NULL ;
}
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
static void trie_collect_stats ( struct trie * t , struct trie_stat * s )
{
struct node * n ;
struct fib_trie_iter iter ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
memset ( s , 0 , sizeof ( * s ) ) ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
rcu_read_lock ( ) ;
for ( n = fib_trie_get_first ( & iter , t ) ; n ;
n = fib_trie_get_next ( & iter ) ) {
if ( IS_LEAF ( n ) ) {
s - > leaves + + ;
s - > totdepth + = iter . depth ;
if ( iter . depth > s - > maxdepth )
s - > maxdepth = iter . depth ;
} else {
const struct tnode * tn = ( const struct tnode * ) n ;
int i ;
s - > tnodes + + ;
2006-03-21 08:35:01 +03:00
if ( tn - > bits < MAX_STAT_DEPTH )
s - > nodesizes [ tn - > bits ] + + ;
2005-09-10 00:35:42 +04:00
for ( i = 0 ; i < ( 1 < < tn - > bits ) ; i + + )
if ( ! tn - > child [ i ] )
s - > nullpointers + + ;
2005-06-21 23:43:18 +04:00
}
}
2005-08-26 00:01:29 +04:00
rcu_read_unlock ( ) ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
/*
* This outputs / proc / net / fib_triestats
*/
static void trie_show_stats ( struct seq_file * seq , struct trie_stat * stat )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
unsigned i , max , pointers , bytes , avdepth ;
2005-07-20 01:01:51 +04:00
2005-09-10 00:35:42 +04:00
if ( stat - > leaves )
avdepth = stat - > totdepth * 100 / stat - > leaves ;
else
avdepth = 0 ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
seq_printf ( seq , " \t Aver depth: %d.%02d \n " , avdepth / 100 , avdepth % 100 ) ;
seq_printf ( seq , " \t Max depth: %u \n " , stat - > maxdepth ) ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
seq_printf ( seq , " \t Leaves: %u \n " , stat - > leaves ) ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
bytes = sizeof ( struct leaf ) * stat - > leaves ;
seq_printf ( seq , " \t Internal nodes: %d \n \t " , stat - > tnodes ) ;
bytes + = sizeof ( struct tnode ) * stat - > tnodes ;
2005-06-21 23:43:18 +04:00
2006-03-21 08:35:01 +03:00
max = MAX_STAT_DEPTH ;
while ( max > 0 & & stat - > nodesizes [ max - 1 ] = = 0 )
2005-09-10 00:35:42 +04:00
max - - ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
pointers = 0 ;
for ( i = 1 ; i < = max ; i + + )
if ( stat - > nodesizes [ i ] ! = 0 ) {
seq_printf ( seq , " %d: %d " , i , stat - > nodesizes [ i ] ) ;
pointers + = ( 1 < < i ) * stat - > nodesizes [ i ] ;
}
seq_putc ( seq , ' \n ' ) ;
seq_printf ( seq , " \t Pointers: %d \n " , pointers ) ;
2005-08-26 00:01:29 +04:00
2005-09-10 00:35:42 +04:00
bytes + = sizeof ( struct node * ) * pointers ;
seq_printf ( seq , " Null ptrs: %d \n " , stat - > nullpointers ) ;
seq_printf ( seq , " Total size: %d kB \n " , ( bytes + 1023 ) / 1024 ) ;
2005-08-26 00:01:29 +04:00
2005-09-10 00:35:42 +04:00
# ifdef CONFIG_IP_FIB_TRIE_STATS
seq_printf ( seq , " Counters: \n --------- \n " ) ;
seq_printf ( seq , " gets = %d \n " , t - > stats . gets ) ;
seq_printf ( seq , " backtracks = %d \n " , t - > stats . backtrack ) ;
seq_printf ( seq , " semantic match passed = %d \n " , t - > stats . semantic_match_passed ) ;
seq_printf ( seq , " semantic match miss = %d \n " , t - > stats . semantic_match_miss ) ;
seq_printf ( seq , " null node hit= %d \n " , t - > stats . null_node_hit ) ;
seq_printf ( seq , " skipped node resize = %d \n " , t - > stats . resize_node_skipped ) ;
# ifdef CLEAR_STATS
memset ( & ( t - > stats ) , 0 , sizeof ( t - > stats ) ) ;
# endif
# endif /* CONFIG_IP_FIB_TRIE_STATS */
}
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static int fib_triestat_seq_show ( struct seq_file * seq , void * v )
{
struct trie_stat * stat ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
stat = kmalloc ( sizeof ( * stat ) , GFP_KERNEL ) ;
if ( ! stat )
return - ENOMEM ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
seq_printf ( seq , " Basic info: size of leaf: %Zd bytes, size of tnode: %Zd bytes. \n " ,
sizeof ( struct leaf ) , sizeof ( struct tnode ) ) ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
if ( trie_local ) {
seq_printf ( seq , " Local: \n " ) ;
trie_collect_stats ( trie_local , stat ) ;
trie_show_stats ( seq , stat ) ;
}
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
if ( trie_main ) {
seq_printf ( seq , " Main: \n " ) ;
trie_collect_stats ( trie_main , stat ) ;
trie_show_stats ( seq , stat ) ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
kfree ( stat ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
return 0 ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
static int fib_triestat_seq_open ( struct inode * inode , struct file * file )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
return single_open ( file , fib_triestat_seq_show , NULL ) ;
2005-06-21 23:43:18 +04:00
}
2007-02-12 11:55:35 +03:00
static const struct file_operations fib_triestat_fops = {
2005-09-10 00:35:42 +04:00
. owner = THIS_MODULE ,
. open = fib_triestat_seq_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct node * fib_trie_get_idx ( struct fib_trie_iter * iter ,
loff_t pos )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
loff_t idx = 0 ;
struct node * n ;
for ( n = fib_trie_get_first ( iter , trie_local ) ;
n ; + + idx , n = fib_trie_get_next ( iter ) ) {
if ( pos = = idx )
return n ;
}
for ( n = fib_trie_get_first ( iter , trie_main ) ;
n ; + + idx , n = fib_trie_get_next ( iter ) ) {
if ( pos = = idx )
return n ;
}
2005-06-21 23:43:18 +04:00
return NULL ;
}
2005-09-10 00:35:42 +04:00
static void * fib_trie_seq_start ( struct seq_file * seq , loff_t * pos )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
rcu_read_lock ( ) ;
if ( * pos = = 0 )
2005-08-10 07:24:39 +04:00
return SEQ_START_TOKEN ;
2005-09-10 00:35:42 +04:00
return fib_trie_get_idx ( seq - > private , * pos - 1 ) ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
static void * fib_trie_seq_next ( struct seq_file * seq , void * v , loff_t * pos )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
struct fib_trie_iter * iter = seq - > private ;
void * l = v ;
2005-06-21 23:43:18 +04:00
+ + * pos ;
2005-08-10 07:24:39 +04:00
if ( v = = SEQ_START_TOKEN )
2005-09-10 00:35:42 +04:00
return fib_trie_get_idx ( iter , 0 ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
v = fib_trie_get_next ( iter ) ;
BUG_ON ( v = = l ) ;
if ( v )
return v ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
/* continue scan in next trie */
if ( iter - > trie = = trie_local )
return fib_trie_get_first ( iter , trie_main ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
return NULL ;
}
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static void fib_trie_seq_stop ( struct seq_file * seq , void * v )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
rcu_read_unlock ( ) ;
}
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
static void seq_indent ( struct seq_file * seq , int n )
{
while ( n - - > 0 ) seq_puts ( seq , " " ) ;
}
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static inline const char * rtn_scope ( enum rt_scope_t s )
{
static char buf [ 32 ] ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
switch ( s ) {
case RT_SCOPE_UNIVERSE : return " universe " ;
case RT_SCOPE_SITE : return " site " ;
case RT_SCOPE_LINK : return " link " ;
case RT_SCOPE_HOST : return " host " ;
case RT_SCOPE_NOWHERE : return " nowhere " ;
default :
snprintf ( buf , sizeof ( buf ) , " scope=%d " , s ) ;
return buf ;
}
}
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static const char * rtn_type_names [ __RTN_MAX ] = {
[ RTN_UNSPEC ] = " UNSPEC " ,
[ RTN_UNICAST ] = " UNICAST " ,
[ RTN_LOCAL ] = " LOCAL " ,
[ RTN_BROADCAST ] = " BROADCAST " ,
[ RTN_ANYCAST ] = " ANYCAST " ,
[ RTN_MULTICAST ] = " MULTICAST " ,
[ RTN_BLACKHOLE ] = " BLACKHOLE " ,
[ RTN_UNREACHABLE ] = " UNREACHABLE " ,
[ RTN_PROHIBIT ] = " PROHIBIT " ,
[ RTN_THROW ] = " THROW " ,
[ RTN_NAT ] = " NAT " ,
[ RTN_XRESOLVE ] = " XRESOLVE " ,
} ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
static inline const char * rtn_type ( unsigned t )
{
static char buf [ 32 ] ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( t < __RTN_MAX & & rtn_type_names [ t ] )
return rtn_type_names [ t ] ;
snprintf ( buf , sizeof ( buf ) , " type %d " , t ) ;
return buf ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
/* Pretty print the trie */
static int fib_trie_seq_show ( struct seq_file * seq , void * v )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
const struct fib_trie_iter * iter = seq - > private ;
struct node * n = v ;
2005-07-20 01:01:51 +04:00
2005-09-10 00:35:42 +04:00
if ( v = = SEQ_START_TOKEN )
return 0 ;
2005-06-21 23:43:18 +04:00
2007-01-27 06:06:01 +03:00
if ( ! NODE_PARENT ( n ) ) {
if ( iter - > trie = = trie_local )
seq_puts ( seq , " <local>: \n " ) ;
else
seq_puts ( seq , " <main>: \n " ) ;
}
2005-09-10 00:35:42 +04:00
if ( IS_TNODE ( n ) ) {
struct tnode * tn = ( struct tnode * ) n ;
2006-09-27 09:21:45 +04:00
__be32 prf = htonl ( MASK_PFX ( tn - > key , tn - > pos ) ) ;
2005-08-10 07:24:39 +04:00
2005-09-20 02:29:52 +04:00
seq_indent ( seq , iter - > depth - 1 ) ;
seq_printf ( seq , " +-- %d.%d.%d.%d/%d %d %d %d \n " ,
2007-02-09 17:24:47 +03:00
NIPQUAD ( prf ) , tn - > pos , tn - > bits , tn - > full_children ,
2005-09-20 02:29:52 +04:00
tn - > empty_children ) ;
2007-02-09 17:24:47 +03:00
2005-09-10 00:35:42 +04:00
} else {
struct leaf * l = ( struct leaf * ) n ;
int i ;
2006-09-27 09:21:45 +04:00
__be32 val = htonl ( l - > key ) ;
2005-09-10 00:35:42 +04:00
seq_indent ( seq , iter - > depth ) ;
seq_printf ( seq , " |-- %d.%d.%d.%d \n " , NIPQUAD ( val ) ) ;
for ( i = 32 ; i > = 0 ; i - - ) {
2005-09-20 02:31:18 +04:00
struct leaf_info * li = find_leaf_info ( l , i ) ;
2005-09-10 00:35:42 +04:00
if ( li ) {
struct fib_alias * fa ;
list_for_each_entry_rcu ( fa , & li - > falh , fa_list ) {
seq_indent ( seq , iter - > depth + 1 ) ;
seq_printf ( seq , " /%d %s %s " , i ,
rtn_scope ( fa - > fa_scope ) ,
rtn_type ( fa - > fa_type ) ) ;
if ( fa - > fa_tos )
seq_printf ( seq , " tos =%d \n " ,
fa - > fa_tos ) ;
seq_putc ( seq , ' \n ' ) ;
}
}
}
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
2005-06-21 23:43:18 +04:00
return 0 ;
}
2005-09-10 00:35:42 +04:00
static struct seq_operations fib_trie_seq_ops = {
. start = fib_trie_seq_start ,
. next = fib_trie_seq_next ,
. stop = fib_trie_seq_stop ,
. show = fib_trie_seq_show ,
2005-06-21 23:43:18 +04:00
} ;
2005-09-10 00:35:42 +04:00
static int fib_trie_seq_open ( struct inode * inode , struct file * file )
2005-06-21 23:43:18 +04:00
{
struct seq_file * seq ;
int rc = - ENOMEM ;
2005-09-10 00:35:42 +04:00
struct fib_trie_iter * s = kmalloc ( sizeof ( * s ) , GFP_KERNEL ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( ! s )
goto out ;
rc = seq_open ( file , & fib_trie_seq_ops ) ;
2005-06-21 23:43:18 +04:00
if ( rc )
goto out_kfree ;
2005-09-10 00:35:42 +04:00
seq = file - > private_data ;
seq - > private = s ;
memset ( s , 0 , sizeof ( * s ) ) ;
2005-06-21 23:43:18 +04:00
out :
return rc ;
out_kfree :
2005-09-10 00:35:42 +04:00
kfree ( s ) ;
2005-06-21 23:43:18 +04:00
goto out ;
}
2007-02-12 11:55:35 +03:00
static const struct file_operations fib_trie_fops = {
2005-09-10 00:35:42 +04:00
. owner = THIS_MODULE ,
. open = fib_trie_seq_open ,
. read = seq_read ,
. llseek = seq_lseek ,
2005-07-20 01:01:51 +04:00
. release = seq_release_private ,
2005-06-21 23:43:18 +04:00
} ;
2006-09-27 09:21:45 +04:00
static unsigned fib_flag_trans ( int type , __be32 mask , const struct fib_info * fi )
2005-06-21 23:43:18 +04:00
{
2005-09-10 00:35:42 +04:00
static unsigned type2flags [ RTN_MAX + 1 ] = {
[ 7 ] = RTF_REJECT , [ 8 ] = RTF_REJECT ,
} ;
unsigned flags = type2flags [ type ] ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( fi & & fi - > fib_nh - > nh_gw )
flags | = RTF_GATEWAY ;
2006-09-27 09:21:45 +04:00
if ( mask = = htonl ( 0xFFFFFFFF ) )
2005-09-10 00:35:42 +04:00
flags | = RTF_HOST ;
flags | = RTF_UP ;
return flags ;
2005-06-21 23:43:18 +04:00
}
2005-09-10 00:35:42 +04:00
/*
* This outputs / proc / net / route .
* The format of the file is not supposed to be changed
* and needs to be same as fib_hash output to avoid breaking
* legacy utilities
*/
static int fib_route_seq_show ( struct seq_file * seq , void * v )
2005-06-21 23:43:18 +04:00
{
2005-11-21 08:09:00 +03:00
const struct fib_trie_iter * iter = seq - > private ;
2005-09-10 00:35:42 +04:00
struct leaf * l = v ;
int i ;
char bf [ 128 ] ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( v = = SEQ_START_TOKEN ) {
seq_printf ( seq , " %-127s \n " , " Iface \t Destination \t Gateway "
" \t Flags \t RefCnt \t Use \t Metric \t Mask \t \t MTU "
" \t Window \t IRTT " ) ;
return 0 ;
}
2005-06-21 23:43:18 +04:00
2005-11-21 08:09:00 +03:00
if ( iter - > trie = = trie_local )
return 0 ;
2005-09-10 00:35:42 +04:00
if ( IS_TNODE ( l ) )
return 0 ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
for ( i = 32 ; i > = 0 ; i - - ) {
2005-09-20 02:31:18 +04:00
struct leaf_info * li = find_leaf_info ( l , i ) ;
2005-09-10 00:35:42 +04:00
struct fib_alias * fa ;
2006-09-27 09:21:45 +04:00
__be32 mask , prefix ;
2005-08-10 07:24:39 +04:00
2005-09-10 00:35:42 +04:00
if ( ! li )
continue ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
mask = inet_make_mask ( li - > plen ) ;
prefix = htonl ( l - > key ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
list_for_each_entry_rcu ( fa , & li - > falh , fa_list ) {
2005-10-15 03:42:39 +04:00
const struct fib_info * fi = fa - > fa_info ;
2005-09-10 00:35:42 +04:00
unsigned flags = fib_flag_trans ( fa - > fa_type , mask , fi ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( fa - > fa_type = = RTN_BROADCAST
| | fa - > fa_type = = RTN_MULTICAST )
continue ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( fi )
snprintf ( bf , sizeof ( bf ) ,
" %s \t %08X \t %08X \t %04X \t %d \t %u \t %d \t %08X \t %d \t %u \t %u " ,
fi - > fib_dev ? fi - > fib_dev - > name : " * " ,
prefix ,
fi - > fib_nh - > nh_gw , flags , 0 , 0 ,
fi - > fib_priority ,
mask ,
( fi - > fib_advmss ? fi - > fib_advmss + 40 : 0 ) ,
fi - > fib_window ,
fi - > fib_rtt > > 3 ) ;
else
snprintf ( bf , sizeof ( bf ) ,
" * \t %08X \t %08X \t %04X \t %d \t %u \t %d \t %08X \t %d \t %u \t %u " ,
prefix , 0 , flags , 0 , 0 , 0 ,
mask , 0 , 0 , 0 ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
seq_printf ( seq , " %-127s \n " , bf ) ;
}
2005-06-21 23:43:18 +04:00
}
return 0 ;
}
2005-09-10 00:35:42 +04:00
static struct seq_operations fib_route_seq_ops = {
. start = fib_trie_seq_start ,
. next = fib_trie_seq_next ,
. stop = fib_trie_seq_stop ,
. show = fib_route_seq_show ,
2005-06-21 23:43:18 +04:00
} ;
2005-09-10 00:35:42 +04:00
static int fib_route_seq_open ( struct inode * inode , struct file * file )
2005-06-21 23:43:18 +04:00
{
struct seq_file * seq ;
int rc = - ENOMEM ;
2005-09-10 00:35:42 +04:00
struct fib_trie_iter * s = kmalloc ( sizeof ( * s ) , GFP_KERNEL ) ;
2005-06-21 23:43:18 +04:00
2005-09-10 00:35:42 +04:00
if ( ! s )
goto out ;
rc = seq_open ( file , & fib_route_seq_ops ) ;
2005-06-21 23:43:18 +04:00
if ( rc )
goto out_kfree ;
2005-09-10 00:35:42 +04:00
seq = file - > private_data ;
seq - > private = s ;
memset ( s , 0 , sizeof ( * s ) ) ;
2005-06-21 23:43:18 +04:00
out :
return rc ;
out_kfree :
2005-09-10 00:35:42 +04:00
kfree ( s ) ;
2005-06-21 23:43:18 +04:00
goto out ;
}
2007-02-12 11:55:35 +03:00
static const struct file_operations fib_route_fops = {
2005-09-10 00:35:42 +04:00
. owner = THIS_MODULE ,
. open = fib_route_seq_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = seq_release_private ,
2005-06-21 23:43:18 +04:00
} ;
int __init fib_proc_init ( void )
{
2005-09-10 00:35:42 +04:00
if ( ! proc_net_fops_create ( " fib_trie " , S_IRUGO , & fib_trie_fops ) )
goto out1 ;
if ( ! proc_net_fops_create ( " fib_triestat " , S_IRUGO , & fib_triestat_fops ) )
goto out2 ;
if ( ! proc_net_fops_create ( " route " , S_IRUGO , & fib_route_fops ) )
goto out3 ;
2005-06-21 23:43:18 +04:00
return 0 ;
2005-09-10 00:35:42 +04:00
out3 :
proc_net_remove ( " fib_triestat " ) ;
out2 :
proc_net_remove ( " fib_trie " ) ;
out1 :
return - ENOMEM ;
2005-06-21 23:43:18 +04:00
}
void __init fib_proc_exit ( void )
{
proc_net_remove ( " fib_trie " ) ;
2005-09-10 00:35:42 +04:00
proc_net_remove ( " fib_triestat " ) ;
proc_net_remove ( " route " ) ;
2005-06-21 23:43:18 +04:00
}
# endif /* CONFIG_PROC_FS */