Chinaunix首页 | 论坛 | 博客
  • 博客访问: 37209
  • 博文数量: 6
  • 博客积分: 391
  • 博客等级: 一等列兵
  • 技术积分: 127
  • 用 户 组: 普通用户
  • 注册时间: 2012-12-07 15:29
文章分类
文章存档

2013年(5)

2012年(1)

我的朋友

分类: C/C++

2012-12-13 16:23:48

  内核版本:2.6.32

  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代码如下:
  1. /* This routine computes an IPv4 TCP checksum. */
  2. void tcp_v4_send_check(struct sock *sk, int len, struct sk_buff *skb)
  3. {
  4.     struct inet_sock *inet = inet_sk(sk);
  5.     struct tcphdr *th = tcp_hdr(skb);

  6.     if (skb->ip_summed == CHECKSUM_PARTIAL) {
  7.         th->check = ~tcp_v4_check(len, inet->saddr,
  8.                      inet->daddr, 0);
  9.         skb->csum_start = skb_transport_header(skb) - skb->head;
  10.         skb->csum_offset = offsetof(struct tcphdr, check);
  11.     } else {
  12.         th->check = tcp_v4_check(len, inet->saddr, inet->daddr,
  13.                      csum_partial(th,
  14.                          th->doff << 2,
  15.                          skb->csum));
  16.     }
  17. }
在这个函数中可以看到第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函数上。看了几遍之后突然看到一句不起眼的代码:
  1. /*
  2.      * Only clear those fields we need to clear, not those that we will
  3.      * actually initialise below. Hence, don't put any more fields after
  4.      * the tail pointer in struct
  5.      */
  6.     memset(skb, 0, offsetof(struct sk_buff, tail));
  7.     skb->truesize = size + sizeof(struct sk_buff);
  8.     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中找,代码片段如下:
  1. /*
  2.                  * Check whether we can use HW checksum.
  3.                  */
  4.                 if (sk->sk_route_caps & NETIF_F_ALL_CSUM)
  5.                     skb->ip_summed = CHECKSUM_PARTIAL;
也就是说如果目的路由网络设备的特性支持NETIF_F_ALL_CSUM时,才将ip_summed设置为CHECKSUM_PARTIAL。
当然这里的讨论只限于TCP协议发送SKB包,到计算校验和的过程中ip_summed成员的变化和值。

阅读(3514) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~