11da177e4SLinus Torvalds /* 215401946SWang Sheng-Hui * Sysfs attributes of bridge 31da177e4SLinus Torvalds * Linux ethernet bridge 41da177e4SLinus Torvalds * 51da177e4SLinus Torvalds * Authors: 61da177e4SLinus Torvalds * Stephen Hemminger <shemminger@osdl.org> 71da177e4SLinus Torvalds * 81da177e4SLinus Torvalds * This program is free software; you can redistribute it and/or 91da177e4SLinus Torvalds * modify it under the terms of the GNU General Public License 101da177e4SLinus Torvalds * as published by the Free Software Foundation; either version 111da177e4SLinus Torvalds * 2 of the License, or (at your option) any later version. 121da177e4SLinus Torvalds */ 131da177e4SLinus Torvalds 144fc268d2SRandy Dunlap #include <linux/capability.h> 151da177e4SLinus Torvalds #include <linux/kernel.h> 161da177e4SLinus Torvalds #include <linux/netdevice.h> 17b3343a2aSJohn Fastabend #include <linux/etherdevice.h> 181da177e4SLinus Torvalds #include <linux/if_bridge.h> 191da177e4SLinus Torvalds #include <linux/rtnetlink.h> 201da177e4SLinus Torvalds #include <linux/spinlock.h> 211da177e4SLinus Torvalds #include <linux/times.h> 22174cd4b1SIngo Molnar #include <linux/sched/signal.h> 231da177e4SLinus Torvalds 241da177e4SLinus Torvalds #include "br_private.h" 251da177e4SLinus Torvalds 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 { 13317120889SStephen Hemminger br_stp_set_enabled(br, val); 13417120889SStephen Hemminger 1354436156bSXin Long return 0; 1364436156bSXin Long } 1374436156bSXin Long 1384436156bSXin Long static ssize_t stp_state_store(struct device *d, 1394436156bSXin Long struct device_attribute *attr, const char *buf, 1404436156bSXin Long size_t len) 1414436156bSXin Long { 1424436156bSXin Long return store_bridge_parm(d, buf, len, set_stp_state); 1431da177e4SLinus Torvalds } 144fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(stp_state); 1451da177e4SLinus Torvalds 146fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_show(struct device *d, 147fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 148fbf2671bSsfeldma@cumulusnetworks.com char *buf) 149515853ccSstephen hemminger { 150515853ccSstephen hemminger struct net_bridge *br = to_bridge(d); 151515853ccSstephen hemminger return sprintf(buf, "%#x\n", br->group_fwd_mask); 152515853ccSstephen hemminger } 153515853ccSstephen hemminger 154347db6b4SXin Long static int set_group_fwd_mask(struct net_bridge *br, unsigned long val) 155347db6b4SXin Long { 156347db6b4SXin Long if (val & BR_GROUPFWD_RESTRICTED) 157347db6b4SXin Long return -EINVAL; 158347db6b4SXin Long 159347db6b4SXin Long br->group_fwd_mask = val; 160347db6b4SXin Long 161347db6b4SXin Long return 0; 162347db6b4SXin Long } 163515853ccSstephen hemminger 164fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_store(struct device *d, 165fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 166fbf2671bSsfeldma@cumulusnetworks.com const char *buf, 167515853ccSstephen hemminger size_t len) 168515853ccSstephen hemminger { 169347db6b4SXin Long return store_bridge_parm(d, buf, len, set_group_fwd_mask); 170515853ccSstephen hemminger } 171fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_fwd_mask); 172515853ccSstephen hemminger 173fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_show(struct device *d, struct device_attribute *attr, 17443cb76d9SGreg Kroah-Hartman char *buf) 1751da177e4SLinus Torvalds { 17643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1771da177e4SLinus Torvalds return sprintf(buf, "%d\n", 1781da177e4SLinus Torvalds (br->bridge_id.prio[0] << 8) | br->bridge_id.prio[1]); 1791da177e4SLinus Torvalds } 1801da177e4SLinus Torvalds 1818d4698f7SStephen Hemminger static int set_priority(struct net_bridge *br, unsigned long val) 1821da177e4SLinus Torvalds { 1831da177e4SLinus Torvalds br_stp_set_bridge_priority(br, (u16) val); 1848d4698f7SStephen Hemminger return 0; 1851da177e4SLinus Torvalds } 1861da177e4SLinus Torvalds 187fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_store(struct device *d, struct device_attribute *attr, 1881da177e4SLinus Torvalds const char *buf, size_t len) 1891da177e4SLinus Torvalds { 19043cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_priority); 1911da177e4SLinus Torvalds } 192fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(priority); 1931da177e4SLinus Torvalds 194fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_id_show(struct device *d, struct device_attribute *attr, 19543cb76d9SGreg Kroah-Hartman char *buf) 1961da177e4SLinus Torvalds { 19743cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->designated_root); 1981da177e4SLinus Torvalds } 199fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_id); 2001da177e4SLinus Torvalds 201fbf2671bSsfeldma@cumulusnetworks.com static ssize_t bridge_id_show(struct device *d, struct device_attribute *attr, 20243cb76d9SGreg Kroah-Hartman char *buf) 2031da177e4SLinus Torvalds { 20443cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->bridge_id); 2051da177e4SLinus Torvalds } 206fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(bridge_id); 2071da177e4SLinus Torvalds 208fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_port_show(struct device *d, struct device_attribute *attr, 20943cb76d9SGreg Kroah-Hartman char *buf) 2101da177e4SLinus Torvalds { 21143cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_port); 2121da177e4SLinus Torvalds } 213fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_port); 2141da177e4SLinus Torvalds 215fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_path_cost_show(struct device *d, 21643cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2171da177e4SLinus Torvalds { 21843cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_path_cost); 2191da177e4SLinus Torvalds } 220fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_path_cost); 2211da177e4SLinus Torvalds 222fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_show(struct device *d, 22343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2241da177e4SLinus Torvalds { 22543cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->topology_change); 2261da177e4SLinus Torvalds } 227fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change); 2281da177e4SLinus Torvalds 229fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_detected_show(struct device *d, 23043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 23143cb76d9SGreg Kroah-Hartman char *buf) 2321da177e4SLinus Torvalds { 23343cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2341da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->topology_change_detected); 2351da177e4SLinus Torvalds } 236fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_detected); 2371da177e4SLinus Torvalds 238fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_timer_show(struct device *d, 23943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2401da177e4SLinus Torvalds { 24143cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2421da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->hello_timer)); 2431da177e4SLinus Torvalds } 244fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(hello_timer); 2451da177e4SLinus Torvalds 246fbf2671bSsfeldma@cumulusnetworks.com static ssize_t tcn_timer_show(struct device *d, struct device_attribute *attr, 24743cb76d9SGreg Kroah-Hartman char *buf) 2481da177e4SLinus Torvalds { 24943cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2501da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->tcn_timer)); 2511da177e4SLinus Torvalds } 252fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(tcn_timer); 2531da177e4SLinus Torvalds 254fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_timer_show(struct device *d, 25543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 25643cb76d9SGreg Kroah-Hartman char *buf) 2571da177e4SLinus Torvalds { 25843cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2591da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->topology_change_timer)); 2601da177e4SLinus Torvalds } 261fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_timer); 2621da177e4SLinus Torvalds 263fbf2671bSsfeldma@cumulusnetworks.com static ssize_t gc_timer_show(struct device *d, struct device_attribute *attr, 26443cb76d9SGreg Kroah-Hartman char *buf) 2651da177e4SLinus Torvalds { 26643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 267f7cdee8aSNikolay Aleksandrov return sprintf(buf, "%ld\n", br_timer_value(&br->gc_work.timer)); 2681da177e4SLinus Torvalds } 269fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(gc_timer); 2701da177e4SLinus Torvalds 271fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_show(struct device *d, 27243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 273fda93d92SStephen Hemminger { 27443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 275223b229bSAndy Shevchenko return sprintf(buf, "%pM\n", br->group_addr); 276fda93d92SStephen Hemminger } 277fda93d92SStephen Hemminger 278fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_store(struct device *d, 27943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 28043cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 281fda93d92SStephen Hemminger { 28243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2834197f24bSBen Hutchings u8 new_addr[6]; 284fda93d92SStephen Hemminger 285cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 286fda93d92SStephen Hemminger return -EPERM; 287fda93d92SStephen Hemminger 288223b229bSAndy Shevchenko if (!mac_pton(buf, new_addr)) 289fda93d92SStephen Hemminger return -EINVAL; 290fda93d92SStephen Hemminger 29146acc460SBen Hutchings if (!is_link_local_ether_addr(new_addr)) 292fda93d92SStephen Hemminger return -EINVAL; 293fda93d92SStephen Hemminger 294f64f9e71SJoe Perches if (new_addr[5] == 1 || /* 802.3x Pause address */ 295f64f9e71SJoe Perches new_addr[5] == 2 || /* 802.3ad Slow protocols */ 296f64f9e71SJoe Perches new_addr[5] == 3) /* 802.1X PAE address */ 297fda93d92SStephen Hemminger return -EINVAL; 298fda93d92SStephen Hemminger 299204177f3SToshiaki Makita if (!rtnl_trylock()) 300204177f3SToshiaki Makita return restart_syscall(); 301204177f3SToshiaki Makita 302fda93d92SStephen Hemminger spin_lock_bh(&br->lock); 303223b229bSAndy Shevchenko ether_addr_copy(br->group_addr, new_addr); 304fda93d92SStephen Hemminger spin_unlock_bh(&br->lock); 305204177f3SToshiaki Makita 306be3664a0SNikolay Aleksandrov br_opt_toggle(br, BROPT_GROUP_ADDR_SET, true); 307204177f3SToshiaki Makita br_recalculate_fwd_mask(br); 308047831a9SXin Long netdev_state_change(br->dev); 309204177f3SToshiaki Makita 310204177f3SToshiaki Makita rtnl_unlock(); 311204177f3SToshiaki Makita 312fda93d92SStephen Hemminger return len; 313fda93d92SStephen Hemminger } 314fda93d92SStephen Hemminger 315fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_addr); 316fda93d92SStephen Hemminger 31714f31bb3SXin Long static int set_flush(struct net_bridge *br, unsigned long val) 31814f31bb3SXin Long { 31914f31bb3SXin Long br_fdb_flush(br); 32014f31bb3SXin Long return 0; 32114f31bb3SXin Long } 32214f31bb3SXin Long 323fbf2671bSsfeldma@cumulusnetworks.com static ssize_t flush_store(struct device *d, 3249cf63747SStephen Hemminger struct device_attribute *attr, 3259cf63747SStephen Hemminger const char *buf, size_t len) 3269cf63747SStephen Hemminger { 32714f31bb3SXin Long return store_bridge_parm(d, buf, len, set_flush); 3289cf63747SStephen Hemminger } 329fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_WO(flush); 330fda93d92SStephen Hemminger 3310909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 332fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_show(struct device *d, 3330909e117SHerbert Xu struct device_attribute *attr, char *buf) 3340909e117SHerbert Xu { 3350909e117SHerbert Xu struct net_bridge *br = to_bridge(d); 3360909e117SHerbert Xu return sprintf(buf, "%d\n", br->multicast_router); 3370909e117SHerbert Xu } 3380909e117SHerbert Xu 339fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_store(struct device *d, 3400909e117SHerbert Xu struct device_attribute *attr, 3410909e117SHerbert Xu const char *buf, size_t len) 3420909e117SHerbert Xu { 3430909e117SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_router); 3440909e117SHerbert Xu } 345fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_router); 346561f1103SHerbert Xu 347fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_show(struct device *d, 348561f1103SHerbert Xu struct device_attribute *attr, 349561f1103SHerbert Xu char *buf) 350561f1103SHerbert Xu { 351561f1103SHerbert Xu struct net_bridge *br = to_bridge(d); 35213cefad2SNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_MULTICAST_ENABLED)); 353561f1103SHerbert Xu } 354561f1103SHerbert Xu 355fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_store(struct device *d, 356561f1103SHerbert Xu struct device_attribute *attr, 357561f1103SHerbert Xu const char *buf, size_t len) 358561f1103SHerbert Xu { 359561f1103SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_toggle); 360561f1103SHerbert Xu } 361fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_snooping); 362b195167fSHerbert Xu 363fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_use_ifaddr_show(struct device *d, 3641c8ad5bfSCong Wang struct device_attribute *attr, 3651c8ad5bfSCong Wang char *buf) 3661c8ad5bfSCong Wang { 3671c8ad5bfSCong Wang struct net_bridge *br = to_bridge(d); 368675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", 369675779adSNikolay Aleksandrov br_opt_get(br, BROPT_MULTICAST_QUERY_USE_IFADDR)); 3701c8ad5bfSCong Wang } 3711c8ad5bfSCong Wang 3721c8ad5bfSCong Wang static int set_query_use_ifaddr(struct net_bridge *br, unsigned long val) 3731c8ad5bfSCong Wang { 374675779adSNikolay Aleksandrov br_opt_toggle(br, BROPT_MULTICAST_QUERY_USE_IFADDR, !!val); 3751c8ad5bfSCong Wang return 0; 3761c8ad5bfSCong Wang } 3771c8ad5bfSCong Wang 3781c8ad5bfSCong Wang static ssize_t 379fbf2671bSsfeldma@cumulusnetworks.com multicast_query_use_ifaddr_store(struct device *d, 3801c8ad5bfSCong Wang struct device_attribute *attr, 3811c8ad5bfSCong Wang const char *buf, size_t len) 3821c8ad5bfSCong Wang { 3831c8ad5bfSCong Wang return store_bridge_parm(d, buf, len, set_query_use_ifaddr); 3841c8ad5bfSCong Wang } 385fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_use_ifaddr); 3861c8ad5bfSCong Wang 387fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_show(struct device *d, 388c5c23260SHerbert Xu struct device_attribute *attr, 389c5c23260SHerbert Xu char *buf) 390c5c23260SHerbert Xu { 391c5c23260SHerbert Xu struct net_bridge *br = to_bridge(d); 392675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_MULTICAST_QUERIER)); 393c5c23260SHerbert Xu } 394c5c23260SHerbert Xu 395fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_store(struct device *d, 396c5c23260SHerbert Xu struct device_attribute *attr, 397c5c23260SHerbert Xu const char *buf, size_t len) 398c5c23260SHerbert Xu { 399c5c23260SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_querier); 400c5c23260SHerbert Xu } 401fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier); 402c5c23260SHerbert Xu 403fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_show(struct device *d, 404b195167fSHerbert Xu struct device_attribute *attr, char *buf) 405b195167fSHerbert Xu { 406b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 407b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_elasticity); 408b195167fSHerbert Xu } 409b195167fSHerbert Xu 410b195167fSHerbert Xu static int set_elasticity(struct net_bridge *br, unsigned long val) 411b195167fSHerbert Xu { 412b195167fSHerbert Xu br->hash_elasticity = val; 413b195167fSHerbert Xu return 0; 414b195167fSHerbert Xu } 415b195167fSHerbert Xu 416fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_store(struct device *d, 417b195167fSHerbert Xu struct device_attribute *attr, 418b195167fSHerbert Xu const char *buf, size_t len) 419b195167fSHerbert Xu { 420b195167fSHerbert Xu return store_bridge_parm(d, buf, len, set_elasticity); 421b195167fSHerbert Xu } 422fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_elasticity); 423b195167fSHerbert Xu 424fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_show(struct device *d, struct device_attribute *attr, 425b195167fSHerbert Xu char *buf) 426b195167fSHerbert Xu { 427b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 428b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_max); 429b195167fSHerbert Xu } 430b195167fSHerbert Xu 431fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_store(struct device *d, struct device_attribute *attr, 432b195167fSHerbert Xu const char *buf, size_t len) 433b195167fSHerbert Xu { 434b195167fSHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_hash_max); 435b195167fSHerbert Xu } 436fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_max); 437d902eee4SHerbert Xu 4385e923585SNikolay Aleksandrov static ssize_t multicast_igmp_version_show(struct device *d, 4395e923585SNikolay Aleksandrov struct device_attribute *attr, 4405e923585SNikolay Aleksandrov char *buf) 4415e923585SNikolay Aleksandrov { 4425e923585SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 4435e923585SNikolay Aleksandrov 4445e923585SNikolay Aleksandrov return sprintf(buf, "%u\n", br->multicast_igmp_version); 4455e923585SNikolay Aleksandrov } 4465e923585SNikolay Aleksandrov 4475e923585SNikolay Aleksandrov static ssize_t multicast_igmp_version_store(struct device *d, 4485e923585SNikolay Aleksandrov struct device_attribute *attr, 4495e923585SNikolay Aleksandrov const char *buf, size_t len) 4505e923585SNikolay Aleksandrov { 4515e923585SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_multicast_set_igmp_version); 4525e923585SNikolay Aleksandrov } 4535e923585SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_igmp_version); 4545e923585SNikolay Aleksandrov 455fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_show(struct device *d, 456d902eee4SHerbert Xu struct device_attribute *attr, 457d902eee4SHerbert Xu char *buf) 458d902eee4SHerbert Xu { 459d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 460d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_last_member_count); 461d902eee4SHerbert Xu } 462d902eee4SHerbert Xu 463d902eee4SHerbert Xu static int set_last_member_count(struct net_bridge *br, unsigned long val) 464d902eee4SHerbert Xu { 465d902eee4SHerbert Xu br->multicast_last_member_count = val; 466d902eee4SHerbert Xu return 0; 467d902eee4SHerbert Xu } 468d902eee4SHerbert Xu 469fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_store(struct device *d, 470d902eee4SHerbert Xu struct device_attribute *attr, 471d902eee4SHerbert Xu const char *buf, size_t len) 472d902eee4SHerbert Xu { 473d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_count); 474d902eee4SHerbert Xu } 475fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_count); 476d902eee4SHerbert Xu 477fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_show( 478d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 479d902eee4SHerbert Xu { 480d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 481d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_startup_query_count); 482d902eee4SHerbert Xu } 483d902eee4SHerbert Xu 484d902eee4SHerbert Xu static int set_startup_query_count(struct net_bridge *br, unsigned long val) 485d902eee4SHerbert Xu { 486d902eee4SHerbert Xu br->multicast_startup_query_count = val; 487d902eee4SHerbert Xu return 0; 488d902eee4SHerbert Xu } 489d902eee4SHerbert Xu 490fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_store( 491d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 492d902eee4SHerbert Xu size_t len) 493d902eee4SHerbert Xu { 494d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_count); 495d902eee4SHerbert Xu } 496fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_count); 497d902eee4SHerbert Xu 498fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_show( 499d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 500d902eee4SHerbert Xu { 501d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 502d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 503d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_last_member_interval)); 504d902eee4SHerbert Xu } 505d902eee4SHerbert Xu 506d902eee4SHerbert Xu static int set_last_member_interval(struct net_bridge *br, unsigned long val) 507d902eee4SHerbert Xu { 508d902eee4SHerbert Xu br->multicast_last_member_interval = clock_t_to_jiffies(val); 509d902eee4SHerbert Xu return 0; 510d902eee4SHerbert Xu } 511d902eee4SHerbert Xu 512fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_store( 513d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 514d902eee4SHerbert Xu size_t len) 515d902eee4SHerbert Xu { 516d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_interval); 517d902eee4SHerbert Xu } 518fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_interval); 519d902eee4SHerbert Xu 520fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_show( 521d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 522d902eee4SHerbert Xu { 523d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 524d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 525d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_membership_interval)); 526d902eee4SHerbert Xu } 527d902eee4SHerbert Xu 528d902eee4SHerbert Xu static int set_membership_interval(struct net_bridge *br, unsigned long val) 529d902eee4SHerbert Xu { 530d902eee4SHerbert Xu br->multicast_membership_interval = clock_t_to_jiffies(val); 531d902eee4SHerbert Xu return 0; 532d902eee4SHerbert Xu } 533d902eee4SHerbert Xu 534fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_store( 535d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 536d902eee4SHerbert Xu size_t len) 537d902eee4SHerbert Xu { 538d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_membership_interval); 539d902eee4SHerbert Xu } 540fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_membership_interval); 541d902eee4SHerbert Xu 542fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_show(struct device *d, 543d902eee4SHerbert Xu struct device_attribute *attr, 544d902eee4SHerbert Xu char *buf) 545d902eee4SHerbert Xu { 546d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 547d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 548d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_querier_interval)); 549d902eee4SHerbert Xu } 550d902eee4SHerbert Xu 551d902eee4SHerbert Xu static int set_querier_interval(struct net_bridge *br, unsigned long val) 552d902eee4SHerbert Xu { 553d902eee4SHerbert Xu br->multicast_querier_interval = clock_t_to_jiffies(val); 554d902eee4SHerbert Xu return 0; 555d902eee4SHerbert Xu } 556d902eee4SHerbert Xu 557fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_store(struct device *d, 558d902eee4SHerbert Xu struct device_attribute *attr, 559d902eee4SHerbert Xu const char *buf, size_t len) 560d902eee4SHerbert Xu { 561d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_querier_interval); 562d902eee4SHerbert Xu } 563fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier_interval); 564d902eee4SHerbert Xu 565fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_show(struct device *d, 566d902eee4SHerbert Xu struct device_attribute *attr, 567d902eee4SHerbert Xu char *buf) 568d902eee4SHerbert Xu { 569d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 570d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 571d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_interval)); 572d902eee4SHerbert Xu } 573d902eee4SHerbert Xu 574d902eee4SHerbert Xu static int set_query_interval(struct net_bridge *br, unsigned long val) 575d902eee4SHerbert Xu { 576d902eee4SHerbert Xu br->multicast_query_interval = clock_t_to_jiffies(val); 577d902eee4SHerbert Xu return 0; 578d902eee4SHerbert Xu } 579d902eee4SHerbert Xu 580fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_store(struct device *d, 581d902eee4SHerbert Xu struct device_attribute *attr, 582d902eee4SHerbert Xu const char *buf, size_t len) 583d902eee4SHerbert Xu { 584d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_interval); 585d902eee4SHerbert Xu } 586fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_interval); 587d902eee4SHerbert Xu 588fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_show( 589d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 590d902eee4SHerbert Xu { 591d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 592d902eee4SHerbert Xu return sprintf( 593d902eee4SHerbert Xu buf, "%lu\n", 594d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_response_interval)); 595d902eee4SHerbert Xu } 596d902eee4SHerbert Xu 597d902eee4SHerbert Xu static int set_query_response_interval(struct net_bridge *br, unsigned long val) 598d902eee4SHerbert Xu { 599d902eee4SHerbert Xu br->multicast_query_response_interval = clock_t_to_jiffies(val); 600d902eee4SHerbert Xu return 0; 601d902eee4SHerbert Xu } 602d902eee4SHerbert Xu 603fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_store( 604d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 605d902eee4SHerbert Xu size_t len) 606d902eee4SHerbert Xu { 607d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_response_interval); 608d902eee4SHerbert Xu } 609fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_response_interval); 610d902eee4SHerbert Xu 611fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_show( 612d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 613d902eee4SHerbert Xu { 614d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 615d902eee4SHerbert Xu return sprintf( 616d902eee4SHerbert Xu buf, "%lu\n", 617d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_startup_query_interval)); 618d902eee4SHerbert Xu } 619d902eee4SHerbert Xu 620d902eee4SHerbert Xu static int set_startup_query_interval(struct net_bridge *br, unsigned long val) 621d902eee4SHerbert Xu { 622d902eee4SHerbert Xu br->multicast_startup_query_interval = clock_t_to_jiffies(val); 623d902eee4SHerbert Xu return 0; 624d902eee4SHerbert Xu } 625d902eee4SHerbert Xu 626fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_store( 627d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 628d902eee4SHerbert Xu size_t len) 629d902eee4SHerbert Xu { 630d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_interval); 631d902eee4SHerbert Xu } 632fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_interval); 6331080ab95SNikolay Aleksandrov 6341080ab95SNikolay Aleksandrov static ssize_t multicast_stats_enabled_show(struct device *d, 6351080ab95SNikolay Aleksandrov struct device_attribute *attr, 6361080ab95SNikolay Aleksandrov char *buf) 6371080ab95SNikolay Aleksandrov { 6381080ab95SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 6391080ab95SNikolay Aleksandrov 640675779adSNikolay Aleksandrov return sprintf(buf, "%d\n", 641675779adSNikolay Aleksandrov br_opt_get(br, BROPT_MULTICAST_STATS_ENABLED)); 6421080ab95SNikolay Aleksandrov } 6431080ab95SNikolay Aleksandrov 6441080ab95SNikolay Aleksandrov static int set_stats_enabled(struct net_bridge *br, unsigned long val) 6451080ab95SNikolay Aleksandrov { 646675779adSNikolay Aleksandrov br_opt_toggle(br, BROPT_MULTICAST_STATS_ENABLED, !!val); 6471080ab95SNikolay Aleksandrov return 0; 6481080ab95SNikolay Aleksandrov } 6491080ab95SNikolay Aleksandrov 6501080ab95SNikolay Aleksandrov static ssize_t multicast_stats_enabled_store(struct device *d, 6511080ab95SNikolay Aleksandrov struct device_attribute *attr, 6521080ab95SNikolay Aleksandrov const char *buf, 6531080ab95SNikolay Aleksandrov size_t len) 6541080ab95SNikolay Aleksandrov { 6551080ab95SNikolay Aleksandrov return store_bridge_parm(d, buf, len, set_stats_enabled); 6561080ab95SNikolay Aleksandrov } 6571080ab95SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_stats_enabled); 658aa2ae3e7SNikolay Aleksandrov 659aa2ae3e7SNikolay Aleksandrov #if IS_ENABLED(CONFIG_IPV6) 660aa2ae3e7SNikolay Aleksandrov static ssize_t multicast_mld_version_show(struct device *d, 661aa2ae3e7SNikolay Aleksandrov struct device_attribute *attr, 662aa2ae3e7SNikolay Aleksandrov char *buf) 663aa2ae3e7SNikolay Aleksandrov { 664aa2ae3e7SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 665aa2ae3e7SNikolay Aleksandrov 666aa2ae3e7SNikolay Aleksandrov return sprintf(buf, "%u\n", br->multicast_mld_version); 667aa2ae3e7SNikolay Aleksandrov } 668aa2ae3e7SNikolay Aleksandrov 669aa2ae3e7SNikolay Aleksandrov static ssize_t multicast_mld_version_store(struct device *d, 670aa2ae3e7SNikolay Aleksandrov struct device_attribute *attr, 671aa2ae3e7SNikolay Aleksandrov const char *buf, size_t len) 672aa2ae3e7SNikolay Aleksandrov { 673aa2ae3e7SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_multicast_set_mld_version); 674aa2ae3e7SNikolay Aleksandrov } 675aa2ae3e7SNikolay Aleksandrov static DEVICE_ATTR_RW(multicast_mld_version); 676aa2ae3e7SNikolay Aleksandrov #endif 6770909e117SHerbert Xu #endif 67834666d46SPablo Neira Ayuso #if IS_ENABLED(CONFIG_BRIDGE_NETFILTER) 679fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_show( 6804df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6814df53d8bSPatrick McHardy { 6824df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6838df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_IPTABLES)); 6844df53d8bSPatrick McHardy } 6854df53d8bSPatrick McHardy 6864df53d8bSPatrick McHardy static int set_nf_call_iptables(struct net_bridge *br, unsigned long val) 6874df53d8bSPatrick McHardy { 6888df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_IPTABLES, !!val); 6894df53d8bSPatrick McHardy return 0; 6904df53d8bSPatrick McHardy } 6914df53d8bSPatrick McHardy 692fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_store( 6934df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6944df53d8bSPatrick McHardy size_t len) 6954df53d8bSPatrick McHardy { 6964df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_iptables); 6974df53d8bSPatrick McHardy } 698fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_iptables); 6994df53d8bSPatrick McHardy 700fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_show( 7014df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 7024df53d8bSPatrick McHardy { 7034df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 7048df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_IP6TABLES)); 7054df53d8bSPatrick McHardy } 7064df53d8bSPatrick McHardy 7074df53d8bSPatrick McHardy static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val) 7084df53d8bSPatrick McHardy { 7098df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_IP6TABLES, !!val); 7104df53d8bSPatrick McHardy return 0; 7114df53d8bSPatrick McHardy } 7124df53d8bSPatrick McHardy 713fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_store( 7144df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 7154df53d8bSPatrick McHardy size_t len) 7164df53d8bSPatrick McHardy { 7174df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_ip6tables); 7184df53d8bSPatrick McHardy } 719fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_ip6tables); 7204df53d8bSPatrick McHardy 721fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_show( 7224df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 7234df53d8bSPatrick McHardy { 7244df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 7258df3510fSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_NF_CALL_ARPTABLES)); 7264df53d8bSPatrick McHardy } 7274df53d8bSPatrick McHardy 7284df53d8bSPatrick McHardy static int set_nf_call_arptables(struct net_bridge *br, unsigned long val) 7294df53d8bSPatrick McHardy { 7308df3510fSNikolay Aleksandrov br_opt_toggle(br, BROPT_NF_CALL_ARPTABLES, !!val); 7314df53d8bSPatrick McHardy return 0; 7324df53d8bSPatrick McHardy } 7334df53d8bSPatrick McHardy 734fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_store( 7354df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 7364df53d8bSPatrick McHardy size_t len) 7374df53d8bSPatrick McHardy { 7384df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_arptables); 7394df53d8bSPatrick McHardy } 740fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_arptables); 7414df53d8bSPatrick McHardy #endif 742243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 743fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_show(struct device *d, 744243a2e63SVlad Yasevich struct device_attribute *attr, 745243a2e63SVlad Yasevich char *buf) 746243a2e63SVlad Yasevich { 747243a2e63SVlad Yasevich struct net_bridge *br = to_bridge(d); 748ae75767eSNikolay Aleksandrov return sprintf(buf, "%d\n", br_opt_get(br, BROPT_VLAN_ENABLED)); 749243a2e63SVlad Yasevich } 750243a2e63SVlad Yasevich 751fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_store(struct device *d, 752243a2e63SVlad Yasevich struct device_attribute *attr, 753243a2e63SVlad Yasevich const char *buf, size_t len) 754243a2e63SVlad Yasevich { 755243a2e63SVlad Yasevich return store_bridge_parm(d, buf, len, br_vlan_filter_toggle); 756243a2e63SVlad Yasevich } 757fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(vlan_filtering); 758204177f3SToshiaki Makita 759204177f3SToshiaki Makita static ssize_t vlan_protocol_show(struct device *d, 760204177f3SToshiaki Makita struct device_attribute *attr, 761204177f3SToshiaki Makita char *buf) 762204177f3SToshiaki Makita { 763204177f3SToshiaki Makita struct net_bridge *br = to_bridge(d); 764204177f3SToshiaki Makita return sprintf(buf, "%#06x\n", ntohs(br->vlan_proto)); 765204177f3SToshiaki Makita } 766204177f3SToshiaki Makita 767204177f3SToshiaki Makita static ssize_t vlan_protocol_store(struct device *d, 768204177f3SToshiaki Makita struct device_attribute *attr, 769204177f3SToshiaki Makita const char *buf, size_t len) 770204177f3SToshiaki Makita { 771204177f3SToshiaki Makita return store_bridge_parm(d, buf, len, br_vlan_set_proto); 772204177f3SToshiaki Makita } 773204177f3SToshiaki Makita static DEVICE_ATTR_RW(vlan_protocol); 77496a20d9dSVlad Yasevich 77596a20d9dSVlad Yasevich static ssize_t default_pvid_show(struct device *d, 77696a20d9dSVlad Yasevich struct device_attribute *attr, 77796a20d9dSVlad Yasevich char *buf) 77896a20d9dSVlad Yasevich { 77996a20d9dSVlad Yasevich struct net_bridge *br = to_bridge(d); 78096a20d9dSVlad Yasevich return sprintf(buf, "%d\n", br->default_pvid); 78196a20d9dSVlad Yasevich } 78296a20d9dSVlad Yasevich 78396a20d9dSVlad Yasevich static ssize_t default_pvid_store(struct device *d, 78496a20d9dSVlad Yasevich struct device_attribute *attr, 78596a20d9dSVlad Yasevich const char *buf, size_t len) 78696a20d9dSVlad Yasevich { 78796a20d9dSVlad Yasevich return store_bridge_parm(d, buf, len, br_vlan_set_default_pvid); 78896a20d9dSVlad Yasevich } 78996a20d9dSVlad Yasevich static DEVICE_ATTR_RW(default_pvid); 7906dada9b1SNikolay Aleksandrov 7916dada9b1SNikolay Aleksandrov static ssize_t vlan_stats_enabled_show(struct device *d, 7926dada9b1SNikolay Aleksandrov struct device_attribute *attr, 7936dada9b1SNikolay Aleksandrov char *buf) 7946dada9b1SNikolay Aleksandrov { 7956dada9b1SNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 796ae75767eSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_VLAN_STATS_ENABLED)); 7976dada9b1SNikolay Aleksandrov } 7986dada9b1SNikolay Aleksandrov 7996dada9b1SNikolay Aleksandrov static ssize_t vlan_stats_enabled_store(struct device *d, 8006dada9b1SNikolay Aleksandrov struct device_attribute *attr, 8016dada9b1SNikolay Aleksandrov const char *buf, size_t len) 8026dada9b1SNikolay Aleksandrov { 8036dada9b1SNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_vlan_set_stats); 8046dada9b1SNikolay Aleksandrov } 8056dada9b1SNikolay Aleksandrov static DEVICE_ATTR_RW(vlan_stats_enabled); 806*9163a0fcSNikolay Aleksandrov 807*9163a0fcSNikolay Aleksandrov static ssize_t vlan_stats_per_port_show(struct device *d, 808*9163a0fcSNikolay Aleksandrov struct device_attribute *attr, 809*9163a0fcSNikolay Aleksandrov char *buf) 810*9163a0fcSNikolay Aleksandrov { 811*9163a0fcSNikolay Aleksandrov struct net_bridge *br = to_bridge(d); 812*9163a0fcSNikolay Aleksandrov return sprintf(buf, "%u\n", br_opt_get(br, BROPT_VLAN_STATS_PER_PORT)); 813*9163a0fcSNikolay Aleksandrov } 814*9163a0fcSNikolay Aleksandrov 815*9163a0fcSNikolay Aleksandrov static ssize_t vlan_stats_per_port_store(struct device *d, 816*9163a0fcSNikolay Aleksandrov struct device_attribute *attr, 817*9163a0fcSNikolay Aleksandrov const char *buf, size_t len) 818*9163a0fcSNikolay Aleksandrov { 819*9163a0fcSNikolay Aleksandrov return store_bridge_parm(d, buf, len, br_vlan_set_stats_per_port); 820*9163a0fcSNikolay Aleksandrov } 821*9163a0fcSNikolay Aleksandrov static DEVICE_ATTR_RW(vlan_stats_per_port); 822243a2e63SVlad Yasevich #endif 8230909e117SHerbert Xu 8241da177e4SLinus Torvalds static struct attribute *bridge_attrs[] = { 82543cb76d9SGreg Kroah-Hartman &dev_attr_forward_delay.attr, 82643cb76d9SGreg Kroah-Hartman &dev_attr_hello_time.attr, 82743cb76d9SGreg Kroah-Hartman &dev_attr_max_age.attr, 82843cb76d9SGreg Kroah-Hartman &dev_attr_ageing_time.attr, 82943cb76d9SGreg Kroah-Hartman &dev_attr_stp_state.attr, 830515853ccSstephen hemminger &dev_attr_group_fwd_mask.attr, 83143cb76d9SGreg Kroah-Hartman &dev_attr_priority.attr, 83243cb76d9SGreg Kroah-Hartman &dev_attr_bridge_id.attr, 83343cb76d9SGreg Kroah-Hartman &dev_attr_root_id.attr, 83443cb76d9SGreg Kroah-Hartman &dev_attr_root_path_cost.attr, 83543cb76d9SGreg Kroah-Hartman &dev_attr_root_port.attr, 83643cb76d9SGreg Kroah-Hartman &dev_attr_topology_change.attr, 83743cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_detected.attr, 83843cb76d9SGreg Kroah-Hartman &dev_attr_hello_timer.attr, 83943cb76d9SGreg Kroah-Hartman &dev_attr_tcn_timer.attr, 84043cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_timer.attr, 84143cb76d9SGreg Kroah-Hartman &dev_attr_gc_timer.attr, 84243cb76d9SGreg Kroah-Hartman &dev_attr_group_addr.attr, 8439cf63747SStephen Hemminger &dev_attr_flush.attr, 8440909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 8450909e117SHerbert Xu &dev_attr_multicast_router.attr, 846561f1103SHerbert Xu &dev_attr_multicast_snooping.attr, 847c5c23260SHerbert Xu &dev_attr_multicast_querier.attr, 8481c8ad5bfSCong Wang &dev_attr_multicast_query_use_ifaddr.attr, 849b195167fSHerbert Xu &dev_attr_hash_elasticity.attr, 850b195167fSHerbert Xu &dev_attr_hash_max.attr, 851d902eee4SHerbert Xu &dev_attr_multicast_last_member_count.attr, 852d902eee4SHerbert Xu &dev_attr_multicast_startup_query_count.attr, 853d902eee4SHerbert Xu &dev_attr_multicast_last_member_interval.attr, 854d902eee4SHerbert Xu &dev_attr_multicast_membership_interval.attr, 855d902eee4SHerbert Xu &dev_attr_multicast_querier_interval.attr, 856d902eee4SHerbert Xu &dev_attr_multicast_query_interval.attr, 857d902eee4SHerbert Xu &dev_attr_multicast_query_response_interval.attr, 858d902eee4SHerbert Xu &dev_attr_multicast_startup_query_interval.attr, 8591080ab95SNikolay Aleksandrov &dev_attr_multicast_stats_enabled.attr, 8605e923585SNikolay Aleksandrov &dev_attr_multicast_igmp_version.attr, 861aa2ae3e7SNikolay Aleksandrov #if IS_ENABLED(CONFIG_IPV6) 862aa2ae3e7SNikolay Aleksandrov &dev_attr_multicast_mld_version.attr, 863aa2ae3e7SNikolay Aleksandrov #endif 8640909e117SHerbert Xu #endif 86534666d46SPablo Neira Ayuso #if IS_ENABLED(CONFIG_BRIDGE_NETFILTER) 8664df53d8bSPatrick McHardy &dev_attr_nf_call_iptables.attr, 8674df53d8bSPatrick McHardy &dev_attr_nf_call_ip6tables.attr, 8684df53d8bSPatrick McHardy &dev_attr_nf_call_arptables.attr, 8694df53d8bSPatrick McHardy #endif 870243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 871243a2e63SVlad Yasevich &dev_attr_vlan_filtering.attr, 872204177f3SToshiaki Makita &dev_attr_vlan_protocol.attr, 87396a20d9dSVlad Yasevich &dev_attr_default_pvid.attr, 8746dada9b1SNikolay Aleksandrov &dev_attr_vlan_stats_enabled.attr, 875*9163a0fcSNikolay Aleksandrov &dev_attr_vlan_stats_per_port.attr, 876243a2e63SVlad Yasevich #endif 8771da177e4SLinus Torvalds NULL 8781da177e4SLinus Torvalds }; 8791da177e4SLinus Torvalds 880cddbb79fSArvind Yadav static const struct attribute_group bridge_group = { 8811da177e4SLinus Torvalds .name = SYSFS_BRIDGE_ATTR, 8821da177e4SLinus Torvalds .attrs = bridge_attrs, 8831da177e4SLinus Torvalds }; 8841da177e4SLinus Torvalds 8851da177e4SLinus Torvalds /* 8861da177e4SLinus Torvalds * Export the forwarding information table as a binary file 8871da177e4SLinus Torvalds * The records are struct __fdb_entry. 8881da177e4SLinus Torvalds * 8891da177e4SLinus Torvalds * Returns the number of bytes read. 8901da177e4SLinus Torvalds */ 8912c3c8beaSChris Wright static ssize_t brforward_read(struct file *filp, struct kobject *kobj, 89291a69029SZhang Rui struct bin_attribute *bin_attr, 89391a69029SZhang Rui char *buf, loff_t off, size_t count) 8941da177e4SLinus Torvalds { 895aeb7ed14SGeliang Tang struct device *dev = kobj_to_dev(kobj); 89643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(dev); 8971da177e4SLinus Torvalds int n; 8981da177e4SLinus Torvalds 8991da177e4SLinus Torvalds /* must read whole records */ 9001da177e4SLinus Torvalds if (off % sizeof(struct __fdb_entry) != 0) 9011da177e4SLinus Torvalds return -EINVAL; 9021da177e4SLinus Torvalds 9031da177e4SLinus Torvalds n = br_fdb_fillbuf(br, buf, 9041da177e4SLinus Torvalds count / sizeof(struct __fdb_entry), 9051da177e4SLinus Torvalds off / sizeof(struct __fdb_entry)); 9061da177e4SLinus Torvalds 9071da177e4SLinus Torvalds if (n > 0) 9081da177e4SLinus Torvalds n *= sizeof(struct __fdb_entry); 9091da177e4SLinus Torvalds 9101da177e4SLinus Torvalds return n; 9111da177e4SLinus Torvalds } 9121da177e4SLinus Torvalds 9131da177e4SLinus Torvalds static struct bin_attribute bridge_forward = { 9141da177e4SLinus Torvalds .attr = { .name = SYSFS_BRIDGE_FDB, 915d6444062SJoe Perches .mode = 0444, }, 9161da177e4SLinus Torvalds .read = brforward_read, 9171da177e4SLinus Torvalds }; 9181da177e4SLinus Torvalds 9191da177e4SLinus Torvalds /* 9201da177e4SLinus Torvalds * Add entries in sysfs onto the existing network class device 9211da177e4SLinus Torvalds * for the bridge. 9221da177e4SLinus Torvalds * Adds a attribute group "bridge" containing tuning parameters. 9231da177e4SLinus Torvalds * Binary attribute containing the forward table 9241da177e4SLinus Torvalds * Sub directory to hold links to interfaces. 9251da177e4SLinus Torvalds * 9261da177e4SLinus Torvalds * Note: the ifobj exists only to be a subdirectory 9271da177e4SLinus Torvalds * to hold links. The ifobj exists in same data structure 9281da177e4SLinus Torvalds * as it's parent the bridge so reference counting works. 9291da177e4SLinus Torvalds */ 9301da177e4SLinus Torvalds int br_sysfs_addbr(struct net_device *dev) 9311da177e4SLinus Torvalds { 93243cb76d9SGreg Kroah-Hartman struct kobject *brobj = &dev->dev.kobj; 9331da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 9341da177e4SLinus Torvalds int err; 9351da177e4SLinus Torvalds 9361da177e4SLinus Torvalds err = sysfs_create_group(brobj, &bridge_group); 9371da177e4SLinus Torvalds if (err) { 9381da177e4SLinus Torvalds pr_info("%s: can't create group %s/%s\n", 9390dc47877SHarvey Harrison __func__, dev->name, bridge_group.name); 9401da177e4SLinus Torvalds goto out1; 9411da177e4SLinus Torvalds } 9421da177e4SLinus Torvalds 9431da177e4SLinus Torvalds err = sysfs_create_bin_file(brobj, &bridge_forward); 9441da177e4SLinus Torvalds if (err) { 9451842c4beSRandy Dunlap pr_info("%s: can't create attribute file %s/%s\n", 9460dc47877SHarvey Harrison __func__, dev->name, bridge_forward.attr.name); 9471da177e4SLinus Torvalds goto out2; 9481da177e4SLinus Torvalds } 9491da177e4SLinus Torvalds 95043b98c4aSGreg Kroah-Hartman br->ifobj = kobject_create_and_add(SYSFS_BRIDGE_PORT_SUBDIR, brobj); 95143b98c4aSGreg Kroah-Hartman if (!br->ifobj) { 9521da177e4SLinus Torvalds pr_info("%s: can't add kobject (directory) %s/%s\n", 9530dc47877SHarvey Harrison __func__, dev->name, SYSFS_BRIDGE_PORT_SUBDIR); 954b5958963SPan Bian err = -ENOMEM; 9551da177e4SLinus Torvalds goto out3; 9561da177e4SLinus Torvalds } 9571da177e4SLinus Torvalds return 0; 9581da177e4SLinus Torvalds out3: 95943cb76d9SGreg Kroah-Hartman sysfs_remove_bin_file(&dev->dev.kobj, &bridge_forward); 9601da177e4SLinus Torvalds out2: 96143cb76d9SGreg Kroah-Hartman sysfs_remove_group(&dev->dev.kobj, &bridge_group); 9621da177e4SLinus Torvalds out1: 9631da177e4SLinus Torvalds return err; 9641da177e4SLinus Torvalds 9651da177e4SLinus Torvalds } 9661da177e4SLinus Torvalds 9671da177e4SLinus Torvalds void br_sysfs_delbr(struct net_device *dev) 9681da177e4SLinus Torvalds { 96943cb76d9SGreg Kroah-Hartman struct kobject *kobj = &dev->dev.kobj; 9701da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 9711da177e4SLinus Torvalds 97278a2d906SGreg Kroah-Hartman kobject_put(br->ifobj); 9731da177e4SLinus Torvalds sysfs_remove_bin_file(kobj, &bridge_forward); 9741da177e4SLinus Torvalds sysfs_remove_group(kobj, &bridge_group); 9751da177e4SLinus Torvalds } 976