2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2015-09-29 20:07:11 -07:00
/*
* net / l3mdev / l3mdev . c - L3 master device implementation
* Copyright ( c ) 2015 Cumulus Networks
* Copyright ( c ) 2015 David Ahern < dsa @ cumulusnetworks . com >
*/
# include <linux/netdevice.h>
2016-06-08 10:55:39 -07:00
# include <net/fib_rules.h>
2015-09-29 20:07:11 -07:00
# include <net/l3mdev.h>
2020-06-20 00:54:43 +02:00
static DEFINE_SPINLOCK ( l3mdev_lock ) ;
struct l3mdev_handler {
lookup_by_table_id_t dev_lookup ;
} ;
static struct l3mdev_handler l3mdev_handlers [ L3MDEV_TYPE_MAX + 1 ] ;
static int l3mdev_check_type ( enum l3mdev_type l3type )
{
if ( l3type < = L3MDEV_TYPE_UNSPEC | | l3type > L3MDEV_TYPE_MAX )
return - EINVAL ;
return 0 ;
}
int l3mdev_table_lookup_register ( enum l3mdev_type l3type ,
lookup_by_table_id_t fn )
{
struct l3mdev_handler * hdlr ;
int res ;
res = l3mdev_check_type ( l3type ) ;
if ( res )
return res ;
hdlr = & l3mdev_handlers [ l3type ] ;
spin_lock ( & l3mdev_lock ) ;
if ( hdlr - > dev_lookup ) {
res = - EBUSY ;
goto unlock ;
}
hdlr - > dev_lookup = fn ;
res = 0 ;
unlock :
spin_unlock ( & l3mdev_lock ) ;
return res ;
}
EXPORT_SYMBOL_GPL ( l3mdev_table_lookup_register ) ;
void l3mdev_table_lookup_unregister ( enum l3mdev_type l3type ,
lookup_by_table_id_t fn )
{
struct l3mdev_handler * hdlr ;
if ( l3mdev_check_type ( l3type ) )
return ;
hdlr = & l3mdev_handlers [ l3type ] ;
spin_lock ( & l3mdev_lock ) ;
if ( hdlr - > dev_lookup = = fn )
hdlr - > dev_lookup = NULL ;
spin_unlock ( & l3mdev_lock ) ;
}
EXPORT_SYMBOL_GPL ( l3mdev_table_lookup_unregister ) ;
int l3mdev_ifindex_lookup_by_table_id ( enum l3mdev_type l3type ,
struct net * net , u32 table_id )
{
lookup_by_table_id_t lookup ;
struct l3mdev_handler * hdlr ;
int ifindex = - EINVAL ;
int res ;
res = l3mdev_check_type ( l3type ) ;
if ( res )
return res ;
hdlr = & l3mdev_handlers [ l3type ] ;
spin_lock ( & l3mdev_lock ) ;
lookup = hdlr - > dev_lookup ;
if ( ! lookup )
goto unlock ;
ifindex = lookup ( net , table_id ) ;
unlock :
spin_unlock ( & l3mdev_lock ) ;
return ifindex ;
}
EXPORT_SYMBOL_GPL ( l3mdev_ifindex_lookup_by_table_id ) ;
2015-09-29 20:07:11 -07:00
/**
* l3mdev_master_ifindex - get index of L3 master device
* @ dev : targeted interface
*/
2016-02-24 11:47:02 -08:00
int l3mdev_master_ifindex_rcu ( const struct net_device * dev )
2015-09-29 20:07:11 -07:00
{
int ifindex = 0 ;
if ( ! dev )
return 0 ;
if ( netif_is_l3_master ( dev ) ) {
ifindex = dev - > ifindex ;
2015-10-05 08:51:24 -07:00
} else if ( netif_is_l3_slave ( dev ) ) {
2015-09-29 20:07:11 -07:00
struct net_device * master ;
2016-02-24 11:47:02 -08:00
struct net_device * _dev = ( struct net_device * ) dev ;
/* netdev_master_upper_dev_get_rcu calls
* list_first_or_null_rcu to walk the upper dev list .
* list_first_or_null_rcu does not handle a const arg . We aren ' t
* making changes , just want the master device from that list so
* typecast to remove the const
*/
master = netdev_master_upper_dev_get_rcu ( _dev ) ;
2015-10-05 08:51:24 -07:00
if ( master )
2015-09-29 20:07:11 -07:00
ifindex = master - > ifindex ;
}
return ifindex ;
}
EXPORT_SYMBOL_GPL ( l3mdev_master_ifindex_rcu ) ;
2018-12-03 10:54:39 +01:00
/**
* l3mdev_master_upper_ifindex_by_index - get index of upper l3 master
* device
* @ net : network namespace for device index lookup
* @ ifindex : targeted interface
*/
int l3mdev_master_upper_ifindex_by_index_rcu ( struct net * net , int ifindex )
{
struct net_device * dev ;
dev = dev_get_by_index_rcu ( net , ifindex ) ;
while ( dev & & ! netif_is_l3_master ( dev ) )
dev = netdev_master_upper_dev_get ( dev ) ;
return dev ? dev - > ifindex : 0 ;
}
EXPORT_SYMBOL_GPL ( l3mdev_master_upper_ifindex_by_index_rcu ) ;
2015-09-29 20:07:11 -07:00
/**
2020-08-27 07:27:49 -04:00
* l3mdev_fib_table_rcu - get FIB table id associated with an L3
2015-09-29 20:07:11 -07:00
* master interface
* @ dev : targeted interface
*/
u32 l3mdev_fib_table_rcu ( const struct net_device * dev )
{
u32 tb_id = 0 ;
if ( ! dev )
return 0 ;
if ( netif_is_l3_master ( dev ) ) {
if ( dev - > l3mdev_ops - > l3mdev_fib_table )
tb_id = dev - > l3mdev_ops - > l3mdev_fib_table ( dev ) ;
2015-10-05 08:51:24 -07:00
} else if ( netif_is_l3_slave ( dev ) ) {
2015-09-29 20:07:11 -07:00
/* Users of netdev_master_upper_dev_get_rcu need non-const,
* but current inet_ * type functions take a const
*/
struct net_device * _dev = ( struct net_device * ) dev ;
const struct net_device * master ;
master = netdev_master_upper_dev_get_rcu ( _dev ) ;
2015-10-05 08:51:24 -07:00
if ( master & &
2015-09-29 20:07:11 -07:00
master - > l3mdev_ops - > l3mdev_fib_table )
tb_id = master - > l3mdev_ops - > l3mdev_fib_table ( master ) ;
}
return tb_id ;
}
EXPORT_SYMBOL_GPL ( l3mdev_fib_table_rcu ) ;
u32 l3mdev_fib_table_by_index ( struct net * net , int ifindex )
{
struct net_device * dev ;
u32 tb_id = 0 ;
if ( ! ifindex )
return 0 ;
rcu_read_lock ( ) ;
dev = dev_get_by_index_rcu ( net , ifindex ) ;
if ( dev )
tb_id = l3mdev_fib_table_rcu ( dev ) ;
rcu_read_unlock ( ) ;
return tb_id ;
}
EXPORT_SYMBOL_GPL ( l3mdev_fib_table_by_index ) ;
2016-05-07 16:48:59 -07:00
/**
2016-09-10 12:09:56 -07:00
* l3mdev_link_scope_lookup - IPv6 route lookup based on flow for link
* local and multicast addresses
2016-05-07 16:48:59 -07:00
* @ net : network namespace for device index lookup
* @ fl6 : IPv6 flow struct for lookup
2019-06-20 17:36:41 -07:00
* This function does not hold refcnt on the returned dst .
* Caller must hold rcu_read_lock ( ) .
2016-05-07 16:48:59 -07:00
*/
2016-09-10 12:09:56 -07:00
struct dst_entry * l3mdev_link_scope_lookup ( struct net * net ,
struct flowi6 * fl6 )
2016-05-07 16:48:59 -07:00
{
struct dst_entry * dst = NULL ;
struct net_device * dev ;
2019-06-20 17:36:41 -07:00
WARN_ON_ONCE ( ! rcu_read_lock_held ( ) ) ;
2016-05-07 16:49:00 -07:00
if ( fl6 - > flowi6_oif ) {
dev = dev_get_by_index_rcu ( net , fl6 - > flowi6_oif ) ;
if ( dev & & netif_is_l3_slave ( dev ) )
dev = netdev_master_upper_dev_get_rcu ( dev ) ;
if ( dev & & netif_is_l3_master ( dev ) & &
2016-09-10 12:09:56 -07:00
dev - > l3mdev_ops - > l3mdev_link_scope_lookup )
dst = dev - > l3mdev_ops - > l3mdev_link_scope_lookup ( dev , fl6 ) ;
2016-05-07 16:48:59 -07:00
}
return dst ;
}
2016-09-10 12:09:56 -07:00
EXPORT_SYMBOL_GPL ( l3mdev_link_scope_lookup ) ;
2016-05-07 16:48:59 -07:00
2016-06-08 10:55:39 -07:00
/**
* l3mdev_fib_rule_match - Determine if flowi references an
* L3 master device
* @ net : network namespace for device index lookup
* @ fl : flow struct
2020-10-28 01:50:59 +01:00
* @ arg : store the table the rule matched with here
2016-06-08 10:55:39 -07:00
*/
int l3mdev_fib_rule_match ( struct net * net , struct flowi * fl ,
struct fib_lookup_arg * arg )
{
struct net_device * dev ;
int rc = 0 ;
rcu_read_lock ( ) ;
dev = dev_get_by_index_rcu ( net , fl - > flowi_oif ) ;
if ( dev & & netif_is_l3_master ( dev ) & &
dev - > l3mdev_ops - > l3mdev_fib_table ) {
arg - > table = dev - > l3mdev_ops - > l3mdev_fib_table ( dev ) ;
rc = 1 ;
goto out ;
}
dev = dev_get_by_index_rcu ( net , fl - > flowi_iif ) ;
if ( dev & & netif_is_l3_master ( dev ) & &
dev - > l3mdev_ops - > l3mdev_fib_table ) {
arg - > table = dev - > l3mdev_ops - > l3mdev_fib_table ( dev ) ;
rc = 1 ;
goto out ;
}
out :
rcu_read_unlock ( ) ;
return rc ;
}
2016-09-10 12:09:52 -07:00
void l3mdev_update_flow ( struct net * net , struct flowi * fl )
{
struct net_device * dev ;
int ifindex ;
rcu_read_lock ( ) ;
if ( fl - > flowi_oif ) {
dev = dev_get_by_index_rcu ( net , fl - > flowi_oif ) ;
if ( dev ) {
ifindex = l3mdev_master_ifindex_rcu ( dev ) ;
if ( ifindex ) {
fl - > flowi_oif = ifindex ;
fl - > flowi_flags | = FLOWI_FLAG_SKIP_NH_OIF ;
goto out ;
}
}
}
if ( fl - > flowi_iif ) {
dev = dev_get_by_index_rcu ( net , fl - > flowi_iif ) ;
if ( dev ) {
ifindex = l3mdev_master_ifindex_rcu ( dev ) ;
if ( ifindex ) {
fl - > flowi_iif = ifindex ;
fl - > flowi_flags | = FLOWI_FLAG_SKIP_NH_OIF ;
}
}
}
out :
rcu_read_unlock ( ) ;
}
EXPORT_SYMBOL_GPL ( l3mdev_update_flow ) ;