11 原始套接字

原始套接口(RAW SOCKET)

 

原始套接口提供三种TCP和UDP套接口不提供的功能:

1). 可以读写ICMPv4, IGMPv4和ICMPv6分组. 例如Ping和Traceroute程序就是利用ICMP分组

2). 可以读写特殊的IPv4数据报, 内核不处理这些数据报IPv4协议字段

3). 使用IP_HDRINCL选项可以构造自己IPv4头部. 可以用这个特性来构造自己的TCP和UTP分组

 

原始套接口创建

1). 设置第二个参数为SOCK_RAW

[cpp]  view plain copy
  1. int     sockfd;  
  2. sockfd = socket(AF_INET, SOCK_RAW, IPPROTO_XXX);  

这个IPPROTO_XXX可以在<netinet/in.h>找到定义, 例如可以为IPPROTO_ICMP,IPPROTO_IP, IPPRPTO_TCP和IPPRPTO_UDP等

2). 可以设置IP_HDRINCL选项

[c-sharp]  view plain copy
  1. const int on = 1;  
  2. if (setsockopt(sockfd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0)  
  3.     error  

3). 可以调用bind绑定本地IP

4). 可以调用connect绑定目的IP

 

原始套接口输出

1). 通过sendto或sendmsg到目的IP, 或write, writev或send到connect的原始套接口

2). 如果IP_HDRINCL没有设置, 用户数据从IP头之后开始

3). 如果IP_HDRINCL设置, 用户数据必须包括IP头, 并且要自己设置IPv4标示字段和校验和之外的所有字段

4). 对以超出外出MTU分组, 内核将其分片

 

原始套接口输入

1). 接收到的TCP和UTP分组绝不会传递给任何原始套接口

2). 绝大多数ICMP分组会传递给相应的原始套接口

3). 所有IGMP分组会传递给给相应的原始套接口

4). 多有带有内核不能识别的协议字段的IP数据报会给相应的原始套接口

5). 如果数据包是分片, 内核要等到接收到所有片才转发

在内核准备传递一个数据报时, 将对所有原始套接口进行检查, 如果满足下面三个条件将传送一个拷贝:

1). 协议字段不为0是必须匹配

2). 如果绑定了本地IP, 必须匹配

3). 如果设置了目的IP, 必须匹配

所以如果没有bind和connect, 而且协议为0, 则接收多有数据报

你可能感兴趣的:(11 原始套接字)