2015-03-03 19:10:47 -06:00
# ifndef MPLS_INTERNAL_H
# define MPLS_INTERNAL_H
2016-09-30 19:08:06 +02:00
# include <net/mpls.h>
2015-03-03 19:10:47 -06:00
struct mpls_entry_decoded {
u32 label ;
u8 ttl ;
u8 tc ;
u8 bos ;
} ;
2017-01-16 14:16:37 +00:00
struct mpls_pcpu_stats {
struct mpls_link_stats stats ;
struct u64_stats_sync syncp ;
} ;
2015-04-22 11:14:37 +01:00
struct mpls_dev {
2017-01-16 14:16:37 +00:00
int input_enabled ;
2017-02-20 08:03:30 -08:00
struct net_device * dev ;
2017-01-16 14:16:37 +00:00
struct mpls_pcpu_stats __percpu * stats ;
struct ctl_table_header * sysctl ;
struct rcu_head rcu ;
2015-04-22 11:14:37 +01:00
} ;
2017-01-16 14:16:37 +00:00
# if BITS_PER_LONG == 32
# define MPLS_INC_STATS_LEN(mdev, len, pkts_field, bytes_field) \
do { \
__typeof__ ( * ( mdev ) - > stats ) * ptr = \
raw_cpu_ptr ( ( mdev ) - > stats ) ; \
local_bh_disable ( ) ; \
u64_stats_update_begin ( & ptr - > syncp ) ; \
ptr - > stats . pkts_field + + ; \
ptr - > stats . bytes_field + = ( len ) ; \
u64_stats_update_end ( & ptr - > syncp ) ; \
local_bh_enable ( ) ; \
} while ( 0 )
# define MPLS_INC_STATS(mdev, field) \
do { \
__typeof__ ( * ( mdev ) - > stats ) * ptr = \
raw_cpu_ptr ( ( mdev ) - > stats ) ; \
local_bh_disable ( ) ; \
u64_stats_update_begin ( & ptr - > syncp ) ; \
ptr - > stats . field + + ; \
u64_stats_update_end ( & ptr - > syncp ) ; \
local_bh_enable ( ) ; \
} while ( 0 )
# else
# define MPLS_INC_STATS_LEN(mdev, len, pkts_field, bytes_field) \
do { \
this_cpu_inc ( ( mdev ) - > stats - > stats . pkts_field ) ; \
this_cpu_add ( ( mdev ) - > stats - > stats . bytes_field , ( len ) ) ; \
} while ( 0 )
# define MPLS_INC_STATS(mdev, field) \
this_cpu_inc ( ( mdev ) - > stats - > stats . field )
# endif
2015-03-03 19:10:47 -06:00
struct sk_buff ;
2015-10-23 06:03:27 -07:00
# define LABEL_NOT_SPECIFIED (1 << 20)
# define MAX_NEW_LABELS 2
/* This maximum ha length copied from the definition of struct neighbour */
2015-10-27 00:37:36 +00:00
# define VIA_ALEN_ALIGN sizeof(unsigned long)
# define MAX_VIA_ALEN (ALIGN(MAX_ADDR_LEN, VIA_ALEN_ALIGN))
2015-10-23 06:03:27 -07:00
enum mpls_payload_type {
MPT_UNSPEC , /* IPv4 or IPv6 */
MPT_IPV4 = 4 ,
MPT_IPV6 = 6 ,
/* Other types not implemented:
* - Pseudo - wire with or without control word ( RFC4385 )
* - GAL ( RFC5586 )
*/
} ;
struct mpls_nh { /* next hop label forwarding entry */
struct net_device __rcu * nh_dev ;
2015-12-01 22:18:11 -08:00
unsigned int nh_flags ;
2015-10-23 06:03:27 -07:00
u32 nh_label [ MAX_NEW_LABELS ] ;
u8 nh_labels ;
u8 nh_via_alen ;
u8 nh_via_table ;
} ;
2017-03-10 20:43:24 +00:00
enum mpls_ttl_propagation {
MPLS_TTL_PROP_DEFAULT ,
MPLS_TTL_PROP_ENABLED ,
MPLS_TTL_PROP_DISABLED ,
} ;
2015-10-27 00:37:36 +00:00
/* The route, nexthops and vias are stored together in the same memory
* block :
*
* + - - - - - - - - - - - - - - - - - - - - - - +
* | mpls_route |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | mpls_nh 0 |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | . . . |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | mpls_nh n - 1 |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | alignment padding |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | via [ rt_max_alen ] 0 |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | . . . |
* + - - - - - - - - - - - - - - - - - - - - - - +
* | via [ rt_max_alen ] n - 1 |
* + - - - - - - - - - - - - - - - - - - - - - - +
*/
2015-10-23 06:03:27 -07:00
struct mpls_route { /* next hop label forwarding entry */
struct rcu_head rt_rcu ;
u8 rt_protocol ;
u8 rt_payload_type ;
2015-10-27 00:37:36 +00:00
u8 rt_max_alen ;
2017-03-10 20:43:24 +00:00
u8 rt_ttl_propagate ;
2015-10-27 00:37:36 +00:00
unsigned int rt_nhn ;
2015-12-01 22:18:11 -08:00
unsigned int rt_nhn_alive ;
2015-10-23 06:03:27 -07:00
struct mpls_nh rt_nh [ 0 ] ;
} ;
# define for_nexthops(rt) { \
int nhsel ; struct mpls_nh * nh ; \
for ( nhsel = 0 , nh = ( rt ) - > rt_nh ; \
nhsel < ( rt ) - > rt_nhn ; \
nh + + , nhsel + + )
# define change_nexthops(rt) { \
int nhsel ; struct mpls_nh * nh ; \
for ( nhsel = 0 , nh = ( struct mpls_nh * ) ( ( rt ) - > rt_nh ) ; \
nhsel < ( rt ) - > rt_nhn ; \
nh + + , nhsel + + )
# define endfor_nexthops(rt) }
2015-03-03 19:10:47 -06:00
static inline struct mpls_shim_hdr mpls_entry_encode ( u32 label , unsigned ttl , unsigned tc , bool bos )
{
struct mpls_shim_hdr result ;
result . label_stack_entry =
cpu_to_be32 ( ( label < < MPLS_LS_LABEL_SHIFT ) |
( tc < < MPLS_LS_TC_SHIFT ) |
( bos ? ( 1 < < MPLS_LS_S_SHIFT ) : 0 ) |
( ttl < < MPLS_LS_TTL_SHIFT ) ) ;
return result ;
}
static inline struct mpls_entry_decoded mpls_entry_decode ( struct mpls_shim_hdr * hdr )
{
struct mpls_entry_decoded result ;
unsigned entry = be32_to_cpu ( hdr - > label_stack_entry ) ;
result . label = ( entry & MPLS_LS_LABEL_MASK ) > > MPLS_LS_LABEL_SHIFT ;
result . ttl = ( entry & MPLS_LS_TTL_MASK ) > > MPLS_LS_TTL_SHIFT ;
result . tc = ( entry & MPLS_LS_TC_MASK ) > > MPLS_LS_TC_SHIFT ;
result . bos = ( entry & MPLS_LS_S_MASK ) > > MPLS_LS_S_SHIFT ;
return result ;
}
2017-01-16 14:16:37 +00:00
static inline struct mpls_dev * mpls_dev_get ( const struct net_device * dev )
{
return rcu_dereference_rtnl ( dev - > mpls_ptr ) ;
}
2015-07-21 10:43:52 +02:00
int nla_put_labels ( struct sk_buff * skb , int attrtype , u8 labels ,
const u32 label [ ] ) ;
2015-10-23 06:03:27 -07:00
int nla_get_labels ( const struct nlattr * nla , u32 max_labels , u8 * labels ,
2015-07-21 10:43:52 +02:00
u32 label [ ] ) ;
2015-10-23 06:03:27 -07:00
int nla_get_via ( const struct nlattr * nla , u8 * via_alen , u8 * via_table ,
u8 via [ ] ) ;
2015-07-21 10:43:52 +02:00
bool mpls_output_possible ( const struct net_device * dev ) ;
unsigned int mpls_dev_mtu ( const struct net_device * dev ) ;
bool mpls_pkt_too_big ( const struct sk_buff * skb , unsigned int mtu ) ;
2017-01-16 14:16:37 +00:00
void mpls_stats_inc_outucastpkts ( struct net_device * dev ,
const struct sk_buff * skb ) ;
2015-03-03 19:13:19 -06:00
2015-03-03 19:10:47 -06:00
# endif /* MPLS_INTERNAL_H */