版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、一个简易网络嗅探器的实现 摘 要:本文介绍一个用C语言和网络数据包分析开发工具实现的简易网络Sniffer。 关 键 词:网络;数据包;Sniffer 1 引言 目前,已经有不少的Sniff工具软件,如Windows环境下,最富盛名的工具是Netxray和Sniffer pro,用它们在 Windows环境下抓包来分析,非常方便。在UNIX环境下如Sniffit,Snoop,Tcpdump,Dsniff 等都是比
2、较常见的。这里介绍一个用C语言和网络数据包和分析开发工具libpcap及winpcap实现的简易网络Sniffer。 2网络嗅探器程序实现在c环境下编程,源码如下:/* June 2nd,2002 * Project for graduation qualification By Bby Team 19 */#include <stdio.h>#include <conio.h> /必须加路径,必须把头文件packet32.h包含进去#include ".Includepacket32.h"#include ".Includentddndi
3、s.h"#define Max_Num_Adapter 10/ Prototypes原形/发包void PrintPackets(LPPACKET lpPacket);/设备列表char AdapterListMax_Num_Adapter1024;/ 主程序开始int main() /define a pointer to an ADAPTER structure设备指针 LPADAPTER lpAdapter = 0; /define a po
4、inter to a PACKET structure包指针 LPPACKET lpPacket; int i; DWORD dwErrorCode; DWORD dwVersion; DWORD
5、 dwWindowsMajorVersion; /Unicode strings (WinNT) WCHAR AdapterName8192; /网络适配器设备列表 WCHAR *temp,*temp1; /ASCII strings (Win9x) char
6、0; AdapterNamea8192; /网络适配器设备列表 char *tempa,*temp1a; int AdapterNum=0,Open; ULONG AdapterLength; char buffer256000; / 容纳来自
7、驱动器的数据的缓冲区struct bpf_stat stat; / 获得本机网卡名 AdapterLength=4096; printf("Packet.dll test application. Library version:%sn", PacketGetVersion(); printf("Adapters installed:n");
8、0; i=0; 下面这段代码是用来在不同版本下得到网络适配器名: Win9x 和WinNT中的网卡名称是分别用ASCII和UNICODE实现的,所以首先要得到本地操作系统的版本号.: dwVersion=GetVersion(); dwWindowsMajorVersion= (DWORD)(LOBYTE(LOWORD(dwVersion);这里首先用到的Packet.dll函数是PacketGetAd
9、apterNames(PTSTR pStr,PULONG BufferSize,通常它是与驱动程序通信并被调用的第一个函数,它将返回的用户本地系统中安装的网络适配器的名字放在缓冲区pStr中;BufferSize是缓冲区的长度: if (!(dwVersion >= 0x80000000 && dwWindowsMajorVersion >= 4) /是Windows NT / 找不到设备列表
10、 if(PacketGetAdapterNames(AdapterName,&AdapterLength)=FALSE) printf("Unable to retrieve the list of the adapters!n"); return -1; / 找到设备列表 temp=AdapterName;
11、0; temp1=AdapterName; while (*temp!='0')|(*(temp-1)!='0') if (*temp='0') memcpy(AdapterList,temp1,(temp-temp1)*2); temp1=te
12、mp 1; i ; temp ; / 显示适配器列表 AdapterNum=i; for (i=0;i<AdapterNum;i ) wprintf(L"n%d- %sn",i 1,AdapterLi
13、st); printf("n"); else /否则就是windows 9x,获取适配器名的方法同WinNT下 if(PacketGetAdapterNames(AdapterNamea,&AdapterLength)=FALSE) printf("Unable to retriev
14、e the list of the adapters!n"); return -1; tempa=AdapterNamea; temp1a=AdapterNamea;while (*tempa!='0')|(*(tempa-1)!='0') if (*tempa='0&
15、#39;) memcpy(AdapterList,temp1a,tempa-temp1a); temp1a=tempa 1; i ; tempa ; AdapterNum=i;
16、0;for (i=0;i<AdapterNum;i ) printf("n%d- %sn",i 1,AdapterList); printf("n"); 下面这段代码就是让用户选择监听的网络适配器号:/ 选择设备 do printf("Select the numb
17、er of the adapter to open : "); scanf("%d",&Open); if (Open>AdapterNum) printf("nThe number must be smaller than %d",AdapterNum); while (Open>AdapterNum); 然后,将所选择的设备打开,这里可以
18、设置为“混杂”模式打开,也可以是“直接”模式打开。代码如下: / 打开设备lpAdapter = PacketOpenAdapter(AdapterListOpen-1);/ 当设备无法打开时,出示错误信息: if (!lpAdapter | (lpAdapter->hFile = INVALID_HANDLE_VALUE) dwErrorCode=GetLastError(); printf("
19、;Unable to open the adapter, Error Code : %lxn",dwErrorCode); return -1; 将网卡设置为“混杂”模式,代码如下:这里用到函数PacketSetHwFilter(LPADAPTER AdapterObject,ULONG Filter),它在到来的包上设置了一个硬件过滤器,如操作成功,返回TRUE。AdapterObject是过滤器所在的网卡设备指针;过滤器的常量Filter定义
20、在头文件ntddndis.h 中,包括有: NDIS-PACKET-TYPE-PROMISCUOUS:设置混杂模式,每个到来的包都会被网卡接受;NDIS-PACKET-TYPE-DIRECTED:只有直接到主机网卡的包才会被接受;NDIS-PACKET-TYPE-BROADCAST:只接受广播包;NDIS-PACKET-TYPE-MULTICAST:只接受到主机所在的组的多播包;NDIS-PACKET-TYPE-ALL-MULTICAS:接受每个多播的包。 / set the network adapt
21、er in promiscuous mode / 如果混杂模式设置失败,提示错误: if(PacketSetHwFilter(lpAdapter,NDIS_PACKET_TYPE_PROMISCUOUS)=FALSE) printf("Warning: unable to set promiscuous mode!n"); 然后在driver中置512K的缓冲:
22、60; 这里用到函数PacketSetBuff(LPADAPTER AdapterObject,int dim),它被用于设置AdapterObject指向的网卡的驱动程序的缓冲区,成功则返回TRUE。Dim是新的缓冲区的大小,当它被设定时,旧缓冲区中的数据将被丢弃,其中存储的包也会失去。 需要注意的地方:驱动器缓冲区的大小设置是否恰当,将影响截包进程的性能,设置应能保证运行快且不会丢包。这里设置的是512000Byte。 / set a 512K buffer in the driver /
23、 当无法设置缓冲区时,提示错误: if(PacketSetBuff(lpAdapter,512000)=FALSE) printf("Unable to set the kernel buffer!n"); return -1; PacketSetReadTimeout(LPADAPTER AdapterObject,int timeout)函数的功能是,设置与AdapterObject指定网
24、卡绑定的读操作超时的值,timeout以毫秒为单位,0表示没有超时,当没有包到时,read就不返回。/ set a 1 second read timeout / 设置1秒的读取操作超时 if(PacketSetReadTimeout(lpAdapter,1000)=FALSE) printf("Warning: unable to set the read tiemout!n"); 接下来,定位设备,代码如下:
25、 这里用到函数PacketAllocatePacket(Void)将在内存中分配一个PACKET结构并返回一个指向它的指针,但这个结构的Buffer字段还没有设定,所以应再调用PacketInitPacket函数来对其进行初始化。 /allocate and initialize a packet structure that will be used to /receive the packets. / 当定位失败时,提示错误: if
26、(lpPacket = PacketAllocatePacket()=NULL) printf("nError: failed to allocate the LPPACKET structure."); return (-1); 然后,就可以初始化设备,开始接受网络包了:用函数PacketInitPacket(LPPACKET lpPacket,PVOID Buffer,UINT Leng
27、th)来初始化PACKET结构。lpPacket是要被初始化的指针;Buffer为指向用户分配的包含包的数据的缓冲区的指针;Length为缓冲区长度。需要注意的地方:PACKET结构关联的缓冲区存储由packet capture driver 截获的包,包的数量被缓冲区大小所限制,最大缓冲区的大小就是应用程序从驱动器中一次能读到的数据的多少。所以设置大的缓冲区可减少系统调用的次数,提高截获效率。这里设置的是256K。 PacketInitPacket(lpPacket,(char*)buffer,256000);
28、 接下来,是截包主循环: /main capture loop 这里又用到函数PacketReceivePacket(LPADAPTER AdapterObject,LPPACKET lpPacket,BOOLEAN Sync),它将接受(截获)一个包的集合。参数包括一个指向用来指定截包的网卡的ADAPTER结构指针、一个指向用来容纳包的PACKET结构、一个指出是同步还是异步方式操作的标记。当操作同步时,函数锁定程序;当操作异步时,函数不锁定程序,必须调用PacketWaitPacket过程
29、来检查是否正确完成。一般采用同步模式。 / 直到有键盘键入: while(!kbhit() / capture the packets 捕获包 / 捕获包失败时,提示错误: if(PacketReceivePacket(lpAdapter,lpPacket,TRUE)=FALSE) printf("Error:
30、 PacketReceivePacket failed"); return (-1); / 打印包中的数据,调用自定义函数PrintPackets() PrintPackets(lpPacket); 最后将得到的统计数据打印出来,代码如下: 这里用到函数PacketGetStats(LPADAPTER AdapterObject,struct bpf_star*s)可以得到两个驱动程序的内部变量的值:从调用PacketOpenAdapter开始,已经被指定网卡接收的包数目;以及已经被网卡接收但被内核丢弃的包数目。这两个值被驱动程序拷贝到应用提供的bpf_stat结构中。 /print the capture statistics / 得到统计值 / 当无法从内核读取状态时,提示错误: if(Pac
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 二零二五版国有土地临时用地合同3篇
- 二零二五版高级别别墅居住权购置与买卖合同3篇
- 医院2025年度物流配送服务合同2篇
- 二零二五年度交通枢纽“四害”灭治与旅客健康服务合同3篇
- 二零二五版数字艺术版权保护与侵权处理合同范本3篇
- 二零二五版宅基地使用权转让及农村土地流转收益分配合同2篇
- 二零二五年户外广告牌场地租赁及新媒体营销合同3篇
- 二零二五年投影机采购与灯光音响租赁服务合同3篇
- 二零二五版建筑工程项目招投标代理中介费合同3篇
- 二零二五版汽车零部件钣金加工及机加服务采购合同模板3篇
- 退学费和解协议书模板
- 2024至2030年中国对氯甲苯行业市场全景调研及发展趋势分析报告
- 智能教育辅助系统运营服务合同
- 心功能分级及护理
- DLT 572-2021 电力变压器运行规程
- 重庆育才中学2025届化学九上期末教学质量检测试题含解析
- 成都市2022级(2025届)高中毕业班摸底测试(零诊)数学试卷(含答案)
- 【云南省中药材出口现状、问题及对策11000字(论文)】
- 服装板房管理制度
- 河北省兴隆县盛嘉恒信矿业有限公司李杖子硅石矿矿山地质环境保护与治理恢复方案
- 第七章力与运动第八章压强第九章浮力综合检测题(一)-2023-2024学年沪科版物理八年级下学期
评论
0/150
提交评论