2005-04-17 02:20:36 +04:00
/*
* Linux ethernet bridge
*
* Authors :
* Lennert Buytenhek < buytenh @ gnu . org >
*
* 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 .
*/
# ifndef _BR_PRIVATE_H
# define _BR_PRIVATE_H
# include <linux/netdevice.h>
# include <linux/if_bridge.h>
2008-07-31 03:27:55 +04:00
# include <net/route.h>
2005-04-17 02:20:36 +04:00
# define BR_HASH_BITS 8
# define BR_HASH_SIZE (1 << BR_HASH_BITS)
# define BR_HOLD_TIME (1*HZ)
# define BR_PORT_BITS 10
# define BR_MAX_PORTS (1<<BR_PORT_BITS)
2007-03-22 00:22:44 +03:00
# define BR_VERSION "2.3"
/* Path to usermode spanning tree program */
# define BR_STP_PROG " / sbin / bridge-stp"
2005-12-22 06:01:30 +03:00
2005-04-17 02:20:36 +04:00
typedef struct bridge_id bridge_id ;
typedef struct mac_addr mac_addr ;
typedef __u16 port_id ;
struct bridge_id
{
unsigned char prio [ 2 ] ;
unsigned char addr [ 6 ] ;
} ;
struct mac_addr
{
unsigned char addr [ 6 ] ;
} ;
2010-04-18 07:42:07 +04:00
struct br_ip
{
union {
__be32 ip4 ;
2010-04-22 20:54:22 +04:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
struct in6_addr ip6 ;
# endif
2010-04-18 07:42:07 +04:00
} u ;
__be16 proto ;
} ;
2005-04-17 02:20:36 +04:00
struct net_bridge_fdb_entry
{
struct hlist_node hlist ;
struct net_bridge_port * dst ;
struct rcu_head rcu ;
unsigned long ageing_timer ;
mac_addr addr ;
unsigned char is_local ;
unsigned char is_static ;
} ;
2010-02-27 22:41:45 +03:00
struct net_bridge_port_group {
struct net_bridge_port * port ;
struct net_bridge_port_group * next ;
struct hlist_node mglist ;
struct rcu_head rcu ;
struct timer_list timer ;
struct timer_list query_timer ;
2010-04-18 07:42:07 +04:00
struct br_ip addr ;
2010-02-27 22:41:45 +03:00
u32 queries_sent ;
} ;
struct net_bridge_mdb_entry
{
struct hlist_node hlist [ 2 ] ;
struct hlist_node mglist ;
struct net_bridge * br ;
struct net_bridge_port_group * ports ;
struct rcu_head rcu ;
struct timer_list timer ;
struct timer_list query_timer ;
2010-04-18 07:42:07 +04:00
struct br_ip addr ;
2010-02-27 22:41:45 +03:00
u32 queries_sent ;
} ;
struct net_bridge_mdb_htable
{
struct hlist_head * mhash ;
struct rcu_head rcu ;
struct net_bridge_mdb_htable * old ;
u32 size ;
u32 max ;
u32 secret ;
u32 ver ;
} ;
2005-04-17 02:20:36 +04:00
struct net_bridge_port
{
struct net_bridge * br ;
struct net_device * dev ;
struct list_head list ;
/* STP */
u8 priority ;
u8 state ;
u16 port_no ;
unsigned char topology_change_ack ;
unsigned char config_pending ;
port_id port_id ;
port_id designated_port ;
bridge_id designated_root ;
bridge_id designated_bridge ;
u32 path_cost ;
u32 designated_cost ;
struct timer_list forward_delay_timer ;
struct timer_list hold_timer ;
struct timer_list message_age_timer ;
struct kobject kobj ;
struct rcu_head rcu ;
2009-08-13 10:55:16 +04:00
unsigned long flags ;
# define BR_HAIRPIN_MODE 0x00000001
2010-02-27 22:41:45 +03:00
# ifdef CONFIG_BRIDGE_IGMP_SNOOPING
u32 multicast_startup_queries_sent ;
unsigned char multicast_router ;
struct timer_list multicast_router_timer ;
struct timer_list multicast_query_timer ;
struct hlist_head mglist ;
struct hlist_node rlist ;
# endif
2005-04-17 02:20:36 +04:00
} ;
2010-04-21 06:06:52 +04:00
struct br_cpu_netstats {
unsigned long rx_packets ;
unsigned long rx_bytes ;
unsigned long tx_packets ;
unsigned long tx_bytes ;
} ;
2005-04-17 02:20:36 +04:00
struct net_bridge
{
spinlock_t lock ;
struct list_head port_list ;
struct net_device * dev ;
2010-03-02 16:32:09 +03:00
2010-04-21 06:06:52 +04:00
struct br_cpu_netstats __percpu * stats ;
2005-04-17 02:20:36 +04:00
spinlock_t hash_lock ;
struct hlist_head hash [ BR_HASH_SIZE ] ;
2005-12-22 06:00:58 +03:00
unsigned long feature_mask ;
2008-07-31 03:27:55 +04:00
# ifdef CONFIG_BRIDGE_NETFILTER
struct rtable fake_rtable ;
# endif
2008-06-18 03:10:06 +04:00
unsigned long flags ;
# define BR_SET_MAC_ADDR 0x00000001
2005-04-17 02:20:36 +04:00
/* STP */
bridge_id designated_root ;
bridge_id bridge_id ;
u32 root_path_cost ;
unsigned long max_age ;
unsigned long hello_time ;
unsigned long forward_delay ;
unsigned long bridge_max_age ;
unsigned long ageing_time ;
unsigned long bridge_hello_time ;
unsigned long bridge_forward_delay ;
2006-03-21 09:59:21 +03:00
u8 group_addr [ ETH_ALEN ] ;
2005-04-17 02:20:36 +04:00
u16 root_port ;
2007-03-22 00:22:44 +03:00
enum {
BR_NO_STP , /* no spanning tree */
BR_KERNEL_STP , /* old STP in kernel */
BR_USER_STP , /* new RSTP in userspace */
} stp_enabled ;
2005-04-17 02:20:36 +04:00
unsigned char topology_change ;
unsigned char topology_change_detected ;
2010-02-27 22:41:45 +03:00
# ifdef CONFIG_BRIDGE_IGMP_SNOOPING
unsigned char multicast_router ;
u8 multicast_disabled : 1 ;
u32 hash_elasticity ;
u32 hash_max ;
u32 multicast_last_member_count ;
u32 multicast_startup_queries_sent ;
u32 multicast_startup_query_count ;
unsigned long multicast_last_member_interval ;
unsigned long multicast_membership_interval ;
unsigned long multicast_querier_interval ;
unsigned long multicast_query_interval ;
unsigned long multicast_query_response_interval ;
unsigned long multicast_startup_query_interval ;
spinlock_t multicast_lock ;
struct net_bridge_mdb_htable * mdb ;
struct hlist_head router_list ;
struct hlist_head mglist ;
struct timer_list multicast_router_timer ;
struct timer_list multicast_querier_timer ;
struct timer_list multicast_query_timer ;
# endif
2005-04-17 02:20:36 +04:00
struct timer_list hello_timer ;
struct timer_list tcn_timer ;
struct timer_list topology_change_timer ;
struct timer_list gc_timer ;
2007-12-17 22:54:39 +03:00
struct kobject * ifobj ;
2005-04-17 02:20:36 +04:00
} ;
2010-02-27 22:41:40 +03:00
struct br_input_skb_cb {
struct net_device * brdev ;
2010-03-16 00:51:18 +03:00
# ifdef CONFIG_BRIDGE_IGMP_SNOOPING
2010-02-27 22:41:45 +03:00
int igmp ;
int mrouters_only ;
2010-03-16 00:51:18 +03:00
# endif
2010-02-27 22:41:40 +03:00
} ;
# define BR_INPUT_SKB_CB(__skb) ((struct br_input_skb_cb *)(__skb)->cb)
2010-03-16 00:51:18 +03:00
# ifdef CONFIG_BRIDGE_IGMP_SNOOPING
# define BR_INPUT_SKB_CB_MROUTERS_ONLY(__skb) (BR_INPUT_SKB_CB(__skb)->mrouters_only)
# else
# define BR_INPUT_SKB_CB_MROUTERS_ONLY(__skb) (0)
# endif
2010-05-10 13:31:09 +04:00
# define br_printk(level, br, format, args...) \
printk ( level " %s: " format , ( br ) - > dev - > name , # # args )
# define br_err(__br, format, args...) \
br_printk ( KERN_ERR , __br , format , # # args )
# define br_warn(__br, format, args...) \
br_printk ( KERN_WARNING , __br , format , # # args )
# define br_notice(__br, format, args...) \
br_printk ( KERN_NOTICE , __br , format , # # args )
# define br_info(__br, format, args...) \
br_printk ( KERN_INFO , __br , format , # # args )
# define br_debug(br, format, args...) \
pr_debug ( " %s: " format , ( br ) - > dev - > name , # # args )
2005-04-17 02:20:36 +04:00
extern struct notifier_block br_device_notifier ;
2006-03-21 09:59:21 +03:00
extern const u8 br_group_address [ ETH_ALEN ] ;
2005-04-17 02:20:36 +04:00
/* called under bridge lock */
static inline int br_is_root_bridge ( const struct net_bridge * br )
{
return ! memcmp ( & br - > bridge_id , & br - > designated_root , 8 ) ;
}
/* br_device.c */
extern void br_dev_setup ( struct net_device * dev ) ;
2009-08-31 23:50:41 +04:00
extern netdev_tx_t br_dev_xmit ( struct sk_buff * skb ,
struct net_device * dev ) ;
2010-05-10 13:31:08 +04:00
# ifdef CONFIG_NET_POLL_CONTROLLER
extern void br_netpoll_cleanup ( struct net_device * dev ) ;
extern void br_netpoll_enable ( struct net_bridge * br ,
struct net_device * dev ) ;
extern void br_netpoll_disable ( struct net_bridge * br ,
struct net_device * dev ) ;
# else
# define br_netpoll_cleanup(br)
# define br_netpoll_enable(br, dev)
# define br_netpoll_disable(br, dev)
# endif
2005-04-17 02:20:36 +04:00
/* br_fdb.c */
2007-04-07 13:57:07 +04:00
extern int br_fdb_init ( void ) ;
2005-04-17 02:20:36 +04:00
extern void br_fdb_fini ( void ) ;
2007-04-09 23:57:54 +04:00
extern void br_fdb_flush ( struct net_bridge * br ) ;
2005-04-17 02:20:36 +04:00
extern void br_fdb_changeaddr ( struct net_bridge_port * p ,
const unsigned char * newaddr ) ;
extern void br_fdb_cleanup ( unsigned long arg ) ;
extern void br_fdb_delete_by_port ( struct net_bridge * br ,
2006-10-13 01:45:38 +04:00
const struct net_bridge_port * p , int do_all ) ;
2005-04-17 02:20:36 +04:00
extern struct net_bridge_fdb_entry * __br_fdb_get ( struct net_bridge * br ,
const unsigned char * addr ) ;
2009-06-05 09:35:28 +04:00
extern int br_fdb_test_addr ( struct net_device * dev , unsigned char * addr ) ;
2007-02-09 17:24:35 +03:00
extern int br_fdb_fillbuf ( struct net_bridge * br , void * buf ,
2005-04-17 02:20:36 +04:00
unsigned long count , unsigned long off ) ;
extern int br_fdb_insert ( struct net_bridge * br ,
struct net_bridge_port * source ,
const unsigned char * addr ) ;
extern void br_fdb_update ( struct net_bridge * br ,
struct net_bridge_port * source ,
const unsigned char * addr ) ;
/* br_forward.c */
extern void br_deliver ( const struct net_bridge_port * to ,
struct sk_buff * skb ) ;
extern int br_dev_queue_push_xmit ( struct sk_buff * skb ) ;
extern void br_forward ( const struct net_bridge_port * to ,
2010-03-16 10:26:22 +03:00
struct sk_buff * skb , struct sk_buff * skb0 ) ;
2005-04-17 02:20:36 +04:00
extern int br_forward_finish ( struct sk_buff * skb ) ;
2007-09-17 03:20:48 +04:00
extern void br_flood_deliver ( struct net_bridge * br , struct sk_buff * skb ) ;
2010-02-27 22:41:41 +03:00
extern void br_flood_forward ( struct net_bridge * br , struct sk_buff * skb ,
struct sk_buff * skb2 ) ;
2005-04-17 02:20:36 +04:00
/* br_if.c */
2007-02-22 12:10:18 +03:00
extern void br_port_carrier_check ( struct net_bridge_port * p ) ;
2008-09-09 03:19:58 +04:00
extern int br_add_bridge ( struct net * net , const char * name ) ;
extern int br_del_bridge ( struct net * net , const char * name ) ;
2008-09-09 03:20:18 +04:00
extern void br_net_exit ( struct net * net ) ;
2005-04-17 02:20:36 +04:00
extern int br_add_if ( struct net_bridge * br ,
struct net_device * dev ) ;
extern int br_del_if ( struct net_bridge * br ,
struct net_device * dev ) ;
extern int br_min_mtu ( const struct net_bridge * br ) ;
2005-05-30 01:15:17 +04:00
extern void br_features_recompute ( struct net_bridge * br ) ;
2005-04-17 02:20:36 +04:00
/* br_input.c */
extern int br_handle_frame_finish ( struct sk_buff * skb ) ;
2007-03-21 23:38:47 +03:00
extern struct sk_buff * br_handle_frame ( struct net_bridge_port * p ,
struct sk_buff * skb ) ;
2005-04-17 02:20:36 +04:00
/* br_ioctl.c */
extern int br_dev_ioctl ( struct net_device * dev , struct ifreq * rq , int cmd ) ;
2007-09-17 22:56:21 +04:00
extern int br_ioctl_deviceless_stub ( struct net * net , unsigned int cmd , void __user * arg ) ;
2005-04-17 02:20:36 +04:00
2010-02-27 22:41:45 +03:00
/* br_multicast.c */
# ifdef CONFIG_BRIDGE_IGMP_SNOOPING
extern int br_multicast_rcv ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb ) ;
extern struct net_bridge_mdb_entry * br_mdb_get ( struct net_bridge * br ,
struct sk_buff * skb ) ;
extern void br_multicast_add_port ( struct net_bridge_port * port ) ;
extern void br_multicast_del_port ( struct net_bridge_port * port ) ;
extern void br_multicast_enable_port ( struct net_bridge_port * port ) ;
extern void br_multicast_disable_port ( struct net_bridge_port * port ) ;
extern void br_multicast_init ( struct net_bridge * br ) ;
extern void br_multicast_open ( struct net_bridge * br ) ;
extern void br_multicast_stop ( struct net_bridge * br ) ;
2010-02-27 22:41:46 +03:00
extern void br_multicast_deliver ( struct net_bridge_mdb_entry * mdst ,
struct sk_buff * skb ) ;
extern void br_multicast_forward ( struct net_bridge_mdb_entry * mdst ,
struct sk_buff * skb , struct sk_buff * skb2 ) ;
2010-02-27 22:41:49 +03:00
extern int br_multicast_set_router ( struct net_bridge * br , unsigned long val ) ;
extern int br_multicast_set_port_router ( struct net_bridge_port * p ,
unsigned long val ) ;
2010-02-27 22:41:50 +03:00
extern int br_multicast_toggle ( struct net_bridge * br , unsigned long val ) ;
2010-02-27 22:41:51 +03:00
extern int br_multicast_set_hash_max ( struct net_bridge * br , unsigned long val ) ;
2010-03-01 12:53:04 +03:00
static inline bool br_multicast_is_router ( struct net_bridge * br )
{
return br - > multicast_router = = 2 | |
( br - > multicast_router = = 1 & &
timer_pending ( & br - > multicast_router_timer ) ) ;
}
2010-02-27 22:41:45 +03:00
# else
static inline int br_multicast_rcv ( struct net_bridge * br ,
struct net_bridge_port * port ,
struct sk_buff * skb )
{
return 0 ;
}
static inline struct net_bridge_mdb_entry * br_mdb_get ( struct net_bridge * br ,
struct sk_buff * skb )
{
return NULL ;
}
static inline void br_multicast_add_port ( struct net_bridge_port * port )
{
}
static inline void br_multicast_del_port ( struct net_bridge_port * port )
{
}
static inline void br_multicast_enable_port ( struct net_bridge_port * port )
{
}
static inline void br_multicast_disable_port ( struct net_bridge_port * port )
{
}
static inline void br_multicast_init ( struct net_bridge * br )
{
}
static inline void br_multicast_open ( struct net_bridge * br )
{
}
static inline void br_multicast_stop ( struct net_bridge * br )
{
}
2010-02-27 22:41:46 +03:00
static inline void br_multicast_deliver ( struct net_bridge_mdb_entry * mdst ,
struct sk_buff * skb )
{
}
static inline void br_multicast_forward ( struct net_bridge_mdb_entry * mdst ,
struct sk_buff * skb ,
struct sk_buff * skb2 )
{
}
2010-02-27 22:41:45 +03:00
static inline bool br_multicast_is_router ( struct net_bridge * br )
{
2010-03-01 12:53:04 +03:00
return 0 ;
2010-02-27 22:41:45 +03:00
}
2010-03-01 12:53:04 +03:00
# endif
2010-02-27 22:41:45 +03:00
2005-04-17 02:20:36 +04:00
/* br_netfilter.c */
2006-05-26 02:59:33 +04:00
# ifdef CONFIG_BRIDGE_NETFILTER
2005-04-17 02:20:36 +04:00
extern int br_netfilter_init ( void ) ;
extern void br_netfilter_fini ( void ) ;
2008-07-31 03:27:55 +04:00
extern void br_netfilter_rtable_init ( struct net_bridge * ) ;
2006-05-26 02:59:33 +04:00
# else
# define br_netfilter_init() (0)
# define br_netfilter_fini() do { } while(0)
2008-07-31 03:27:55 +04:00
# define br_netfilter_rtable_init(x)
2006-05-26 02:59:33 +04:00
# endif
2005-04-17 02:20:36 +04:00
/* br_stp.c */
extern void br_log_state ( const struct net_bridge_port * p ) ;
extern struct net_bridge_port * br_get_port ( struct net_bridge * br ,
2007-02-09 17:24:35 +03:00
u16 port_no ) ;
2005-04-17 02:20:36 +04:00
extern void br_init_port ( struct net_bridge_port * p ) ;
extern void br_become_designated_port ( struct net_bridge_port * p ) ;
/* br_stp_if.c */
extern void br_stp_enable_bridge ( struct net_bridge * br ) ;
extern void br_stp_disable_bridge ( struct net_bridge * br ) ;
2007-03-22 00:22:44 +03:00
extern void br_stp_set_enabled ( struct net_bridge * br , unsigned long val ) ;
2005-04-17 02:20:36 +04:00
extern void br_stp_enable_port ( struct net_bridge_port * p ) ;
extern void br_stp_disable_port ( struct net_bridge_port * p ) ;
extern void br_stp_recalculate_bridge_id ( struct net_bridge * br ) ;
2005-12-22 05:51:49 +03:00
extern void br_stp_change_bridge_id ( struct net_bridge * br , const unsigned char * a ) ;
2005-04-17 02:20:36 +04:00
extern void br_stp_set_bridge_priority ( struct net_bridge * br ,
u16 newprio ) ;
extern void br_stp_set_port_priority ( struct net_bridge_port * p ,
u8 newprio ) ;
extern void br_stp_set_path_cost ( struct net_bridge_port * p ,
u32 path_cost ) ;
extern ssize_t br_show_bridge_id ( char * buf , const struct bridge_id * id ) ;
/* br_stp_bpdu.c */
2008-07-06 08:25:56 +04:00
struct stp_proto ;
extern void br_stp_rcv ( const struct stp_proto * proto , struct sk_buff * skb ,
struct net_device * dev ) ;
2005-04-17 02:20:36 +04:00
/* br_stp_timer.c */
extern void br_stp_timer_init ( struct net_bridge * br ) ;
extern void br_stp_port_timer_init ( struct net_bridge_port * p ) ;
extern unsigned long br_timer_value ( const struct timer_list * timer ) ;
/* br.c */
2009-06-05 09:35:28 +04:00
# if defined(CONFIG_ATM_LANE) || defined(CONFIG_ATM_LANE_MODULE)
extern int ( * br_fdb_test_addr_hook ) ( struct net_device * dev , unsigned char * addr ) ;
# endif
2005-04-17 02:20:36 +04:00
2006-05-26 03:00:12 +04:00
/* br_netlink.c */
2007-03-22 21:59:03 +03:00
extern int br_netlink_init ( void ) ;
2006-05-26 03:00:12 +04:00
extern void br_netlink_fini ( void ) ;
extern void br_ifinfo_notify ( int event , struct net_bridge_port * port ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SYSFS
/* br_sysfs_if.c */
2010-01-19 04:58:23 +03:00
extern const struct sysfs_ops brport_sysfs_ops ;
2005-04-17 02:20:36 +04:00
extern int br_sysfs_addif ( struct net_bridge_port * p ) ;
/* br_sysfs_br.c */
extern int br_sysfs_addbr ( struct net_device * dev ) ;
extern void br_sysfs_delbr ( struct net_device * dev ) ;
# else
# define br_sysfs_addif(p) (0)
# define br_sysfs_addbr(dev) (0)
# define br_sysfs_delbr(dev) do { } while(0)
# endif /* CONFIG_SYSFS */
# endif