实验六基于ICMP协议的ping程序设计实验Word文件下载.docx
《实验六基于ICMP协议的ping程序设计实验Word文件下载.docx》由会员分享,可在线阅读,更多相关《实验六基于ICMP协议的ping程序设计实验Word文件下载.docx(18页珍藏版)》请在冰豆网上搜索。
因此,保证数据送达的工作应该由其他的模块来完成。
其中一个重要的模块就是ICMP(网络控制报文)协议。
当传送IP数据包发生错误,比如主机不可达,路由不可达等等,ICMP协议将会把错误信息封包,然后传送回给主机。
给主机一个处理错误的机会,这也就是为什么说建立在IP层以上的协议是可能做到安全的原因。
ICMP数据包由8bit的错误类型和8bit的代码和16bit的校验和组成。
而前16bit就组成了ICMP所要传递的信息。
PING利用ICMP协议包来侦测另一个主机是否可达。
其原理是用类型码为0的ICMP发请求,受到请求的主机则用类型码为8的ICMP回应。
ping程序来计算间隔时间,并计算有多少个包被送达。
用户就可以判断网络大致的情况。
2、RAW模式的SOCKET编程
PING程序是面向用户的应用程序,该程序使用ICMP的封装机制,通过IP协议来工作。
为了实现直接对IP和ICMP包进行操作,实验中使用RAW模式的SOCKET编程。
熟悉SOCKET的编程,包括基本的系统调用如SOCKET、BIND等。
3、ICMP协议
网络本身是不可靠的,在网络传输过程中,可能会发生许多突发事件并导致数据传输失败。
位于网络层的IP协议是一个无连接的协议,它不会处理网络层传输中的故障,而位于网络层的ICMP协议却恰好弥补了IP的缺限,它使用IP协议进行信息传递,向数据包中的源端节点提供发生在网络层的错误信息反馈。
ICMP全称InternetControlMessageProtocol,中文名为因特网控制报文协议,它的报头长8字节,结构如下图所示:
比特0 7 8
15
16
比特31
类型(0或8)
代码(0)
检验和
标识符
序号
数据
ICMP协议提供的诊断报文类型如下表所示:
类型
描述
回应应答(Ping应答,与类型8的Ping请求一起使用)
3
目的不可达
4
源消亡
5
重定向
8
回应请求(Ping请求,与类型8的Ping应答一起使用)
9
路由器公告(与类型10一起使用)
10
路由器请求(与类型9一起使用)
11
超时
12
参数问题
13
时标请求(与类型14一起使用)
14
时标应答(与类型13一起使用)
15
信息请求(与类型16一起使用)
16
信息应答(与类型15一起使用)
17
地址掩码请求(与类型18一起使用)
18
地址掩码应答(与类型17一起使用)
ICMP提供多种类型的消息为源端节点提供网络层的故障信息反馈,它的报文类型可以归纳为以下5个大类:
1)诊断报文(类型8,代码0;
类型0,代码0);
2)目的不可达报文(类型3,代码0-15);
3)重定向报文(类型5,代码0-4);
4)超时报文(类型11,代码0-1);
5)信息报文(类型12-18)。
六、实验步骤及注意点
1)熟悉IP以及ICMP协议的工作机制;
2)熟悉RAW模式的SOCKET编程;
3)编写PING的实现程序;
4)在模拟实现环境下调试并运行自己编写的PING程序;
5)编译环境中需要包括SOCKET库WS2_32.lib。
七、实验报告要求
提交源程序,并撰写实验报告。
八、相关参考资料
1)WinsockProgrammer'
sFAQExamples:
Ping:
RawSocketsMethod,
2)透析ICMP协议:
协议原理,
3)ping原理与ICMP协议,
九、SOCKET编程基本知识
1、SOCKET规范概述
WindowsSockets规范以U.C.Berkeley大学BSDUNIX中流行的Socket接口为范例定义了一套MicorosoftWindows下网络编程接口。
它不仅包含了人们所熟悉的BerkeleySocket风格的库函数;
也包含了一组针对Windows的扩展库函数,以使程序员能充分地利用Windows消息驱动机制进行编程。
WindowsSockets规范本意在于提供给应用程序开发者一套简单的API,并让各家网络软件供应商共同遵守。
此外,在一个特定版本Windows的基础上,WindowsSockets也定义了一个二进制接口(ABI),以此来保证应用WindowsSocketsAPI的应用程序能够在任何网络软件供应商的符合WindowsSockets协议的实现上工作。
因此这份规范定义了应用程序开发者能够使用,并且网络软件供应商能够实现的一套库函数调用和相关语义。
遵守这套WindowsSockets规范的网络软件,我们称之为WindowsSockets兼容的,而WindowsSockets兼容实现的提供者,我们称之为WindowsSockets提供者。
一个网络软件供应商必须百分之百地实现WindowsSockets规范才能做到现WindowsSockets兼容。
任何能够与WindowsSockets兼容实现协同工作的应用程序就被认为是具有WindowsSockets接口。
我们称这种应用程序为WindowsSockets应用程序。
WindowsSockets规范定义并记录了如何使用API与Internet协议族(IPS,通常我们指的是TCP/IP)连接,尤其要指出的是所有的WindowsSockets实现都支持流套接口和数据报套接口.
应用程序调用WindowsSockets的API实现相互之间的通讯。
WindowsSockets又利用下层的网络通讯协议功能和操作系统调用实现实际的通讯工作。
2、WINDOWS环境下SOCKET基本函数
(1)WSAStartup函数;
intWSAStartup(WORDwVersionRequested,LPWSADATAlpWSAData);
使用Socket的程序在使用Socket之前必须调用WSAStartup函数。
该函数的第一个参数指明程序请求使用的Socket版本,其中高位字节指明副版本、低位字节指明主版本;
操作系统利用第二个参数返回请求的Socket的版本信息。
当一个应用程序调用WSAStartup函数时,操作系统根据请求的Socket版本来搜索相应的Socket库,然后绑定找到的Socket库到该应用程序中。
以后应用程序就可以调用所请求的Socket库中的其它Socket函数了。
该函数执行成功后返回0。
例:
假如一个程序要使用2.1版本的Socket,那么程序代码如下
wVersionRequested=MAKEWORD(2,1);
err=WSAStartup(wVersionRequested,&
wsaData);
(2)WSACleanup函数
intWSACleanup(void);
应用程序在完成对请求的Socket库的使用后,要调用WSACleanup函数来解
除与Socket库的绑定并且释放Socket库所占用的系统资源。
(3)socket函数
SOCKETsocket(intaf,inttype,intprotocol);
应用程序调用socket函数来创建一个能够进行网络通信的套接字。
第一个
参数指定应用程序使用的通信协议的协议族,对于TCP/IP协议族,该参数置PF_INET;
第二个参数指定要创建的套接字类型,流套接字类型为SOCK_STREAM、数据报套接字类型为SOCK_DGRAM;
第三个参数指定应用程序所使用的通信协议。
该函数如果调用成功就返回新创建的套接字的描述符,如果失败就返回INVALID_SOCKET。
套接字描述符是一个整数类型的值。
每个进程的进程空间里都有一个套接字描述符表,该表中存放着套接字描述符和套接字数据结构的对应关系。
该表中有一个字段存放新创建的套接字的描述符,另一个字段存放套接字数据结构的地址,因此根据套接字描述符就可以找到其对应的套接字数据结构。
每个进程在自己的进程空间里都有一个套接字描述符表但是套接字数据结构都是在操作系统的内核缓冲里。
下面是一个创建流套接字的例子:
structprotoent*ppe;
ppe=getprotobyname("
tcp"
);
SOCKETListenSocket=socket(PF_INET,SOCK_STREAM,ppe->
p_proto);
(4)closesocket函数
intclosesocket(SOCKETs);
closesocket函数用来关闭一个描述符为s套接字。
由于每个进程中都有
一个套接字描述符表,表中的每个套接字描述符都对应了一个位于操作系统缓冲区中的套接字数据结构,因此有可能有几个套接字描述符指向同一个套接字数据结构。
套接字数据结构中专门有一个字段存放该结构的被引用次数,即有多少个套接字描述符指向该结构。
当调用closesocket函数时,操作系统先检查套接字数据结构中的该字段的值,如果为1,就表明只有一个套接字描述符指向它,因此操作系统就先把s在套接字描述符表中对应的那条表项清除,并且释放s对应的套接字数据结构;
如果该字段大于1,那么操作系统仅仅清除s在套接字描述符表中的对应表项,并且把s对应的套接字数据结构的引用次数减1。
closesocket函数如果执行成功就返回0,否则返回SOCKET_ERROR。
(5)send函数
intsend(SOCKETs,constcharFAR*buf,intlen,intflags);
不论是客户还是服务器应用程序都用send函数来向TCP连接的另一端发送数据。
客户程序一般用send函数向服务器发送请求,而服务器则通常用send函数来向客户程序发送应答。
该函数的第一个参数指定发送端套接字描述符;
第二个参数指明一个存放应用程序要发送数据的缓冲区;
第三个参数指明实际要发送的数据的字节数;
第四个参数一般置0。
这里只描述同步Socket的send函数的执行流程。
当调用该函数时,send先比较待发送数据的长度len和套接字s的发送缓冲区的长度,如果len大于s的发送缓冲区的长度,该函数返回SOCKET_ERROR;
如果len小于或者等于s的发送缓冲区的长度,那么send先检查协议是否正在发送s的发送缓冲中的数据,如果是就等待协议把数据发送完,如果协议还没有开始发送s的发送缓冲中的数据或者s的发送缓冲中没有数据,那么send就比较s的发送缓冲区的剩余空间和len,如果len大于剩余空间大小send就一直等待协议把s的发送缓冲中的数据发送完,如果len小于剩余空间大小send就仅仅把buf中的数据copy到剩余空间里(注意并不是send把s的发送缓冲中的数据传到连接的另一端的,而是协议传的,send仅仅是把buf中的数据copy到s的发送缓冲区的剩余空间里)。
如果send函数copy数据成功,就返回实际copy的字节数,如果send在copy数据时出现错误,那么send就返回SOCKET_ERROR;
如果send在等待协议传送数据时网络断开的话,那么send函数也返回SOCKET_ERROR。
要注意send函数把buf中的数据成功copy到s的发送缓冲的剩余空间里后它就返回了,但是此时这些数据并不一定马上被传到连接的另一端。
如果协议在后续的传送过程中出现网络错误的话,那么下一个Socket函数就会返回SOCKET_ERROR。
(每一个除send外的Socket函数在执行的最开始总要先等待套接字的发送缓冲中的数据被协议传送完毕才能继续,如果在等待时出现网络错误,那么该Socket函数就返回SOCKET_ERROR)
注意:
在Unix系统下,如果send在等待协议传送数据时网络断开的话,调用send的进程会接收到一个SIGPIPE信号,进程对该信号的默认处理是进程终止。
(6)recv函数
intrecv(SOCKETs,charFAR*buf,intlen,intflags);
不论是客户还是服务器应用程序都用recv函数从TCP连接的另一端接收数
据。
该函数的第一个参数指定接收端套接字描述符;
第二个参数指明一个缓冲区,该缓冲区用来存放recv函数接收到的数据;
第三个参数指明buf的长度;
这里只描述同步Socket的recv函数的执行流程。
当应用程序调用recv函数时,recv先等待s的发送缓冲中的数据被协议传送完毕,如果协议在传送s的发送缓冲中的数据时出现网络错误,那么recv函数返回SOCKET_ERROR,如果s的发送缓冲中没有数据或者数据被协议成功发送完毕后,recv先检查套接字s的接收缓冲区,如果s接收缓冲区中没有数据或者协议正在接收数据,那么recv就一直等待,只到协议把数据接收完毕。
当协议把数据接收完毕,recv函数就把s的接收缓冲中的数据copy到buf中(注意协议接收到的数据可能大于buf的长度,所以在这种情况下要调用几次recv函数才能把s的接收缓冲中的数据copy完。
recv函数仅仅是copy数据,真正的接收数据是协议来完成的),recv函数返回其实际copy的字节数。
如果recv在copy时出错,那么它返回SOCKET_ERROR;
如果recv函数在等待协议接收数据时网络中断了,那么它返回0。
在Unix系统下,如果recv函数在等待协议接收数据时网络断开了,那么调用recv的进程会接收到一个SIGPIPE信号,进程对该信号的默认处理是进程终止。
(7)bind函数
intbind(SOCKETs,conststructsockaddrFAR*name,intnamelen);
当创建了一个Socket以后,套接字数据结构中有一个默认的IP地址和默认的端口号。
一个服务程序必须调用bind函数来给其绑定一个IP地址和一个特定的端口号。
客户程序一般不必调用bind函数来为其Socket绑定IP地址和断口号。
该函数的第一个参数指定待绑定的Socket描述符;
第二个参数指定一个sockaddr结构,该结构是这样定义的:
structsockaddr{
u_shortsa_family;
charsa_data[14];
};
sa_family指定地址族,对于TCP/IP协议族的套接字,给其置AF_INET。
当对TCP/IP协议族的套接字进行绑定时,我们通常使用另一个地址结构:
structsockaddr_in{
shortsin_family;
u_shortsin_port;
structin_addrsin_addr;
charsin_zero[8];
其中sin_family置AF_INET;
sin_port指明端口号;
sin_addr结构体中只有一个唯一的字段s_addr,表示IP地址,该字段是一个整数,一般用函数inet_addr()把字符串形式的IP地址转换成unsignedlong型的整数值后再置给s_addr。
有的服务器是多宿主机,至少有两个网卡,那么运行在这样的服务器上的服务程序在为其Socket绑定IP地址时可以把htonl(INADDR_ANY)置给s_addr,这样做的好处是不论哪个网段上的客户程序都能与该服务程序通信;
如果只给运行在多宿主机上的服务程序的Socket绑定一个固定的IP地址,那么就只有与该IP地址处于同一个网段上的客户程序才能与该服务程序通信。
我们用0来填充sin_zero数组,目的是让sockaddr_in结构的大小与sockaddr结构的大小一致。
下面是一个bind函数调用的例子:
structsockaddr_insaddr;
saddr.sin_family=AF_INET;
saddr.sin_port=htons(8888);
saddr.sin_addr.s_addr=htonl(INADDR_ANY);
bind(ListenSocket,(structsockaddr*)&
saddr,sizeof(saddr));
(8)listen函数
intlisten(SOCKETs,intbacklog);
服务程序可以调用listen函数使其流套接字s处于监听状态。
处于监听状态的流套接字s将维护一个客户连接请求队列,该队列最多容纳backlog个客户连接请求。
假如该函数执行成功,则返回0;
如果执行失败,则返回SOCKET_ERROR。
(9)accept函数
SOCKETaccept(SOCKETs,structsockaddrFAR*addr,intFAR*addrlen);
服务程序调用accept函数从处于监听状态的流套接字s的客户连接请求队列中取出排在最前的一个客户请求,并且创建一个新的套接字来与客户套接字创建连接通道,如果连接成功,就返回新创建的套接字的描述符,以后与客户套接字交换数据的是新创建的套接字;
如果失败就返回INVALID_SOCKET。
该函数的第一个参数指定处于监听状态的流套接字;
操作系统利用第二个参数来返回新创建的套接字的地址结构;
操作系统利用第三个参数来返回新创建的套接字的地址结构的长度。
下面是一个调用accept的例子:
structsockaddr_inServerSocketAddr;
intaddrlen;
addrlen=sizeof(ServerSocketAddr);
ServerSocket=accept(ListenSocket,(structsockaddr*)&
ServerSocketAddr,&
addrlen);
(10)connect函数
intconnect(SOCKETs,conststructsockaddrFAR*name,intnamelen);
客户程序调用connect函数来使客户Sockets与监听于name所指定的计算机的特定端口上的服务Socket进行连接。
如果连接成功,connect返回0;
如果失败则返回SOCKET_ERROR。
下面是一个例子:
structsockaddr_indaddr;
memset((void*)&
daddr,0,sizeof(daddr));
daddr.sin_family=AF_INET;
daddr.sin_port=htons(8888);
daddr.sin_addr.s_addr=inet_addr("
133.197.22.4"
connect(ClientSocket,(structsockaddr*)&
daddr,sizeof(daddr));
3、RAW模式的SOCKET编程
WindowsSockets是一个编程接口,它是在加州大学伯克利分校开发的套接字接口的基础上定义的。
它包括了一组扩展件,以充分利用MicrosoftWindows消息驱动的特点。
规范的1.1版是在1993年1月发行的,2.2.0版在1996年5月发行。
Windows2000支持Winsock2.2版。
在Winsock2中,支持多个传输协议的原始套接字,重叠I/O模型、服务质量控制等。
这里介绍WindowsSockets的一些关于原始套接字(RawSocket)的编程。
同Winsock1相比,最明显的就是支持了RawSocket套接字类型,通过原始套接字,我们可以更加自如地控制Windows下的多种协议,而且能够对网络底层的传输机制进行控制。
(1)创建一个原始套接字,并设置IP头选项。
SOCKETsock;
sock=socket(AF_INET,SOCK_RAW,IPPROTO_IP);
或者:
s=WSASoccket(AF_INET,SOCK_RAW,IPPROTO_IP,NULL,0,WSA_FLAG_OVERLAPPED);
这里,我们设置了SOCK_RAW标志,表示我们声明的是一个原始套接字类型。
创建原始套接字后,IP头就会包含在接收的数据中,如果我们设定IP_HDRINCL选项,那么,就需要自己来构造IP头。
注意,如果设置IP_HDRINCL选项,那么必须具有administrator权限,要不就必须修改注册表:
HKEY_LOCAL_MACHINESystemCurrentControlSetServicesAfdParameter
修改键:
DisableRawSecurity(类型为DWORD),把值修改为1。
如果没有,就添加。
BOOLblnFlag=TRUE;
setsockopt(sock,IPPROTO_IP,IP_HDRINCL,(char*)&
blnFlag,sizeof(blnFlag);
对于原始套接字在接收数据报的时候,要注意这么几点:
如果接收的数据报中协议类型和定义的原始套接字匹配,那么,接收的所有数据就拷贝到套接字中。
如果绑定了本地地址,那么只有接收数据IP头中对应的远端地址匹配,接收的数据就拷贝到套接字中。
如果定义的是外部地址,比如使用connect(),那么,只有接收数据IP头中对应的源地址匹配,接收的数据就拷贝到套接字中。
(2)构造IP头和TCP头
这里,提供IP头和TCP头的结构:
//StandardTCPflags
#defineURG0x20
#defineACK0x10
#def