取得Unix/Linux系统中的IP、MAC地址等信息
获得Unix/Linux系统中的IP、MAC地址等信息分类参数二(宏)参数三描述接口SIOCGIFCONFstruct ifconf获得所有
获得Unix/Linux系统中的IP、MAC地址等信息
分类参数二(宏)参数三描述接口SIOCGIFCONFstruct ifconf获得所有接口列表 SIOCGIFADDRstruct ifreq获得接口地址 SIOCGIFFLAGSstruct ifreq获得接口标志 SIOCGIFBRDADDRstruct ifreq获得广播地址 SIOCGIFNETMASKstruct ifreq获得子网掩码
上表中列出了两个相关的结构体:struct ifconf 和 struct ifreq,要了解ioctl函数的具体运用,首先要了解这两个结构:

ioctl函数中的struct ifconf 和 struct ifreq结构关系
通常运用ioctl函数的第一步是从内核获取系统的所有接口,然后再针对每个接口获取其地址信息。获取所有接口通过SIOCGIFCONF请求来实现:
- struct ifconf ifc; /* ifconf结构 */
- struct ifreq ifrs[16]; /* ifreq结构数组(这里估计了接口的最大数量16) */
-
- /* 初始化ifconf结构 */
- ifc.ifc_len = sizeof(ifrs);
- ifc.ifc_buf = (caddr_t) ifrs;
-
- /* 获得接口列表 */
- ioctl(fd, SIOCGIFCONF,(char *) &ifc);
获得了接口列表,就可以通过struct ifconf结构中*ifcu_req的指针得到struct ifreq结构数组的地址,通过遍历获得每隔接口的详细地址信息:
- printf("接口名称:%s/n", ifrs[n].ifr_name);/* 接口名称 */
-
- /* 获得IP地址 */
- ioctl(fd, SIOCGIFADDR,(char *) &ifrs[n]);
- printf("IP地址:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));
-
- /* 获得子网掩码 */
- ioctl(fd, SIOCGIFNETMASK,(char *) &ifrs[n]);
- printf("子网掩码:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));
-
- /* 获得广播地址 */
- ioctl(fd, SIOCGIFBRDADDR,(char *) &ifrs[n]);
- printf("广播地址:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));
-
- /* 获得MAC地址 */
- ioctl(fd, SIOCGIFHWADDR,(char *) &ifrs[n]);
- printf("MAC地址:%02x:%02x:%02x:%02x:%02x:%02x/n",
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[0],
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[1],
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[2],
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[3],
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[4],
- (unsignedchar) ifrs[n].ifr_hwaddr.sa_data[5]);
最后,给出一个参考程序代码。
ioctl函数没有纳入POXIS规范,各系统对ioctl的实现也不尽相同,下面的代码在我的Ubuntu10.04 linux上可执行通过,但在其他Unix系统上不一定能够通过编译,例如在Power AIX 5.3上需要将获得MAC地址的那段代码注释掉。
- #include <arpa/inet.h>
- #include <net/if.h>
- #include <net/if_arp.h>
- #include <netinet/in.h>
- #include <stdio.h>
- #include <sys/ioctl.h>
- #include <sys/socket.h>
- #include <unistd.h>
-
- #define MAXINTERFACES 16 /* 最大接口数 */
-
- int fd; /* 套接字 */
- int if_len; /* 接口数量 */
- struct ifreq buf[MAXINTERFACES]; /* ifreq结构数组 */
- struct ifconf ifc; /* ifconf结构 */
-
- int main(argc, argv)
- {
- /* 建立IPv4的UDP套接字fd */
- if((fd = socket(AF_INET, SOCK_DGRAM,0)) ==-1)
- {
- perror("socket(AF_INET, SOCK_DGRAM, 0)");
- return-1;
- }
-
- /* 初始化ifconf结构 */
- ifc.ifc_len = sizeof(buf);
- ifc.ifc_buf = (caddr_t) buf;
-
- /* 获得接口列表 */
- if(ioctl(fd, SIOCGIFCONF,(char *) &ifc) ==-1)
- {
- perror("SIOCGIFCONF ioctl");
- return-1;
- }
-
- if_len = ifc.ifc_len / sizeof(struct ifreq);/* 接口数量 */
- printf("接口数量:%d/n/n", if_len);
-
- while(if_len– > 0)/* 遍历每个接口 */
- {
- printf("接口:%s/n", buf[if_len].ifr_name);/* 接口名称 */
-
- /* 获得接口标志 */
- if(!(ioctl(fd, SIOCGIFFLAGS,(char *) &buf[if_len])))
- {
- /* 接口状态 */
- if(buf[if_len].ifr_flags & IFF_UP)
- {
- printf("接口状态: UP/n");
- }
- else
- {
- printf("接口状态: DOWN/n");
- }
- }
- else
- {
- char str[256];
- sprintf(str,"SIOCGIFFLAGS ioctl %s", buf[if_len].ifr_name);
- perror(str);
- }
-
-
- /* IP地址 */
- if(!(ioctl(fd, SIOCGIFADDR,(char *) &buf[if_len])))
- {
- printf("IP地址:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));
- }
- else
- {
- char str[256];
- sprintf(str,"SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);
- perror(str);
- }
-
- /* 子网掩码 */
- if(!(ioctl(fd, SIOCGIFNETMASK,(char *) &buf[if_len])))
- {
- printf("子网掩码:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));
- }
- else
- {
- char str[256];
- sprintf(str,"SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);
- perror(str);
- }
-
- /* 广播地址 */
- if(!(ioctl(fd, SIOCGIFBRDADDR,(char *) &buf[if_len])))
- {
- printf("广播地址:%s/n",
- (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));
- }
- else
- {
- char str[256];
- sprintf(str,"SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);
- perror(str);
- }
-
- /*MAC地址 */
- if(!(ioctl(fd, SIOCGIFHWADDR,(char *) &buf[if_len])))
- {
- printf("MAC地址:%02x:%02x:%02x:%02x:%02x:%02x/n/n",
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[0],
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[1],
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[2],
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[3],
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[4],
- (unsignedchar) buf[if_len].ifr_hwaddr.sa_data[5]);
- }
- else
- {
- char str[256];
- sprintf(str,"SIOCGIFHWADDR ioctl %s", buf[if_len].ifr_name);
- perror(str);
- }
- }//–while end
-
- //关闭socket
- close(fd);
- return0;
- }
在我的系统上,程序输出:
接口数量:4
接口:wlan0
接口状态: UP
IP地址:192.168.1.142
子网掩码:255.255.255.0
广播地址:192.168.1.255
MAC地址:00:14:a5:65:47:57
接口:eth0:0
接口状态: UP
IP地址:192.168.4.113
子网掩码:255.255.255.0
广播地址:192.168.4.255
MAC地址:00:14:c2:e5:45:57
接口:eth0
接口状态: UP
IP地址:192.168.4.111
子网掩码:255.255.255.0
广播地址:192.168.4.255
MAC地址:00:14:c2:e5:45:57
接口:lo
接口状态: UP
IP地址:127.0.0.1
子网掩码:255.0.0.0
广播地址:0.0.0.0
MAC地址:00:00:00:00:00:00
从输出可以看出,系统有4个接口,”wlan0″表示第一块无线网卡接口,”eth0″(IP地址:192.168.4.111)表示第一块连线网卡接口(我们最长用的RJ45连接口网卡),”lo”是回路地址接口(我们常用的127.0.0.1)。
注意:”eth0:0″(IP地址:192.168.4.113)是有线网卡的别名(单网卡绑定多个IP),这是为了测试这个参考程序特意在eth0上添加的一个IP地址。