当前位置:编程学习 > C/C++ >>

TCP协议发送SKB时ip_summed成员的设置

tcp_send_ack()函数是内核用来发送ACK的函数,该函数比较简单,就是先分配一个SKB包,然后简单的初始化(初始化操作中没有设置ip_summed)后,调用tcp_transmit_skb()来将SKB包传递到IP层。tcp_transmit_skb中先做一些操作后(仔细看过,到计算校验和之前,没有更改过ip_summed),会调用tcp_v4_send_check(语句是:icsk->icsk_af_ops->send_check(sk, skb->len, skb);)来计算校验和。
 tcp_v4_send_check代码如下:
[cpp]  
/* This routine computes an IPv4 TCP checksum. */  
void tcp_v4_send_check(struct sock *sk, int len, struct sk_buff *skb)  
{  
    struct inet_sock *inet = inet_sk(sk);  
    struct tcphdr *th = tcp_hdr(skb);  
  
    if (skb->ip_summed == CHECKSUM_PARTIAL) {  
        th->check = ~tcp_v4_check(len, inet->saddr,  
                     inet->daddr, 0);  
        skb->csum_start = skb_transport_header(skb) - skb->head;  
        skb->csum_offset = offsetof(struct tcphdr, check);  
    } else {  
        th->check = tcp_v4_check(len, inet->saddr, inet->daddr,  
                     csum_partial(th,  
                         th->doff << 2,  
                         skb->csum));  
    }  
}  
在这个函数中可以看到第7行中已经开始使用skb->ip_summed来做比较,那也就是说skb->ip_summed肯定在之前已经初始化过,反反复复地仔细阅读tcp_send_ack和tcp_transmit_skb函数都没有发现初始化的地方,真的是很奇怪。因为之前我看过alloc_skb的实现,而且又看了一次,没看到在哪个地方“明显地”初始化了ip_summed成员。
  只能全文搜索ip_summed的所有设置的地方,看了半天还是没有找到在tcp_send_ack到tcp_transmit_skb之间的地方有初始化ip_summed的操作,最后还是把目光锁定在alloc_skb函数上。看了几遍之后突然看到一句不起眼的代码:
[cpp]  
/* 
     * Only clear those fields we need to clear, not those that we will 
     * actually initialise below. Hence, don't put any more fields after 
     * the tail pointer in struct  
     */  
    memset(skb, 0, offsetof(struct sk_buff, tail));  
    skb->truesize = size + sizeof(struct sk_buff);  
    atomic_set(&skb->users, 1);  
看到第6行代码,才恍然大悟,原来是在这个地方把skb从head成员到tail的所有成员都初始化为0,当前ip_summed也在这个范围内,也就是说ip_summed的值为0,对应的就是CHECKSUM_NONE。哎,只怪自己粗心大意,看的不够仔细!
 
  tcp_v4_send_check中第7行if (skb->ip_summed == CHECKSUM_PARTIAL),判断ip_summed是否等于CHECKSUM_PARTIAL,那在什么是否会将ip_summed置为CHECKSUM_PARTIAL呢?这个答案要在tcp_sendmsg中找,代码片段如下:
[cpp]  
/* 
                 * Check whether we can use HW checksum. 
                 */  
                if (sk->sk_route_caps & NETIF_F_ALL_CSUM)  
                    skb->ip_summed = CHECKSUM_PARTIAL;  
 
也就是说如果目的路由网络设备的特性支持NETIF_F_ALL_CSUM时,才将ip_summed设置为CHECKSUM_PARTIAL。
当然这里的讨论只限于TCP协议发送SKB包,到计算校验和的过程中ip_summed成员的变化和值。
 
补充:软件开发 , C++ ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,