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

时间:2021-11-15 01:17:55

获得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函数的具体运用,首先要了解这两个结构:

  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. struct ifreq
  12. {
  13. # define IFHWADDRLEN    6
  14. # define IFNAMSIZ   IF_NAMESIZE
  15. union
  16. {
  17. char ifrn_name[IFNAMSIZ];   /* Interface name, e.g. "en0".  */
  18. } ifr_ifrn;
  19. union
  20. {
  21. struct sockaddr ifru_addr;
  22. struct sockaddr ifru_dstaddr;
  23. struct sockaddr ifru_broadaddr;
  24. struct sockaddr ifru_netmask;
  25. struct sockaddr ifru_hwaddr;
  26. short int ifru_flags;
  27. int ifru_ivalue;
  28. int ifru_mtu;
  29. struct ifmap ifru_map;
  30. char ifru_slave[IFNAMSIZ];  /* Just fits the size */
  31. char ifru_newname[IFNAMSIZ];
  32. __caddr_t ifru_data;
  33. } ifr_ifru;
  34. };

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

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

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

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

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

  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. struct ifreq
  12. {
  13. # define IFHWADDRLEN    6
  14. # define IFNAMSIZ   IF_NAMESIZE
  15. union
  16. {
  17. char ifrn_name[IFNAMSIZ];   /* Interface name, e.g. "en0".  */
  18. } ifr_ifrn;
  19. union
  20. {
  21. struct sockaddr ifru_addr;
  22. struct sockaddr ifru_dstaddr;
  23. struct sockaddr ifru_broadaddr;
  24. struct sockaddr ifru_netmask;
  25. struct sockaddr ifru_hwaddr;
  26. short int ifru_flags;
  27. int ifru_ivalue;
  28. int ifru_mtu;
  29. struct ifmap ifru_map;
  30. char ifru_slave[IFNAMSIZ];  /* Just fits the size */
  31. char ifru_newname[IFNAMSIZ];
  32. __caddr_t ifru_data;
  33. } ifr_ifru;
  34. };

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

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

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

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

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

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

接口数量: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操作

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

  1. Unix&sol;Linux系统中僵尸进程是如何产生的?有什么危害?如何避免?

    如题 Unix/Linux系统中僵尸进程是如何产生的?有什么危害?如何避免? 一个进程在调用exit命令结束自己的生命的时候,其实他并没有真正的被销毁,而是留下一个称为僵尸进程(Zombie)的数据结 ...

  2. 在Linux系统中修改IP地址

    在Linux系统中,通过编辑网络配置文件,设置系统IP地址,当然要在root权限下执行,具体步骤如下: 1.切换路径到/etc/sysconfig/network-scripts [root@Comp ...

  3. java工具类&comma;在Windows&comma;Linux系统获取电脑的MAC地址、本地IP、电脑名

    package com.cloudssaas.util; import java.io.BufferedReader; import java.io.IOException; import java. ...

  4. 在windows系统和linux系统中查询IP地址命令的不同

    在linux和windows系统上查询IP地址的命令是不一样的.         在linux中的命令行模式下,输入ifconfig即可查询到IP.而在windows系统下要查询IP地址需要先打开do ...

  5. linux 系统获取网络ip&comma; mask&comma; gateway&comma; dns信息小程序

    net_util.c #define WIRED_DEV                   "eth0"     #define WIRELESS_DEV             ...

  6. 用户管理 之 Linux 系统中的超级权限的控制

    在Linux操作系统中,root的权限是最高的,也被称为超级权限的拥有者.普通用户无法执行的操作,root用户都能完成,所以也被称之为超级管理用户. 在系统中,每个文件.目录和进程,都归属于某一个用户 ...

  7. 在 Linux 系统中安装Load Generator ,并在windows 调用方法

    在 Linux 系统中安装Load Generator ,并在windows 调用 由于公司需要测试系统的最大用户承受能力,所以需要学习使用loadrunner.在安装的时候碰到了不少问题,所以写下此 ...

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

    原文地址:http://www.blogjava.net/qileilove/archive/2012/03/14/371861.html 由于公司需要测试系统的最大用户承受能力,所以需要学习使用lo ...

  9. Linux系统中&OpenCurlyDoubleQuote;动态库”和&OpenCurlyDoubleQuote;静态库”那点事儿 &sol;etc&sol;ld&period;so&period;conf 动态库的后缀为&ast;&period;so 静态库的后缀为 libxxx&period;a ldconfig 目录名

    Linux系统中“动态库”和“静态库”那点事儿 /etc/ld.so.conf  动态库的后缀为*.so  静态库的后缀为 libxxx.a   ldconfig   目录名 转载自:http://b ...

随机推荐

  1. winrt组件库(包括翻书组件)

    http://www.mindscapehq.com/products/metroelements/controls/book-control-for-winrt 点击“down free trial ...

  2. 搭建高性能计算环境(六)、应用软件的安装之lammps

    1,上传需要的软件包lammps-stable.tar.gz. 2,解压缩并进入安装目录 tar xvf lammps-stable.tar.gz cd lammps-30Oct14 3,如果需要re ...

  3. 基于Emgu CV 的手势识别实现PPT的控制放映

    Emgu CV 简介         众所周知,Emgu CV是.NET平台下对OpenCV图像处理库的封装,也就是.NET版的OpenCV.开发者可以很方便的通过C#,VB等语言调用OpenCV函数 ...

  4. 案例:利用累加器计算前N个学生的总成绩和平均成绩

    /* *录入N个学生的成绩,并求出这些学生的总成绩和平均成绩! * */ import java.util.Scanner; public class SumTest{ public static v ...

  5. hdu1507 Uncle Tom&&num;39&semi;s Inherited Land&ast; 二分匹配

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1507 将i+j为奇数的构成x集合中 将i+j为偶数的构成y集合中 然后就是构建二部图 关键就是构图 然 ...

  6. 分享如何使用PHP将URL地址参数进行加密传输提高网站安全性

    大家在使用PHP进行GET或POST提交数据时,经常会在URL带着参数进行传递,比如www.mdaima.com/get.php?id=1&page=5,这里就将id编号和page页码进行了参 ...

  7. Protobuf 从入门到实战

    简介 从第一次接触Protobuf到实际使用已经有半年多,刚开始可能被它的名字所唬住,其实就它是一种轻便高效的数据格式,平台无关.语言无关.可扩展,可用于通讯协议和数据存储等领域. 优点 平台无关,语 ...

  8. 基于gtid的复制

    Ⅰ.GTID的介绍 global transaction id identifier 全局事务id gtid = server_uuid + transaction_id server_uuid是全局 ...

  9. js-权威指南学习笔记18

    1.除mouseenter和mouseleave外的所有鼠标事件都能冒泡. 2.传递给鼠标事件处理程序的事件对象有clientX和clientY属性,它们制订了鼠标指针相对于包含窗口的坐标. 3.一个 ...

  10. 解题:NOI 2016 优秀的拆分

    题面 其实题目不算很难,但是我调试的时候被玄学了,for循环里不写空格会RE,写了才能过.神**调了一个多小时是这么个不知道是什么的玩意(真事,可以问i207M=.=),心态爆炸 发现我们只要找AA或 ...