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

📄 rtnetlink.c

📁 Linux Kernel 2.6.9 for OMAP1710
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * INET		An implementation of the TCP/IP protocol suite for the LINUX *		operating system.  INET is implemented using the  BSD Socket *		interface as the means of communication with the user level. * *		Routing netlink socket interface: protocol independent part. * * Authors:	Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru> * *		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. * *	Fixes: *	Vitaly E. Lavrov		RTA_OK arithmetics was wrong. */#include <linux/config.h>#include <linux/errno.h>#include <linux/module.h>#include <linux/types.h>#include <linux/socket.h>#include <linux/kernel.h>#include <linux/major.h>#include <linux/sched.h>#include <linux/timer.h>#include <linux/string.h>#include <linux/sockios.h>#include <linux/net.h>#include <linux/fcntl.h>#include <linux/mm.h>#include <linux/slab.h>#include <linux/interrupt.h>#include <linux/capability.h>#include <linux/skbuff.h>#include <linux/init.h>#include <linux/security.h>#include <asm/uaccess.h>#include <asm/system.h>#include <asm/string.h>#include <linux/inet.h>#include <linux/netdevice.h>#include <net/ip.h>#include <net/protocol.h>#include <net/arp.h>#include <net/route.h>#include <net/udp.h>#include <net/sock.h>#include <net/pkt_sched.h>DECLARE_MUTEX(rtnl_sem);void rtnl_lock(void){	rtnl_shlock();} void rtnl_unlock(void){	rtnl_shunlock();	netdev_run_todo();}int rtattr_parse(struct rtattr *tb[], int maxattr, struct rtattr *rta, int len){	memset(tb, 0, sizeof(struct rtattr*)*maxattr);	while (RTA_OK(rta, len)) {		unsigned flavor = rta->rta_type;		if (flavor && flavor <= maxattr)			tb[flavor-1] = rta;		rta = RTA_NEXT(rta, len);	}	return 0;}struct sock *rtnl;struct rtnetlink_link * rtnetlink_links[NPROTO];static const int rtm_min[(RTM_MAX+1-RTM_BASE)/4] ={	NLMSG_LENGTH(sizeof(struct ifinfomsg)),	NLMSG_LENGTH(sizeof(struct ifaddrmsg)),	NLMSG_LENGTH(sizeof(struct rtmsg)),	NLMSG_LENGTH(sizeof(struct ndmsg)),	NLMSG_LENGTH(sizeof(struct rtmsg)),	NLMSG_LENGTH(sizeof(struct tcmsg)),	NLMSG_LENGTH(sizeof(struct tcmsg)),	NLMSG_LENGTH(sizeof(struct tcmsg)),	NLMSG_LENGTH(sizeof(struct tcamsg))};static const int rta_max[(RTM_MAX+1-RTM_BASE)/4] ={	IFLA_MAX,	IFA_MAX,	RTA_MAX,	NDA_MAX,	RTA_MAX,	TCA_MAX,	TCA_MAX,	TCA_MAX,	TCAA_MAX};void __rta_fill(struct sk_buff *skb, int attrtype, int attrlen, const void *data){	struct rtattr *rta;	int size = RTA_LENGTH(attrlen);	rta = (struct rtattr*)skb_put(skb, RTA_ALIGN(size));	rta->rta_type = attrtype;	rta->rta_len = size;	memcpy(RTA_DATA(rta), data, attrlen);}int rtnetlink_send(struct sk_buff *skb, u32 pid, unsigned group, int echo){	int err = 0;	NETLINK_CB(skb).dst_groups = group;	if (echo)		atomic_inc(&skb->users);	netlink_broadcast(rtnl, skb, pid, group, GFP_KERNEL);	if (echo)		err = netlink_unicast(rtnl, skb, pid, MSG_DONTWAIT);	return err;}int rtnetlink_put_metrics(struct sk_buff *skb, u32 *metrics){	struct rtattr *mx = (struct rtattr*)skb->tail;	int i;	RTA_PUT(skb, RTA_METRICS, 0, NULL);	for (i=0; i<RTAX_MAX; i++) {		if (metrics[i])			RTA_PUT(skb, i+1, sizeof(u32), metrics+i);	}	mx->rta_len = skb->tail - (u8*)mx;	if (mx->rta_len == RTA_LENGTH(0))		skb_trim(skb, (u8*)mx - skb->data);	return 0;rtattr_failure:	skb_trim(skb, (u8*)mx - skb->data);	return -1;}static int rtnetlink_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,				 int type, u32 pid, u32 seq, u32 change){	struct ifinfomsg *r;	struct nlmsghdr  *nlh;	unsigned char	 *b = skb->tail;	nlh = NLMSG_PUT(skb, pid, seq, type, sizeof(*r));	if (pid) nlh->nlmsg_flags |= NLM_F_MULTI;	r = NLMSG_DATA(nlh);	r->ifi_family = AF_UNSPEC;	r->ifi_type = dev->type;	r->ifi_index = dev->ifindex;	r->ifi_flags = dev_get_flags(dev);	r->ifi_change = change;	RTA_PUT(skb, IFLA_IFNAME, strlen(dev->name)+1, dev->name);	if (1) {		u32 txqlen = dev->tx_queue_len;		RTA_PUT(skb, IFLA_TXQLEN, sizeof(txqlen), &txqlen);	}	if (1) {		u32 weight = dev->weight;		RTA_PUT(skb, IFLA_WEIGHT, sizeof(weight), &weight);	}	if (1) {		struct rtnl_link_ifmap map = {			.mem_start   = dev->mem_start,			.mem_end     = dev->mem_end,			.base_addr   = dev->base_addr,			.irq         = dev->irq,			.dma         = dev->dma,			.port        = dev->if_port,		};		RTA_PUT(skb, IFLA_MAP, sizeof(map), &map);	}	if (dev->addr_len) {		RTA_PUT(skb, IFLA_ADDRESS, dev->addr_len, dev->dev_addr);		RTA_PUT(skb, IFLA_BROADCAST, dev->addr_len, dev->broadcast);	}	if (1) {		u32 mtu = dev->mtu;		RTA_PUT(skb, IFLA_MTU, sizeof(mtu), &mtu);	}	if (dev->ifindex != dev->iflink) {		u32 iflink = dev->iflink;		RTA_PUT(skb, IFLA_LINK, sizeof(iflink), &iflink);	}	if (dev->qdisc_sleeping)		RTA_PUT(skb, IFLA_QDISC,			strlen(dev->qdisc_sleeping->ops->id) + 1,			dev->qdisc_sleeping->ops->id);		if (dev->master) {		u32 master = dev->master->ifindex;		RTA_PUT(skb, IFLA_MASTER, sizeof(master), &master);	}	if (dev->get_stats) {		unsigned long *stats = (unsigned long*)dev->get_stats(dev);		if (stats) {			struct rtattr  *a;			__u32	       *s;			int		i;			int		n = sizeof(struct rtnl_link_stats)/4;			a = __RTA_PUT(skb, IFLA_STATS, n*4);			s = RTA_DATA(a);			for (i=0; i<n; i++)				s[i] = stats[i];		}	}	nlh->nlmsg_len = skb->tail - b;	return skb->len;nlmsg_failure:rtattr_failure:	skb_trim(skb, b - skb->data);	return -1;}int rtnetlink_dump_ifinfo(struct sk_buff *skb, struct netlink_callback *cb){	int idx;	int s_idx = cb->args[0];	struct net_device *dev;	read_lock(&dev_base_lock);	for (dev=dev_base, idx=0; dev; dev = dev->next, idx++) {		if (idx < s_idx)			continue;		if (rtnetlink_fill_ifinfo(skb, dev, RTM_NEWLINK, NETLINK_CB(cb->skb).pid, cb->nlh->nlmsg_seq, 0) <= 0)			break;	}	read_unlock(&dev_base_lock);	cb->args[0] = idx;	return skb->len;}static int do_setlink(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg){	struct ifinfomsg  *ifm = NLMSG_DATA(nlh);	struct rtattr    **ida = arg;	struct net_device *dev;	int err, send_addr_notify = 0;	dev = dev_get_by_index(ifm->ifi_index);	if (!dev)		return -ENODEV;	err = -EINVAL;	if (ifm->ifi_flags)		dev_change_flags(dev, ifm->ifi_flags);	if (ida[IFLA_MAP - 1]) {		struct rtnl_link_ifmap *u_map;		struct ifmap k_map;		if (!dev->set_config) {			err = -EOPNOTSUPP;			goto out;		}		if (!netif_device_present(dev)) {			err = -ENODEV;			goto out;		}				if (ida[IFLA_MAP - 1]->rta_len != RTA_LENGTH(sizeof(*u_map)))			goto out;		u_map = RTA_DATA(ida[IFLA_MAP - 1]);		k_map.mem_start = (unsigned long) u_map->mem_start;		k_map.mem_end = (unsigned long) u_map->mem_end;		k_map.base_addr = (unsigned short) u_map->base_addr;		k_map.irq = (unsigned char) u_map->irq;		k_map.dma = (unsigned char) u_map->dma;		k_map.port = (unsigned char) u_map->port;		err = dev->set_config(dev, &k_map);		if (err)			goto out;	}	if (ida[IFLA_ADDRESS - 1]) {		if (!dev->set_mac_address) {			err = -EOPNOTSUPP;			goto out;		}		if (!netif_device_present(dev)) {			err = -ENODEV;			goto out;		}		if (ida[IFLA_ADDRESS - 1]->rta_len != RTA_LENGTH(dev->addr_len))			goto out;		err = dev->set_mac_address(dev, RTA_DATA(ida[IFLA_ADDRESS - 1]));		if (err)			goto out;		send_addr_notify = 1;	}	if (ida[IFLA_BROADCAST - 1]) {		if (ida[IFLA_BROADCAST - 1]->rta_len != RTA_LENGTH(dev->addr_len))			goto out;		memcpy(dev->broadcast, RTA_DATA(ida[IFLA_BROADCAST - 1]),		       dev->addr_len);		send_addr_notify = 1;	}	if (ida[IFLA_MTU - 1]) {		if (ida[IFLA_MTU - 1]->rta_len != RTA_LENGTH(sizeof(u32)))			goto out;		err = dev_set_mtu(dev, *((u32 *) RTA_DATA(ida[IFLA_MTU - 1])));		if (err)			goto out;

⌨️ 快捷键说明

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