2019-05-24 14:43:04 -07:00
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Generic nexthop implementation
*
* Copyright ( c ) 2017 - 19 Cumulus Networks
* Copyright ( c ) 2017 - 19 David Ahern < dsa @ cumulusnetworks . com >
*/
# ifndef __LINUX_NEXTHOP_H
# define __LINUX_NEXTHOP_H
# include <linux/netdevice.h>
2020-05-21 22:26:15 -07:00
# include <linux/notifier.h>
2019-06-03 20:19:52 -07:00
# include <linux/route.h>
2019-05-24 14:43:04 -07:00
# include <linux/types.h>
# include <net/ip_fib.h>
2019-05-24 14:43:06 -07:00
# include <net/ip6_fib.h>
2019-05-24 14:43:04 -07:00
# include <net/netlink.h>
# define NEXTHOP_VALID_USER_FLAGS RTNH_F_ONLINK
struct nexthop ;
struct nh_config {
u32 nh_id ;
u8 nh_family ;
u8 nh_protocol ;
u8 nh_blackhole ;
2020-05-21 22:26:13 -07:00
u8 nh_fdb ;
2019-05-24 14:43:04 -07:00
u32 nh_flags ;
int nh_ifindex ;
struct net_device * dev ;
2019-05-24 14:43:05 -07:00
union {
__be32 ipv4 ;
2019-05-24 14:43:06 -07:00
struct in6_addr ipv6 ;
2019-05-24 14:43:05 -07:00
} gw ;
2019-05-24 14:43:08 -07:00
struct nlattr * nh_grp ;
u16 nh_grp_type ;
2019-05-24 14:43:07 -07:00
struct nlattr * nh_encap ;
u16 nh_encap_type ;
2019-05-24 14:43:04 -07:00
u32 nlflags ;
struct nl_info nlinfo ;
} ;
struct nh_info {
struct hlist_node dev_hash ; /* entry on netns devhash */
struct nexthop * nh_parent ;
u8 family ;
bool reject_nh ;
2020-05-21 22:26:13 -07:00
bool fdb_nh ;
2019-05-24 14:43:04 -07:00
union {
struct fib_nh_common fib_nhc ;
2019-05-24 14:43:05 -07:00
struct fib_nh fib_nh ;
2019-05-24 14:43:06 -07:00
struct fib6_nh fib6_nh ;
2019-05-24 14:43:04 -07:00
} ;
} ;
2019-05-24 14:43:08 -07:00
struct nh_grp_entry {
struct nexthop * nh ;
u8 weight ;
atomic_t upper_bound ;
struct list_head nh_list ;
struct nexthop * nh_parent ; /* nexthop of group with this entry */
} ;
struct nh_group {
2020-05-26 12:56:15 -06:00
struct nh_group * spare ; /* spare group for removals */
2019-05-24 14:43:08 -07:00
u16 num_nh ;
bool mpath ;
2020-06-08 20:54:43 -06:00
bool fdb_nh ;
2019-05-24 14:43:08 -07:00
bool has_v4 ;
2020-02-28 18:14:11 -06:00
struct nh_grp_entry nh_entries [ ] ;
2019-05-24 14:43:08 -07:00
} ;
2019-05-24 14:43:04 -07:00
struct nexthop {
struct rb_node rb_node ; /* entry on netns rbtree */
2019-06-03 20:19:51 -07:00
struct list_head fi_list ; /* v4 entries using nh */
2019-06-03 20:19:52 -07:00
struct list_head f6i_list ; /* v6 entries using nh */
2020-05-21 22:26:13 -07:00
struct list_head fdb_list ; /* fdb entries using this nh */
2019-05-24 14:43:08 -07:00
struct list_head grp_list ; /* nh group entries using this nh */
2019-05-24 14:43:04 -07:00
struct net * net ;
u32 id ;
u8 protocol ; /* app managing this nh */
u8 nh_flags ;
2019-05-24 14:43:08 -07:00
bool is_group ;
2019-05-24 14:43:04 -07:00
refcount_t refcnt ;
struct rcu_head rcu ;
union {
struct nh_info __rcu * nh_info ;
2019-05-24 14:43:08 -07:00
struct nh_group __rcu * nh_grp ;
2019-05-24 14:43:04 -07:00
} ;
} ;
2020-05-21 22:26:15 -07:00
enum nexthop_event_type {
NEXTHOP_EVENT_DEL
} ;
int register_nexthop_notifier ( struct net * net , struct notifier_block * nb ) ;
int unregister_nexthop_notifier ( struct net * net , struct notifier_block * nb ) ;
2019-05-24 14:43:04 -07:00
/* caller is holding rcu or rtnl; no reference taken to nexthop */
struct nexthop * nexthop_find_by_id ( struct net * net , u32 id ) ;
void nexthop_free_rcu ( struct rcu_head * head ) ;
static inline bool nexthop_get ( struct nexthop * nh )
{
return refcount_inc_not_zero ( & nh - > refcnt ) ;
}
static inline void nexthop_put ( struct nexthop * nh )
{
if ( refcount_dec_and_test ( & nh - > refcnt ) )
call_rcu ( & nh - > rcu , nexthop_free_rcu ) ;
}
2019-06-03 20:19:51 -07:00
static inline bool nexthop_cmp ( const struct nexthop * nh1 ,
const struct nexthop * nh2 )
{
return nh1 = = nh2 ;
2020-06-08 20:54:43 -06:00
}
static inline bool nexthop_is_fdb ( const struct nexthop * nh )
{
if ( nh - > is_group ) {
const struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
return nh_grp - > fdb_nh ;
} else {
const struct nh_info * nhi ;
nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
return nhi - > fdb_nh ;
}
2020-06-09 17:27:28 -06:00
}
static inline bool nexthop_has_v4 ( const struct nexthop * nh )
{
if ( nh - > is_group ) {
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
return nh_grp - > has_v4 ;
}
return false ;
2019-06-03 20:19:51 -07:00
}
2019-05-24 14:43:08 -07:00
static inline bool nexthop_is_multipath ( const struct nexthop * nh )
{
if ( nh - > is_group ) {
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
return nh_grp - > mpath ;
}
return false ;
}
struct nexthop * nexthop_select_path ( struct nexthop * nh , int hash ) ;
static inline unsigned int nexthop_num_path ( const struct nexthop * nh )
{
unsigned int rc = 1 ;
2020-05-26 12:56:16 -06:00
if ( nh - > is_group ) {
2019-05-24 14:43:08 -07:00
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
2020-05-26 12:56:16 -06:00
if ( nh_grp - > mpath )
rc = nh_grp - > num_nh ;
2019-05-24 14:43:08 -07:00
}
return rc ;
}
static inline
2020-05-26 12:56:16 -06:00
struct nexthop * nexthop_mpath_select ( const struct nh_group * nhg , int nhsel )
2019-05-24 14:43:08 -07:00
{
/* for_nexthops macros in fib_semantics.c grabs a pointer to
* the nexthop before checking nhsel
*/
2019-06-07 18:31:07 +03:00
if ( nhsel > = nhg - > num_nh )
2019-05-24 14:43:08 -07:00
return NULL ;
return nhg - > nh_entries [ nhsel ] . nh ;
}
static inline
2019-09-04 10:11:58 -04:00
int nexthop_mpath_fill_node ( struct sk_buff * skb , struct nexthop * nh ,
u8 rt_family )
2019-05-24 14:43:08 -07:00
{
struct nh_group * nhg = rtnl_dereference ( nh - > nh_grp ) ;
int i ;
for ( i = 0 ; i < nhg - > num_nh ; i + + ) {
struct nexthop * nhe = nhg - > nh_entries [ i ] . nh ;
struct nh_info * nhi = rcu_dereference_rtnl ( nhe - > nh_info ) ;
struct fib_nh_common * nhc = & nhi - > fib_nhc ;
int weight = nhg - > nh_entries [ i ] . weight ;
2019-09-04 10:11:58 -04:00
if ( fib_add_nexthop ( skb , nhc , weight , rt_family ) < 0 )
2019-05-24 14:43:08 -07:00
return - EMSGSIZE ;
}
return 0 ;
}
2019-05-24 14:43:04 -07:00
/* called with rcu lock */
static inline bool nexthop_is_blackhole ( const struct nexthop * nh )
{
const struct nh_info * nhi ;
2020-05-26 12:56:16 -06:00
if ( nh - > is_group ) {
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
if ( nh_grp - > num_nh > 1 )
2019-05-24 14:43:08 -07:00
return false ;
2020-05-26 12:56:16 -06:00
nh = nh_grp - > nh_entries [ 0 ] . nh ;
2019-05-24 14:43:08 -07:00
}
nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
2019-05-24 14:43:04 -07:00
return nhi - > reject_nh ;
}
2019-06-03 20:19:49 -07:00
2019-06-03 20:19:51 -07:00
static inline void nexthop_path_fib_result ( struct fib_result * res , int hash )
{
struct nh_info * nhi ;
struct nexthop * nh ;
nh = nexthop_select_path ( res - > fi - > nh , hash ) ;
nhi = rcu_dereference ( nh - > nh_info ) ;
res - > nhc = & nhi - > fib_nhc ;
}
/* called with rcu read lock or rtnl held */
static inline
struct fib_nh_common * nexthop_fib_nhc ( struct nexthop * nh , int nhsel )
{
struct nh_info * nhi ;
BUILD_BUG_ON ( offsetof ( struct fib_nh , nh_common ) ! = 0 ) ;
BUILD_BUG_ON ( offsetof ( struct fib6_nh , nh_common ) ! = 0 ) ;
2020-05-26 12:56:16 -06:00
if ( nh - > is_group ) {
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
if ( nh_grp - > mpath ) {
nh = nexthop_mpath_select ( nh_grp , nhsel ) ;
if ( ! nh )
return NULL ;
}
2019-06-03 20:19:51 -07:00
}
nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
return & nhi - > fib_nhc ;
}
2020-05-26 12:56:17 -06:00
/* called from fib_table_lookup with rcu_lock */
static inline
struct fib_nh_common * nexthop_get_nhc_lookup ( const struct nexthop * nh ,
int fib_flags ,
const struct flowi4 * flp ,
int * nhsel )
{
struct nh_info * nhi ;
if ( nh - > is_group ) {
struct nh_group * nhg = rcu_dereference ( nh - > nh_grp ) ;
int i ;
for ( i = 0 ; i < nhg - > num_nh ; i + + ) {
struct nexthop * nhe = nhg - > nh_entries [ i ] . nh ;
nhi = rcu_dereference ( nhe - > nh_info ) ;
if ( fib_lookup_good_nhc ( & nhi - > fib_nhc , fib_flags , flp ) ) {
* nhsel = i ;
return & nhi - > fib_nhc ;
}
}
} else {
nhi = rcu_dereference ( nh - > nh_info ) ;
if ( fib_lookup_good_nhc ( & nhi - > fib_nhc , fib_flags , flp ) ) {
* nhsel = 0 ;
return & nhi - > fib_nhc ;
}
}
return NULL ;
}
2020-05-26 12:56:18 -06:00
static inline bool nexthop_uses_dev ( const struct nexthop * nh ,
const struct net_device * dev )
{
struct nh_info * nhi ;
if ( nh - > is_group ) {
struct nh_group * nhg = rcu_dereference ( nh - > nh_grp ) ;
int i ;
for ( i = 0 ; i < nhg - > num_nh ; i + + ) {
struct nexthop * nhe = nhg - > nh_entries [ i ] . nh ;
nhi = rcu_dereference ( nhe - > nh_info ) ;
if ( nhc_l3mdev_matches_dev ( & nhi - > fib_nhc , dev ) )
return true ;
}
} else {
nhi = rcu_dereference ( nh - > nh_info ) ;
if ( nhc_l3mdev_matches_dev ( & nhi - > fib_nhc , dev ) )
return true ;
}
return false ;
}
2019-06-03 20:19:49 -07:00
static inline unsigned int fib_info_num_path ( const struct fib_info * fi )
{
2019-06-03 20:19:51 -07:00
if ( unlikely ( fi - > nh ) )
return nexthop_num_path ( fi - > nh ) ;
2019-06-03 20:19:49 -07:00
return fi - > fib_nhs ;
}
2019-06-03 20:19:51 -07:00
int fib_check_nexthop ( struct nexthop * nh , u8 scope ,
struct netlink_ext_ack * extack ) ;
2019-06-03 20:19:49 -07:00
static inline struct fib_nh_common * fib_info_nhc ( struct fib_info * fi , int nhsel )
{
2019-06-03 20:19:51 -07:00
if ( unlikely ( fi - > nh ) )
return nexthop_fib_nhc ( fi - > nh , nhsel ) ;
2019-06-03 20:19:49 -07:00
return & fi - > fib_nh [ nhsel ] . nh_common ;
}
2019-06-03 20:19:51 -07:00
/* only used when fib_nh is built into fib_info */
2019-06-03 20:19:49 -07:00
static inline struct fib_nh * fib_info_nh ( struct fib_info * fi , int nhsel )
{
2019-06-03 20:19:51 -07:00
WARN_ON ( fi - > nh ) ;
2019-06-03 20:19:49 -07:00
return & fi - > fib_nh [ nhsel ] ;
}
2019-06-03 20:19:52 -07:00
/*
* IPv6 variants
*/
int fib6_check_nexthop ( struct nexthop * nh , struct fib6_config * cfg ,
struct netlink_ext_ack * extack ) ;
static inline struct fib6_nh * nexthop_fib6_nh ( struct nexthop * nh )
{
struct nh_info * nhi ;
2020-05-26 12:56:16 -06:00
if ( nh - > is_group ) {
struct nh_group * nh_grp ;
nh_grp = rcu_dereference_rtnl ( nh - > nh_grp ) ;
nh = nexthop_mpath_select ( nh_grp , 0 ) ;
2019-06-03 20:19:52 -07:00
if ( ! nh )
return NULL ;
}
nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
if ( nhi - > family = = AF_INET6 )
return & nhi - > fib6_nh ;
return NULL ;
}
static inline struct net_device * fib6_info_nh_dev ( struct fib6_info * f6i )
{
struct fib6_nh * fib6_nh ;
fib6_nh = f6i - > nh ? nexthop_fib6_nh ( f6i - > nh ) : f6i - > fib6_nh ;
return fib6_nh - > fib_nh_dev ;
}
static inline void nexthop_path_fib6_result ( struct fib6_result * res , int hash )
{
struct nexthop * nh = res - > f6i - > nh ;
struct nh_info * nhi ;
nh = nexthop_select_path ( nh , hash ) ;
nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
if ( nhi - > reject_nh ) {
res - > fib6_type = RTN_BLACKHOLE ;
res - > fib6_flags | = RTF_REJECT ;
res - > nh = nexthop_fib6_nh ( nh ) ;
} else {
res - > nh = & nhi - > fib6_nh ;
}
}
2019-06-08 14:53:22 -07:00
int nexthop_for_each_fib6_nh ( struct nexthop * nh ,
int ( * cb ) ( struct fib6_nh * nh , void * arg ) ,
void * arg ) ;
2020-05-21 22:26:13 -07:00
static inline int nexthop_get_family ( struct nexthop * nh )
{
struct nh_info * nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
return nhi - > family ;
}
static inline
struct fib_nh_common * nexthop_fdb_nhc ( struct nexthop * nh )
{
struct nh_info * nhi = rcu_dereference_rtnl ( nh - > nh_info ) ;
return & nhi - > fib_nhc ;
}
static inline struct fib_nh_common * nexthop_path_fdb_result ( struct nexthop * nh ,
int hash )
{
struct nh_info * nhi ;
struct nexthop * nhp ;
nhp = nexthop_select_path ( nh , hash ) ;
if ( unlikely ( ! nhp ) )
return NULL ;
nhi = rcu_dereference ( nhp - > nh_info ) ;
return & nhi - > fib_nhc ;
}
2019-05-24 14:43:04 -07:00
# endif