如何从c中的IP获取以太网适配器名称

时间:2015-04-24 11:16:54

标签: c++ c linux network-interface

有两个以太网适配器,所以我有两个不同的IP地址。现在我蚂蚁用相应的ip找到适配器的名称。就像,我有ip 192.168.10.1的intel卡。如何在没有任何第三方安装的情况下使用C或C ++在centos(linux)中检索此适配器名称?

我需要找到制造商名称(不是eth0等)。此制造商列表位于“/usr/share/hwdata/pci.ids”中,但我无法使用IP地址映射该名称。我可以使用'lscpu |获取适配器名称列表grep“以太网”'。但问题再次出现在用IP地址映射名称。

3 个答案:

答案 0 :(得分:3)

标准libc中有getifaddrs个函数。我从手册页修改了一个例子。

您无法从内核获取名称,但它在/sys文件系统中提供PCI ID。您可以使用libpci将这些数字解析为文件名。当前代码不支持USB设备和子设备号。

#define _GNU_SOURCE     /* To get defns of NI_MAXSERV and NI_MAXHOST */
#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>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <limits.h>

#include <pci/pci.h>

/* PCI IDs are contained in /sys filesystem. */
unsigned long read_sysfs_uint(const char* ifa_name, const char* info) {
    char path[PATH_MAX];
    char buf[12];
    int fd;

    snprintf(path, PATH_MAX, "/sys/class/net/%s/device/%s", ifa_name, info);

    fd = open(path, O_RDONLY);
    if(fd == -1)
        return 0;

    if(read(fd, buf, 12) == -1) {
        close(fd);
        return 0;
    }

    close(fd);
    return strtoul(buf, NULL, 16);
}

/* Try to get PCI IDs and get PCI device name for it.
   XXX: doesn't check for subsystem's numbers */
void print_pci_ids(const char* ifa_name) {
    int vendor = (int) read_sysfs_uint(ifa_name, "vendor");
    int device = (int) read_sysfs_uint(ifa_name, "device");
    int subsystem_vendor = (int) read_sysfs_uint(ifa_name, "subsystem_vendor");
    int subsystem_device = (int) read_sysfs_uint(ifa_name, "subsystem_device");

    struct pci_access *pacc = pci_alloc();
    char namebuf[256];

    printf("PCI IDs: %x %x %x %x\n", vendor, device, subsystem_device, subsystem_vendor);

    pci_init(pacc);

    if(pci_lookup_name(pacc, namebuf, 256, 
                    PCI_LOOKUP_VENDOR | PCI_LOOKUP_DEVICE,
                    vendor, device)) {
        printf("PCI Name: %s\n", namebuf);
    }

    pci_cleanup(pacc);
}

int main(int argc, char *argv[])
{
    struct ifaddrs *ifaddr, *ifa;
    struct in_addr* ifa_inaddr;
    struct in_addr addr;
    int family, s, n;

    if(argc != 2) {
        fprintf(stderr, "Usage: getifaddr <IP>\n");
        return EXIT_FAILURE;
    }

    if (inet_aton(argv[1], &addr) == 0) {
        perror("inet_aton");
        return EXIT_FAILURE;
    }

    if (getifaddrs(&ifaddr) == -1) {
        perror("getifaddrs");
        return 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;

        /* We seek only for IPv4 addresses */
        if(ifa->ifa_addr->sa_family != AF_INET)
            continue;

        ifa_inaddr = &(((struct sockaddr_in*) ifa->ifa_addr)->sin_addr);
        if(memcmp(ifa_inaddr, &addr, sizeof(struct in_addr)) == 0) {
            printf("Interface: %s\n", ifa->ifa_name);
            print_pci_ids(ifa->ifa_name);
        }
    }

    freeifaddrs(ifaddr);
    return EXIT_SUCCESS;
}

使用libpci编译它(您需要安装相应的devel包):

$ gcc getifname.c -lpci -o ./getifname

以下是其用法示例:

$ ./getifname 
Usage: getifaddr <IP>
$ ./getifname  dlks
inet_aton: Success
$ ./getifname 127.0.0.1
Interface: lo
PCI IDs: 0 0 0 0
PCI Name: Device 0000:0000
$ ./getifname 192.168.13.144
Interface: wlan0
PCI IDs: 8086 88e 4060 8086
PCI Name: Intel Corporation Centrino Advanced-N 6235

答案 1 :(得分:0)

我假设适配器名称是指eth0 / eth1 / etc。而不是制造商/型号。如果是这样,一个可能的解决方案(有点复杂但它的工作原理)将执行ifconfig sys调用并将其传递给文本文件。从那里你可以搜索文本文件以查找IP地址,然后从那里开始输出是常量,你可以使用IP的起始位置作为获取适配器名称的基础。

答案 2 :(得分:0)

这实际上有些棘手,因为linux没有像Windows一样的通用驱动程序堆栈API - 基本上归结为3个选项:

  1. 读取内核导出的特殊文件:https://stackoverflow.com/a/5611176/351861
  2. 致电lspci并解析其输出:http://prefetch.net/articles/linuxpci.html
  3. 复制lspci的功能并实际编写自己的应用程序,因为您可以看到您将需要几个内核数据结构,如pcimap_entry和诸如此类的东西,但它应该是直截了当的,因为您可以直接虹吸你们内核大师的知识:https://github.com/gittup/pciutils/blob/gittup/ls-kernel.c