私はUbuntuに取り組んでいます。Cプログラムを使用して、自分のマシンまたはeth0などのインターフェースのMACアドレスを取得するにはどうすればよいですか。
12 に答える
このソケットやシェルの狂気よりもはるかに優れているのは、単純に sysfs を使用することです。
このファイルには、 / //sys/class/net/eth0/address
で読み取ることができる単純な文字列として Mac アドレスが含まれています。それより簡単なことはありません。fopen()
fscanf()
fclose()
また、eth0 以外のネットワーク インターフェイスをサポートしたい場合 (そして、おそらくそうしたい場合) は、単純に // opendir()
onを使用しreaddir()
ます。closedir()
/sys/class/net/
マシンで使用可能なすべてのインターフェイスを反復処理し、ioctl
with SIOCGIFHWADDR
flag を使用して 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);
}
getifaddrs(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;
}
私はそれを書いて、virtualboxのgentooでテストしました。
// 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;
}
Linux では、DBus 経由で「Network Manager」のサービスを使用します。
呼び出して結果を取得できる古き良きシェル プログラムもあります(C ではexec関数を使用します)。
$ /sbin/ifconfig | grep HWaddr
非常に移植性の高い方法は、このコマンドの出力を解析することです。
ifconfig | awk '$0 ~ /HWaddr/ { print $5 }'
ifconfig を現在のユーザーとして実行でき (通常は実行可能)、awk がインストールされている (多くの場合は実行可能)。これにより、マシンのMACアドレスが得られます。