在使用ioctl / SIOCGIFADDR / SIOCGIFCONF确定如何在Mac OS X上获取以太网接口信息时遇到问题?

您是否在使用ioctl / SIOCGIFADDR / SIOCGIFCONF了解如何在Mac OS X上获取接口信息时遇到问题? 我很难在Linux上获得能够在Mac OS X上运行的代码。     
已邀请:
复制粘贴到
main.c
gcc main.c && ./a.out
应该有效(列出所有网络接口,它们的ipv4 / 6地址,网络掩码和MAC地址,如果相关): 适用于Mac OSX和iOS 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 : %sn", 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 : %sn",
             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 : %sn",
               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:%02xn", 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 %sn", errno, strerror(errno));
    return -1;
  }
}
    
获取MAC地址的机制在BSD派生的操作系统上与在Linux上完全不同。这包括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地址感兴趣,则传递另一个值。     
这个帖子在某种程度上与我的问 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上的手册页有一个很好的例子,说明如何使用getifaddrs并使用它作为参考帮助我找出适用于Mac和Linux的代码。我不希望任何其他人在如此简单的事情上浪费时间,所以我在这里张贴并回答自己。希望我的帖子可以帮助你......     

要回复问题请先登录注册