使用 C 代码获取与 ifconfig 相同的信息

Posted

技术标签:

【中文标题】使用 C 代码获取与 ifconfig 相同的信息【英文标题】:using C code to get same info as ifconfig 【发布时间】:2011-06-24 11:43:39 【问题描述】:

在 Linux 中有没有一种方法可以使用 C 代码来获得与“ifconfig eth0”返回的信息相同的信息?我对 IP 地址、链接状态和 MAC 地址等内容感兴趣。

这是 ifconfig 的示例输出:

eth0      Link encap:Ethernet  HWaddr 00:0F:20:CF:8B:42
          inet addr:217.149.127.10  Bcast:217.149.127.63  Mask:255.255.255.192
          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
          RX packets:2472694671 errors:1 dropped:0 overruns:0 frame:0
          TX packets:44641779 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:1000
          RX bytes:1761467179 (1679.8 Mb)  TX bytes:2870928587 (2737.9 Mb)
          Interrupt:28 

【问题讨论】:

'ifconfig' 已多年未在 Linux 上维护。 请改用 'ip'。 serverfault.com/questions/633087/… 【参考方案1】:

解决此类问题的一种方法是strace。

它会为您提供您传递给它的任何程序进行的所有系统调用的列表,以及它们的参数和返回值。如果您的程序只是转储一些信息并退出而不是长时间运行,那么只需对您看到的所有系统调用执行人工操作就可以非常简单,看起来它们可能会提供您正在寻找的信息。

当我跑步时

strace ifconfig

一些有趣的调用是:

open("/proc/net/dev", O_RDONLY)         = 6

随后是一堆 ioctl,证实了@payne 的回答:

ioctl(5, SIOCGIFFLAGS, ifr_name="eth0",    ifr_flags=IFF_UP|IFF_BROADCAST|IFF_RUNNING|IFF_MULTICAST) = 0
ioctl(5, SIOCGIFHWADDR, ifr_name="eth0", ifr_hwaddr=84:2b:2b:b7:9e:6d) = 0
ioctl(5, SIOCGIFMETRIC, ifr_name="eth0", ifr_metric=0) = 0
ioctl(5, SIOCGIFMTU, ifr_name="eth0", ifr_mtu=1500) = 0

【讨论】:

+1 表示您如何找到答案,以及如何将该技术应用于其他问题 聪明,当然它确实忽略了这样一个事实,即对于 Linux ifconfig 命令,source 是现成的。 这必须是我见过的用于模拟命令和过滤器功能的最佳工具之一。有时你很匆忙,需要知道什么样的系统调用序列可以让你得到你需要做的事情,而无需调用 popen() 来执行命令。我很幸运地找到了关于 Linux 编程的最佳技巧之一。非常感谢!【参考方案2】:

是的,ifconfig 本身是用 C 编写的。:) 请参阅:http://cvsweb.netbsd.org/bsdweb.cgi/src/sbin/ifconfig/ifconfig.c?rev=1.169&content-type=text/x-cvsweb-markup

man netdevice 以查看详细信息(在 Linux 上)。您使用ioctl() 系统调用。

【讨论】:

【参考方案3】:

有更简单的方法。复制自http://man7.org/linux/man-pages/man3/getifaddrs.3.html

   #include <arpa/inet.h>
   #include <sys/socket.h>
   #include <netdb.h>
   #include <ifaddrs.h>
   #include <stdio.h>
   #include <stdlib.h>
   #include <unistd.h>
   #include <linux/if_link.h>

   int main(int argc, char *argv[])
   
       struct ifaddrs *ifaddr, *ifa;
       int family, s, n;
       char host[NI_MAXHOST];

       if (getifaddrs(&ifaddr) == -1) 
           perror("getifaddrs");
           exit(EXIT_FAILURE);
       

       /* Walk through linked list, maintaining head pointer so we
          can free list later */

       for (ifa = ifaddr, n = 0; ifa != NULL; ifa = ifa->ifa_next, n++) 
           if (ifa->ifa_addr == NULL)
               continue;

           family = ifa->ifa_addr->sa_family;

           /* Display interface name and family (including symbolic
              form of the latter for the common families) */

           printf("%-8s %s (%d)\n",
                  ifa->ifa_name,
                  (family == AF_PACKET) ? "AF_PACKET" :
                  (family == AF_INET) ? "AF_INET" :
                  (family == AF_INET6) ? "AF_INET6" : "???",
                  family);

           /* For an AF_INET* interface address, display the address */

           if (family == AF_INET || family == AF_INET6) 
               s = getnameinfo(ifa->ifa_addr,
                       (family == AF_INET) ? sizeof(struct sockaddr_in) :
                                             sizeof(struct sockaddr_in6),
                       host, NI_MAXHOST,
                       NULL, 0, NI_NUMERICHOST);
               if (s != 0) 
                   printf("getnameinfo() failed: %s\n", gai_strerror(s));
                   exit(EXIT_FAILURE);
               

               printf("\t\taddress: <%s>\n", host);

            else if (family == AF_PACKET && ifa->ifa_data != NULL) 
               struct rtnl_link_stats *stats = (struct rtnl_link_stats *)ifa->ifa_data;

               printf("\t\ttx_packets = %10u; rx_packets = %10u\n"
                      "\t\ttx_bytes   = %10u; rx_bytes   = %10u\n",
                      stats->tx_packets, stats->rx_packets,
                      stats->tx_bytes, stats->rx_bytes);
           
       

       freeifaddrs(ifaddr);
       exit(EXIT_SUCCESS);
   

【讨论】:

【参考方案4】:

一种简单的方法是使用 popen 函数,请参阅: http://pubs.opengroup.org/onlinepubs/009696899/functions/popen.html

使用类似的东西:

FILE *fp;

char returnData[64];

fp = popen("/sbin/ifconfig eth0", "r");

while (fgets(returnData, 64, fp) != NULL)

    printf("%s", returnData);


pclose(fp);

【讨论】:

人们不会喜欢它,但它是一个很好的便携式答案,并且使用 ioctls 滚动你自己不适合胆小的人! 有趣的答案。如果您还将它传送到awk 以解析值,我会赞成。由于您已经使用 popen 分叉并调用了 shell,我建议使用 awk 作为解析数据的经典 Unix 选择,而不是 C 中的 strtokstrstr 等。【参考方案5】:

这是我在代码中获取 MAC 和 MTU 的方法:

void getMACAddress(std::string _iface,unsigned char MAC[6]) 
        int fd = socket(AF_INET, SOCK_DGRAM, 0);
        struct ifreq ifr;
        ifr.ifr_addr.sa_family = AF_INET;
        strncpy(ifr.ifr_name , _iface.c_str() , IFNAMSIZ-1);
        ioctl(fd, SIOCGIFHWADDR, &ifr);
        for(unsigned int i=0;i<6;i++)
            MAC[i] = ifr.ifr_hwaddr.sa_data[i];
        ioctl(fd, SIOCGIFMTU, &ifr);
        close(fd);
        printf("MTU: %d\n",ifr.ifr_mtu);
        printf("MAC:%.2x:%.2x:%.2x:%.2x:%.2x:%.2x\n",MAC[0],MAC[1],MAC[2],MAC[3],MAC[4],MAC[5]);
    

【讨论】:

我应该向unsigned char MAC[6] 传递什么?? @ppumkin: MAC[6] 是一个输出参数,他使用它而不是返回值。因此,在此处输入一个指向您自己的无符号字符数组的指针。像这样:unsigned char outMAC[6]; getMACAddress("eth0", outMac); 但是在只有绑定地址的情况下如何获取接口名称呢?【参考方案6】:
void parse_ioctl(const char *ifname)

    printf("%s\n", "scarf rosari...");
    int sock;
    struct ifreq ifr;
    struct sockaddr_in *ipaddr;
    char address[INET_ADDRSTRLEN];
    size_t ifnamelen;

    /* copy ifname to ifr object */
    ifnamelen = strlen(ifname);
    if (ifnamelen >= sizeof(ifr.ifr_name)) 
        printf("error :%s\n", ifr.ifr_name);
        return ;
    
    memcpy(ifr.ifr_name, ifname, ifnamelen);
    ifr.ifr_name[ifnamelen] = '\0';

    /* open socket */
    sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP);
    if (sock < 0) 
        printf("error :%s\n", "unable to open socket..");
        return;
    

    /* process mac */
    if (ioctl(sock, SIOCGIFHWADDR, &ifr) != -1) 
        printf("Mac address: %02x:%02x:%02x:%02x:%02x:%02x\n",
                (unsigned char)ifr.ifr_hwaddr.sa_data[0],
                (unsigned char)ifr.ifr_hwaddr.sa_data[1],
                (unsigned char)ifr.ifr_hwaddr.sa_data[2],
                (unsigned char)ifr.ifr_hwaddr.sa_data[3],
                (unsigned char)ifr.ifr_hwaddr.sa_data[4],
                (unsigned char)ifr.ifr_hwaddr.sa_data[5]);
    

    /* process mtu */
    if (ioctl(sock, SIOCGIFMTU, &ifr) != -1) 
        printf("MTU: %d\n", ifr.ifr_mtu);
    

    /* die if cannot get address */
    if (ioctl(sock, SIOCGIFADDR, &ifr) == -1) 
        close(sock);
        return;
    

    /* process ip */
    ipaddr = (struct sockaddr_in *)&ifr.ifr_addr;
    if (inet_ntop(AF_INET, &ipaddr->sin_addr, address, sizeof(address)) != NULL) 
        printf("Ip address: %s\n", address);
    

    /* try to get broadcast */
    if (ioctl(sock, SIOCGIFBRDADDR, &ifr) != -1) 
        ipaddr = (struct sockaddr_in *)&ifr.ifr_broadaddr;
        if (inet_ntop(AF_INET, &ipaddr->sin_addr, address, sizeof(address)) != NULL) 
            printf("Broadcast: %s\n", address);
        
    

    /* try to get mask */
    if (ioctl(sock, SIOCGIFNETMASK, &ifr) != -1) 
        ipaddr = (struct sockaddr_in *)&ifr.ifr_netmask;
        if (inet_ntop(AF_INET, &ipaddr->sin_addr, address, sizeof(address)) != NULL) 
            printf("Netmask: %s\n", address);
        
    

    close(sock);

用法:

parse_ioctl("eth0");

【讨论】:

以上是关于使用 C 代码获取与 ifconfig 相同的信息的主要内容,如果未能解决你的问题,请参考以下文章

liunx

有没有办法使用 C 程序在 .txt 文件上获取 linux 命令(如 ifconfig)的输出? [复制]

ifconfig命令

37ifconfig命令

ifconfig 命令

Linux常用命令40:ifconfig命令