概要

プログラムソース内で定義されているIPアドレス範囲(192.168.0.0~192.168.0.254)にpingを実行し、応答があればホスト名及びmacアドレスを取得します。
プログラムはコマンドラインで動作します。(IPv4専用)
表示内容は、応答があったIPアドレスとtimeはパケット送信から受信までの時間(ミリ秒)、TTLはIPパケットの生存時間、macアドレス、ホスト名、ベンダー名である。
ベンダー名の一覧は、IEEE-SAのホームページの下のほうのDownload a copyより本プログラムが自動的に取得し、macアドレスと一致するベンダーを表示します。

テスト環境

コンパイラ

Visual C++ 2008/2013 Express 32/64bit マルチバイト/UNICODE

実行環境

Windows XP Professional Service Pack 3 32bit(Virtual Box上の仮想マシーン)
Windows 7 Enterprise Service Pack 1 64bit

プログラムソースの概要

ping2.cpp

_tmain

カレントフォルダーにIEEE-SAサイトからダウンロードしたoui.txtファイルがあればこれを開きます。
ファイルがなければインターネットからダウンロードします。
get_mac_vendor関数によりファイルを読み取り、macアドレスとベンダー名の一覧(vendor_vec)を作成します。
IcmpCreateFile APIによりハンドルを作成します。
IPアドレスは、for文の中の配列ipで定義しています。(ビックエンディアンで設定)
IcmpSendEcho APIでpingを実行します。引数で送信するデータサイズやタイムアウトなどを指定できます。ここではマクロBUF_SIZEで32byteのデータサイズ及び2000ミリ秒をタイムアウト値としています。
APIが正常値を返した場合、SendARP APIによりIPアドレスからmacアドレスを取得します。
macアドレスよりmac2vendor_name関数によりベンダー名を取得します。
gethostbyaddr APIによりIPアドレスからホスト名を取得します。このAPI実行にはWINSOCKの初期化が必要であるため、WSAStartup APIをあらかじめ実行しておきます。
なぜかこのAPIはUNICODE版がないので、MultiByteToWideChar APIによりUNICODEに変換します。
for文のループから抜けたら、IcmpCloseHandle APIによりハンドルの開放、WSACleanup APIによりWINSOCKの開放を行います。

get_mac_vendor

ファイルよりmacアドレスとベンダー名の対応を一覧を作成します。
一覧はvendor_vecに保存されます。

mac2vendor_name

macアドレスよりベンダー名を取得します。該当するmacアドレスがない場合は文字列errorを返します。

ソースコード

ping2.cpp


//      指定した範囲のIPアドレスにPINGを実行し、レスポンスがあればマックアドレスとベンダー名およびホスト名を表示
//      2014/07/20

#include <stdio.h>
#include <winsock2.h>
#include <ws2tcpip.h>
#include <iphlpapi.h>
#include <icmpapi.h>
#include <wininet.h>
#include <locale.h>
#include <tchar.h>
#include <vector>

#pragma comment(lib,"iphlpapi.lib")
#pragma comment(lib,"Ws2_32.lib")       //      gethostbyaddr関数に必要
#pragma comment(lib,"wininet.lib")

#define BUF_SIZE 32


struct VENDOR_ID{
        unsigned char mac[3];   //      macアドレス 上位3byte
        TCHAR* name;                    //      ベンダー名
        void put(FILE* fp){
                _ftprintf(fp,_TEXT("%02x:%02x:%02x\t%s\n"),mac[0],mac[1],mac[2],name);
        }
        bool cmp(unsigned char* src_mac){
                if(mac[0]==src_mac[0] && mac[1]==src_mac[1] && mac[2]==src_mac[2])
                        return true;
                else
                        return false;
        }
};

using namespace std;

vector<VENDOR_ID> vendor_vec;     //      macアドレスとベンダー名の一覧

//      macアドレスよりベンダー名を取得する
TCHAR* mac2vendor_name(unsigned char* src_mac);

//      macアドレスとベンダー名の対応を一覧する
void vendor_fput(FILE* fp);

//      macアドレスとベンダー名の対応を一覧する
void vendor_fput(FILE* fp);

//      ファイルよりmacアドレスとベンダー名の対応を一覧を作成する
void get_mac_vendor(FILE* fp);


void _tmain(void){
        _tsetlocale(LC_ALL,_TEXT(""));  //      UNICODE文字を標準出力に正しく表示できるように設定


        FILE* fp;
        //      ファイルが開けない場合、インターネットから取得しファイルに保存後、再度ファイルを開く
        if(_tfopen_s(&fp,_TEXT("oui.txt"),_TEXT("r"))){
                _tprintf(_TEXT("ファイルが開けないのでインターネットから取得します\nhttp://standards.ieee.org/develop/regauth/oui/oui.txt\n"));
                if(_tfopen_s(&fp,_TEXT("oui.txt"),_TEXT("wb"))){
                        _tprintf(_TEXT("ファイルが開けません。\n"));
                        return;
                }
                HINTERNET hInet, hUrl;
                //インターネット(WinInet)開始
                hInet = InternetOpen(_TEXT("xyz-123"),
                        INTERNET_OPEN_TYPE_PRECONFIG,   
                        NULL, NULL, 0);
                if (hInet == NULL) {
                        _tprintf(_TEXT("インタネットから取得できませんでした。\n"));
                        return;
                }

                //HTTPセッションの開始, 指定のURLオープン
                hUrl = InternetOpenUrl(hInet, _TEXT("http://standards.ieee.org/develop/regauth/oui/oui.txt"), NULL, 0, 0, 0);
                if (hUrl == NULL) {
                        _tprintf(_TEXT("インタネットから取得できませんでした。\n"));
                        InternetCloseHandle(hInet);
                        return;
                }
                char szBuf[1024];
                DWORD dwRead = 0;
                //読み出す物がなくなるまで読み出す
                while (1) {
                        InternetReadFile(hUrl, szBuf, (DWORD)sizeof(szBuf), &dwRead); 

                        //読み出す物がなくなったのでループ脱出
                        if (dwRead == 0)
                                break;

                        fwrite(szBuf,1,dwRead,fp);
                }
                fclose(fp);

                if(_tfopen_s(&fp,_TEXT("oui.txt"),_TEXT("r"))){
                        _tprintf(_TEXT("ファイルが開けません。\n"));
                        return;
                }
                //インターネットハンドルの解放
                InternetCloseHandle(hUrl);
                InternetCloseHandle(hInet);

        }
        get_mac_vendor(fp);
        fclose(fp);

#ifdef _DEBUG
        if(_tfopen_s(&fp,_TEXT("vendor.txt"),_TEXT("w"))){
                _tprintf(_TEXT("ファイルが開けません。\n"));
                return;
        }
        vendor_fput(fp);
        fclose(fp);
        
#endif

        HANDLE hIcmp;
        IPAddr ipaddr;

        hIcmp=IcmpCreateFile();
        char cbRequest[BUF_SIZE];       //      送信するデータ
        memset(cbRequest,'a',BUF_SIZE); //      送信するデータを設定
        DWORD cbReply=sizeof(ICMP_ECHO_REPLY)+BUF_SIZE;
        char* pReply=new char[cbReply]; //      受信するヘッダー + 受信するデータ
        PICMP_ECHO_REPLY p=(PICMP_ECHO_REPLY)pReply;

        WSADATA wsaData;
        int iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);     //      WINSOCKを初期化

        unsigned char* ip=(unsigned char*)&ipaddr;
        _tprintf(_TEXT("IPアドレス      time  TTL マックアドレス    ベンダー名\tホスト名\n"));

        for(unsigned n=0;n<=244;n++){        //      IPアドレスの検索範囲を設定している
                ip[0]=192;      //      IPアドレス  ビックエンディアンで設定
                ip[1]=168;
                ip[2]=0;
                ip[3]=n;
                _tprintf(_TEXT("%3i.%3i.%3i.%3i\r"),ip[0],ip[1],ip[2],ip[3]);
                // pingを実行
                //                           IPアドレス 送信データ データーサイズ                     タイムアウトms
                DWORD ret=IcmpSendEcho( hIcmp ,ipaddr , cbRequest ,BUF_SIZE , NULL , pReply , cbReply , 2000);
                if(ret){        //      pingが成功した
                        unsigned char mac[6];
                        struct hostent* hostp;
                        ULONG len=sizeof(mac)/sizeof(unsigned char);
                        if(SendARP(ipaddr,NULL,mac,&len)==NO_ERROR){    //      IPアドレスからmacアドレスを取得
                                hostp=gethostbyaddr((const char*)ip,4,AF_INET); //      IPアドレスからホスト名を取得
                                TCHAR buf[64];
                                if(hostp)
#ifdef UNICODE
                                        //      UNICODEに変換
                                        MultiByteToWideChar(932,0,hostp->h_name,-1,buf,sizeof(buf)/sizeof(TCHAR));
#else
                                        _tcscpy_s(buf,sizeof(buf)/sizeof(TCHAR),hostp->h_name);
                                        
#endif
                                else
                                        buf[0]=_T('\0');
                                _tprintf(_TEXT("%3i.%3i.%3i.%3i "),ip[0],ip[1],ip[2],ip[3]);
                                _tprintf(_TEXT("%4d %4d %02x:%02x:%02x:%02x:%02x:%02x %s\t%s\n"),p->RoundTripTime,p->Options.Ttl,mac[0],mac[1],mac[2],mac[3],mac[4],mac[5],mac2vendor_name(mac),buf);
                        }

                }else{  //      pingに失敗した場合
                }
        }
        delete [] pReply;
        IcmpCloseHandle(hIcmp);
        WSACleanup();
}

//      macアドレスよりベンダー名を取得する

TCHAR* mac2vendor_name(unsigned char* src_mac){
        for(unsigned n=0;n<vendor_vec.size();n++){
                if(vendor_vec[n].cmp(src_mac)==true)
                        return vendor_vec[n].name;
        }
        return _TEXT("error");
}

//      macアドレスとベンダー名の対応を一覧する

void vendor_fput(FILE* fp){
        for(unsigned n=0;n<vendor_vec.size();n++){
                vendor_vec[n].put(fp);
        }
}


//      ファイルよりmacアドレスとベンダー名の対応を一覧を作成する

void get_mac_vendor(FILE* fp){
        TCHAR buf[256];
        int f=1;
        int len;
        TCHAR* top;
        TCHAR* p;

        VENDOR_ID id;
        while(_fgetts(buf,sizeof(buf)/sizeof(TCHAR),fp)){
                top=buf;
                while(*top){
                        if(*top!=_T(' '))
                                break;
                        ++top;
                }

                if(f==1){
                        len=(int)_tcslen(top);
                        if(10<=len){
                                if(_istxdigit(top[0]) && top[2]==_T('-')){
                                        f=2;
                                }
                        }

                }
                switch(f){
                        case 2:{
                                TCHAR* name=top;
                                id.mac[0]=(unsigned char)_tcstol(top,NULL,16);
                                id.mac[1]=(unsigned char)_tcstol(top+3,NULL,16);
                                id.mac[2]=(unsigned char)_tcstol(top+6,NULL,16);

                                while(*name){
                                        if(*name==_T(' '))
                                                break;
                                        ++name;
                                }
                                while(*name){
                                        if(*name!=_T(' '))
                                                break;
                                        ++name;
                                }
                                while(*name){
                                        if(*name==_T(')'))
                                                break;
                                        ++name;
                                }
                                while(*name){
                                        if(*name!=_T('\t'))
                                                break;
                                        ++name;
                                }
                                name+=3;
                                p=name;
                                while(*p){
                                        if(*p==_T('\n') || *p==_T('\r') || *p==_T('\t'))
                                                *p=_T('\0');
                                        ++p;
                                }
                                id.name=_tcsdup(name);
                                vendor_vec.push_back(id);
                                f=1;
                                break;
                        }
                }

        }
}

実行ファイルとソースファイルのダウンロード