linux应用层监控网线插拔状态的实现

摘要:
因此,实现Linux应用程序层监控网线插入状态相当于自己编写ifconfig函数。基于这个想法,首先参考ifconfig的源代码。您可以参考Linux本身提供的ifconfig源代码,也可以在Internet上找到它。

1、对于基于linux2.4内核的uclinux系统如何实现在应用层监控网线插拔状态?

2、硬件环境:IPS100ARM7TDMI

3、实现过程

由于linux下的ifconfig命令就能够实现在应用层监控网线插拔状态,例如当网线连接正常时,使用ifconfig eth0命令,打印的信息中会有RUNNING,而拔掉网线后,再使用ifconfig eth0命令,RUNNING就不见了。所以,实现Linux应用层监控网线插入状态就相当于自己写一个ifconfig函数。

基于这个思路,首先参考ifconfig的源码,可以参考linux自身提供的ifconfig的源码(linux提供的有ifconfig.c函数),也可以在网上查找。首先找到linux自身提供的ifconfig.c函数,既然ifconfig通过RUNNING来判断网络的通断的状况的,首先找到RUNNING的出处,搜索一下发现这句话,

If(ptr->flags &IFF_RUNNIG)

{

Printf(__("RUNNING"));

}

以这个为切入点,层层向上找,分别是被些函数调用,最后我们进入了main函数(这是理所当然的),在这里的到RUNNING→ife_print()→if_pirint()→main().。先看ife_prinf()函数,这里没有和内核通信,这时我们再看if_print()函数,这里刚好有和内核通信的函数,else {                       
        struct interface *ife;

        ife = lookup_interface(ifname);
        res = do_if_fetch(ife);
        if (res >= 0)
            ife_print(ife);
    }

    这时需要看到lookup_interface(ifname)和do_if_fetch(ife)的原型,由于头文件很多,我没办法知道这两个函数在那个头文件中,所以干脆在网上找到这两个函数的原型,我们找到了一篇ifconfing源码分析的文章,http://viscar.blog.sohu.com/2574772.html,这里面找到了函数原型,我们看到 do_if_fetch()函数里面又调用了if_fetch()函数,太好了,这个函数正是我们所需要的,在这里面有个很重要的函数ioctl(),ioctl用于向设备发送控制和配置命令,驱动程序可以接收ioctl的数据,并返回数据,ioctl的原型为

    ioctl(int d, int cmd, ......),

d是某个设备的文件描述符,cmdioctl的命令,可变参数取决于cmd,是指向变量或结构体的指针。

这里面用到的设备文件描述符skfd = socketAF_INET,SOCK_DGRAM,0);这是一个套接字,作用是打开一个网络通讯端口,成功的话返回skfd,相当于一个文件描述符。

   有了这些之后我们就可以写一个自己的简洁版的ifconfig函数了,现在ubuntu10.04上编写代码,代码里面的ioctl函数这样写ioctl(skfd, SIOCGIFFLAGS, &ifr) ,其中SIOCGIGGLAGS表示得到sock i/oflags,这时因为,RUNNIGN的条件是ptr->flags &IFF_RUNNING 是否为真,代码里面直接体现eth0,函数为strcpy(ifr.ifr_name, “eth0”);完整的代码如下:
(以下代码在ubuntu 10.04下运行通过)

#include "icconst.h"

#include <sys/socket.h>

#include <sys/ioctl.h>

#include <linux/if.h>

#include <string.h>

#include <stdio.h>

#include "net_detect.h"

 

int net_detect(char* net_name)

{

int skfd = 0;

struct ifreq ifr;

 

skfd = socket(AF_INET, SOCK_DGRAM, 0);

if(skfd < 0)

{

printf("%s:%d Open socket error!\n", __FILE__, __LINE__);

return -1;

}

 

strcpy(ifr.ifr_name, net_name);

 

if(ioctl(skfd, SIOCGIFFLAGS, &ifr) <0 )

{

printf("%s:%d IOCTL error!\n", __FILE__, __LINE__);

printf("Maybe ethernet inferface %s is not valid!", ifr.ifr_name);

close(skfd);

return -1;

}

 

if(ifr.ifr_flags & IFF_RUNNING)

{

printf("%s is running :)\n", ifr.ifr_name);

}

else

{

printf("%s is not running :(\n", ifr.ifr_name);

}

/*****************以下为识别ip地址,mac地址***************************************/

if(ioctl(skfd,SIOCGIFADDR,&ifr)<0)

{

  printf("SIOCGIFADDR IOCTL error!\n");

  close(skfd);

  return -1;

}

printf("ip addr :[%s]\n",inet_ntoa(((struct sockaddr_in*)&(ifr.ifr_addr))->sin_addr));

if(ioctl(skfd,SIOCGIFHWADDR,&ifr)<0)

{

  printf("SIOCGIFHWADDR IOCTL error!\n");

  close(skfd);

  return -1;

}

printf("mac addr: %02x:%02x:%02x:%02x:%02x:%02x\n"

    (unsigned char)ifr.ifr_hwaddr.sa_data[0],

    (unsigned char)ifr.ifr_hwaddr.sa_data[1],

    (unsigned char)ifr.ifr_hwaddr.sa_data[2],

    (unsigned char)ifr.ifr_hwaddr.sa_data[3],

    (unsigned char)ifr.ifr_hwaddr.sa_data[4],

    (unsigned char)ifr.ifr_hwaddr.sa_data[5]  );

/*************************************************************************************************************************************/

close(skfd);

 

return 0;

}

代码里面的struct ifreq 是一个设备请求的结构体,在<linux/if.h>中定义,SIOCGIFFLAGS使用了ifreq结构,

  在ubuntu10.04环境下编译之后,运行#./a.out eth0  即可实现网线插拔的监控。

这时把代码加到uclinux下运行,发现并不能实现ubuntu的效果,这是为什么呢?

初步分析原因,可能是在驱动程序中没有将插拔状态的信息通知内核,所以我们使用ioctl实际上不能获得内核网络设备的状态的信息。我们在《linux设备驱动程序这本书上》看到有两个函数刚好做这件事情:

Void netif_carrier_off(struct net_device*dev);

Void netif_carrier_on(struct net_device*dev);

当驱动检测到设备没有连接好,可以调用netif_carrier_off通知内核这一事情;当设备再次连接好时,调用netif_carrier_on通知内核现在连接好了。

现在我们将这两个函数分别加到驱动程序中,放到监控网线插拔状态的位置,再在unlinux中插拔网线时,会在串口终端打印出相应的状态信息,这时的状态信息完全是在应用层实现的。

 

4.参考文档:uclinux内核网络驱动源码和ifconfig的源码

           《linux设备驱动程序》

 

免责声明:文章转载自《linux应用层监控网线插拔状态的实现》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇使用 IDEA和Maven 整合SSH框架springboot 显式配置数据源 解决mybatis注解无法被识别问题下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

C语言精要总结-内存地址对齐与struct大小判断篇

在笔试时,经常会遇到结构体大小的问题,实际就是在考内存地址对齐。在实际开发中,如果一个结构体会在内存中高频地分配创建,那么掌握内存地址对齐规则,通过简单地自定义对齐方式,或者调整结构体成员的顺序,可以有效地减少内存使用。另外,一些不用边界对齐、可以在任何地址(包括奇数地址)引用任何数据类型的的机器,不在本文讨论范围之内。 什么是地址对齐 计算机读取或者写入...

查看linux操作系统版本

转自:https://www.cnblogs.com/vaelailai/p/7545166.html 一、linux下如何查看已安装的centos版本信息: 1.Linux查看当前操作系统版本信息  cat /proc/version Linux version 2.6.32-696.el6.x86_64 (mockbuild@c1bm.rdu2.cen...

zookeeper集群(linux)和单机(windows)搭建

Zookeeper集群(linxu)搭建 环境要求:必须要有jdk环境,本次讲课使用jdk1.8 3.1结构 一共三个节点集群的服务器数量一般为2n+1个,(zk服务器集群规模不小于3个节点),要求服务器之间系统时间保持一致。 3.2上传zk并且解压进行解压: tar -zxvf zookeeper-3.4.6.tar.gz重命名: mv zookeep...

Linux系统时间和硬件时间设置

在linux系统中有两个时间,硬件时钟和系统时间,硬件时钟是指主机板上的时钟设备,也就是通常可在BIOS画面设定的时钟。系统时钟则是指kernel中的时钟。所有Linux相关指令与函数都是读取系统时钟的设定。因为存在两种不同的时钟,那么它们之间就会存在差异。当Linux启动时,系统时钟会去读取硬件时钟的设定,之后系统时钟即独立运作。 用date命令对系统时...

Linux配置阿里epl源

去阿里云 有源仓库 阿里云镜像官方站点https://developer.aliyun.com/mirror/ 先备份本机上的源 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup 根据对应版本下载新的 CentOS-Base.repo 到 /etc/...

Redis在Linux下的安装

一、下载地址 ①redis中文网下载地址:http://www.redis.cn/ ②百度云网盘下载地址:https://pan.baidu.com/s/1UQcF9V3lwA0fxquM_JFMZw 提取码:lnwk 二、编译软件安装 yum -y install gcc gcc-c++ make 三、安装 解压后进入目录,编译安装命令: make ma...