⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 libbridge_devif.c

📁 linux下桥接软件源码
💻 C
字号:
/* * Copyright (C) 2000 Lennert Buytenhek * * 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. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <errno.h>#include <string.h>#include <sys/fcntl.h>#include "libbridge.h"#include "libbridge_private.h"#ifdef HAVE_LIBSYSFS/* Given two two character "0a" convert it to a byte */static unsigned char getoctet(const char *cp) {	char t[3] = { cp[0], cp[1], 0 };	return strtoul(t, NULL, 16);}static struct sysfs_directory *bridge_sysfs_directory(const char *devname,						      const char *subname){	struct sysfs_directory *sdir;	struct sysfs_class_device *dev;	char path[SYSFS_PATH_MAX];	if (!devname)		return NULL;	if (!br_class_net) {		dprintf("can't find class_net\n");		return NULL;	}	dev = sysfs_get_class_device(br_class_net, (char *) devname);	if (!dev) {		dprintf("can't find device %s in %s\n", devname, br_class_net->path);		return NULL;	}	snprintf(path, SYSFS_PATH_MAX, "%s/%s", dev->path, subname);	sdir = sysfs_open_directory(path);	if (!sdir)		dprintf("can't open directory: %s\n", path);	return sdir;}static void fetch_id(struct sysfs_directory *sdir, const char *name, 		     struct bridge_id *id){	struct sysfs_attribute *attr;	memset(id, 0, sizeof(id));	attr = sysfs_get_directory_attribute(sdir, (char *) name);	if (!attr) {		dprintf("Can't find attribute %s/%s\n", sdir->path, name);		return;	}	if (strlen(attr->value) < 17) 		dprintf("Bad format for %s: '%s'\n", name, attr->value);	else {		const char *cp = attr->value;		id->prio[0] = getoctet(cp); cp += 2;		id->prio[1] = getoctet(cp); cp += 3;		id->addr[0] = getoctet(cp); cp += 2;		id->addr[1] = getoctet(cp); cp += 2;		id->addr[2] = getoctet(cp); cp += 2;		id->addr[3] = getoctet(cp); cp += 2;		id->addr[4] = getoctet(cp); cp += 2;		id->addr[5] = getoctet(cp);	}}/* Get a time value out of sysfs */static void fetch_tv(struct sysfs_directory *sdir, const char *name,		     struct timeval *tv){	struct sysfs_attribute *attr		= sysfs_get_directory_attribute(sdir, (char *) name);	if (!attr) {		dprintf("Can't find attribute %s/%s\n", sdir->path, name);		memset(tv, 0, sizeof(tv));		return;	}	__jiffies_to_tv(tv, strtoul(attr->value, NULL, 0));}/* Fetch an integer attribute out of sysfs. */static int fetch_int(struct sysfs_directory *sdir, const char *name){	struct sysfs_attribute *attr		= sysfs_get_directory_attribute(sdir, (char *) name);	int val = 0;	if (!attr) 		dprintf("Can't find attribute %s/%s\n", sdir->path, name);	else 		val = strtol(attr->value, NULL, 0);	return val;}#endif/* * Convert device name to an index in the list of ports in bridge. * * Old API does bridge operations as if ports were an array * inside bridge structure. */static int get_portno(const char *brname, const char *ifname){	int i;	int ifindex = if_nametoindex(ifname);	int ifindices[MAX_PORTS];	unsigned long args[4] = { BRCTL_GET_PORT_LIST,				  (unsigned long)ifindices, MAX_PORTS, 0 };	struct ifreq ifr;	if (ifindex <= 0)		goto error;	memset(ifindices, 0, sizeof(ifindices));	strncpy(ifr.ifr_name, brname, IFNAMSIZ);	ifr.ifr_data = (char *) &args;	if (ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr) < 0) {		dprintf("get_portno: get ports of %s failed: %s\n", 			brname, strerror(errno));		goto error;	}	for (i = 0; i < MAX_PORTS; i++) {		if (ifindices[i] == ifindex)			return i;	}	dprintf("%s is not a in bridge %s\n", ifname, brname); error:	return -1;}/* get information via ioctl */static int old_get_bridge_info(const char *bridge, struct bridge_info *info){	struct ifreq ifr;	struct __bridge_info i;	unsigned long args[4] = { BRCTL_GET_BRIDGE_INFO,				  (unsigned long) &i, 0, 0 };	memset(info, 0, sizeof(*info));	strncpy(ifr.ifr_name, bridge, IFNAMSIZ);	ifr.ifr_data = (char *) &args;	if (ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr) < 0) {		dprintf("%s: can't get info %s\n",			bridge, strerror(errno));		return errno;	}	memcpy(&info->designated_root, &i.designated_root, 8);	memcpy(&info->bridge_id, &i.bridge_id, 8);	info->root_path_cost = i.root_path_cost;	info->root_port = i.root_port;	info->topology_change = i.topology_change;	info->topology_change_detected = i.topology_change_detected;	info->stp_enabled = i.stp_enabled;	__jiffies_to_tv(&info->max_age, i.max_age);	__jiffies_to_tv(&info->hello_time, i.hello_time);	__jiffies_to_tv(&info->forward_delay, i.forward_delay);	__jiffies_to_tv(&info->bridge_max_age, i.bridge_max_age);	__jiffies_to_tv(&info->bridge_hello_time, i.bridge_hello_time);	__jiffies_to_tv(&info->bridge_forward_delay, i.bridge_forward_delay);	__jiffies_to_tv(&info->ageing_time, i.ageing_time);	__jiffies_to_tv(&info->hello_timer_value, i.hello_timer_value);	__jiffies_to_tv(&info->tcn_timer_value, i.tcn_timer_value);	__jiffies_to_tv(&info->topology_change_timer_value, 			i.topology_change_timer_value);	__jiffies_to_tv(&info->gc_timer_value, i.gc_timer_value);	return 0;}/* * Get bridge parameters using either sysfs or old * ioctl. */int br_get_bridge_info(const char *bridge, struct bridge_info *info){#ifndef HAVE_LIBSYSFS	return old_get_bridge_info(bridge, info);#else	struct sysfs_directory *sdir;	sdir = bridge_sysfs_directory(bridge, SYSFS_BRIDGE_ATTR);	if (!sdir) 		return old_get_bridge_info(bridge,info);	memset(info, 0, sizeof(*info));	fetch_id(sdir, "root_id", &info->designated_root);	fetch_id(sdir, "bridge_id", &info->bridge_id);	info->root_path_cost = fetch_int(sdir, "root_path_cost");	fetch_tv(sdir, "max_age", &info->max_age);	fetch_tv(sdir, "hello_time", &info->hello_time);	fetch_tv(sdir, "forward_delay", &info->forward_delay);	fetch_tv(sdir, "max_age", &info->bridge_max_age);	fetch_tv(sdir, "hello_time", &info->bridge_hello_time);	fetch_tv(sdir, "forward_delay", &info->bridge_forward_delay);	fetch_tv(sdir, "ageing_time", &info->ageing_time);	fetch_tv(sdir, "hello_timer", &info->hello_timer_value);	fetch_tv(sdir, "tcn_timer", &info->tcn_timer_value);	fetch_tv(sdir, "topology_change_timer", 		 &info->topology_change_timer_value);;	fetch_tv(sdir, "gc_timer", &info->gc_timer_value);	info->root_port = fetch_int(sdir, "root_port");	info->stp_enabled = fetch_int(sdir, "stp_state");	info->topology_change = fetch_int(sdir, "topology_change");	info->topology_change_detected = fetch_int(sdir, "topology_change_detected");	sysfs_close_directory(sdir);	return 0;#endif}static int old_get_port_info(const char *brname, const char *port,			     struct port_info *info){	struct __port_info i;	int index;	memset(info, 0, sizeof(*info));	index = get_portno(brname, port);	if (index < 0)		return errno;		else {		struct ifreq ifr;		unsigned long args[4] = { BRCTL_GET_PORT_INFO,					   (unsigned long) &i, index, 0 };			strncpy(ifr.ifr_name, brname, IFNAMSIZ);		ifr.ifr_data = (char *) &args;				if (ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr) < 0) {			dprintf("old can't get port %s(%d) info %s\n",				brname, index, strerror(errno));			return errno;		}	}	info->port_no = index;	memcpy(&info->designated_root, &i.designated_root, 8);	memcpy(&info->designated_bridge, &i.designated_bridge, 8);	info->port_id = i.port_id;	info->designated_port = i.designated_port;	info->path_cost = i.path_cost;	info->designated_cost = i.designated_cost;	info->state = i.state;	info->top_change_ack = i.top_change_ack;	info->config_pending = i.config_pending;	__jiffies_to_tv(&info->message_age_timer_value, 			i.message_age_timer_value);	__jiffies_to_tv(&info->forward_delay_timer_value, 			i.forward_delay_timer_value);	__jiffies_to_tv(&info->hold_timer_value, i.hold_timer_value);	return 0;}/* * Get information about port on bridge. */int br_get_port_info(const char *brname, const char *port, 		     struct port_info *info){#ifndef HAVE_LIBSYSFS	return old_get_port_info(brname, port, info);#else	struct sysfs_directory *sdir		= bridge_sysfs_directory(port, SYSFS_BRIDGE_PORT_ATTR);	if (!sdir) 		return old_get_port_info(brname, port, info);	memset(info, 0, sizeof(*info));	fetch_id(sdir, "designated_root", &info->designated_root);	fetch_id(sdir, "designated_bridge", &info->designated_bridge);	info->port_no = fetch_int(sdir, "port_no");	info->port_id = fetch_int(sdir, "port_id");	info->designated_port = fetch_int(sdir, "designated_port");	info->path_cost = fetch_int(sdir, "path_cost");	info->designated_cost = fetch_int(sdir, "designated_cost");	info->state = fetch_int(sdir, "state");	info->top_change_ack = fetch_int(sdir, "change_ack");	info->config_pending = fetch_int(sdir, "config_pending");	fetch_tv(sdir, "message_age_timer", 		 &info->message_age_timer_value);	fetch_tv(sdir, "forward_delay_timer",		 &info->forward_delay_timer_value);	fetch_tv(sdir, "hold_timer",		 &info->hold_timer_value);	sysfs_close_directory(sdir);	return 0;#endif}static int br_set(const char *bridge, const char *name,		  unsigned long value, unsigned long oldcode){	int ret;#ifdef HAVE_LIBSYSFS	struct sysfs_directory *sdir;		sdir = bridge_sysfs_directory(bridge, SYSFS_BRIDGE_ATTR);	if (sdir) {		struct sysfs_attribute *attr;		char buf[32];		sprintf(buf, "%ld", value);				attr = sysfs_get_directory_attribute(sdir, (char *) name);		if (attr) 			ret = sysfs_write_attribute(attr, buf, strlen(buf));		else {			ret = -1;			errno = EINVAL;		}		sysfs_close_directory(sdir);	} else #endif	{		struct ifreq ifr;		unsigned long args[4] = { oldcode, value, 0, 0 };				strncpy(ifr.ifr_name, bridge, IFNAMSIZ);		ifr.ifr_data = (char *) &args;		ret = ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr);	}	return ret < 0 ? errno : 0;}int br_set_bridge_forward_delay(const char *br, struct timeval *tv){	return br_set(br, "forward_delay", __tv_to_jiffies(tv),		      BRCTL_SET_BRIDGE_FORWARD_DELAY);}int br_set_bridge_hello_time(const char *br, struct timeval *tv){	return br_set(br, "hello_time", __tv_to_jiffies(tv),		      BRCTL_SET_BRIDGE_HELLO_TIME);}int br_set_bridge_max_age(const char *br, struct timeval *tv){	return br_set(br, "max_age", __tv_to_jiffies(tv),		      BRCTL_SET_BRIDGE_MAX_AGE);}int br_set_ageing_time(const char *br, struct timeval *tv){	return br_set(br, "ageing_time", __tv_to_jiffies(tv),		      BRCTL_SET_AGEING_TIME);}int br_set_stp_state(const char *br, int stp_state){	return br_set(br, "stp_state", stp_state, BRCTL_SET_BRIDGE_STP_STATE);}int br_set_bridge_priority(const char *br, int bridge_priority){	return br_set(br, "priority", bridge_priority, 		      BRCTL_SET_BRIDGE_PRIORITY);}static int port_set(const char *bridge, const char *ifname, 		    const char *name, unsigned long value, 		    unsigned long oldcode){	int ret, index;#ifdef HAVE_LIBSYSFS	struct sysfs_directory *sdir;	sdir = bridge_sysfs_directory(ifname, SYSFS_BRIDGE_PORT_ATTR);	if (sdir) {		struct sysfs_attribute *attr;		char buf[32];		sprintf(buf, "%ld", value);		attr = sysfs_get_directory_attribute(sdir, (char *) name);		if (attr) 			ret = sysfs_write_attribute(attr, buf, strlen(buf));		else {			ret = -1; 			errno = EINVAL;		}		sysfs_close_directory(sdir);	} else#endif	if ( (index = get_portno(bridge, ifname)) < 0)		ret = index;	else {		struct ifreq ifr;		unsigned long args[4] = { oldcode, index, value, 0 };		strncpy(ifr.ifr_name, bridge, IFNAMSIZ);		ifr.ifr_data = (char *) &args;		ret = ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr);	}	return ret < 0 ? errno : 0;}int br_set_port_priority(const char *bridge, const char *port, int priority){	return port_set(bridge, port, "priority", priority, BRCTL_SET_PORT_PRIORITY);}int br_set_path_cost(const char *bridge, const char *port, int cost){	return port_set(bridge, port, "path_cost", cost, BRCTL_SET_PATH_COST);}static inline void __copy_fdb(struct fdb_entry *ent, 			      const struct __fdb_entry *f){	memcpy(ent->mac_addr, f->mac_addr, 6);	ent->port_no = f->port_no;	ent->is_local = f->is_local;	__jiffies_to_tv(&ent->ageing_timer_value, f->ageing_timer_value);}int br_read_fdb(const char *bridge, struct fdb_entry *fdbs, 		unsigned long offset, int num){	int i, fd = -1, n;	struct __fdb_entry fe[num];#ifdef HAVE_LIBSYSFS	struct sysfs_class_device *dev;		/* open /sys/class/net/brXXX/brforward */	if (br_class_net &&	    (dev = sysfs_get_class_device(br_class_net, (char *) bridge))) {		char path[SYSFS_PATH_MAX];		snprintf(path, SYSFS_PATH_MAX, "%s/%s", dev->path, 			 SYSFS_BRIDGE_FDB);		fd = open(path, O_RDONLY, 0);	}	if (fd != -1) {		/* read records from file */		lseek(fd, offset*sizeof(struct __fdb_entry), SEEK_SET);		n = read(fd, fe, num*sizeof(struct __fdb_entry));		if (n > 0)			n /= sizeof(struct __fdb_entry);	} else#endif	{		/* old kernel, use ioctl */		unsigned long args[4] = { BRCTL_GET_FDB_ENTRIES,					  (unsigned long) fe,					  num, offset };		struct ifreq ifr;		int retries = 0;		strncpy(ifr.ifr_name, bridge, IFNAMSIZ);		ifr.ifr_data = (char *) args;	retry:		n = ioctl(br_socket_fd, SIOCDEVPRIVATE, &ifr);		/* table can change during ioctl processing */		if (n < 0 && errno == EAGAIN && ++retries < 10) {			sleep(0);			goto retry;		}	}	for (i = 0; i < n; i++) 		__copy_fdb(fdbs+i, fe+i);	if (fd > 0)		close(fd);		return n;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -