Linux内核网络UDP数据包发送(三)——IP协议层分析
扫描二维码
随时随地手机看文章
1. 前言
Linux内核网络 UDP 协议层通过调用ip_send_skb
将 skb 交给 IP 协议层,本文通过分析内核 IP 协议层的关键函数来分享内核数据包发送在 IP 协议层的处理,并分享了监控IP层的方法。2. ip_send_skb
ip_send_skb
函数定义在 net/ipv4/ip_output.c 中,非常简短。它只是调用ip_local_out
,如果调用失败,就更新相应的错误计数:int ip_send_skb(struct net *net, struct sk_buff *skb)
{
int err;
err = ip_local_out(skb);
if (err) {
if (err > 0)
err = net_xmit_errno(err);
if (err)
IP_INC_STATS(net, IPSTATS_MIB_OUTDISCARDS);
}
return err;
}
net_xmit_errno
函数将低层错误转换为 IP 和 UDP 协议层所能理解的错误。如果发生错误, IP 协议计数器 OutDiscards
会递增。稍后我们将看到读取哪些文件可以获取此统计信息。接下来看 ip_local_out
。3. ip_local_out
and __ip_local_out
ip_local_out
和__ip_local_out
都很简单。ip_local_out
只需调用__ip_local_out
,如果返回值为 1,则调用路由层 dst_output
发送数据包:int ip_local_out(struct sk_buff *skb)
{
int err;
err = __ip_local_out(skb);
if (likely(err == 1))
err = dst_output(skb);
return err;
}
接下来看__ip_local_out
的代码:int __ip_local_out(struct sk_buff *skb)
{
struct iphdr *iph = ip_hdr(skb);
iph->tot_len = htons(skb->len);
ip_send_check(iph);
return nf_hook(NFPROTO_IPV4, NF_INET_LOCAL_OUT, skb, NULL,
skb_dst(skb)->dev, dst_output);
}
可以看到,该函数首先做了两件重要的事情:- 设置 IP 数据包的长度
- 调用
ip_send_check
来计算要写入 IP 头的校验和。ip_send_check
函数将进一步调用名为ip_fast_csum
的函数来计算校验和。在 x86 和 x86_64 体系结构上,此函数用汇编实 现。
nf_hook
进入 netfilter,其返回值将传递回 ip_local_out
。如果 nf_hook
返回 1,则表示允许数据包通过,并且调用者应该自己发送数据包。这正是我们在上面看到的情况:ip_local_out
检查返回值 1 时,自己通过调用 dst_output
发送数据包。3.1 netfilter and nf_hook
nf_hook
只是一个 wrapper,它调用 nf_hook_thresh
,首先检查是否有为这个协议族和hook 类型(这里分别为 NFPROTO_IPV4
和 NF_INET_LOCAL_OUT
)安装的过滤器,然后将返回到 IP 协议层,避免深入到 netfilter 或更下面,比如 iptables 和 conntrack。如果有非常多或者非常复杂的 netfilter 或 iptables 规则,那些规则将在触发 sendmsg
系统调的用户进程的上下文中执行。如果对这个用户进程设置了 CPU 亲和性,相应的 CPU 将花费系统时间(system time)处理出站(outbound)iptables 规则。如果做性能回归测试,那可能要考虑根据系统的负载,将相应的用户进程绑到到特定的 CPU,或者是减少 netfilter/iptables 规则的复杂度,以减少对性能测试的影响。出于讨论目的,我们假设 nf_hook
返回 1,表示调用者(在这种情况下是 IP 协议层)应该自己发送数据包。3.2 目的(路由)缓存
dst 代码在 Linux 内核中实现协议无关的目标缓存。为了继续学习发送 UDP 数据报的流程 ,我们需要了解 dst 条目是如何被设置的,首先来看 dst 条目和路由是如何生成的。目标缓存,路由和邻居子系统,任何一个都可以拿来单独详细的介绍。现在不深入细节,只是快速地看一下它们是如何组合到一起的。上面看到的代码调用了dst_output(skb)
。此函数只是查找关联到这个 skb 的 dst 条目 ,然后调用 output
方法。代码如下:/* Output packet to network from transport. */
static inline int dst_output(struct sk_buff *skb)
{
return skb_dst(skb)->output(skb);
}
看起来很简单,但是 output
方法之前是如何关联到 dst 条目的?首先很重要的一点,目标缓存条目是以多种不同方式添加的。到目前为止,我们已经在代码中看到的一种方法是从 udp_sendmsg
调用ip_route_output_flow
。ip_route_output_flow
函数调用 __ip_route_output_key
,后者进而调用 __mkroute_output
。 __mkroute_output
函数创建路由和目标缓存条目。当它执行创建操作时,它会判断哪个 output
方法适合此 dst。大多数时候,这个函数是 ip_output
。4. ip_output
在 UDP IPv4 情况下,上面的 output
方法指向的是 ip_output
:int ip_output(struct sk_buff *skb)
{
struct net_device *dev = skb_dst(skb)->dev;
IP_UPD_PO_STATS(dev_net(dev), IPSTATS_MIB_OUT, skb->len);
skb->dev = dev;
skb->protocol = htons(ETH_P_IP);
return NF_HOOK_COND(NFPROTO_IPV4, NF_INET_POST_ROUTING, skb, NULL, dev,
ip_finish_output,
!(IPCB(skb)->flags