11da177e4SLinus Torvalds /* 21da177e4SLinus Torvalds * Sysfs attributes of bridge ports 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> 171da177e4SLinus Torvalds #include <linux/if_bridge.h> 181da177e4SLinus Torvalds #include <linux/rtnetlink.h> 191da177e4SLinus Torvalds #include <linux/spinlock.h> 201da177e4SLinus Torvalds #include <linux/times.h> 211da177e4SLinus Torvalds 221da177e4SLinus Torvalds #include "br_private.h" 231da177e4SLinus Torvalds 2443cb76d9SGreg Kroah-Hartman #define to_dev(obj) container_of(obj, struct device, kobj) 25524ad0a7SWang Chen #define to_bridge(cd) ((struct net_bridge *)netdev_priv(to_net_dev(cd))) 261da177e4SLinus Torvalds 271da177e4SLinus Torvalds /* 281da177e4SLinus Torvalds * Common code for storing bridge parameters. 291da177e4SLinus Torvalds */ 3043cb76d9SGreg Kroah-Hartman static ssize_t store_bridge_parm(struct device *d, 311da177e4SLinus Torvalds const char *buf, size_t len, 328d4698f7SStephen Hemminger int (*set)(struct net_bridge *, unsigned long)) 331da177e4SLinus Torvalds { 3443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 351da177e4SLinus Torvalds char *endp; 361da177e4SLinus Torvalds unsigned long val; 378d4698f7SStephen Hemminger int err; 381da177e4SLinus Torvalds 391da177e4SLinus Torvalds if (!capable(CAP_NET_ADMIN)) 401da177e4SLinus Torvalds return -EPERM; 411da177e4SLinus Torvalds 421da177e4SLinus Torvalds val = simple_strtoul(buf, &endp, 0); 431da177e4SLinus Torvalds if (endp == buf) 441da177e4SLinus Torvalds return -EINVAL; 451da177e4SLinus Torvalds 468d4698f7SStephen Hemminger err = (*set)(br, val); 478d4698f7SStephen Hemminger return err ? err : len; 481da177e4SLinus Torvalds } 491da177e4SLinus Torvalds 501da177e4SLinus Torvalds 5143cb76d9SGreg Kroah-Hartman static ssize_t show_forward_delay(struct device *d, 5243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 531da177e4SLinus Torvalds { 5443cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 551da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->forward_delay)); 561da177e4SLinus Torvalds } 571da177e4SLinus Torvalds 5843cb76d9SGreg Kroah-Hartman static ssize_t store_forward_delay(struct device *d, 5943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 6043cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 611da177e4SLinus Torvalds { 62*14f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_forward_delay); 631da177e4SLinus Torvalds } 6443cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(forward_delay, S_IRUGO | S_IWUSR, 651da177e4SLinus Torvalds show_forward_delay, store_forward_delay); 661da177e4SLinus Torvalds 6743cb76d9SGreg Kroah-Hartman static ssize_t show_hello_time(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 7443cb76d9SGreg Kroah-Hartman static ssize_t store_hello_time(struct device *d, 7543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, const char *buf, 761da177e4SLinus Torvalds size_t len) 771da177e4SLinus Torvalds { 78*14f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_hello_time); 791da177e4SLinus Torvalds } 8043cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(hello_time, S_IRUGO | S_IWUSR, show_hello_time, 811da177e4SLinus Torvalds store_hello_time); 821da177e4SLinus Torvalds 8343cb76d9SGreg Kroah-Hartman static ssize_t show_max_age(struct device *d, struct device_attribute *attr, 8443cb76d9SGreg Kroah-Hartman char *buf) 851da177e4SLinus Torvalds { 861da177e4SLinus Torvalds return sprintf(buf, "%lu\n", 8743cb76d9SGreg Kroah-Hartman jiffies_to_clock_t(to_bridge(d)->max_age)); 881da177e4SLinus Torvalds } 891da177e4SLinus Torvalds 9043cb76d9SGreg Kroah-Hartman static ssize_t store_max_age(struct device *d, struct device_attribute *attr, 9143cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 921da177e4SLinus Torvalds { 93*14f98f25Sstephen hemminger return store_bridge_parm(d, buf, len, br_set_max_age); 941da177e4SLinus Torvalds } 9543cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(max_age, S_IRUGO | S_IWUSR, show_max_age, store_max_age); 961da177e4SLinus Torvalds 9743cb76d9SGreg Kroah-Hartman static ssize_t show_ageing_time(struct device *d, 9843cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 991da177e4SLinus Torvalds { 10043cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1011da177e4SLinus Torvalds return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->ageing_time)); 1021da177e4SLinus Torvalds } 1031da177e4SLinus Torvalds 1048d4698f7SStephen Hemminger static int set_ageing_time(struct net_bridge *br, unsigned long val) 1051da177e4SLinus Torvalds { 1061da177e4SLinus Torvalds br->ageing_time = clock_t_to_jiffies(val); 1078d4698f7SStephen Hemminger return 0; 1081da177e4SLinus Torvalds } 1091da177e4SLinus Torvalds 11043cb76d9SGreg Kroah-Hartman static ssize_t store_ageing_time(struct device *d, 11143cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 11243cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 1131da177e4SLinus Torvalds { 11443cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_ageing_time); 1151da177e4SLinus Torvalds } 11643cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(ageing_time, S_IRUGO | S_IWUSR, show_ageing_time, 1171da177e4SLinus Torvalds store_ageing_time); 11843cb76d9SGreg Kroah-Hartman 11943cb76d9SGreg Kroah-Hartman static ssize_t show_stp_state(struct device *d, 12043cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 1211da177e4SLinus Torvalds { 12243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1231da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->stp_enabled); 1241da177e4SLinus Torvalds } 1251da177e4SLinus Torvalds 1261da177e4SLinus Torvalds 12743cb76d9SGreg Kroah-Hartman static ssize_t store_stp_state(struct device *d, 12843cb76d9SGreg Kroah-Hartman struct device_attribute *attr, const char *buf, 12943cb76d9SGreg Kroah-Hartman size_t len) 1301da177e4SLinus Torvalds { 13117120889SStephen Hemminger struct net_bridge *br = to_bridge(d); 13217120889SStephen Hemminger char *endp; 13317120889SStephen Hemminger unsigned long val; 13417120889SStephen Hemminger 13517120889SStephen Hemminger if (!capable(CAP_NET_ADMIN)) 13617120889SStephen Hemminger return -EPERM; 13717120889SStephen Hemminger 13817120889SStephen Hemminger val = simple_strtoul(buf, &endp, 0); 13917120889SStephen Hemminger if (endp == buf) 14017120889SStephen Hemminger return -EINVAL; 14117120889SStephen Hemminger 142af38f298SEric W. Biederman if (!rtnl_trylock()) 143af38f298SEric W. Biederman return restart_syscall(); 14417120889SStephen Hemminger br_stp_set_enabled(br, val); 14517120889SStephen Hemminger rtnl_unlock(); 14617120889SStephen Hemminger 14735b426c3SAl Viro return len; 1481da177e4SLinus Torvalds } 14943cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(stp_state, S_IRUGO | S_IWUSR, show_stp_state, 1501da177e4SLinus Torvalds store_stp_state); 1511da177e4SLinus Torvalds 15243cb76d9SGreg Kroah-Hartman static ssize_t show_priority(struct device *d, struct device_attribute *attr, 15343cb76d9SGreg Kroah-Hartman char *buf) 1541da177e4SLinus Torvalds { 15543cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 1561da177e4SLinus Torvalds return sprintf(buf, "%d\n", 1571da177e4SLinus Torvalds (br->bridge_id.prio[0] << 8) | br->bridge_id.prio[1]); 1581da177e4SLinus Torvalds } 1591da177e4SLinus Torvalds 1608d4698f7SStephen Hemminger static int set_priority(struct net_bridge *br, unsigned long val) 1611da177e4SLinus Torvalds { 1621da177e4SLinus Torvalds br_stp_set_bridge_priority(br, (u16) val); 1638d4698f7SStephen Hemminger return 0; 1641da177e4SLinus Torvalds } 1651da177e4SLinus Torvalds 16643cb76d9SGreg Kroah-Hartman static ssize_t store_priority(struct device *d, struct device_attribute *attr, 1671da177e4SLinus Torvalds const char *buf, size_t len) 1681da177e4SLinus Torvalds { 16943cb76d9SGreg Kroah-Hartman return store_bridge_parm(d, buf, len, set_priority); 1701da177e4SLinus Torvalds } 17143cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(priority, S_IRUGO | S_IWUSR, show_priority, store_priority); 1721da177e4SLinus Torvalds 17343cb76d9SGreg Kroah-Hartman static ssize_t show_root_id(struct device *d, struct device_attribute *attr, 17443cb76d9SGreg Kroah-Hartman char *buf) 1751da177e4SLinus Torvalds { 17643cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->designated_root); 1771da177e4SLinus Torvalds } 17843cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(root_id, S_IRUGO, show_root_id, NULL); 1791da177e4SLinus Torvalds 18043cb76d9SGreg Kroah-Hartman static ssize_t show_bridge_id(struct device *d, struct device_attribute *attr, 18143cb76d9SGreg Kroah-Hartman char *buf) 1821da177e4SLinus Torvalds { 18343cb76d9SGreg Kroah-Hartman return br_show_bridge_id(buf, &to_bridge(d)->bridge_id); 1841da177e4SLinus Torvalds } 18543cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(bridge_id, S_IRUGO, show_bridge_id, NULL); 1861da177e4SLinus Torvalds 18743cb76d9SGreg Kroah-Hartman static ssize_t show_root_port(struct device *d, struct device_attribute *attr, 18843cb76d9SGreg Kroah-Hartman char *buf) 1891da177e4SLinus Torvalds { 19043cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_port); 1911da177e4SLinus Torvalds } 19243cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(root_port, S_IRUGO, show_root_port, NULL); 1931da177e4SLinus Torvalds 19443cb76d9SGreg Kroah-Hartman static ssize_t show_root_path_cost(struct device *d, 19543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 1961da177e4SLinus Torvalds { 19743cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->root_path_cost); 1981da177e4SLinus Torvalds } 19943cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(root_path_cost, S_IRUGO, show_root_path_cost, NULL); 2001da177e4SLinus Torvalds 20143cb76d9SGreg Kroah-Hartman static ssize_t show_topology_change(struct device *d, 20243cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2031da177e4SLinus Torvalds { 20443cb76d9SGreg Kroah-Hartman return sprintf(buf, "%d\n", to_bridge(d)->topology_change); 2051da177e4SLinus Torvalds } 20643cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(topology_change, S_IRUGO, show_topology_change, NULL); 2071da177e4SLinus Torvalds 20843cb76d9SGreg Kroah-Hartman static ssize_t show_topology_change_detected(struct device *d, 20943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 21043cb76d9SGreg Kroah-Hartman char *buf) 2111da177e4SLinus Torvalds { 21243cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2131da177e4SLinus Torvalds return sprintf(buf, "%d\n", br->topology_change_detected); 2141da177e4SLinus Torvalds } 21543cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(topology_change_detected, S_IRUGO, 21643cb76d9SGreg Kroah-Hartman show_topology_change_detected, NULL); 2171da177e4SLinus Torvalds 21843cb76d9SGreg Kroah-Hartman static ssize_t show_hello_timer(struct device *d, 21943cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 2201da177e4SLinus Torvalds { 22143cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2221da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->hello_timer)); 2231da177e4SLinus Torvalds } 22443cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(hello_timer, S_IRUGO, show_hello_timer, NULL); 2251da177e4SLinus Torvalds 22643cb76d9SGreg Kroah-Hartman static ssize_t show_tcn_timer(struct device *d, struct device_attribute *attr, 22743cb76d9SGreg Kroah-Hartman char *buf) 2281da177e4SLinus Torvalds { 22943cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2301da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->tcn_timer)); 2311da177e4SLinus Torvalds } 23243cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(tcn_timer, S_IRUGO, show_tcn_timer, NULL); 2331da177e4SLinus Torvalds 23443cb76d9SGreg Kroah-Hartman static ssize_t show_topology_change_timer(struct device *d, 23543cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 23643cb76d9SGreg Kroah-Hartman char *buf) 2371da177e4SLinus Torvalds { 23843cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2391da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->topology_change_timer)); 2401da177e4SLinus Torvalds } 24143cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(topology_change_timer, S_IRUGO, show_topology_change_timer, 24243cb76d9SGreg Kroah-Hartman NULL); 2431da177e4SLinus Torvalds 24443cb76d9SGreg Kroah-Hartman static ssize_t show_gc_timer(struct device *d, struct device_attribute *attr, 24543cb76d9SGreg Kroah-Hartman char *buf) 2461da177e4SLinus Torvalds { 24743cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 2481da177e4SLinus Torvalds return sprintf(buf, "%ld\n", br_timer_value(&br->gc_timer)); 2491da177e4SLinus Torvalds } 25043cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(gc_timer, S_IRUGO, show_gc_timer, NULL); 2511da177e4SLinus Torvalds 25243cb76d9SGreg Kroah-Hartman static ssize_t show_group_addr(struct device *d, 25343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, char *buf) 254fda93d92SStephen Hemminger { 25543cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 256fda93d92SStephen Hemminger return sprintf(buf, "%x:%x:%x:%x:%x:%x\n", 257fda93d92SStephen Hemminger br->group_addr[0], br->group_addr[1], 258fda93d92SStephen Hemminger br->group_addr[2], br->group_addr[3], 259fda93d92SStephen Hemminger br->group_addr[4], br->group_addr[5]); 260fda93d92SStephen Hemminger } 261fda93d92SStephen Hemminger 26243cb76d9SGreg Kroah-Hartman static ssize_t store_group_addr(struct device *d, 26343cb76d9SGreg Kroah-Hartman struct device_attribute *attr, 26443cb76d9SGreg Kroah-Hartman const char *buf, size_t len) 265fda93d92SStephen Hemminger { 26643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(d); 267fda93d92SStephen Hemminger unsigned new_addr[6]; 268fda93d92SStephen Hemminger int i; 269fda93d92SStephen Hemminger 270fda93d92SStephen Hemminger if (!capable(CAP_NET_ADMIN)) 271fda93d92SStephen Hemminger return -EPERM; 272fda93d92SStephen Hemminger 273fda93d92SStephen Hemminger if (sscanf(buf, "%x:%x:%x:%x:%x:%x", 274fda93d92SStephen Hemminger &new_addr[0], &new_addr[1], &new_addr[2], 275fda93d92SStephen Hemminger &new_addr[3], &new_addr[4], &new_addr[5]) != 6) 276fda93d92SStephen Hemminger return -EINVAL; 277fda93d92SStephen Hemminger 278fda93d92SStephen Hemminger /* Must be 01:80:c2:00:00:0X */ 279fda93d92SStephen Hemminger for (i = 0; i < 5; i++) 280fda93d92SStephen Hemminger if (new_addr[i] != br_group_address[i]) 281fda93d92SStephen Hemminger return -EINVAL; 282fda93d92SStephen Hemminger 283fda93d92SStephen Hemminger if (new_addr[5] & ~0xf) 284fda93d92SStephen Hemminger return -EINVAL; 285fda93d92SStephen Hemminger 286f64f9e71SJoe Perches if (new_addr[5] == 1 || /* 802.3x Pause address */ 287f64f9e71SJoe Perches new_addr[5] == 2 || /* 802.3ad Slow protocols */ 288f64f9e71SJoe Perches new_addr[5] == 3) /* 802.1X PAE address */ 289fda93d92SStephen Hemminger return -EINVAL; 290fda93d92SStephen Hemminger 291fda93d92SStephen Hemminger spin_lock_bh(&br->lock); 292fda93d92SStephen Hemminger for (i = 0; i < 6; i++) 293fda93d92SStephen Hemminger br->group_addr[i] = new_addr[i]; 294fda93d92SStephen Hemminger spin_unlock_bh(&br->lock); 295fda93d92SStephen Hemminger return len; 296fda93d92SStephen Hemminger } 297fda93d92SStephen Hemminger 29843cb76d9SGreg Kroah-Hartman static DEVICE_ATTR(group_addr, S_IRUGO | S_IWUSR, 299fda93d92SStephen Hemminger show_group_addr, store_group_addr); 300fda93d92SStephen Hemminger 3019cf63747SStephen Hemminger static ssize_t store_flush(struct device *d, 3029cf63747SStephen Hemminger struct device_attribute *attr, 3039cf63747SStephen Hemminger const char *buf, size_t len) 3049cf63747SStephen Hemminger { 3059cf63747SStephen Hemminger struct net_bridge *br = to_bridge(d); 3069cf63747SStephen Hemminger 3079cf63747SStephen Hemminger if (!capable(CAP_NET_ADMIN)) 3089cf63747SStephen Hemminger return -EPERM; 3099cf63747SStephen Hemminger 3109cf63747SStephen Hemminger br_fdb_flush(br); 3119cf63747SStephen Hemminger return len; 3129cf63747SStephen Hemminger } 3139cf63747SStephen Hemminger static DEVICE_ATTR(flush, S_IWUSR, NULL, store_flush); 314fda93d92SStephen Hemminger 3150909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 3160909e117SHerbert Xu static ssize_t show_multicast_router(struct device *d, 3170909e117SHerbert Xu struct device_attribute *attr, char *buf) 3180909e117SHerbert Xu { 3190909e117SHerbert Xu struct net_bridge *br = to_bridge(d); 3200909e117SHerbert Xu return sprintf(buf, "%d\n", br->multicast_router); 3210909e117SHerbert Xu } 3220909e117SHerbert Xu 3230909e117SHerbert Xu static ssize_t store_multicast_router(struct device *d, 3240909e117SHerbert Xu struct device_attribute *attr, 3250909e117SHerbert Xu const char *buf, size_t len) 3260909e117SHerbert Xu { 3270909e117SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_router); 3280909e117SHerbert Xu } 3290909e117SHerbert Xu static DEVICE_ATTR(multicast_router, S_IRUGO | S_IWUSR, show_multicast_router, 3300909e117SHerbert Xu store_multicast_router); 331561f1103SHerbert Xu 332561f1103SHerbert Xu static ssize_t show_multicast_snooping(struct device *d, 333561f1103SHerbert Xu struct device_attribute *attr, 334561f1103SHerbert Xu char *buf) 335561f1103SHerbert Xu { 336561f1103SHerbert Xu struct net_bridge *br = to_bridge(d); 337561f1103SHerbert Xu return sprintf(buf, "%d\n", !br->multicast_disabled); 338561f1103SHerbert Xu } 339561f1103SHerbert Xu 340561f1103SHerbert Xu static ssize_t store_multicast_snooping(struct device *d, 341561f1103SHerbert Xu struct device_attribute *attr, 342561f1103SHerbert Xu const char *buf, size_t len) 343561f1103SHerbert Xu { 344561f1103SHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_toggle); 345561f1103SHerbert Xu } 346561f1103SHerbert Xu static DEVICE_ATTR(multicast_snooping, S_IRUGO | S_IWUSR, 347561f1103SHerbert Xu show_multicast_snooping, store_multicast_snooping); 348b195167fSHerbert Xu 349b195167fSHerbert Xu static ssize_t show_hash_elasticity(struct device *d, 350b195167fSHerbert Xu struct device_attribute *attr, char *buf) 351b195167fSHerbert Xu { 352b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 353b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_elasticity); 354b195167fSHerbert Xu } 355b195167fSHerbert Xu 356b195167fSHerbert Xu static int set_elasticity(struct net_bridge *br, unsigned long val) 357b195167fSHerbert Xu { 358b195167fSHerbert Xu br->hash_elasticity = val; 359b195167fSHerbert Xu return 0; 360b195167fSHerbert Xu } 361b195167fSHerbert Xu 362b195167fSHerbert Xu static ssize_t store_hash_elasticity(struct device *d, 363b195167fSHerbert Xu struct device_attribute *attr, 364b195167fSHerbert Xu const char *buf, size_t len) 365b195167fSHerbert Xu { 366b195167fSHerbert Xu return store_bridge_parm(d, buf, len, set_elasticity); 367b195167fSHerbert Xu } 368b195167fSHerbert Xu static DEVICE_ATTR(hash_elasticity, S_IRUGO | S_IWUSR, show_hash_elasticity, 369b195167fSHerbert Xu store_hash_elasticity); 370b195167fSHerbert Xu 371b195167fSHerbert Xu static ssize_t show_hash_max(struct device *d, struct device_attribute *attr, 372b195167fSHerbert Xu char *buf) 373b195167fSHerbert Xu { 374b195167fSHerbert Xu struct net_bridge *br = to_bridge(d); 375b195167fSHerbert Xu return sprintf(buf, "%u\n", br->hash_max); 376b195167fSHerbert Xu } 377b195167fSHerbert Xu 378b195167fSHerbert Xu static ssize_t store_hash_max(struct device *d, struct device_attribute *attr, 379b195167fSHerbert Xu const char *buf, size_t len) 380b195167fSHerbert Xu { 381b195167fSHerbert Xu return store_bridge_parm(d, buf, len, br_multicast_set_hash_max); 382b195167fSHerbert Xu } 383b195167fSHerbert Xu static DEVICE_ATTR(hash_max, S_IRUGO | S_IWUSR, show_hash_max, 384b195167fSHerbert Xu store_hash_max); 385d902eee4SHerbert Xu 386d902eee4SHerbert Xu static ssize_t show_multicast_last_member_count(struct device *d, 387d902eee4SHerbert Xu struct device_attribute *attr, 388d902eee4SHerbert Xu char *buf) 389d902eee4SHerbert Xu { 390d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 391d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_last_member_count); 392d902eee4SHerbert Xu } 393d902eee4SHerbert Xu 394d902eee4SHerbert Xu static int set_last_member_count(struct net_bridge *br, unsigned long val) 395d902eee4SHerbert Xu { 396d902eee4SHerbert Xu br->multicast_last_member_count = val; 397d902eee4SHerbert Xu return 0; 398d902eee4SHerbert Xu } 399d902eee4SHerbert Xu 400d902eee4SHerbert Xu static ssize_t store_multicast_last_member_count(struct device *d, 401d902eee4SHerbert Xu struct device_attribute *attr, 402d902eee4SHerbert Xu const char *buf, size_t len) 403d902eee4SHerbert Xu { 404d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_count); 405d902eee4SHerbert Xu } 406d902eee4SHerbert Xu static DEVICE_ATTR(multicast_last_member_count, S_IRUGO | S_IWUSR, 407d902eee4SHerbert Xu show_multicast_last_member_count, 408d902eee4SHerbert Xu store_multicast_last_member_count); 409d902eee4SHerbert Xu 410d902eee4SHerbert Xu static ssize_t show_multicast_startup_query_count( 411d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 412d902eee4SHerbert Xu { 413d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 414d902eee4SHerbert Xu return sprintf(buf, "%u\n", br->multicast_startup_query_count); 415d902eee4SHerbert Xu } 416d902eee4SHerbert Xu 417d902eee4SHerbert Xu static int set_startup_query_count(struct net_bridge *br, unsigned long val) 418d902eee4SHerbert Xu { 419d902eee4SHerbert Xu br->multicast_startup_query_count = val; 420d902eee4SHerbert Xu return 0; 421d902eee4SHerbert Xu } 422d902eee4SHerbert Xu 423d902eee4SHerbert Xu static ssize_t store_multicast_startup_query_count( 424d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 425d902eee4SHerbert Xu size_t len) 426d902eee4SHerbert Xu { 427d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_count); 428d902eee4SHerbert Xu } 429d902eee4SHerbert Xu static DEVICE_ATTR(multicast_startup_query_count, S_IRUGO | S_IWUSR, 430d902eee4SHerbert Xu show_multicast_startup_query_count, 431d902eee4SHerbert Xu store_multicast_startup_query_count); 432d902eee4SHerbert Xu 433d902eee4SHerbert Xu static ssize_t show_multicast_last_member_interval( 434d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 435d902eee4SHerbert Xu { 436d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 437d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 438d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_last_member_interval)); 439d902eee4SHerbert Xu } 440d902eee4SHerbert Xu 441d902eee4SHerbert Xu static int set_last_member_interval(struct net_bridge *br, unsigned long val) 442d902eee4SHerbert Xu { 443d902eee4SHerbert Xu br->multicast_last_member_interval = clock_t_to_jiffies(val); 444d902eee4SHerbert Xu return 0; 445d902eee4SHerbert Xu } 446d902eee4SHerbert Xu 447d902eee4SHerbert Xu static ssize_t store_multicast_last_member_interval( 448d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 449d902eee4SHerbert Xu size_t len) 450d902eee4SHerbert Xu { 451d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_last_member_interval); 452d902eee4SHerbert Xu } 453d902eee4SHerbert Xu static DEVICE_ATTR(multicast_last_member_interval, S_IRUGO | S_IWUSR, 454d902eee4SHerbert Xu show_multicast_last_member_interval, 455d902eee4SHerbert Xu store_multicast_last_member_interval); 456d902eee4SHerbert Xu 457d902eee4SHerbert Xu static ssize_t show_multicast_membership_interval( 458d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 459d902eee4SHerbert Xu { 460d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 461d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 462d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_membership_interval)); 463d902eee4SHerbert Xu } 464d902eee4SHerbert Xu 465d902eee4SHerbert Xu static int set_membership_interval(struct net_bridge *br, unsigned long val) 466d902eee4SHerbert Xu { 467d902eee4SHerbert Xu br->multicast_membership_interval = clock_t_to_jiffies(val); 468d902eee4SHerbert Xu return 0; 469d902eee4SHerbert Xu } 470d902eee4SHerbert Xu 471d902eee4SHerbert Xu static ssize_t store_multicast_membership_interval( 472d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 473d902eee4SHerbert Xu size_t len) 474d902eee4SHerbert Xu { 475d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_membership_interval); 476d902eee4SHerbert Xu } 477d902eee4SHerbert Xu static DEVICE_ATTR(multicast_membership_interval, S_IRUGO | S_IWUSR, 478d902eee4SHerbert Xu show_multicast_membership_interval, 479d902eee4SHerbert Xu store_multicast_membership_interval); 480d902eee4SHerbert Xu 481d902eee4SHerbert Xu static ssize_t show_multicast_querier_interval(struct device *d, 482d902eee4SHerbert Xu struct device_attribute *attr, 483d902eee4SHerbert Xu char *buf) 484d902eee4SHerbert Xu { 485d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 486d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 487d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_querier_interval)); 488d902eee4SHerbert Xu } 489d902eee4SHerbert Xu 490d902eee4SHerbert Xu static int set_querier_interval(struct net_bridge *br, unsigned long val) 491d902eee4SHerbert Xu { 492d902eee4SHerbert Xu br->multicast_querier_interval = clock_t_to_jiffies(val); 493d902eee4SHerbert Xu return 0; 494d902eee4SHerbert Xu } 495d902eee4SHerbert Xu 496d902eee4SHerbert Xu static ssize_t store_multicast_querier_interval(struct device *d, 497d902eee4SHerbert Xu struct device_attribute *attr, 498d902eee4SHerbert Xu const char *buf, size_t len) 499d902eee4SHerbert Xu { 500d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_querier_interval); 501d902eee4SHerbert Xu } 502d902eee4SHerbert Xu static DEVICE_ATTR(multicast_querier_interval, S_IRUGO | S_IWUSR, 503d902eee4SHerbert Xu show_multicast_querier_interval, 504d902eee4SHerbert Xu store_multicast_querier_interval); 505d902eee4SHerbert Xu 506d902eee4SHerbert Xu static ssize_t show_multicast_query_interval(struct device *d, 507d902eee4SHerbert Xu struct device_attribute *attr, 508d902eee4SHerbert Xu char *buf) 509d902eee4SHerbert Xu { 510d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 511d902eee4SHerbert Xu return sprintf(buf, "%lu\n", 512d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_interval)); 513d902eee4SHerbert Xu } 514d902eee4SHerbert Xu 515d902eee4SHerbert Xu static int set_query_interval(struct net_bridge *br, unsigned long val) 516d902eee4SHerbert Xu { 517d902eee4SHerbert Xu br->multicast_query_interval = clock_t_to_jiffies(val); 518d902eee4SHerbert Xu return 0; 519d902eee4SHerbert Xu } 520d902eee4SHerbert Xu 521d902eee4SHerbert Xu static ssize_t store_multicast_query_interval(struct device *d, 522d902eee4SHerbert Xu struct device_attribute *attr, 523d902eee4SHerbert Xu const char *buf, size_t len) 524d902eee4SHerbert Xu { 525d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_interval); 526d902eee4SHerbert Xu } 527d902eee4SHerbert Xu static DEVICE_ATTR(multicast_query_interval, S_IRUGO | S_IWUSR, 528d902eee4SHerbert Xu show_multicast_query_interval, 529d902eee4SHerbert Xu store_multicast_query_interval); 530d902eee4SHerbert Xu 531d902eee4SHerbert Xu static ssize_t show_multicast_query_response_interval( 532d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 533d902eee4SHerbert Xu { 534d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 535d902eee4SHerbert Xu return sprintf( 536d902eee4SHerbert Xu buf, "%lu\n", 537d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_query_response_interval)); 538d902eee4SHerbert Xu } 539d902eee4SHerbert Xu 540d902eee4SHerbert Xu static int set_query_response_interval(struct net_bridge *br, unsigned long val) 541d902eee4SHerbert Xu { 542d902eee4SHerbert Xu br->multicast_query_response_interval = clock_t_to_jiffies(val); 543d902eee4SHerbert Xu return 0; 544d902eee4SHerbert Xu } 545d902eee4SHerbert Xu 546d902eee4SHerbert Xu static ssize_t store_multicast_query_response_interval( 547d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 548d902eee4SHerbert Xu size_t len) 549d902eee4SHerbert Xu { 550d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_query_response_interval); 551d902eee4SHerbert Xu } 552d902eee4SHerbert Xu static DEVICE_ATTR(multicast_query_response_interval, S_IRUGO | S_IWUSR, 553d902eee4SHerbert Xu show_multicast_query_response_interval, 554d902eee4SHerbert Xu store_multicast_query_response_interval); 555d902eee4SHerbert Xu 556d902eee4SHerbert Xu static ssize_t show_multicast_startup_query_interval( 557d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, char *buf) 558d902eee4SHerbert Xu { 559d902eee4SHerbert Xu struct net_bridge *br = to_bridge(d); 560d902eee4SHerbert Xu return sprintf( 561d902eee4SHerbert Xu buf, "%lu\n", 562d902eee4SHerbert Xu jiffies_to_clock_t(br->multicast_startup_query_interval)); 563d902eee4SHerbert Xu } 564d902eee4SHerbert Xu 565d902eee4SHerbert Xu static int set_startup_query_interval(struct net_bridge *br, unsigned long val) 566d902eee4SHerbert Xu { 567d902eee4SHerbert Xu br->multicast_startup_query_interval = clock_t_to_jiffies(val); 568d902eee4SHerbert Xu return 0; 569d902eee4SHerbert Xu } 570d902eee4SHerbert Xu 571d902eee4SHerbert Xu static ssize_t store_multicast_startup_query_interval( 572d902eee4SHerbert Xu struct device *d, struct device_attribute *attr, const char *buf, 573d902eee4SHerbert Xu size_t len) 574d902eee4SHerbert Xu { 575d902eee4SHerbert Xu return store_bridge_parm(d, buf, len, set_startup_query_interval); 576d902eee4SHerbert Xu } 577d902eee4SHerbert Xu static DEVICE_ATTR(multicast_startup_query_interval, S_IRUGO | S_IWUSR, 578d902eee4SHerbert Xu show_multicast_startup_query_interval, 579d902eee4SHerbert Xu store_multicast_startup_query_interval); 5800909e117SHerbert Xu #endif 5814df53d8bSPatrick McHardy #ifdef CONFIG_BRIDGE_NETFILTER 5824df53d8bSPatrick McHardy static ssize_t show_nf_call_iptables( 5834df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 5844df53d8bSPatrick McHardy { 5854df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 5864df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_iptables); 5874df53d8bSPatrick McHardy } 5884df53d8bSPatrick McHardy 5894df53d8bSPatrick McHardy static int set_nf_call_iptables(struct net_bridge *br, unsigned long val) 5904df53d8bSPatrick McHardy { 5914df53d8bSPatrick McHardy br->nf_call_iptables = val ? true : false; 5924df53d8bSPatrick McHardy return 0; 5934df53d8bSPatrick McHardy } 5944df53d8bSPatrick McHardy 5954df53d8bSPatrick McHardy static ssize_t store_nf_call_iptables( 5964df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 5974df53d8bSPatrick McHardy size_t len) 5984df53d8bSPatrick McHardy { 5994df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_iptables); 6004df53d8bSPatrick McHardy } 6014df53d8bSPatrick McHardy static DEVICE_ATTR(nf_call_iptables, S_IRUGO | S_IWUSR, 6024df53d8bSPatrick McHardy show_nf_call_iptables, store_nf_call_iptables); 6034df53d8bSPatrick McHardy 6044df53d8bSPatrick McHardy static ssize_t show_nf_call_ip6tables( 6054df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6064df53d8bSPatrick McHardy { 6074df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6084df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_ip6tables); 6094df53d8bSPatrick McHardy } 6104df53d8bSPatrick McHardy 6114df53d8bSPatrick McHardy static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val) 6124df53d8bSPatrick McHardy { 6134df53d8bSPatrick McHardy br->nf_call_ip6tables = val ? true : false; 6144df53d8bSPatrick McHardy return 0; 6154df53d8bSPatrick McHardy } 6164df53d8bSPatrick McHardy 6174df53d8bSPatrick McHardy static ssize_t store_nf_call_ip6tables( 6184df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6194df53d8bSPatrick McHardy size_t len) 6204df53d8bSPatrick McHardy { 6214df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_ip6tables); 6224df53d8bSPatrick McHardy } 6234df53d8bSPatrick McHardy static DEVICE_ATTR(nf_call_ip6tables, S_IRUGO | S_IWUSR, 6244df53d8bSPatrick McHardy show_nf_call_ip6tables, store_nf_call_ip6tables); 6254df53d8bSPatrick McHardy 6264df53d8bSPatrick McHardy static ssize_t show_nf_call_arptables( 6274df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, char *buf) 6284df53d8bSPatrick McHardy { 6294df53d8bSPatrick McHardy struct net_bridge *br = to_bridge(d); 6304df53d8bSPatrick McHardy return sprintf(buf, "%u\n", br->nf_call_arptables); 6314df53d8bSPatrick McHardy } 6324df53d8bSPatrick McHardy 6334df53d8bSPatrick McHardy static int set_nf_call_arptables(struct net_bridge *br, unsigned long val) 6344df53d8bSPatrick McHardy { 6354df53d8bSPatrick McHardy br->nf_call_arptables = val ? true : false; 6364df53d8bSPatrick McHardy return 0; 6374df53d8bSPatrick McHardy } 6384df53d8bSPatrick McHardy 6394df53d8bSPatrick McHardy static ssize_t store_nf_call_arptables( 6404df53d8bSPatrick McHardy struct device *d, struct device_attribute *attr, const char *buf, 6414df53d8bSPatrick McHardy size_t len) 6424df53d8bSPatrick McHardy { 6434df53d8bSPatrick McHardy return store_bridge_parm(d, buf, len, set_nf_call_arptables); 6444df53d8bSPatrick McHardy } 6454df53d8bSPatrick McHardy static DEVICE_ATTR(nf_call_arptables, S_IRUGO | S_IWUSR, 6464df53d8bSPatrick McHardy show_nf_call_arptables, store_nf_call_arptables); 6474df53d8bSPatrick McHardy #endif 6480909e117SHerbert Xu 6491da177e4SLinus Torvalds static struct attribute *bridge_attrs[] = { 65043cb76d9SGreg Kroah-Hartman &dev_attr_forward_delay.attr, 65143cb76d9SGreg Kroah-Hartman &dev_attr_hello_time.attr, 65243cb76d9SGreg Kroah-Hartman &dev_attr_max_age.attr, 65343cb76d9SGreg Kroah-Hartman &dev_attr_ageing_time.attr, 65443cb76d9SGreg Kroah-Hartman &dev_attr_stp_state.attr, 65543cb76d9SGreg Kroah-Hartman &dev_attr_priority.attr, 65643cb76d9SGreg Kroah-Hartman &dev_attr_bridge_id.attr, 65743cb76d9SGreg Kroah-Hartman &dev_attr_root_id.attr, 65843cb76d9SGreg Kroah-Hartman &dev_attr_root_path_cost.attr, 65943cb76d9SGreg Kroah-Hartman &dev_attr_root_port.attr, 66043cb76d9SGreg Kroah-Hartman &dev_attr_topology_change.attr, 66143cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_detected.attr, 66243cb76d9SGreg Kroah-Hartman &dev_attr_hello_timer.attr, 66343cb76d9SGreg Kroah-Hartman &dev_attr_tcn_timer.attr, 66443cb76d9SGreg Kroah-Hartman &dev_attr_topology_change_timer.attr, 66543cb76d9SGreg Kroah-Hartman &dev_attr_gc_timer.attr, 66643cb76d9SGreg Kroah-Hartman &dev_attr_group_addr.attr, 6679cf63747SStephen Hemminger &dev_attr_flush.attr, 6680909e117SHerbert Xu #ifdef CONFIG_BRIDGE_IGMP_SNOOPING 6690909e117SHerbert Xu &dev_attr_multicast_router.attr, 670561f1103SHerbert Xu &dev_attr_multicast_snooping.attr, 671b195167fSHerbert Xu &dev_attr_hash_elasticity.attr, 672b195167fSHerbert Xu &dev_attr_hash_max.attr, 673d902eee4SHerbert Xu &dev_attr_multicast_last_member_count.attr, 674d902eee4SHerbert Xu &dev_attr_multicast_startup_query_count.attr, 675d902eee4SHerbert Xu &dev_attr_multicast_last_member_interval.attr, 676d902eee4SHerbert Xu &dev_attr_multicast_membership_interval.attr, 677d902eee4SHerbert Xu &dev_attr_multicast_querier_interval.attr, 678d902eee4SHerbert Xu &dev_attr_multicast_query_interval.attr, 679d902eee4SHerbert Xu &dev_attr_multicast_query_response_interval.attr, 680d902eee4SHerbert Xu &dev_attr_multicast_startup_query_interval.attr, 6810909e117SHerbert Xu #endif 6824df53d8bSPatrick McHardy #ifdef CONFIG_BRIDGE_NETFILTER 6834df53d8bSPatrick McHardy &dev_attr_nf_call_iptables.attr, 6844df53d8bSPatrick McHardy &dev_attr_nf_call_ip6tables.attr, 6854df53d8bSPatrick McHardy &dev_attr_nf_call_arptables.attr, 6864df53d8bSPatrick McHardy #endif 6871da177e4SLinus Torvalds NULL 6881da177e4SLinus Torvalds }; 6891da177e4SLinus Torvalds 6901da177e4SLinus Torvalds static struct attribute_group bridge_group = { 6911da177e4SLinus Torvalds .name = SYSFS_BRIDGE_ATTR, 6921da177e4SLinus Torvalds .attrs = bridge_attrs, 6931da177e4SLinus Torvalds }; 6941da177e4SLinus Torvalds 6951da177e4SLinus Torvalds /* 6961da177e4SLinus Torvalds * Export the forwarding information table as a binary file 6971da177e4SLinus Torvalds * The records are struct __fdb_entry. 6981da177e4SLinus Torvalds * 6991da177e4SLinus Torvalds * Returns the number of bytes read. 7001da177e4SLinus Torvalds */ 7012c3c8beaSChris Wright static ssize_t brforward_read(struct file *filp, struct kobject *kobj, 70291a69029SZhang Rui struct bin_attribute *bin_attr, 70391a69029SZhang Rui char *buf, loff_t off, size_t count) 7041da177e4SLinus Torvalds { 70543cb76d9SGreg Kroah-Hartman struct device *dev = to_dev(kobj); 70643cb76d9SGreg Kroah-Hartman struct net_bridge *br = to_bridge(dev); 7071da177e4SLinus Torvalds int n; 7081da177e4SLinus Torvalds 7091da177e4SLinus Torvalds /* must read whole records */ 7101da177e4SLinus Torvalds if (off % sizeof(struct __fdb_entry) != 0) 7111da177e4SLinus Torvalds return -EINVAL; 7121da177e4SLinus Torvalds 7131da177e4SLinus Torvalds n = br_fdb_fillbuf(br, buf, 7141da177e4SLinus Torvalds count / sizeof(struct __fdb_entry), 7151da177e4SLinus Torvalds off / sizeof(struct __fdb_entry)); 7161da177e4SLinus Torvalds 7171da177e4SLinus Torvalds if (n > 0) 7181da177e4SLinus Torvalds n *= sizeof(struct __fdb_entry); 7191da177e4SLinus Torvalds 7201da177e4SLinus Torvalds return n; 7211da177e4SLinus Torvalds } 7221da177e4SLinus Torvalds 7231da177e4SLinus Torvalds static struct bin_attribute bridge_forward = { 7241da177e4SLinus Torvalds .attr = { .name = SYSFS_BRIDGE_FDB, 7257b595756STejun Heo .mode = S_IRUGO, }, 7261da177e4SLinus Torvalds .read = brforward_read, 7271da177e4SLinus Torvalds }; 7281da177e4SLinus Torvalds 7291da177e4SLinus Torvalds /* 7301da177e4SLinus Torvalds * Add entries in sysfs onto the existing network class device 7311da177e4SLinus Torvalds * for the bridge. 7321da177e4SLinus Torvalds * Adds a attribute group "bridge" containing tuning parameters. 7331da177e4SLinus Torvalds * Binary attribute containing the forward table 7341da177e4SLinus Torvalds * Sub directory to hold links to interfaces. 7351da177e4SLinus Torvalds * 7361da177e4SLinus Torvalds * Note: the ifobj exists only to be a subdirectory 7371da177e4SLinus Torvalds * to hold links. The ifobj exists in same data structure 7381da177e4SLinus Torvalds * as it's parent the bridge so reference counting works. 7391da177e4SLinus Torvalds */ 7401da177e4SLinus Torvalds int br_sysfs_addbr(struct net_device *dev) 7411da177e4SLinus Torvalds { 74243cb76d9SGreg Kroah-Hartman struct kobject *brobj = &dev->dev.kobj; 7431da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 7441da177e4SLinus Torvalds int err; 7451da177e4SLinus Torvalds 7461da177e4SLinus Torvalds err = sysfs_create_group(brobj, &bridge_group); 7471da177e4SLinus Torvalds if (err) { 7481da177e4SLinus Torvalds pr_info("%s: can't create group %s/%s\n", 7490dc47877SHarvey Harrison __func__, dev->name, bridge_group.name); 7501da177e4SLinus Torvalds goto out1; 7511da177e4SLinus Torvalds } 7521da177e4SLinus Torvalds 7531da177e4SLinus Torvalds err = sysfs_create_bin_file(brobj, &bridge_forward); 7541da177e4SLinus Torvalds if (err) { 7551842c4beSRandy Dunlap pr_info("%s: can't create attribute file %s/%s\n", 7560dc47877SHarvey Harrison __func__, dev->name, bridge_forward.attr.name); 7571da177e4SLinus Torvalds goto out2; 7581da177e4SLinus Torvalds } 7591da177e4SLinus Torvalds 76043b98c4aSGreg Kroah-Hartman br->ifobj = kobject_create_and_add(SYSFS_BRIDGE_PORT_SUBDIR, brobj); 76143b98c4aSGreg Kroah-Hartman if (!br->ifobj) { 7621da177e4SLinus Torvalds pr_info("%s: can't add kobject (directory) %s/%s\n", 7630dc47877SHarvey Harrison __func__, dev->name, SYSFS_BRIDGE_PORT_SUBDIR); 7641da177e4SLinus Torvalds goto out3; 7651da177e4SLinus Torvalds } 7661da177e4SLinus Torvalds return 0; 7671da177e4SLinus Torvalds out3: 76843cb76d9SGreg Kroah-Hartman sysfs_remove_bin_file(&dev->dev.kobj, &bridge_forward); 7691da177e4SLinus Torvalds out2: 77043cb76d9SGreg Kroah-Hartman sysfs_remove_group(&dev->dev.kobj, &bridge_group); 7711da177e4SLinus Torvalds out1: 7721da177e4SLinus Torvalds return err; 7731da177e4SLinus Torvalds 7741da177e4SLinus Torvalds } 7751da177e4SLinus Torvalds 7761da177e4SLinus Torvalds void br_sysfs_delbr(struct net_device *dev) 7771da177e4SLinus Torvalds { 77843cb76d9SGreg Kroah-Hartman struct kobject *kobj = &dev->dev.kobj; 7791da177e4SLinus Torvalds struct net_bridge *br = netdev_priv(dev); 7801da177e4SLinus Torvalds 78178a2d906SGreg Kroah-Hartman kobject_put(br->ifobj); 7821da177e4SLinus Torvalds sysfs_remove_bin_file(kobj, &bridge_forward); 7831da177e4SLinus Torvalds sysfs_remove_group(kobj, &bridge_group); 7841da177e4SLinus Torvalds } 785