br_sysfs_br.c 22.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 *	Sysfs attributes of bridge ports
 *	Linux ethernet bridge
 *
 *	Authors:
 *	Stephen Hemminger		<shemminger@osdl.org>
 *
 *	This program is free software; you can redistribute it and/or
 *	modify it under the terms of the GNU General Public License
 *	as published by the Free Software Foundation; either version
 *	2 of the License, or (at your option) any later version.
 */

14
#include <linux/capability.h>
L
Linus Torvalds 已提交
15 16 17 18 19 20 21 22 23
#include <linux/kernel.h>
#include <linux/netdevice.h>
#include <linux/if_bridge.h>
#include <linux/rtnetlink.h>
#include <linux/spinlock.h>
#include <linux/times.h>

#include "br_private.h"

24
#define to_dev(obj)	container_of(obj, struct device, kobj)
25
#define to_bridge(cd)	((struct net_bridge *)netdev_priv(to_net_dev(cd)))
L
Linus Torvalds 已提交
26 27 28 29

/*
 * Common code for storing bridge parameters.
 */
30
static ssize_t store_bridge_parm(struct device *d,
L
Linus Torvalds 已提交
31
				 const char *buf, size_t len,
32
				 int (*set)(struct net_bridge *, unsigned long))
L
Linus Torvalds 已提交
33
{
34
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
35 36
	char *endp;
	unsigned long val;
37
	int err;
L
Linus Torvalds 已提交
38 39 40 41 42 43 44 45

	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

	val = simple_strtoul(buf, &endp, 0);
	if (endp == buf)
		return -EINVAL;

46 47
	err = (*set)(br, val);
	return err ? err : len;
L
Linus Torvalds 已提交
48 49 50
}


51 52
static ssize_t show_forward_delay(struct device *d,
				  struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
53
{
54
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
55 56 57
	return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->forward_delay));
}

58 59 60
static ssize_t store_forward_delay(struct device *d,
				   struct device_attribute *attr,
				   const char *buf, size_t len)
L
Linus Torvalds 已提交
61
{
62
	return store_bridge_parm(d, buf, len, br_set_forward_delay);
L
Linus Torvalds 已提交
63
}
64 65
static DEVICE_ATTR(forward_delay, S_IRUGO | S_IWUSR,
		   show_forward_delay, store_forward_delay);
L
Linus Torvalds 已提交
66

67 68
static ssize_t show_hello_time(struct device *d, struct device_attribute *attr,
			       char *buf)
L
Linus Torvalds 已提交
69 70
{
	return sprintf(buf, "%lu\n",
71
		       jiffies_to_clock_t(to_bridge(d)->hello_time));
L
Linus Torvalds 已提交
72 73
}

74 75
static ssize_t store_hello_time(struct device *d,
				struct device_attribute *attr, const char *buf,
L
Linus Torvalds 已提交
76 77
				size_t len)
{
78
	return store_bridge_parm(d, buf, len, br_set_hello_time);
L
Linus Torvalds 已提交
79
}
80 81
static DEVICE_ATTR(hello_time, S_IRUGO | S_IWUSR, show_hello_time,
		   store_hello_time);
L
Linus Torvalds 已提交
82

83 84
static ssize_t show_max_age(struct device *d, struct device_attribute *attr,
			    char *buf)
L
Linus Torvalds 已提交
85 86
{
	return sprintf(buf, "%lu\n",
87
		       jiffies_to_clock_t(to_bridge(d)->max_age));
L
Linus Torvalds 已提交
88 89
}

90 91
static ssize_t store_max_age(struct device *d, struct device_attribute *attr,
			     const char *buf, size_t len)
L
Linus Torvalds 已提交
92
{
93
	return store_bridge_parm(d, buf, len, br_set_max_age);
L
Linus Torvalds 已提交
94
}
95
static DEVICE_ATTR(max_age, S_IRUGO | S_IWUSR, show_max_age, store_max_age);
L
Linus Torvalds 已提交
96

97 98
static ssize_t show_ageing_time(struct device *d,
				struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
99
{
100
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
101 102 103
	return sprintf(buf, "%lu\n", jiffies_to_clock_t(br->ageing_time));
}

104
static int set_ageing_time(struct net_bridge *br, unsigned long val)
L
Linus Torvalds 已提交
105 106
{
	br->ageing_time = clock_t_to_jiffies(val);
107
	return 0;
L
Linus Torvalds 已提交
108 109
}

110 111 112
static ssize_t store_ageing_time(struct device *d,
				 struct device_attribute *attr,
				 const char *buf, size_t len)
L
Linus Torvalds 已提交
113
{
114
	return store_bridge_parm(d, buf, len, set_ageing_time);
L
Linus Torvalds 已提交
115
}
116 117
static DEVICE_ATTR(ageing_time, S_IRUGO | S_IWUSR, show_ageing_time,
		   store_ageing_time);
L
Linus Torvalds 已提交
118

119 120
static ssize_t show_stp_state(struct device *d,
			      struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
121
{
122
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
123 124 125 126
	return sprintf(buf, "%d\n", br->stp_enabled);
}


127 128 129
static ssize_t store_stp_state(struct device *d,
			       struct device_attribute *attr, const char *buf,
			       size_t len)
L
Linus Torvalds 已提交
130
{
S
Stephen Hemminger 已提交
131 132 133 134 135 136 137 138 139 140 141
	struct net_bridge *br = to_bridge(d);
	char *endp;
	unsigned long val;

	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

	val = simple_strtoul(buf, &endp, 0);
	if (endp == buf)
		return -EINVAL;

142 143
	if (!rtnl_trylock())
		return restart_syscall();
S
Stephen Hemminger 已提交
144 145 146
	br_stp_set_enabled(br, val);
	rtnl_unlock();

A
Al Viro 已提交
147
	return len;
L
Linus Torvalds 已提交
148
}
149 150
static DEVICE_ATTR(stp_state, S_IRUGO | S_IWUSR, show_stp_state,
		   store_stp_state);
L
Linus Torvalds 已提交
151

152 153
static ssize_t show_priority(struct device *d, struct device_attribute *attr,
			     char *buf)
L
Linus Torvalds 已提交
154
{
155
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
156 157 158 159
	return sprintf(buf, "%d\n",
		       (br->bridge_id.prio[0] << 8) | br->bridge_id.prio[1]);
}

160
static int set_priority(struct net_bridge *br, unsigned long val)
L
Linus Torvalds 已提交
161 162
{
	br_stp_set_bridge_priority(br, (u16) val);
163
	return 0;
L
Linus Torvalds 已提交
164 165
}

166
static ssize_t store_priority(struct device *d, struct device_attribute *attr,
L
Linus Torvalds 已提交
167 168
			       const char *buf, size_t len)
{
169
	return store_bridge_parm(d, buf, len, set_priority);
L
Linus Torvalds 已提交
170
}
171
static DEVICE_ATTR(priority, S_IRUGO | S_IWUSR, show_priority, store_priority);
L
Linus Torvalds 已提交
172

173 174
static ssize_t show_root_id(struct device *d, struct device_attribute *attr,
			    char *buf)
L
Linus Torvalds 已提交
175
{
176
	return br_show_bridge_id(buf, &to_bridge(d)->designated_root);
L
Linus Torvalds 已提交
177
}
178
static DEVICE_ATTR(root_id, S_IRUGO, show_root_id, NULL);
L
Linus Torvalds 已提交
179

180 181
static ssize_t show_bridge_id(struct device *d, struct device_attribute *attr,
			      char *buf)
L
Linus Torvalds 已提交
182
{
183
	return br_show_bridge_id(buf, &to_bridge(d)->bridge_id);
L
Linus Torvalds 已提交
184
}
185
static DEVICE_ATTR(bridge_id, S_IRUGO, show_bridge_id, NULL);
L
Linus Torvalds 已提交
186

187 188
static ssize_t show_root_port(struct device *d, struct device_attribute *attr,
			      char *buf)
L
Linus Torvalds 已提交
189
{
190
	return sprintf(buf, "%d\n", to_bridge(d)->root_port);
L
Linus Torvalds 已提交
191
}
192
static DEVICE_ATTR(root_port, S_IRUGO, show_root_port, NULL);
L
Linus Torvalds 已提交
193

194 195
static ssize_t show_root_path_cost(struct device *d,
				   struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
196
{
197
	return sprintf(buf, "%d\n", to_bridge(d)->root_path_cost);
L
Linus Torvalds 已提交
198
}
199
static DEVICE_ATTR(root_path_cost, S_IRUGO, show_root_path_cost, NULL);
L
Linus Torvalds 已提交
200

201 202
static ssize_t show_topology_change(struct device *d,
				    struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
203
{
204
	return sprintf(buf, "%d\n", to_bridge(d)->topology_change);
L
Linus Torvalds 已提交
205
}
206
static DEVICE_ATTR(topology_change, S_IRUGO, show_topology_change, NULL);
L
Linus Torvalds 已提交
207

208 209 210
static ssize_t show_topology_change_detected(struct device *d,
					     struct device_attribute *attr,
					     char *buf)
L
Linus Torvalds 已提交
211
{
212
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
213 214
	return sprintf(buf, "%d\n", br->topology_change_detected);
}
215 216
static DEVICE_ATTR(topology_change_detected, S_IRUGO,
		   show_topology_change_detected, NULL);
L
Linus Torvalds 已提交
217

218 219
static ssize_t show_hello_timer(struct device *d,
				struct device_attribute *attr, char *buf)
L
Linus Torvalds 已提交
220
{
221
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
222 223
	return sprintf(buf, "%ld\n", br_timer_value(&br->hello_timer));
}
224
static DEVICE_ATTR(hello_timer, S_IRUGO, show_hello_timer, NULL);
L
Linus Torvalds 已提交
225

226 227
static ssize_t show_tcn_timer(struct device *d, struct device_attribute *attr,
			      char *buf)
L
Linus Torvalds 已提交
228
{
229
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
230 231
	return sprintf(buf, "%ld\n", br_timer_value(&br->tcn_timer));
}
232
static DEVICE_ATTR(tcn_timer, S_IRUGO, show_tcn_timer, NULL);
L
Linus Torvalds 已提交
233

234 235 236
static ssize_t show_topology_change_timer(struct device *d,
					  struct device_attribute *attr,
					  char *buf)
L
Linus Torvalds 已提交
237
{
238
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
239 240
	return sprintf(buf, "%ld\n", br_timer_value(&br->topology_change_timer));
}
241 242
static DEVICE_ATTR(topology_change_timer, S_IRUGO, show_topology_change_timer,
		   NULL);
L
Linus Torvalds 已提交
243

244 245
static ssize_t show_gc_timer(struct device *d, struct device_attribute *attr,
			     char *buf)
L
Linus Torvalds 已提交
246
{
247
	struct net_bridge *br = to_bridge(d);
L
Linus Torvalds 已提交
248 249
	return sprintf(buf, "%ld\n", br_timer_value(&br->gc_timer));
}
250
static DEVICE_ATTR(gc_timer, S_IRUGO, show_gc_timer, NULL);
L
Linus Torvalds 已提交
251

252 253
static ssize_t show_group_addr(struct device *d,
			       struct device_attribute *attr, char *buf)
254
{
255
	struct net_bridge *br = to_bridge(d);
256 257 258 259 260 261
	return sprintf(buf, "%x:%x:%x:%x:%x:%x\n",
		       br->group_addr[0], br->group_addr[1],
		       br->group_addr[2], br->group_addr[3],
		       br->group_addr[4], br->group_addr[5]);
}

262 263 264
static ssize_t store_group_addr(struct device *d,
				struct device_attribute *attr,
				const char *buf, size_t len)
265
{
266
	struct net_bridge *br = to_bridge(d);
267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285
	unsigned new_addr[6];
	int i;

	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

	if (sscanf(buf, "%x:%x:%x:%x:%x:%x",
		   &new_addr[0], &new_addr[1], &new_addr[2],
		   &new_addr[3], &new_addr[4], &new_addr[5]) != 6)
		return -EINVAL;

	/* Must be 01:80:c2:00:00:0X */
	for (i = 0; i < 5; i++)
		if (new_addr[i] != br_group_address[i])
			return -EINVAL;

	if (new_addr[5] & ~0xf)
		return -EINVAL;

286 287 288
	if (new_addr[5] == 1 ||		/* 802.3x Pause address */
	    new_addr[5] == 2 ||		/* 802.3ad Slow protocols */
	    new_addr[5] == 3)		/* 802.1X PAE address */
289 290 291 292 293 294 295 296 297
		return -EINVAL;

	spin_lock_bh(&br->lock);
	for (i = 0; i < 6; i++)
		br->group_addr[i] = new_addr[i];
	spin_unlock_bh(&br->lock);
	return len;
}

298 299
static DEVICE_ATTR(group_addr, S_IRUGO | S_IWUSR,
		   show_group_addr, store_group_addr);
300

301 302 303 304 305 306 307 308 309 310 311 312 313
static ssize_t store_flush(struct device *d,
			   struct device_attribute *attr,
			   const char *buf, size_t len)
{
	struct net_bridge *br = to_bridge(d);

	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

	br_fdb_flush(br);
	return len;
}
static DEVICE_ATTR(flush, S_IWUSR, NULL, store_flush);
314

315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330
#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
static ssize_t show_multicast_router(struct device *d,
				     struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%d\n", br->multicast_router);
}

static ssize_t store_multicast_router(struct device *d,
				      struct device_attribute *attr,
				      const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, br_multicast_set_router);
}
static DEVICE_ATTR(multicast_router, S_IRUGO | S_IWUSR, show_multicast_router,
		   store_multicast_router);
331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347

static ssize_t show_multicast_snooping(struct device *d,
				       struct device_attribute *attr,
				       char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%d\n", !br->multicast_disabled);
}

static ssize_t store_multicast_snooping(struct device *d,
					struct device_attribute *attr,
					const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, br_multicast_toggle);
}
static DEVICE_ATTR(multicast_snooping, S_IRUGO | S_IWUSR,
		   show_multicast_snooping, store_multicast_snooping);
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384

static ssize_t show_hash_elasticity(struct device *d,
				    struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->hash_elasticity);
}

static int set_elasticity(struct net_bridge *br, unsigned long val)
{
	br->hash_elasticity = val;
	return 0;
}

static ssize_t store_hash_elasticity(struct device *d,
				     struct device_attribute *attr,
				     const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, set_elasticity);
}
static DEVICE_ATTR(hash_elasticity, S_IRUGO | S_IWUSR, show_hash_elasticity,
		   store_hash_elasticity);

static ssize_t show_hash_max(struct device *d, struct device_attribute *attr,
			     char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->hash_max);
}

static ssize_t store_hash_max(struct device *d, struct device_attribute *attr,
			      const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, br_multicast_set_hash_max);
}
static DEVICE_ATTR(hash_max, S_IRUGO | S_IWUSR, show_hash_max,
		   store_hash_max);
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579

static ssize_t show_multicast_last_member_count(struct device *d,
						struct device_attribute *attr,
						char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->multicast_last_member_count);
}

static int set_last_member_count(struct net_bridge *br, unsigned long val)
{
	br->multicast_last_member_count = val;
	return 0;
}

static ssize_t store_multicast_last_member_count(struct device *d,
						 struct device_attribute *attr,
						 const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, set_last_member_count);
}
static DEVICE_ATTR(multicast_last_member_count, S_IRUGO | S_IWUSR,
		   show_multicast_last_member_count,
		   store_multicast_last_member_count);

static ssize_t show_multicast_startup_query_count(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->multicast_startup_query_count);
}

static int set_startup_query_count(struct net_bridge *br, unsigned long val)
{
	br->multicast_startup_query_count = val;
	return 0;
}

static ssize_t store_multicast_startup_query_count(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_startup_query_count);
}
static DEVICE_ATTR(multicast_startup_query_count, S_IRUGO | S_IWUSR,
		   show_multicast_startup_query_count,
		   store_multicast_startup_query_count);

static ssize_t show_multicast_last_member_interval(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%lu\n",
		       jiffies_to_clock_t(br->multicast_last_member_interval));
}

static int set_last_member_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_last_member_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_last_member_interval(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_last_member_interval);
}
static DEVICE_ATTR(multicast_last_member_interval, S_IRUGO | S_IWUSR,
		   show_multicast_last_member_interval,
		   store_multicast_last_member_interval);

static ssize_t show_multicast_membership_interval(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%lu\n",
		       jiffies_to_clock_t(br->multicast_membership_interval));
}

static int set_membership_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_membership_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_membership_interval(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_membership_interval);
}
static DEVICE_ATTR(multicast_membership_interval, S_IRUGO | S_IWUSR,
		   show_multicast_membership_interval,
		   store_multicast_membership_interval);

static ssize_t show_multicast_querier_interval(struct device *d,
					       struct device_attribute *attr,
					       char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%lu\n",
		       jiffies_to_clock_t(br->multicast_querier_interval));
}

static int set_querier_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_querier_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_querier_interval(struct device *d,
						struct device_attribute *attr,
						const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, set_querier_interval);
}
static DEVICE_ATTR(multicast_querier_interval, S_IRUGO | S_IWUSR,
		   show_multicast_querier_interval,
		   store_multicast_querier_interval);

static ssize_t show_multicast_query_interval(struct device *d,
					     struct device_attribute *attr,
					     char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%lu\n",
		       jiffies_to_clock_t(br->multicast_query_interval));
}

static int set_query_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_query_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_query_interval(struct device *d,
					      struct device_attribute *attr,
					      const char *buf, size_t len)
{
	return store_bridge_parm(d, buf, len, set_query_interval);
}
static DEVICE_ATTR(multicast_query_interval, S_IRUGO | S_IWUSR,
		   show_multicast_query_interval,
		   store_multicast_query_interval);

static ssize_t show_multicast_query_response_interval(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(
		buf, "%lu\n",
		jiffies_to_clock_t(br->multicast_query_response_interval));
}

static int set_query_response_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_query_response_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_query_response_interval(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_query_response_interval);
}
static DEVICE_ATTR(multicast_query_response_interval, S_IRUGO | S_IWUSR,
		   show_multicast_query_response_interval,
		   store_multicast_query_response_interval);

static ssize_t show_multicast_startup_query_interval(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(
		buf, "%lu\n",
		jiffies_to_clock_t(br->multicast_startup_query_interval));
}

static int set_startup_query_interval(struct net_bridge *br, unsigned long val)
{
	br->multicast_startup_query_interval = clock_t_to_jiffies(val);
	return 0;
}

static ssize_t store_multicast_startup_query_interval(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_startup_query_interval);
}
static DEVICE_ATTR(multicast_startup_query_interval, S_IRUGO | S_IWUSR,
		   show_multicast_startup_query_interval,
		   store_multicast_startup_query_interval);
580
#endif
581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647
#ifdef CONFIG_BRIDGE_NETFILTER
static ssize_t show_nf_call_iptables(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->nf_call_iptables);
}

static int set_nf_call_iptables(struct net_bridge *br, unsigned long val)
{
	br->nf_call_iptables = val ? true : false;
	return 0;
}

static ssize_t store_nf_call_iptables(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_nf_call_iptables);
}
static DEVICE_ATTR(nf_call_iptables, S_IRUGO | S_IWUSR,
		   show_nf_call_iptables, store_nf_call_iptables);

static ssize_t show_nf_call_ip6tables(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->nf_call_ip6tables);
}

static int set_nf_call_ip6tables(struct net_bridge *br, unsigned long val)
{
	br->nf_call_ip6tables = val ? true : false;
	return 0;
}

static ssize_t store_nf_call_ip6tables(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_nf_call_ip6tables);
}
static DEVICE_ATTR(nf_call_ip6tables, S_IRUGO | S_IWUSR,
		   show_nf_call_ip6tables, store_nf_call_ip6tables);

static ssize_t show_nf_call_arptables(
	struct device *d, struct device_attribute *attr, char *buf)
{
	struct net_bridge *br = to_bridge(d);
	return sprintf(buf, "%u\n", br->nf_call_arptables);
}

static int set_nf_call_arptables(struct net_bridge *br, unsigned long val)
{
	br->nf_call_arptables = val ? true : false;
	return 0;
}

static ssize_t store_nf_call_arptables(
	struct device *d, struct device_attribute *attr, const char *buf,
	size_t len)
{
	return store_bridge_parm(d, buf, len, set_nf_call_arptables);
}
static DEVICE_ATTR(nf_call_arptables, S_IRUGO | S_IWUSR,
		   show_nf_call_arptables, store_nf_call_arptables);
#endif
648

L
Linus Torvalds 已提交
649
static struct attribute *bridge_attrs[] = {
650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666
	&dev_attr_forward_delay.attr,
	&dev_attr_hello_time.attr,
	&dev_attr_max_age.attr,
	&dev_attr_ageing_time.attr,
	&dev_attr_stp_state.attr,
	&dev_attr_priority.attr,
	&dev_attr_bridge_id.attr,
	&dev_attr_root_id.attr,
	&dev_attr_root_path_cost.attr,
	&dev_attr_root_port.attr,
	&dev_attr_topology_change.attr,
	&dev_attr_topology_change_detected.attr,
	&dev_attr_hello_timer.attr,
	&dev_attr_tcn_timer.attr,
	&dev_attr_topology_change_timer.attr,
	&dev_attr_gc_timer.attr,
	&dev_attr_group_addr.attr,
667
	&dev_attr_flush.attr,
668 669
#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
	&dev_attr_multicast_router.attr,
670
	&dev_attr_multicast_snooping.attr,
671 672
	&dev_attr_hash_elasticity.attr,
	&dev_attr_hash_max.attr,
673 674 675 676 677 678 679 680
	&dev_attr_multicast_last_member_count.attr,
	&dev_attr_multicast_startup_query_count.attr,
	&dev_attr_multicast_last_member_interval.attr,
	&dev_attr_multicast_membership_interval.attr,
	&dev_attr_multicast_querier_interval.attr,
	&dev_attr_multicast_query_interval.attr,
	&dev_attr_multicast_query_response_interval.attr,
	&dev_attr_multicast_startup_query_interval.attr,
681 682 683 684 685
#endif
#ifdef CONFIG_BRIDGE_NETFILTER
	&dev_attr_nf_call_iptables.attr,
	&dev_attr_nf_call_ip6tables.attr,
	&dev_attr_nf_call_arptables.attr,
686
#endif
L
Linus Torvalds 已提交
687 688 689 690 691 692 693 694 695 696 697 698 699 700
	NULL
};

static struct attribute_group bridge_group = {
	.name = SYSFS_BRIDGE_ATTR,
	.attrs = bridge_attrs,
};

/*
 * Export the forwarding information table as a binary file
 * The records are struct __fdb_entry.
 *
 * Returns the number of bytes read.
 */
701
static ssize_t brforward_read(struct file *filp, struct kobject *kobj,
702 703
			      struct bin_attribute *bin_attr,
			      char *buf, loff_t off, size_t count)
L
Linus Torvalds 已提交
704
{
705 706
	struct device *dev = to_dev(kobj);
	struct net_bridge *br = to_bridge(dev);
L
Linus Torvalds 已提交
707 708 709 710 711 712
	int n;

	/* must read whole records */
	if (off % sizeof(struct __fdb_entry) != 0)
		return -EINVAL;

713
	n =  br_fdb_fillbuf(br, buf,
L
Linus Torvalds 已提交
714 715 716 717 718
			    count / sizeof(struct __fdb_entry),
			    off / sizeof(struct __fdb_entry));

	if (n > 0)
		n *= sizeof(struct __fdb_entry);
719

L
Linus Torvalds 已提交
720 721 722 723 724
	return n;
}

static struct bin_attribute bridge_forward = {
	.attr = { .name = SYSFS_BRIDGE_FDB,
725
		  .mode = S_IRUGO, },
L
Linus Torvalds 已提交
726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741
	.read = brforward_read,
};

/*
 * Add entries in sysfs onto the existing network class device
 * for the bridge.
 *   Adds a attribute group "bridge" containing tuning parameters.
 *   Binary attribute containing the forward table
 *   Sub directory to hold links to interfaces.
 *
 * Note: the ifobj exists only to be a subdirectory
 *   to hold links.  The ifobj exists in same data structure
 *   as it's parent the bridge so reference counting works.
 */
int br_sysfs_addbr(struct net_device *dev)
{
742
	struct kobject *brobj = &dev->dev.kobj;
L
Linus Torvalds 已提交
743 744 745 746 747 748
	struct net_bridge *br = netdev_priv(dev);
	int err;

	err = sysfs_create_group(brobj, &bridge_group);
	if (err) {
		pr_info("%s: can't create group %s/%s\n",
749
			__func__, dev->name, bridge_group.name);
L
Linus Torvalds 已提交
750 751 752 753 754
		goto out1;
	}

	err = sysfs_create_bin_file(brobj, &bridge_forward);
	if (err) {
755
		pr_info("%s: can't create attribute file %s/%s\n",
756
			__func__, dev->name, bridge_forward.attr.name);
L
Linus Torvalds 已提交
757 758 759
		goto out2;
	}

760 761
	br->ifobj = kobject_create_and_add(SYSFS_BRIDGE_PORT_SUBDIR, brobj);
	if (!br->ifobj) {
L
Linus Torvalds 已提交
762
		pr_info("%s: can't add kobject (directory) %s/%s\n",
763
			__func__, dev->name, SYSFS_BRIDGE_PORT_SUBDIR);
L
Linus Torvalds 已提交
764 765 766 767
		goto out3;
	}
	return 0;
 out3:
768
	sysfs_remove_bin_file(&dev->dev.kobj, &bridge_forward);
L
Linus Torvalds 已提交
769
 out2:
770
	sysfs_remove_group(&dev->dev.kobj, &bridge_group);
L
Linus Torvalds 已提交
771 772 773 774 775 776 777
 out1:
	return err;

}

void br_sysfs_delbr(struct net_device *dev)
{
778
	struct kobject *kobj = &dev->dev.kobj;
L
Linus Torvalds 已提交
779 780
	struct net_bridge *br = netdev_priv(dev);

781
	kobject_put(br->ifobj);
L
Linus Torvalds 已提交
782 783 784
	sysfs_remove_bin_file(kobj, &bridge_forward);
	sysfs_remove_group(kobj, &bridge_group);
}