发送网络实验指导----icmp协议的分析与实现

icmp协议  时间:2021-02-27  阅读:()

实验 ICMP协议的分析与实现

实验目的

分析ICMP报文理解ICMP协议在In ternet网中的具体应用及其实现原理深入了解 TCP/IP网络的容错控制学会运用网络套接字 Win sock开发网络通信程序。

实验内容

使用Visual Studio C++6.0和网络接口套接字Socket进行Windows环境下的网络编程运用原始嵌套字RAW_SOCKET从IP层开始构造整个ICMP报文通过ICMP协议所提供的回送请求(echo request)和回送应答(echoreply)这两种报文实现检测目的站的可达性与状态 。

1 . IP报头、 I C M P报文的基本描述

IP协议并不能保证绝对的可靠所以就设计了ICMP协议,进行差错报告.

ICMP消息使用IP头作为基本控制.

IP头的格式如下

0 1 2 3

01 23456789012345678901 2345678901

|Versio n| IHL|Type of Service|Total Len gth |

| Ide ntificati on |Flags| Fragme nt Offset |

|Time to Live|Protocol |Header Checksum|

| Source Address |

| Dest in ati on Address |

Version=4

IHL In ternet头长

Type of Service=0

Total Len gth IP 包的总长度

Identification, Flags, Fragment Offset 用于IP包分段

Timeto Live IP 包的存活时长

Protocol ICMP= 1

Header Checksum 头校验和(检查整个 IP报头)

Addresses发送Echo消息的源地址是发送 Echo reply消息的目的地址 相反 发送Echo

消息的目的地址是发送 Echo reply消息的源地址.

Echo或Echo Reply消息格式如下:

01 23

01 23456789012345678901 2345678901

+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+|Type|Code|Checksum|

+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+| Identifier | Sequence Number|

+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+|Data|

+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+Typeecho消息的类型为8echo reply 的消息类型为0.

Code=0

Checksum

为从TYPE开始到IP包结束的校验和也就是校验整个 ICMP报文

Identifier

如果code= 0, identifier 用来匹配echo和echo reply 消息

Sequence Number

如果code= 0, identifier 用来匹配echo和echo reply 消息

功能描述

收到echo消息必须回应 echo reply消息. identifier 和sequence number可能被发送echo的主机用来匹配返回的 echo reply消息.例如 identifier 可能用于类似于TCP或UDP的port用来标示一个会话,而sequence number会在每次发送echo请求后递增.收到echo的主机或路由器返回同一个值与之匹配

2数据结构

1 IP报头格式

//定义IP首部typedef struct_iphdr{unsigned charh_lenver; unsigned char tos; //4位IP版本号+4位首部长度unsigned short total_len; //8位服务类型TOSunsigned short ident; unsigned //16位IP包总长度字节short frag_and_flags; unsigned 〃 1 6位标识 用于辅助IP包的拆装 本实验不用 置零char ttl ; unsigned char proto; //3位标志位+13位偏移位,也是用于IP包的拆装,本实验不用,置零//8位IPunsigned short checksum; 包生存时间TTLunsigned int sourceIP; 〃8位协议TCP,UDP或其他 ,本实验置ICMP,置为1unsigned int destIP; //16位IP首部校验和最初置零等所有包头都填写正确后计算并替换. //32}IP_HEADER; 位源IP地址

2 ICMP报头格式 //32位目的IP地址

//定义ICMP首部

typedef struct_icmphdr{ 〃8位类型 本实验用8:ECHO 0:ECHO REPLYun sig ned char i_type; 〃8位代码,本实验置零un sig ned char i_code; 〃 16位校验和 从TYPE开始,直到最后一位用户数据 如果为字节数为奇数unsigned short i_cksum; 则补充一位

//识别号一般用进程号作为识别号  用于匹配ECHO和ECHO REPLY包unsigned short i_id; 〃报文序列号 用于标记ECHO报文顺序

//时间戳un sig ned short i_seq ;un sig ned int timestamp;

}ICMP_HEADER;

3总体设计

ICMP协议中的发送

4.VC中网络套接字Win sock编程基础

在VC中进行WINSOC的API编程开发的时候需要在项目中使用下面三个文件 否则会出现编译错误。

1 .WINSOCK.H:这是WINSOCKAP的头文件需要包含在项目中。

2 .WSOCK32.LIB:WINSOCK API连接库文件。在使用中一定要把它作为项目的非缺省的连接库包含到项目文件中去。

3. WINSOCK.DLL:WINSOC的动态连接库位于WINDOW的安装目录下。

几个基本的套接字

1 、 创建套接字——socket()功能使用前创建一个新的套接字

格式 SOCKET PASCAL FAR socket。 nt af, int type, int procotol);

参数 af:通信发生的区域type: 要建立的套接字类型procotol : 使用的特定协议

2、指定本地地址——bind()

功能 将套接字地址与所创建的套接字号联系起来。

格式 int PASCAL FAR bind(SOCKET s,const struct sockaddr FAR*name, int namelen);

参数 s:是由socket()调用返回的并且未作连接的套接字描述符(套接字号)。 其它 没有错误 bind()

返回0 否则SOCKET_ERROR

地址结构说明struct sockaddr_in

{short sin_fami ly;//AF_INETu_short sin_port;//16 位端口号 网络字节顺序struct in_addr sin_addr;//32 位IP地址 网络字节顺序char sin_zero[8];// 保留

}

3 建立套接字连接-----------connect()和accept()

功能 共同完成连接工作

格式 int PASCAL FAR connect(SOCKET s,const struct sockaddr FAR*name, int namelen);

SOCKET PASCAL FAR accept(SOCKET s,struct sockaddr FAR*name, int FAR

*addrlen);

参数 同上

4、 监听连接——l isten()

功能用于面向连接服务器表明它愿意接收连接。格式 int PASCAL FAR l isten(SOCKET s, int backlog);

5、 数据传输——send()与recv()

功能 数据的发送与接收

格式 int PASCAL FAR send(SOCKET s,const char FAR*buf, int len, int flags);int PASCAL FAR recv(SOCKET s,const char FAR*buf, int len, int flags);

参数 b uf:指向存有传输数据的缓冲区的指针。

6、 多路复用——se lect()

功能 用来检测一个或多个套接字状态。

格式 int PASCAL FAR select(int nfds,fd_set FAR* readfds,fd_set FAR*writefds,fd_set FAR*exceptfds,const struct timeval FAR* timeout);

参数

7、 关闭套接字——closesocket()

功能关闭套接字s

格式 BOOL PASCAL FAR closesocket(SOCKET s);

5部分程序代码

//初始化SOCKET

WSADATA wsaData;iErrorCode=WSAStartup(MAKEWORD(2,2),&wsaData);CheckSockError(iErrorCode, "WSAStartup");sockRaw=socket(AF_INET,SOCK_RAW, IPPROTO_ICMP); // 原始套接字

CheckSockError(sockRaw, "socket");

//设置超时时间timeout=time;iErrorCode=setsockopt(sockRaw,SOL_SOCKET,SO_RCVTIMEO,(char*)&timeout,sizeof(timeout));//时CheckSockError(iErrorCode, "SO_RCVTIMEO"); 设置接受延timeout=time;iErrorCode=setsockopt(sockRaw,SOL_SOCKET,SO_SNDTIMEO,(char*)&timeout,sizeof(timeout));//时

CheckSockError(iErrorCode, "SO_SNDTIMEO");

设置发送延//获得目标主机IP memset(&dest,0,sizeof(dest));//初始化dest结构dest.sin_fami ly=AF_I NET; // 填充SOCKADDR」结构内容if((dest.sin_addr.s_addr=inet_addr(lpdest))==INADDR_NONE)

{

if((hp=gethostbyname(lpdest)) !=NULL) // 目的主机名字不为空

{memcpy(&(dest.sin_addr),hp->h_addr_l ist[0],hp->h_length);dest.sin_fami ly=hp->h_addrtype;printf("dest.sin_addr=%s\n", inet_ntoa(dest.sin_addr));

}else

{

CheckSockError(SOCKET_ERROR, "gethostbyname()");

}

}

{

//创建ICMP数据包datasize+= // 包长sizeof(ICMP_HEADER); icmp_data= 创建icmp数据报内存空间

(char *)mal loc(1024); // recvbuf= 接收icmp包缓冲区

(char *)mal loc(1024); //if((! icmp_data) | | (!recvb uf))

CheckSockError(SOCKET_ERROR, "mal loc()");

}memset(icmp_data,0,MAX_PACKET); //初始化icmp_data Fi l l ICMPData(icmp_data,datasize); //

填充icmp包printf("Pinging%s with%d bytes of data(timeout=%dms):\n\n", inet_ntoa(dest.sin_addr),datasize,timeout );

//发送与接收ICMP数据包whi le(1)

{memset(recvbuf,0,MAX_PACKET); //初始化接受缓冲区static int nCount=0; // 设置发送icmp包的次数一般为4if(nCount++==4)break;

((ICMP_HEADER*)icmp_data)->i_cksum=0; // 初设校验和为0

((ICMP_HEADER*)icmp_data)->timestamp=GetTickCount(); // 获得目前时间

((ICMP_HEADER*)icmp_data)->i_seq=seq_no++; //icmp 数据报的序列号((ICMP_HEADER*)icmp_data)->i_cksum=checksum((USHO RT*)icmp_data,datasize);// 计算校验和iErrorCode=sendto(sockRaw, icmp_data,datasize,0,(structsockaddr*)&dest,sizeof(dest));//发送icmp数据报if(iErrorCode==SOCKET_ERROR) //错误检查

{if(WSAGetLastError()==WSAETIMEDO UT)

{printf("timed out\n");continue;

}

CheckSockError(SOCKET_ERROR, "sendto()");

} if(iErrorCode<datasize)

{printf("Wrote%d bytes\n", iErrorCode);

}int fromlen=sizeof(from);// 接受icmp包长度

iErrorCode=recvfrom(sockRaw,recvbuf,MAX_PACKET,0,(structsockaddr*)&from,&fromlen);//接受icmp包if(iErrorCode==SOCKET_ERROR)

{if(WSAGetLastError()==WSAETIMEDO UT)

{printf("timed out\n");continue;

}

CheckSockError(SOCKET_ERROR, "recvfrom()");

}DecodeICMPHeader(recvbuf, iErrorCode,&from); //分解icmp包头

Sleep(1000); //休眠一段时间

}

//SOCK错误处理程序void CheckSockError(int iErrorCode,char *pErrorMsg) {if(iErrorCode==SOCKET_ERROR)

{printf("%s Error:%d\n", pErrorMsg,GetLastError()); closesocket(sockRaw);

ExitProcess(0);

//填充数据void Fi l l ICMPData(char *icmp_data, int datasize)

{

ICMP_HEADER*icmp_hdr=NULL;char *datapart=NULL;icmp_hdr=(ICMP_HEADER*)icmp_data; icmp_hdr->i_type=

ICMP_ECHO; //发送ping//Request an ICMP echoicmp_hdr->i_code=0; //代码字段为0 icmp_hdr->i_id=

(USHORT)GetCurrentProcessId(); // icmp_hdr->i_cksum=0;icmp_hdr->i_seq=0; // 初始化序列号datapart=icmp_data+sizeof(ICMP_HEADER); ////

//Place somejunk in the buffer

// 加上icmp包头memset(datapart, 'E',datasize-sizeof(ICMP_HEADER)); // }

//计算检验和

unsigned long cksum=0;whi le(size> 1)

{cks u m+=*buffe r++;size-=sizeof(US HORT);

}if (size)

{cksu m+=*(UC HAR*)b uffer;

}cksum=(cksum>>16)+(cksum&0xffff);

cksum+=(cksum>>16); return(USHO RT)(~cksum); }

//ICMP解包程序void DecodeICMPHeader(char *buf, int bytes, struct sockaddr_in*from)

{

IP_HEADER*iphdr=NULL;

ICMP_HEADER*icmphdr=NULL; unsigned short iphdrlen;

DWORD tick;iphdr=(IP_HEADER*)buf;

//Number of 32-bit words*4=bytesiphdrlen=sizeof(unsigned long) * (iphdr->h_lenver&0xf); // 计算ip包头长度tick=GetTickCount();if(bytes<iphdrlen+ICMP_MIN) // 数据报太短丢弃

{printf("Too few bytes from%s\n", inet_ntoa(from->sin_addr));

}icmphdr=(ICMP_HEADER*)(buf+iphdrlen);if (icmphdr->i_type!=ICMP_ECHOREPLY) // 不是回送响应(ping应答)丢弃

{printf("nonecho type%d recvd\n", icmphdr->i_type); return;

}

//Make sure this is an ICMP reply to something we sent!

//if (icmphdr->i_id !=(USHORT)GetCurrentProcessId()) //id 号不符合丢弃

{printf("someone else‘s packet! \n");return;

}printf("%d bytes from%s:",bytes, inet_ntoa(from->sin_addr)); // 输出正在使用的ip地址printf(" icmp_seq=%d. ", icmphdr->i_seq); //printf(" time:%d输出序列号ms", tick- icmphdr->timestamp); //printf("\n"); 输出所用时间return;

}

6实验结果

该程序用来检验网络中的一台目标主机是否可达其功能相当与 Windows系统自带的ping命令。例如当程序检验地址为可以返回如下信息。

CloudCone,美国洛杉矶独立服务器特价优惠,美国洛杉矶MC机房,100Mbps带宽不限流量,可选G口,E3-1270 v2处理器32G内存1Gbps带宽,69美元/月

今天CloudCone发布了最新的消息,推送了几款特价独立服务器/杜甫产品,美国洛杉矶MC机房,分配100Mbps带宽不限流量,可以选择G口限制流量计划方案,存储分配的比较大,选择HDD硬盘的话2TB起,MC机房到大陆地区线路还不错,有需要美国特价独立服务器的朋友可以关注一下。CloudCone怎么样?CloudCone服务器好不好?CloudCone值不值得购买?CloudCone是一家成立于2...

[黑五]ProfitServer新加坡/德国/荷兰/西班牙VPS五折,不限流量KVM月付2.88美元起

ProfitServer已开启了黑色星期五的促销活动,一直到本月底,商家新加坡、荷兰、德国和西班牙机房VPS直接5折,无码直购最低每月2.88美元起,不限制流量,提供IPv4+IPv6。这是一家始于2003年的俄罗斯主机商,提供虚拟主机、VPS、独立服务器、SSL证书、域名等产品,可选数据中心包括俄罗斯、法国、荷兰、美国、新加坡、拉脱维亚、捷克、保加利亚等多个国家和地区。我们随便以一个数据中心为例...

Megalayer美国独立服务器配置及性能速度综合评测

Megalayer 商家在之前也有记录过,商家开始只有提供香港站群服务器和独立服务器,后来也有增加到美国独立服务器,以及前几天也有介绍到有增加香港VPS主机。对于香港服务器之前有过评测(Megalayer香港服务器配置一览及E3-1230 8GB服务器评测记录),这里申请到一台美国独立服务器,所以也准备简单的评测记录。目前市场上我们看到很多商家提供VPS或者云服务器基本上没有什么特别的,但是独立服...

icmp协议为你推荐
fontfamilyfont-family:"microsoft yahei",simhei; 这句到底设置为微软雅黑还是黑体,为什么写2个字体?可以发外链的论坛给几个可以发外链的论坛,还有分类信息网,不要有限制的哪种,收录不收录无所谓邮箱怎么写邮箱地址怎么写google竞价排名谷歌竞价排名现在是显示在什么位置?主页改不了怎么改不了主页快速美白好方法快速美白的好点子!?(不是晒黑的)手机区号手机号码网店推广网站什么平台适合做淘宝店铺推广直播加速手机上什么软件可以帮助直播加速,大神们推荐推荐qq怎么发邮件怎样在QQ上发送邮件?
新网域名管理 smartvps cybermonday bandwagonhost 外贸主机 56折 主机屋免费空间 godaddy 国外php空间 panel1 合租空间 中国电信测网速 国外代理服务器地址 免费申请个人网站 如何用qq邮箱发邮件 hktv 789电视剧 中国电信宽带测速器 免费邮件服务器 上海电信测速网站 更多