两个结构的TCP校验和

时间:2018-02-15 15:05:41

标签: c tcp checksum

我正在尝试创建一个为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;
}

我自己没有,我从互联网上偷了它。

解决方案:我删除了两个结构并将其组合到一个结构中,然后我创建了一个伪标头结构,我用它来制作校验和!

1 个答案:

答案 0 :(得分:0)

您的结构在u_int_16_t成员之前有一个奇数个u_int_32_t成员,所以为了确保32位成员正确对齐,它将有一个隐藏的16位块,其中您将永远无法通过成员引用进行设置,其内容将是未定义的。因此,比较缓冲区中的每个16位字将包括比较该16位值。如果您没有使用某种方法来确保该空间的值始终相同,例如在为其成员分配值之前的memset(0)结构,那么您将不得不使用不同的比较过程(比较成员明确地说。)