免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1963 | 回复: 0
打印 上一主题 下一主题

TCP raw socket chksum 死活不对,求助! [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-12-23 15:40 |只看该作者 |倒序浏览
看网上有说tcp的chksum = 0的话 kernel 会自动补全,但是我这里一直是0, 如果加上tcp chksum语句的话,校验值却总是错误的。google了好多文档都没有解决。另外 IP 的那个 chksum 倒是 计算不计算都没什么区别,tcpdump显示都一样,很奇怪的说。 我这里的环境是x86_64 Linux。
  
tcpdump:
127.0.0.1.22345 > 127.0.0.1.22346: Flags [S], cksum 0x0000 (incorrect -> 0x109c), seq 1732610923, win 0, length 0
  
加上tcp chksum
127.0.0.1.22345 > 127.0.0.1.22346: Flags [S], cksum 0x0eb9 (incorrect -> 0x109c), seq 1732610923, win 0, length 0
  
代码:

  1. #include <unistd.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <sys/socket.h>
  6. #include <netinet/ip.h>
  7. #include <netinet/tcp.h>

  8. #define MAX_PACKET_SIZE 4096

  9. /* function for header checksums */
  10. unsigned short chksum(unsigned short *buf, int nwords)
  11. {
  12.         unsigned long sum;
  13.         for (sum = 0; nwords > 0; nwords--)
  14.                 sum += *buf++;
  15.         sum = (sum >> 16) + (sum & 0xffff);
  16.         sum += (sum >> 16);
  17.         return (unsigned short)(~sum);
  18. }

  19. void ip_header_init(struct iphdr *iph)
  20. {
  21.         iph->version = 4;
  22.         iph->ihl = 5;
  23.         iph->tos = 0;
  24.         iph->tot_len = sizeof(struct iphdr) + sizeof(struct tcphdr);
  25.         iph->id = htonl(54321);                /* no need to set this value if we set DF */
  26.         iph->frag_off = 0;
  27.         iph->ttl = MAXTTL;                        /* MAXTTL = 255 */
  28.         iph->protocol = 6;                        /* upper layer protocol, TCP */
  29.         iph->check        = 0;
  30.        
  31.         /* Initial IP */
  32.         iph->saddr = inet_addr("127.0.0.1");
  33.         iph->daddr = inet_addr("127.0.0.1");
  34. }

  35. void tcp_header_init(struct tcphdr *tcph)
  36. {
  37.         tcph->source = htons(22345);
  38. //        tcph->dest = htons(22346);
  39.         tcph->seq = random();
  40.         tcph->ack_seq = 0;
  41.         tcph->res2 = 0;
  42.         tcph->doff = 5;
  43.         tcph->syn = 1;
  44.         tcph->window = htonl(65535);
  45.         tcph->check = 0;
  46.         tcph->urg_ptr = 0;
  47. }

  48. int main(int argc, char *argv[])
  49. {
  50.         int sockfd, on =1;
  51.         unsigned int dport;
  52.         char data[MAX_PACKET_SIZE];
  53.         struct iphdr *iph = (struct iphdr *)data;
  54.         struct tcphdr *tcph = (struct tcphdr *)(data + sizeof(struct iphdr));
  55.         struct sockaddr_in dst;
  56.        
  57.         if(argc != 3) {
  58.                 fprintf(stderr, "Invalid parameters!\n");
  59.                 fprintf(stdout, "Usage: %s <target> <port>\n", argv[0]);
  60.                 exit(1);
  61.         }
  62.        
  63.         if ( (sockfd = socket(PF_INET, SOCK_RAW, IPPROTO_TCP)) < 0) {
  64.                         perror("Socket raw error ");
  65.                         exit(1);
  66.         }
  67.        
  68.         dport = atoi(argv[2]);
  69.         dst.sin_family = AF_INET;
  70.         dst.sin_port = htons(dport);
  71.         dst.sin_addr.s_addr = inet_addr(argv[1]);
  72.        
  73.         /* init ip and tcp headers */
  74.         bzero(data, sizeof(data));
  75.         ip_header_init(iph);
  76.         tcp_header_init(tcph);
  77.         iph->daddr = dst.sin_addr.s_addr;
  78.         tcph->dest = htons(dport);
  79.         iph->check = chksum((unsigned short *)data, iph->tot_len);
  80.         tcph->check = chksum((unsigned short *)tcph, sizeof(struct tcphdr));

  81.        
  82.         if ( setsockopt(sockfd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0) {
  83.                         perror("setsockopt IP_HDRINCL error");
  84.                         exit(1);
  85.         }
  86.        
  87.         if (sendto(sockfd, data, iph->tot_len, 0, (struct sockaddr *)&dst, \
  88.                                 sizeof(dst)) < 0) {
  89.                         perror("send to error");
  90.                         exit(1);
  91.         }
  92.        
  93.         return 0;       
  94. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP