当前位置: 首页 > news >正文

struct net_device网络设备结构体详解

转自:http://blog.csdn.net/viewsky11/article/details/53046787

 

在linux中使用struct net_device结构体来描述每一个网络设备。同时这个用来刻画网络设备的struct net_device结构体包含的字段非常的多,以至于内核的开发者都觉得在现在的linux内核中,这个struct net_device是一个大的错误。 
在本篇文章中,只介绍struct net_device中的一些字段,其他的字段在以后使用的时候再说。

#define IFNAMSIZ 32
   struct net_device
   {
       //用于存放网络设备的设备名称;
       char name[IFNAMSIZ];

       //网络设备的别名; char *ifalias; //网络设备的接口索引值,独一无二的网络设备标识符; int ifindex; //这个字段用于构建网络设备名的哈希散列表,而struct net中的 //name_hlist就指向每个哈希散列表的链表头; struct hlist_node name_hlist; //用于构建网络设备的接口索引值哈希散列表,在struct net中的 //index_hlist用于指向接口索引值哈希散列表的链表头; struct hlist_node index_hlist; //用于将每一个网络设备加入到一个网络命名空间中的网络设备双链表中 struct list_head dev_list; //网络设备接口的标识符,其状态类型被定义在<linux/if.h>之中; unsigned int flags; //网络设备接口的标识符,但对用户空间不可见; unsigned short priv_flags; //接口硬件类型,在<if_arp.h>中定义了每一个接口硬件类型; unsigned short type; //网络设备接口的最大传输单元; unsigned mtu; //硬件接口头长度; unsigned short hard_header_len; //网络设备接口的MAC地址; unsigned char *dev_addr; //网络设备接口的单播模式 int uc_promisc; //网络设备接口的混杂模式; unsigned int promiscuity; //网络设备接口的全组播模式; unsigend int allmulti; //secondary unicast mac address struct netdev_hw_addr_list uc; //list of device hw address; struct netdev_hw_addr_list dev_addrs; //hw broadcast address; unsigned char broadcast[MAX_ADDR_LEN]; //multicast mac address; struct dev_addr_list *mac_list; //网络设备接口的数据包接收队列; struct netdev_queue rx_queue; //网络设备接口的数据包发送队列; struct netdev_queue *tx; //Number of TX queues allocated at alloc_netdev_mq() time unsigned int num_tx_queues; //Number of TX queues currently active in device; unsigned int real_num_tx_queues; //Max frame per queue allowned; unsigned long tx_queue_len; //网络设备接口的状态; unsigned long state; //网络设备接口的统计情况; struct net_device_state states; //用于执行网络设备所在的命名空间; struct net *nd_net; } 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89

下面的这幅图用于展示linux内核如何利用struct net和struct net_device来构成一个网络命名空间: 
这里写图片描述

1.分配一个网络设备函数,即分配一个struct net_device结构体:alloc_netdev(sizeof_priv, name, setup); 

这个alloc_netdev()函数本质上是一个宏定义:

    #define alloc_netdev(sizeof_priv, name, setup) \
 alloc_netdev_mq(sizeof_priv, name, setup, 1)

struct net_device *alloc_netdev_mq(int sizeof_priv,
    const char *name, void(*setup)(struct net_device*),
    unsigned int queue_count)
 EXPORT_SYMBOL(alloc_netdev_mq)  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

参数介绍: 
- sizeof_priv:为分配给网络设备私有空间的大小; 
- name :网络设备的名称; 
- setup:对分配的网络设备进行初始化的回调函数; 
- queue_count:分配给网络设备的子队列数;

struct net_device *alloc_netdev_mq( int sizeof_priv, const char *name, 
            void (*setup)( struct net_device *), unsigned int queue_count ) { struct net_device *dev ; struct netdev_queue *tx ; int alloc_size ; alloc_size = sizeof(struct net_device); dev = kzalloc(alloc_size, GFP_KERNEL); tx = kcalloc(queue_count, sizeof(struct netdev_queue), GFP_KERNEL); dev_addr_init(dev);//对struct net_device中的dev_addrs成员进行初始化; dev_unicast_init(dev); //对struct net_device中的uc成员进行初始化; dev_net_set(dev, &init_net);//对网络设备的命名空间进行初始化,默认为init_net; dev->_tx = tx ;//设置网络设备的发送队列; dev->num_tx_queues = queue_count ; dev->real_num_tx_queues = queue_count ; netdev_init_queues(dev);//对struct net_device中的rx_queue成员进行初始化; setup(dev); //对struct net_device结构体进行初始化; strcpy(dev->name, name);//设置网路设备的设备名称; } EXPORT_SYMBOL(alloc_netdev_mq); 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35

2.释放一个网络设备:

void free_netdev(struct net_device *dev);
EXPORT_SYMBOL(free_netdev);
  • 1
  • 2
  • 3

3.注册一个网络设备,只有对一个网络设备进行注册以后,这个网络设备才会在内核中起作用:

 int register_netdev(struct net_device *dev)
EXPORT_SYMBOL(register_netdev);   
  • 1
  • 2
  • 3

返回值: 
- 0:表示注册成功; 
- a negative errno code :表示注册失败;

int register_netdev(struct net_device *dev)
{
    struct hlist_head *head;
    struct hlist_node *p;
    int ret;
    struct net *net = dev_net(dev); // 获取网络设备所在的命名空间; rtnl_lock();//获取rtnl信号量; if(!dev_valid_name(dev->name)) //判断网络设备的设备名是否有效; {} dev->ifindex = dev_new_index(net); //从网络设备所在的命名空间中找到一个全局唯一的网络 //接口索引值; dev->iflink = dev->ifindex ; //用于判断网络命名空间中是否有相同名字的网络设备存在; head = dev_name_hash(net, dev->name); hlist_for_each(p,head) { struct net_device *d = hlist_entry(p, struct net_device, name_hlist); if(!strncmp(d->name, dev->name, 32)) { ret = -EEXIST ; } } set_bit(__LINK_STATE_PRESENT, &dev->state);//设置网络设备的状态; list_netdevice(dev); //将网络设备叫入到相应的命名空间之中; } EXPORT_SYMBOL(register_netdev);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

3.注销一个网络设备结构体:

void unregister_netdev(struct net_device *dev);
EXPORT_SYMBOL(unregister_netdev);
  • 1
  • 2
  • 3

4.设置一个网络设备的MAC地址:

int dev_set_mac_address(struct net_device *dev, 
                    struct sockaddr *sa);
EXPORT_SYMBOL(dev_set_mac_address);
  • 1
  • 2
  • 3
  • 4

5.设置一个网络设备的最大传输单元:

int dev_set_mtu(struct net_device *dev, int new_mtu);
EXPORT_SYMBOL(dev_set_mtu);
  • 1
  • 2
  • 3

6.改变一个网络设备的flag标识符:

int dev_change_flags(struct net_device *dev, unsigend flags);
EXPORT_SYMBOL(dev_change_flags);
  • 1
  • 2
  • 3

7.获取一个网络设备的flag标识符:

unsigned dev_get_flags(struct net_device *dev);
EXPORT_SYMBOL(dev_get_flags);
  • 1
  • 2
  • 3

8.给网络设备添加一个单播MAC地址,当网络设备在发送单播时使用单播MAC地址

int dev_unicast_add(struct net_device *dev, void *addr);
EXPORT_SYMBOL(dev_unicast_add);
  • 1
  • 2
  • 3

9.删除网络设备中的单播MAC地址:

int dev_unicast_delete(struct net_device *dev, void *addr);
EXPORT_SYMBOL(dev_unicast_delete);
  • 1
  • 2
  • 3

10.给网络设备添加一个设备地址:

int dev_addr_add(struct net_device *dev, unsigned char *addr,
                  unsigned char addr_type)
EXPORT_SYMBOL(dev_addr_add);
addr_type : address type;
  • 1
  • 2
  • 3
  • 4
  • 5

11.删除网络设备中的一个设备地址:

int dev_addr_del(struct net_device *dev, unsigend char *addr,
                  unsigned char addr_type);
EXPORT_SYMBOL(dev_addr_del);
  • 1
  • 2
  • 3
  • 4

12.设置网络设备的接口为混杂模式:

 int dev_set_promiscuity(struct net_device *dev, int inc);
 EXPORT_SYMBOL(dev_set_promiscuity);
  • 1
  • 2
  • 3

当 inc > 0 将网络设备设置为混杂模式; 
当 inc = 0 将网络设备设置为正常模式; 
当 inc < 0 将去掉网络设备的混杂模式;

13.设置网络设备的接口为allmulticast模式:

 int dev_set_allmulti(struct net_device *dev, int inc);
 EXPORT_SYMBOL(dev_set_allmulti);
 inc 含义同上;
  • 1
  • 2
  • 3
  • 4

14.在给一个网络设备的设备名称进行赋值时,先要检测这个名字是否有效:

 int dev_valid_name(const char *name);
 EXPORT_SYMBOL(dev_valid_name);
  • 1
  • 2
  • 3

返回: 
1 :表示有效; 
0 :表示无效;

15.通过设备的MAC地址以及设备类型来获取网络设备的设备结构体:

  struct net_device *dev_getbyhwaddr(struct net *net,
              unsigned short type, char *hwaddr);
 EXPORT_SYMBOL(dev_getbyhwaddr);
  • 1
  • 2
  • 3
  • 4

net : 网络命名空间; 
type: media type of device; 
hwaddr:hardware address; 
通过memcmp(dev->dev_addr, hwaddr, dev->addr_len)来实现;

16.通过网络设备的接口索引值来获取网络设备结构体:

 struct net_device *dev_get_by_index(struct net *net, int ifindex);
 EXPORT_SYMBOL(dev_get_by_index);
  • 1
  • 2
  • 3

17.通过网络设备的设备名来获取网络设备结构体:

struct net_device *dev_get_by_name(struct net *net, const char *name);
  EXPORT_SYMBOL(dev_get_by_name);
  • 1
  • 2
  • 3

以上所介绍的所有函数位于/net/core/dev.c之中;

本文转载自 http://weiguozhihui.blog.51cto.com/3060615/1584894

 

相关文章:

  • Python操作MySQL数据库的三种方法
  • 如何下载腾讯视频的视频转为MP4常用格式视频
  • FreeBSD-musb_otg文件详解
  • centos7 安装wps 后 演示无法启动
  • NSString属性什么时候用copy,什么时候用strong?
  • 使用Visual Studio Code对Node.js进行断点调试
  • CentOS7.2升级openSSH为7.5P1无法登录的处理过程
  • linux复盘:mysql双主与mysql-proxy实现读写分离
  • 10.28 rsync工具介绍 10.29/10.30 rsync常用选项 10.31 rsync通
  • 三角形内随机生成一个点
  • 04.spring security oauth2认证中心 集成zuul网关的代码分析
  • 2018 掌握好这几点方法学习Linux,一定比别人更快入门运维!
  • python小白项目推荐
  • 使用Jackson来实现java对象和json格式的相互转换
  • Python设计模式——观察者模式
  • 【译】JS基础算法脚本:字符串结尾
  • @angular/forms 源码解析之双向绑定
  • [ 一起学React系列 -- 8 ] React中的文件上传
  • [原]深入对比数据科学工具箱:Python和R 非结构化数据的结构化
  • 【知识碎片】第三方登录弹窗效果
  • Date型的使用
  • java架构面试锦集:开源框架+并发+数据结构+大企必备面试题
  • JDK 6和JDK 7中的substring()方法
  • js中forEach回调同异步问题
  • nodejs:开发并发布一个nodejs包
  • PHP的类修饰符与访问修饰符
  • Puppeteer:浏览器控制器
  • PyCharm搭建GO开发环境(GO语言学习第1课)
  • Sublime Text 2/3 绑定Eclipse快捷键
  • vue-cli3搭建项目
  • vue-loader 源码解析系列之 selector
  • 基于Javascript, Springboot的管理系统报表查询页面代码设计
  • 聚簇索引和非聚簇索引
  • 开放才能进步!Angular和Wijmo一起走过的日子
  • 利用DataURL技术在网页上显示图片
  • 如何学习JavaEE,项目又该如何做?
  • 使用 QuickBI 搭建酷炫可视化分析
  • scrapy中间件源码分析及常用中间件大全
  • ​Kaggle X光肺炎检测比赛第二名方案解析 | CVPR 2020 Workshop
  • ​第20课 在Android Native开发中加入新的C++类
  • ​一些不规范的GTID使用场景
  • %@ page import=%的用法
  • ()、[]、{}、(())、[[]]命令替换
  • (07)Hive——窗口函数详解
  • (4) PIVOT 和 UPIVOT 的使用
  • (LeetCode C++)盛最多水的容器
  • (Note)C++中的继承方式
  • (附源码)ssm教师工作量核算统计系统 毕业设计 162307
  • (一)硬件制作--从零开始自制linux掌上电脑(F1C200S) <嵌入式项目>
  • (已解决)什么是vue导航守卫
  • (转)总结使用Unity 3D优化游戏运行性能的经验
  • ./configure,make,make install的作用
  • .cn根服务器被攻击之后
  • .NET 5种线程安全集合
  • .NET Core WebAPI中使用swagger版本控制,添加注释