获得Unix/Linux系统中的IP、MAC地址等信息

获得Unix/Linux系统中的IP、MAC地址等信息

中高级  |  2010-07-13 16:03  |  分类:①C语言、 Unix/Linux、 网络编程 ②手册  |  4,471 次阅读

作者:diaoyf  |  文章来源:http://programmerdigest.cn

实际环境和特殊需求往往会将简单问题复杂化,比如计算机IP地址,对于一个连接中socket,可以直接获得本端和对端的IP、端口信息。但在一些特殊场合我们可能需要更多的信息,比如系统中有几块网卡,他们的Mac地址是多少,每块网卡分配了几个IP(一个网卡对应多个IP)等等。

这些信息往往需要通过ifconfig指令来获得,对于程序员来说,在代码中调用外部的shell指令可不是个最佳方案,因为没人能保障不同平台、不同版本的ifconfig指令输出的格式是一致的。本篇文章中将介绍通过ioctl函数实现上述需求。

 

#include <sys/ioctl.h>
int ioctl(int fd, int request, … /* void *arg */);
返回:成功返回0,失败返回-1

ioctl函数的参数只有3个,但却是Unix中少有的几个“家族类”复杂函数,这里摘录一段《Unix网络编程》一书中对ioctl函数的描述:

在传统上ioctl函数是用于那些普遍使用、但不适合归入其他类别的任何特殊的系统接口……网络程序(一般是服务器程序)中ioctl常用于在程序启动时获得主机上所有接口的信息:接口的地址、接口是否支持广播、是否支持多播,等等。

ioctl函数的第一个参数fd,可以表示一个打开的文件(文件句柄)或网络套接字,第二个和第三个参数体现了函数的家族特色,参数二request根据函数功能分类定义了多组宏,而参数三总是一个指针,指针的类型依赖于参数二request。因为ioctl的种类实在太多,这里只列出和本文相关的几个参数定义:

分类 参数二(宏) 参数三 描述
接口 SIOCGIFCONF struct ifconf 获得所有接口列表
  SIOCGIFADDR struct ifreq 获得接口地址
  SIOCGIFFLAGS struct ifreq 获得接口标志
  SIOCGIFBRDADDR struct ifreq 获得广播地址
  SIOCGIFNETMASK struct ifreq 获得子网掩码

 

上表中列出了两个相关的结构体:struct ifconf 和 struct ifreq,要了解ioctl函数的具体运用,首先要了解这两个结构:

 

 

[cpp] view plain copy

 

  1. /* net/if.h */  
  2. struct ifconf  
  3. {  
  4.     int ifc_len;            /* Size of buffer.  */  
  5.     union  
  6.     {  
  7.         __caddr_t ifcu_buf;  
  8.         struct ifreq *ifcu_req;  
  9.     } ifc_ifcu;  
  10. };  
  11.    
  12. struct ifreq  
  13. {  
  14. # define IFHWADDRLEN    6  
  15. # define IFNAMSIZ   IF_NAMESIZE  
  16.      
  17.     union  
  18.     {  
  19.         char ifrn_name[IFNAMSIZ];   /* Interface name, e.g. "en0".  */  
  20.     } ifr_ifrn;  
  21.      
  22.     union  
  23.     {  
  24.         struct sockaddr ifru_addr;  
  25.         struct sockaddr ifru_dstaddr;  
  26.         struct sockaddr ifru_broadaddr;  
  27.         struct sockaddr ifru_netmask;  
  28.         struct sockaddr ifru_hwaddr;  
  29.         short int ifru_flags;  
  30.         int ifru_ivalue;  
  31.         int ifru_mtu;  
  32.         struct ifmap ifru_map;  
  33.         char ifru_slave[IFNAMSIZ];  /* Just fits the size */  
  34.         char ifru_newname[IFNAMSIZ];  
  35.         __caddr_t ifru_data;  
  36.     } ifr_ifru;  
  37. };  

 

 

 

struct ifconf的第二个元素ifc_ifcu是一个联合,是指向struct ifreq结构的地址,通常是一组struct ifreq结构空间(每一个描述一个接口),struct ifconf的第一个元素ifc_len描述了struct ifreq结构空间的大小;结构struct ifreq也有两个元素,第一个元素ifr_ifrn内含一个字符串,用来描述接口的名称,比如“eth0″、”wlan0”等,第二个元素是联合,比较复杂,用来描述套接口的地址结构。

struct ifconf 和 struct ifreq的关系可以参考下图:


ioctl函数中的struct ifconf 和 struct ifreq结构关系

通常运用ioctl函数的第一步是从内核获取系统的所有接口,然后再针对每个接口获取其地址信息。获取所有接口通过SIOCGIFCONF请求来实现:

 

 

[cpp] view plain copy

 

  1. /* net/if.h */  
  2. struct ifconf  
  3. {  
  4.     int ifc_len;            /* Size of buffer.  */  
  5.     union  
  6.     {  
  7.         __caddr_t ifcu_buf;  
  8.         struct ifreq *ifcu_req;  
  9.     } ifc_ifcu;  
  10. };  
  11.    
  12. struct ifreq  
  13. {  
  14. # define IFHWADDRLEN    6  
  15. # define IFNAMSIZ   IF_NAMESIZE  
  16.      
  17.     union  
  18.     {  
  19.         char ifrn_name[IFNAMSIZ];   /* Interface name, e.g. "en0".  */  
  20.     } ifr_ifrn;  
  21.      
  22.     union  
  23.     {  
  24.         struct sockaddr ifru_addr;  
  25.         struct sockaddr ifru_dstaddr;  
  26.         struct sockaddr ifru_broadaddr;  
  27.         struct sockaddr ifru_netmask;  
  28.         struct sockaddr ifru_hwaddr;  
  29.         short int ifru_flags;  
  30.         int ifru_ivalue;  
  31.         int ifru_mtu;  
  32.         struct ifmap ifru_map;  
  33.         char ifru_slave[IFNAMSIZ];  /* Just fits the size */  
  34.         char ifru_newname[IFNAMSIZ];  
  35.         __caddr_t ifru_data;  
  36.     } ifr_ifru;  
  37. };  

 

 

获得了接口列表,就可以通过struct ifconf结构中*ifcu_req的指针得到struct ifreq结构数组的地址,通过遍历获得每隔接口的详细地址信息:

 

[cpp] view plain copy

 

  1. printf("接口名称:%s\n", ifrs[n].ifr_name); /* 接口名称 */  
  2.    
  3. /* 获得IP地址 */  
  4. ioctl(fd, SIOCGIFADDR, (char *) &ifrs[n]);  
  5. printf("IP地址:%s\n",  
  6.     (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));  
  7.    
  8. /* 获得子网掩码 */  
  9. ioctl(fd, SIOCGIFNETMASK, (char *) &ifrs[n]);  
  10. printf("子网掩码:%s\n",  
  11.     (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));  
  12.    
  13. /* 获得广播地址 */  
  14. ioctl(fd, SIOCGIFBRDADDR, (char *) &ifrs[n]);  
  15. printf("广播地址:%s\n",  
  16.     (char*)inet_ntoa(((struct sockaddr_in*) (&ifrs[n].ifr_addr))->sin_addr));  
  17.    
  18. /* 获得MAC地址 */  
  19. ioctl(fd, SIOCGIFHWADDR, (char *) &ifrs[n]);  
  20. printf("MAC地址:%02x:%02x:%02x:%02x:%02x:%02x\n",  
  21.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[0],  
  22.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[1],  
  23.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[2],  
  24.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[3],  
  25.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[4],  
  26.     (unsigned char) ifrs[n].ifr_hwaddr.sa_data[5]);  

 

 

最后,给出一个参考程序代码。

ioctl函数没有纳入POXIS规范,各系统对ioctl的实现也不尽相同,下面的代码在我的Ubuntu10.04 linux上可执行通过,但在其他Unix系统上不一定能够通过编译,例如在Power AIX 5.3上需要将获得MAC地址的那段代码注释掉。

 

 

[cpp] view plain copy

 

  1. #include <arpa/inet.h>  
  2. #include <net/if.h>  
  3. #include <net/if_arp.h>  
  4. #include <netinet/in.h>  
  5. #include <stdio.h>  
  6. #include <sys/ioctl.h>  
  7. #include <sys/socket.h>  
  8. #include <unistd.h>  
  9.    
  10. #define MAXINTERFACES 16    /* 最大接口数 */  
  11.    
  12. int fd;         /* 套接字 */  
  13. int if_len;     /* 接口数量 */  
  14. struct ifreq buf[MAXINTERFACES];    /* ifreq结构数组 */  
  15. struct ifconf ifc;                  /* ifconf结构 */  
  16.    
  17. int main(argc, argv)  
  18. {  
  19.     /* 建立IPv4的UDP套接字fd */  
  20.     if ((fd = socket(AF_INET, SOCK_DGRAM, 0)) == -1)  
  21.     {  
  22.         perror("socket(AF_INET, SOCK_DGRAM, 0)");  
  23.         return -1;  
  24.     }  
  25.    
  26.     /* 初始化ifconf结构 */  
  27.     ifc.ifc_len = sizeof(buf);  
  28.     ifc.ifc_buf = (caddr_t) buf;  
  29.    
  30.     /* 获得接口列表 */  
  31.     if (ioctl(fd, SIOCGIFCONF, (char *) &ifc) == -1)  
  32.     {  
  33.         perror("SIOCGIFCONF ioctl");  
  34.         return -1;  
  35.     }  
  36.    
  37.     if_len = ifc.ifc_len / sizeof(struct ifreq); /* 接口数量 */  
  38.     printf("接口数量:%d\n\n", if_len);  
  39.    
  40.     while (if_len– > 0) /* 遍历每个接口 */  
  41.     {  
  42.         printf("接口:%s\n", buf[if_len].ifr_name); /* 接口名称 */  
  43.    
  44.         /* 获得接口标志 */  
  45.         if (!(ioctl(fd, SIOCGIFFLAGS, (char *) &buf[if_len])))  
  46.         {  
  47.             /* 接口状态 */  
  48.             if (buf[if_len].ifr_flags & IFF_UP)  
  49.             {  
  50.                 printf("接口状态: UP\n");  
  51.             }  
  52.             else  
  53.             {  
  54.                 printf("接口状态: DOWN\n");  
  55.             }  
  56.         }  
  57.         else  
  58.         {  
  59.             char str[256];  
  60.             sprintf(str, "SIOCGIFFLAGS ioctl %s", buf[if_len].ifr_name);  
  61.             perror(str);  
  62.         }  
  63.    
  64.    
  65.         /* IP地址 */  
  66.         if (!(ioctl(fd, SIOCGIFADDR, (char *) &buf[if_len])))  
  67.         {  
  68.             printf("IP地址:%s\n",  
  69.                     (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));  
  70.         }  
  71.         else  
  72.         {  
  73.             char str[256];  
  74.             sprintf(str, "SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);  
  75.             perror(str);  
  76.         }  
  77.    
  78.         /* 子网掩码 */  
  79.         if (!(ioctl(fd, SIOCGIFNETMASK, (char *) &buf[if_len])))  
  80.         {  
  81.             printf("子网掩码:%s\n",  
  82.                     (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));  
  83.         }  
  84.         else  
  85.         {  
  86.             char str[256];  
  87.             sprintf(str, "SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);  
  88.             perror(str);  
  89.         }  
  90.    
  91.         /* 广播地址 */  
  92.         if (!(ioctl(fd, SIOCGIFBRDADDR, (char *) &buf[if_len])))  
  93.         {  
  94.             printf("广播地址:%s\n",  
  95.                     (char*)inet_ntoa(((struct sockaddr_in*) (&buf[if_len].ifr_addr))->sin_addr));  
  96.         }  
  97.         else  
  98.         {  
  99.             char str[256];  
  100.             sprintf(str, "SIOCGIFADDR ioctl %s", buf[if_len].ifr_name);  
  101.             perror(str);  
  102.         }  
  103.    
  104.         /*MAC地址 */  
  105.         if (!(ioctl(fd, SIOCGIFHWADDR, (char *) &buf[if_len])))  
  106.         {  
  107.             printf("MAC地址:%02x:%02x:%02x:%02x:%02x:%02x\n\n",  
  108.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[0],  
  109.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[1],  
  110.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[2],  
  111.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[3],  
  112.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[4],  
  113.                     (unsigned char) buf[if_len].ifr_hwaddr.sa_data[5]);  
  114.         }  
  115.         else  
  116.         {  
  117.             char str[256];  
  118.             sprintf(str, "SIOCGIFHWADDR ioctl %s", buf[if_len].ifr_name);  
  119.             perror(str);  
  120.         }  
  121.     }//–while end  
  122.    
  123.     //关闭socket  
  124.     close(fd);  
  125.     return 0;  
  126. }  

 

 

 

在我的系统上,程序输出:

接口数量: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地址。



参考资料:《Unix网络编程》第16章 ioctl操作

时间: 2024-10-01 00:14:53

获得Unix/Linux系统中的IP、MAC地址等信息的相关文章

linux系统中修改IP和DNS的命令

  Linux下修改IP.DNS.路由命令行设置 ubuntu 版本命令行设置IP cat /etc/network/interfaces # This file describes the network interfaces available on your system # and how to activate them. For more information, see interfaces(5). # The loopback network interface auto lo

linux 系统获取网络ip, mask, gateway, dns信息小程序

net_util.c          #define WIRED_DEV                   "eth0"     #define WIRELESS_DEV                "ra0"               #define PPPOE_DEV                   "ppp0" #define DEBUG_PRT(fmt, arg...)      printf(fmt,##arg) /** *

教你如何在windows与linux系统中设置MySQL数据库名、表名大小写敏感_Mysql

在 MySQL 中,数据库和表其实就是数据目录下的目录和文. 因而,操作系统的敏感性决定数据库和表命名的大小写敏感.这就意味着数据库和表名在 Windows 中是大小写不敏感的,而在大多数类型的 Unix/Linux 系统中是大小写敏感的. MySQL大小写敏感可以通过配置文件的lower_case_table_names参数来控制. WINDOWS: 编辑MySQL安装目录下的my.ini 文件,在[mysqld]节下 添加 lower_case_table_names=0 (备注:为0时大小

深入解析Linux系统中的SELinux访问控制功能

  SELinux(Security-Enhanced Linux) 是美国国家安全局(NSA)对于强制访问控制的实现,是 Linux历史上最杰出的新安全子系统.NSA是在Linux社区的帮助下开发了一种访问控制体系,在这种访问控制体系的限制下,进程只能访问那些在他的任务中所需要文件.SELinux 默认安装在 Fedora 和 Red Hat Enterprise Linux 上,也可以作为其他发行版上容易安装的包得到. SELinux 是 2.6 版本的 Linux 内核中提供的强制访问控制

在 Linux 系统中安装Load Generator ,并在windows 调用

原文地址:http://www.blogjava.net/qileilove/archive/2012/03/14/371861.html 由于公司需要测试系统的最大用户承受能力,所以需要学习使用loadrunner.在安装的时候碰到了不少问题,所以写下此文章总结遇到的问题以及解决方案,希望能帮到大家.也希望大家转载注明出处. Winsows 的Loadrunner 安装就不多讲了,这个太容易了. 以下是Linux 中安装 Load Generator 说明: Linux 系统版本:CentOS

在 Mac OS X 系统中创建用于 Mac 的 Ubuntu USB 启动盘

在 Mac OS X 系统中创建用于 Mac 的 Ubuntu USB 启动盘 上个月,在戴尔的服务中心丢失我的笔记本后,我买了一台 Macbook Air 笔记本.买回来后我首先做的一些事就是给机器装上双系统,使 Ubuntu Linux 和 Mac OS X 都可用.随后的文章我会介绍如何在 Macbook 上安装 Linux ,刚开始我们需要学习 如何在 Mac OS X 系统中创建用于 Mac 的 Ubuntu USB 启动盘. 在 Ubuntu 系统或 Windows 系统中创建可启动

linux系统中root用户不能登录怎么办?

问题描述 linux系统中root用户不能登录怎么办? 自从我更改了Linux系统的IP地址,root用户就不能登录了,root用户登录提示"鉴定故障".普通用户可以登录,能用su命令登录root用户,但是不能直接登录. 解决方案 如果没有给ROOT设置密码,就用普通用户登录,没有普通用户这是不可能的登录进入终端,直接输入 sudo passwd root输入普通用户密码输入你要设置的root密码重复root密码这样用root就行了 解决方案二: 终端输入:sudo gedit /us

MySQL在Linux系统中隐藏命令行中的密码的方法

  这篇文章主要介绍了MySQL在Linux系统中隐藏命令行中的密码的方法,作者利用简单的C程序实现,需要的朋友可以参考下 在命令行中输入命令并不是一个好主意,会造成安全问题.但是如果你决定去写一个应用,而这个应用需要在命令行中使用密码或者其他敏感信息.那么,你能通过以下方法禁止系统的其他用户轻易的看到这些敏感数据 呢?,类似MySQL在ps命令下隐藏密码. 假设我这里系统里两个用户,一个是root ,一个是dabu .测试系统为centos 6.5在按照下面的步骤做: ? 1 2 3 4 [r

Linux系统中的usr目录有哪些

Linux 系统文件结构中,有这么一个神奇的目录那就是usr.之前一直都没有去关注过它,反正程序已经安装在里面,我也不知道有什么用就放在哪里了,后来 fedora 要简化整个文件系统体系,这才看到才想到它的存在,usr 到底是什么的缩写呢,它又是怎么来的呢?这里小编要给你们讲述的是Linux系统中的usr目录,下面也列举了usr目录下一些重要子目录所包含文件的功能,有需要的网友可以参考学习下. 1.usr 是 unix system resources 的缩写; 2.usr 是 user 的缩写