在Mac OS X上使用ioctl/SIOCGIFADDR/SIOCGIFCONF获取以太网接口信息遇到问题了吗?

5

您是否在使用ioctl/SIOCGIFADDR/SIOCGIFCONF获取Mac OS X上的接口信息时遇到问题?

今天我在Mac OS X上遇到了许多问题,无法使在Linux上运行良好的代码在Mac OS X上运行。

3个回答

12

将以下代码复制粘贴到main.c中,然后运行命令gcc main.c && ./a.out(列出所有网络接口、它们的IPv4/6地址、子网掩码和MAC地址(如果有关联)):

Mac OSXiOS iPad/iPhone上可正常工作:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <net/if.h>
#include <net/if_dl.h>
#include <ifaddrs.h>
#include <errno.h>

int main() {
  struct ifaddrs *if_addrs = NULL;
  struct ifaddrs *if_addr = NULL;
  void *tmp = NULL;
  char buf[INET6_ADDRSTRLEN];
  if (0 == getifaddrs(&if_addrs)) {    
    for (if_addr = if_addrs; if_addr != NULL; if_addr = if_addr->ifa_next) {

      printf("name : %s\n", if_addr->ifa_name);

      // Address
      if (if_addr->ifa_addr->sa_family == AF_INET) {
        tmp = &((struct sockaddr_in *)if_addr->ifa_addr)->sin_addr;
      } else {
        tmp = &((struct sockaddr_in6 *)if_addr->ifa_addr)->sin6_addr;
      }
      printf("addr : %s\n",
             inet_ntop(if_addr->ifa_addr->sa_family,
                       tmp,
                       buf,
                       sizeof(buf)));

      // Mask
      if (if_addr->ifa_netmask != NULL) {
        if (if_addr->ifa_netmask->sa_family == AF_INET) {
          tmp = &((struct sockaddr_in *)if_addr->ifa_netmask)->sin_addr;
        } else {
          tmp = &((struct sockaddr_in6 *)if_addr->ifa_netmask)->sin6_addr;
        }
        printf("mask : %s\n",
               inet_ntop(if_addr->ifa_netmask->sa_family,
                         tmp,
                         buf,
                         sizeof(buf)));
      }

      // MAC address
      if (if_addr->ifa_addr != NULL && if_addr->ifa_addr->sa_family == AF_LINK) {
        struct sockaddr_dl* sdl = (struct sockaddr_dl *)if_addr->ifa_addr;
        unsigned char mac[6];
        if (6 == sdl->sdl_alen) {
          memcpy(mac, LLADDR(sdl), sdl->sdl_alen);
          printf("mac  : %02x:%02x:%02x:%02x:%02x:%02x\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
        }
      }

      printf("\n");
    }
    freeifaddrs(if_addrs);
    if_addrs = NULL;
  } else {
    printf("getifaddrs() failed with errno =  %i %s\n", errno, strerror(errno));
    return -1;
  }
}

10

在基于BSD的操作系统和Linux上获取MAC地址的机制是完全不同的。这也包括OS X。

这是我在Linux和OS X上使用的代码,并且很可能也适用于BSD:

#if defined(HAVE_SIOCGIFHWADDR)
bool get_mac_address(char* mac_addr, const char* if_name = "eth0")
{
    struct ifreq ifinfo;
    strcpy(ifinfo.ifr_name, if_name);
    int sd = socket(AF_INET, SOCK_DGRAM, 0);
    int result = ioctl(sd, SIOCGIFHWADDR, &ifinfo);
    close(sd);

    if ((result == 0) && (ifinfo.ifr_hwaddr.sa_family == 1)) {
        memcpy(mac_addr, ifinfo.ifr_hwaddr.sa_data, IFHWADDRLEN);
        return true;
    }
    else {
        return false;
    }
}
#elif defined(HAVE_GETIFADDRS)
bool get_mac_address(char* mac_addr, const char* if_name = "en0")
{
    ifaddrs* iflist;
    bool found = false;
    if (getifaddrs(&iflist) == 0) {
        for (ifaddrs* cur = iflist; cur; cur = cur->ifa_next) {
            if ((cur->ifa_addr->sa_family == AF_LINK) &&
                    (strcmp(cur->ifa_name, if_name) == 0) &&
                    cur->ifa_addr) {
                sockaddr_dl* sdl = (sockaddr_dl*)cur->ifa_addr;
                memcpy(mac_addr, LLADDR(sdl), sdl->sdl_alen);
                found = true;
                break;
            }
        }

        freeifaddrs(iflist);
    }
    return found;
}
#else
#   error no definition for get_mac_address() on this platform!
#endif

如何获取适合平台的正确HAVE_*宏定义是由您决定的。我恰好使用autoconf来完成这个任务,但您可能有另外一种处理平台差异的方式。

请注意,这些函数的默认接口名称参数是在Linux和OS X设备上第一个以太网接口的默认值。对于其他操作系统,您可能需要覆盖此参数或传递另一个值,如果您想获取不同接口的MAC地址,则需要传递其他值。


2

这个帖子在一定程度上解决了我的问题:

http://discussions.apple.com/thread.jspa?messageID=10935410&tstart=0

这个帖子帮了我很多:

https://lists.isc.org/pipermail/dhcp-hackers/2007-September/000767.html

因为那个帖子最终提到应该使用getifaddrs()。Ubuntu 10.04的手册示例非常好,以它为参考帮助我找到了适用于Mac和Linux的代码。我不希望其他人浪费时间在这么简单的事情上,所以我在这里发布并回答自己。希望我的文章能帮助到您...


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接