检测当前网卡是否处于混杂模式 - Flier????s Sky - 博客园

    技术2022-05-12  10

    导读: 检测当前网卡是否处于混杂模式http://www.blogcn.com/user8/flier_lu/index.html?id=1245590&run=.0C9B086    今天比较巧,刚刚把重起网卡的文章贴上来,就有同事要我写个检测网卡混杂模式的小工具。本以为 WinPCap会提供此功能,但翻了一遍Packet32.c和其驱动代码后,发现居然没有提供接口。只好下班、跑步、吃饭,然后老老实实google找解决方法,呵呵      实际上方法很简单,打开一个网卡设备,查询其全局统计信息(IOCTL_NDIS_QUERY_GLOBAL_STATS),然后判断相应的标志位(NDIS_PACKET_TYPE_PROMISCUOUS)是否设置,即可判断此网卡是否进入混杂模式。      首先还是枚举网卡ID,我这儿偷懒直接用WinPCap提供的PacketGetAdapterNames函数,获取网卡列表。WinPCap 3.x返回的适配器设备名是类似DeviceNPF_{4BA2EE23-C4FE-488E-98CD-FE129206458A}这种格式的,因此要字符串操作去掉DeviceNPF_前缀,组装成/.{4BA2EE23-C4FE-488E-98CD-FE129206458A}这种形式的设备名,用CreateFile打开。   以下为引用:  string::size_type idx = name.find_last_of('{');

     if(idx == string::npos)   return;

     string strDev = "//./" + name.substr(idx);

     HANDLE hNic = CreateFile(strDev.c_str(), 0, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, INVALID_HANDLE_VALUE); 

         如果打开设备成功,则可以使用DeviceIoControl函数向其发送获取全局统计信息的请求,其实这儿叫网卡设备配置信息更合适 :)   以下为引用:  DWORD dwInBuf = OID_GEN_CURRENT_PACKET_FILTER, dwOutBuf = 0, dwByteReturn = 0;

     if(DeviceIoControl(hNic, IOCTL_NDIS_QUERY_GLOBAL_STATS,    &dwInBuf, sizeof(dwInBuf), &dwOutBuf, sizeof(dwOutBuf), &dwByteReturn, NULL)) {   cout << endl << " mode=";      static const char *NdisFilterTypes[] =    {     "Directed", "Multicast", "All Multicast", "Broadcast",        "Source Routing", "Promiscuous", "SMT", "All Local",       "Group", "All Functional", "Functional", "MAC Frame"   };      for(int i=0; i<(sizeof(NdisFilterTypes) / sizeof(NdisFilterTypes[0])); i++)   {     if((dwOutBuf & (1 << i)) != 0)     {       cout << NdisFilterTypes[i] << " ";     }   }         } 

         网卡当前模式有多个标志位,由DDK的ntddndis.h文件指定标志位意义   以下为引用:  //  // Ndis Packet Filter Bits (OID_GEN_CURRENT_PACKET_FILTER).  //  #define NDIS_PACKET_TYPE_DIRECTED      0x0001  #define NDIS_PACKET_TYPE_MULTICAST    0x0002  #define NDIS_PACKET_TYPE_ALL_MULTICAST  0x0004  #define NDIS_PACKET_TYPE_BROADCAST    0x0008  #define NDIS_PACKET_TYPE_SOURCE_ROUTING  0x0010  #define NDIS_PACKET_TYPE_PROMISCUOUS   0x0020  #define NDIS_PACKET_TYPE_SMT         0x0040  #define NDIS_PACKET_TYPE_ALL_LOCAL    0x0080  #define NDIS_PACKET_TYPE_MAC_FRAME    0x8000  #define NDIS_PACKET_TYPE_FUNCTIONAL    0x4000  #define NDIS_PACKET_TYPE_ALL_FUNCTIONAL  0x2000  #define NDIS_PACKET_TYPE_GROUP       0x1000              有兴趣仔细看看的朋友可以参考这个讨论 Detecting if Adapter is in Promiscuous mode。有一个小工具也完成了类似的功能 PromiscDetect,可惜不开源,不然我也不用折腾了,呵呵      此外PCAUSA提供的两个小工具和例子也很不错,对了解NDIS很有帮助

         PCAUSA NDIS Developer Tools     MACADDR II IOCTL_NDIS_QUERY_GLOBAL_STATS Sample Application

    本文转自 http://www.cnblogs.com/flier/archive/2004/07/08/22301.html

    最新回复(0)