- 论坛徽章:
- 0
|
看网上有说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
代码:
- #include <unistd.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <sys/socket.h>
- #include <netinet/ip.h>
- #include <netinet/tcp.h>
-
- #define MAX_PACKET_SIZE 4096
- /* function for header checksums */
- unsigned short chksum(unsigned short *buf, int nwords)
- {
- unsigned long sum;
- for (sum = 0; nwords > 0; nwords--)
- sum += *buf++;
- sum = (sum >> 16) + (sum & 0xffff);
- sum += (sum >> 16);
- return (unsigned short)(~sum);
- }
- void ip_header_init(struct iphdr *iph)
- {
- iph->version = 4;
- iph->ihl = 5;
- iph->tos = 0;
- iph->tot_len = sizeof(struct iphdr) + sizeof(struct tcphdr);
- iph->id = htonl(54321); /* no need to set this value if we set DF */
- iph->frag_off = 0;
- iph->ttl = MAXTTL; /* MAXTTL = 255 */
- iph->protocol = 6; /* upper layer protocol, TCP */
- iph->check = 0;
-
- /* Initial IP */
- iph->saddr = inet_addr("127.0.0.1");
- iph->daddr = inet_addr("127.0.0.1");
- }
- void tcp_header_init(struct tcphdr *tcph)
- {
- tcph->source = htons(22345);
- // tcph->dest = htons(22346);
- tcph->seq = random();
- tcph->ack_seq = 0;
- tcph->res2 = 0;
- tcph->doff = 5;
- tcph->syn = 1;
- tcph->window = htonl(65535);
- tcph->check = 0;
- tcph->urg_ptr = 0;
- }
- int main(int argc, char *argv[])
- {
- int sockfd, on =1;
- unsigned int dport;
- char data[MAX_PACKET_SIZE];
- struct iphdr *iph = (struct iphdr *)data;
- struct tcphdr *tcph = (struct tcphdr *)(data + sizeof(struct iphdr));
- struct sockaddr_in dst;
-
- if(argc != 3) {
- fprintf(stderr, "Invalid parameters!\n");
- fprintf(stdout, "Usage: %s <target> <port>\n", argv[0]);
- exit(1);
- }
-
- if ( (sockfd = socket(PF_INET, SOCK_RAW, IPPROTO_TCP)) < 0) {
- perror("Socket raw error ");
- exit(1);
- }
-
- dport = atoi(argv[2]);
- dst.sin_family = AF_INET;
- dst.sin_port = htons(dport);
- dst.sin_addr.s_addr = inet_addr(argv[1]);
-
- /* init ip and tcp headers */
- bzero(data, sizeof(data));
- ip_header_init(iph);
- tcp_header_init(tcph);
- iph->daddr = dst.sin_addr.s_addr;
- tcph->dest = htons(dport);
- iph->check = chksum((unsigned short *)data, iph->tot_len);
- tcph->check = chksum((unsigned short *)tcph, sizeof(struct tcphdr));
-
- if ( setsockopt(sockfd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0) {
- perror("setsockopt IP_HDRINCL error");
- exit(1);
- }
-
- if (sendto(sockfd, data, iph->tot_len, 0, (struct sockaddr *)&dst, \
- sizeof(dst)) < 0) {
- perror("send to error");
- exit(1);
- }
-
- return 0;
- }
复制代码 |
|