12874c5fdSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later 21da177e4SLinus Torvalds /* 315401946SWang Sheng-Hui * Sysfs attributes of bridge 41da177e4SLinus Torvalds * Linux ethernet bridge 51da177e4SLinus Torvalds * 61da177e4SLinus Torvalds * Authors: 71da177e4SLinus Torvalds * Stephen Hemminger <shemminger@osdl.org> 81da177e4SLinus Torvalds */ 91da177e4SLinus Torvalds 104fc268d2SRandy Dunlap #include <linux/capability.h> 111da177e4SLinus Torvalds #include <linux/kernel.h> 121da177e4SLinus Torvalds #include <linux/netdevice.h> 13b3343a2aSJohn Fastabend #include <linux/etherdevice.h> 141da177e4SLinus Torvalds #include <linux/if_bridge.h> 151da177e4SLinus Torvalds #include <linux/rtnetlink.h> 161da177e4SLinus Torvalds #include <linux/spinlock.h> 171da177e4SLinus Torvalds #include <linux/times.h> 18174cd4b1SIngo Molnar #include <linux/sched/signal.h> 191da177e4SLinus Torvalds 201da177e4SLinus Torvalds #include "br_private.h" 211da177e4SLinus Torvalds 22*1e16f382SNikolay Aleksandrov /* IMPORTANT: new bridge options must be added with netlink support only 23*1e16f382SNikolay Aleksandrov * please do not add new sysfs entries 24*1e16f382SNikolay Aleksandrov */ 25*1e16f382SNikolay Aleksandrov 26524ad0a7SWang Chen #define to_bridge(cd) ((struct net_bridge *)netdev_priv(to_net_dev(cd))) 271da177e4SLinus Torvalds 281da177e4SLinus Torvalds /* 291da177e4SLinus Torvalds * Common code for storing bridge parameters. 301da177e4SLinus Torvalds */ 3143cb76d9SGreg Kroah-Hartman static ssize_t store_bridge_parm(struct device *d, 321da177e4SLinus Torvalds const char *buf, size_t len, 338d4698f7SStephen Hemminger int (*set)(struct net_bridge *, unsigned long)) 341da177e4SLinus Torvalds { 3543cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 361da177e4SLinus Torvalds char *endp; 371da177e4SLinus Torvalds unsigned long val; 388d4698f7SStephen Hemminger int err; 391da177e4SLinus Torvalds 40cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 411da177e4SLinus Torvalds return -EPERM; 421da177e4SLinus Torvalds 431da177e4SLinus Torvalds val = simple_strtoul(buf, &endp, 0); 441da177e4SLinus Torvalds if (endp == buf) 451da177e4SLinus Torvalds return -EINVAL; 461da177e4SLinus Torvalds 47047831a9SXin Long if (!rtnl_trylock()) 48047831a9SXin Long return restart_syscall(); 49047831a9SXin Long 508d4698f7SStephen Hemminger err = (*set)(br, val); 51047831a9SXin Long if (!err) 52047831a9SXin Long netdev_state_change(br->dev); 53047831a9SXin Long rtnl_unlock(); 54047831a9SXin Long 558d4698f7SStephen Hemminger return err ? err : len; 561da177e4SLinus Torvalds } 571da177e4SLinus Torvalds 581da177e4SLinus Torvalds 59fbf2671bSsfeldma@cumulusnetworks.com static ssize_t forward_delay_show(struct device *d, 6043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 611da177e4SLinus Torvalds { 6243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 631da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->forward_delay)); 641da177e4SLinus Torvalds } 651da177e4SLinus Torvalds 66fbf2671bSsfeldma@cumulusnetworks.com static ssize_t forward_delay_store(struct device *d, 6743cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 6843cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 691da177e4SLinus Torvalds { 7014f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_forward_delay); 711da177e4SLinus Torvalds } 72fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(forward_delay); 731da177e4SLinus Torvalds 74fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_time_show(struct device *d, struct device_attribute *attr, 7543cb76d9SGreg Kroah-Hartman char *buf) 761da177e4SLinus Torvalds { 771da177e4SLinus Torvalds return sprintf(buf, "%lu\n", 7843cb76d9SGreg Kroah-Hartman jiffies_to_clock_t(to_bridge(d)->hello_time)); 791da177e4SLinus Torvalds } 801da177e4SLinus Torvalds 81fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_time_store(struct device *d, 8243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, const char *buf, 831da177e4SLinus Torvalds size_t len) 841da177e4SLinus Torvalds { 8514f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_hello_time); 861da177e4SLinus Torvalds } 87fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hello_time); 881da177e4SLinus Torvalds 89fbf2671bSsfeldma@cumulusnetworks.com static ssize_t max_age_show(struct device *d, struct device_attribute *attr, 9043cb76d9SGreg Kroah-Hartman char *buf) 911da177e4SLinus Torvalds { 921da177e4SLinus Torvalds return sprintf(buf, "%lu\n", 9343cb76d9SGreg Kroah-Hartman jiffies_to_clock_t(to_bridge(d)->max_age)); 941da177e4SLinus Torvalds } 951da177e4SLinus Torvalds 96fbf2671bSsfeldma@cumulusnetworks.com static ssize_t max_age_store(struct device *d, struct device_attribute *attr, 9743cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 981da177e4SLinus Torvalds { 9914f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_max_age); 1001da177e4SLinus Torvalds } 101fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(max_age); 1021da177e4SLinus Torvalds 103fbf2671bSsfeldma@cumulusnetworks.com static ssize_t ageing_time_show(struct device *d, 10443cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 1051da177e4SLinus Torvalds { 10643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1071da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->ageing_time)); 1081da177e4SLinus Torvalds } 1091da177e4SLinus Torvalds 1108d4698f7SStephen Hemminger static int set_ageing_time(struct net_bridge *br, unsigned long val) 1111da177e4SLinus Torvalds { 112047831a9SXin Long return br_set_ageing_time(br, val); 1131da177e4SLinus Torvalds } 1141da177e4SLinus Torvalds 115fbf2671bSsfeldma@cumulusnetworks.com static ssize_t ageing_time_store(struct device *d, 11643cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 11743cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 1181da177e4SLinus Torvalds { 11943cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_ageing_time); 1201da177e4SLinus Torvalds } 121fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(ageing_time); 12243cb76d9SGreg Kroah-Hartman 123fbf2671bSsfeldma@cumulusnetworks.com static ssize_t stp_state_show(struct device *d, 12443cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 1251da177e4SLinus Torvalds { 12643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1271da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->stp_enabled); 1281da177e4SLinus Torvalds } 1291da177e4SLinus Torvalds 1301da177e4SLinus Torvalds 1314436156bSXin Long static int set_stp_state(struct net_bridge *br, unsigned long val) 1321da177e4SLinus Torvalds { 133419dba8aSHoratiu Vultur return br_stp_set_enabled(br, val, NULL); 1344436156bSXin Long } 1354436156bSXin Long 1364436156bSXin Long static ssize_t stp_state_store(struct device *d, 1374436156bSXin Long struct device_attribute *attr, const char *buf, 1384436156bSXin Long size_t len) 1394436156bSXin Long { 1404436156bSXin Long return store_bridge_parm(d, buf, len, set_stp_state); 1411da177e4SLinus Torvalds } 142fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(stp_state); 1431da177e4SLinus Torvalds 144fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_show(struct device *d, 145fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 146fbf2671bSsfeldma@cumulusnetworks.com char *buf) 147515853ccSstephen hemminger { 148515853ccSstephen hemminger struct net_bridge *br = to_bridge(d); 149515853ccSstephen hemminger return sprintf(buf, "%#x\n", br->group_fwd_mask); 150515853ccSstephen hemminger } 151515853ccSstephen hemminger 152347db6b4SXin Long static int set_group_fwd_mask(struct net_bridge *br, unsigned long val) 153347db6b4SXin Long { 154347db6b4SXin Long if (val & BR_GROUPFWD_RESTRICTED) 155347db6b4SXin Long return -EINVAL; 156347db6b4SXin Long 157347db6b4SXin Long br->group_fwd_mask = val; 158347db6b4SXin Long 159347db6b4SXin Long return 0; 160347db6b4SXin Long } 161515853ccSstephen hemminger 162fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_store(struct device *d, 163fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 164fbf2671bSsfeldma@cumulusnetworks.com const char *buf, 165515853ccSstephen hemminger size_t len) 166515853ccSstephen hemminger { 167347db6b4SXin Long return store_bridge_parm(d, buf, len, set_group_fwd_mask); 168515853ccSstephen hemminger } 169fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_fwd_mask); 170515853ccSstephen hemminger 171fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_show(struct device *d, struct device_attribute *attr, 17243cb76d9SGreg Kroah-Hartman char *buf) 1731da177e4SLinus Torvalds { 17443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1751da177e4SLinus Torvalds return sprintf(buf, "%d\n", 1761da177e4SLinus Torvalds (br->bridge_id.prio[0] << 8) | br->bridge_id.prio[1]); 1771da177e4SLinus Torvalds } 1781da177e4SLinus Torvalds 1798d4698f7SStephen Hemminger static int set_priority(struct net_bridge *br, unsigned long val) 1801da177e4SLinus Torvalds { 1811da177e4SLinus Torvalds br_stp_set_bridge_priority(br, (u16) val); 1828d4698f7SStephen Hemminger return 0; 1831da177e4SLinus Torvalds } 1841da177e4SLinus Torvalds 185fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_store(struct device *d, struct device_attribute *attr, 1861da177e4SLinus Torvalds const char *buf, size_t len) 1871da177e4SLinus Torvalds { 18843cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_priority); 1891da177e4SLinus Torvalds } 190fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(priority); 1911da177e4SLinus Torvalds 192fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_id_show(struct device *d, struct device_attribute *attr, 19343cb76d9SGreg Kroah-Hartman char *buf) 1941da177e4SLinus Torvalds { 19543cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->designated_root); 1961da177e4SLinus Torvalds } 197fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_id); 1981da177e4SLinus Torvalds 199fbf2671bSsfeldma@cumulusnetworks.com static ssize_t bridge_id_show(struct device *d, struct device_attribute *attr, 20043cb76d9SGreg Kroah-Hartman char *buf) 2011da177e4SLinus Torvalds { 20243cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->bridge_id); 2031da177e4SLinus Torvalds } 204fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(bridge_id); 2051da177e4SLinus Torvalds 206fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_port_show(struct device *d, struct device_attribute *attr, 20743cb76d9SGreg Kroah-Hartman char *buf) 2081da177e4SLinus Torvalds { 20943cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_port); 2101da177e4SLinus Torvalds } 211fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_port); 2121da177e4SLinus Torvalds 213fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_path_cost_show(struct device *d, 21443cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2151da177e4SLinus Torvalds { 21643cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_path_cost); 2171da177e4SLinus Torvalds } 218fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_path_cost); 2191da177e4SLinus Torvalds 220fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_show(struct device *d, 22143cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2221da177e4SLinus Torvalds { 22343cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->topology_change); 2241da177e4SLinus Torvalds } 225fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change); 2261da177e4SLinus Torvalds 227fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_detected_show(struct device *d, 22843cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 22943cb76d9SGreg Kroah-Hartman char *buf) 2301da177e4SLinus Torvalds { 23143cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2321da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->topology_change_detected); 2331da177e4SLinus Torvalds } 234fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_detected); 2351da177e4SLinus Torvalds 236fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_timer_show(struct device *d, 23743cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2381da177e4SLinus Torvalds { 23943cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2401da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->hello_timer)); 2411da177e4SLinus Torvalds } 242fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(hello_timer); 2431da177e4SLinus Torvalds 244fbf2671bSsfeldma@cumulusnetworks.com static ssize_t tcn_timer_show(struct device *d, struct device_attribute *attr, 24543cb76d9SGreg Kroah-Hartman char *buf) 2461da177e4SLinus Torvalds { 24743cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2481da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->tcn_timer)); 2491da177e4SLinus Torvalds } 250fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(tcn_timer); 2511da177e4SLinus Torvalds 252fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_timer_show(struct device *d, 25343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 25443cb76d9SGreg Kroah-Hartman char *buf) 2551da177e4SLinus Torvalds { 25643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2571da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->topology_change_timer)); 2581da177e4SLinus Torvalds } 259fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_timer); 2601da177e4SLinus Torvalds 261fbf2671bSsfeldma@cumulusnetworks.com static ssize_t gc_timer_show(struct device *d, struct device_attribute *attr, 26243cb76d9SGreg Kroah-Hartman char *buf) 2631da177e4SLinus Torvalds { 26443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 265f7cdee8aSNikolay Aleksandrov return sprintf(buf, "%ld\n", br_timer_value(&br->gc_work.timer)); 2661da177e4SLinus Torvalds } 267fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(gc_timer); 2681da177e4SLinus Torvalds 269fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_show(struct device *d, 27043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 271fda93d92SStephen Hemminger { 27243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 273223b229bSAndy Shevchenko return sprintf(buf, "%pM\n", br->group_addr); 274fda93d92SStephen Hemminger } 275fda93d92SStephen Hemminger 276fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_store(struct device *d, 27743cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 27843cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 279fda93d92SStephen Hemminger { 28043cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2814197f24bSBen Hutchings u8 new_addr[6]; 282fda93d92SStephen Hemminger 283cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 284fda93d92SStephen Hemminger return -EPERM; 285fda93d92SStephen Hemminger 286223b229bSAndy Shevchenko if (!mac_pton(buf, new_addr)) 287fda93d92SStephen Hemminger return -EINVAL; 288fda93d92SStephen Hemminger 28946acc460SBen Hutchings if (!is_link_local_ether_addr(new_addr)) 290fda93d92SStephen Hemminger return -EINVAL; 291fda93d92SStephen Hemminger 292f64f9e71SJoe Perches if (new_addr[5] == 1 || /* 802.3x Pause address */ 293f64f9e71SJoe Perches new_addr[5] == 2 || /* 802.3ad Slow protocols */ 294f64f9e71SJoe Perches new_addr[5] == 3) /* 802.1X PAE address */ 295fda93d92SStephen Hemminger return -EINVAL; 296fda93d92SStephen Hemminger 297204177f3SToshiaki Makita if (!rtnl_trylock()) 298204177f3SToshiaki Makita return restart_syscall(); 299204177f3SToshiaki Makita 300fda93d92SStephen Hemminger spin_lock_bh(&br->lock); 301223b229bSAndy Shevchenko ether_addr_copy(br->group_addr, new_addr); 302fda93d92SStephen Hemminger spin_unlock_bh(&br->lock); 303204177f3SToshiaki Makita 304be3664a0SNikolay Aleksandrov br_opt_toggle(br, BROPT_GROUP_ADDR_SET, true); 305204177f3SToshiaki Makita br_recalculate_fwd_mask(br); 306047831a9SXin Long netdev_state_change(br->dev); 307204177f3SToshiaki Makita 308204177f3SToshiaki Makita rtnl_unlock(); 309204177f3SToshiaki Makita 310fda93d92SStephen Hemminger return len; 311fda93d92SStephen Hemminger } 312fda93d92SStephen Hemminger 313fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_addr); 314fda93d92SStephen Hemminger 31514f31bb3SXin Long static int set_flush(struct net_bridge *br, unsigned long val) 31614f31bb3SXin Long { 31714f31bb3SXin Long br_fdb_flush(br); 31814f31bb3SXin Long return 0; 31914f31bb3SXin Long } 32014f31bb3SXin Long 321fbf2671bSsfeldma@cumulusnetworks.com static ssize_t flush_store(struct device *d, 3229cf63747SStephen Hemminger struct device_attribute *attr, 3239cf63747SStephen Hemminger const char *buf, size_t len) 3249cf63747SStephen Hemminger { 32514f31bb3SXin Long return store_bridge_parm(d, buf, len, set_flush); 3269cf63747SStephen Hemminger } 327fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_WO(flush); 328fda93d92SStephen Hemminger 32970e4272bSNikolay Aleksandrov static ssize_t no_linklocal_learn_show(struct device *d, 33070e4272bSNikolay Aleksandrov struct device_attribute *attr, 33170e4272bSNikolay Aleksandrov char *buf) 33270e4272bSNikolay Aleksandrov { 33370e4272bSNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 33470e4272bSNikolay Aleksandrov return sprintf(buf, "%d\n", br_boolopt_get(br, BR_BOOLOPT_NO_LL_LEARN)); 33570e4272bSNikolay Aleksandrov } 33670e4272bSNikolay Aleksandrov 33770e4272bSNikolay Aleksandrov static int set_no_linklocal_learn(struct net_bridge *br, unsigned long val) 33870e4272bSNikolay Aleksandrov { 33970e4272bSNikolay Aleksandrov return br_boolopt_toggle(br, BR_BOOLOPT_NO_LL_LEARN, !!val, NULL); 34070e4272bSNikolay Aleksandrov } 34170e4272bSNikolay Aleksandrov 34270e4272bSNikolay Aleksandrov static ssize_t no_linklocal_learn_store(struct device *d, 34370e4272bSNikolay Aleksandrov struct device_attribute *attr, 34470e4272bSNikolay Aleksandrov const char *buf, size_t len) 34570e4272bSNikolay Aleksandrov { 34670e4272bSNikolay Aleksandrov return store_bridge_parm(d, buf, len, set_no_linklocal_learn); 34770e4272bSNikolay Aleksandrov } 34870e4272bSNikolay Aleksandrov static DEVICE_ATTR_RW(no_linklocal_learn); 34970e4272bSNikolay Aleksandrov 3500909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 351fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_show(struct device *d, 3520909e117SHerbert Xu struct device_attribute *attr, char *buf) 3530909e117SHerbert Xu { 3540909e117SHerbert Xu struct net_bridge *br = to_bridge(d); 3550909e117SHerbert Xu return sprintf(buf, "%d\n", br->multicast_router); 3560909e117SHerbert Xu } 3570909e117SHerbert Xu 358fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_store(struct device *d, 3590909e117SHerbert Xu struct device_attribute *attr, 3600909e117SHerbert Xu const char *buf, size_t len) 3610909e117SHerbert Xu { 3620909e117SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_router); 3630909e117SHerbert Xu } 364fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_router); 365561f1103SHerbert Xu 366fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_show(struct device *d, 367561f1103SHerbert Xu struct device_attribute *attr, 368561f1103SHerbert Xu char *buf) 369561f1103SHerbert Xu { 370561f1103SHerbert Xu struct net_bridge *br = to_bridge(d); 37113cefad2SNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_MULTICAST_ENABLED)); 372561f1103SHerbert Xu } 373561f1103SHerbert Xu 374fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_store(struct device *d, 375561f1103SHerbert Xu struct device_attribute *attr, 376561f1103SHerbert Xu const char *buf, size_t len) 377561f1103SHerbert Xu { 378561f1103SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_toggle); 379561f1103SHerbert Xu } 380fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_snooping); 381b195167fSHerbert Xu 382fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_use_ifaddr_show(struct device *d, 3831c8ad5bfSCong Wang struct device_attribute *attr, 3841c8ad5bfSCong Wang char *buf) 3851c8ad5bfSCong Wang { 3861c8ad5bfSCong Wang struct net_bridge *br = to_bridge(d); 387675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", 388675779adSNikolay Aleksandrov br_opt_get(br, BROPT_MULTICAST_QUERY_USE_IFADDR)); 3891c8ad5bfSCong Wang } 3901c8ad5bfSCong Wang 3911c8ad5bfSCong Wang static int set_query_use_ifaddr(struct net_bridge *br, unsigned long val) 3921c8ad5bfSCong Wang { 393675779adSNikolay Aleksandrov br_opt_toggle(br, BROPT_MULTICAST_QUERY_USE_IFADDR, !!val); 3941c8ad5bfSCong Wang return 0; 3951c8ad5bfSCong Wang } 3961c8ad5bfSCong Wang 3971c8ad5bfSCong Wang static ssize_t 398fbf2671bSsfeldma@cumulusnetworks.com multicast_query_use_ifaddr_store(struct device *d, 3991c8ad5bfSCong Wang struct device_attribute *attr, 4001c8ad5bfSCong Wang const char *buf, size_t len) 4011c8ad5bfSCong Wang { 4021c8ad5bfSCong Wang return store_bridge_parm(d, buf, len, set_query_use_ifaddr); 4031c8ad5bfSCong Wang } 404fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_use_ifaddr); 4051c8ad5bfSCong Wang 406fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_show(struct device *d, 407c5c23260SHerbert Xu struct device_attribute *attr, 408c5c23260SHerbert Xu char *buf) 409c5c23260SHerbert Xu { 410c5c23260SHerbert Xu struct net_bridge *br = to_bridge(d); 411675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_MULTICAST_QUERIER)); 412c5c23260SHerbert Xu } 413c5c23260SHerbert Xu 414fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_store(struct device *d, 415c5c23260SHerbert Xu struct device_attribute *attr, 416c5c23260SHerbert Xu const char *buf, size_t len) 417c5c23260SHerbert Xu { 418c5c23260SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_querier); 419c5c23260SHerbert Xu } 420fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier); 421c5c23260SHerbert Xu 422fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_show(struct device *d, 423b195167fSHerbert Xu struct device_attribute *attr, char *buf) 424b195167fSHerbert Xu { 425cf332bcaSNikolay Aleksandrov return sprintf(buf, "%u\n", RHT_ELASTICITY); 426b195167fSHerbert Xu } 427b195167fSHerbert Xu 428b195167fSHerbert Xu static int set_elasticity(struct net_bridge *br, unsigned long val) 429b195167fSHerbert Xu { 430cf332bcaSNikolay Aleksandrov br_warn(br, "the hash_elasticity option has been deprecated and is always %u\n", 431cf332bcaSNikolay Aleksandrov RHT_ELASTICITY); 432b195167fSHerbert Xu return 0; 433b195167fSHerbert Xu } 434b195167fSHerbert Xu 435fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_store(struct device *d, 436b195167fSHerbert Xu struct device_attribute *attr, 437b195167fSHerbert Xu const char *buf, size_t len) 438b195167fSHerbert Xu { 439b195167fSHerbert Xu return store_bridge_parm(d, buf, len, set_elasticity); 440b195167fSHerbert Xu } 441fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_elasticity); 442b195167fSHerbert Xu 443fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_show(struct device *d, struct device_attribute *attr, 444b195167fSHerbert Xu char *buf) 445b195167fSHerbert Xu { 446b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 447b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_max); 448b195167fSHerbert Xu } 449b195167fSHerbert Xu 45019e3a9c9SNikolay Aleksandrov static int set_hash_max(struct net_bridge *br, unsigned long val) 45119e3a9c9SNikolay Aleksandrov { 45219e3a9c9SNikolay Aleksandrov br->hash_max = val; 45319e3a9c9SNikolay Aleksandrov return 0; 45419e3a9c9SNikolay Aleksandrov } 45519e3a9c9SNikolay Aleksandrov 456fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_store(struct device *d, struct device_attribute *attr, 457b195167fSHerbert Xu const char *buf, size_t len) 458b195167fSHerbert Xu { 45919e3a9c9SNikolay Aleksandrov return store_bridge_parm(d, buf, len, set_hash_max); 460b195167fSHerbert Xu } 461fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_max); 462d902eee4SHerbert Xu 4635e923585SNikolay Aleksandrov static ssize_t multicast_igmp_version_show(struct device *d, 4645e923585SNikolay Aleksandrov struct device_attribute *attr, 4655e923585SNikolay Aleksandrov char *buf) 4665e923585SNikolay Aleksandrov { 4675e923585SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 4685e923585SNikolay Aleksandrov 4695e923585SNikolay Aleksandrov return sprintf(buf, "%u\n", br->multicast_igmp_version); 4705e923585SNikolay Aleksandrov } 4715e923585SNikolay Aleksandrov 4725e923585SNikolay Aleksandrov static ssize_t multicast_igmp_version_store(struct device *d, 4735e923585SNikolay Aleksandrov struct device_attribute *attr, 4745e923585SNikolay Aleksandrov const char *buf, size_t len) 4755e923585SNikolay Aleksandrov { 4765e923585SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_multicast_set_igmp_version); 4775e923585SNikolay Aleksandrov } 4785e923585SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_igmp_version); 4795e923585SNikolay Aleksandrov 480fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_show(struct device *d, 481d902eee4SHerbert Xu struct device_attribute *attr, 482d902eee4SHerbert Xu char *buf) 483d902eee4SHerbert Xu { 484d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 485d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_last_member_count); 486d902eee4SHerbert Xu } 487d902eee4SHerbert Xu 488d902eee4SHerbert Xu static int set_last_member_count(struct net_bridge *br, unsigned long val) 489d902eee4SHerbert Xu { 490d902eee4SHerbert Xu br->multicast_last_member_count = val; 491d902eee4SHerbert Xu return 0; 492d902eee4SHerbert Xu } 493d902eee4SHerbert Xu 494fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_store(struct device *d, 495d902eee4SHerbert Xu struct device_attribute *attr, 496d902eee4SHerbert Xu const char *buf, size_t len) 497d902eee4SHerbert Xu { 498d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_count); 499d902eee4SHerbert Xu } 500fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_count); 501d902eee4SHerbert Xu 502fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_show( 503d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 504d902eee4SHerbert Xu { 505d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 506d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_startup_query_count); 507d902eee4SHerbert Xu } 508d902eee4SHerbert Xu 509d902eee4SHerbert Xu static int set_startup_query_count(struct net_bridge *br, unsigned long val) 510d902eee4SHerbert Xu { 511d902eee4SHerbert Xu br->multicast_startup_query_count = val; 512d902eee4SHerbert Xu return 0; 513d902eee4SHerbert Xu } 514d902eee4SHerbert Xu 515fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_store( 516d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 517d902eee4SHerbert Xu size_t len) 518d902eee4SHerbert Xu { 519d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_count); 520d902eee4SHerbert Xu } 521fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_count); 522d902eee4SHerbert Xu 523fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_show( 524d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 525d902eee4SHerbert Xu { 526d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 527d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 528d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_last_member_interval)); 529d902eee4SHerbert Xu } 530d902eee4SHerbert Xu 531d902eee4SHerbert Xu static int set_last_member_interval(struct net_bridge *br, unsigned long val) 532d902eee4SHerbert Xu { 533d902eee4SHerbert Xu br->multicast_last_member_interval = clock_t_to_jiffies(val); 534d902eee4SHerbert Xu return 0; 535d902eee4SHerbert Xu } 536d902eee4SHerbert Xu 537fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_store( 538d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 539d902eee4SHerbert Xu size_t len) 540d902eee4SHerbert Xu { 541d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_interval); 542d902eee4SHerbert Xu } 543fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_interval); 544d902eee4SHerbert Xu 545fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_show( 546d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 547d902eee4SHerbert Xu { 548d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 549d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 550d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_membership_interval)); 551d902eee4SHerbert Xu } 552d902eee4SHerbert Xu 553d902eee4SHerbert Xu static int set_membership_interval(struct net_bridge *br, unsigned long val) 554d902eee4SHerbert Xu { 555d902eee4SHerbert Xu br->multicast_membership_interval = clock_t_to_jiffies(val); 556d902eee4SHerbert Xu return 0; 557d902eee4SHerbert Xu } 558d902eee4SHerbert Xu 559fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_store( 560d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 561d902eee4SHerbert Xu size_t len) 562d902eee4SHerbert Xu { 563d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_membership_interval); 564d902eee4SHerbert Xu } 565fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_membership_interval); 566d902eee4SHerbert Xu 567fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_show(struct device *d, 568d902eee4SHerbert Xu struct device_attribute *attr, 569d902eee4SHerbert Xu char *buf) 570d902eee4SHerbert Xu { 571d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 572d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 573d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_querier_interval)); 574d902eee4SHerbert Xu } 575d902eee4SHerbert Xu 576d902eee4SHerbert Xu static int set_querier_interval(struct net_bridge *br, unsigned long val) 577d902eee4SHerbert Xu { 578d902eee4SHerbert Xu br->multicast_querier_interval = clock_t_to_jiffies(val); 579d902eee4SHerbert Xu return 0; 580d902eee4SHerbert Xu } 581d902eee4SHerbert Xu 582fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_store(struct device *d, 583d902eee4SHerbert Xu struct device_attribute *attr, 584d902eee4SHerbert Xu const char *buf, size_t len) 585d902eee4SHerbert Xu { 586d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_querier_interval); 587d902eee4SHerbert Xu } 588fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier_interval); 589d902eee4SHerbert Xu 590fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_show(struct device *d, 591d902eee4SHerbert Xu struct device_attribute *attr, 592d902eee4SHerbert Xu char *buf) 593d902eee4SHerbert Xu { 594d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 595d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 596d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_interval)); 597d902eee4SHerbert Xu } 598d902eee4SHerbert Xu 599d902eee4SHerbert Xu static int set_query_interval(struct net_bridge *br, unsigned long val) 600d902eee4SHerbert Xu { 601d902eee4SHerbert Xu br->multicast_query_interval = clock_t_to_jiffies(val); 602d902eee4SHerbert Xu return 0; 603d902eee4SHerbert Xu } 604d902eee4SHerbert Xu 605fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_store(struct device *d, 606d902eee4SHerbert Xu struct device_attribute *attr, 607d902eee4SHerbert Xu const char *buf, size_t len) 608d902eee4SHerbert Xu { 609d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_interval); 610d902eee4SHerbert Xu } 611fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_interval); 612d902eee4SHerbert Xu 613fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_show( 614d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 615d902eee4SHerbert Xu { 616d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 617d902eee4SHerbert Xu return sprintf( 618d902eee4SHerbert Xu buf, "%lu\n", 619d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_response_interval)); 620d902eee4SHerbert Xu } 621d902eee4SHerbert Xu 622d902eee4SHerbert Xu static int set_query_response_interval(struct net_bridge *br, unsigned long val) 623d902eee4SHerbert Xu { 624d902eee4SHerbert Xu br->multicast_query_response_interval = clock_t_to_jiffies(val); 625d902eee4SHerbert Xu return 0; 626d902eee4SHerbert Xu } 627d902eee4SHerbert Xu 628fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_store( 629d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 630d902eee4SHerbert Xu size_t len) 631d902eee4SHerbert Xu { 632d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_response_interval); 633d902eee4SHerbert Xu } 634fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_response_interval); 635d902eee4SHerbert Xu 636fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_show( 637d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 638d902eee4SHerbert Xu { 639d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 640d902eee4SHerbert Xu return sprintf( 641d902eee4SHerbert Xu buf, "%lu\n", 642d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_startup_query_interval)); 643d902eee4SHerbert Xu } 644d902eee4SHerbert Xu 645d902eee4SHerbert Xu static int set_startup_query_interval(struct net_bridge *br, unsigned long val) 646d902eee4SHerbert Xu { 647d902eee4SHerbert Xu br->multicast_startup_query_interval = clock_t_to_jiffies(val); 648d902eee4SHerbert Xu return 0; 649d902eee4SHerbert Xu } 650d902eee4SHerbert Xu 651fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_store( 652d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 653d902eee4SHerbert Xu size_t len) 654d902eee4SHerbert Xu { 655d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_interval); 656d902eee4SHerbert Xu } 657fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_interval); 6581080ab95SNikolay Aleksandrov 6591080ab95SNikolay Aleksandrov static ssize_t multicast_stats_enabled_show(struct device *d, 6601080ab95SNikolay Aleksandrov struct device_attribute *attr, 6611080ab95SNikolay Aleksandrov char *buf) 6621080ab95SNikolay Aleksandrov { 6631080ab95SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 6641080ab95SNikolay Aleksandrov 665675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", 666675779adSNikolay Aleksandrov br_opt_get(br, BROPT_MULTICAST_STATS_ENABLED)); 6671080ab95SNikolay Aleksandrov } 6681080ab95SNikolay Aleksandrov 6691080ab95SNikolay Aleksandrov static int set_stats_enabled(struct net_bridge *br, unsigned long val) 6701080ab95SNikolay Aleksandrov { 671675779adSNikolay Aleksandrov br_opt_toggle(br, BROPT_MULTICAST_STATS_ENABLED, !!val); 6721080ab95SNikolay Aleksandrov return 0; 6731080ab95SNikolay Aleksandrov } 6741080ab95SNikolay Aleksandrov 6751080ab95SNikolay Aleksandrov static ssize_t multicast_stats_enabled_store(struct device *d, 6761080ab95SNikolay Aleksandrov struct device_attribute *attr, 6771080ab95SNikolay Aleksandrov const char *buf, 6781080ab95SNikolay Aleksandrov size_t len) 6791080ab95SNikolay Aleksandrov { 6801080ab95SNikolay Aleksandrov return store_bridge_parm(d, buf, len, set_stats_enabled); 6811080ab95SNikolay Aleksandrov } 6821080ab95SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_stats_enabled); 683aa2ae3e7SNikolay Aleksandrov 684aa2ae3e7SNikolay Aleksandrov #if IS_ENABLED(CONFIG_IPV6) 685aa2ae3e7SNikolay Aleksandrov static ssize_t multicast_mld_version_show(struct device *d, 686aa2ae3e7SNikolay Aleksandrov struct device_attribute *attr, 687aa2ae3e7SNikolay Aleksandrov char *buf) 688aa2ae3e7SNikolay Aleksandrov { 689aa2ae3e7SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 690aa2ae3e7SNikolay Aleksandrov 691aa2ae3e7SNikolay Aleksandrov return sprintf(buf, "%u\n", br->multicast_mld_version); 692aa2ae3e7SNikolay Aleksandrov } 693aa2ae3e7SNikolay Aleksandrov 694aa2ae3e7SNikolay Aleksandrov static ssize_t multicast_mld_version_store(struct device *d, 695aa2ae3e7SNikolay Aleksandrov struct device_attribute *attr, 696aa2ae3e7SNikolay Aleksandrov const char *buf, size_t len) 697aa2ae3e7SNikolay Aleksandrov { 698aa2ae3e7SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_multicast_set_mld_version); 699aa2ae3e7SNikolay Aleksandrov } 700aa2ae3e7SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_mld_version); 701aa2ae3e7SNikolay Aleksandrov #endif 7020909e117SHerbert Xu #endif 70334666d46SPablo Neira Ayuso #if IS_ENABLED(CONFIG_BRIDGE_NETFILTER) 704fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_show( 7054df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 7064df53d8bSPatrick McHardy { 7074df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 7088df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_IPTABLES)); 7094df53d8bSPatrick McHardy } 7104df53d8bSPatrick McHardy 7114df53d8bSPatrick McHardy static int set_nf_call_iptables(struct net_bridge *br, unsigned long val) 7124df53d8bSPatrick McHardy { 7138df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_IPTABLES, !!val); 7144df53d8bSPatrick McHardy return 0; 7154df53d8bSPatrick McHardy } 7164df53d8bSPatrick McHardy 717fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_store( 7184df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 7194df53d8bSPatrick McHardy size_t len) 7204df53d8bSPatrick McHardy { 7214df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_iptables); 7224df53d8bSPatrick McHardy } 723fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_iptables); 7244df53d8bSPatrick McHardy 725fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_show( 7264df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 7274df53d8bSPatrick McHardy { 7284df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 7298df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_IP6TABLES)); 7304df53d8bSPatrick McHardy } 7314df53d8bSPatrick McHardy 7324df53d8bSPatrick McHardy static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val) 7334df53d8bSPatrick McHardy { 7348df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_IP6TABLES, !!val); 7354df53d8bSPatrick McHardy return 0; 7364df53d8bSPatrick McHardy } 7374df53d8bSPatrick McHardy 738fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_store( 7394df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 7404df53d8bSPatrick McHardy size_t len) 7414df53d8bSPatrick McHardy { 7424df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_ip6tables); 7434df53d8bSPatrick McHardy } 744fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_ip6tables); 7454df53d8bSPatrick McHardy 746fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_show( 7474df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 7484df53d8bSPatrick McHardy { 7494df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 7508df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_ARPTABLES)); 7514df53d8bSPatrick McHardy } 7524df53d8bSPatrick McHardy 7534df53d8bSPatrick McHardy static int set_nf_call_arptables(struct net_bridge *br, unsigned long val) 7544df53d8bSPatrick McHardy { 7558df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_ARPTABLES, !!val); 7564df53d8bSPatrick McHardy return 0; 7574df53d8bSPatrick McHardy } 7584df53d8bSPatrick McHardy 759fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_store( 7604df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 7614df53d8bSPatrick McHardy size_t len) 7624df53d8bSPatrick McHardy { 7634df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_arptables); 7644df53d8bSPatrick McHardy } 765fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_arptables); 7664df53d8bSPatrick McHardy #endif 767243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 768fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_show(struct device *d, 769243a2e63SVlad Yasevich struct device_attribute *attr, 770243a2e63SVlad Yasevich char *buf) 771243a2e63SVlad Yasevich { 772243a2e63SVlad Yasevich struct net_bridge *br = to_bridge(d); 773ae75767eSNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_VLAN_ENABLED)); 774243a2e63SVlad Yasevich } 775243a2e63SVlad Yasevich 776fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_store(struct device *d, 777243a2e63SVlad Yasevich struct device_attribute *attr, 778243a2e63SVlad Yasevich const char *buf, size_t len) 779243a2e63SVlad Yasevich { 780243a2e63SVlad Yasevich return store_bridge_parm(d, buf, len, br_vlan_filter_toggle); 781243a2e63SVlad Yasevich } 782fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(vlan_filtering); 783204177f3SToshiaki Makita 784204177f3SToshiaki Makita static ssize_t vlan_protocol_show(struct device *d, 785204177f3SToshiaki Makita struct device_attribute *attr, 786204177f3SToshiaki Makita char *buf) 787204177f3SToshiaki Makita { 788204177f3SToshiaki Makita struct net_bridge *br = to_bridge(d); 789204177f3SToshiaki Makita return sprintf(buf, "%#06x\n", ntohs(br->vlan_proto)); 790204177f3SToshiaki Makita } 791204177f3SToshiaki Makita 792204177f3SToshiaki Makita static ssize_t vlan_protocol_store(struct device *d, 793204177f3SToshiaki Makita struct device_attribute *attr, 794204177f3SToshiaki Makita const char *buf, size_t len) 795204177f3SToshiaki Makita { 796204177f3SToshiaki Makita return store_bridge_parm(d, buf, len, br_vlan_set_proto); 797204177f3SToshiaki Makita } 798204177f3SToshiaki Makita static DEVICE_ATTR_RW(vlan_protocol); 79996a20d9dSVlad Yasevich 80096a20d9dSVlad Yasevich static ssize_t default_pvid_show(struct device *d, 80196a20d9dSVlad Yasevich struct device_attribute *attr, 80296a20d9dSVlad Yasevich char *buf) 80396a20d9dSVlad Yasevich { 80496a20d9dSVlad Yasevich struct net_bridge *br = to_bridge(d); 80596a20d9dSVlad Yasevich return sprintf(buf, "%d\n", br->default_pvid); 80696a20d9dSVlad Yasevich } 80796a20d9dSVlad Yasevich 80896a20d9dSVlad Yasevich static ssize_t default_pvid_store(struct device *d, 80996a20d9dSVlad Yasevich struct device_attribute *attr, 81096a20d9dSVlad Yasevich const char *buf, size_t len) 81196a20d9dSVlad Yasevich { 81296a20d9dSVlad Yasevich return store_bridge_parm(d, buf, len, br_vlan_set_default_pvid); 81396a20d9dSVlad Yasevich } 81496a20d9dSVlad Yasevich static DEVICE_ATTR_RW(default_pvid); 8156dada9b1SNikolay Aleksandrov 8166dada9b1SNikolay Aleksandrov static ssize_t vlan_stats_enabled_show(struct device *d, 8176dada9b1SNikolay Aleksandrov struct device_attribute *attr, 8186dada9b1SNikolay Aleksandrov char *buf) 8196dada9b1SNikolay Aleksandrov { 8206dada9b1SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 821ae75767eSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_VLAN_STATS_ENABLED)); 8226dada9b1SNikolay Aleksandrov } 8236dada9b1SNikolay Aleksandrov 8246dada9b1SNikolay Aleksandrov static ssize_t vlan_stats_enabled_store(struct device *d, 8256dada9b1SNikolay Aleksandrov struct device_attribute *attr, 8266dada9b1SNikolay Aleksandrov const char *buf, size_t len) 8276dada9b1SNikolay Aleksandrov { 8286dada9b1SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_vlan_set_stats); 8296dada9b1SNikolay Aleksandrov } 8306dada9b1SNikolay Aleksandrov static DEVICE_ATTR_RW(vlan_stats_enabled); 8319163a0fcSNikolay Aleksandrov 8329163a0fcSNikolay Aleksandrov static ssize_t vlan_stats_per_port_show(struct device *d, 8339163a0fcSNikolay Aleksandrov struct device_attribute *attr, 8349163a0fcSNikolay Aleksandrov char *buf) 8359163a0fcSNikolay Aleksandrov { 8369163a0fcSNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 8379163a0fcSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_VLAN_STATS_PER_PORT)); 8389163a0fcSNikolay Aleksandrov } 8399163a0fcSNikolay Aleksandrov 8409163a0fcSNikolay Aleksandrov static ssize_t vlan_stats_per_port_store(struct device *d, 8419163a0fcSNikolay Aleksandrov struct device_attribute *attr, 8429163a0fcSNikolay Aleksandrov const char *buf, size_t len) 8439163a0fcSNikolay Aleksandrov { 8449163a0fcSNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_vlan_set_stats_per_port); 8459163a0fcSNikolay Aleksandrov } 8469163a0fcSNikolay Aleksandrov static DEVICE_ATTR_RW(vlan_stats_per_port); 847243a2e63SVlad Yasevich #endif 8480909e117SHerbert Xu 8491da177e4SLinus Torvalds static struct attribute *bridge_attrs[] = { 85043cb76d9SGreg Kroah-Hartman &dev_attr_forward_delay.attr, 85143cb76d9SGreg Kroah-Hartman &dev_attr_hello_time.attr, 85243cb76d9SGreg Kroah-Hartman &dev_attr_max_age.attr, 85343cb76d9SGreg Kroah-Hartman &dev_attr_ageing_time.attr, 85443cb76d9SGreg Kroah-Hartman &dev_attr_stp_state.attr, 855515853ccSstephen hemminger &dev_attr_group_fwd_mask.attr, 85643cb76d9SGreg Kroah-Hartman &dev_attr_priority.attr, 85743cb76d9SGreg Kroah-Hartman &dev_attr_bridge_id.attr, 85843cb76d9SGreg Kroah-Hartman &dev_attr_root_id.attr, 85943cb76d9SGreg Kroah-Hartman &dev_attr_root_path_cost.attr, 86043cb76d9SGreg Kroah-Hartman &dev_attr_root_port.attr, 86143cb76d9SGreg Kroah-Hartman &dev_attr_topology_change.attr, 86243cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_detected.attr, 86343cb76d9SGreg Kroah-Hartman &dev_attr_hello_timer.attr, 86443cb76d9SGreg Kroah-Hartman &dev_attr_tcn_timer.attr, 86543cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_timer.attr, 86643cb76d9SGreg Kroah-Hartman &dev_attr_gc_timer.attr, 86743cb76d9SGreg Kroah-Hartman &dev_attr_group_addr.attr, 8689cf63747SStephen Hemminger &dev_attr_flush.attr, 86970e4272bSNikolay Aleksandrov &dev_attr_no_linklocal_learn.attr, 8700909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 8710909e117SHerbert Xu &dev_attr_multicast_router.attr, 872561f1103SHerbert Xu &dev_attr_multicast_snooping.attr, 873c5c23260SHerbert Xu &dev_attr_multicast_querier.attr, 8741c8ad5bfSCong Wang &dev_attr_multicast_query_use_ifaddr.attr, 875b195167fSHerbert Xu &dev_attr_hash_elasticity.attr, 876b195167fSHerbert Xu &dev_attr_hash_max.attr, 877d902eee4SHerbert Xu &dev_attr_multicast_last_member_count.attr, 878d902eee4SHerbert Xu &dev_attr_multicast_startup_query_count.attr, 879d902eee4SHerbert Xu &dev_attr_multicast_last_member_interval.attr, 880d902eee4SHerbert Xu &dev_attr_multicast_membership_interval.attr, 881d902eee4SHerbert Xu &dev_attr_multicast_querier_interval.attr, 882d902eee4SHerbert Xu &dev_attr_multicast_query_interval.attr, 883d902eee4SHerbert Xu &dev_attr_multicast_query_response_interval.attr, 884d902eee4SHerbert Xu &dev_attr_multicast_startup_query_interval.attr, 8851080ab95SNikolay Aleksandrov &dev_attr_multicast_stats_enabled.attr, 8865e923585SNikolay Aleksandrov &dev_attr_multicast_igmp_version.attr, 887aa2ae3e7SNikolay Aleksandrov #if IS_ENABLED(CONFIG_IPV6) 888aa2ae3e7SNikolay Aleksandrov &dev_attr_multicast_mld_version.attr, 889aa2ae3e7SNikolay Aleksandrov #endif 8900909e117SHerbert Xu #endif 89134666d46SPablo Neira Ayuso #if IS_ENABLED(CONFIG_BRIDGE_NETFILTER) 8924df53d8bSPatrick McHardy &dev_attr_nf_call_iptables.attr, 8934df53d8bSPatrick McHardy &dev_attr_nf_call_ip6tables.attr, 8944df53d8bSPatrick McHardy &dev_attr_nf_call_arptables.attr, 8954df53d8bSPatrick McHardy #endif 896243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 897243a2e63SVlad Yasevich &dev_attr_vlan_filtering.attr, 898204177f3SToshiaki Makita &dev_attr_vlan_protocol.attr, 89996a20d9dSVlad Yasevich &dev_attr_default_pvid.attr, 9006dada9b1SNikolay Aleksandrov &dev_attr_vlan_stats_enabled.attr, 9019163a0fcSNikolay Aleksandrov &dev_attr_vlan_stats_per_port.attr, 902243a2e63SVlad Yasevich #endif 9031da177e4SLinus Torvalds NULL 9041da177e4SLinus Torvalds }; 9051da177e4SLinus Torvalds 906cddbb79fSArvind Yadav static const struct attribute_group bridge_group = { 9071da177e4SLinus Torvalds .name = SYSFS_BRIDGE_ATTR, 9081da177e4SLinus Torvalds .attrs = bridge_attrs, 9091da177e4SLinus Torvalds }; 9101da177e4SLinus Torvalds 9111da177e4SLinus Torvalds /* 9121da177e4SLinus Torvalds * Export the forwarding information table as a binary file 9131da177e4SLinus Torvalds * The records are struct __fdb_entry. 9141da177e4SLinus Torvalds * 9151da177e4SLinus Torvalds * Returns the number of bytes read. 9161da177e4SLinus Torvalds */ 9172c3c8beaSChris Wright static ssize_t brforward_read(struct file *filp, struct kobject *kobj, 91891a69029SZhang Rui struct bin_attribute *bin_attr, 91991a69029SZhang Rui char *buf, loff_t off, size_t count) 9201da177e4SLinus Torvalds { 921aeb7ed14SGeliang Tang struct device *dev = kobj_to_dev(kobj); 92243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(dev); 9231da177e4SLinus Torvalds int n; 9241da177e4SLinus Torvalds 9251da177e4SLinus Torvalds /* must read whole records */ 9261da177e4SLinus Torvalds if (off % sizeof(struct __fdb_entry) != 0) 9271da177e4SLinus Torvalds return -EINVAL; 9281da177e4SLinus Torvalds 9291da177e4SLinus Torvalds n = br_fdb_fillbuf(br, buf, 9301da177e4SLinus Torvalds count / sizeof(struct __fdb_entry), 9311da177e4SLinus Torvalds off / sizeof(struct __fdb_entry)); 9321da177e4SLinus Torvalds 9331da177e4SLinus Torvalds if (n > 0) 9341da177e4SLinus Torvalds n *= sizeof(struct __fdb_entry); 9351da177e4SLinus Torvalds 9361da177e4SLinus Torvalds return n; 9371da177e4SLinus Torvalds } 9381da177e4SLinus Torvalds 9391da177e4SLinus Torvalds static struct bin_attribute bridge_forward = { 9401da177e4SLinus Torvalds .attr = { .name = SYSFS_BRIDGE_FDB, 941d6444062SJoe Perches .mode = 0444, }, 9421da177e4SLinus Torvalds .read = brforward_read, 9431da177e4SLinus Torvalds }; 9441da177e4SLinus Torvalds 9451da177e4SLinus Torvalds /* 9461da177e4SLinus Torvalds * Add entries in sysfs onto the existing network class device 9471da177e4SLinus Torvalds * for the bridge. 9481da177e4SLinus Torvalds * Adds a attribute group "bridge" containing tuning parameters. 9491da177e4SLinus Torvalds * Binary attribute containing the forward table 9501da177e4SLinus Torvalds * Sub directory to hold links to interfaces. 9511da177e4SLinus Torvalds * 9521da177e4SLinus Torvalds * Note: the ifobj exists only to be a subdirectory 9531da177e4SLinus Torvalds * to hold links. The ifobj exists in same data structure 9541da177e4SLinus Torvalds * as it's parent the bridge so reference counting works. 9551da177e4SLinus Torvalds */ 9561da177e4SLinus Torvalds int br_sysfs_addbr(struct net_device *dev) 9571da177e4SLinus Torvalds { 95843cb76d9SGreg Kroah-Hartman struct kobject *brobj = &dev->dev.kobj; 9591da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 9601da177e4SLinus Torvalds int err; 9611da177e4SLinus Torvalds 9621da177e4SLinus Torvalds err = sysfs_create_group(brobj, &bridge_group); 9631da177e4SLinus Torvalds if (err) { 9641da177e4SLinus Torvalds pr_info("%s: can't create group %s/%s\n", 9650dc47877SHarvey Harrison __func__, dev->name, bridge_group.name); 9661da177e4SLinus Torvalds goto out1; 9671da177e4SLinus Torvalds } 9681da177e4SLinus Torvalds 9691da177e4SLinus Torvalds err = sysfs_create_bin_file(brobj, &bridge_forward); 9701da177e4SLinus Torvalds if (err) { 9711842c4beSRandy Dunlap pr_info("%s: can't create attribute file %s/%s\n", 9720dc47877SHarvey Harrison __func__, dev->name, bridge_forward.attr.name); 9731da177e4SLinus Torvalds goto out2; 9741da177e4SLinus Torvalds } 9751da177e4SLinus Torvalds 97643b98c4aSGreg Kroah-Hartman br->ifobj = kobject_create_and_add(SYSFS_BRIDGE_PORT_SUBDIR, brobj); 97743b98c4aSGreg Kroah-Hartman if (!br->ifobj) { 9781da177e4SLinus Torvalds pr_info("%s: can't add kobject (directory) %s/%s\n", 9790dc47877SHarvey Harrison __func__, dev->name, SYSFS_BRIDGE_PORT_SUBDIR); 980b5958963SPan Bian err = -ENOMEM; 9811da177e4SLinus Torvalds goto out3; 9821da177e4SLinus Torvalds } 9831da177e4SLinus Torvalds return 0; 9841da177e4SLinus Torvalds out3: 98543cb76d9SGreg Kroah-Hartman sysfs_remove_bin_file(&dev->dev.kobj, &bridge_forward); 9861da177e4SLinus Torvalds out2: 98743cb76d9SGreg Kroah-Hartman sysfs_remove_group(&dev->dev.kobj, &bridge_group); 9881da177e4SLinus Torvalds out1: 9891da177e4SLinus Torvalds return err; 9901da177e4SLinus Torvalds 9911da177e4SLinus Torvalds } 9921da177e4SLinus Torvalds 9931da177e4SLinus Torvalds void br_sysfs_delbr(struct net_device *dev) 9941da177e4SLinus Torvalds { 99543cb76d9SGreg Kroah-Hartman struct kobject *kobj = &dev->dev.kobj; 9961da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 9971da177e4SLinus Torvalds 99878a2d906SGreg Kroah-Hartman kobject_put(br->ifobj); 9991da177e4SLinus Torvalds sysfs_remove_bin_file(kobj, &bridge_forward); 10001da177e4SLinus Torvalds sysfs_remove_group(kobj, &bridge_group); 10011da177e4SLinus Torvalds } 1002