2021年10月28日 星期四

UNIX網絡編程——ioctl 函數的用法詳解

 1.介紹

       Linux網絡程序與內核交互的方法是通過ioctl來實現的,ioctl與網絡協議棧進行交互,可得到網絡接口的信息,網卡設備的映射屬性和配置網絡接口。並且還能夠查看,修改,刪除ARP高速緩存的信息,所以,我們有必要了解一下ioctl函數的具體實現。

 

2.相關結構體與相關函數

 

 
  1. #include <sys/ioctl.h>

  2.  
  3. int ioctl(int d,int request,....);

參數:

d-文件描述符,這裏是對網絡套接字操作,顯然是套接字描述符。

request-請求碼

省略的部分對應不同的內存緩衝區,而具體的內存緩衝區是由請求碼request來決定的,下面看一下具體都有哪些相關緩衝區。
 

 

(1)網絡接口請求結構ifreq

 

 
  1. struct ifreq

  2. {

  3. #define IFHWADDRLEN 6 //6個字節的硬件地址,即MAC

  4. union{

  5. char ifrn_name[IFNAMESIZ];//網絡接口名稱

  6. }ifr_ifrn;

  7. union{

  8. struct sockaddr ifru_addr;//本地IP地址

  9. struct sockaddr ifru_dstaddr;//目標IP地址

  10. struct sockaddr ifru_broadaddr;//廣播IP地址

  11. struct sockaddr ifru_netmask;//本地子網掩碼地址

  12. struct sockaddr ifru_hwaddr;//本地MAC地址

  13. short ifru_flags;//網絡接口標記

  14. int ifru_ivalue;//不同的請求含義不同

  15. struct ifmap ifru_map;//網卡地址映射

  16. int ifru_mtu;//最大傳輸單元

  17. char ifru_slave[IFNAMSIZ];//佔位符

  18. char ifru_newname[IFNAMSIZE];//新名稱

  19. void __user* ifru_data;//用戶數據

  20. struct if_settings ifru_settings;//設備協議設置

  21. }ifr_ifru;

  22. }

  23. #define ifr_name ifr_ifrn.ifrn_name;//接口名稱

  24. #define ifr_hwaddr ifr_ifru.ifru_hwaddr;//MAC

  25. #define ifr_addr ifr_ifru.ifru_addr;//本地IP

  26. #define ifr_dstaddr ifr_ifru.dstaddr;//目標IP

  27. #define ifr_broadaddr ifr_ifru.broadaddr;//廣播IP

  28. #define ifr_netmask ifr_ifru.ifru_netmask;//子網掩碼

  29. #define ifr_flags ifr_ifru.ifru_flags;//標誌

  30. #define ifr_metric ifr_ifru.ifru_ivalue;//接口側度

  31. #define ifr_mtu ifr_ifru.ifru_mtu;//最大傳輸單元

  32. #define ifr_map ifr_ifru.ifru_map;//設備地址映射

  33. #define ifr_slave ifr_ifru.ifru_slave;//副設備

  34. #define ifr_data ifr_ifru.ifru_data;//接口使用

  35. #define ifr_ifrindex ifr_ifru.ifru_ivalue;//網絡接口序號

  36. #define ifr_bandwidth ifr_ifru.ifru_ivalue;//連接帶寬

  37. #define ifr_qlen ifr_ifru.ifru_ivalue;//傳輸單元長度

  38. #define ifr_newname ifr_ifru.ifru_newname;//新名稱

  39. #define ifr_seeting ifr_ifru.ifru_settings;//設備協議設置

       如果想獲得網絡接口的相關信息,就傳入ifreq結構體。

 

 

(2)網卡設備屬性ifmap

 

 
  1. struct ifmap{//網卡設備的映射屬性

  2. unsigned long mem_start;//開始地址

  3. unsigned long mem_end;//結束地址

  4. unsigned short base_addr;//基地址

  5. unsigned char irq;//中斷號

  6. unsigned char dma;//DMA

  7. unsigned char port;//端口

  8. }


(3)網絡配置接口ifconf

 

 

 
  1. struct ifconf{//網絡配置結構體是一種緩衝區

  2. int ifc_len;//緩衝區ifr_buf的大小

  3. union{

  4. char__user *ifcu_buf;//繪衝區指針

  5. struct ifreq__user* ifcu_req;//指向ifreq指針

  6. }ifc_ifcu;

  7. };

  8. #define ifc_buf ifc_ifcu.ifcu_buf;//緩衝區地址

  9. #define ifc_req ifc_ifcu.ifcu_req;//ifc_req地址


(4)ARP高速緩存操作arpreq

 

 

 
  1. /* ARP高速緩存操作,包含IP地址和硬件地址的映射表,操作ARP高速緩存的命令字 SIOCDARP,SIOCGARP,SIOCSARP分別是刪除ARP高速緩存的一條記錄,獲得ARP高速緩存的一條記錄和修改ARP高速緩存的一條記錄 */

  2. struct arpreq{

  3. struct sockaddr arp_pa;//協議地址

  4. struct sockaddr arp_ha;//硬件地址

  5. int arp_flags;//標記

  6. struct sockaddr arp_netmask;//協議地址的子網掩碼

  7. char arp_dev[16];//查詢網絡接口的名稱

  8. }

 

 

3. 請求碼request
 

類別

Request

說明

數據類型

SIOCATMARK

SIOCSPGRP

SIOCGPGRP

是否位於帶外標記

設置套接口的進程ID或進程組ID

獲取套接口的進程ID或進程組ID

int

int

int

 

 

 

 

FIONBIN

FIOASYNC

FIONREAD

FIOSETOWN

FIOGETOWN

 

設置/清除非阻塞I/O標誌

設置/清除信號驅動異步I/O標誌

獲取接收緩存區中的字節數

設置文件的進程ID或進程組ID

獲取文件的進程ID或進程組ID

int

int

int

int

int

 

 

 

 

 

 

 

 

 

 

 

 

 

 

SIOCGIFCONF

SIOCSIFADDR

SIOCGIFADDR

SIOCSIFFLAGS

SIOCGIFFLAGS

SIOCSIFDSTADDR

SIOCGIFDSTADDR

SIOCGIFBRDADDR

SIOCSIFBRDADDR

SIOCGIFNETMASK

SIOCSIFNETMASK

SIOCGIFMETRIC

SIOCSIFMETRIC

SIOCGIFMTU

SIOCxxx

獲取所有接口的清單

設置接口地址

獲取接口地址

設置接口標誌

獲取接口標誌

設置點到點地址

獲取點到點地址

獲取廣播地址

設置廣播地址

獲取子網掩碼

設置子網掩碼

獲取接口的測度

設置接口的測度

獲取接口MTU

(還有很多取決於系統的實現)

struct ifconf

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

struct ifreq

 

ARP

SIOCSARP

SIOCGARP

SIOCDARP

創建/修改ARP表項

獲取ARP表項

刪除ARP表項

struct arpreq

struct arpreq

struct arpreq

SIOCADDRT

SIOCDELRT

增加路徑

刪除路徑

struct rtentry

struct rtentry

I_xxx

 

 


 

4. 相關例子

(1)網絡接口信息
       選項獲取填充struct ifreq的ifr_name:

 

 
  1. #include <stdio.h>

  2. #include <stdlib.h>

  3. #include <string.h>

  4. #include <sys/types.h>

  5. #include <sys/socket.h>

  6. #include <netinet/in.h>

  7. #include <linux/if.h>

  8. #include <arpa/inet.h>

  9. #include <linux/sockios.h>

  10. /**

  11. ioctl函數是與內核交互的一種方法,使用ioctl函數與內核協議棧進行交互

  12. ioctl函數可操作I/O請求,文件請求與網絡接口請求

  13. 網絡接口請求的幾個結構體:

  14. struct ifreq{

  15. #define IFHWADDRLEN 6 //6個字節的硬件地址,即MAC

  16. union{

  17. char ifrn_name[IFNAMESIZ];//網絡接口名稱

  18. }ifr_ifrn;

  19. union{

  20. struct sockaddr ifru_addr;//本地IP地址

  21. struct sockaddr ifru_dstaddr;//目標IP地址

  22. struct sockaddr ifru_broadaddr;//廣播IP地址

  23. struct sockaddr ifru_netmask;//本地子網掩碼地址

  24. struct sockaddr ifru_hwaddr;//本地MAC地址

  25. short ifru_flags;//網絡接口標記

  26. int ifru_ivalue;//不同的請求含義不同

  27. struct ifmap ifru_map;//網卡地址映射

  28. int ifru_mtu;//最大傳輸單元

  29. char ifru_slave[IFNAMSIZ];//佔位符

  30. char ifru_newname[IFNAMSIZE];//新名稱

  31. void __user* ifru_data;//用戶數據

  32. struct if_settings ifru_settings;//設備協議設置

  33.  
  34. }ifr_ifru;

  35. }

  36. #define ifr_name ifr_ifrn.ifrn_name;//接口名稱

  37. #define ifr_hwaddr ifr_ifru.ifru_hwaddr;//MAC

  38. #define ifr_addr ifr_ifru.ifru_addr;//本地IP

  39. #define ifr_dstaddr ifr_ifru.dstaddr;//目標IP

  40. #define ifr_broadaddr ifr_ifru.broadaddr;//廣播IP

  41. #define ifr_netmask ifr_ifru.ifru_netmask;//子網掩碼

  42. #define ifr_flags ifr_ifru.ifru_flags;//標誌

  43. #define ifr_metric ifr_ifru.ifru_ivalue;//接口側度

  44. #define ifr_mtu ifr_ifru.ifru_mtu;//最大傳輸單元

  45. #define ifr_map ifr_ifru.ifru_map;//設備地址映射

  46. #define ifr_slave ifr_ifru.ifru_slave;//副設備

  47. #define ifr_data ifr_ifru.ifru_data;//接口使用

  48. #define ifr_ifrindex ifr_ifru.ifru_ivalue;//網絡接口序號

  49. #define ifr_bandwidth ifr_ifru.ifru_ivalue;//連接帶寬

  50. #define ifr_qlen ifr_ifru.ifru_ivalue;//傳輸單元長度

  51. #define ifr_newname ifr_ifru.ifru_newname;//新名稱

  52. #define ifr_seeting ifr_ifru.ifru_settings;//設備協議設置

  53.  
  54. struct ifmap{//網卡設備的映射屬性

  55. unsigned long mem_start;//開始地址

  56. unsigned long mem_end;//結束地址

  57. unsigned short base_addr;//基地址

  58. unsigned char irq;//中斷號

  59. unsigned char dma;//DMA

  60. unsigned char port;//端口

  61. }

  62.  
  63. struct ifconf{//網絡配置結構體是一種緩衝區

  64. int ifc_len;//緩衝區ifr_buf的大小

  65. union{

  66. char__user *ifcu_buf;//繪衝區指針

  67. struct ifreq__user* ifcu_req;//指向ifreq指針

  68. }ifc_ifcu;

  69.  
  70. };

  71. #define ifc_buf ifc_ifcu.ifcu_buf;//緩衝區地址

  72. #define ifc_req ifc_ifcu.ifcu_req;//ifc_req地址

  73.  
  74. (1)獲得配置選項SIOCGIFCONF獲得網絡接口的配置情況 需要填充struct ifreq中ifr_name變量

  75. (2)其它選項獲取填充struct ifreq的ifr_name

  76. **/

  77.  
  78. int main(int argc,char*argv[]){

  79. int s;

  80. int err;

  81. s=socket(AF_INET,SOCK_DGRAM,0);

  82. if(s<0){

  83. perror("socket error");

  84. return;

  85. }

  86.  
  87. //傳入網絡接口序號,獲得網絡接口的名稱

  88. struct ifreq ifr;

  89.  
  90. ifr.ifr_ifindex=2;//獲得第2個網絡接口的名稱

  91. err=ioctl(s,SIOCGIFNAME,&ifr);

  92.  
  93. if(err)

  94. perror("index error");

  95. else

  96. printf("the %dst interface is:%s\n",ifr.ifr_ifindex,ifr.ifr_name);

  97.  
  98. //傳入網絡接口名稱,獲得標誌

  99. memcpy(ifr.ifr_name,"eth0",5);

  100. err=ioctl(s,SIOCGIFFLAGS,&ifr);

  101. if(!err)

  102. printf("SIOCGIFFLAGS:%d\n",ifr.ifr_flags);

  103.  
  104. //獲得MTU和MAC

  105. err=ioctl(s,SIOCGIFMTU,&ifr);

  106. if(!err)

  107. printf("SIOCGIFMTU:%d\n",ifr.ifr_mtu);

  108.  
  109. //獲得MAC地址

  110. err=ioctl(s,SIOCGIFHWADDR,&ifr);

  111. if(!err){

  112. unsigned char* hw=ifr.ifr_hwaddr.sa_data;

  113. printf("SIOCGIFHWADDR:%02x:%02x:%02x:%02x:%02x:%02x\n",hw[0],hw[1],hw[2],hw[3],hw[4],hw[5]);

  114. }

  115.  
  116. //獲得網卡映射參數 命令字SIOCGIFMAP

  117. err=ioctl(s,SIOCGIFMAP,&ifr);

  118. if(!err)

  119. printf("SIOCGIFMAP,mem_start:%d,mem_end:%d,base_addr:%d,ifr_map:%d,dma:%d,port:%d\n",ifr.ifr_map.mem_start,ifr.ifr_map.mem_end,ifr.ifr_map.base_addr,ifr.ifr_map.irq,ifr.ifr_map.dma,ifr.ifr_map.port);

  120.  
  121.  
  122. //獲得網卡序號

  123. err=ioctl(s,SIOCGIFINDEX,&ifr);

  124. if(!err)

  125. printf("SIOCGIFINDEX:%d\n",ifr.ifr_ifindex);

  126.  
  127. //獲取發送隊列的長度

  128. err=ioctl(s,SIOCGIFTXQLEN,&ifr);

  129. if(!err)

  130. printf("SIOCGIFTXQLEN:%d\n",ifr.ifr_qlen);

  131.  
  132. //獲取網絡接口IP

  133.  
  134. struct sockaddr_in *sin=(struct sockaddr_in*)&ifr.ifr_addr;//保存的是二進制IP

  135. char ip[16];//字符數組,存放字符串

  136. memset(ip,0,16);

  137. err=ioctl(s,SIOCGIFADDR,&ifr);

  138. if(!err){

  139. inet_ntop(AF_INET,&sin->sin_addr.s_addr,ip,16);//轉換的字符串保存到ip數組中,第二個參數是要轉換的二進制IP指針,第三個參數是轉換完成存放IP的緩衝區,最後一個參數是緩衝區的長度

  140. printf("SIOCGIFADDR:%s\n",ip);

  141. }

  142.  
  143. //查詢目標IP地址

  144. err=ioctl(s,SIOCGIFDSTADDR,&ifr);

  145. if(!err){

  146. inet_ntop(AF_INET,&sin->sin_addr.s_addr,ip,16);

  147. printf("SIOCGIFDSTADDR:%s\n",ip);

  148. }

  149.  
  150. //查詢子網掩碼

  151. err=ioctl(s,SIOCGIFNETMASK,&ifr);

  152. if(!err){

  153. inet_ntop(AF_INET,&sin->sin_addr.s_addr,ip,16);

  154. printf("SIOCGIFNETMASK:%s\n",ip);

  155. }

  156.  
  157. //設置IP地址,設置網絡接口

  158. inet_pton(AF_INET,"222.27.253.108",&sin->sin_addr.s_addr);//將字符串IP轉換成二進制

  159. err=ioctl(s,SIOCSIFADDR,&ifr);//發送設置本機ip地址請求命令

  160. if(!err){

  161. printf("check IP-----");

  162. memset(&ifr,0,sizeof(ifr));

  163. memcpy(ifr.ifr_name,"eth0",5);

  164. ioctl(s,SIOCGIFADDR,&ifr);

  165. inet_ntop(AF_INET,&sin->sin_addr.s_addr,ip,16);

  166. printf("%s\n",ip);

  167. }

  168.  
  169. //得到接口的廣播地址

  170. memset(&ifr,0,sizeof(ifr));

  171. memcpy(ifr.ifr_name,"eth0",5);

  172. ioctl(s,SIOCGIFBRDADDR,&ifr);

  173. struct sockaddr_in *broadcast=(struct sockaddr_in*)&ifr.ifr_broadaddr;

  174. //轉換成字符串

  175. inet_ntop(AF_INET,&broadcast->sin_addr.s_addr,ip,16);//inet_ntop將二進制IP轉換成點分十進制的字符串

  176. printf("BROADCAST IP:%s\n",ip);

  177. close(s);

  178. }

運行結果:

 

 

(2)查看arp高速緩存信息

 
  1. #include <stdio.h>

  2. #include <sys/types.h>

  3. #include <sys/socket.h>

  4. #include <netinet/in.h>

  5. #include <arpa/inet.h>

  6. #include <net/if_arp.h>

  7. #include <string.h>

  8. #include <stdlib.h>

  9. #include <linux/sockios.h>

  10. /**

  11. ARP高速緩存操作,包含IP地址和硬件地址的映射表

  12. 操作ARP高速緩存的命令字 SIOCDARP,SIOCGARP,SIOCSARP分別是刪除ARP高速緩存的一條記錄,獲得ARP高速緩存的一條記錄和修改ARP高速緩存的一條記錄

  13. struct arpreq{

  14. struct sockaddr arp_pa;//協議地址

  15. struct sockaddr arp_ha;//硬件地址

  16. int arp_flags;//標記

  17. struct sockaddr arp_netmask;//協議地址的子網掩碼

  18. char arp_dev[16];//查詢網絡接口的名稱

  19. }

  20.  
  21. **/

  22.  
  23. //根據IP地址查找硬件地址

  24. int main(int argc,char*argv[]){

  25. int s;

  26. int err;

  27. struct arpreq arpreq;

  28. struct sockaddr_in *addr=(struct sockaddr_in*)&arpreq.arp_pa;//IP地址

  29. s=socket(AF_INET,SOCK_DGRAM,0);

  30. if(s<0)

  31. perror("socket error");

  32.  
  33. addr->sin_family=AF_INET;

  34. addr->sin_addr.s_addr=inet_addr(argv[1]);//轉換成二進制IP

  35. if(addr->sin_addr.s_addr==INADDR_NONE)

  36. printf("IP地址格式錯誤\n");

  37.  
  38. strcpy(arpreq.arp_dev,"eth0");

  39. err=ioctl(s,SIOCGARP,&arpreq);

  40. if(err==-1){

  41. perror("arp");

  42. return -1;

  43. }

  44.  
  45. unsigned char* hw=(unsigned char*)&arpreq.arp_ha.sa_data;//硬件地址

  46. printf("%s\n",argv[1]);

  47. printf("%02x:%02x:%02x:%02x:%02x:%02x\n",hw[0],hw[1],hw[2],hw[3],hw[4],hw[5]);

  48. close(s);

  49. return 0;

  50. }

運行結果:

                                                         

 

       查看網關的MAC。在查看ARP高速緩存時要傳入IP地址與接口信息。而獲得接口信息要傳入接口名ifr_name,如eth0。

總結:

       本文主要介紹了獲得網絡接口請求信息,獲得網卡設備映射屬性、配置網絡接口、獲得ARP高速緩存等。其它ioctl函數還能對操作文件,操作I/O、操作路由等。最後對於網絡接口的操作與ARP高速緩存的操作分別給出了實例。

沒有留言:

張貼留言