原始套接字能够访问ICMP和ICMP等协议包,能够读写内核不处理的IP数据包。能够建立自定义的IP数据包首部。一句话,使用原始套接字能够
编写基于IP协议的通信程序。
1.建立原始套接字具体格式以下:int sockfd;sockfd = socktet(AF_INET, SOCK_RAW, IPPROTO_ICMP);第一个参数:协议族 AF_INET 表明TCP/IP协议第二个参数:SOCKET类型第三个参数:协议类型注意:@若是指定协议为0时,原始套接字能够接收内核传递给原始套接字的任何IP数据包,且只有超级用户才能够建立原始套接字。
@当须要编写本身的IP数据包首部时,能够在原始套接字上设置套接字选项IP_HDRINCL.在不设置这个选项的状况下,IP协议自动填充IP数据包的首部。
int on = 1;if(setsockopt(sockfd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0)
{ fprintf(stderr, "setsockopt IP_HDRINCL ERROR! /n");exit(1);}
原始套接字直接使用IP协议的套接字,因此是非面向链接的。在这个套接字上能够调用connect和bind函数,分别执行绑定对方和本地地址。
说明:
bind函数:调用bind函数后,发送数据包的源IP地址将是bind函数指定的地址。如是不调用bind,则内核将以发接口的主IP地址填充。若是设置了IP_HDRINCL,那么必须手工填充每一个发送数据包的源IP地址。
connetc函数:调用connect函数后,能够用write和send发送数据包。内核将用这个绑定的地址填充IP数据包的目的IP地址。
发送数据包
使用原始套接字发送数据包必须遵循如下规则:1.若是没有用connect函数绑定对方地址时,则应使用sendto或sendmsg函数发送数据包,在函数参数中指定对方地址。如?饔昧薱onnect函数,则能够直接使用send,write或writev来发送数据包。
2.若是没有设置IP_HDRINCL选项时,包内可写的内容为数据部分,内核将自动建立IP首部。若是设置了IP_HDRINCL选项,则包内要填充的内容为IP数据包和首部。内核只负责填充下面两个域:。若是将IP数据包的标识域设置为0,内核将设置这个域。内核老是计算和填充IP数据包首部的校验和。
注意:IP数据包首部各个域的内容都是网络字节顺序。
接收数据包
内核遵循如下规则接收数据包:1.UDP和TCP数据包从不传送给一个原始套接字。若是要查看这两类数据包,只能经过直接访问数据链路层来实现。
2.大多数ICMP数据包的一个拷贝传送给匹配的原始套接字。
3.内核处理的全部其它类型的数据包的一个拷贝都传给匹配的原始套接字。
4.全部内核不能识别的协议类型的IP数据包都传送给匹配的原始套接字。对于这些IP数据包,内核只作必要的检验工做。
在将一个IP数据包传送给原始套接字以前,内核须要选择匹配的原始套接字1.数据包的协议域必须与接收原始套接字的协议类型匹配。
2.若是原始套接字调用了bind函数绑定了本地IP地址,那么到达的IP数据包的源IP地址必须和对方的IP相匹配。
3.若是原始套接字调用connect函数指定了对方的IP地址,则到达的IP数据包的源IP地址秘须与这它相同网络