2006-08-04 03:38:38 -07:00
/*
* net / core / fib_rules . c Generic Routing Rules
*
* 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 , version 2.
*
* Authors : Thomas Graf < tgraf @ suug . ch >
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/list.h>
2007-09-12 13:02:17 +02:00
# include <net/net_namespace.h>
2007-09-17 11:56:21 -07:00
# include <net/sock.h>
2006-08-04 03:38:38 -07:00
# include <net/fib_rules.h>
2007-11-10 22:12:03 -08:00
int fib_default_rule_add ( struct fib_rules_ops * ops ,
u32 pref , u32 table , u32 flags )
{
struct fib_rule * r ;
r = kzalloc ( ops - > rule_size , GFP_KERNEL ) ;
if ( r = = NULL )
return - ENOMEM ;
atomic_set ( & r - > refcnt , 1 ) ;
r - > action = FR_ACT_TO_TBL ;
r - > pref = pref ;
r - > table = table ;
r - > flags = flags ;
2008-04-16 02:01:56 -07:00
r - > fr_net = hold_net ( ops - > fro_net ) ;
2007-11-10 22:12:03 -08:00
/* The lock is not required here, the list in unreacheable
* at the moment this function is called */
list_add_tail ( & r - > list , & ops - > rules_list ) ;
return 0 ;
}
EXPORT_SYMBOL ( fib_default_rule_add ) ;
2008-01-20 16:46:41 -08:00
static void notify_rule_change ( int event , struct fib_rule * rule ,
2006-08-15 00:32:48 -07:00
struct fib_rules_ops * ops , struct nlmsghdr * nlh ,
u32 pid ) ;
2006-08-04 03:38:38 -07:00
2008-01-10 03:20:28 -08:00
static struct fib_rules_ops * lookup_rules_ops ( struct net * net , int family )
2006-08-04 03:38:38 -07:00
{
struct fib_rules_ops * ops ;
rcu_read_lock ( ) ;
2008-01-10 03:20:28 -08:00
list_for_each_entry_rcu ( ops , & net - > rules_ops , list ) {
2006-08-04 03:38:38 -07:00
if ( ops - > family = = family ) {
if ( ! try_module_get ( ops - > owner ) )
ops = NULL ;
rcu_read_unlock ( ) ;
return ops ;
}
}
rcu_read_unlock ( ) ;
return NULL ;
}
static void rules_ops_put ( struct fib_rules_ops * ops )
{
if ( ops )
module_put ( ops - > owner ) ;
}
2007-03-27 13:56:52 -07:00
static void flush_route_cache ( struct fib_rules_ops * ops )
{
if ( ops - > flush_cache )
ops - > flush_cache ( ) ;
}
2008-01-20 16:46:41 -08:00
int fib_rules_register ( struct fib_rules_ops * ops )
2006-08-04 03:38:38 -07:00
{
int err = - EEXIST ;
struct fib_rules_ops * o ;
2008-01-20 16:46:41 -08:00
struct net * net ;
net = ops - > fro_net ;
2006-08-04 03:38:38 -07:00
if ( ops - > rule_size < sizeof ( struct fib_rule ) )
return - EINVAL ;
if ( ops - > match = = NULL | | ops - > configure = = NULL | |
ops - > compare = = NULL | | ops - > fill = = NULL | |
ops - > action = = NULL )
return - EINVAL ;
2008-01-10 03:20:28 -08:00
spin_lock ( & net - > rules_mod_lock ) ;
list_for_each_entry ( o , & net - > rules_ops , list )
2006-08-04 03:38:38 -07:00
if ( ops - > family = = o - > family )
goto errout ;
2008-01-10 03:20:28 -08:00
hold_net ( net ) ;
list_add_tail_rcu ( & ops - > list , & net - > rules_ops ) ;
2006-08-04 03:38:38 -07:00
err = 0 ;
errout :
2008-01-10 03:20:28 -08:00
spin_unlock ( & net - > rules_mod_lock ) ;
2006-08-04 03:38:38 -07:00
return err ;
}
EXPORT_SYMBOL_GPL ( fib_rules_register ) ;
2007-12-07 00:42:52 -08:00
void fib_rules_cleanup_ops ( struct fib_rules_ops * ops )
2006-08-04 03:38:38 -07:00
{
struct fib_rule * rule , * tmp ;
2007-09-16 15:44:27 -07:00
list_for_each_entry_safe ( rule , tmp , & ops - > rules_list , list ) {
2006-08-04 03:38:38 -07:00
list_del_rcu ( & rule - > list ) ;
fib_rule_put ( rule ) ;
}
}
2007-12-07 00:42:52 -08:00
EXPORT_SYMBOL_GPL ( fib_rules_cleanup_ops ) ;
2006-08-04 03:38:38 -07:00
2008-01-20 16:46:41 -08:00
void fib_rules_unregister ( struct fib_rules_ops * ops )
2006-08-04 03:38:38 -07:00
{
2008-01-20 16:46:41 -08:00
struct net * net = ops - > fro_net ;
2006-08-04 03:38:38 -07:00
2008-01-10 03:20:28 -08:00
spin_lock ( & net - > rules_mod_lock ) ;
2008-01-14 22:59:30 -08:00
list_del_rcu ( & ops - > list ) ;
fib_rules_cleanup_ops ( ops ) ;
2008-01-10 03:20:28 -08:00
spin_unlock ( & net - > rules_mod_lock ) ;
2006-08-04 03:38:38 -07:00
synchronize_rcu ( ) ;
2008-01-14 22:59:30 -08:00
release_net ( net ) ;
2006-08-04 03:38:38 -07:00
}
EXPORT_SYMBOL_GPL ( fib_rules_unregister ) ;
2006-11-09 15:23:20 -08:00
static int fib_rule_match ( struct fib_rule * rule , struct fib_rules_ops * ops ,
struct flowi * fl , int flags )
{
int ret = 0 ;
if ( rule - > ifindex & & ( rule - > ifindex ! = fl - > iif ) )
goto out ;
if ( ( rule - > mark ^ fl - > mark ) & rule - > mark_mask )
goto out ;
ret = ops - > match ( rule , fl , flags ) ;
out :
return ( rule - > flags & FIB_RULE_INVERT ) ? ! ret : ret ;
}
2006-08-04 03:38:38 -07:00
int fib_rules_lookup ( struct fib_rules_ops * ops , struct flowi * fl ,
int flags , struct fib_lookup_arg * arg )
{
struct fib_rule * rule ;
int err ;
rcu_read_lock ( ) ;
2007-09-16 15:44:27 -07:00
list_for_each_entry_rcu ( rule , & ops - > rules_list , list ) {
2007-03-26 17:14:15 -07:00
jumped :
2006-11-09 15:23:20 -08:00
if ( ! fib_rule_match ( rule , ops , fl , flags ) )
2006-08-04 03:38:38 -07:00
continue ;
2007-03-26 17:14:15 -07:00
if ( rule - > action = = FR_ACT_GOTO ) {
struct fib_rule * target ;
target = rcu_dereference ( rule - > ctarget ) ;
if ( target = = NULL ) {
continue ;
} else {
rule = target ;
goto jumped ;
}
2007-03-26 17:38:53 -07:00
} else if ( rule - > action = = FR_ACT_NOP )
continue ;
else
2007-03-26 17:14:15 -07:00
err = ops - > action ( rule , fl , flags , arg ) ;
2006-08-04 03:38:38 -07:00
if ( err ! = - EAGAIN ) {
fib_rule_get ( rule ) ;
arg - > rule = rule ;
goto out ;
}
}
2007-03-30 13:34:27 -07:00
err = - ESRCH ;
2006-08-04 03:38:38 -07:00
out :
rcu_read_unlock ( ) ;
return err ;
}
EXPORT_SYMBOL_GPL ( fib_rules_lookup ) ;
2007-03-24 12:46:02 -07:00
static int validate_rulemsg ( struct fib_rule_hdr * frh , struct nlattr * * tb ,
struct fib_rules_ops * ops )
{
int err = - EINVAL ;
if ( frh - > src_len )
if ( tb [ FRA_SRC ] = = NULL | |
frh - > src_len > ( ops - > addr_size * 8 ) | |
nla_len ( tb [ FRA_SRC ] ) ! = ops - > addr_size )
goto errout ;
if ( frh - > dst_len )
if ( tb [ FRA_DST ] = = NULL | |
frh - > dst_len > ( ops - > addr_size * 8 ) | |
nla_len ( tb [ FRA_DST ] ) ! = ops - > addr_size )
goto errout ;
err = 0 ;
errout :
return err ;
}
2007-03-25 23:20:05 -07:00
static int fib_nl_newrule ( struct sk_buff * skb , struct nlmsghdr * nlh , void * arg )
2006-08-04 03:38:38 -07:00
{
2008-03-26 02:26:21 +09:00
struct net * net = sock_net ( skb - > sk ) ;
2006-08-04 03:38:38 -07:00
struct fib_rule_hdr * frh = nlmsg_data ( nlh ) ;
struct fib_rules_ops * ops = NULL ;
struct fib_rule * rule , * r , * last = NULL ;
struct nlattr * tb [ FRA_MAX + 1 ] ;
2007-03-26 17:14:15 -07:00
int err = - EINVAL , unresolved = 0 ;
2006-08-04 03:38:38 -07:00
if ( nlh - > nlmsg_len < nlmsg_msg_size ( sizeof ( * frh ) ) )
goto errout ;
2008-01-10 03:20:28 -08:00
ops = lookup_rules_ops ( net , frh - > family ) ;
2006-08-04 03:38:38 -07:00
if ( ops = = NULL ) {
err = EAFNOSUPPORT ;
goto errout ;
}
err = nlmsg_parse ( nlh , sizeof ( * frh ) , tb , FRA_MAX , ops - > policy ) ;
if ( err < 0 )
goto errout ;
2007-03-24 12:46:02 -07:00
err = validate_rulemsg ( frh , tb , ops ) ;
if ( err < 0 )
goto errout ;
2006-08-04 03:38:38 -07:00
rule = kzalloc ( ops - > rule_size , GFP_KERNEL ) ;
if ( rule = = NULL ) {
err = - ENOMEM ;
goto errout ;
}
2008-04-16 02:01:56 -07:00
rule - > fr_net = hold_net ( net ) ;
2006-08-04 03:38:38 -07:00
if ( tb [ FRA_PRIORITY ] )
rule - > pref = nla_get_u32 ( tb [ FRA_PRIORITY ] ) ;
if ( tb [ FRA_IFNAME ] ) {
struct net_device * dev ;
rule - > ifindex = - 1 ;
2006-08-26 20:13:18 -07:00
nla_strlcpy ( rule - > ifname , tb [ FRA_IFNAME ] , IFNAMSIZ ) ;
2007-09-17 11:56:21 -07:00
dev = __dev_get_by_name ( net , rule - > ifname ) ;
2006-08-04 03:38:38 -07:00
if ( dev )
rule - > ifindex = dev - > ifindex ;
}
2006-11-09 15:22:18 -08:00
if ( tb [ FRA_FWMARK ] ) {
rule - > mark = nla_get_u32 ( tb [ FRA_FWMARK ] ) ;
if ( rule - > mark )
/* compatibility: if the mark value is non-zero all bits
* are compared unless a mask is explicitly specified .
*/
rule - > mark_mask = 0xFFFFFFFF ;
}
if ( tb [ FRA_FWMASK ] )
rule - > mark_mask = nla_get_u32 ( tb [ FRA_FWMASK ] ) ;
2006-08-04 03:38:38 -07:00
rule - > action = frh - > action ;
rule - > flags = frh - > flags ;
2006-08-10 23:09:48 -07:00
rule - > table = frh_get_table ( frh , tb ) ;
2006-08-04 03:38:38 -07:00
if ( ! rule - > pref & & ops - > default_pref )
2008-01-10 03:18:25 -08:00
rule - > pref = ops - > default_pref ( ops ) ;
2006-08-04 03:38:38 -07:00
2007-03-26 17:14:15 -07:00
err = - EINVAL ;
if ( tb [ FRA_GOTO ] ) {
if ( rule - > action ! = FR_ACT_GOTO )
goto errout_free ;
rule - > target = nla_get_u32 ( tb [ FRA_GOTO ] ) ;
/* Backward jumps are prohibited to avoid endless loops */
if ( rule - > target < = rule - > pref )
goto errout_free ;
2007-09-16 15:44:27 -07:00
list_for_each_entry ( r , & ops - > rules_list , list ) {
2007-03-26 17:14:15 -07:00
if ( r - > pref = = rule - > target ) {
rule - > ctarget = r ;
break ;
}
}
if ( rule - > ctarget = = NULL )
unresolved = 1 ;
} else if ( rule - > action = = FR_ACT_GOTO )
goto errout_free ;
2006-08-04 03:38:38 -07:00
err = ops - > configure ( rule , skb , nlh , frh , tb ) ;
if ( err < 0 )
goto errout_free ;
2007-09-16 15:44:27 -07:00
list_for_each_entry ( r , & ops - > rules_list , list ) {
2006-08-04 03:38:38 -07:00
if ( r - > pref > rule - > pref )
break ;
last = r ;
}
fib_rule_get ( rule ) ;
2007-03-26 17:14:15 -07:00
if ( ops - > unresolved_rules ) {
/*
* There are unresolved goto rules in the list , check if
* any of them are pointing to this new rule .
*/
2007-09-16 15:44:27 -07:00
list_for_each_entry ( r , & ops - > rules_list , list ) {
2007-03-26 17:14:15 -07:00
if ( r - > action = = FR_ACT_GOTO & &
r - > target = = rule - > pref ) {
BUG_ON ( r - > ctarget ! = NULL ) ;
rcu_assign_pointer ( r - > ctarget , rule ) ;
if ( - - ops - > unresolved_rules = = 0 )
break ;
}
}
}
if ( rule - > action = = FR_ACT_GOTO )
ops - > nr_goto_rules + + ;
if ( unresolved )
ops - > unresolved_rules + + ;
2006-08-04 03:38:38 -07:00
if ( last )
list_add_rcu ( & rule - > list , & last - > list ) ;
else
2007-09-16 15:44:27 -07:00
list_add_rcu ( & rule - > list , & ops - > rules_list ) ;
2006-08-04 03:38:38 -07:00
2008-01-20 16:46:41 -08:00
notify_rule_change ( RTM_NEWRULE , rule , ops , nlh , NETLINK_CB ( skb ) . pid ) ;
2007-03-27 13:56:52 -07:00
flush_route_cache ( ops ) ;
2006-08-04 03:38:38 -07:00
rules_ops_put ( ops ) ;
return 0 ;
errout_free :
2008-04-16 02:01:56 -07:00
release_net ( rule - > fr_net ) ;
2006-08-04 03:38:38 -07:00
kfree ( rule ) ;
errout :
rules_ops_put ( ops ) ;
return err ;
}
2007-03-25 23:20:05 -07:00
static int fib_nl_delrule ( struct sk_buff * skb , struct nlmsghdr * nlh , void * arg )
2006-08-04 03:38:38 -07:00
{
2008-03-26 02:26:21 +09:00
struct net * net = sock_net ( skb - > sk ) ;
2006-08-04 03:38:38 -07:00
struct fib_rule_hdr * frh = nlmsg_data ( nlh ) ;
struct fib_rules_ops * ops = NULL ;
2007-03-26 17:14:15 -07:00
struct fib_rule * rule , * tmp ;
2006-08-04 03:38:38 -07:00
struct nlattr * tb [ FRA_MAX + 1 ] ;
int err = - EINVAL ;
if ( nlh - > nlmsg_len < nlmsg_msg_size ( sizeof ( * frh ) ) )
goto errout ;
2008-01-10 03:20:28 -08:00
ops = lookup_rules_ops ( net , frh - > family ) ;
2006-08-04 03:38:38 -07:00
if ( ops = = NULL ) {
err = EAFNOSUPPORT ;
goto errout ;
}
err = nlmsg_parse ( nlh , sizeof ( * frh ) , tb , FRA_MAX , ops - > policy ) ;
if ( err < 0 )
goto errout ;
2007-03-24 12:46:02 -07:00
err = validate_rulemsg ( frh , tb , ops ) ;
if ( err < 0 )
goto errout ;
2007-09-16 15:44:27 -07:00
list_for_each_entry ( rule , & ops - > rules_list , list ) {
2006-08-04 03:38:38 -07:00
if ( frh - > action & & ( frh - > action ! = rule - > action ) )
continue ;
2006-08-10 23:09:48 -07:00
if ( frh - > table & & ( frh_get_table ( frh , tb ) ! = rule - > table ) )
2006-08-04 03:38:38 -07:00
continue ;
if ( tb [ FRA_PRIORITY ] & &
( rule - > pref ! = nla_get_u32 ( tb [ FRA_PRIORITY ] ) ) )
continue ;
if ( tb [ FRA_IFNAME ] & &
nla_strcmp ( tb [ FRA_IFNAME ] , rule - > ifname ) )
continue ;
2006-11-09 15:22:18 -08:00
if ( tb [ FRA_FWMARK ] & &
( rule - > mark ! = nla_get_u32 ( tb [ FRA_FWMARK ] ) ) )
continue ;
if ( tb [ FRA_FWMASK ] & &
( rule - > mark_mask ! = nla_get_u32 ( tb [ FRA_FWMASK ] ) ) )
continue ;
2006-08-04 03:38:38 -07:00
if ( ! ops - > compare ( rule , frh , tb ) )
continue ;
if ( rule - > flags & FIB_RULE_PERMANENT ) {
err = - EPERM ;
goto errout ;
}
list_del_rcu ( & rule - > list ) ;
2007-03-26 17:14:15 -07:00
if ( rule - > action = = FR_ACT_GOTO )
ops - > nr_goto_rules - - ;
/*
* Check if this rule is a target to any of them . If so ,
* disable them . As this operation is eventually very
* expensive , it is only performed if goto rules have
* actually been added .
*/
if ( ops - > nr_goto_rules > 0 ) {
2007-09-16 15:44:27 -07:00
list_for_each_entry ( tmp , & ops - > rules_list , list ) {
2007-03-26 17:14:15 -07:00
if ( tmp - > ctarget = = rule ) {
rcu_assign_pointer ( tmp - > ctarget , NULL ) ;
ops - > unresolved_rules + + ;
}
}
}
2006-08-04 03:38:38 -07:00
synchronize_rcu ( ) ;
2008-01-20 16:46:41 -08:00
notify_rule_change ( RTM_DELRULE , rule , ops , nlh ,
2006-08-15 00:32:48 -07:00
NETLINK_CB ( skb ) . pid ) ;
2006-08-04 03:38:38 -07:00
fib_rule_put ( rule ) ;
2007-03-27 13:56:52 -07:00
flush_route_cache ( ops ) ;
2006-08-04 03:38:38 -07:00
rules_ops_put ( ops ) ;
return 0 ;
}
err = - ENOENT ;
errout :
rules_ops_put ( ops ) ;
return err ;
}
2006-11-10 14:10:15 -08:00
static inline size_t fib_rule_nlmsg_size ( struct fib_rules_ops * ops ,
struct fib_rule * rule )
{
size_t payload = NLMSG_ALIGN ( sizeof ( struct fib_rule_hdr ) )
+ nla_total_size ( IFNAMSIZ ) /* FRA_IFNAME */
+ nla_total_size ( 4 ) /* FRA_PRIORITY */
+ nla_total_size ( 4 ) /* FRA_TABLE */
+ nla_total_size ( 4 ) /* FRA_FWMARK */
+ nla_total_size ( 4 ) ; /* FRA_FWMASK */
if ( ops - > nlmsg_payload )
payload + = ops - > nlmsg_payload ( rule ) ;
return payload ;
}
2006-08-04 03:38:38 -07:00
static int fib_nl_fill_rule ( struct sk_buff * skb , struct fib_rule * rule ,
u32 pid , u32 seq , int type , int flags ,
struct fib_rules_ops * ops )
{
struct nlmsghdr * nlh ;
struct fib_rule_hdr * frh ;
nlh = nlmsg_put ( skb , pid , seq , type , sizeof ( * frh ) , flags ) ;
if ( nlh = = NULL )
2007-01-31 23:16:40 -08:00
return - EMSGSIZE ;
2006-08-04 03:38:38 -07:00
frh = nlmsg_data ( nlh ) ;
frh - > table = rule - > table ;
2006-08-10 23:09:48 -07:00
NLA_PUT_U32 ( skb , FRA_TABLE , rule - > table ) ;
2006-08-04 03:38:38 -07:00
frh - > res1 = 0 ;
frh - > res2 = 0 ;
frh - > action = rule - > action ;
frh - > flags = rule - > flags ;
2007-03-26 17:14:15 -07:00
if ( rule - > action = = FR_ACT_GOTO & & rule - > ctarget = = NULL )
frh - > flags | = FIB_RULE_UNRESOLVED ;
2007-03-26 17:37:59 -07:00
if ( rule - > ifname [ 0 ] ) {
2006-08-04 03:38:38 -07:00
NLA_PUT_STRING ( skb , FRA_IFNAME , rule - > ifname ) ;
2007-03-26 17:37:59 -07:00
if ( rule - > ifindex = = - 1 )
frh - > flags | = FIB_RULE_DEV_DETACHED ;
}
2006-08-04 03:38:38 -07:00
if ( rule - > pref )
NLA_PUT_U32 ( skb , FRA_PRIORITY , rule - > pref ) ;
2006-11-09 15:22:18 -08:00
if ( rule - > mark )
NLA_PUT_U32 ( skb , FRA_FWMARK , rule - > mark ) ;
if ( rule - > mark_mask | | rule - > mark )
NLA_PUT_U32 ( skb , FRA_FWMASK , rule - > mark_mask ) ;
2007-03-26 17:14:15 -07:00
if ( rule - > target )
NLA_PUT_U32 ( skb , FRA_GOTO , rule - > target ) ;
2006-08-04 03:38:38 -07:00
if ( ops - > fill ( rule , skb , nlh , frh ) < 0 )
goto nla_put_failure ;
return nlmsg_end ( skb , nlh ) ;
nla_put_failure :
2007-01-31 23:16:40 -08:00
nlmsg_cancel ( skb , nlh ) ;
return - EMSGSIZE ;
2006-08-04 03:38:38 -07:00
}
2007-03-25 23:24:24 -07:00
static int dump_rules ( struct sk_buff * skb , struct netlink_callback * cb ,
struct fib_rules_ops * ops )
2006-08-04 03:38:38 -07:00
{
int idx = 0 ;
struct fib_rule * rule ;
2007-09-16 15:44:27 -07:00
list_for_each_entry ( rule , & ops - > rules_list , list ) {
2007-03-25 23:24:24 -07:00
if ( idx < cb - > args [ 1 ] )
2006-08-04 03:38:38 -07:00
goto skip ;
if ( fib_nl_fill_rule ( skb , rule , NETLINK_CB ( cb - > skb ) . pid ,
cb - > nlh - > nlmsg_seq , RTM_NEWRULE ,
NLM_F_MULTI , ops ) < 0 )
break ;
skip :
idx + + ;
}
2007-03-25 23:24:24 -07:00
cb - > args [ 1 ] = idx ;
2006-08-04 03:38:38 -07:00
rules_ops_put ( ops ) ;
return skb - > len ;
}
2007-03-25 23:24:24 -07:00
static int fib_nl_dumprule ( struct sk_buff * skb , struct netlink_callback * cb )
{
2008-03-26 02:26:21 +09:00
struct net * net = sock_net ( skb - > sk ) ;
2007-03-25 23:24:24 -07:00
struct fib_rules_ops * ops ;
int idx = 0 , family ;
family = rtnl_msg_family ( cb - > nlh ) ;
if ( family ! = AF_UNSPEC ) {
/* Protocol specific dump request */
2008-01-10 03:20:28 -08:00
ops = lookup_rules_ops ( net , family ) ;
2007-03-25 23:24:24 -07:00
if ( ops = = NULL )
return - EAFNOSUPPORT ;
return dump_rules ( skb , cb , ops ) ;
}
rcu_read_lock ( ) ;
2008-01-10 03:20:28 -08:00
list_for_each_entry_rcu ( ops , & net - > rules_ops , list ) {
2007-03-25 23:24:24 -07:00
if ( idx < cb - > args [ 0 ] | | ! try_module_get ( ops - > owner ) )
goto skip ;
if ( dump_rules ( skb , cb , ops ) < 0 )
break ;
cb - > args [ 1 ] = 0 ;
skip :
idx + + ;
}
rcu_read_unlock ( ) ;
cb - > args [ 0 ] = idx ;
return skb - > len ;
}
2006-08-04 03:38:38 -07:00
2008-01-20 16:46:41 -08:00
static void notify_rule_change ( int event , struct fib_rule * rule ,
2006-08-15 00:32:48 -07:00
struct fib_rules_ops * ops , struct nlmsghdr * nlh ,
u32 pid )
2006-08-04 03:38:38 -07:00
{
2008-01-20 16:46:41 -08:00
struct net * net ;
2006-08-15 00:32:48 -07:00
struct sk_buff * skb ;
int err = - ENOBUFS ;
2006-08-04 03:38:38 -07:00
2008-01-20 16:46:41 -08:00
net = ops - > fro_net ;
2006-11-10 14:10:15 -08:00
skb = nlmsg_new ( fib_rule_nlmsg_size ( ops , rule ) , GFP_KERNEL ) ;
2006-08-04 03:38:38 -07:00
if ( skb = = NULL )
2006-08-15 00:32:48 -07:00
goto errout ;
err = fib_nl_fill_rule ( skb , rule , pid , nlh - > nlmsg_seq , event , 0 , ops ) ;
2007-01-31 23:16:40 -08:00
if ( err < 0 ) {
/* -EMSGSIZE implies BUG in fib_rule_nlmsg_size() */
WARN_ON ( err = = - EMSGSIZE ) ;
kfree_skb ( skb ) ;
goto errout ;
}
2008-01-20 16:46:41 -08:00
2008-01-10 03:20:28 -08:00
err = rtnl_notify ( skb , net , pid , ops - > nlgroup , nlh , GFP_KERNEL ) ;
2006-08-15 00:32:48 -07:00
errout :
if ( err < 0 )
2008-01-10 03:20:28 -08:00
rtnl_set_sk_err ( net , ops - > nlgroup , err ) ;
2006-08-04 03:38:38 -07:00
}
static void attach_rules ( struct list_head * rules , struct net_device * dev )
{
struct fib_rule * rule ;
list_for_each_entry ( rule , rules , list ) {
if ( rule - > ifindex = = - 1 & &
strcmp ( dev - > name , rule - > ifname ) = = 0 )
rule - > ifindex = dev - > ifindex ;
}
}
static void detach_rules ( struct list_head * rules , struct net_device * dev )
{
struct fib_rule * rule ;
list_for_each_entry ( rule , rules , list )
if ( rule - > ifindex = = dev - > ifindex )
rule - > ifindex = - 1 ;
}
static int fib_rules_event ( struct notifier_block * this , unsigned long event ,
void * ptr )
{
struct net_device * dev = ptr ;
2008-03-25 21:47:49 +09:00
struct net * net = dev_net ( dev ) ;
2006-08-04 03:38:38 -07:00
struct fib_rules_ops * ops ;
ASSERT_RTNL ( ) ;
rcu_read_lock ( ) ;
switch ( event ) {
case NETDEV_REGISTER :
2008-01-10 03:20:28 -08:00
list_for_each_entry ( ops , & net - > rules_ops , list )
2007-09-16 15:44:27 -07:00
attach_rules ( & ops - > rules_list , dev ) ;
2006-08-04 03:38:38 -07:00
break ;
case NETDEV_UNREGISTER :
2008-01-10 03:20:28 -08:00
list_for_each_entry ( ops , & net - > rules_ops , list )
2007-09-16 15:44:27 -07:00
detach_rules ( & ops - > rules_list , dev ) ;
2006-08-04 03:38:38 -07:00
break ;
}
rcu_read_unlock ( ) ;
return NOTIFY_DONE ;
}
static struct notifier_block fib_rules_notifier = {
. notifier_call = fib_rules_event ,
} ;
2008-01-10 03:20:28 -08:00
static int fib_rules_net_init ( struct net * net )
{
INIT_LIST_HEAD ( & net - > rules_ops ) ;
spin_lock_init ( & net - > rules_mod_lock ) ;
return 0 ;
}
static struct pernet_operations fib_rules_net_ops = {
. init = fib_rules_net_init ,
} ;
2006-08-04 03:38:38 -07:00
static int __init fib_rules_init ( void )
{
2008-01-10 03:20:28 -08:00
int err ;
2007-03-25 23:20:05 -07:00
rtnl_register ( PF_UNSPEC , RTM_NEWRULE , fib_nl_newrule , NULL ) ;
rtnl_register ( PF_UNSPEC , RTM_DELRULE , fib_nl_delrule , NULL ) ;
2007-03-25 23:24:24 -07:00
rtnl_register ( PF_UNSPEC , RTM_GETRULE , NULL , fib_nl_dumprule ) ;
2007-03-25 23:20:05 -07:00
2008-01-10 03:20:28 -08:00
err = register_netdevice_notifier ( & fib_rules_notifier ) ;
if ( err < 0 )
goto fail ;
err = register_pernet_subsys ( & fib_rules_net_ops ) ;
if ( err < 0 )
goto fail_unregister ;
return 0 ;
fail_unregister :
unregister_netdevice_notifier ( & fib_rules_notifier ) ;
fail :
rtnl_unregister ( PF_UNSPEC , RTM_NEWRULE ) ;
rtnl_unregister ( PF_UNSPEC , RTM_DELRULE ) ;
rtnl_unregister ( PF_UNSPEC , RTM_GETRULE ) ;
return err ;
2006-08-04 03:38:38 -07:00
}
subsys_initcall ( fib_rules_init ) ;