- 论坛徽章:
- 1
|
本帖最后由 Fart_King 于 2015-06-18 09:54 编辑
uint16_t checksum(void *buffer, uint32_t length)
{
uint16_t *pAddr = (uint16_t *)buffer;
uint32_t sum = 0;
while (length > 1)
{
sum += *(uint16_t *)pAddr++;
if (sum & 0x80000000)
{
sum = (sum & 0xffff) + (sum >> 16);
}
length -= 2;
}
if (length > 1)
{
sum += *(uint8_t *)pAddr;
}
while (sum >> 16)
{
sum = (sum & 0xffff) + (sum >> 16);
}
return ~sum;
}
void test_checksum(char *buf, PSDHEADER *psdHeader,unsigned char* calcBuf)
{
struct ether_header *ethHdr = (struct ether_header*)buf;
struct ip *ipHdr = (struct ip*)(buf + sizeof(struct ether_header));
struct tcphdr *tcpHdr = (struct tcphdr*)(buf + sizeof(struct ether_header) + sizeof(struct ip));
u_short ip_sum = 0;
uint16_t tcp_check = 0;
char *httpbuf = (char*)(buf + sizeof(struct ether_header) + sizeof(struct ip) + sizeof(struct tcphdr));
//strcpy(httpbuf, request);
int len = 0; //http + tcp len
fprintf(stdout, "auto tcp->check = %d\n", tcpHdr->check);
ipHdr->ip_sum = 0;
tcpHdr->check = 0;
len = ntohs(ipHdr->ip_len)- ((ipHdr->ip_hl)&15)*4;
fprintf(stdout, "ip len = %d http + tcp len = %d\n", ((ipHdr->ip_hl)&15)*4, len);
memset(calcBuf, 0, sizeof(calcBuf));
memcpy(calcBuf, (void*)ipHdr, sizeof(struct ip));
ip_sum = checksum((uint16_t*)(buf + sizeof(struct ether_header)), sizeof(struct ip));
psdHeader->saddr = (unsigned long)ipHdr->ip_src.s_addr;
psdHeader->daddr = (unsigned long)ipHdr->ip_dst.s_addr;
psdHeader->mbz = 0;
psdHeader->ptcl = 0x06;
// psdHeader->tcpl = htons((u_short)(sizeof (struct tcphdr) + http_len));
psdHeader->tcpl = htons((u_short)(len));
//tcpHdr->check = CalcTCPSum((u_int16_t *)psdHeader, (u_int16_t *)tcpHdr,sizeof(struct tcphdr) + http_len);
memset(calcBuf, 0, sizeof(calcBuf));
memcpy(calcBuf, (void*)psdHeader, sizeof(PSDHEADER));
memcpy(calcBuf + sizeof(PSDHEADER), (void*)tcpHdr, sizeof(struct tcphdr));
//memcpy(calcBuf + sizeof(PSDHEADER), (void*)tcpHdr, len);
memcpy(calcBuf + sizeof(PSDHEADER) + sizeof(struct tcphdr), (void*)httpbuf, len - sizeof(struct tcphdr));
tcp_check = checksum((u_int16_t*)calcBuf, sizeof(PSDHEADER) + len); //
// tcpHdr->check = tcp4_checksum (*ipHdr, *tcpHdr, (uint8_t *) httpbuf, http_len);
fprintf(stdout, "tcp_check = %d\n", tcp_check);
}
这个函数是我用来测试tcp checksum是否正确的function。
参数 buf 就是raw sock 接收的以太网帧 psdheader 是tcp 伪首部 。
为什么一旦tcp包中带了数据tcp->check 就不正确了呢。求大神指点 |
|