Как получить MAC-адрес вашего компьютера с помощью программы на языке Си?

StackOverflow https://stackoverflow.com/questions/1779715

  •  21-09-2019
  •  | 
  •  

Вопрос

Я работаю над Ubuntu.Как я могу получить MAC-адрес моего компьютера или интерфейс, скажем, eth0, используя программу C?

Это было полезно?

Решение

Вам нужно перебрать все доступные интерфейсы на вашем компьютере и использовать ioctl с SIOCGIFHWADDR установите флажок, чтобы получить mac-адрес.Mac-адрес будет получен в виде 6-октетного двоичного массива.Вы также хотите пропустить интерфейс обратной связи.

#include <sys/ioctl.h>
#include <net/if.h> 
#include <unistd.h>
#include <netinet/in.h>
#include <string.h>

int main()
{
    struct ifreq ifr;
    struct ifconf ifc;
    char buf[1024];
    int success = 0;

    int sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_IP);
    if (sock == -1) { /* handle error*/ };

    ifc.ifc_len = sizeof(buf);
    ifc.ifc_buf = buf;
    if (ioctl(sock, SIOCGIFCONF, &ifc) == -1) { /* handle error */ }

    struct ifreq* it = ifc.ifc_req;
    const struct ifreq* const end = it + (ifc.ifc_len / sizeof(struct ifreq));

    for (; it != end; ++it) {
        strcpy(ifr.ifr_name, it->ifr_name);
        if (ioctl(sock, SIOCGIFFLAGS, &ifr) == 0) {
            if (! (ifr.ifr_flags & IFF_LOOPBACK)) { // don't count loopback
                if (ioctl(sock, SIOCGIFHWADDR, &ifr) == 0) {
                    success = 1;
                    break;
                }
            }
        }
        else { /* handle error */ }
    }

    unsigned char mac_address[6];

    if (success) memcpy(mac_address, ifr.ifr_hwaddr.sa_data, 6);
}

Другие советы

Гораздо приятнее, чем все это безумие с сокетами или оболочками, просто использовать sysfs для этого:

файл /sys/class/net/eth0/address содержит ваш mac-адрес в виде простой строки, которую вы можете прочитать с помощью fopen()/fscanf()/fclose().Нет ничего проще, чем это.

И если вы хотите поддерживать другие сетевые интерфейсы, кроме eth0 (а вы, вероятно, хотите), то просто используйте opendir()/readdir()/closedir() на /sys/class/net/.

Вы хотите взглянуть на получите дополнительные файлы (3) страница руководства.На самой странице руководства есть пример на C, который вы можете использовать.Вы хотите получить адрес с типом AF_LINK.

#include <sys/socket.h>
#include <sys/ioctl.h>
#include <linux/if.h>
#include <netdb.h>
#include <stdio.h>
#include <string.h>

int main()
{
  struct ifreq s;
  int fd = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP);

  strcpy(s.ifr_name, "eth0");
  if (0 == ioctl(fd, SIOCGIFHWADDR, &s)) {
    int i;
    for (i = 0; i < 6; ++i)
      printf(" %02x", (unsigned char) s.ifr_addr.sa_data[i]);
    puts("\n");
    return 0;
  }
  return 1;
}

Используя getifaddr - ы вы можете получить MAC-адрес от семьи AF_PACKET.

Чтобы отобразить MAC-адрес для каждого интерфейса, вы можете поступить следующим образом:

#include <stdio.h>
#include <ifaddrs.h>
#include <netpacket/packet.h>

int main (int argc, const char * argv[])
{
    struct ifaddrs *ifaddr=NULL;
    struct ifaddrs *ifa = NULL;
    int i = 0;

    if (getifaddrs(&ifaddr) == -1)
    {
         perror("getifaddrs");
    }
    else
    {
         for ( ifa = ifaddr; ifa != NULL; ifa = ifa->ifa_next)
         {
             if ( (ifa->ifa_addr) && (ifa->ifa_addr->sa_family == AF_PACKET) )
             {
                  struct sockaddr_ll *s = (struct sockaddr_ll*)ifa->ifa_addr;
                  printf("%-8s ", ifa->ifa_name);
                  for (i=0; i <s->sll_halen; i++)
                  {
                      printf("%02x%c", (s->sll_addr[i]), (i+1!=s->sll_halen)?':':'\n');
                  }
             }
         }
         freeifaddrs(ifaddr);
    }
    return 0;
}

Идеон

Я только что написал один и протестировал его на gentoo в virtualbox.

// get_mac.c
#include <stdio.h>    //printf
#include <string.h>   //strncpy
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <net/if.h>   //ifreq
#include <unistd.h>   //close

int main()
{
    int fd;
    struct ifreq ifr;
    char *iface = "enp0s3";
    unsigned char *mac = NULL;

    memset(&ifr, 0, sizeof(ifr));

    fd = socket(AF_INET, SOCK_DGRAM, 0);

    ifr.ifr_addr.sa_family = AF_INET;
    strncpy(ifr.ifr_name , iface , IFNAMSIZ-1);

    if (0 == ioctl(fd, SIOCGIFHWADDR, &ifr)) {
        mac = (unsigned char *)ifr.ifr_hwaddr.sa_data;

        //display mac address
        printf("Mac : %.2X:%.2X:%.2X:%.2X:%.2X:%.2X\n" , mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
    }

    close(fd);

    return 0;
}

Предполагая, что код c ++ (c ++ 11) также в порядке и интерфейс известен.

#include <cstdint>
#include <fstream>
#include <streambuf>
#include <regex>

using namespace std;

uint64_t getIFMAC(const string &ifname) {
  ifstream iface("/sys/class/net/"+ifname+"/address");
  string str((istreambuf_iterator<char>(iface)), istreambuf_iterator<char>());
  if (str.length() > 0) {
    string hex = regex_replace(str, std::regex(":"), "");
    return stoull(hex, 0, 16);
  } else {
    return 0;
  }
} 
int main()
{
  string iface="eth0";
  printf("%s: mac=%016lX\n", iface.c_str(), getIFMAC(iface));
}
  1. В Linux используйте службу "Network Manager" через DBus.

  2. Существует также хороший ол программа-оболочка, которую можно вызвать и получить результат (используйте исполнительный функция в соответствии с C):

$ /sbin/ifconfig | grep HWaddr

Очень переносимым способом является синтаксический анализ выходных данных этой команды.

ifconfig | awk '$0 ~ /HWaddr/ { print $5 }'

При условии, что ifconfig может быть запущен от имени текущего пользователя (обычно может) и установлен awk (это часто бывает).Это даст вам mac-адрес компьютера.

Это строка Bash, которая выводит все доступные mac-адреса, за исключением обратного цикла:

for x in `ls /sys/class/net |grep -v lo`; do cat /sys/class/net/$x/address; done

Может быть выполнен из программы на языке Си.

Расширяя ответ , данный @user175104 ...

std::vector<std::string> GetAllFiles(const std::string& folder, bool recursive = false)
{
  // uses opendir, readdir, and struct dirent.
  // left as an exercise to the reader, as it isn't the point of this OP and answer.
}

bool ReadFileContents(const std::string& folder, const std::string& fname, std::string& contents)
{
  // uses ifstream to read entire contents
  // left as an exercise to the reader, as it isn't the point of this OP and answer.
}

std::vector<std::string> GetAllMacAddresses()
{
  std::vector<std::string> macs;
  std::string address;

  // from: https://stackoverflow.com/questions/9034575/c-c-linux-mac-address-of-all-interfaces
  //  ... just read /sys/class/net/eth0/address

  // NOTE: there may be more than one: /sys/class/net/*/address
  //  (1) so walk /sys/class/net/* to find the names to read the address of.

  std::vector<std::string> nets = GetAllFiles("/sys/class/net/", false);
  for (auto it = nets.begin(); it != nets.end(); ++it)
  {
    // we don't care about the local loopback interface
    if (0 == strcmp((*it).substr(-3).c_str(), "/lo"))
      continue;
    address.clear();
    if (ReadFileContents(*it, "address", address))
    {
      if (!address.empty())
      {
        macs.push_back(address);
      }
    }
  }
  return macs;
}
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top