用ioctl獲得本地ip地址

用ioctl獲得本地ip地址時要用到兩個結構體ifconf和ifreq,它們對於大多數人
來說都是比較陌生的,這裏給大家一種比較簡單的理解方法,當然只一種幫助
理解的方法,在描述中可能會有一些地方與真實定義有所出入,僅供參考.

首先先認識一下ifconf和ifreq:

//ifconf通常是用來保存所有接口信息的
//if.h
struct ifconf 
{
    int    ifc_len;            /* size of buffer    */
    union 
    {
        char *ifcu_buf;                        /* input from user->kernel*/
        struct ifreq *ifcu_req;        /* return from kernel->user*/
    } ifc_ifcu;
};
#define    ifc_buf    ifc_ifcu.ifcu_buf        /* buffer address    */
#define    ifc_req    ifc_ifcu.ifcu_req        /* array of structures    */
 
//ifreq用來保存某個接口的信息
//if.h
struct ifreq {
    char ifr_name[IFNAMSIZ];
    union {
        struct sockaddr ifru_addr;
        struct sockaddr ifru_dstaddr;
        struct sockaddr ifru_broadaddr;
        short ifru_flags;
        int ifru_metric;
        caddr_t ifru_data;
    } ifr_ifru;
};
#define ifr_addr ifr_ifru.ifru_addr
#define ifr_dstaddr ifr_ifru.ifru_dstaddr
#define ifr_broadaddr ifr_ifru.ifru_broadaddr

 

上邊這兩個結構看起來比較複雜,我們現在把它們簡單化一些:
比如說現在我們向實現獲得本地IP的功能。

我們的做法是:
1. 先通過ioctl獲得本地所有接口的信息,並保存在ifconf中
2. 再從ifconf中取出每一個ifreq中表示ip地址的信息

具體使用時我們可以認爲ifconf就有兩個成員:
ifc_len 和 ifc_buf,
如圖一所示:    

  

ifc_len:表示用來存放所有接口信息的緩衝區長度
ifc_buf:表示存放接口信息的緩衝區

所以我們需要在程序開始時對ifconf的ifc_led和ifc_buf進行初始化
接下來使用ioctl獲取所有接口信息,完成後ifc_len內存放實際獲得的藉口信息總長度
並且信息被存放在ifc_buf中。

如下圖示:(假設讀到兩個接口信息)

 

 

接下來我們只需要從一個一個的接口信息獲取ip地址信息即可。

下面有一個簡單的參考:

 

 

 

#include 
#include 
#include 
#include 
#include in.h>
#include <string.h>
#include if.h>
#include 
 
int main()
{
    int i=0;
    int sockfd;
  struct ifconf ifconf;
  unsigned char buf[512];
  struct ifreq *ifreq;
  
  //初始化ifconf
  ifconf.ifc_len = 512;
  ifconf.ifc_buf = buf;
  
    if((sockfd = socket(AF_INET, SOCK_DGRAM, 0))<0)
    {
        perror("socket");
        exit(1);
    }  
  ioctl(sockfd, SIOCGIFCONF, &ifconf);    //獲取所有接口信息
  
  //接下來一個一個的獲取IP地址
  ifreq = (struct ifreq*)buf;  
  for(i=(ifconf.ifc_len/sizeof(struct ifreq)); i>0; i--)
  {
//      if(ifreq->ifr_flags == AF_INET){            //for ipv4
          printf("name = [%s]/n", ifreq->ifr_name);
      printf("local addr = [%s]/n", 
                      inet_ntoa(((struct sockaddr_in*)&(ifreq->ifr_addr))->sin_addr));
      ifreq++;
//  }
  }
    return 0;
}
 
此方法僅供參考,也適用於獲取其他信息
發佈了20 篇原創文章 · 獲贊 4 · 訪問量 17萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章