Linux 网络编程——原始套接字实例:发送 UDP 数据包

以太网报文格式:


详细的说明,请看《MAC 头部报文分析》。


IP 报文格式:



详细的说明,请看《IP 数据报格式详解》。


UDP 报文格式:



详细的说明,请看《UDP 数据报格式详解》。


校验和函数:

[objc] view plaincopy
  1. /******************************************************* 
  2. 功能: 
  3.     校验和函数 
  4. 参数: 
  5.     buf: 需要校验数据的首地址 
  6.     nword: 需要校验数据长度的一半 
  7. 返回值: 
  8.     校验和 
  9. *******************************************************/  
  10. unsigned short checksum(unsigned shortshort *buf, int nword)  
  11. {  
  12.     unsigned long sum;  
  13.     for(sum = 0; nword > 0; nword--)  
  14.     {  
  15.         sum += htons(*buf);  
  16.         buf++;  
  17.     }  
  18.     sum = (sum>>16) + (sum&0xffff);  
  19.     sum += (sum>>16);  
  20.     return ~sum;  
  21. }  


这里是在 ubuntu 下通过原始套接字组一个 udp 数据包,给 PC 机的网络调试助手发送信息:

[objc] view plaincopy
  1. #include   
  2. #include   
  3. #include   
  4. #include                //struct ifreq  
  5. #include             //ioctl、SIOCGIFADDR  
  6. #include   
  7. #include         //ETH_P_ALL  
  8. #include  //struct sockaddr_ll  
  9.   
  10.   
  11. unsigned short checksum(unsigned shortshort *buf, int nword);//校验和函数  
  12. int main(int argc, charchar *argv[])  
  13. {  
  14.     //1.创建通信用的原始套接字  
  15.     int sock_raw_fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));  
  16.       
  17.     //2.根据各种协议首部格式构建发送数据报  
  18.     unsigned char send_msg[1024] = {  
  19.         //--------------组MAC--------14------  
  20.         0x740x270xea, 0xb50xef, 0xd8//dst_mac: 74-27-EA-B5-FF-D8  
  21.         0xc80x9c, 0xdc, 0xb70x0f, 0x19//src_mac: c8:9c:dc:b7:0f:19  
  22.         0x080x00,                         //类型:0x0800 IP协议  
  23.         //--------------组IP---------20------  
  24.         0x450x000x000x00,             //版本号:4, 首部长度:20字节, TOS:0, --总长度--:  
  25.         0x000x000x000x00,             //16位标识、3位标志、13位片偏移都设置0  
  26.         0x8017,   0x000x00,             //TTL:128、协议:UDP(17)、16位首部校验和  
  27.         10,  221,   20,  11,                //src_ip: 10.221.20.11  
  28.         10,  221,   20,  10,                //dst_ip: 10.221.20.10  
  29.         //--------------组UDP--------8+78=86------  
  30.         0x1f, 0x900x1f, 0x90,             //src_port:0x1f90(8080), dst_port:0x1f90(8080)  
  31.         0x000x000x000x00,               //#--16位UDP长度--30个字节、#16位校验和  
  32.     };  
  33.       
  34.     int len = sprintf(send_msg+42"%s""this is for the udp test");  
  35.     if(len % 2 == 1)//判断len是否为奇数  
  36.     {  
  37.         len++;//如果是奇数,len就应该加1(因为UDP的数据部分如果不为偶数需要用0填补)  
  38.     }  
  39.       
  40.     *((unsigned shortshort *)&send_msg[16]) = htons(20+8+len);//IP总长度 = 20 + 8 + len  
  41.     *((unsigned shortshort *)&send_msg[14+20+4]) = htons(8+len);//udp总长度 = 8 + len  
  42.     //3.UDP伪头部  
  43.     unsigned char pseudo_head[1024] = {  
  44.         //------------UDP伪头部--------12--  
  45.         10,  221,   20,  11,                //src_ip: 10.221.20.11  
  46.         10,  221,   20,  10,                //dst_ip: 10.221.20.10  
  47.         0x0017,   0x000x00,                 //0,17,#--16位UDP长度--20个字节  
  48.     };  
  49.       
  50.     *((unsigned shortshort *)&pseudo_head[10]) = htons(8 + len);//为头部中的udp长度(和真实udp长度是同一个值)  
  51.     //4.构建udp校验和需要的数据报 = udp伪头部 + udp数据报  
  52.     memcpy(pseudo_head+12, send_msg+348+len);//--计算udp校验和时需要加上伪头部--  
  53.     //5.对IP首部进行校验  
  54.     *((unsigned shortshort *)&send_msg[24]) = htons(checksum((unsigned shortshort *)(send_msg+14),20/2));  
  55.     //6.--对UDP数据进行校验--  
  56.     *((unsigned shortshort *)&send_msg[40]) = htons(checksum((unsigned shortshort *)pseudo_head,(12+8+len)/2));  
  57.       
  58.       
  59.     //6.发送数据  
  60.     struct sockaddr_ll sll;                 //原始套接字地址结构  
  61.     struct ifreq ethreq;                    //网络接口地址  
  62.       
  63.     strncpy(ethreq.ifr_name"eth0", IFNAMSIZ);         //指定网卡名称  
  64.     if(-1 == ioctl(sock_raw_fd, SIOCGIFINDEX, ðreq))    //获取网络接口  
  65.     {  
  66.         perror("ioctl");  
  67.         close(sock_raw_fd);  
  68.         exit(-1);  
  69.     }  
  70.       
  71.     /*将网络接口赋值给原始套接字地址结构*/  
  72.     bzero(&sll, sizeof(sll));  
  73.     sll.sll_ifindex = ethreq.ifr_ifindex;  
  74.     len = sendto(sock_raw_fd, send_msg, 14+20+8+len, 0 , (struct sockaddr *)&sll, sizeof(sll));  
  75.     if(len == -1)  
  76.     {  
  77.         perror("sendto");  
  78.     }  
  79.     return 0;  
  80. }  
  81.   
  82. unsigned short checksum(unsigned shortshort *buf, int nword)  
  83. {  
  84.     unsigned long sum;  
  85.     for(sum = 0; nword > 0; nword--)  
  86.     {  
  87.         sum += htons(*buf);  
  88.         buf++;  
  89.     }  
  90.     sum = (sum>>16) + (sum&0xffff);  
  91.     sum += (sum>>16);  
  92.     return ~sum;  
  93. }  

运行结果如下:


转自:http://blog.csdn.net/tennysonsky/article/details/44925057

你可能感兴趣的:(Linux_高级网络编程)