在virtio数据传输中,如果一个报文使用多个desc,那么这些desc会通过VRING_DESC_F_NEXT进行成链动作。在virtio中有一个can_push的参数,如果can_push为0,那么virtio_net_hdr会单独真用一个desc(sg或mbuf),报文pkt占用一个单独的desc(sg或mbuf),virtio_net_hdr和pkt的两个desc通过VRING_DESC_F_NEXT形成链。也就是说,不论报文的大小,如果can_push为0,都会存在发送端成链的现象,下面是相关的不同实现
1.1 内核实现
can_push的判定
根据类型选择hdr的大小(是否有merge_buf), 没有can_push的情况,hdr放入第一个sg,报文skb放在第二个sg(sg+1), 总num_sg数量加1
发送sg
发送时候,sg多于1个,会进行成链操作
1.2 DPDK实现
can_push的判断
can_push 为 0 时,第一个desc记录hdr,然后idx下移一个,通过F_NEXT进行成链操作
细心的同学会看到在indirect的时候,hdr指向的是 txr[idx].tx_hdr,start_dp[idx=0]也没有初始化为virtio_net_hdr啊,怎么就让第一个desc指向了virtio_net_hdr呢?
这是因为在初始化的时候,start_dp[idx=0] 都被初始化指向了 virtio_net_hdr了