我正在尝试创建一个为TCP生成校验和的函数,但我使用两种TCP结构,因为在tcp.h头中为Mac OS X制作的tcphdr结构不包含制作数据包所需的所有内容。我的数据包包含tcphdr结构和这个结构:
typedef struct tcp_option{
u_int16_t mss_opt:8,
mss_len:8;
u_int16_t mss;
u_int16_t sack_kind:8,
sack_len:8;
u_int16_t win_opt:8,
win_len:8;
u_int32_t win:8,
win_nop:8,
time_opt:8,
time_len:8;
u_int32_t time;
u_int32_t time_echo;
} tcp_option;
所以有人有想法制作校验和以将此结构和tcphdr结构传递给。
我的校验和功能适用于IP校验和,但不适用于其中包含两个结构的TCP校验和。这是校验和功能:
unsigned short checksum(const char *buf, unsigned size)
{
unsigned long long sum = 0;
const unsigned long long *b = (unsigned long long *) buf;
unsigned t1, t2;
unsigned short t3, t4;
/* Main loop - 8 bytes at a time */
while (size >= sizeof(unsigned long long))
{
unsigned long long s = *b++;
sum += s;
if (sum < s) sum++;
size -= 8;
}
/* Handle tail less than 8-bytes long */
buf = (const char *) b;
if (size & 4)
{
unsigned s = *(unsigned *)buf;
sum += s;
if (sum < s) sum++;
buf += 4;
}
if (size & 2)
{
unsigned short s = *(unsigned short *) buf;
sum += s;
if (sum < s) sum++;
buf += 2;
}
if (size)
{
unsigned char s = *(unsigned char *) buf;
sum += s;
if (sum < s) sum++;
}
/* Fold down to 16 bits */
t1 = sum;
t2 = sum >> 32;
t1 += t2;
if (t1 < t2) t1++;
t3 = t1;
t4 = t1 >> 16;
t3 += t4;
if (t3 < t4) t3++;
return ~t3;
}
我自己没有,我从互联网上偷了它。
解决方案:我删除了两个结构并将其组合到一个结构中,然后我创建了一个伪标头结构,我用它来制作校验和!
答案 0 :(得分:0)
您的结构在u_int_16_t
成员之前有一个奇数个u_int_32_t
成员,所以为了确保32位成员正确对齐,它将有一个隐藏的16位块,其中您将永远无法通过成员引用进行设置,其内容将是未定义的。因此,比较缓冲区中的每个16位字将包括比较该16位值。如果您没有使用某种方法来确保该空间的值始终相同,例如在为其成员分配值之前的memset(0)
结构,那么您将不得不使用不同的比较过程(比较成员明确地说。)