《ifconfig源码分析之与内核交互数据》
本文档的Copyleft归rosetta所有,使用GPL发布,可以自由拷贝、转载,转载时请保持文档的完整性。
参考资料:《Linux设备驱动程序 第三版》,scull源码,Linux内核源码
来源:http://blog.csdn.net/rosetta/article/details/7563615

ifconifg是Linux提供的一个操作网络接口的应用层程序,虽然和设备驱动编写没什么联系,但分析它的部分核心代码有助于理解应用层和内核层交互过程。
    这也是对《字符设备驱动程序编写基础》最后提出的问题的一个解答。
    ifconifg.c文件一千多行再加上相关公共文件大概会达到二千行,只分析其与内核交互过程,其它部分有兴趣的朋友可以自行分析。

知识点:
* 获取ifconfig源码方法。
* ifconfig 输出结果解释。
* 应用层和内核层交互过程。
* ioctl的使用。
* 认识/proc/net/dev。

一、获取ifconifg源码包并编译。
  [root@xxx net-tools-1.60]# type ifconfig          
  ifconfig is hashed (/sbin/ifconfig)
  [root@xxx net-tools-1.60]# rpm -qf /sbin/ifconfig 
  net-tools-1.60-78.el5
  可知ifconfig属于net-tools源码包,下载之。net-tools源码包不仅包含ifconifg,还包含常用的arp、route、netstat等工具源码。
  
  直接make,应该会有错误,按着错误提示修改下源码即可。

二、ifconifg eth0执行结果解释
  [root@ xxx]# ./ifconfig eth0
  eth0      Link encap:Ethernet  HWaddr 00:0C:29:9a:26:37  
            inet addr:192.168.95.162  Bcast:192.168.95.255  Mask:255.255.255.0
            inet6 addr: fe80::21c:29ff:fe9b:2637/64 Scope:Link
            UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
            RX packets:2495308 errors:0 dropped:0 overruns:0 frame:0
            TX packets:2215616 errors:0 dropped:0 overruns:0 carrier:0
            collisions:0 txqueuelen:1000 
            RX bytes:998016881 (951.7 MiB)  TX bytes:886972155 (845.8 MiB)
            Interrupt:18 Base address:0x2000 
  Link encap:Ethernet   本网卡接入的网络的类型是以太网。
  HWaddr 00:0C:29:9a:26:37   本网卡的硬件地址。
  inet6 addr: fe80::21c:29ff:fe9b:2637/64 Scope:Link  ipv6地址。
  UP 网卡状态为开启。
  BROADCAST 支持广播。
  RUNNING 网卡的网线被接上。
  MULTICAST  支持多播。
  MTU:1500 IP数据包的最大长度,带IP头。
  RX表示接收数据包的情况。
  TX表示发送数据包的情况。
  如果网卡已经完成配置却还是无法与其它设备通信,那么从RX 和TX 的显示数据上可以简单地分析一下故障原因。在这种情况下,如果接收和传送的包的计数(packets)增加,那有可能是系统的IP地址出现了冲突;如果看到大量的错误(errors)和冲突(Collisions),那么这很有可能是网络的传输介质出了问题,例如网线不通或hub损坏。
  collisions: 网络讯号碰撞的情况说明
  txqueuelen: 传输缓区长度大小

三、认识/proc/net/dev
    这里列出了所有网络设备的其属性状态和收发包情况。ifconfig会open这个设备查找匹配信息。
  [root@xxx ipsec]# cat /proc/net/dev    
  Inter-|   Receive                                                |  Transmit
   face |bytes    packets errs drop fifo frame compressed multicast|bytes    packets errs drop fifo colls carrier compressed
      lo:   14920     167    0    0    0     0          0         0    14920     167    0    0    0     0       0          0
    eth0:104165628  231316    5    5    0     0          0         0 27195571  185064    0    0    0     0       0          0
    eth1:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
    eth2:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
    sit0:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
  ipsec0:     128       2    0    0    0     0          0         0      900       6    0    0    0     0       0          0
  ipsec1:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
  ipsec2:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
  ipsec3:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
     sn0:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0
     sn1:       0       0    0    0    0     0          0         0        0       0    0    0    0     0       0          0

四、分析./ifconfig eth0 源码执行流程
  前面部分是对选项的解析判断,给出函数调用过程,具体内容跳过。
  //ifconfig.c
  main()
   ->if_print()//输入参数为"eth0"
     ->lookup_interface()
     ->do_if_fetch()
       ->if_fetch()//从内核获取网卡信息,也是和内核交互的核心
       ->ife_print()//再把接收到的数据以第二步的格式打出

int if_fetch(struct interface *ife)
  {
      struct ifreq ifr;
      int fd;
      char *ifname = ife->name;
  
      strcpy(ifr.ifr_name, ifname);
      if (ioctl(skfd, SIOCGIFFLAGS, &ifr) < 0)//skfd为本地域套接字,SIOCGIFFLAGS为传给内核的cmd,ifr接收从内核返回的数据。
      return (-1);
      ife->flags = ifr.ifr_flags;
  
      strcpy(ifr.ifr_name, ifname);
      if (ioctl(skfd, SIOCGIFHWADDR, &ifr) < 0)
      memset(ife->hwaddr, 0, 32);
      else
      memcpy(ife->hwaddr, ifr.ifr_hwaddr.sa_data, 8);
  
      ife->type = ifr.ifr_hwaddr.sa_family;
      
      ……  
  }

讲到这里,我觉得就讲完了,虽然没有很高深的内容,但原本在脑海中模糊的概念已经变得清晰。

再帖上一段内核有关ioctl处理的源码:
int dev_ioctl(unsigned int cmd, void __user *arg)
{
    struct ifreq ifr;
    int ret;
    char *colon;

/* One special case: SIOCGIFCONF takes ifconf argument
       and requires shared lock, because it sleeps writing
       to user space.
     */

if (cmd == SIOCGIFCONF) {
        rtnl_shlock();
        ret = dev_ifconf((char __user *) arg);
        rtnl_shunlock();
        return ret;
    }
    if (cmd == SIOCGIFNAME)
        return dev_ifname((struct ifreq __user *)arg);

if (copy_from_user(&ifr, arg, sizeof(struct ifreq)))
        return -EFAULT;

ifr.ifr_name[IFNAMSIZ-1] = 0;
    colon = strchr(ifr.ifr_name, ':');
    if (colon)
        *colon = 0;

/*
     *  See which interface the caller is talking about.
     */

switch (cmd) {
        /*
         *  These ioctl calls:
         *  - can be done by all.
         *  - atomic and do not require locking.
         *  - return a value
         */
        case SIOCGIFFLAGS://here case
        case SIOCGIFMETRIC:
        case SIOCGIFMTU:
        case SIOCGIFHWADDR:
        case SIOCGIFSLAVE:
        case SIOCGIFMAP:
        case SIOCGIFINDEX:
        case SIOCGIFTXQLEN:
            dev_load(ifr.ifr_name);
            read_lock(&dev_base_lock);
            ret = dev_ifsioc(&ifr, cmd);//here
            read_unlock(&dev_base_lock);
            if (!ret) {
                if (colon)
                  *colon = ':';
                                if (copy_to_user(arg, &ifr,
                                         sizeof(struct ifreq)))
                                    ret = -EFAULT;
                            }
                            return ret;
    ……
    }
    
     /*          
     *  Perform the SIOCxIFxxx calls.
     */             
    static int dev_ifsioc(struct ifreq *ifr, unsigned int cmd)
    {               
        int err;             
        struct net_device *dev = __dev_get_by_name(ifr->ifr_name);
                
        if (!dev)
            return -ENODEV;
            
        switch (cmd) {
            case SIOCGIFFLAGS:  /* Get interface flags */
                ifr->ifr_flags = dev_get_flags(dev);//给ifr赋值
                return 0;
                
            case SIOCSIFFLAGS:  /* Set interface flags */
                return dev_change_flags(dev, ifr->ifr_flags);
    ……
    }

ifconfig源码分析之与内核交互数据的更多相关文章

  1. 鸿蒙内核源码分析(管道文件篇) | 如何降低数据流动成本 | 百篇博客分析OpenHarmony源码 | v70.01

    百篇博客系列篇.本篇为: v70.xx 鸿蒙内核源码分析(管道文件篇) | 如何降低数据流动成本 | 51.c.h.o 文件系统相关篇为: v62.xx 鸿蒙内核源码分析(文件概念篇) | 为什么说一 ...

  2. 鸿蒙内核源码分析(调度队列篇) | 内核有多少个调度队列 | 百篇博客分析OpenHarmony源码 | v6.05

    百篇博客系列篇.本篇为: v06.xx 鸿蒙内核源码分析(调度队列篇) | 内核有多少个调度队列 | 51.c.h .o 任务管理相关篇为: v03.xx 鸿蒙内核源码分析(时钟任务篇) | 触发调度 ...

  3. springMVC源码分析--FlashMap和FlashMapManager重定向数据保存

    在上一篇博客springMVC源码分析--页面跳转RedirectView(三)中我们看到了在RedirectView跳转时会将跳转之前的请求中的参数保存到fFlashMap中,然后通过FlashMa ...

  4. MongoDB源码分析——mongo与JavaScript交互

    mongo与JavaScript交互 源码版本为MongoDB 2.6分支     之前已经说过mongo是MongoDB提供的一个执行JavaScript脚本的客户端工具,执行js其实就是一个js和 ...

  5. yolov3源码分析keras(一)数据的处理

    一.前言 本次分析的源码为大佬复现的keras版本,上一波地址:https://github.com/qqwweee/keras-yolo3 初步打算重点分析两部分,第一部分为数据,即分析图像如何做等 ...

  6. MPTCP 源码分析(四) 发送和接收数据

    简述:      MPTCP在发送数据方面和TCP的区别是可以从多条路径中选择一条 路径来发送数据.MPTCP在接收数据方面与TCP的区别是子路径对无序包 进行重排后,MPTCP的mpcb需要多所有子 ...

  7. ViewPager部分源码分析一:加载数据

    onMeasure()调用populate(),完成首次数据初始化. populate()维护ViewPager的page,包括mItems和mAdapter. populate(): if (cur ...

  8. 5. SOFAJRaft源码分析— RheaKV中如何存放数据?

    概述 上一篇讲了RheaKV是如何进行初始化的,因为RheaKV主要是用来做KV存储的,RheaKV读写的是相当的复杂,一起写会篇幅太长,所以这一篇主要来讲一下RheaKV中如何存放数据. 我们这里使 ...

  9. 内核通信之Netlink源码分析-用户内核通信原理2

    2017-07-05 上文以一个简单的案例描述了通过Netlink进行用户.内核通信的流程,本节针对流程中的各个要点进行深入分析 sock的创建 sock管理结构 sendmsg源码分析  sock的 ...

随机推荐

  1. 我觉得epoll和select最大的区别

    最近在用epoll,网速资料很多,大家都说epoll和select的区别比较大,而且select要不停遍历所有的fd,效率要低,而且fd有限制. 但是我认为二者最大的区别在于 先看代码 while ( ...

  2. 网络虚拟化技术(二): TUN/TAP MACVLAN MACVTAP

    TUN 设备 TUN 设备是一种虚拟网络设备,通过此设备,程序可以方便得模拟网络行为.先来看看物理设备是如何工作的:

  3. LeetCode——Contains Duplicate II

    Description: Given an array of integers and an integer k, find out whether there there are two disti ...

  4. 二、微信小游戏开发 多线程Worker

    微信多线程Worker教程 微信多线程Worker API 一.创建Worker,并和当前线程通讯 多线程worker只能创建1个.能和当前线程互传数据. 创建worker 在微信开发者工具中,在当前 ...

  5. Fluent Nhibernate Mapping for Sql Views

    Views are mapped the same way tables are mapped except that you should put Readonly() in the mapping ...

  6. 【BZOJ5008】方师傅的房子 三角剖分

    [BZOJ5008]方师傅的房子 Description 方师傅来到了一个二维平面.他站在原点上,觉得这里风景不错,就建了一个房子.这个房子是n个点的凸多边形,原点一定严格在凸多边形内部.有m个人也到 ...

  7. iOS 如何在视图中添加一个用xib创建的view

    NSArray *nib = [[NSBundle mainBundle]loadNibNamed:[pages objectAtIndex:] owner:self options:nil]; // ...

  8. MUI窗口管理

    参考:窗口管理 http://dev.dcloud.net.cn/mui/window/ 页面初始化:在app开发中,若要使用HTML5+扩展api,必须等plusready事件发生后才能正常使用,m ...

  9. 微信开发(3):微信公众号发现金红包功能开发,利用第三方SDK实现(转)

    最近需求是 用户兑换微信红包,需要一些验证,加密,以及证书: 工欲善其事必先利其器 感谢前辈的微信SDK 已经维护三年了,还在维护中! 官方文档走一波 文档还是一如既往的 坑人啊,写的很简单,对简单明 ...

  10. 限制ip ssh远程登录

    有时候为了服务器的安全考虑,我们可以在服务器上做限制,禁止其他ip地址连接服务器. 方法一:修改ssh配置文件 其实做这个操作很简单,只需要改/etc/ssh/sshd_config配置文件,再最后一 ...