udp_tunnel_xmit_skb函数是OVS2.5发送UDP报文的内核入口,在调用该函数之前,headroom空间需要准备完成,且vxlan头已经创建,skb结构体的data指向vxlan头的首地址。
1、udp_tunnel_xmit_skb函数
int udp_tunnel_xmit_skb(struct rtable *rt, struct sock *sk, struct sk_buff *skb,
__be32 src, __be32 dst, __u8 tos, __u8 ttl,
__be16 df, __be16 src_port, __be16 dst_port,
bool xnet, bool nocheck)
{
struct udphdr *uh;
__skb_push(skb, sizeof(*uh)); //skb增加UDP头,skb报文的headroom大小在vxlan头封装前就完成准备,线性区的空间是充足的
skb_reset_transport_header(skb); //重置报文(外层报文)的transport header的偏移
uh = udp_hdr(skb);
uh->dest = dst_port; //设置目的端口
uh->source = src_port; //设置源端端口
uh->len = htons(skb->len); //设置UDP头中的长度,该长度包括UDP头 + vxlan头 + 用户数据(payload)
udp_set_csum(nocheck, skb, src, dst, skb->len); //UDP头csum计算
return iptunnel_xmit(sk, rt, skb, src, dst, IPPROTO_UDP, //IP层封装tunnel发送报文
tos, ttl, df, xnet);
}
2、udp_set_csum函数
/* Function to set UDP checksum for an IPv4 UDP packet. This is intended
* for the simple case like when setting the checksum for a UDP tunnel.
*/
void udp_set_csum(bool nocheck, struct sk_buff *skb,
__be32 saddr, __be32 daddr, int len)
{
struct udphdr *uh = udp_hdr(skb);
if (nocheck) //如果ovs的vxlan设备支持VXLAN_F_UDP_CSUM标记,则该值为true,默认为false
uh->check = 0;
else if (skb_is_gso(skb)) //对于gso报文,仅需要计算UDP头的csum值
uh->check = ~udp_v4_check(len, saddr, daddr, 0); //如果是gso报文,仅计算UDP头的csum值
else if (skb_dst(skb) && skb_dst(skb)->dev &&
(skb_dst(skb)->dev->features & NETIF_F_V4_CSUM)) { //virtio-net设备不支持该特性,硬件网卡一般都支持
BUG_ON(skb->ip_summed == CHECKSUM_PARTIAL); //非gso报文,ip_summed不可能为CHECKSUM_PARTIAL?
skb->ip_summed = CHECKSUM_PARTIAL;
skb->csum_start = skb_transport_header(skb) - skb->head; //即UDP头的起始位置偏移
skb->csum_offset = offsetof(struct udphdr, check); //check在UDP头中的偏移
uh->check = ~udp_v4_check(len, saddr, daddr, 0); //仅需要计算UDP头的csum值,payload的csum值计算由硬件来做
} else {
__wsum csum;
BUG_ON(skb->ip_summed == CHECKSUM_PARTIAL);
uh->check = 0;
csum = skb_checksum(skb, 0, len, 0); //计算整个报文的csum,比较耗CPU资源
uh->check = udp_v4_check(len, saddr, daddr, csum); //计算UDP头的check值
if (uh->check == 0)
uh->check = CSUM_MANGLED_0;
skb->ip_summed = CHECKSUM_UNNECESSARY; //软件已完成csum计算,硬件不再需要计算
}
}
udp_tunnel_xmit_skb函数完成UDP头的封装,并根据nocheck入参决定如何计算csum值,随后递交给iptunnel_xmit函数进行IP报文的封装并发送。