bridge: add per bridge device controls for invoking iptables
Support more fine grained control of bridge netfilter iptables invocation by adding seperate brnf_call_*tables parameters for each device using the sysfs interface. Packets are passed to layer 3 netfilter when either the global parameter or the per bridge parameter is enabled. Acked-by: Stephen Hemminger <shemminger@vyatta.com> Acked-by: David S. Miller <davem@davemloft.net> Signed-off-by: Patrick McHardy <kaber@trash.net>
This commit is contained in:
@@ -55,6 +55,9 @@ static int brnf_call_arptables __read_mostly = 1;
|
|||||||
static int brnf_filter_vlan_tagged __read_mostly = 0;
|
static int brnf_filter_vlan_tagged __read_mostly = 0;
|
||||||
static int brnf_filter_pppoe_tagged __read_mostly = 0;
|
static int brnf_filter_pppoe_tagged __read_mostly = 0;
|
||||||
#else
|
#else
|
||||||
|
#define brnf_call_iptables 1
|
||||||
|
#define brnf_call_ip6tables 1
|
||||||
|
#define brnf_call_arptables 1
|
||||||
#define brnf_filter_vlan_tagged 0
|
#define brnf_filter_vlan_tagged 0
|
||||||
#define brnf_filter_pppoe_tagged 0
|
#define brnf_filter_pppoe_tagged 0
|
||||||
#endif
|
#endif
|
||||||
@@ -543,25 +546,30 @@ static unsigned int br_nf_pre_routing(unsigned int hook, struct sk_buff *skb,
|
|||||||
const struct net_device *out,
|
const struct net_device *out,
|
||||||
int (*okfn)(struct sk_buff *))
|
int (*okfn)(struct sk_buff *))
|
||||||
{
|
{
|
||||||
|
struct net_bridge_port *p;
|
||||||
|
struct net_bridge *br;
|
||||||
struct iphdr *iph;
|
struct iphdr *iph;
|
||||||
__u32 len = nf_bridge_encap_header_len(skb);
|
__u32 len = nf_bridge_encap_header_len(skb);
|
||||||
|
|
||||||
if (unlikely(!pskb_may_pull(skb, len)))
|
if (unlikely(!pskb_may_pull(skb, len)))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
p = rcu_dereference(in->br_port);
|
||||||
|
if (p == NULL)
|
||||||
|
goto out;
|
||||||
|
br = p->br;
|
||||||
|
|
||||||
if (skb->protocol == htons(ETH_P_IPV6) || IS_VLAN_IPV6(skb) ||
|
if (skb->protocol == htons(ETH_P_IPV6) || IS_VLAN_IPV6(skb) ||
|
||||||
IS_PPPOE_IPV6(skb)) {
|
IS_PPPOE_IPV6(skb)) {
|
||||||
#ifdef CONFIG_SYSCTL
|
if (!brnf_call_ip6tables && !br->nf_call_ip6tables)
|
||||||
if (!brnf_call_ip6tables)
|
|
||||||
return NF_ACCEPT;
|
return NF_ACCEPT;
|
||||||
#endif
|
|
||||||
nf_bridge_pull_encap_header_rcsum(skb);
|
nf_bridge_pull_encap_header_rcsum(skb);
|
||||||
return br_nf_pre_routing_ipv6(hook, skb, in, out, okfn);
|
return br_nf_pre_routing_ipv6(hook, skb, in, out, okfn);
|
||||||
}
|
}
|
||||||
#ifdef CONFIG_SYSCTL
|
|
||||||
if (!brnf_call_iptables)
|
if (!brnf_call_iptables && !br->nf_call_iptables)
|
||||||
return NF_ACCEPT;
|
return NF_ACCEPT;
|
||||||
#endif
|
|
||||||
|
|
||||||
if (skb->protocol != htons(ETH_P_IP) && !IS_VLAN_IP(skb) &&
|
if (skb->protocol != htons(ETH_P_IP) && !IS_VLAN_IP(skb) &&
|
||||||
!IS_PPPOE_IP(skb))
|
!IS_PPPOE_IP(skb))
|
||||||
@@ -714,12 +722,17 @@ static unsigned int br_nf_forward_arp(unsigned int hook, struct sk_buff *skb,
|
|||||||
const struct net_device *out,
|
const struct net_device *out,
|
||||||
int (*okfn)(struct sk_buff *))
|
int (*okfn)(struct sk_buff *))
|
||||||
{
|
{
|
||||||
|
struct net_bridge_port *p;
|
||||||
|
struct net_bridge *br;
|
||||||
struct net_device **d = (struct net_device **)(skb->cb);
|
struct net_device **d = (struct net_device **)(skb->cb);
|
||||||
|
|
||||||
#ifdef CONFIG_SYSCTL
|
p = rcu_dereference(out->br_port);
|
||||||
if (!brnf_call_arptables)
|
if (p == NULL)
|
||||||
|
return NF_ACCEPT;
|
||||||
|
br = p->br;
|
||||||
|
|
||||||
|
if (!brnf_call_arptables && !br->nf_call_arptables)
|
||||||
return NF_ACCEPT;
|
return NF_ACCEPT;
|
||||||
#endif
|
|
||||||
|
|
||||||
if (skb->protocol != htons(ETH_P_ARP)) {
|
if (skb->protocol != htons(ETH_P_ARP)) {
|
||||||
if (!IS_VLAN_ARP(skb))
|
if (!IS_VLAN_ARP(skb))
|
||||||
|
@@ -164,6 +164,9 @@ struct net_bridge
|
|||||||
unsigned long feature_mask;
|
unsigned long feature_mask;
|
||||||
#ifdef CONFIG_BRIDGE_NETFILTER
|
#ifdef CONFIG_BRIDGE_NETFILTER
|
||||||
struct rtable fake_rtable;
|
struct rtable fake_rtable;
|
||||||
|
bool nf_call_iptables;
|
||||||
|
bool nf_call_ip6tables;
|
||||||
|
bool nf_call_arptables;
|
||||||
#endif
|
#endif
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
#define BR_SET_MAC_ADDR 0x00000001
|
#define BR_SET_MAC_ADDR 0x00000001
|
||||||
|
@@ -611,6 +611,73 @@ static DEVICE_ATTR(multicast_startup_query_interval, S_IRUGO | S_IWUSR,
|
|||||||
show_multicast_startup_query_interval,
|
show_multicast_startup_query_interval,
|
||||||
store_multicast_startup_query_interval);
|
store_multicast_startup_query_interval);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef CONFIG_BRIDGE_NETFILTER
|
||||||
|
static ssize_t show_nf_call_iptables(
|
||||||
|
struct device *d, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct net_bridge *br = to_bridge(d);
|
||||||
|
return sprintf(buf, "%u\n", br->nf_call_iptables);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_nf_call_iptables(struct net_bridge *br, unsigned long val)
|
||||||
|
{
|
||||||
|
br->nf_call_iptables = val ? true : false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t store_nf_call_iptables(
|
||||||
|
struct device *d, struct device_attribute *attr, const char *buf,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
return store_bridge_parm(d, buf, len, set_nf_call_iptables);
|
||||||
|
}
|
||||||
|
static DEVICE_ATTR(nf_call_iptables, S_IRUGO | S_IWUSR,
|
||||||
|
show_nf_call_iptables, store_nf_call_iptables);
|
||||||
|
|
||||||
|
static ssize_t show_nf_call_ip6tables(
|
||||||
|
struct device *d, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct net_bridge *br = to_bridge(d);
|
||||||
|
return sprintf(buf, "%u\n", br->nf_call_ip6tables);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val)
|
||||||
|
{
|
||||||
|
br->nf_call_ip6tables = val ? true : false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t store_nf_call_ip6tables(
|
||||||
|
struct device *d, struct device_attribute *attr, const char *buf,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
return store_bridge_parm(d, buf, len, set_nf_call_ip6tables);
|
||||||
|
}
|
||||||
|
static DEVICE_ATTR(nf_call_ip6tables, S_IRUGO | S_IWUSR,
|
||||||
|
show_nf_call_ip6tables, store_nf_call_ip6tables);
|
||||||
|
|
||||||
|
static ssize_t show_nf_call_arptables(
|
||||||
|
struct device *d, struct device_attribute *attr, char *buf)
|
||||||
|
{
|
||||||
|
struct net_bridge *br = to_bridge(d);
|
||||||
|
return sprintf(buf, "%u\n", br->nf_call_arptables);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int set_nf_call_arptables(struct net_bridge *br, unsigned long val)
|
||||||
|
{
|
||||||
|
br->nf_call_arptables = val ? true : false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t store_nf_call_arptables(
|
||||||
|
struct device *d, struct device_attribute *attr, const char *buf,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
return store_bridge_parm(d, buf, len, set_nf_call_arptables);
|
||||||
|
}
|
||||||
|
static DEVICE_ATTR(nf_call_arptables, S_IRUGO | S_IWUSR,
|
||||||
|
show_nf_call_arptables, store_nf_call_arptables);
|
||||||
|
#endif
|
||||||
|
|
||||||
static struct attribute *bridge_attrs[] = {
|
static struct attribute *bridge_attrs[] = {
|
||||||
&dev_attr_forward_delay.attr,
|
&dev_attr_forward_delay.attr,
|
||||||
@@ -644,6 +711,11 @@ static struct attribute *bridge_attrs[] = {
|
|||||||
&dev_attr_multicast_query_interval.attr,
|
&dev_attr_multicast_query_interval.attr,
|
||||||
&dev_attr_multicast_query_response_interval.attr,
|
&dev_attr_multicast_query_response_interval.attr,
|
||||||
&dev_attr_multicast_startup_query_interval.attr,
|
&dev_attr_multicast_startup_query_interval.attr,
|
||||||
|
#endif
|
||||||
|
#ifdef CONFIG_BRIDGE_NETFILTER
|
||||||
|
&dev_attr_nf_call_iptables.attr,
|
||||||
|
&dev_attr_nf_call_ip6tables.attr,
|
||||||
|
&dev_attr_nf_call_arptables.attr,
|
||||||
#endif
|
#endif
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
Reference in New Issue
Block a user