在计算机通信领域中,IP头是构成所有TCP/IP网络报文的基础。因此,了解IP头的结构和各个字段的含义,对于网络通信的理解和网络应用点问题的调试都非常重要。本文将从多个方面详细阐述IP头的相关内容。
一、IPHeader的结构
typedef struct ip_hdr {
#if BYTE_ORDER == LITTLE_ENDIAN
u_char ip_hl:4, /* 小端模式,低字节存储在内存的低地址中 */
ip_v:4;
#endif
#if BYTE_ORDER == BIG_ENDIAN
u_char ip_v:4, /* 大端模式,低字节存储在内存的高地址中 */
ip_hl:4;
#endif
u_char ip_tos; /* 服务类型字段 */
short ip_len; /* 报文总长度 */
u_short ip_id; /* 报文标志 */
short ip_off; /* 分片偏移 */
u_char ip_ttl; /* 生存时间 */
u_char ip_p; /* 协议类型 */
u_short ip_sum; /* 校验和 */
struct in_addr ip_src,ip_dst; /* 源IP地址和目的IP地址 */
} IP_HDR;
IP头包含多个字段,其中包括版本、头部长度、服务类型、总长度、标识、分片偏移、TTL、协议类型、校验和、源IP地址以及目的IP地址等。位于每个字段后面的注释中,有关于该字段的简单解释。
二、IPHeader的版本
IP头中的版本字段(ip_v)记录了IP协议的版本。IPv4的版本为4,而IPv6的版本为6。IPv4是当前使用最广泛的IP协议版本,而IPv6协议是IPv4的下一代协议,具有更大的地址空间,能够支持更多的设备与应用程序连接到互联网。
三、IPHeader的TTL字段
Time-To-Live(TTL)字段是IP头的一个重要字段,它标识了IP数据包在网络上可以传输的最大跳数。每次经过一个路由器,TTL值就会减少1。当TTL值为0时,数据包将被丢弃,并向发送的主机返回一个ICMP TTL超时消息。因此,TTL的设置对于网络路由的优化非常重要。
四、IPHeader的协议类型字段
IP头中的协议类型字段(ip_p)标识了上层协议类型,例如TCP、UDP、ICMP和IGMP等。它的值对应于IANA协议号分配,详情请参考IANA的网站。
五、IPHeader的校验和字段
IP头中的校验和(ip_sum)是UDP、TCP和ICMP协议中使用的校验和的基础。它用于保证报文在从源到目的地的传输过程中的完整性。计算校验和时,所有16位字的1's complement被累加在一起。在累加结束后,将结果的1's complement取反,得到的值就是校验和。在将ICMP、TCP和UDP包发送到网络时,将自动计算生成校验和,并存储在校验和字段中。
六、IPHeader的源IP地址和目的IP地址
IP协议使用IP地址来唯一标识每个连接到互联网或者私有网络上的设备,就像人类通过姓名和地址之间的联系一样,IP地址在网络通信中扮演了相同的角色。源IP地址表示报文数据包来自哪个设备,目的IP地址表示接收数据包的设备在互联网上的位置。
七、IPHeader的应用实例
#include <netinet/ip.h>
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#define PACKET_SIZE 4096
int main()
{
int saddr_size , data_size;
struct sockaddr saddr;
unsigned char *buffer = (unsigned char *)malloc(PACKET_SIZE);
int sock_raw = socket(AF_INET , SOCK_RAW , IPPROTO_TCP);
if(sock_raw < 0)
{
perror("Could not create socket");
return 1;
}
while(1)
{
saddr_size = sizeof saddr;
// 接收原始数据包
data_size = recvfrom(sock_raw , buffer , PACKET_SIZE , 0 , &saddr , (socklen_t*)&saddr_size);
if(data_size < 0)
{
printf("Recvfrom error , failed to get packets\n");
return 1;
}
// 从缓冲区中解析IP头
struct iphdr *iph = (struct iphdr*)buffer;
// 根据需求对IP头进行操作
printf("Version : %d\n" , iph->version);
printf("Header length : %d\n" , iph->ihl);
printf("Type Of Service : %d\n" , iph->tos);
printf("Total length : %d\n" , iph->tot_len);
printf("Identification : %d\n" , iph->id);
printf("Time to live : %d\n" , iph->ttl);
printf("Protocol : %d\n" , iph->protocol);
printf("Checksum : %d\n" , iph->check);
printf("Source address : %d.%d.%d.%d\n" , (iph->saddr>>24)&0xff , (iph->saddr>>16)&0xff , (iph->saddr>>8)&0xff , iph->saddr&0xff);
printf("Destination address : %d.%d.%d.%d\n" , (iph->daddr>>24)&0xff , (iph->daddr>>16)&0xff , (iph->daddr>>8)&0xff , iph->daddr&0xff);
}
close(sock_raw);
return 0;
}
上述代码中使用socket函数创建一个原始套接字,然后使用recvfrom函数从缓冲区中解析IP头,并对IP头进行操作,最后输出相关信息。通常情况下,处理IP头时需要使用网络库库函数或者相关的系统调用。