Add interface config support to enable or disable ARP proxy in the kernel per device.
Signed-off-by: Hrvoje Varga <hrvoje.va...@sartura.hr> --- bridge.c | 5 +++++ device.c | 9 +++++++++ device.h | 3 +++ system-linux.c | 18 ++++++++++++++++++ system.h | 1 + 5 files changed, 36 insertions(+) diff --git a/bridge.c b/bridge.c index 8e6c9a6..8f2bd30 100644 --- a/bridge.c +++ b/bridge.c @@ -176,6 +176,11 @@ bridge_enable_interface(struct bridge_state *bst) return ret; bst->active = true; + + if (bst->dev.settings.flags & DEV_OPT_PROXY_ARP) { + system_set_proxy_arp(&bst->dev, bst->dev.settings.proxy_arp ? "1" : "0"); + } + return 0; } diff --git a/device.c b/device.c index 82596e4..aa06d37 100644 --- a/device.c +++ b/device.c @@ -58,6 +58,7 @@ static const struct blobmsg_policy dev_attrs[__DEV_ATTR_MAX] = { [DEV_ATTR_MULTICAST] = { .name ="multicast", .type = BLOBMSG_TYPE_BOOL }, [DEV_ATTR_LEARNING] = { .name ="learning", .type = BLOBMSG_TYPE_BOOL }, [DEV_ATTR_UNICAST_FLOOD] = { .name ="unicast_flood", .type = BLOBMSG_TYPE_BOOL }, + [DEV_ATTR_PROXY_ARP] = { .name ="proxy_arp", .type = BLOBMSG_TYPE_BOOL }, }; const struct uci_blob_param_list device_attr_list = { @@ -225,6 +226,7 @@ device_merge_settings(struct device *dev, struct device_settings *n) n->multicast_fast_leave = s->multicast_fast_leave; n->learning = s->learning; n->unicast_flood = s->unicast_flood; + n->proxy_arp = s->flags & DEV_OPT_PROXY_ARP ? s->proxy_arp : os->proxy_arp; n->flags = s->flags | os->flags | os->valid_flags; } @@ -363,6 +365,11 @@ device_init_settings(struct device *dev, struct blob_attr **tb) s->flags |= DEV_OPT_UNICAST_FLOOD; } + if ((cur = tb[DEV_ATTR_PROXY_ARP])) { + s->proxy_arp = blobmsg_get_bool(cur); + s->flags |= DEV_OPT_PROXY_ARP; + } + device_set_disabled(dev, disabled); } @@ -1050,6 +1057,8 @@ device_dump_status(struct blob_buf *b, struct device *dev) blobmsg_add_u8(b, "learning", st.learning); if (st.flags & DEV_OPT_UNICAST_FLOOD) blobmsg_add_u8(b, "unicast_flood", st.unicast_flood); + if (st.flags & DEV_OPT_PROXY_ARP) + blobmsg_add_u8(b, "proxy_arp", st.proxy_arp); } s = blobmsg_open_table(b, "statistics"); diff --git a/device.h b/device.h index 2af93bb..699c7cc 100644 --- a/device.h +++ b/device.h @@ -50,6 +50,7 @@ enum { DEV_ATTR_LEARNING, DEV_ATTR_UNICAST_FLOOD, DEV_ATTR_NEIGHGCSTALETIME, + DEV_ATTR_PROXY_ARP, __DEV_ATTR_MAX, }; @@ -101,6 +102,7 @@ enum { DEV_OPT_UNICAST_FLOOD = (1 << 18), DEV_OPT_NEIGHGCSTALETIME = (1 << 19), DEV_OPT_MULTICAST_FAST_LEAVE = (1 << 20), + DEV_OPT_PROXY_ARP = (1 << 21), }; /* events broadcasted to all users of a device */ @@ -167,6 +169,7 @@ struct device_settings { bool multicast; bool learning; bool unicast_flood; + bool proxy_arp; }; /* diff --git a/system-linux.c b/system-linux.c index 6e4a194..d0c8655 100644 --- a/system-linux.c +++ b/system-linux.c @@ -271,6 +271,11 @@ static void system_set_dev_sysctl(const char *path, const char *device, const ch system_set_sysctl(dev_buf, val); } +void system_set_proxy_arp(struct device *dev, const char *val) +{ + system_set_dev_sysctl("/proc/sys/net/ipv4/conf/%s/proxy_arp", dev->ifname, val); +} + static void system_set_disable_ipv6(struct device *dev, const char *val) { system_set_dev_sysctl("/proc/sys/net/ipv6/conf/%s/disable_ipv6", dev->ifname, val); @@ -485,6 +490,12 @@ static int system_get_dadtransmits(struct device *dev, char *buf, const size_t b dev->ifname, buf, buf_sz); } +static int system_get_proxy_arp(struct device *dev, char *buf, const size_t buf_sz) +{ + return system_get_dev_sysctl("/proc/sys/net/ipv4/conf/%s/proxy_arp", + dev->ifname, buf, buf_sz); +} + // Evaluate netlink messages static int cb_rtnl_event(struct nl_msg *msg, void *arg) { @@ -1288,6 +1299,11 @@ system_if_get_settings(struct device *dev, struct device_settings *s) s->dadtransmits = strtoul(buf, NULL, 0); s->flags |= DEV_OPT_DADTRANSMITS; } + + if (!system_get_proxy_arp(dev, buf, sizeof(buf))) { + s->proxy_arp = strtoul(buf, NULL, 0); + s->flags |= DEV_OPT_PROXY_ARP; + } } static void @@ -1393,6 +1409,8 @@ system_if_apply_settings(struct device *dev, struct device_settings *s, unsigned !s->multicast ? IFF_MULTICAST : 0) < 0) s->flags &= ~DEV_OPT_MULTICAST; } + if (s->flags & DEV_OPT_PROXY_ARP & apply_mask) + system_set_proxy_arp(dev, s->proxy_arp ? "1" : "0"); system_if_apply_rps_xps(dev, s); } diff --git a/system.h b/system.h index d5cb4e3..a41a1b2 100644 --- a/system.h +++ b/system.h @@ -165,4 +165,5 @@ void system_fd_set_cloexec(int fd); int system_update_ipv6_mtu(struct device *device, int mtu); +void system_set_proxy_arp(struct device *dev, const char *val); #endif -- 2.10.0 _______________________________________________ openwrt-devel mailing list openwrt-devel@lists.openwrt.org https://lists.openwrt.org/cgi-bin/mailman/listinfo/openwrt-devel