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> 221da177e4SLinus Torvalds 231da177e4SLinus Torvalds #include "br_private.h" 241da177e4SLinus Torvalds 2543cb76d9SGreg Kroah-Hartman #define to_dev(obj) container_of(obj, struct device, kobj) 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 478d4698f7SStephen Hemminger err = (*set)(br, val); 488d4698f7SStephen Hemminger return err ? err : len; 491da177e4SLinus Torvalds } 501da177e4SLinus Torvalds 511da177e4SLinus Torvalds 52*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t forward_delay_show(struct device *d, 5343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 541da177e4SLinus Torvalds { 5543cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 561da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->forward_delay)); 571da177e4SLinus Torvalds } 581da177e4SLinus Torvalds 59*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t forward_delay_store(struct device *d, 6043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 6143cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 621da177e4SLinus Torvalds { 6314f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_forward_delay); 641da177e4SLinus Torvalds } 65*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(forward_delay); 661da177e4SLinus Torvalds 67*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_time_show(struct device *d, struct device_attribute *attr, 6843cb76d9SGreg Kroah-Hartman char *buf) 691da177e4SLinus Torvalds { 701da177e4SLinus Torvalds return sprintf(buf, "%lu\n", 7143cb76d9SGreg Kroah-Hartman jiffies_to_clock_t(to_bridge(d)->hello_time)); 721da177e4SLinus Torvalds } 731da177e4SLinus Torvalds 74*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_time_store(struct device *d, 7543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, const char *buf, 761da177e4SLinus Torvalds size_t len) 771da177e4SLinus Torvalds { 7814f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_hello_time); 791da177e4SLinus Torvalds } 80*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hello_time); 811da177e4SLinus Torvalds 82*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t max_age_show(struct device *d, struct device_attribute *attr, 8343cb76d9SGreg Kroah-Hartman char *buf) 841da177e4SLinus Torvalds { 851da177e4SLinus Torvalds return sprintf(buf, "%lu\n", 8643cb76d9SGreg Kroah-Hartman jiffies_to_clock_t(to_bridge(d)->max_age)); 871da177e4SLinus Torvalds } 881da177e4SLinus Torvalds 89*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t max_age_store(struct device *d, struct device_attribute *attr, 9043cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 911da177e4SLinus Torvalds { 9214f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_max_age); 931da177e4SLinus Torvalds } 94*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(max_age); 951da177e4SLinus Torvalds 96*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t ageing_time_show(struct device *d, 9743cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 981da177e4SLinus Torvalds { 9943cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1001da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->ageing_time)); 1011da177e4SLinus Torvalds } 1021da177e4SLinus Torvalds 1038d4698f7SStephen Hemminger static int set_ageing_time(struct net_bridge *br, unsigned long val) 1041da177e4SLinus Torvalds { 1051da177e4SLinus Torvalds br->ageing_time = clock_t_to_jiffies(val); 1068d4698f7SStephen Hemminger return 0; 1071da177e4SLinus Torvalds } 1081da177e4SLinus Torvalds 109*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t ageing_time_store(struct device *d, 11043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 11143cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 1121da177e4SLinus Torvalds { 11343cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_ageing_time); 1141da177e4SLinus Torvalds } 115*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(ageing_time); 11643cb76d9SGreg Kroah-Hartman 117*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t stp_state_show(struct device *d, 11843cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 1191da177e4SLinus Torvalds { 12043cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1211da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->stp_enabled); 1221da177e4SLinus Torvalds } 1231da177e4SLinus Torvalds 1241da177e4SLinus Torvalds 125*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t stp_state_store(struct device *d, 12643cb76d9SGreg Kroah-Hartman struct device_attribute *attr, const char *buf, 12743cb76d9SGreg Kroah-Hartman size_t len) 1281da177e4SLinus Torvalds { 12917120889SStephen Hemminger struct net_bridge *br = to_bridge(d); 13017120889SStephen Hemminger char *endp; 13117120889SStephen Hemminger unsigned long val; 13217120889SStephen Hemminger 133cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 13417120889SStephen Hemminger return -EPERM; 13517120889SStephen Hemminger 13617120889SStephen Hemminger val = simple_strtoul(buf, &endp, 0); 13717120889SStephen Hemminger if (endp == buf) 13817120889SStephen Hemminger return -EINVAL; 13917120889SStephen Hemminger 140af38f298SEric W. Biederman if (!rtnl_trylock()) 141af38f298SEric W. Biederman return restart_syscall(); 14217120889SStephen Hemminger br_stp_set_enabled(br, val); 14317120889SStephen Hemminger rtnl_unlock(); 14417120889SStephen Hemminger 14535b426c3SAl Viro return len; 1461da177e4SLinus Torvalds } 147*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(stp_state); 1481da177e4SLinus Torvalds 149*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_show(struct device *d, 150*fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 151*fbf2671bSsfeldma@cumulusnetworks.com char *buf) 152515853ccSstephen hemminger { 153515853ccSstephen hemminger struct net_bridge *br = to_bridge(d); 154515853ccSstephen hemminger return sprintf(buf, "%#x\n", br->group_fwd_mask); 155515853ccSstephen hemminger } 156515853ccSstephen hemminger 157515853ccSstephen hemminger 158*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_fwd_mask_store(struct device *d, 159*fbf2671bSsfeldma@cumulusnetworks.com struct device_attribute *attr, 160*fbf2671bSsfeldma@cumulusnetworks.com const char *buf, 161515853ccSstephen hemminger size_t len) 162515853ccSstephen hemminger { 163515853ccSstephen hemminger struct net_bridge *br = to_bridge(d); 164515853ccSstephen hemminger char *endp; 165515853ccSstephen hemminger unsigned long val; 166515853ccSstephen hemminger 167cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 168515853ccSstephen hemminger return -EPERM; 169515853ccSstephen hemminger 170515853ccSstephen hemminger val = simple_strtoul(buf, &endp, 0); 171515853ccSstephen hemminger if (endp == buf) 172515853ccSstephen hemminger return -EINVAL; 173515853ccSstephen hemminger 174515853ccSstephen hemminger if (val & BR_GROUPFWD_RESTRICTED) 175515853ccSstephen hemminger return -EINVAL; 176515853ccSstephen hemminger 177515853ccSstephen hemminger br->group_fwd_mask = val; 178515853ccSstephen hemminger 179515853ccSstephen hemminger return len; 180515853ccSstephen hemminger } 181*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_fwd_mask); 182515853ccSstephen hemminger 183*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_show(struct device *d, struct device_attribute *attr, 18443cb76d9SGreg Kroah-Hartman char *buf) 1851da177e4SLinus Torvalds { 18643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1871da177e4SLinus Torvalds return sprintf(buf, "%d\n", 1881da177e4SLinus Torvalds (br->bridge_id.prio[0] << 8) | br->bridge_id.prio[1]); 1891da177e4SLinus Torvalds } 1901da177e4SLinus Torvalds 1918d4698f7SStephen Hemminger static int set_priority(struct net_bridge *br, unsigned long val) 1921da177e4SLinus Torvalds { 1931da177e4SLinus Torvalds br_stp_set_bridge_priority(br, (u16) val); 1948d4698f7SStephen Hemminger return 0; 1951da177e4SLinus Torvalds } 1961da177e4SLinus Torvalds 197*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t priority_store(struct device *d, struct device_attribute *attr, 1981da177e4SLinus Torvalds const char *buf, size_t len) 1991da177e4SLinus Torvalds { 20043cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_priority); 2011da177e4SLinus Torvalds } 202*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(priority); 2031da177e4SLinus Torvalds 204*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_id_show(struct device *d, struct device_attribute *attr, 20543cb76d9SGreg Kroah-Hartman char *buf) 2061da177e4SLinus Torvalds { 20743cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->designated_root); 2081da177e4SLinus Torvalds } 209*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_id); 2101da177e4SLinus Torvalds 211*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t bridge_id_show(struct device *d, struct device_attribute *attr, 21243cb76d9SGreg Kroah-Hartman char *buf) 2131da177e4SLinus Torvalds { 21443cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->bridge_id); 2151da177e4SLinus Torvalds } 216*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(bridge_id); 2171da177e4SLinus Torvalds 218*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_port_show(struct device *d, struct device_attribute *attr, 21943cb76d9SGreg Kroah-Hartman char *buf) 2201da177e4SLinus Torvalds { 22143cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_port); 2221da177e4SLinus Torvalds } 223*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_port); 2241da177e4SLinus Torvalds 225*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t root_path_cost_show(struct device *d, 22643cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2271da177e4SLinus Torvalds { 22843cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_path_cost); 2291da177e4SLinus Torvalds } 230*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(root_path_cost); 2311da177e4SLinus Torvalds 232*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_show(struct device *d, 23343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2341da177e4SLinus Torvalds { 23543cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->topology_change); 2361da177e4SLinus Torvalds } 237*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change); 2381da177e4SLinus Torvalds 239*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_detected_show(struct device *d, 24043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 24143cb76d9SGreg Kroah-Hartman char *buf) 2421da177e4SLinus Torvalds { 24343cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2441da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->topology_change_detected); 2451da177e4SLinus Torvalds } 246*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_detected); 2471da177e4SLinus Torvalds 248*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hello_timer_show(struct device *d, 24943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2501da177e4SLinus Torvalds { 25143cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2521da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->hello_timer)); 2531da177e4SLinus Torvalds } 254*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(hello_timer); 2551da177e4SLinus Torvalds 256*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t tcn_timer_show(struct device *d, struct device_attribute *attr, 25743cb76d9SGreg Kroah-Hartman char *buf) 2581da177e4SLinus Torvalds { 25943cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2601da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->tcn_timer)); 2611da177e4SLinus Torvalds } 262*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(tcn_timer); 2631da177e4SLinus Torvalds 264*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t topology_change_timer_show(struct device *d, 26543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 26643cb76d9SGreg Kroah-Hartman char *buf) 2671da177e4SLinus Torvalds { 26843cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2691da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->topology_change_timer)); 2701da177e4SLinus Torvalds } 271*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(topology_change_timer); 2721da177e4SLinus Torvalds 273*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t gc_timer_show(struct device *d, struct device_attribute *attr, 27443cb76d9SGreg Kroah-Hartman char *buf) 2751da177e4SLinus Torvalds { 27643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2771da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->gc_timer)); 2781da177e4SLinus Torvalds } 279*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RO(gc_timer); 2801da177e4SLinus Torvalds 281*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_show(struct device *d, 28243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 283fda93d92SStephen Hemminger { 28443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 285fda93d92SStephen Hemminger return sprintf(buf, "%x:%x:%x:%x:%x:%x\n", 286fda93d92SStephen Hemminger br->group_addr[0], br->group_addr[1], 287fda93d92SStephen Hemminger br->group_addr[2], br->group_addr[3], 288fda93d92SStephen Hemminger br->group_addr[4], br->group_addr[5]); 289fda93d92SStephen Hemminger } 290fda93d92SStephen Hemminger 291*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t group_addr_store(struct device *d, 29243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 29343cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 294fda93d92SStephen Hemminger { 29543cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2964197f24bSBen Hutchings u8 new_addr[6]; 297fda93d92SStephen Hemminger int i; 298fda93d92SStephen Hemminger 299cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 300fda93d92SStephen Hemminger return -EPERM; 301fda93d92SStephen Hemminger 3024197f24bSBen Hutchings if (sscanf(buf, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx", 303fda93d92SStephen Hemminger &new_addr[0], &new_addr[1], &new_addr[2], 304fda93d92SStephen Hemminger &new_addr[3], &new_addr[4], &new_addr[5]) != 6) 305fda93d92SStephen Hemminger return -EINVAL; 306fda93d92SStephen Hemminger 30746acc460SBen Hutchings if (!is_link_local_ether_addr(new_addr)) 308fda93d92SStephen Hemminger return -EINVAL; 309fda93d92SStephen Hemminger 310f64f9e71SJoe Perches if (new_addr[5] == 1 || /* 802.3x Pause address */ 311f64f9e71SJoe Perches new_addr[5] == 2 || /* 802.3ad Slow protocols */ 312f64f9e71SJoe Perches new_addr[5] == 3) /* 802.1X PAE address */ 313fda93d92SStephen Hemminger return -EINVAL; 314fda93d92SStephen Hemminger 315fda93d92SStephen Hemminger spin_lock_bh(&br->lock); 316fda93d92SStephen Hemminger for (i = 0; i < 6; i++) 317fda93d92SStephen Hemminger br->group_addr[i] = new_addr[i]; 318fda93d92SStephen Hemminger spin_unlock_bh(&br->lock); 319fda93d92SStephen Hemminger return len; 320fda93d92SStephen Hemminger } 321fda93d92SStephen Hemminger 322*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(group_addr); 323fda93d92SStephen Hemminger 324*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t flush_store(struct device *d, 3259cf63747SStephen Hemminger struct device_attribute *attr, 3269cf63747SStephen Hemminger const char *buf, size_t len) 3279cf63747SStephen Hemminger { 3289cf63747SStephen Hemminger struct net_bridge *br = to_bridge(d); 3299cf63747SStephen Hemminger 330cb990503SEric W. Biederman if (!ns_capable(dev_net(br->dev)->user_ns, CAP_NET_ADMIN)) 3319cf63747SStephen Hemminger return -EPERM; 3329cf63747SStephen Hemminger 3339cf63747SStephen Hemminger br_fdb_flush(br); 3349cf63747SStephen Hemminger return len; 3359cf63747SStephen Hemminger } 336*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_WO(flush); 337fda93d92SStephen Hemminger 3380909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 339*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_show(struct device *d, 3400909e117SHerbert Xu struct device_attribute *attr, char *buf) 3410909e117SHerbert Xu { 3420909e117SHerbert Xu struct net_bridge *br = to_bridge(d); 3430909e117SHerbert Xu return sprintf(buf, "%d\n", br->multicast_router); 3440909e117SHerbert Xu } 3450909e117SHerbert Xu 346*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_router_store(struct device *d, 3470909e117SHerbert Xu struct device_attribute *attr, 3480909e117SHerbert Xu const char *buf, size_t len) 3490909e117SHerbert Xu { 3500909e117SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_router); 3510909e117SHerbert Xu } 352*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_router); 353561f1103SHerbert Xu 354*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_show(struct device *d, 355561f1103SHerbert Xu struct device_attribute *attr, 356561f1103SHerbert Xu char *buf) 357561f1103SHerbert Xu { 358561f1103SHerbert Xu struct net_bridge *br = to_bridge(d); 359561f1103SHerbert Xu return sprintf(buf, "%d\n", !br->multicast_disabled); 360561f1103SHerbert Xu } 361561f1103SHerbert Xu 362*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_snooping_store(struct device *d, 363561f1103SHerbert Xu struct device_attribute *attr, 364561f1103SHerbert Xu const char *buf, size_t len) 365561f1103SHerbert Xu { 366561f1103SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_toggle); 367561f1103SHerbert Xu } 368*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_snooping); 369b195167fSHerbert Xu 370*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_use_ifaddr_show(struct device *d, 3711c8ad5bfSCong Wang struct device_attribute *attr, 3721c8ad5bfSCong Wang char *buf) 3731c8ad5bfSCong Wang { 3741c8ad5bfSCong Wang struct net_bridge *br = to_bridge(d); 3751c8ad5bfSCong Wang return sprintf(buf, "%d\n", br->multicast_query_use_ifaddr); 3761c8ad5bfSCong Wang } 3771c8ad5bfSCong Wang 3781c8ad5bfSCong Wang static int set_query_use_ifaddr(struct net_bridge *br, unsigned long val) 3791c8ad5bfSCong Wang { 3801c8ad5bfSCong Wang br->multicast_query_use_ifaddr = !!val; 3811c8ad5bfSCong Wang return 0; 3821c8ad5bfSCong Wang } 3831c8ad5bfSCong Wang 3841c8ad5bfSCong Wang static ssize_t 385*fbf2671bSsfeldma@cumulusnetworks.com multicast_query_use_ifaddr_store(struct device *d, 3861c8ad5bfSCong Wang struct device_attribute *attr, 3871c8ad5bfSCong Wang const char *buf, size_t len) 3881c8ad5bfSCong Wang { 3891c8ad5bfSCong Wang return store_bridge_parm(d, buf, len, set_query_use_ifaddr); 3901c8ad5bfSCong Wang } 391*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_use_ifaddr); 3921c8ad5bfSCong Wang 393*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_show(struct device *d, 394c5c23260SHerbert Xu struct device_attribute *attr, 395c5c23260SHerbert Xu char *buf) 396c5c23260SHerbert Xu { 397c5c23260SHerbert Xu struct net_bridge *br = to_bridge(d); 398c5c23260SHerbert Xu return sprintf(buf, "%d\n", br->multicast_querier); 399c5c23260SHerbert Xu } 400c5c23260SHerbert Xu 401*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_store(struct device *d, 402c5c23260SHerbert Xu struct device_attribute *attr, 403c5c23260SHerbert Xu const char *buf, size_t len) 404c5c23260SHerbert Xu { 405c5c23260SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_querier); 406c5c23260SHerbert Xu } 407*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier); 408c5c23260SHerbert Xu 409*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_show(struct device *d, 410b195167fSHerbert Xu struct device_attribute *attr, char *buf) 411b195167fSHerbert Xu { 412b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 413b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_elasticity); 414b195167fSHerbert Xu } 415b195167fSHerbert Xu 416b195167fSHerbert Xu static int set_elasticity(struct net_bridge *br, unsigned long val) 417b195167fSHerbert Xu { 418b195167fSHerbert Xu br->hash_elasticity = val; 419b195167fSHerbert Xu return 0; 420b195167fSHerbert Xu } 421b195167fSHerbert Xu 422*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_elasticity_store(struct device *d, 423b195167fSHerbert Xu struct device_attribute *attr, 424b195167fSHerbert Xu const char *buf, size_t len) 425b195167fSHerbert Xu { 426b195167fSHerbert Xu return store_bridge_parm(d, buf, len, set_elasticity); 427b195167fSHerbert Xu } 428*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_elasticity); 429b195167fSHerbert Xu 430*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_show(struct device *d, struct device_attribute *attr, 431b195167fSHerbert Xu char *buf) 432b195167fSHerbert Xu { 433b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 434b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_max); 435b195167fSHerbert Xu } 436b195167fSHerbert Xu 437*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t hash_max_store(struct device *d, struct device_attribute *attr, 438b195167fSHerbert Xu const char *buf, size_t len) 439b195167fSHerbert Xu { 440b195167fSHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_hash_max); 441b195167fSHerbert Xu } 442*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(hash_max); 443d902eee4SHerbert Xu 444*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_show(struct device *d, 445d902eee4SHerbert Xu struct device_attribute *attr, 446d902eee4SHerbert Xu char *buf) 447d902eee4SHerbert Xu { 448d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 449d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_last_member_count); 450d902eee4SHerbert Xu } 451d902eee4SHerbert Xu 452d902eee4SHerbert Xu static int set_last_member_count(struct net_bridge *br, unsigned long val) 453d902eee4SHerbert Xu { 454d902eee4SHerbert Xu br->multicast_last_member_count = val; 455d902eee4SHerbert Xu return 0; 456d902eee4SHerbert Xu } 457d902eee4SHerbert Xu 458*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_count_store(struct device *d, 459d902eee4SHerbert Xu struct device_attribute *attr, 460d902eee4SHerbert Xu const char *buf, size_t len) 461d902eee4SHerbert Xu { 462d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_count); 463d902eee4SHerbert Xu } 464*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_count); 465d902eee4SHerbert Xu 466*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_show( 467d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 468d902eee4SHerbert Xu { 469d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 470d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_startup_query_count); 471d902eee4SHerbert Xu } 472d902eee4SHerbert Xu 473d902eee4SHerbert Xu static int set_startup_query_count(struct net_bridge *br, unsigned long val) 474d902eee4SHerbert Xu { 475d902eee4SHerbert Xu br->multicast_startup_query_count = val; 476d902eee4SHerbert Xu return 0; 477d902eee4SHerbert Xu } 478d902eee4SHerbert Xu 479*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_count_store( 480d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 481d902eee4SHerbert Xu size_t len) 482d902eee4SHerbert Xu { 483d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_count); 484d902eee4SHerbert Xu } 485*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_count); 486d902eee4SHerbert Xu 487*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_show( 488d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 489d902eee4SHerbert Xu { 490d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 491d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 492d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_last_member_interval)); 493d902eee4SHerbert Xu } 494d902eee4SHerbert Xu 495d902eee4SHerbert Xu static int set_last_member_interval(struct net_bridge *br, unsigned long val) 496d902eee4SHerbert Xu { 497d902eee4SHerbert Xu br->multicast_last_member_interval = clock_t_to_jiffies(val); 498d902eee4SHerbert Xu return 0; 499d902eee4SHerbert Xu } 500d902eee4SHerbert Xu 501*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_last_member_interval_store( 502d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 503d902eee4SHerbert Xu size_t len) 504d902eee4SHerbert Xu { 505d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_interval); 506d902eee4SHerbert Xu } 507*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_last_member_interval); 508d902eee4SHerbert Xu 509*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_show( 510d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 511d902eee4SHerbert Xu { 512d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 513d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 514d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_membership_interval)); 515d902eee4SHerbert Xu } 516d902eee4SHerbert Xu 517d902eee4SHerbert Xu static int set_membership_interval(struct net_bridge *br, unsigned long val) 518d902eee4SHerbert Xu { 519d902eee4SHerbert Xu br->multicast_membership_interval = clock_t_to_jiffies(val); 520d902eee4SHerbert Xu return 0; 521d902eee4SHerbert Xu } 522d902eee4SHerbert Xu 523*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_membership_interval_store( 524d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 525d902eee4SHerbert Xu size_t len) 526d902eee4SHerbert Xu { 527d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_membership_interval); 528d902eee4SHerbert Xu } 529*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_membership_interval); 530d902eee4SHerbert Xu 531*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_show(struct device *d, 532d902eee4SHerbert Xu struct device_attribute *attr, 533d902eee4SHerbert Xu char *buf) 534d902eee4SHerbert Xu { 535d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 536d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 537d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_querier_interval)); 538d902eee4SHerbert Xu } 539d902eee4SHerbert Xu 540d902eee4SHerbert Xu static int set_querier_interval(struct net_bridge *br, unsigned long val) 541d902eee4SHerbert Xu { 542d902eee4SHerbert Xu br->multicast_querier_interval = clock_t_to_jiffies(val); 543d902eee4SHerbert Xu return 0; 544d902eee4SHerbert Xu } 545d902eee4SHerbert Xu 546*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_querier_interval_store(struct device *d, 547d902eee4SHerbert Xu struct device_attribute *attr, 548d902eee4SHerbert Xu const char *buf, size_t len) 549d902eee4SHerbert Xu { 550d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_querier_interval); 551d902eee4SHerbert Xu } 552*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_querier_interval); 553d902eee4SHerbert Xu 554*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_show(struct device *d, 555d902eee4SHerbert Xu struct device_attribute *attr, 556d902eee4SHerbert Xu char *buf) 557d902eee4SHerbert Xu { 558d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 559d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 560d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_interval)); 561d902eee4SHerbert Xu } 562d902eee4SHerbert Xu 563d902eee4SHerbert Xu static int set_query_interval(struct net_bridge *br, unsigned long val) 564d902eee4SHerbert Xu { 565d902eee4SHerbert Xu br->multicast_query_interval = clock_t_to_jiffies(val); 566d902eee4SHerbert Xu return 0; 567d902eee4SHerbert Xu } 568d902eee4SHerbert Xu 569*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_interval_store(struct device *d, 570d902eee4SHerbert Xu struct device_attribute *attr, 571d902eee4SHerbert Xu const char *buf, size_t len) 572d902eee4SHerbert Xu { 573d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_interval); 574d902eee4SHerbert Xu } 575*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_interval); 576d902eee4SHerbert Xu 577*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_show( 578d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 579d902eee4SHerbert Xu { 580d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 581d902eee4SHerbert Xu return sprintf( 582d902eee4SHerbert Xu buf, "%lu\n", 583d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_response_interval)); 584d902eee4SHerbert Xu } 585d902eee4SHerbert Xu 586d902eee4SHerbert Xu static int set_query_response_interval(struct net_bridge *br, unsigned long val) 587d902eee4SHerbert Xu { 588d902eee4SHerbert Xu br->multicast_query_response_interval = clock_t_to_jiffies(val); 589d902eee4SHerbert Xu return 0; 590d902eee4SHerbert Xu } 591d902eee4SHerbert Xu 592*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_query_response_interval_store( 593d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 594d902eee4SHerbert Xu size_t len) 595d902eee4SHerbert Xu { 596d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_response_interval); 597d902eee4SHerbert Xu } 598*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_query_response_interval); 599d902eee4SHerbert Xu 600*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_show( 601d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 602d902eee4SHerbert Xu { 603d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 604d902eee4SHerbert Xu return sprintf( 605d902eee4SHerbert Xu buf, "%lu\n", 606d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_startup_query_interval)); 607d902eee4SHerbert Xu } 608d902eee4SHerbert Xu 609d902eee4SHerbert Xu static int set_startup_query_interval(struct net_bridge *br, unsigned long val) 610d902eee4SHerbert Xu { 611d902eee4SHerbert Xu br->multicast_startup_query_interval = clock_t_to_jiffies(val); 612d902eee4SHerbert Xu return 0; 613d902eee4SHerbert Xu } 614d902eee4SHerbert Xu 615*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t multicast_startup_query_interval_store( 616d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 617d902eee4SHerbert Xu size_t len) 618d902eee4SHerbert Xu { 619d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_interval); 620d902eee4SHerbert Xu } 621*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(multicast_startup_query_interval); 6220909e117SHerbert Xu #endif 6234df53d8bSPatrick McHardy #ifdef CONFIG_BRIDGE_NETFILTER 624*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_show( 6254df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6264df53d8bSPatrick McHardy { 6274df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6284df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_iptables); 6294df53d8bSPatrick McHardy } 6304df53d8bSPatrick McHardy 6314df53d8bSPatrick McHardy static int set_nf_call_iptables(struct net_bridge *br, unsigned long val) 6324df53d8bSPatrick McHardy { 6334df53d8bSPatrick McHardy br->nf_call_iptables = val ? true : false; 6344df53d8bSPatrick McHardy return 0; 6354df53d8bSPatrick McHardy } 6364df53d8bSPatrick McHardy 637*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_iptables_store( 6384df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6394df53d8bSPatrick McHardy size_t len) 6404df53d8bSPatrick McHardy { 6414df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_iptables); 6424df53d8bSPatrick McHardy } 643*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_iptables); 6444df53d8bSPatrick McHardy 645*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_show( 6464df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6474df53d8bSPatrick McHardy { 6484df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6494df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_ip6tables); 6504df53d8bSPatrick McHardy } 6514df53d8bSPatrick McHardy 6524df53d8bSPatrick McHardy static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val) 6534df53d8bSPatrick McHardy { 6544df53d8bSPatrick McHardy br->nf_call_ip6tables = val ? true : false; 6554df53d8bSPatrick McHardy return 0; 6564df53d8bSPatrick McHardy } 6574df53d8bSPatrick McHardy 658*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_ip6tables_store( 6594df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6604df53d8bSPatrick McHardy size_t len) 6614df53d8bSPatrick McHardy { 6624df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_ip6tables); 6634df53d8bSPatrick McHardy } 664*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_ip6tables); 6654df53d8bSPatrick McHardy 666*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_show( 6674df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6684df53d8bSPatrick McHardy { 6694df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6704df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_arptables); 6714df53d8bSPatrick McHardy } 6724df53d8bSPatrick McHardy 6734df53d8bSPatrick McHardy static int set_nf_call_arptables(struct net_bridge *br, unsigned long val) 6744df53d8bSPatrick McHardy { 6754df53d8bSPatrick McHardy br->nf_call_arptables = val ? true : false; 6764df53d8bSPatrick McHardy return 0; 6774df53d8bSPatrick McHardy } 6784df53d8bSPatrick McHardy 679*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t nf_call_arptables_store( 6804df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6814df53d8bSPatrick McHardy size_t len) 6824df53d8bSPatrick McHardy { 6834df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_arptables); 6844df53d8bSPatrick McHardy } 685*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(nf_call_arptables); 6864df53d8bSPatrick McHardy #endif 687243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 688*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_show(struct device *d, 689243a2e63SVlad Yasevich struct device_attribute *attr, 690243a2e63SVlad Yasevich char *buf) 691243a2e63SVlad Yasevich { 692243a2e63SVlad Yasevich struct net_bridge *br = to_bridge(d); 693243a2e63SVlad Yasevich return sprintf(buf, "%d\n", br->vlan_enabled); 694243a2e63SVlad Yasevich } 695243a2e63SVlad Yasevich 696*fbf2671bSsfeldma@cumulusnetworks.com static ssize_t vlan_filtering_store(struct device *d, 697243a2e63SVlad Yasevich struct device_attribute *attr, 698243a2e63SVlad Yasevich const char *buf, size_t len) 699243a2e63SVlad Yasevich { 700243a2e63SVlad Yasevich return store_bridge_parm(d, buf, len, br_vlan_filter_toggle); 701243a2e63SVlad Yasevich } 702*fbf2671bSsfeldma@cumulusnetworks.com static DEVICE_ATTR_RW(vlan_filtering); 703243a2e63SVlad Yasevich #endif 7040909e117SHerbert Xu 7051da177e4SLinus Torvalds static struct attribute *bridge_attrs[] = { 70643cb76d9SGreg Kroah-Hartman &dev_attr_forward_delay.attr, 70743cb76d9SGreg Kroah-Hartman &dev_attr_hello_time.attr, 70843cb76d9SGreg Kroah-Hartman &dev_attr_max_age.attr, 70943cb76d9SGreg Kroah-Hartman &dev_attr_ageing_time.attr, 71043cb76d9SGreg Kroah-Hartman &dev_attr_stp_state.attr, 711515853ccSstephen hemminger &dev_attr_group_fwd_mask.attr, 71243cb76d9SGreg Kroah-Hartman &dev_attr_priority.attr, 71343cb76d9SGreg Kroah-Hartman &dev_attr_bridge_id.attr, 71443cb76d9SGreg Kroah-Hartman &dev_attr_root_id.attr, 71543cb76d9SGreg Kroah-Hartman &dev_attr_root_path_cost.attr, 71643cb76d9SGreg Kroah-Hartman &dev_attr_root_port.attr, 71743cb76d9SGreg Kroah-Hartman &dev_attr_topology_change.attr, 71843cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_detected.attr, 71943cb76d9SGreg Kroah-Hartman &dev_attr_hello_timer.attr, 72043cb76d9SGreg Kroah-Hartman &dev_attr_tcn_timer.attr, 72143cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_timer.attr, 72243cb76d9SGreg Kroah-Hartman &dev_attr_gc_timer.attr, 72343cb76d9SGreg Kroah-Hartman &dev_attr_group_addr.attr, 7249cf63747SStephen Hemminger &dev_attr_flush.attr, 7250909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 7260909e117SHerbert Xu &dev_attr_multicast_router.attr, 727561f1103SHerbert Xu &dev_attr_multicast_snooping.attr, 728c5c23260SHerbert Xu &dev_attr_multicast_querier.attr, 7291c8ad5bfSCong Wang &dev_attr_multicast_query_use_ifaddr.attr, 730b195167fSHerbert Xu &dev_attr_hash_elasticity.attr, 731b195167fSHerbert Xu &dev_attr_hash_max.attr, 732d902eee4SHerbert Xu &dev_attr_multicast_last_member_count.attr, 733d902eee4SHerbert Xu &dev_attr_multicast_startup_query_count.attr, 734d902eee4SHerbert Xu &dev_attr_multicast_last_member_interval.attr, 735d902eee4SHerbert Xu &dev_attr_multicast_membership_interval.attr, 736d902eee4SHerbert Xu &dev_attr_multicast_querier_interval.attr, 737d902eee4SHerbert Xu &dev_attr_multicast_query_interval.attr, 738d902eee4SHerbert Xu &dev_attr_multicast_query_response_interval.attr, 739d902eee4SHerbert Xu &dev_attr_multicast_startup_query_interval.attr, 7400909e117SHerbert Xu #endif 7414df53d8bSPatrick McHardy #ifdef CONFIG_BRIDGE_NETFILTER 7424df53d8bSPatrick McHardy &dev_attr_nf_call_iptables.attr, 7434df53d8bSPatrick McHardy &dev_attr_nf_call_ip6tables.attr, 7444df53d8bSPatrick McHardy &dev_attr_nf_call_arptables.attr, 7454df53d8bSPatrick McHardy #endif 746243a2e63SVlad Yasevich #ifdef CONFIG_BRIDGE_VLAN_FILTERING 747243a2e63SVlad Yasevich &dev_attr_vlan_filtering.attr, 748243a2e63SVlad Yasevich #endif 7491da177e4SLinus Torvalds NULL 7501da177e4SLinus Torvalds }; 7511da177e4SLinus Torvalds 7521da177e4SLinus Torvalds static struct attribute_group bridge_group = { 7531da177e4SLinus Torvalds .name = SYSFS_BRIDGE_ATTR, 7541da177e4SLinus Torvalds .attrs = bridge_attrs, 7551da177e4SLinus Torvalds }; 7561da177e4SLinus Torvalds 7571da177e4SLinus Torvalds /* 7581da177e4SLinus Torvalds * Export the forwarding information table as a binary file 7591da177e4SLinus Torvalds * The records are struct __fdb_entry. 7601da177e4SLinus Torvalds * 7611da177e4SLinus Torvalds * Returns the number of bytes read. 7621da177e4SLinus Torvalds */ 7632c3c8beaSChris Wright static ssize_t brforward_read(struct file *filp, struct kobject *kobj, 76491a69029SZhang Rui struct bin_attribute *bin_attr, 76591a69029SZhang Rui char *buf, loff_t off, size_t count) 7661da177e4SLinus Torvalds { 76743cb76d9SGreg Kroah-Hartman struct device *dev = to_dev(kobj); 76843cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(dev); 7691da177e4SLinus Torvalds int n; 7701da177e4SLinus Torvalds 7711da177e4SLinus Torvalds /* must read whole records */ 7721da177e4SLinus Torvalds if (off % sizeof(struct __fdb_entry) != 0) 7731da177e4SLinus Torvalds return -EINVAL; 7741da177e4SLinus Torvalds 7751da177e4SLinus Torvalds n = br_fdb_fillbuf(br, buf, 7761da177e4SLinus Torvalds count / sizeof(struct __fdb_entry), 7771da177e4SLinus Torvalds off / sizeof(struct __fdb_entry)); 7781da177e4SLinus Torvalds 7791da177e4SLinus Torvalds if (n > 0) 7801da177e4SLinus Torvalds n *= sizeof(struct __fdb_entry); 7811da177e4SLinus Torvalds 7821da177e4SLinus Torvalds return n; 7831da177e4SLinus Torvalds } 7841da177e4SLinus Torvalds 7851da177e4SLinus Torvalds static struct bin_attribute bridge_forward = { 7861da177e4SLinus Torvalds .attr = { .name = SYSFS_BRIDGE_FDB, 7877b595756STejun Heo .mode = S_IRUGO, }, 7881da177e4SLinus Torvalds .read = brforward_read, 7891da177e4SLinus Torvalds }; 7901da177e4SLinus Torvalds 7911da177e4SLinus Torvalds /* 7921da177e4SLinus Torvalds * Add entries in sysfs onto the existing network class device 7931da177e4SLinus Torvalds * for the bridge. 7941da177e4SLinus Torvalds * Adds a attribute group "bridge" containing tuning parameters. 7951da177e4SLinus Torvalds * Binary attribute containing the forward table 7961da177e4SLinus Torvalds * Sub directory to hold links to interfaces. 7971da177e4SLinus Torvalds * 7981da177e4SLinus Torvalds * Note: the ifobj exists only to be a subdirectory 7991da177e4SLinus Torvalds * to hold links. The ifobj exists in same data structure 8001da177e4SLinus Torvalds * as it's parent the bridge so reference counting works. 8011da177e4SLinus Torvalds */ 8021da177e4SLinus Torvalds int br_sysfs_addbr(struct net_device *dev) 8031da177e4SLinus Torvalds { 80443cb76d9SGreg Kroah-Hartman struct kobject *brobj = &dev->dev.kobj; 8051da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 8061da177e4SLinus Torvalds int err; 8071da177e4SLinus Torvalds 8081da177e4SLinus Torvalds err = sysfs_create_group(brobj, &bridge_group); 8091da177e4SLinus Torvalds if (err) { 8101da177e4SLinus Torvalds pr_info("%s: can't create group %s/%s\n", 8110dc47877SHarvey Harrison __func__, dev->name, bridge_group.name); 8121da177e4SLinus Torvalds goto out1; 8131da177e4SLinus Torvalds } 8141da177e4SLinus Torvalds 8151da177e4SLinus Torvalds err = sysfs_create_bin_file(brobj, &bridge_forward); 8161da177e4SLinus Torvalds if (err) { 8171842c4beSRandy Dunlap pr_info("%s: can't create attribute file %s/%s\n", 8180dc47877SHarvey Harrison __func__, dev->name, bridge_forward.attr.name); 8191da177e4SLinus Torvalds goto out2; 8201da177e4SLinus Torvalds } 8211da177e4SLinus Torvalds 82243b98c4aSGreg Kroah-Hartman br->ifobj = kobject_create_and_add(SYSFS_BRIDGE_PORT_SUBDIR, brobj); 82343b98c4aSGreg Kroah-Hartman if (!br->ifobj) { 8241da177e4SLinus Torvalds pr_info("%s: can't add kobject (directory) %s/%s\n", 8250dc47877SHarvey Harrison __func__, dev->name, SYSFS_BRIDGE_PORT_SUBDIR); 8261da177e4SLinus Torvalds goto out3; 8271da177e4SLinus Torvalds } 8281da177e4SLinus Torvalds return 0; 8291da177e4SLinus Torvalds out3: 83043cb76d9SGreg Kroah-Hartman sysfs_remove_bin_file(&dev->dev.kobj, &bridge_forward); 8311da177e4SLinus Torvalds out2: 83243cb76d9SGreg Kroah-Hartman sysfs_remove_group(&dev->dev.kobj, &bridge_group); 8331da177e4SLinus Torvalds out1: 8341da177e4SLinus Torvalds return err; 8351da177e4SLinus Torvalds 8361da177e4SLinus Torvalds } 8371da177e4SLinus Torvalds 8381da177e4SLinus Torvalds void br_sysfs_delbr(struct net_device *dev) 8391da177e4SLinus Torvalds { 84043cb76d9SGreg Kroah-Hartman struct kobject *kobj = &dev->dev.kobj; 8411da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 8421da177e4SLinus Torvalds 84378a2d906SGreg Kroah-Hartman kobject_put(br->ifobj); 8441da177e4SLinus Torvalds sysfs_remove_bin_file(kobj, &bridge_forward); 8451da177e4SLinus Torvalds sysfs_remove_group(kobj, &bridge_group); 8461da177e4SLinus Torvalds } 847